typst/tests/typeset.rs

674 lines
21 KiB
Rust
Raw Normal View History

use std::env;
use std::ffi::OsStr;
use std::fs;
use std::path::Path;
use std::rc::Rc;
2020-11-30 22:07:08 +01:00
use image::{GenericImageView, Rgba};
2021-07-08 22:33:44 +02:00
use tiny_skia as sk;
use ttf_parser::{GlyphId, OutlineBuilder};
use walkdir::WalkDir;
use typst::color::{Color, RgbaColor};
use typst::diag::Error;
use typst::eval::{State, Value};
2021-08-27 13:29:50 +02:00
use typst::font::Face;
2021-07-08 22:33:44 +02:00
use typst::geom::{self, Length, PathElement, Point, Sides, Size};
2021-08-27 13:29:50 +02:00
use typst::image::Image;
#[cfg(feature = "layout-cache")]
use typst::layout::LayoutTree;
use typst::layout::{layout, Element, Frame, Geometry, Paint, Text};
2021-08-09 11:06:37 +02:00
use typst::loading::FsLoader;
use typst::parse::Scanner;
use typst::source::SourceFile;
2021-08-13 12:21:14 +02:00
use typst::syntax::{Pos, Span};
2021-07-20 20:21:56 +02:00
use typst::Context;
2021-02-20 17:53:40 +01:00
const TYP_DIR: &str = "./typ";
const REF_DIR: &str = "./ref";
const PNG_DIR: &str = "./png";
const PDF_DIR: &str = "./pdf";
const FONT_DIR: &str = "../fonts";
fn main() {
env::set_current_dir(env::current_dir().unwrap().join("tests")).unwrap();
let args = Args::new(env::args().skip(1));
let mut filtered = Vec::new();
for entry in WalkDir::new(".").into_iter() {
let entry = entry.unwrap();
if entry.depth() <= 1 {
continue;
}
let src_path = entry.into_path();
if src_path.extension() != Some(OsStr::new("typ")) {
continue;
}
if args.matches(&src_path.to_string_lossy()) {
filtered.push(src_path);
}
}
let len = filtered.len();
2021-01-13 21:33:22 +01:00
if len == 1 {
println!("Running test ...");
2021-01-13 21:33:22 +01:00
} else if len > 1 {
println!("Running {} tests", len);
}
2021-07-20 20:21:56 +02:00
// We want to have "unbounded" pages, so we allow them to be infinitely
// large and fit them to match their content.
let mut state = State::default();
2021-07-29 13:21:25 +02:00
let page = state.page_mut();
page.size = Size::new(Length::pt(120.0), Length::inf());
page.margins = Sides::splat(Some(Length::pt(10.0).into()));
2021-07-20 20:21:56 +02:00
// Hook up an assert function into the global scope.
2021-07-21 11:25:49 +02:00
let mut std = typst::library::new();
std.def_func("test", move |_, args| {
let lhs = args.expect::<Value>("left-hand side")?;
let rhs = args.expect::<Value>("right-hand side")?;
if lhs != rhs {
2021-08-12 13:39:33 +02:00
return Err(Error::boxed(
args.span,
2021-08-12 13:39:33 +02:00
format!("Assertion failed: {:?} != {:?}", lhs, rhs),
));
}
Ok(Value::None)
});
2021-07-21 11:25:49 +02:00
// Create loader and context.
let loader = FsLoader::new().with_path(FONT_DIR).wrap();
2021-08-09 11:06:37 +02:00
let mut ctx = Context::builder().std(std).state(state).build(loader);
2021-07-21 11:25:49 +02:00
// Run all the tests.
2021-01-13 21:33:22 +01:00
let mut ok = true;
for src_path in filtered {
let path = src_path.strip_prefix(TYP_DIR).unwrap();
let png_path = Path::new(PNG_DIR).join(path).with_extension("png");
let ref_path = Path::new(REF_DIR).join(path).with_extension("png");
let pdf_path =
args.pdf.then(|| Path::new(PDF_DIR).join(path).with_extension("pdf"));
ok &= test(
2021-07-20 20:21:56 +02:00
&mut ctx,
&src_path,
&png_path,
&ref_path,
pdf_path.as_deref(),
);
2021-01-13 14:07:38 +01:00
}
if !ok {
std::process::exit(1);
}
}
struct Args {
filter: Vec<String>,
pdf: bool,
perfect: bool,
}
impl Args {
fn new(args: impl Iterator<Item = String>) -> Self {
let mut filter = Vec::new();
let mut perfect = false;
let mut pdf = false;
for arg in args {
match arg.as_str() {
"--nocapture" => {}
"--pdf" => pdf = true,
"=" => perfect = true,
_ => filter.push(arg),
}
}
Self { filter, pdf, perfect }
}
fn matches(&self, name: &str) -> bool {
if self.perfect {
self.filter.iter().any(|p| name == p)
} else {
2020-08-30 22:18:55 +02:00
self.filter.is_empty() || self.filter.iter().any(|p| name.contains(p))
}
}
}
fn test(
2021-07-20 20:21:56 +02:00
ctx: &mut Context,
src_path: &Path,
png_path: &Path,
2021-02-20 17:53:40 +01:00
ref_path: &Path,
pdf_path: Option<&Path>,
) -> bool {
let name = src_path.strip_prefix(TYP_DIR).unwrap_or(src_path);
println!("Testing {}", name.display());
let src = fs::read_to_string(src_path).unwrap();
2021-01-13 14:07:38 +01:00
let mut ok = true;
let mut frames = vec![];
let mut line = 0;
2021-01-31 22:43:11 +01:00
let mut compare_ref = true;
let mut compare_ever = false;
2021-01-31 22:43:11 +01:00
2021-06-29 16:05:05 +02:00
let parts: Vec<_> = src.split("\n---").collect();
for (i, &part) in parts.iter().enumerate() {
2021-01-31 22:43:11 +01:00
let is_header = i == 0
&& parts.len() > 1
&& part
.lines()
.all(|s| s.starts_with("//") || s.chars().all(|c| c.is_whitespace()));
if is_header {
for line in part.lines() {
if line.starts_with("// Ref: false") {
compare_ref = false;
}
}
} else {
let (part_ok, compare_here, part_frames) =
2021-08-09 11:06:37 +02:00
test_part(ctx, src_path, part.into(), i, compare_ref, line);
2021-01-31 22:43:11 +01:00
ok &= part_ok;
compare_ever |= compare_here;
2021-01-31 22:43:11 +01:00
frames.extend(part_frames);
}
line += part.lines().count() + 1;
2021-01-13 14:07:38 +01:00
}
if compare_ever {
if let Some(pdf_path) = pdf_path {
2021-07-20 20:21:56 +02:00
let pdf_data = typst::export::pdf(ctx, &frames);
fs::create_dir_all(&pdf_path.parent().unwrap()).unwrap();
fs::write(pdf_path, pdf_data).unwrap();
}
2021-01-13 14:07:38 +01:00
2021-07-20 20:21:56 +02:00
let canvas = draw(ctx, &frames, 2.0);
fs::create_dir_all(&png_path.parent().unwrap()).unwrap();
2021-03-23 13:17:00 +01:00
canvas.save_png(png_path).unwrap();
2021-01-13 23:19:44 +01:00
2021-07-08 22:33:44 +02:00
if let Ok(ref_pixmap) = sk::Pixmap::load_png(ref_path) {
2021-03-23 13:17:00 +01:00
if canvas != ref_pixmap {
2021-02-20 17:53:40 +01:00
println!(" Does not match reference image. ❌");
2021-01-13 14:07:38 +01:00
ok = false;
}
} else if !frames.is_empty() {
2021-02-20 17:53:40 +01:00
println!(" Failed to open reference image. ❌");
ok = false;
2021-01-13 14:07:38 +01:00
}
}
if ok {
println!("\x1b[1ATesting {}", name.display());
2021-01-13 14:07:38 +01:00
}
ok
}
2021-01-31 22:43:11 +01:00
fn test_part(
2021-07-20 20:21:56 +02:00
ctx: &mut Context,
2021-08-09 11:06:37 +02:00
src_path: &Path,
src: String,
2021-01-31 22:43:11 +01:00
i: usize,
compare_ref: bool,
line: usize,
2021-06-18 13:01:55 +02:00
) -> (bool, bool, Vec<Rc<Frame>>) {
2021-08-09 11:06:37 +02:00
let id = ctx.sources.provide(src_path, src);
let source = ctx.sources.get(id);
let (local_compare_ref, mut ref_errors) = parse_metadata(&source);
2021-01-31 22:43:11 +01:00
let compare_ref = local_compare_ref.unwrap_or(compare_ref);
let mut ok = true;
let (frames, mut errors) = match ctx.execute(id) {
Ok(tree) => {
let mut frames = layout(ctx, &tree);
2021-06-18 13:00:36 +02:00
#[cfg(feature = "layout-cache")]
(ok &= test_incremental(ctx, i, &tree, &frames));
2021-06-27 18:06:39 +02:00
if !compare_ref {
frames.clear();
}
(frames, vec![])
}
Err(errors) => (vec![], *errors),
};
// TODO: Also handle errors from other files.
2021-08-13 12:21:14 +02:00
errors.retain(|error| error.span.source == id);
for error in &mut errors {
error.trace.clear();
}
2021-08-21 16:38:51 +02:00
// The comparison never fails since all spans are from the same source file.
ref_errors.sort_by(|a, b| a.span.partial_cmp(&b.span).unwrap());
errors.sort_by(|a, b| a.span.partial_cmp(&b.span).unwrap());
2021-07-21 11:25:49 +02:00
if errors != ref_errors {
println!(" Subtest {} does not match expected errors. ❌", i);
ok = false;
2021-08-09 11:06:37 +02:00
let source = ctx.sources.get(id);
for error in errors.iter() {
2021-08-13 12:21:14 +02:00
if error.span.source == id && !ref_errors.contains(error) {
print!(" Not annotated | ");
print_error(&source, line, error);
}
}
for error in ref_errors.iter() {
if !errors.contains(error) {
print!(" Not emitted | ");
print_error(&source, line, error);
}
}
}
(ok, compare_ref, frames)
}
#[cfg(feature = "layout-cache")]
fn test_incremental(
ctx: &mut Context,
i: usize,
tree: &LayoutTree,
frames: &[Rc<Frame>],
) -> bool {
let mut ok = true;
let reference = ctx.layouts.clone();
for level in 0 .. reference.levels() {
ctx.layouts = reference.clone();
ctx.layouts.retain(|x| x == level);
if ctx.layouts.is_empty() {
continue;
}
2021-07-20 20:21:56 +02:00
ctx.layouts.turnaround();
let cached = layout(ctx, tree);
let misses = ctx
.layouts
.entries()
.filter(|e| e.level() == level && !e.hit() && e.age() == 2)
.count();
if misses > 0 {
println!(
" Subtest {} relayout had {} cache misses on level {} ❌",
i, misses, level
);
ok = false;
}
if cached != frames {
println!(" Subtest {} relayout differs from clean pass ❌", i);
ok = false;
}
2021-06-27 18:06:39 +02:00
}
ctx.layouts = reference;
ctx.layouts.turnaround();
ok
}
fn parse_metadata(source: &SourceFile) -> (Option<bool>, Vec<Error>) {
2021-01-31 22:43:11 +01:00
let mut compare_ref = None;
let mut errors = vec![];
let lines: Vec<_> = source.src().lines().map(str::trim).collect();
for (i, line) in lines.iter().enumerate() {
2021-01-31 22:43:11 +01:00
if line.starts_with("// Ref: false") {
compare_ref = Some(false);
}
if line.starts_with("// Ref: true") {
compare_ref = Some(true);
}
2021-01-13 14:07:38 +01:00
let rest = if let Some(rest) = line.strip_prefix("// Error: ") {
rest
} else {
continue;
};
fn num(s: &mut Scanner) -> usize {
s.eat_while(|c| c.is_numeric()).parse().unwrap()
}
let comments =
lines[i ..].iter().take_while(|line| line.starts_with("//")).count();
let pos = |s: &mut Scanner| -> Pos {
let first = num(s) - 1;
let (delta, column) =
if s.eat_if(':') { (first, num(s) - 1) } else { (0, first) };
let line = (i + comments) + delta;
2021-08-13 12:21:14 +02:00
source.line_column_to_byte(line, column).unwrap().into()
};
2021-01-13 14:07:38 +01:00
let mut s = Scanner::new(rest);
2021-02-17 21:30:20 +01:00
let start = pos(&mut s);
let end = if s.eat_if('-') { pos(&mut s) } else { start };
2021-08-13 12:21:14 +02:00
let span = Span::new(source.id(), start, end);
2021-02-17 21:30:20 +01:00
2021-08-13 12:21:14 +02:00
errors.push(Error::new(span, s.rest().trim()));
}
(compare_ref, errors)
}
fn print_error(source: &SourceFile, line: usize, error: &Error) {
2021-08-13 12:21:14 +02:00
let start_line = 1 + line + source.byte_to_line(error.span.start.to_usize()).unwrap();
let start_col = 1 + source.byte_to_column(error.span.start.to_usize()).unwrap();
let end_line = 1 + line + source.byte_to_line(error.span.end.to_usize()).unwrap();
let end_col = 1 + source.byte_to_column(error.span.end.to_usize()).unwrap();
println!(
"Error: {}:{}-{}:{}: {}",
start_line, start_col, end_line, end_col, error.message
);
}
2021-08-27 13:29:50 +02:00
fn draw(ctx: &Context, frames: &[Rc<Frame>], dpp: f32) -> sk::Pixmap {
2020-11-25 18:46:47 +01:00
let pad = Length::pt(5.0);
let width = 2.0 * pad + frames.iter().map(|l| l.size.w).max().unwrap_or_default();
let height = pad + frames.iter().map(|l| l.size.h + pad).sum::<Length>();
2020-08-30 22:18:55 +02:00
2021-08-27 13:29:50 +02:00
let pixel_width = (dpp * width.to_pt() as f32) as u32;
let pixel_height = (dpp * height.to_pt() as f32) as u32;
2021-01-13 23:19:44 +01:00
if pixel_width > 4000 || pixel_height > 4000 {
panic!(
"overlarge image: {} by {} ({:?} x {:?})",
pixel_width, pixel_height, width, height,
);
2021-01-13 23:19:44 +01:00
}
2021-07-08 22:33:44 +02:00
let mut canvas = sk::Pixmap::new(pixel_width, pixel_height).unwrap();
2021-08-27 13:29:50 +02:00
let ts = sk::Transform::from_scale(dpp, dpp);
2021-07-08 22:33:44 +02:00
canvas.fill(sk::Color::BLACK);
2021-05-17 22:55:31 +02:00
let mut origin = Point::splat(pad);
2021-01-03 00:12:09 +01:00
for frame in frames {
2021-07-08 22:33:44 +02:00
let mut paint = sk::Paint::default();
paint.set_color(sk::Color::WHITE);
2020-11-25 18:46:47 +01:00
canvas.fill_rect(
2021-07-08 22:33:44 +02:00
sk::Rect::from_xywh(
2020-11-25 18:46:47 +01:00
origin.x.to_pt() as f32,
origin.y.to_pt() as f32,
frame.size.w.to_pt() as f32,
frame.size.h.to_pt() as f32,
2020-11-25 18:46:47 +01:00
)
.unwrap(),
&paint,
2021-03-23 13:17:00 +01:00
ts,
None,
);
2021-06-18 13:01:55 +02:00
for (pos, element) in frame.elements() {
let global = origin + pos;
let x = global.x.to_pt() as f32;
let y = global.y.to_pt() as f32;
let ts = ts.pre_translate(x, y);
match *element {
Element::Text(ref text) => {
2021-08-27 13:29:50 +02:00
draw_text(&mut canvas, ts, ctx.fonts.get(text.face_id), text);
}
2021-07-08 22:33:44 +02:00
Element::Geometry(ref geometry, paint) => {
draw_geometry(&mut canvas, ts, geometry, paint);
}
Element::Image(id, size) => {
2021-08-27 13:29:50 +02:00
draw_image(&mut canvas, ts, ctx.images.get(id), size);
}
Element::Link(_, s) => {
let outline = Geometry::Rect(s);
let paint = Paint::Color(Color::Rgba(RgbaColor::new(40, 54, 99, 40)));
draw_geometry(&mut canvas, ts, &outline, paint);
}
}
}
origin.y += frame.size.h + pad;
}
2020-11-25 18:46:47 +01:00
canvas
}
2021-08-27 13:29:50 +02:00
fn draw_text(canvas: &mut sk::Pixmap, ts: sk::Transform, face: &Face, text: &Text) {
let ttf = face.ttf();
let size = text.size.to_pt() as f32;
let units_per_em = ttf.units_per_em() as f32;
let pixels_per_em = text.size.to_pt() as f32 * ts.sy;
let scale = size / units_per_em;
2021-08-27 13:29:50 +02:00
let mut x = 0.0;
for glyph in &text.glyphs {
2021-08-27 13:29:50 +02:00
let glyph_id = GlyphId(glyph.id);
let offset = x + glyph.x_offset.to_length(text.size).to_pt() as f32;
let ts = ts.pre_translate(offset, 0.0);
2020-11-25 18:46:47 +01:00
if let Some(tree) = ttf
2021-08-27 13:29:50 +02:00
.glyph_svg_image(glyph_id)
2021-03-23 15:32:36 +01:00
.and_then(|data| std::str::from_utf8(data).ok())
.map(|svg| {
let viewbox = format!("viewBox=\"0 0 {0} {0}\" xmlns", units_per_em);
svg.replace("xmlns", &viewbox)
})
.and_then(|s| usvg::Tree::from_str(&s, &usvg::Options::default()).ok())
{
for child in tree.root().children() {
if let usvg::NodeKind::Path(node) = &*child.borrow() {
2021-08-27 13:29:50 +02:00
// SVG is already Y-down, no flipping required.
let ts = convert_usvg_transform(node.transform)
2021-08-27 13:29:50 +02:00
.post_scale(scale, scale)
2021-03-23 15:32:36 +01:00
.post_concat(ts);
2021-08-27 13:29:50 +02:00
2021-03-23 15:32:36 +01:00
if let Some(fill) = &node.fill {
2021-08-27 13:29:50 +02:00
let path = convert_usvg_path(&node.data);
2021-03-23 15:32:36 +01:00
let (paint, fill_rule) = convert_usvg_fill(fill);
canvas.fill_path(&path, &paint, fill_rule, ts, None);
}
}
}
2021-08-27 13:29:50 +02:00
} else if let Some(raster) =
ttf.glyph_raster_image(glyph_id, pixels_per_em as u16)
{
// TODO: Vertical alignment isn't quite right for Apple Color Emoji,
// and maybe also for Noto Color Emoji. And: Is the size calculation
// correct?
let img = Image::parse(&raster.data).unwrap();
let h = text.size;
let w = (img.width() as f64 / img.height() as f64) * h;
let dx = (raster.x as f32) / (img.width() as f32) * size;
let dy = (raster.y as f32) / (img.height() as f32) * size;
let ts = ts.pre_translate(dx, -size - dy);
draw_image(canvas, ts, &img, Size::new(w, h));
} else {
// Otherwise, draw normal outline.
2021-07-08 22:33:44 +02:00
let mut builder = WrappedPathBuilder(sk::PathBuilder::new());
2021-08-27 13:29:50 +02:00
if ttf.outline_glyph(glyph_id, &mut builder).is_some() {
// Flip vertically because font designed coordinate system is Y-up.
let ts = ts.pre_scale(scale, -scale);
let path = builder.0.finish().unwrap();
2021-07-08 22:33:44 +02:00
let mut paint = convert_typst_paint(text.fill);
paint.anti_alias = true;
2021-07-08 22:33:44 +02:00
canvas.fill_path(&path, &paint, sk::FillRule::default(), ts, None);
}
2021-03-23 15:32:36 +01:00
}
2021-08-23 13:18:20 +02:00
x += glyph.x_advance.to_length(text.size).to_pt() as f32;
}
}
2021-07-08 22:33:44 +02:00
fn draw_geometry(
canvas: &mut sk::Pixmap,
ts: sk::Transform,
geometry: &Geometry,
paint: Paint,
) {
let paint = convert_typst_paint(paint);
let rule = sk::FillRule::default();
2021-02-04 21:30:18 +01:00
2021-07-08 22:33:44 +02:00
match *geometry {
Geometry::Rect(Size { w: width, h: height }) => {
2021-03-20 20:19:30 +01:00
let w = width.to_pt() as f32;
let h = height.to_pt() as f32;
2021-07-08 22:33:44 +02:00
let rect = sk::Rect::from_xywh(0.0, 0.0, w, h).unwrap();
2021-03-23 13:17:00 +01:00
canvas.fill_rect(rect, &paint, ts, None);
2021-03-20 20:19:30 +01:00
}
2021-07-08 22:33:44 +02:00
Geometry::Ellipse(size) => {
2021-05-17 22:55:31 +02:00
let path = convert_typst_path(&geom::Path::ellipse(size));
2021-03-23 13:17:00 +01:00
canvas.fill_path(&path, &paint, rule, ts, None);
2021-03-20 20:19:30 +01:00
}
2021-07-08 22:33:44 +02:00
Geometry::Line(target, thickness) => {
2021-06-10 23:08:52 +02:00
let path = {
2021-07-08 22:33:44 +02:00
let mut builder = sk::PathBuilder::new();
2021-06-10 23:08:52 +02:00
builder.line_to(target.x.to_pt() as f32, target.y.to_pt() as f32);
builder.finish().unwrap()
};
2021-07-08 22:33:44 +02:00
let mut stroke = sk::Stroke::default();
2021-06-10 23:08:52 +02:00
stroke.width = thickness.to_pt() as f32;
canvas.stroke_path(&path, &paint, &stroke, ts, None);
}
2021-07-08 22:33:44 +02:00
Geometry::Path(ref path) => {
2021-03-23 15:32:36 +01:00
let path = convert_typst_path(path);
2021-03-23 13:17:00 +01:00
canvas.fill_path(&path, &paint, rule, ts, None);
}
2021-02-06 12:30:44 +01:00
};
2021-02-04 21:30:18 +01:00
}
2021-08-27 13:29:50 +02:00
fn draw_image(canvas: &mut sk::Pixmap, ts: sk::Transform, img: &Image, size: Size) {
2021-07-08 22:33:44 +02:00
let mut pixmap = sk::Pixmap::new(img.buf.width(), img.buf.height()).unwrap();
2021-01-01 17:54:31 +01:00
for ((_, _, src), dest) in img.buf.pixels().zip(pixmap.pixels_mut()) {
let Rgba([r, g, b, a]) = src;
2021-07-08 22:33:44 +02:00
*dest = sk::ColorU8::from_rgba(r, g, b, a).premultiply();
}
let view_width = size.w.to_pt() as f32;
let view_height = size.h.to_pt() as f32;
2020-11-25 18:46:47 +01:00
let scale_x = view_width as f32 / pixmap.width() as f32;
let scale_y = view_height as f32 / pixmap.height() as f32;
2021-07-08 22:33:44 +02:00
let mut paint = sk::Paint::default();
paint.shader = sk::Pattern::new(
2021-03-23 13:17:00 +01:00
pixmap.as_ref(),
2021-07-08 22:33:44 +02:00
sk::SpreadMode::Pad,
sk::FilterQuality::Bilinear,
2020-11-25 18:46:47 +01:00
1.0,
2021-08-27 13:29:50 +02:00
sk::Transform::from_scale(scale_x, scale_y),
2020-11-25 18:46:47 +01:00
);
2021-07-08 22:33:44 +02:00
let rect = sk::Rect::from_xywh(0.0, 0.0, view_width, view_height).unwrap();
2021-03-23 13:17:00 +01:00
canvas.fill_rect(rect, &paint, ts, None);
}
2021-07-08 22:33:44 +02:00
fn convert_typst_paint(paint: Paint) -> sk::Paint<'static> {
let Paint::Color(Color::Rgba(c)) = paint;
let mut paint = sk::Paint::default();
paint.set_color_rgba8(c.r, c.g, c.b, c.a);
2021-03-20 20:19:30 +01:00
paint
}
2021-07-08 22:33:44 +02:00
fn convert_typst_path(path: &geom::Path) -> sk::Path {
let mut builder = sk::PathBuilder::new();
let f = |v: Length| v.to_pt() as f32;
2021-03-20 20:19:30 +01:00
for elem in &path.0 {
match elem {
2021-07-08 22:33:44 +02:00
PathElement::MoveTo(p) => {
builder.move_to(f(p.x), f(p.y));
}
2021-07-08 22:33:44 +02:00
PathElement::LineTo(p) => {
builder.line_to(f(p.x), f(p.y));
}
2021-07-08 22:33:44 +02:00
PathElement::CubicTo(p1, p2, p3) => {
builder.cubic_to(f(p1.x), f(p1.y), f(p2.x), f(p2.y), f(p3.x), f(p3.y));
}
2021-07-08 22:33:44 +02:00
PathElement::ClosePath => {
builder.close();
2021-03-23 15:32:36 +01:00
}
2021-03-20 20:19:30 +01:00
};
}
builder.finish().unwrap()
}
2021-07-08 22:33:44 +02:00
fn convert_usvg_transform(transform: usvg::Transform) -> sk::Transform {
let g = |v: f64| v as f32;
let usvg::Transform { a, b, c, d, e, f } = transform;
2021-07-08 22:33:44 +02:00
sk::Transform::from_row(g(a), g(b), g(c), g(d), g(e), g(f))
}
2021-07-08 22:33:44 +02:00
fn convert_usvg_fill(fill: &usvg::Fill) -> (sk::Paint<'static>, sk::FillRule) {
let mut paint = sk::Paint::default();
2021-03-23 15:32:36 +01:00
paint.anti_alias = true;
match fill.paint {
usvg::Paint::Color(usvg::Color { red, green, blue }) => {
paint.set_color_rgba8(red, green, blue, fill.opacity.to_u8())
}
2021-03-23 15:32:36 +01:00
usvg::Paint::Link(_) => {}
}
let rule = match fill.rule {
2021-07-08 22:33:44 +02:00
usvg::FillRule::NonZero => sk::FillRule::Winding,
usvg::FillRule::EvenOdd => sk::FillRule::EvenOdd,
2021-03-23 15:32:36 +01:00
};
(paint, rule)
}
2021-07-08 22:33:44 +02:00
fn convert_usvg_path(path: &usvg::PathData) -> sk::Path {
let mut builder = sk::PathBuilder::new();
let f = |v: f64| v as f32;
2021-03-23 15:32:36 +01:00
for seg in path.iter() {
match *seg {
usvg::PathSegment::MoveTo { x, y } => {
builder.move_to(f(x), f(y));
}
2021-03-23 15:32:36 +01:00
usvg::PathSegment::LineTo { x, y } => {
builder.line_to(f(x), f(y));
}
usvg::PathSegment::CurveTo { x1, y1, x2, y2, x, y } => {
builder.cubic_to(f(x1), f(y1), f(x2), f(y2), f(x), f(y));
}
usvg::PathSegment::ClosePath => {
builder.close();
2021-03-23 15:32:36 +01:00
}
}
}
builder.finish().unwrap()
}
2021-07-08 22:33:44 +02:00
struct WrappedPathBuilder(sk::PathBuilder);
impl OutlineBuilder for WrappedPathBuilder {
fn move_to(&mut self, x: f32, y: f32) {
self.0.move_to(x, y);
}
fn line_to(&mut self, x: f32, y: f32) {
self.0.line_to(x, y);
}
fn quad_to(&mut self, x1: f32, y1: f32, x: f32, y: f32) {
self.0.quad_to(x1, y1, x, y);
}
fn curve_to(&mut self, x1: f32, y1: f32, x2: f32, y2: f32, x: f32, y: f32) {
self.0.cubic_to(x1, y1, x2, y2, x, y);
}
fn close(&mut self) {
self.0.close();
}
}