mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-07-29 12:19:54 +00:00
LibWeb: Make SVGs respect their CSS transforms
This commit is contained in:
parent
15741350ba
commit
6f52064c4f
Notes:
github-actions[bot]
2024-11-24 13:32:12 +00:00
Author: https://github.com/Psychpsyo
Commit: 6f52064c4f
Pull-request: https://github.com/LadybirdBrowser/ladybird/pull/2510
5 changed files with 76 additions and 48 deletions
|
@ -51,60 +51,61 @@ static Gfx::FloatMatrix4x4 matrix_with_scaled_translation(Gfx::FloatMatrix4x4 ma
|
||||||
return matrix;
|
return matrix;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SVGSVGPaintable::paint_svg_box(PaintContext& context, PaintableBox const& svg_box, PaintPhase phase)
|
||||||
|
{
|
||||||
|
auto const& computed_values = svg_box.computed_values();
|
||||||
|
|
||||||
|
auto filters = svg_box.computed_values().filter();
|
||||||
|
auto masking_area = svg_box.get_masking_area();
|
||||||
|
auto needs_to_save_state = computed_values.opacity() < 1 || svg_box.has_css_transform() || svg_box.get_masking_area().has_value() || !filters.is_none();
|
||||||
|
|
||||||
|
if (needs_to_save_state) {
|
||||||
|
context.display_list_recorder().save();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (computed_values.opacity() < 1) {
|
||||||
|
context.display_list_recorder().apply_opacity(computed_values.opacity());
|
||||||
|
}
|
||||||
|
|
||||||
|
context.display_list_recorder().apply_filters(filters);
|
||||||
|
|
||||||
|
if (svg_box.has_css_transform()) {
|
||||||
|
auto transform_matrix = svg_box.transform();
|
||||||
|
Gfx::FloatPoint transform_origin = svg_box.transform_origin().template to_type<float>();
|
||||||
|
auto to_device_pixels_scale = float(context.device_pixels_per_css_pixel());
|
||||||
|
context.display_list_recorder().apply_transform(transform_origin.scaled(to_device_pixels_scale), matrix_with_scaled_translation(transform_matrix, to_device_pixels_scale));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (masking_area.has_value()) {
|
||||||
|
if (masking_area->is_empty())
|
||||||
|
return;
|
||||||
|
auto mask_bitmap = svg_box.calculate_mask(context, *masking_area);
|
||||||
|
if (mask_bitmap) {
|
||||||
|
auto source_paintable_rect = context.enclosing_device_rect(*masking_area).template to_type<int>();
|
||||||
|
auto origin = source_paintable_rect.location();
|
||||||
|
context.display_list_recorder().apply_mask_bitmap(origin, mask_bitmap.release_nonnull(), *svg_box.get_mask_type());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
svg_box.before_paint(context, PaintPhase::Foreground);
|
||||||
|
svg_box.paint(context, PaintPhase::Foreground);
|
||||||
|
svg_box.after_paint(context, PaintPhase::Foreground);
|
||||||
|
|
||||||
|
paint_descendants(context, svg_box, phase);
|
||||||
|
|
||||||
|
if (needs_to_save_state) {
|
||||||
|
context.display_list_recorder().restore();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void SVGSVGPaintable::paint_descendants(PaintContext& context, PaintableBox const& paintable, PaintPhase phase)
|
void SVGSVGPaintable::paint_descendants(PaintContext& context, PaintableBox const& paintable, PaintPhase phase)
|
||||||
{
|
{
|
||||||
if (phase != PaintPhase::Foreground)
|
if (phase != PaintPhase::Foreground)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
auto paint_svg_box = [&](auto& svg_box) {
|
|
||||||
auto const& computed_values = svg_box.computed_values();
|
|
||||||
|
|
||||||
auto filters = paintable.computed_values().filter();
|
|
||||||
auto masking_area = svg_box.get_masking_area();
|
|
||||||
auto needs_to_save_state = computed_values.opacity() < 1 || svg_box.has_css_transform() || svg_box.get_masking_area().has_value() || !filters.is_none();
|
|
||||||
|
|
||||||
if (needs_to_save_state) {
|
|
||||||
context.display_list_recorder().save();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (computed_values.opacity() < 1) {
|
|
||||||
context.display_list_recorder().apply_opacity(computed_values.opacity());
|
|
||||||
}
|
|
||||||
|
|
||||||
context.display_list_recorder().apply_filters(filters);
|
|
||||||
|
|
||||||
if (svg_box.has_css_transform()) {
|
|
||||||
auto transform_matrix = svg_box.transform();
|
|
||||||
Gfx::FloatPoint transform_origin = svg_box.transform_origin().template to_type<float>();
|
|
||||||
auto to_device_pixels_scale = float(context.device_pixels_per_css_pixel());
|
|
||||||
context.display_list_recorder().apply_transform(transform_origin.scaled(to_device_pixels_scale), matrix_with_scaled_translation(transform_matrix, to_device_pixels_scale));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (masking_area.has_value()) {
|
|
||||||
if (masking_area->is_empty())
|
|
||||||
return;
|
|
||||||
auto mask_bitmap = svg_box.calculate_mask(context, *masking_area);
|
|
||||||
if (mask_bitmap) {
|
|
||||||
auto source_paintable_rect = context.enclosing_device_rect(*masking_area).template to_type<int>();
|
|
||||||
auto origin = source_paintable_rect.location();
|
|
||||||
context.display_list_recorder().apply_mask_bitmap(origin, mask_bitmap.release_nonnull(), *svg_box.get_mask_type());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
svg_box.before_paint(context, PaintPhase::Foreground);
|
|
||||||
svg_box.paint(context, PaintPhase::Foreground);
|
|
||||||
svg_box.after_paint(context, PaintPhase::Foreground);
|
|
||||||
|
|
||||||
paint_descendants(context, svg_box, phase);
|
|
||||||
|
|
||||||
if (needs_to_save_state) {
|
|
||||||
context.display_list_recorder().restore();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
paintable.before_children_paint(context, PaintPhase::Foreground);
|
paintable.before_children_paint(context, PaintPhase::Foreground);
|
||||||
paintable.for_each_child_of_type<PaintableBox>([&](PaintableBox& child) {
|
paintable.for_each_child_of_type<PaintableBox>([&](PaintableBox& child) {
|
||||||
paint_svg_box(child);
|
paint_svg_box(context, child, phase);
|
||||||
return IterationDecision::Continue;
|
return IterationDecision::Continue;
|
||||||
});
|
});
|
||||||
paintable.after_children_paint(context, PaintPhase::Foreground);
|
paintable.after_children_paint(context, PaintPhase::Foreground);
|
||||||
|
|
|
@ -23,6 +23,7 @@ public:
|
||||||
|
|
||||||
Layout::SVGSVGBox const& layout_box() const;
|
Layout::SVGSVGBox const& layout_box() const;
|
||||||
|
|
||||||
|
static void paint_svg_box(PaintContext& context, PaintableBox const& svg_box, PaintPhase phase);
|
||||||
static void paint_descendants(PaintContext& context, PaintableBox const& paintable, PaintPhase phase);
|
static void paint_descendants(PaintContext& context, PaintableBox const& paintable, PaintPhase phase);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
|
@ -104,7 +104,7 @@ void StackingContext::paint_svg(PaintContext& context, PaintableBox const& paint
|
||||||
paintable.apply_clip_overflow_rect(context, PaintPhase::Foreground);
|
paintable.apply_clip_overflow_rect(context, PaintPhase::Foreground);
|
||||||
paint_node(paintable, context, PaintPhase::Background);
|
paint_node(paintable, context, PaintPhase::Background);
|
||||||
paint_node(paintable, context, PaintPhase::Border);
|
paint_node(paintable, context, PaintPhase::Border);
|
||||||
SVGSVGPaintable::paint_descendants(context, paintable, phase);
|
SVGSVGPaintable::paint_svg_box(context, paintable, phase);
|
||||||
paintable.clear_clip_overflow_rect(context, PaintPhase::Foreground);
|
paintable.clear_clip_overflow_rect(context, PaintPhase::Foreground);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
12
Tests/LibWeb/Ref/expected/svg-with-css-transform.html
Normal file
12
Tests/LibWeb/Ref/expected/svg-with-css-transform.html
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<style>
|
||||||
|
svg {
|
||||||
|
position: absolute;
|
||||||
|
top: 50px;
|
||||||
|
left: 50px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<svg viewBox="0 0 100 100" xmlns="http://www.w3.org/2000/svg" width="100" height="100">
|
||||||
|
<rect width="100" height="100" />
|
||||||
|
</svg>
|
14
Tests/LibWeb/Ref/input/svg-with-css-transform.html
Normal file
14
Tests/LibWeb/Ref/input/svg-with-css-transform.html
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<link rel="match" href="../expected/svg-with-css-transform.html" />
|
||||||
|
<style>
|
||||||
|
body {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
svg {
|
||||||
|
transform: translate(50px, 50px);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<svg viewBox="0 0 100 100" xmlns="http://www.w3.org/2000/svg" width="100" height="100">
|
||||||
|
<rect width="100" height="100" />
|
||||||
|
</svg>
|
Loading…
Add table
Add a link
Reference in a new issue