terminal/
paths.rs

1// Copyright 2022 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5use carnelian::drawing::TextGrid;
6use carnelian::render::{Context as RenderContext, Path};
7use carnelian::Size;
8use euclid::{point2, vec2};
9use term_model::ansi::CursorStyle;
10
11// Thickness of lines is determined by multiplying thickness factor
12// with the cell height. 1/16 has been chosen as that results in 1px
13// thick lines for a 16px cell height.
14const 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        // top
94        .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        // bottom
100        .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        // left
106        .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        // right
112        .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
139// Box Drawings Light Horizontal, "─".
140fn 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
154// Box Drawings Light Vertical, "│".
155fn 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
169// Box Drawings Light Arc Down and Right, "╭".
170fn 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
201// Box Drawings Light Arc Down and Left, "╮".
202fn 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
233// Box Drawings Light Arc Up and Left, "╯".
234fn 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
265// Box Drawings Light Arc Up and Right, "╰".
266fn 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
297// Light Shade, "░".
298fn 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            // Fill every fourth grid cell.
307            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
324// Medium Shade, "▒".
325fn 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            // Fill every other grid cell.
334            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
351// Dark Shade, "▓".
352fn 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            // Skip every fourth grid cell.
361            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
378// Heavy Check Mark, "✔".
379fn 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
401// Heavy Ballot X, "✘".
402fn 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
430// Heavy Left-Pointing Angle Quotation Mark Ornament, "❮".
431fn 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
451// Heavy Right-Pointing Angle Quotation Mark Ornament, "❯".
452fn 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
472// Triangle right, "".
473fn 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
483// Angle right, "".
484fn path_for_unicode_e0b1(size: &Size, render_context: &mut RenderContext) -> Path {
485    let mut path_builder = render_context.path_builder().expect("path_builder");
486    // Requires 45 degree angle for correct thickness.
487    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
500// Triangle left, "".
501fn 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
511// Angle left, "".
512fn path_for_unicode_e0b3(size: &Size, render_context: &mut RenderContext) -> Path {
513    let mut path_builder = render_context.path_builder().expect("path_builder");
514    // Requires 45 degree angle for correct thickness.
515    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
528// Lower right triangle, "".
529fn 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
539// Upper left triangle, "".
540fn 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    // This font creation method isn't ideal. The correct method would be to ask the Fuchsia
602    // font service for the font data.
603    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}