diff --git a/exporter/canvas/src/lib.rs b/exporter/canvas/src/lib.rs index 7e5214d4..b0fde61a 100644 --- a/exporter/canvas/src/lib.rs +++ b/exporter/canvas/src/lib.rs @@ -178,7 +178,7 @@ impl CanvasElem for CanvasPathElem { if fill { // todo: canvas gradient - if fill_color.starts_with("url") { + if fill_color.starts_with('@') { fill_color = "black".into() } canvas.set_fill_style(&fill_color.as_ref().into()); @@ -187,7 +187,7 @@ impl CanvasElem for CanvasPathElem { if stroke && stroke_width.abs() > 1e-5 { // todo: canvas gradient - if stroke_color.starts_with("url") { + if stroke_color.starts_with('@') { stroke_color = "black".into() } diff --git a/exporter/svg/src/backend/mod.rs b/exporter/svg/src/backend/mod.rs index 7b23233f..6a04ea13 100644 --- a/exporter/svg/src/backend/mod.rs +++ b/exporter/svg/src/backend/mod.rs @@ -631,7 +631,7 @@ fn render_path(path: &ir::PathItem, state: RenderState, abs_ref: &Fingerprint) - for style in &path.styles { match style { PathStyle::Fill(color) => { - fill_color = if color.starts_with("url") { + fill_color = if color.starts_with('@') { ft = format!(r#"url(#{})"#, state.at(abs_ref).as_svg_id("pf")); &ft } else { @@ -640,7 +640,7 @@ fn render_path(path: &ir::PathItem, state: RenderState, abs_ref: &Fingerprint) - } PathStyle::Stroke(color) => { // compress the stroke color - p.push(if color.starts_with("url") { + p.push(if color.starts_with('@') { let ps = state.at(abs_ref).as_svg_id("ps"); format!(r##"stroke="url(#{})" "##, &ps) } else { diff --git a/exporter/svg/src/frontend/context.rs b/exporter/svg/src/frontend/context.rs index 6e8abed6..97bfee0d 100644 --- a/exporter/svg/src/frontend/context.rs +++ b/exporter/svg/src/frontend/context.rs @@ -232,20 +232,10 @@ impl<'m, 't, Feat: ExportFeature> RenderVm for RenderContext<'m, 't, Feat> { let mut group_ctx = text.shape.add_transform(self, group_ctx, upem); - if let Some(fill) = &group_ctx.text_fill { + let width = if let Some(fill) = &group_ctx.text_fill { // clip path rect let clip_id = fill.as_svg_id("tc"); - group_ctx.content.push(SvgText::Plain(format!( - r#""#, - clip_id - ))); - } - let width = text.render_glyphs(upem, |x, g| { - group_ctx.render_glyph(self, x, g); - }); - if let Some(fill) = &group_ctx.text_fill { let fill_id = fill.as_svg_id("tf"); - let clip_id = fill.as_svg_id("tc"); // because the text is already scaled by the font size, // we need to scale it back to the original size. @@ -256,16 +246,26 @@ impl<'m, 't, Feat: ExportFeature> RenderVm for RenderContext<'m, 't, Feat> { .descender .at(TypstAbs::raw(upem.0 as f64)) .to_pt() as f32; - let width = width.0 * upem.0 / text.shape.size.0; + + group_ctx.content.push(SvgText::Plain(format!( + r#""#, + clip_id + ))); + + let width = text.render_glyphs(upem, |x, g| { + group_ctx.render_glyph(self, x, g); + group_ctx.content.push(SvgText::Plain("".into())); + }); group_ctx .content .push(SvgText::Plain(r#""#.to_owned())); // clip path rect + let scaled_width = width.0 * upem.0 / text.shape.size.0; group_ctx.content.push(SvgText::Plain(format!( r##""##, - fill_id, width, upem.0, descender, clip_id + fill_id, scaled_width, upem.0, descender, clip_id ))); // image glyphs @@ -276,7 +276,13 @@ impl<'m, 't, Feat: ExportFeature> RenderVm for RenderContext<'m, 't, Feat> { } group_ctx.render_glyph(self, x, g); }); - } + + width + } else { + text.render_glyphs(upem, |x, g| { + group_ctx.render_glyph(self, x, g); + }) + }; if self.should_render_text_element() { group_ctx.render_text_semantics_inner( @@ -346,35 +352,35 @@ impl<'m, 't, Feat: ExportFeature> FlatRenderVm<'m> for RenderContext<'m, 't, Fea group_ctx = text.shape.add_transform(self, group_ctx, upem); - if let Some(fill) = &group_ctx.text_fill { + let width = if let Some(fill) = &group_ctx.text_fill { // clip path rect let clip_id = fill.as_svg_id("tc"); - group_ctx.content.push(SvgText::Plain(format!( - r#""#, - clip_id - ))); - } - let width = text.render_glyphs(upem, |x, g| { - group_ctx.render_glyph_ref(self, x, g); - }); - if let Some(fill) = &group_ctx.text_fill { let fill_id = fill.as_svg_id("tf"); - let clip_id = fill.as_svg_id("tc"); // because the text is already scaled by the font size, // we need to scale it back to the original size. // todo: infinite multiplication let descender = font.descender.0 * upem.0; - let width = width.0 * upem.0 / text.shape.size.0; + + group_ctx.content.push(SvgText::Plain(format!( + r#""#, + clip_id + ))); + + let width = text.render_glyphs(upem, |x, g| { + group_ctx.render_glyph_ref(self, x, g); + group_ctx.content.push(SvgText::Plain("".into())); + }); group_ctx .content .push(SvgText::Plain(r#""#.to_owned())); // clip path rect + let scaled_width = width.0 * upem.0 / text.shape.size.0; group_ctx.content.push(SvgText::Plain(format!( r##""##, - fill_id, width, upem.0, descender, clip_id + fill_id, scaled_width, upem.0, descender, clip_id ))); // image glyphs @@ -388,7 +394,13 @@ impl<'m, 't, Feat: ExportFeature> FlatRenderVm<'m> for RenderContext<'m, 't, Fea } group_ctx.render_glyph_ref(self, x, g); }); - } + + width + } else { + text.render_glyphs(upem, |x, g| { + group_ctx.render_glyph_ref(self, x, g); + }) + }; if self.should_render_text_element() { group_ctx.render_text_semantics_inner( diff --git a/fuzzers/corpora/visualize/gradient-official_00.typ b/fuzzers/corpora/visualize/gradient-official_00.typ new file mode 100644 index 00000000..9e39bd65 --- /dev/null +++ b/fuzzers/corpora/visualize/gradient-official_00.typ @@ -0,0 +1,10 @@ + + +#set text(fill: gradient.linear(red, blue), font: "Open Sans") +#let rainbow(content) = { + set text(fill: gradient.linear(..color.map.rainbow)) + box(content) +} + +This is a gradient on text, but with a #rainbow[twist]! +