fix crash in json encoding

This commit is contained in:
lauchmelder 2025-03-06 23:55:12 +01:00
parent 7c03386380
commit b893835866
4 changed files with 66 additions and 5 deletions

View file

@ -8,7 +8,10 @@
interface ContourPoint { interface ContourPoint {
x: number, x: number,
y: number y: number,
on_curve: boolean,
is_virtual: boolean,
}; };
interface ContourElement { interface ContourElement {
@ -31,8 +34,10 @@
contours: [] contours: []
}); });
let pointTooltip: string = $state("");
function getPathString(contour: Contour): string { function getPathString(contour: Contour): string {
if (contour.elements?.length === 0) { if (contour.elements?.length == 0 || contour.points?.length == 0) {
return ""; return "";
} }
@ -45,7 +50,11 @@
return ""; return "";
} }
let result = `M${contour.points[firstPoint].x} -${contour.points[firstPoint].y} ` if (firstPoint < 0 || firstPoint >= contour.points.length) {
return "";
}
let result = `M${contour.points[firstPoint].x} -${contour.points[firstPoint]?.y} `
contour.elements.forEach((element) => { contour.elements.forEach((element) => {
if (element.Bezier) { if (element.Bezier) {
let control = contour.points[element.Bezier[1]]; let control = contour.points[element.Bezier[1]];
@ -62,6 +71,31 @@
return result; return result;
} }
function getViewBox(data: GlyphData): string {
const padding = 0.1;
let width = data.bounding_box.xmax - data.bounding_box.xmin;
let height = data.bounding_box.ymax - data.bounding_box.ymin;
let bbox: BoundingBox = {
xmin: data.bounding_box.xmin - padding * width,
xmax: (data.bounding_box.xmax - data.bounding_box.xmin) + 2 * padding * width,
ymin: (-data.bounding_box.ymax) - padding * width,
ymax: (data.bounding_box.ymax - data.bounding_box.ymin) + 2 * padding * width
};
return `${bbox.xmin} ${bbox.ymin} ${bbox.xmax} ${bbox.ymax}`;
}
function getFillColor(point: ContourPoint): string {
if (point.on_curve) {
return "blue";
} else if (point.is_virtual) {
return "green";
}
return "red";
}
let { selectedChar = "", loading = $bindable() } = $props(); let { selectedChar = "", loading = $bindable() } = $props();
$effect(() => { $effect(() => {
@ -87,8 +121,10 @@
</script> </script>
<div class="glyph"> <div class="glyph">
<p>{ pointTooltip }</p>
<svg <svg
viewBox="{glyphData.bounding_box.xmin} -{glyphData.bounding_box.ymax} {glyphData.bounding_box.xmax - glyphData.bounding_box.xmin} {glyphData.bounding_box.ymax - glyphData.bounding_box.ymin}" viewBox="{ getViewBox(glyphData) }"
width="1000" width="1000"
height="1000" height="1000"
> >
@ -98,6 +134,15 @@
style="fill:none; stroke:black; stroke-width: 3" style="fill:none; stroke:black; stroke-width: 3"
d="{getPathString(contour)}" d="{getPathString(contour)}"
/> />
{#each contour.points as point}
<circle
class="glyph-point"
cx="{point.x}" cy="{-point.y}"
fill="{getFillColor(point)}"
onclick="{ () => { pointTooltip = JSON.stringify(point); } }"
/>
{/each}
{/each} {/each}
</svg> </svg>
</div> </div>
@ -107,4 +152,14 @@
max-height: 50vh; max-height: 50vh;
max-width: 50vw; max-width: 50vw;
} }
.glyph-point {
r: 13;
transition: ease-out 0.2s r;
}
.glyph-point:hover {
r: 20;
transition: ease-out 0.2s r;
}
</style> </style>

View file

@ -287,6 +287,7 @@ impl<'a> SplineIter<'a> {
if point.on_curve { if point.on_curve {
self.last_point = Some(IndexedGlyphPoint { point, index }); self.last_point = Some(IndexedGlyphPoint { point, index });
self.first_point = Some(IndexedGlyphPoint { point, index }); self.first_point = Some(IndexedGlyphPoint { point, index });
self.final_control = None;
return Ok(()); return Ok(());
} else { } else {
self.final_control = Some(IndexedGlyphPoint { point, index }); self.final_control = Some(IndexedGlyphPoint { point, index });

View file

@ -24,6 +24,8 @@ pub trait Visitor: Sized {
self.write_spline(&spline_element) self.write_spline(&spline_element)
}); });
debug!("done writing splines");
self.write_suffix(); self.write_suffix();
} }
} }

View file

@ -1,5 +1,6 @@
use std::io::Write; use std::io::Write;
use log::debug;
use serde::Serialize; use serde::Serialize;
use crate::font::{GlyphHeader, GlyphPoint, SplineElement, SplineElementData}; use crate::font::{GlyphHeader, GlyphPoint, SplineElement, SplineElementData};
@ -85,11 +86,13 @@ impl<W: Sized + Write> Visitor for JsonWriter<W> {
} }
fn write_spline(&mut self, spline: &SplineElement) { fn write_spline(&mut self, spline: &SplineElement) {
debug!("index correction: {}", self.index_correction);
let indexed_spline = match spline.data { let indexed_spline = match spline.data {
SplineElementData::Line(start, end) => ContourElement::Line(start.index - self.index_correction, end.index - self.index_correction), SplineElementData::Line(start, end) => ContourElement::Line(start.index - self.index_correction, end.index - self.index_correction),
SplineElementData::Bezier(start, control, end) => ContourElement::Bezier(start.index - self.index_correction, control.index - self.index_correction, end.index - self.index_correction) SplineElementData::Bezier(start, control, end) => ContourElement::Bezier(start.index - self.index_correction, control.index - self.index_correction, end.index - self.index_correction)
}; };
debug!("pushing spline to contours array at position {}/{}", self.current_contour, self.body.contours.len());
self.body.contours[self.current_contour].elements.push(indexed_spline); self.body.contours[self.current_contour].elements.push(indexed_spline);
if spline.is_last { if spline.is_last {
self.index_correction = self.body.contours[self.current_contour].points.len(); self.index_correction = self.body.contours[self.current_contour].points.len();