mirror of
https://github.com/RGBCube/serenity
synced 2025-05-31 13:48:12 +00:00
LibWeb: Add support for 'view-box' attribute to SVGs
This patch begins the support for the 'view-box' attribute that can be attached to <svg>'s. The FormattingContext determines the size of the Element according to the specified 'width' and 'height' or if they are not given by the 'viewbox' or by the bounding box of the path if nothing is specified. When we try to paint a SVG Path that belongs to a <svg> that has the 'view-box' and a specified 'height'/'width', all the parts of the path get scaled/moved accordingly. There probably are many edge cases and bugs still to be found, but this is a nice start. :^)
This commit is contained in:
parent
4f1df48bd9
commit
8461791ce2
3 changed files with 131 additions and 9 deletions
|
@ -1,6 +1,7 @@
|
|||
/*
|
||||
* Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
|
||||
* Copyright (c) 2022, Sam Atkins <atkinssj@serenityos.org>
|
||||
* Copyright (c) 2022, Tobias Christiansen <tobyase@serenityos.org>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
@ -8,6 +9,7 @@
|
|||
#include <AK/Format.h>
|
||||
#include <LibWeb/Layout/SVGFormattingContext.h>
|
||||
#include <LibWeb/Layout/SVGGeometryBox.h>
|
||||
#include <LibWeb/SVG/SVGSVGElement.h>
|
||||
|
||||
namespace Web::Layout {
|
||||
|
||||
|
@ -22,20 +24,57 @@ SVGFormattingContext::~SVGFormattingContext()
|
|||
|
||||
void SVGFormattingContext::run(Box const& box, LayoutMode)
|
||||
{
|
||||
box.for_each_in_subtree_of_type<SVGBox>([&](auto const& descendant) {
|
||||
box.for_each_in_subtree_of_type<SVGBox>([&](SVGBox const& descendant) {
|
||||
if (is<SVGGeometryBox>(descendant)) {
|
||||
auto const& geometry_box = static_cast<SVGGeometryBox const&>(descendant);
|
||||
auto& path = const_cast<SVGGeometryBox&>(geometry_box).dom_node().get_path();
|
||||
auto bounding_box = path.bounding_box();
|
||||
|
||||
auto& geometry_box_state = m_state.get_mutable(geometry_box);
|
||||
|
||||
auto& dom_node = const_cast<SVGGeometryBox&>(geometry_box).dom_node();
|
||||
|
||||
SVG::SVGSVGElement* svg_element = dom_node.first_ancestor_of_type<SVG::SVGSVGElement>();
|
||||
|
||||
if (svg_element->has_attribute(HTML::AttributeNames::width) && svg_element->has_attribute(HTML::AttributeNames::width)) {
|
||||
geometry_box_state.offset = { 0, 0 };
|
||||
auto& layout_node = static_cast<Layout::Node&>(*(svg_element->layout_node()));
|
||||
|
||||
// FIXME: Allow for relative lengths here
|
||||
geometry_box_state.content_width = layout_node.computed_values().width().value().resolved(layout_node, { 0, CSS::Length::Type::Px }).to_px(layout_node);
|
||||
geometry_box_state.content_height = layout_node.computed_values().height().value().resolved(layout_node, { 0, CSS::Length::Type::Px }).to_px(layout_node);
|
||||
|
||||
return IterationDecision::Continue;
|
||||
}
|
||||
|
||||
// FIXME: Allow for one of {width, height} to not be specified}
|
||||
if (svg_element->has_attribute(HTML::AttributeNames::width)) {
|
||||
}
|
||||
|
||||
if (svg_element->has_attribute(HTML::AttributeNames::height)) {
|
||||
}
|
||||
|
||||
auto& path = dom_node.get_path();
|
||||
auto path_bounding_box = path.bounding_box();
|
||||
|
||||
// Stroke increases the path's size by stroke_width/2 per side.
|
||||
auto stroke_width = geometry_box.dom_node().stroke_width().value_or(0);
|
||||
bounding_box.inflate(stroke_width, stroke_width);
|
||||
path_bounding_box.inflate(stroke_width, stroke_width);
|
||||
|
||||
auto& geometry_box_state = m_state.get_mutable(geometry_box);
|
||||
geometry_box_state.offset = bounding_box.top_left();
|
||||
geometry_box_state.content_width = bounding_box.width();
|
||||
geometry_box_state.content_height = bounding_box.height();
|
||||
auto& maybe_view_box = svg_element->view_box();
|
||||
|
||||
if (maybe_view_box.has_value()) {
|
||||
auto view_box = maybe_view_box.value();
|
||||
Gfx::FloatPoint viewbox_offset = { view_box.min_x, view_box.min_y };
|
||||
geometry_box_state.offset = path_bounding_box.top_left() + viewbox_offset;
|
||||
|
||||
geometry_box_state.content_width = view_box.width;
|
||||
geometry_box_state.content_height = view_box.height;
|
||||
|
||||
return IterationDecision::Continue;
|
||||
}
|
||||
|
||||
geometry_box_state.offset = path_bounding_box.top_left();
|
||||
geometry_box_state.content_width = path_bounding_box.width();
|
||||
geometry_box_state.content_height = path_bounding_box.height();
|
||||
}
|
||||
|
||||
return IterationDecision::Continue;
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue