1use carnelian::drawing::TextGrid;
6use carnelian::render::{Context as RenderContext, Path};
7use carnelian::Size;
8use euclid::{point2, vec2};
9use term_model::ansi::CursorStyle;
10
11const LINE_THICKNESS_FACTOR: f32 = 1.0 / 16.0;
15
16fn path_for_block(size: &Size, render_context: &mut RenderContext) -> Path {
17 let mut path_builder = render_context.path_builder().expect("path_builder");
18 path_builder
19 .move_to(point2(0.0, 0.0))
20 .line_to(point2(size.width, 0.0))
21 .line_to(point2(size.width, size.height))
22 .line_to(point2(0.0, size.height))
23 .line_to(point2(0.0, 0.0));
24 path_builder.build()
25}
26
27pub struct Line {
28 position_y: f32,
29 thickness: f32,
30}
31
32impl Line {
33 pub fn new(line_metrics: ttf_parser::LineMetrics, textgrid: &TextGrid) -> Self {
34 let scale_y = textgrid.scale.y;
35 let offset_y = textgrid.offset.y;
36 let position = line_metrics.position as f32 * scale_y;
37 let thickness = line_metrics.thickness as f32 * scale_y;
38 let position_y = offset_y - position;
39 Self { thickness, position_y }
40 }
41}
42
43fn line_bounds(line: Option<Line>, default_bounds: (f32, f32), cell_height: f32) -> (f32, f32) {
44 line.map_or(default_bounds, |Line { thickness, position_y: line_y }| {
45 let top = line_y - thickness / 2.0;
46 let bottom = line_y + thickness / 2.0;
47 let (_, default_bottom) = default_bounds;
48
49 if bottom > cell_height {
50 return (default_bottom - thickness, default_bottom);
51 }
52 (top, bottom)
53 })
54}
55
56pub fn path_for_underline(
57 size: &Size,
58 render_context: &mut RenderContext,
59 line: Option<Line>,
60) -> Path {
61 let mut path_builder = render_context.path_builder().expect("path_builder");
62 let default_top = size.height - size.height * LINE_THICKNESS_FACTOR;
63 let default_bottom = size.height;
64 let (top, bottom) = line_bounds(line, (default_top, default_bottom), size.height);
65
66 path_builder
67 .move_to(point2(0.0, top))
68 .line_to(point2(size.width, top))
69 .line_to(point2(size.width, bottom))
70 .line_to(point2(0.0, bottom))
71 .line_to(point2(0.0, top));
72 path_builder.build()
73}
74
75fn path_for_beam(size: &Size, render_context: &mut RenderContext) -> Path {
76 let mut path_builder = render_context.path_builder().expect("path_builder");
77 let right = size.height * LINE_THICKNESS_FACTOR;
78 path_builder
79 .move_to(point2(0.0, 0.0))
80 .line_to(point2(right, 0.0))
81 .line_to(point2(right, size.height))
82 .line_to(point2(0.0, size.height))
83 .line_to(point2(0.0, 0.0));
84 path_builder.build()
85}
86
87fn path_for_hollow_block(size: &Size, render_context: &mut RenderContext) -> Path {
88 let mut path_builder = render_context.path_builder().expect("path_builder");
89 let inset = size.height * LINE_THICKNESS_FACTOR;
90 let bottom_start = size.height - inset;
91 let right_start = size.width - inset;
92 path_builder
93 .move_to(point2(0.0, 0.0))
95 .line_to(point2(size.width, 0.0))
96 .line_to(point2(size.width, inset))
97 .line_to(point2(0.0, inset))
98 .line_to(point2(0.0, 0.0))
99 .move_to(point2(0.0, bottom_start))
101 .line_to(point2(size.width, bottom_start))
102 .line_to(point2(size.width, size.height))
103 .line_to(point2(0.0, size.height))
104 .line_to(point2(0.0, bottom_start))
105 .move_to(point2(0.0, inset))
107 .line_to(point2(inset, inset))
108 .line_to(point2(inset, bottom_start))
109 .line_to(point2(0.0, bottom_start))
110 .line_to(point2(0.0, inset))
111 .move_to(point2(right_start, inset))
113 .line_to(point2(size.width, inset))
114 .line_to(point2(size.width, bottom_start))
115 .line_to(point2(right_start, bottom_start))
116 .line_to(point2(right_start, inset));
117 path_builder.build()
118}
119
120pub fn path_for_strikeout(
121 size: &Size,
122 render_context: &mut RenderContext,
123 line: Option<Line>,
124) -> Path {
125 let mut path_builder = render_context.path_builder().expect("path_builder");
126 let default_top = size.height / 2.0;
127 let default_bottom = default_top + size.height * LINE_THICKNESS_FACTOR;
128 let (top, bottom) = line_bounds(line, (default_top, default_bottom), size.height);
129
130 path_builder
131 .move_to(point2(0.0, top))
132 .line_to(point2(size.width, top))
133 .line_to(point2(size.width, bottom))
134 .line_to(point2(0.0, bottom))
135 .line_to(point2(0.0, top));
136 path_builder.build()
137}
138
139fn path_for_unicode_2500(size: &Size, render_context: &mut RenderContext) -> Path {
141 let mut path_builder = render_context.path_builder().expect("path_builder");
142 let thickness = size.height * LINE_THICKNESS_FACTOR;
143 let top = size.height / 2.0 - thickness / 2.0;
144 let bottom = top + thickness;
145 path_builder
146 .move_to(point2(0.0, top))
147 .line_to(point2(size.width, top))
148 .line_to(point2(size.width, bottom))
149 .line_to(point2(0.0, bottom))
150 .line_to(point2(0.0, top));
151 path_builder.build()
152}
153
154fn path_for_unicode_2502(size: &Size, render_context: &mut RenderContext) -> Path {
156 let mut path_builder = render_context.path_builder().expect("path_builder");
157 let thickness = size.height * LINE_THICKNESS_FACTOR;
158 let left = size.width / 2.0 - thickness / 2.0;
159 let right = left + thickness;
160 path_builder
161 .move_to(point2(left, 0.0))
162 .line_to(point2(right, 0.0))
163 .line_to(point2(right, size.height))
164 .line_to(point2(left, size.height))
165 .line_to(point2(left, 0.0));
166 path_builder.build()
167}
168
169fn path_for_unicode_256d(size: &Size, render_context: &mut RenderContext) -> Path {
171 let mut path_builder = render_context.path_builder().expect("path_builder");
172 let thickness = size.height * LINE_THICKNESS_FACTOR;
173 let bottom_left = size.width / 2.0 - thickness / 2.0;
174 let bottom_right = bottom_left + thickness;
175 let right_top = size.height / 2.0 - thickness / 2.0;
176 let right_bottom = right_top + thickness;
177 let radius = (size.height * 0.25).min(size.width * 0.25);
178 let inner_radius = radius - thickness / 2.0;
179 let outer_radius = inner_radius + thickness;
180 let kappa = 4.0 / 3.0 * (std::f32::consts::PI / 8.0).tan();
181 let outer_control_dist = kappa * outer_radius;
182 let inner_control_dist = kappa * inner_radius;
183 let center = point2(size.width / 2.0, size.height / 2.0) + vec2(radius, radius);
184 let inner_p1 = center + vec2(-inner_control_dist, -inner_radius);
185 let inner_p2 = center + vec2(-inner_radius, -inner_control_dist);
186 let outer_p1 = center + vec2(-outer_radius, -outer_control_dist);
187 let outer_p2 = center + vec2(-outer_control_dist, -outer_radius);
188 path_builder
189 .move_to(point2(size.width, right_top))
190 .line_to(point2(size.width, right_bottom))
191 .line_to(point2(center.x, right_bottom))
192 .cubic_to(inner_p1, inner_p2, point2(bottom_right, center.y))
193 .line_to(point2(bottom_right, size.height))
194 .line_to(point2(bottom_left, size.height))
195 .line_to(point2(bottom_left, center.y))
196 .cubic_to(outer_p1, outer_p2, point2(center.x, right_top))
197 .line_to(point2(size.width, right_top));
198 path_builder.build()
199}
200
201fn path_for_unicode_256e(size: &Size, render_context: &mut RenderContext) -> Path {
203 let mut path_builder = render_context.path_builder().expect("path_builder");
204 let thickness = size.height * LINE_THICKNESS_FACTOR;
205 let bottom_left = size.width / 2.0 - thickness / 2.0;
206 let bottom_right = bottom_left + thickness;
207 let left_top = size.height / 2.0 - thickness / 2.0;
208 let left_bottom = left_top + thickness;
209 let radius = (size.height * 0.25).min(size.width * 0.25);
210 let inner_radius = radius - thickness / 2.0;
211 let outer_radius = inner_radius + thickness;
212 let kappa = 4.0 / 3.0 * (std::f32::consts::PI / 8.0).tan();
213 let outer_control_dist = kappa * outer_radius;
214 let inner_control_dist = kappa * inner_radius;
215 let center = point2(size.width / 2.0, size.height / 2.0) + vec2(-radius, radius);
216 let inner_p1 = center + vec2(inner_radius, -inner_control_dist);
217 let inner_p2 = center + vec2(inner_control_dist, -inner_radius);
218 let outer_p1 = center + vec2(outer_control_dist, -outer_radius);
219 let outer_p2 = center + vec2(outer_radius, -outer_control_dist);
220 path_builder
221 .move_to(point2(0.0, left_top))
222 .line_to(point2(center.x, left_top))
223 .cubic_to(outer_p1, outer_p2, point2(bottom_right, center.y))
224 .line_to(point2(bottom_right, size.height))
225 .line_to(point2(bottom_left, size.height))
226 .line_to(point2(bottom_left, center.y))
227 .cubic_to(inner_p1, inner_p2, point2(center.x, left_bottom))
228 .line_to(point2(0.0, left_bottom))
229 .line_to(point2(0.0, left_top));
230 path_builder.build()
231}
232
233fn path_for_unicode_256f(size: &Size, render_context: &mut RenderContext) -> Path {
235 let mut path_builder = render_context.path_builder().expect("path_builder");
236 let thickness = size.height * LINE_THICKNESS_FACTOR;
237 let top_left = size.width / 2.0 - thickness / 2.0;
238 let top_right = top_left + thickness;
239 let left_top = size.height / 2.0 - thickness / 2.0;
240 let left_bottom = left_top + thickness;
241 let radius = (size.height * 0.25).min(size.width * 0.25);
242 let inner_radius = radius - thickness / 2.0;
243 let outer_radius = inner_radius + thickness;
244 let kappa = 4.0 / 3.0 * (std::f32::consts::PI / 8.0).tan();
245 let outer_control_dist = kappa * outer_radius;
246 let inner_control_dist = kappa * inner_radius;
247 let center = point2(size.width / 2.0, size.height / 2.0) + vec2(-radius, -radius);
248 let inner_p1 = center + vec2(inner_control_dist, inner_radius);
249 let inner_p2 = center + vec2(inner_radius, inner_control_dist);
250 let outer_p1 = center + vec2(outer_radius, outer_control_dist);
251 let outer_p2 = center + vec2(outer_control_dist, outer_radius);
252 path_builder
253 .move_to(point2(top_left, 0.0))
254 .line_to(point2(top_right, 0.0))
255 .line_to(point2(top_right, center.y))
256 .cubic_to(outer_p1, outer_p2, point2(center.x, left_bottom))
257 .line_to(point2(0.0, left_bottom))
258 .line_to(point2(0.0, left_top))
259 .line_to(point2(center.x, left_top))
260 .cubic_to(inner_p1, inner_p2, point2(top_left, center.y))
261 .line_to(point2(top_left, 0.0));
262 path_builder.build()
263}
264
265fn path_for_unicode_2570(size: &Size, render_context: &mut RenderContext) -> Path {
267 let mut path_builder = render_context.path_builder().expect("path_builder");
268 let thickness = size.height * LINE_THICKNESS_FACTOR;
269 let top_left = size.width / 2.0 - thickness / 2.0;
270 let top_right = top_left + thickness;
271 let right_top = size.height / 2.0 - thickness / 2.0;
272 let right_bottom = right_top + thickness;
273 let radius = (size.height * 0.25).min(size.width * 0.25);
274 let inner_radius = radius - thickness / 2.0;
275 let outer_radius = inner_radius + thickness;
276 let kappa = 4.0 / 3.0 * (std::f32::consts::PI / 8.0).tan();
277 let outer_control_dist = kappa * outer_radius;
278 let inner_control_dist = kappa * inner_radius;
279 let center = point2(size.width / 2.0, size.height / 2.0) + vec2(radius, -radius);
280 let inner_p1 = center + vec2(-inner_radius, inner_control_dist);
281 let inner_p2 = center + vec2(-inner_control_dist, inner_radius);
282 let outer_p1 = center + vec2(-outer_control_dist, outer_radius);
283 let outer_p2 = center + vec2(-outer_radius, outer_control_dist);
284 path_builder
285 .move_to(point2(top_left, 0.0))
286 .line_to(point2(top_right, 0.0))
287 .line_to(point2(top_right, center.y))
288 .cubic_to(inner_p1, inner_p2, point2(center.x, right_top))
289 .line_to(point2(size.width, right_top))
290 .line_to(point2(size.width, right_bottom))
291 .line_to(point2(center.x, right_bottom))
292 .cubic_to(outer_p1, outer_p2, point2(top_left, center.y))
293 .line_to(point2(top_left, 0.0));
294 path_builder.build()
295}
296
297fn path_for_unicode_2591(size: &Size, render_context: &mut RenderContext) -> Path {
299 let mut path_builder = render_context.path_builder().expect("path_builder");
300 const GRID_SIZE: usize = 8;
301 let scale_x = size.width / GRID_SIZE as f32;
302 let scale_y = size.height / GRID_SIZE as f32;
303 for y in 0..GRID_SIZE {
304 for x in 0..GRID_SIZE {
305 let offset = if y % 2 == 0 { x } else { x + 2 };
306 if offset % 4 == 0 {
308 let x0 = x as f32 * scale_x;
309 let y0 = y as f32 * scale_y;
310 let x1 = (x + 1) as f32 * scale_x;
311 let y1 = (y + 1) as f32 * scale_y;
312 path_builder
313 .move_to(point2(x0, y0))
314 .line_to(point2(x1, y0))
315 .line_to(point2(x1, y1))
316 .line_to(point2(x0, y1))
317 .line_to(point2(x0, y0));
318 }
319 }
320 }
321 path_builder.build()
322}
323
324fn path_for_unicode_2592(size: &Size, render_context: &mut RenderContext) -> Path {
326 let mut path_builder = render_context.path_builder().expect("path_builder");
327 const GRID_SIZE: usize = 9;
328 let scale_x = size.width / GRID_SIZE as f32;
329 let scale_y = size.height / GRID_SIZE as f32;
330 for y in 0..GRID_SIZE {
331 for x in 0..GRID_SIZE {
332 let offset = if y % 2 == 0 { x } else { x + 1 };
333 if offset % 2 == 0 {
335 let x0 = x as f32 * scale_x;
336 let y0 = y as f32 * scale_y;
337 let x1 = (x + 1) as f32 * scale_x;
338 let y1 = (y + 1) as f32 * scale_y;
339 path_builder
340 .move_to(point2(x0, y0))
341 .line_to(point2(x1, y0))
342 .line_to(point2(x1, y1))
343 .line_to(point2(x0, y1))
344 .line_to(point2(x0, y0));
345 }
346 }
347 }
348 path_builder.build()
349}
350
351fn path_for_unicode_2593(size: &Size, render_context: &mut RenderContext) -> Path {
353 let mut path_builder = render_context.path_builder().expect("path_builder");
354 const GRID_SIZE: usize = 8;
355 let scale_x = size.width / GRID_SIZE as f32;
356 let scale_y = size.height / GRID_SIZE as f32;
357 for y in 0..GRID_SIZE {
358 for x in 0..GRID_SIZE {
359 let offset = if y % 2 == 0 { x } else { x + 2 };
360 if offset % 4 != 0 {
362 let x0 = x as f32 * scale_x;
363 let y0 = y as f32 * scale_y;
364 let x1 = (x + 1) as f32 * scale_x;
365 let y1 = (y + 1) as f32 * scale_y;
366 path_builder
367 .move_to(point2(x0, y0))
368 .line_to(point2(x1, y0))
369 .line_to(point2(x1, y1))
370 .line_to(point2(x0, y1))
371 .line_to(point2(x0, y0));
372 }
373 }
374 }
375 path_builder.build()
376}
377
378fn path_for_unicode_2714(size: &Size, render_context: &mut RenderContext) -> Path {
380 let mut path_builder = render_context.path_builder().expect("path_builder");
381 const OFFSET_FACTOR: f32 = 1.0 / 12.0;
382 let offset = size.height * OFFSET_FACTOR;
383 let center_offset = offset * 3.0;
384 let left_top = 0.4 * size.height;
385 let right_top = 0.25 * size.height;
386 let center = 0.3 * size.width;
387 let bottom = 0.75 * size.height;
388 let left = 0.05;
389 let right = size.width - 0.05;
390 path_builder
391 .move_to(point2(left, left_top + offset))
392 .line_to(point2(left + offset, left_top))
393 .line_to(point2(center, bottom - center_offset))
394 .line_to(point2(right - offset, right_top))
395 .line_to(point2(right, right_top + offset))
396 .line_to(point2(center, bottom))
397 .line_to(point2(left, left_top + offset));
398 path_builder.build()
399}
400
401fn path_for_unicode_2718(size: &Size, render_context: &mut RenderContext) -> Path {
403 let mut path_builder = render_context.path_builder().expect("path_builder");
404 const OFFSET_FACTOR: f32 = 1.0 / 12.0;
405 let offset = size.height * OFFSET_FACTOR;
406 let center_offset = offset;
407 let top = 0.25 * size.height;
408 let center_x = size.width / 2.0;
409 let center_y = size.height / 2.0;
410 let bottom = 0.75 * size.height;
411 let left = 0.05;
412 let right = size.width - 0.05;
413 path_builder
414 .move_to(point2(left, top + offset))
415 .line_to(point2(left + offset, top))
416 .line_to(point2(center_x, center_y - center_offset))
417 .line_to(point2(right - offset, top))
418 .line_to(point2(right, top + offset))
419 .line_to(point2(center_x + center_offset, center_y))
420 .line_to(point2(right, bottom - offset))
421 .line_to(point2(right - offset, bottom))
422 .line_to(point2(center_x, center_y + center_offset))
423 .line_to(point2(left + offset, bottom))
424 .line_to(point2(left, bottom - offset))
425 .line_to(point2(center_x - center_offset, center_y))
426 .line_to(point2(left, top + offset));
427 path_builder.build()
428}
429
430fn path_for_unicode_276e(size: &Size, render_context: &mut RenderContext) -> Path {
432 let mut path_builder = render_context.path_builder().expect("path_builder");
433 const THICKNESS_FACTOR: f32 = 1.0 / 8.0;
434 let thickness = size.height * THICKNESS_FACTOR;
435 let top = 0.25 * size.height;
436 let bottom = 0.85 * size.height;
437 let left = 0.2 * size.width;
438 let right = 0.8 * size.width;
439 let center_y = top + (bottom - top) / 2.0;
440 path_builder
441 .move_to(point2(right - thickness, top))
442 .line_to(point2(right, top))
443 .line_to(point2(left + thickness, center_y))
444 .line_to(point2(right, bottom))
445 .line_to(point2(right - thickness, bottom))
446 .line_to(point2(left, center_y))
447 .line_to(point2(right - thickness, top));
448 path_builder.build()
449}
450
451fn path_for_unicode_276f(size: &Size, render_context: &mut RenderContext) -> Path {
453 let mut path_builder = render_context.path_builder().expect("path_builder");
454 const THICKNESS_FACTOR: f32 = 1.0 / 8.0;
455 let thickness = size.height * THICKNESS_FACTOR;
456 let top = 0.25 * size.height;
457 let bottom = 0.85 * size.height;
458 let left = 0.2 * size.width;
459 let right = 0.8 * size.width;
460 let center_y = top + (bottom - top) / 2.0;
461 path_builder
462 .move_to(point2(left, top))
463 .line_to(point2(left + thickness, top))
464 .line_to(point2(right, center_y))
465 .line_to(point2(left + thickness, bottom))
466 .line_to(point2(left, bottom))
467 .line_to(point2(right - thickness, center_y))
468 .line_to(point2(left, top));
469 path_builder.build()
470}
471
472fn path_for_unicode_e0b0(size: &Size, render_context: &mut RenderContext) -> Path {
474 let mut path_builder = render_context.path_builder().expect("path_builder");
475 path_builder
476 .move_to(point2(0.0, 0.0))
477 .line_to(point2(size.width, size.height / 2.0))
478 .line_to(point2(0.0, size.height))
479 .line_to(point2(0.0, 0.0));
480 path_builder.build()
481}
482
483fn path_for_unicode_e0b1(size: &Size, render_context: &mut RenderContext) -> Path {
485 let mut path_builder = render_context.path_builder().expect("path_builder");
486 let thickness = size.height * LINE_THICKNESS_FACTOR;
488 let offset = (2.0 * thickness * thickness).sqrt() / 2.0;
489 path_builder
490 .move_to(point2(offset, 0.0))
491 .line_to(point2(size.width + offset, size.height / 2.0))
492 .line_to(point2(offset, size.height))
493 .line_to(point2(-offset, size.height))
494 .line_to(point2(size.width - offset, size.height / 2.0))
495 .line_to(point2(-offset, 0.0))
496 .line_to(point2(offset, 0.0));
497 path_builder.build()
498}
499
500fn path_for_unicode_e0b2(size: &Size, render_context: &mut RenderContext) -> Path {
502 let mut path_builder = render_context.path_builder().expect("path_builder");
503 path_builder
504 .move_to(point2(size.width, 0.0))
505 .line_to(point2(size.width, size.height))
506 .line_to(point2(0.0, size.height / 2.0))
507 .line_to(point2(size.width, 0.0));
508 path_builder.build()
509}
510
511fn path_for_unicode_e0b3(size: &Size, render_context: &mut RenderContext) -> Path {
513 let mut path_builder = render_context.path_builder().expect("path_builder");
514 let thickness = size.height * LINE_THICKNESS_FACTOR;
516 let offset = (2.0 * thickness * thickness).sqrt() / 2.0;
517 path_builder
518 .move_to(point2(size.width + offset, 0.0))
519 .line_to(point2(offset, size.height / 2.0))
520 .line_to(point2(size.width + offset, size.height))
521 .line_to(point2(size.width - offset, size.height))
522 .line_to(point2(-offset, size.height / 2.0))
523 .line_to(point2(size.width - offset, 0.0))
524 .line_to(point2(size.width + offset, 0.0));
525 path_builder.build()
526}
527
528fn path_for_unicode_e0ba(size: &Size, render_context: &mut RenderContext) -> Path {
530 let mut path_builder = render_context.path_builder().expect("path_builder");
531 path_builder
532 .move_to(point2(size.width, 0.0))
533 .line_to(point2(size.width, size.height))
534 .line_to(point2(0.0, size.height))
535 .line_to(point2(size.width, 0.0));
536 path_builder.build()
537}
538
539fn path_for_unicode_e0bc(size: &Size, render_context: &mut RenderContext) -> Path {
541 let mut path_builder = render_context.path_builder().expect("path_builder");
542 path_builder
543 .move_to(point2(0.0, 0.0))
544 .line_to(point2(size.width, 0.0))
545 .line_to(point2(0.0, size.height))
546 .line_to(point2(0.0, 0.0));
547 path_builder.build()
548}
549
550pub fn maybe_path_for_cursor_style(
551 render_context: &mut RenderContext,
552 cursor_style: CursorStyle,
553 cell_size: &Size,
554) -> Option<Path> {
555 match cursor_style {
556 CursorStyle::Block => Some(path_for_block(cell_size, render_context)),
557 CursorStyle::Underline => Some(path_for_underline(cell_size, render_context, None)),
558 CursorStyle::Beam => Some(path_for_beam(cell_size, render_context)),
559 CursorStyle::HollowBlock => Some(path_for_hollow_block(cell_size, render_context)),
560 CursorStyle::Hidden => None,
561 }
562}
563
564pub fn maybe_path_for_char(
565 render_context: &mut RenderContext,
566 c: char,
567 cell_size: &Size,
568) -> Option<Path> {
569 match c {
570 '\u{2500}' => Some(path_for_unicode_2500(cell_size, render_context)),
571 '\u{2502}' => Some(path_for_unicode_2502(cell_size, render_context)),
572 '\u{256d}' => Some(path_for_unicode_256d(cell_size, render_context)),
573 '\u{256e}' => Some(path_for_unicode_256e(cell_size, render_context)),
574 '\u{256f}' => Some(path_for_unicode_256f(cell_size, render_context)),
575 '\u{2570}' => Some(path_for_unicode_2570(cell_size, render_context)),
576 '\u{2591}' => Some(path_for_unicode_2591(cell_size, render_context)),
577 '\u{2592}' => Some(path_for_unicode_2592(cell_size, render_context)),
578 '\u{2593}' => Some(path_for_unicode_2593(cell_size, render_context)),
579 '\u{2714}' => Some(path_for_unicode_2714(cell_size, render_context)),
580 '\u{2718}' => Some(path_for_unicode_2718(cell_size, render_context)),
581 '\u{276e}' => Some(path_for_unicode_276e(cell_size, render_context)),
582 '\u{276f}' => Some(path_for_unicode_276f(cell_size, render_context)),
583 '\u{e0b0}' => Some(path_for_unicode_e0b0(cell_size, render_context)),
584 '\u{e0b1}' => Some(path_for_unicode_e0b1(cell_size, render_context)),
585 '\u{e0b2}' => Some(path_for_unicode_e0b2(cell_size, render_context)),
586 '\u{e0b3}' => Some(path_for_unicode_e0b3(cell_size, render_context)),
587 '\u{e0ba}' => Some(path_for_unicode_e0ba(cell_size, render_context)),
588 '\u{e0bc}' => Some(path_for_unicode_e0bc(cell_size, render_context)),
589 _ => None,
590 }
591}
592
593#[cfg(test)]
594mod tests {
595 use super::*;
596 use anyhow::Error;
597 use carnelian::drawing::{DisplayRotation, FontFace};
598 use carnelian::render::{generic, ContextInner};
599 use euclid::size2;
600
601 static FONT_DATA: &'static [u8] = include_bytes!(
604 "../../../../../prebuilt/third_party/fonts/robotomono/RobotoMono-Regular.ttf"
605 );
606 static FONT_FACE: std::sync::LazyLock<FontFace> =
607 std::sync::LazyLock::new(|| FontFace::new(&FONT_DATA).expect("Failed to create font"));
608
609 #[test]
610 fn check_cursor_paths() -> Result<(), Error> {
611 const SUPPORTED_CURSOR_STYLES: &[CursorStyle] = &[
612 CursorStyle::Block,
613 CursorStyle::Underline,
614 CursorStyle::Beam,
615 CursorStyle::HollowBlock,
616 ];
617 let size = size2(64, 64);
618 let forma_context = generic::Forma::new_context_without_token(size, DisplayRotation::Deg0);
619 let mut render_context = RenderContext { inner: ContextInner::Forma(forma_context) };
620 let cell_size = Size::new(8.0, 16.0);
621 for cursor_style in SUPPORTED_CURSOR_STYLES {
622 let result =
623 maybe_path_for_cursor_style(&mut render_context, *cursor_style, &cell_size);
624 assert_eq!(result.is_some(), true);
625 }
626 Ok(())
627 }
628
629 #[test]
630 fn check_strikeout_path() -> Result<(), Error> {
631 let size = size2(64, 64);
632 let forma_context = generic::Forma::new_context_without_token(size, DisplayRotation::Deg0);
633 let mut render_context = RenderContext { inner: ContextInner::Forma(forma_context) };
634 let cell_size = Size::new(8.0, 16.0);
635 let textgrid = TextGrid::new(&FONT_FACE, &cell_size);
636 let _ = path_for_strikeout(
637 &cell_size,
638 &mut render_context,
639 FONT_FACE
640 .face
641 .strikeout_metrics()
642 .map(|line_metrics| Line::new(line_metrics, &textgrid)),
643 );
644 Ok(())
645 }
646
647 #[test]
648 fn check_underline_path() -> Result<(), Error> {
649 let size = size2(64, 64);
650 let forma_context = generic::Forma::new_context_without_token(size, DisplayRotation::Deg0);
651 let mut render_context = RenderContext { inner: ContextInner::Forma(forma_context) };
652 let cell_size = Size::new(8.0, 16.0);
653 let textgrid = TextGrid::new(&FONT_FACE, &cell_size);
654 let _ = path_for_underline(
655 &cell_size,
656 &mut render_context,
657 FONT_FACE
658 .face
659 .underline_metrics()
660 .map(|line_metrics| Line::new(line_metrics, &textgrid)),
661 );
662 Ok(())
663 }
664
665 #[test]
666 fn check_unicode_paths() -> Result<(), Error> {
667 const SUPPORTED_UNICODE_CHARS: &[char] = &[
668 '\u{2500}', '\u{2502}', '\u{256d}', '\u{256e}', '\u{256f}', '\u{2570}', '\u{2591}',
669 '\u{2592}', '\u{2593}', '\u{2714}', '\u{2718}', '\u{276e}', '\u{276f}', '\u{e0b0}',
670 '\u{e0b1}', '\u{e0b2}', '\u{e0b3}', '\u{e0ba}', '\u{e0bc}',
671 ];
672 let size = size2(64, 64);
673 let forma_context = generic::Forma::new_context_without_token(size, DisplayRotation::Deg0);
674 let mut render_context = RenderContext { inner: ContextInner::Forma(forma_context) };
675 let cell_size = Size::new(8.0, 16.0);
676 for c in SUPPORTED_UNICODE_CHARS {
677 let result = maybe_path_for_char(&mut render_context, *c, &cell_size);
678 assert_eq!(result.is_some(), true);
679 }
680 Ok(())
681 }
682}