fix crash in json encoding
This commit is contained in:
parent
7c03386380
commit
b893835866
|
@ -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 {
|
||||||
|
@ -26,13 +29,15 @@
|
||||||
contours: Contour[]
|
contours: Contour[]
|
||||||
};
|
};
|
||||||
|
|
||||||
let glyphData: GlyphData= $state({
|
let glyphData: GlyphData = $state({
|
||||||
bounding_box: { xmin: 0, xmax: 0, ymin: 0, ymax: 0 },
|
bounding_box: { xmin: 0, xmax: 0, ymin: 0, ymax: 0 },
|
||||||
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>
|
||||||
|
|
|
@ -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 });
|
||||||
|
|
|
@ -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();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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();
|
||||||
|
|
Loading…
Reference in a new issue