mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-04-25 14:05:15 +00:00
CSSStyleDeclaration is a base class that's used by various collections of style properties or descriptors. This commit moves all style-property-related code into CSSStyleProperties, where it belongs. As noted in the previous commit, we also apply the CSSStyleProperties prototype now.
1233 lines
62 KiB
C++
1233 lines
62 KiB
C++
/*
|
||
* Copyright (c) 2018-2024, Andreas Kling <andreas@ladybird.org>
|
||
* Copyright (c) 2023-2025, Sam Atkins <sam@ladybird.org>
|
||
*
|
||
* SPDX-License-Identifier: BSD-2-Clause
|
||
*/
|
||
|
||
#include <LibWeb/Bindings/CSSStylePropertiesPrototype.h>
|
||
#include <LibWeb/Bindings/ExceptionOrUtils.h>
|
||
#include <LibWeb/Bindings/Intrinsics.h>
|
||
#include <LibWeb/CSS/CSSStyleProperties.h>
|
||
#include <LibWeb/CSS/ComputedProperties.h>
|
||
#include <LibWeb/CSS/Parser/Parser.h>
|
||
#include <LibWeb/CSS/StyleComputer.h>
|
||
#include <LibWeb/CSS/StyleValues/CSSKeywordValue.h>
|
||
#include <LibWeb/CSS/StyleValues/FitContentStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/ImageStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/LengthStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/NumberStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/PercentageStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/ShorthandStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/StyleValueList.h>
|
||
#include <LibWeb/CSS/StyleValues/TransformationStyleValue.h>
|
||
#include <LibWeb/DOM/Document.h>
|
||
#include <LibWeb/DOM/Element.h>
|
||
#include <LibWeb/Infra/Strings.h>
|
||
#include <LibWeb/Layout/Node.h>
|
||
#include <LibWeb/Painting/BoxModelMetrics.h>
|
||
#include <LibWeb/Painting/PaintableBox.h>
|
||
|
||
namespace Web::CSS {
|
||
|
||
GC_DEFINE_ALLOCATOR(CSSStyleProperties);
|
||
|
||
GC::Ref<CSSStyleProperties> CSSStyleProperties::create(JS::Realm& realm, Vector<StyleProperty> properties, HashMap<FlyString, StyleProperty> custom_properties)
|
||
{
|
||
// https://drafts.csswg.org/cssom/#dom-cssstylerule-style
|
||
// The style attribute must return a CSSStyleProperties object for the style rule, with the following properties:
|
||
// computed flag: Unset.
|
||
// readonly flag: Unset.
|
||
// declarations: The declared declarations in the rule, in specified order.
|
||
// parent CSS rule: The context object.
|
||
// owner node: Null.
|
||
return realm.create<CSSStyleProperties>(realm, Computed::No, Readonly::No, move(properties), move(custom_properties), OptionalNone {});
|
||
}
|
||
|
||
GC::Ref<CSSStyleProperties> CSSStyleProperties::create_resolved_style(DOM::ElementReference element_reference)
|
||
{
|
||
// https://drafts.csswg.org/cssom/#dom-window-getcomputedstyle
|
||
// 6. Return a live CSSStyleProperties object with the following properties:
|
||
// computed flag: Set.
|
||
// readonly flag: Set.
|
||
// declarations: decls.
|
||
// parent CSS rule: Null.
|
||
// owner node: obj.
|
||
// AD-HOC: Rather than instantiate with a list of decls, they're generated on demand.
|
||
auto& realm = element_reference.element().realm();
|
||
return realm.create<CSSStyleProperties>(realm, Computed::Yes, Readonly::Yes, Vector<StyleProperty> {}, HashMap<FlyString, StyleProperty> {}, move(element_reference));
|
||
}
|
||
|
||
GC::Ref<CSSStyleProperties> CSSStyleProperties::create_element_inline_style(DOM::ElementReference element_reference, Vector<StyleProperty> properties, HashMap<FlyString, StyleProperty> custom_properties)
|
||
{
|
||
// https://drafts.csswg.org/cssom/#dom-elementcssinlinestyle-style
|
||
// The style attribute must return a CSS declaration block object whose readonly flag is unset, whose parent CSS
|
||
// rule is null, and whose owner node is the context object.
|
||
auto& realm = element_reference.element().realm();
|
||
return realm.create<CSSStyleProperties>(realm, Computed::No, Readonly::No, move(properties), move(custom_properties), move(element_reference));
|
||
}
|
||
|
||
CSSStyleProperties::CSSStyleProperties(JS::Realm& realm, Computed computed, Readonly readonly, Vector<StyleProperty> properties, HashMap<FlyString, StyleProperty> custom_properties, Optional<DOM::ElementReference> owner_node)
|
||
: CSSStyleDeclaration(realm, computed, readonly)
|
||
, m_properties(move(properties))
|
||
, m_custom_properties(move(custom_properties))
|
||
{
|
||
set_owner_node(move(owner_node));
|
||
}
|
||
|
||
void CSSStyleProperties::initialize(JS::Realm& realm)
|
||
{
|
||
Base::initialize(realm);
|
||
WEB_SET_PROTOTYPE_FOR_INTERFACE(CSSStyleProperties);
|
||
}
|
||
|
||
void CSSStyleProperties::visit_edges(Visitor& visitor)
|
||
{
|
||
Base::visit_edges(visitor);
|
||
for (auto& property : m_properties) {
|
||
if (property.value->is_image())
|
||
property.value->as_image().visit_edges(visitor);
|
||
}
|
||
}
|
||
|
||
// https://drafts.csswg.org/cssom/#dom-cssstyledeclaration-length
|
||
size_t CSSStyleProperties::length() const
|
||
{
|
||
// The length attribute must return the number of CSS declarations in the declarations.
|
||
// FIXME: Include the number of custom properties.
|
||
|
||
if (is_computed())
|
||
return to_underlying(last_longhand_property_id) - to_underlying(first_longhand_property_id) + 1;
|
||
|
||
return m_properties.size();
|
||
}
|
||
|
||
String CSSStyleProperties::item(size_t index) const
|
||
{
|
||
// The item(index) method must return the property name of the CSS declaration at position index.
|
||
// FIXME: Include custom properties.
|
||
|
||
if (index >= length())
|
||
return {};
|
||
|
||
if (is_computed()) {
|
||
auto property_id = static_cast<PropertyID>(index + to_underlying(first_longhand_property_id));
|
||
return string_from_property_id(property_id).to_string();
|
||
}
|
||
|
||
return CSS::string_from_property_id(m_properties[index].property_id).to_string();
|
||
}
|
||
|
||
Optional<StyleProperty> CSSStyleProperties::property(PropertyID property_id) const
|
||
{
|
||
if (is_computed()) {
|
||
auto& element = owner_node()->element();
|
||
auto pseudo_element = owner_node()->pseudo_element();
|
||
|
||
// https://www.w3.org/TR/cssom-1/#dom-window-getcomputedstyle
|
||
// NB: This is a partial enforcement of step 5 ("If elt is connected, ...")
|
||
if (!element.is_connected())
|
||
return {};
|
||
|
||
auto get_layout_node = [&]() {
|
||
if (pseudo_element.has_value())
|
||
return element.get_pseudo_element_node(pseudo_element.value());
|
||
return element.layout_node();
|
||
};
|
||
|
||
Layout::NodeWithStyle* layout_node = get_layout_node();
|
||
|
||
// FIXME: Be smarter about updating layout if there's no layout node.
|
||
// We may legitimately have no layout node if we're not visible, but this protects against situations
|
||
// where we're requesting the computed style before layout has happened.
|
||
if (!layout_node || property_affects_layout(property_id)) {
|
||
element.document().update_layout(DOM::UpdateLayoutReason::ResolvedCSSStyleDeclarationProperty);
|
||
layout_node = get_layout_node();
|
||
} else {
|
||
// FIXME: If we had a way to update style for a single element, this would be a good place to use it.
|
||
element.document().update_style();
|
||
}
|
||
|
||
if (!layout_node) {
|
||
auto style = element.document().style_computer().compute_style(element, pseudo_element);
|
||
|
||
// FIXME: This is a stopgap until we implement shorthand -> longhand conversion.
|
||
auto const* value = style->maybe_null_property(property_id);
|
||
if (!value) {
|
||
dbgln("FIXME: CSSStyleProperties::property(property_id={:#x}) No value for property ID in newly computed style case.", to_underlying(property_id));
|
||
return {};
|
||
}
|
||
return StyleProperty {
|
||
.property_id = property_id,
|
||
.value = *value,
|
||
};
|
||
}
|
||
|
||
auto value = style_value_for_computed_property(*layout_node, property_id);
|
||
if (!value)
|
||
return {};
|
||
return StyleProperty {
|
||
.property_id = property_id,
|
||
.value = *value,
|
||
};
|
||
}
|
||
|
||
for (auto& property : m_properties) {
|
||
if (property.property_id == property_id)
|
||
return property;
|
||
}
|
||
return {};
|
||
}
|
||
|
||
Optional<StyleProperty const&> CSSStyleProperties::custom_property(FlyString const& custom_property_name) const
|
||
{
|
||
if (is_computed()) {
|
||
auto& element = owner_node()->element();
|
||
auto pseudo_element = owner_node()->pseudo_element();
|
||
|
||
element.document().update_style();
|
||
|
||
auto const* element_to_check = &element;
|
||
while (element_to_check) {
|
||
if (auto property = element_to_check->custom_properties(pseudo_element).get(custom_property_name); property.has_value())
|
||
return *property;
|
||
|
||
element_to_check = element_to_check->parent_element();
|
||
}
|
||
|
||
return {};
|
||
}
|
||
|
||
return m_custom_properties.get(custom_property_name);
|
||
}
|
||
|
||
// https://drafts.csswg.org/cssom/#dom-cssstyledeclaration-setproperty
|
||
WebIDL::ExceptionOr<void> CSSStyleProperties::set_property(StringView property_name, StringView value, StringView priority)
|
||
{
|
||
// 1. If the computed flag is set, then throw a NoModificationAllowedError exception.
|
||
if (is_computed())
|
||
return WebIDL::NoModificationAllowedError::create(realm(), "Cannot modify properties in result of getComputedStyle()"_string);
|
||
|
||
// FIXME: 2. If property is not a custom property, follow these substeps:
|
||
// FIXME: 1. Let property be property converted to ASCII lowercase.
|
||
// FIXME: 2. If property is not a case-sensitive match for a supported CSS property, then return.
|
||
// NB: This must be handled before we've turned the property string into a PropertyID.
|
||
|
||
auto maybe_property_id = property_id_from_string(property_name);
|
||
if (!maybe_property_id.has_value())
|
||
return {};
|
||
auto property_id = maybe_property_id.value();
|
||
|
||
// 3. If value is the empty string, invoke removeProperty() with property as argument and return.
|
||
if (value.is_empty()) {
|
||
MUST(remove_property(property_name));
|
||
return {};
|
||
}
|
||
|
||
// 4. If priority is not the empty string and is not an ASCII case-insensitive match for the string "important", then return.
|
||
if (!priority.is_empty() && !Infra::is_ascii_case_insensitive_match(priority, "important"sv))
|
||
return {};
|
||
|
||
// 5. Let component value list be the result of parsing value for property property.
|
||
auto component_value_list = owner_node().has_value()
|
||
? parse_css_value(CSS::Parser::ParsingParams { owner_node()->element().document() }, value, property_id)
|
||
: parse_css_value(CSS::Parser::ParsingParams {}, value, property_id);
|
||
|
||
// 6. If component value list is null, then return.
|
||
if (!component_value_list)
|
||
return {};
|
||
|
||
// 7. Let updated be false.
|
||
bool updated = false;
|
||
|
||
// 8. If property is a shorthand property,
|
||
if (property_is_shorthand(property_id)) {
|
||
// then for each longhand property longhand that property maps to, in canonical order, follow these substeps:
|
||
StyleComputer::for_each_property_expanding_shorthands(property_id, *component_value_list, StyleComputer::AllowUnresolved::Yes, [this, &updated, priority](PropertyID longhand_property_id, CSSStyleValue const& longhand_value) {
|
||
// 1. Let longhand result be the result of set the CSS declaration longhand with the appropriate value(s) from component value list,
|
||
// with the important flag set if priority is not the empty string, and unset otherwise, and with the list of declarations being the declarations.
|
||
// 2. If longhand result is true, let updated be true.
|
||
updated |= set_a_css_declaration(longhand_property_id, longhand_value, !priority.is_empty() ? Important::Yes : Important::No);
|
||
});
|
||
}
|
||
// 9. Otherwise,
|
||
else {
|
||
if (property_id == PropertyID::Custom) {
|
||
auto custom_name = FlyString::from_utf8_without_validation(property_name.bytes());
|
||
StyleProperty style_property {
|
||
.important = !priority.is_empty() ? Important::Yes : Important::No,
|
||
.property_id = property_id,
|
||
.value = component_value_list.release_nonnull(),
|
||
.custom_name = custom_name,
|
||
};
|
||
m_custom_properties.set(custom_name, style_property);
|
||
updated = true;
|
||
} else {
|
||
// let updated be the result of set the CSS declaration property with value component value list,
|
||
// with the important flag set if priority is not the empty string, and unset otherwise,
|
||
// and with the list of declarations being the declarations.
|
||
updated = set_a_css_declaration(property_id, *component_value_list, !priority.is_empty() ? Important::Yes : Important::No);
|
||
}
|
||
}
|
||
|
||
// 10. If updated is true, update style attribute for the CSS declaration block.
|
||
if (updated)
|
||
update_style_attribute();
|
||
|
||
return {};
|
||
}
|
||
|
||
WebIDL::ExceptionOr<void> CSSStyleProperties::set_property(PropertyID property_id, StringView css_text, StringView priority)
|
||
{
|
||
return set_property(string_from_property_id(property_id), css_text, priority);
|
||
}
|
||
|
||
static NonnullRefPtr<CSSStyleValue const> style_value_for_length_percentage(LengthPercentage const& length_percentage)
|
||
{
|
||
if (length_percentage.is_auto())
|
||
return CSSKeywordValue::create(Keyword::Auto);
|
||
if (length_percentage.is_percentage())
|
||
return PercentageStyleValue::create(length_percentage.percentage());
|
||
if (length_percentage.is_length())
|
||
return LengthStyleValue::create(length_percentage.length());
|
||
return length_percentage.calculated();
|
||
}
|
||
|
||
static NonnullRefPtr<CSSStyleValue const> style_value_for_size(Size const& size)
|
||
{
|
||
if (size.is_none())
|
||
return CSSKeywordValue::create(Keyword::None);
|
||
if (size.is_percentage())
|
||
return PercentageStyleValue::create(size.percentage());
|
||
if (size.is_length())
|
||
return LengthStyleValue::create(size.length());
|
||
if (size.is_auto())
|
||
return CSSKeywordValue::create(Keyword::Auto);
|
||
if (size.is_calculated())
|
||
return size.calculated();
|
||
if (size.is_min_content())
|
||
return CSSKeywordValue::create(Keyword::MinContent);
|
||
if (size.is_max_content())
|
||
return CSSKeywordValue::create(Keyword::MaxContent);
|
||
if (size.is_fit_content())
|
||
return FitContentStyleValue::create(size.fit_content_available_space());
|
||
TODO();
|
||
}
|
||
|
||
enum class LogicalSide {
|
||
BlockStart,
|
||
BlockEnd,
|
||
InlineStart,
|
||
InlineEnd,
|
||
};
|
||
static RefPtr<CSSStyleValue const> style_value_for_length_box_logical_side(Layout::NodeWithStyle const&, LengthBox const& box, LogicalSide logical_side)
|
||
{
|
||
// FIXME: Actually determine the logical sides based on layout_node's writing-mode and direction.
|
||
switch (logical_side) {
|
||
case LogicalSide::BlockStart:
|
||
return style_value_for_length_percentage(box.top());
|
||
case LogicalSide::BlockEnd:
|
||
return style_value_for_length_percentage(box.bottom());
|
||
case LogicalSide::InlineStart:
|
||
return style_value_for_length_percentage(box.left());
|
||
case LogicalSide::InlineEnd:
|
||
return style_value_for_length_percentage(box.right());
|
||
}
|
||
VERIFY_NOT_REACHED();
|
||
}
|
||
|
||
static CSSPixels pixels_for_pixel_box_logical_side(Layout::NodeWithStyle const&, Painting::PixelBox const& box, LogicalSide logical_side)
|
||
{
|
||
// FIXME: Actually determine the logical sides based on layout_node's writing-mode and direction.
|
||
switch (logical_side) {
|
||
case LogicalSide::BlockStart:
|
||
return box.top;
|
||
case LogicalSide::BlockEnd:
|
||
return box.bottom;
|
||
case LogicalSide::InlineStart:
|
||
return box.left;
|
||
case LogicalSide::InlineEnd:
|
||
return box.right;
|
||
}
|
||
VERIFY_NOT_REACHED();
|
||
}
|
||
|
||
static RefPtr<CSSStyleValue const> style_value_for_shadow(Vector<ShadowData> const& shadow_data)
|
||
{
|
||
if (shadow_data.is_empty())
|
||
return CSSKeywordValue::create(Keyword::None);
|
||
|
||
auto make_shadow_style_value = [](ShadowData const& shadow) {
|
||
return ShadowStyleValue::create(
|
||
CSSColorValue::create_from_color(shadow.color, ColorSyntax::Modern),
|
||
style_value_for_length_percentage(shadow.offset_x),
|
||
style_value_for_length_percentage(shadow.offset_y),
|
||
style_value_for_length_percentage(shadow.blur_radius),
|
||
style_value_for_length_percentage(shadow.spread_distance),
|
||
shadow.placement);
|
||
};
|
||
|
||
if (shadow_data.size() == 1)
|
||
return make_shadow_style_value(shadow_data.first());
|
||
|
||
StyleValueVector style_values;
|
||
style_values.ensure_capacity(shadow_data.size());
|
||
for (auto& shadow : shadow_data)
|
||
style_values.unchecked_append(make_shadow_style_value(shadow));
|
||
|
||
return StyleValueList::create(move(style_values), StyleValueList::Separator::Comma);
|
||
}
|
||
|
||
// https://drafts.csswg.org/cssom/#dom-cssstyledeclaration-getpropertyvalue
|
||
String CSSStyleProperties::get_property_value(StringView property_name) const
|
||
{
|
||
auto property_id = property_id_from_string(property_name);
|
||
if (!property_id.has_value())
|
||
return {};
|
||
|
||
if (property_id.value() == PropertyID::Custom) {
|
||
auto maybe_custom_property = custom_property(FlyString::from_utf8_without_validation(property_name.bytes()));
|
||
if (maybe_custom_property.has_value()) {
|
||
return maybe_custom_property.value().value->to_string(
|
||
is_computed() ? CSSStyleValue::SerializationMode::ResolvedValue
|
||
: CSSStyleValue::SerializationMode::Normal);
|
||
}
|
||
return {};
|
||
}
|
||
|
||
auto maybe_property = get_property_internal(property_id.value());
|
||
if (!maybe_property.has_value())
|
||
return {};
|
||
return maybe_property->value->to_string(
|
||
is_computed() ? CSSStyleValue::SerializationMode::ResolvedValue
|
||
: CSSStyleValue::SerializationMode::Normal);
|
||
}
|
||
|
||
// https://drafts.csswg.org/cssom/#dom-cssstyledeclaration-getpropertypriority
|
||
StringView CSSStyleProperties::get_property_priority(StringView property_name) const
|
||
{
|
||
auto property_id = property_id_from_string(property_name);
|
||
if (!property_id.has_value())
|
||
return {};
|
||
if (property_id.value() == PropertyID::Custom) {
|
||
auto maybe_custom_property = custom_property(FlyString::from_utf8_without_validation(property_name.bytes()));
|
||
if (!maybe_custom_property.has_value())
|
||
return {};
|
||
return maybe_custom_property.value().important == Important::Yes ? "important"sv : ""sv;
|
||
}
|
||
auto maybe_property = property(property_id.value());
|
||
if (!maybe_property.has_value())
|
||
return {};
|
||
return maybe_property->important == Important::Yes ? "important"sv : ""sv;
|
||
}
|
||
|
||
static Optional<StyleProperty> style_property_for_sided_shorthand(PropertyID property_id, Optional<StyleProperty> const& top, Optional<StyleProperty> const& right, Optional<StyleProperty> const& bottom, Optional<StyleProperty> const& left)
|
||
{
|
||
if (!top.has_value() || !right.has_value() || !bottom.has_value() || !left.has_value())
|
||
return {};
|
||
|
||
if (top->important != right->important || top->important != bottom->important || top->important != left->important)
|
||
return {};
|
||
|
||
ValueComparingNonnullRefPtr<CSSStyleValue> const top_value { top->value };
|
||
ValueComparingNonnullRefPtr<CSSStyleValue> const right_value { right->value };
|
||
ValueComparingNonnullRefPtr<CSSStyleValue> const bottom_value { bottom->value };
|
||
ValueComparingNonnullRefPtr<CSSStyleValue> const left_value { left->value };
|
||
|
||
bool const top_and_bottom_same = top_value == bottom_value;
|
||
bool const left_and_right_same = left_value == right_value;
|
||
|
||
RefPtr<CSSStyleValue const> value;
|
||
|
||
if (top_and_bottom_same && left_and_right_same && top_value == left_value) {
|
||
value = top_value;
|
||
} else if (top_and_bottom_same && left_and_right_same) {
|
||
value = StyleValueList::create(StyleValueVector { top_value, right_value }, StyleValueList::Separator::Space);
|
||
} else if (left_and_right_same) {
|
||
value = StyleValueList::create(StyleValueVector { top_value, right_value, bottom_value }, StyleValueList::Separator::Space);
|
||
} else {
|
||
value = StyleValueList::create(StyleValueVector { top_value, right_value, bottom_value, left_value }, StyleValueList::Separator::Space);
|
||
}
|
||
|
||
return StyleProperty {
|
||
.important = top->important,
|
||
.property_id = property_id,
|
||
.value = value.release_nonnull(),
|
||
};
|
||
}
|
||
|
||
// https://drafts.csswg.org/cssom/#dom-cssstyledeclaration-getpropertyvalue
|
||
Optional<StyleProperty> CSSStyleProperties::get_property_internal(PropertyID property_id) const
|
||
{
|
||
// 2. If property is a shorthand property, then follow these substeps:
|
||
if (property_is_shorthand(property_id)) {
|
||
|
||
// AD-HOC: Handle shorthands that require manual construction.
|
||
switch (property_id) {
|
||
case PropertyID::Border: {
|
||
auto width = get_property_internal(PropertyID::BorderWidth);
|
||
auto style = get_property_internal(PropertyID::BorderStyle);
|
||
auto color = get_property_internal(PropertyID::BorderColor);
|
||
// `border` only has a reasonable value if all four sides are the same.
|
||
if (!width.has_value() || width->value->is_value_list() || !style.has_value() || style->value->is_value_list() || !color.has_value() || color->value->is_value_list())
|
||
return {};
|
||
if (width->important != style->important || width->important != color->important)
|
||
return {};
|
||
return StyleProperty {
|
||
.important = width->important,
|
||
.property_id = property_id,
|
||
.value = ShorthandStyleValue::create(property_id,
|
||
{ PropertyID::BorderWidth, PropertyID::BorderStyle, PropertyID::BorderColor },
|
||
{ width->value, style->value, color->value })
|
||
};
|
||
}
|
||
case PropertyID::BorderColor: {
|
||
auto top = get_property_internal(PropertyID::BorderTopColor);
|
||
auto right = get_property_internal(PropertyID::BorderRightColor);
|
||
auto bottom = get_property_internal(PropertyID::BorderBottomColor);
|
||
auto left = get_property_internal(PropertyID::BorderLeftColor);
|
||
return style_property_for_sided_shorthand(property_id, top, right, bottom, left);
|
||
}
|
||
case PropertyID::BorderStyle: {
|
||
auto top = get_property_internal(PropertyID::BorderTopStyle);
|
||
auto right = get_property_internal(PropertyID::BorderRightStyle);
|
||
auto bottom = get_property_internal(PropertyID::BorderBottomStyle);
|
||
auto left = get_property_internal(PropertyID::BorderLeftStyle);
|
||
return style_property_for_sided_shorthand(property_id, top, right, bottom, left);
|
||
}
|
||
case PropertyID::BorderWidth: {
|
||
auto top = get_property_internal(PropertyID::BorderTopWidth);
|
||
auto right = get_property_internal(PropertyID::BorderRightWidth);
|
||
auto bottom = get_property_internal(PropertyID::BorderBottomWidth);
|
||
auto left = get_property_internal(PropertyID::BorderLeftWidth);
|
||
return style_property_for_sided_shorthand(property_id, top, right, bottom, left);
|
||
}
|
||
case PropertyID::FontVariant: {
|
||
auto ligatures = get_property_internal(PropertyID::FontVariantLigatures);
|
||
auto caps = get_property_internal(PropertyID::FontVariantCaps);
|
||
auto alternates = get_property_internal(PropertyID::FontVariantAlternates);
|
||
auto numeric = get_property_internal(PropertyID::FontVariantNumeric);
|
||
auto east_asian = get_property_internal(PropertyID::FontVariantEastAsian);
|
||
auto position = get_property_internal(PropertyID::FontVariantPosition);
|
||
auto emoji = get_property_internal(PropertyID::FontVariantEmoji);
|
||
|
||
if (!ligatures.has_value() || !caps.has_value() || !alternates.has_value() || !numeric.has_value() || !east_asian.has_value() || !position.has_value() || !emoji.has_value())
|
||
return {};
|
||
|
||
if (ligatures->important != caps->important || ligatures->important != alternates->important || ligatures->important != numeric->important || ligatures->important != east_asian->important || ligatures->important != position->important || ligatures->important != emoji->important)
|
||
return {};
|
||
|
||
// If ligatures is `none` and any other value isn't `normal`, that's invalid.
|
||
if (ligatures->value->to_keyword() == Keyword::None
|
||
&& (caps->value->to_keyword() != Keyword::Normal
|
||
|| alternates->value->to_keyword() != Keyword::Normal
|
||
|| numeric->value->to_keyword() != Keyword::Normal
|
||
|| east_asian->value->to_keyword() != Keyword::Normal
|
||
|| position->value->to_keyword() != Keyword::Normal
|
||
|| emoji->value->to_keyword() != Keyword::Normal)) {
|
||
return {};
|
||
}
|
||
|
||
return StyleProperty {
|
||
.important = ligatures->important,
|
||
.property_id = property_id,
|
||
.value = ShorthandStyleValue::create(property_id,
|
||
{ PropertyID::FontVariantLigatures, PropertyID::FontVariantCaps, PropertyID::FontVariantAlternates, PropertyID::FontVariantNumeric, PropertyID::FontVariantEastAsian, PropertyID::FontVariantPosition, PropertyID::FontVariantEmoji },
|
||
{ ligatures->value, caps->value, alternates->value, numeric->value, east_asian->value, position->value, emoji->value })
|
||
};
|
||
}
|
||
case PropertyID::Margin: {
|
||
auto top = get_property_internal(PropertyID::MarginTop);
|
||
auto right = get_property_internal(PropertyID::MarginRight);
|
||
auto bottom = get_property_internal(PropertyID::MarginBottom);
|
||
auto left = get_property_internal(PropertyID::MarginLeft);
|
||
return style_property_for_sided_shorthand(property_id, top, right, bottom, left);
|
||
}
|
||
case PropertyID::Padding: {
|
||
auto top = get_property_internal(PropertyID::PaddingTop);
|
||
auto right = get_property_internal(PropertyID::PaddingRight);
|
||
auto bottom = get_property_internal(PropertyID::PaddingBottom);
|
||
auto left = get_property_internal(PropertyID::PaddingLeft);
|
||
return style_property_for_sided_shorthand(property_id, top, right, bottom, left);
|
||
}
|
||
default:
|
||
break;
|
||
}
|
||
|
||
// 1. Let list be a new empty array.
|
||
Vector<ValueComparingNonnullRefPtr<CSSStyleValue const>> list;
|
||
Optional<Important> last_important_flag;
|
||
|
||
// 2. For each longhand property longhand that property maps to, in canonical order, follow these substeps:
|
||
Vector<PropertyID> longhand_ids = longhands_for_shorthand(property_id);
|
||
for (auto longhand_property_id : longhand_ids) {
|
||
// 1. If longhand is a case-sensitive match for a property name of a CSS declaration in the declarations,
|
||
// let declaration be that CSS declaration, or null otherwise.
|
||
auto declaration = get_property_internal(longhand_property_id);
|
||
|
||
// 2. If declaration is null, then return the empty string.
|
||
if (!declaration.has_value())
|
||
return {};
|
||
|
||
// 3. Append the declaration to list.
|
||
list.append(declaration->value);
|
||
|
||
if (last_important_flag.has_value() && declaration->important != *last_important_flag)
|
||
return {};
|
||
last_important_flag = declaration->important;
|
||
}
|
||
|
||
// 3. If important flags of all declarations in list are same, then return the serialization of list.
|
||
// NOTE: Currently we implement property-specific shorthand serialization in ShorthandStyleValue::to_string().
|
||
return StyleProperty {
|
||
.important = last_important_flag.value(),
|
||
.property_id = property_id,
|
||
.value = ShorthandStyleValue::create(property_id, longhand_ids, list),
|
||
};
|
||
|
||
// 4. Return the empty string.
|
||
// NOTE: This is handled by the loop.
|
||
}
|
||
|
||
return property(property_id);
|
||
}
|
||
|
||
RefPtr<CSSStyleValue const> CSSStyleProperties::style_value_for_computed_property(Layout::NodeWithStyle const& layout_node, PropertyID property_id) const
|
||
{
|
||
auto used_value_for_property = [&layout_node, property_id](Function<CSSPixels(Painting::PaintableBox const&)>&& used_value_getter) -> Optional<CSSPixels> {
|
||
auto const& display = layout_node.computed_values().display();
|
||
if (!display.is_none() && !display.is_contents() && layout_node.first_paintable()) {
|
||
if (layout_node.first_paintable()->is_paintable_box()) {
|
||
auto const& paintable_box = static_cast<Painting::PaintableBox const&>(*layout_node.first_paintable());
|
||
return used_value_getter(paintable_box);
|
||
}
|
||
dbgln("FIXME: Support getting used value for property `{}` on {}", string_from_property_id(property_id), layout_node.debug_description());
|
||
}
|
||
return {};
|
||
};
|
||
|
||
auto& element = owner_node()->element();
|
||
auto pseudo_element = owner_node()->pseudo_element();
|
||
|
||
auto used_value_for_inset = [&layout_node, used_value_for_property](LengthPercentage const& start_side, LengthPercentage const& end_side, Function<CSSPixels(Painting::PaintableBox const&)>&& used_value_getter) -> Optional<CSSPixels> {
|
||
if (!layout_node.is_positioned())
|
||
return {};
|
||
|
||
// FIXME: Support getting the used value when position is sticky.
|
||
if (layout_node.is_sticky_position())
|
||
return {};
|
||
|
||
if (!start_side.is_percentage() && !start_side.is_calculated() && !start_side.is_auto() && !end_side.is_auto())
|
||
return {};
|
||
|
||
return used_value_for_property(move(used_value_getter));
|
||
};
|
||
|
||
auto get_computed_value = [&element, pseudo_element](PropertyID property_id) -> auto const& {
|
||
if (pseudo_element.has_value())
|
||
return element.pseudo_element_computed_properties(pseudo_element.value())->property(property_id);
|
||
return element.computed_properties()->property(property_id);
|
||
};
|
||
|
||
// A limited number of properties have special rules for producing their "resolved value".
|
||
// We also have to manually construct shorthands from their longhands here.
|
||
// Everything else uses the computed value.
|
||
// https://drafts.csswg.org/cssom/#resolved-values
|
||
|
||
// The resolved value for a given longhand property can be determined as follows:
|
||
switch (property_id) {
|
||
// -> background-color
|
||
// -> border-block-end-color
|
||
// -> border-block-start-color
|
||
// -> border-bottom-color
|
||
// -> border-inline-end-color
|
||
// -> border-inline-start-color
|
||
// -> border-left-color
|
||
// -> border-right-color
|
||
// -> border-top-color
|
||
// -> box-shadow
|
||
// -> caret-color
|
||
// -> color
|
||
// -> outline-color
|
||
// -> A resolved value special case property like color defined in another specification
|
||
// The resolved value is the used value.
|
||
case PropertyID::BackgroundColor:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().background_color(), ColorSyntax::Modern);
|
||
case PropertyID::BorderBlockEndColor:
|
||
// FIXME: Honor writing-mode, direction and text-orientation.
|
||
return style_value_for_computed_property(layout_node, PropertyID::BorderBottomColor);
|
||
case PropertyID::BorderBlockStartColor:
|
||
// FIXME: Honor writing-mode, direction and text-orientation.
|
||
return style_value_for_computed_property(layout_node, PropertyID::BorderTopColor);
|
||
case PropertyID::BorderBottomColor:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().border_bottom().color, ColorSyntax::Modern);
|
||
case PropertyID::BorderInlineEndColor:
|
||
// FIXME: Honor writing-mode, direction and text-orientation.
|
||
return style_value_for_computed_property(layout_node, PropertyID::BorderRightColor);
|
||
case PropertyID::BorderInlineStartColor:
|
||
// FIXME: Honor writing-mode, direction and text-orientation.
|
||
return style_value_for_computed_property(layout_node, PropertyID::BorderLeftColor);
|
||
case PropertyID::BorderLeftColor:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().border_left().color, ColorSyntax::Modern);
|
||
case PropertyID::BorderRightColor:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().border_right().color, ColorSyntax::Modern);
|
||
case PropertyID::BorderTopColor:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().border_top().color, ColorSyntax::Modern);
|
||
case PropertyID::BoxShadow:
|
||
return style_value_for_shadow(layout_node.computed_values().box_shadow());
|
||
case PropertyID::CaretColor:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().caret_color(), ColorSyntax::Modern);
|
||
case PropertyID::Color:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().color(), ColorSyntax::Modern);
|
||
case PropertyID::OutlineColor:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().outline_color(), ColorSyntax::Modern);
|
||
case PropertyID::TextDecorationColor:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().text_decoration_color(), ColorSyntax::Modern);
|
||
// NB: text-shadow isn't listed, but is computed the same as box-shadow.
|
||
case PropertyID::TextShadow:
|
||
return style_value_for_shadow(layout_node.computed_values().text_shadow());
|
||
|
||
// -> line-height
|
||
// The resolved value is normal if the computed value is normal, or the used value otherwise.
|
||
case PropertyID::LineHeight: {
|
||
auto const& line_height = get_computed_value(property_id);
|
||
if (line_height.is_keyword() && line_height.to_keyword() == Keyword::Normal)
|
||
return line_height;
|
||
return LengthStyleValue::create(Length::make_px(layout_node.computed_values().line_height()));
|
||
}
|
||
|
||
// -> block-size
|
||
// -> height
|
||
// -> inline-size
|
||
// -> margin-block-end
|
||
// -> margin-block-start
|
||
// -> margin-bottom
|
||
// -> margin-inline-end
|
||
// -> margin-inline-start
|
||
// -> margin-left
|
||
// -> margin-right
|
||
// -> margin-top
|
||
// -> padding-block-end
|
||
// -> padding-block-start
|
||
// -> padding-bottom
|
||
// -> padding-inline-end
|
||
// -> padding-inline-start
|
||
// -> padding-left
|
||
// -> padding-right
|
||
// -> padding-top
|
||
// -> width
|
||
// If the property applies to the element or pseudo-element and the resolved value of the
|
||
// display property is not none or contents, then the resolved value is the used value.
|
||
// Otherwise the resolved value is the computed value.
|
||
case PropertyID::BlockSize: {
|
||
auto writing_mode = layout_node.computed_values().writing_mode();
|
||
auto is_vertically_oriented = first_is_one_of(writing_mode, WritingMode::VerticalLr, WritingMode::VerticalRl);
|
||
if (is_vertically_oriented)
|
||
return style_value_for_computed_property(layout_node, PropertyID::Width);
|
||
return style_value_for_computed_property(layout_node, PropertyID::Height);
|
||
}
|
||
case PropertyID::Height: {
|
||
auto maybe_used_height = used_value_for_property([](auto const& paintable_box) { return paintable_box.content_height(); });
|
||
if (maybe_used_height.has_value())
|
||
return style_value_for_size(Size::make_px(maybe_used_height.release_value()));
|
||
return style_value_for_size(layout_node.computed_values().height());
|
||
}
|
||
case PropertyID::InlineSize: {
|
||
auto writing_mode = layout_node.computed_values().writing_mode();
|
||
auto is_vertically_oriented = first_is_one_of(writing_mode, WritingMode::VerticalLr, WritingMode::VerticalRl);
|
||
if (is_vertically_oriented)
|
||
return style_value_for_computed_property(layout_node, PropertyID::Height);
|
||
return style_value_for_computed_property(layout_node, PropertyID::Width);
|
||
}
|
||
case PropertyID::MarginBlockEnd:
|
||
if (auto maybe_used_value = used_value_for_property([&](auto const& paintable_box) { return pixels_for_pixel_box_logical_side(layout_node, paintable_box.box_model().margin, LogicalSide::BlockEnd); }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().margin(), LogicalSide::BlockEnd);
|
||
case PropertyID::MarginBlockStart:
|
||
if (auto maybe_used_value = used_value_for_property([&](auto const& paintable_box) { return pixels_for_pixel_box_logical_side(layout_node, paintable_box.box_model().margin, LogicalSide::BlockStart); }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().margin(), LogicalSide::BlockStart);
|
||
case PropertyID::MarginBottom:
|
||
if (auto maybe_used_value = used_value_for_property([](auto const& paintable_box) { return paintable_box.box_model().margin.bottom; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_percentage(layout_node.computed_values().margin().bottom());
|
||
case PropertyID::MarginInlineEnd:
|
||
if (auto maybe_used_value = used_value_for_property([&](auto const& paintable_box) { return pixels_for_pixel_box_logical_side(layout_node, paintable_box.box_model().margin, LogicalSide::InlineEnd); }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().margin(), LogicalSide::InlineEnd);
|
||
case PropertyID::MarginInlineStart:
|
||
if (auto maybe_used_value = used_value_for_property([&](auto const& paintable_box) { return pixels_for_pixel_box_logical_side(layout_node, paintable_box.box_model().margin, LogicalSide::InlineStart); }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().margin(), LogicalSide::InlineStart);
|
||
case PropertyID::MarginLeft:
|
||
if (auto maybe_used_value = used_value_for_property([](auto const& paintable_box) { return paintable_box.box_model().margin.left; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_percentage(layout_node.computed_values().margin().left());
|
||
case PropertyID::MarginRight:
|
||
if (auto maybe_used_value = used_value_for_property([](auto const& paintable_box) { return paintable_box.box_model().margin.right; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_percentage(layout_node.computed_values().margin().right());
|
||
case PropertyID::MarginTop:
|
||
if (auto maybe_used_value = used_value_for_property([](auto const& paintable_box) { return paintable_box.box_model().margin.top; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_percentage(layout_node.computed_values().margin().top());
|
||
case PropertyID::PaddingBlockEnd:
|
||
if (auto maybe_used_value = used_value_for_property([&](auto const& paintable_box) { return pixels_for_pixel_box_logical_side(layout_node, paintable_box.box_model().padding, LogicalSide::BlockEnd); }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().padding(), LogicalSide::BlockEnd);
|
||
case PropertyID::PaddingBlockStart:
|
||
if (auto maybe_used_value = used_value_for_property([&](auto const& paintable_box) { return pixels_for_pixel_box_logical_side(layout_node, paintable_box.box_model().padding, LogicalSide::BlockStart); }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().padding(), LogicalSide::BlockStart);
|
||
case PropertyID::PaddingBottom:
|
||
if (auto maybe_used_value = used_value_for_property([](auto const& paintable_box) { return paintable_box.box_model().padding.bottom; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_percentage(layout_node.computed_values().padding().bottom());
|
||
case PropertyID::PaddingInlineEnd:
|
||
if (auto maybe_used_value = used_value_for_property([&](auto const& paintable_box) { return pixels_for_pixel_box_logical_side(layout_node, paintable_box.box_model().padding, LogicalSide::InlineEnd); }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().padding(), LogicalSide::InlineEnd);
|
||
case PropertyID::PaddingInlineStart:
|
||
if (auto maybe_used_value = used_value_for_property([&](auto const& paintable_box) { return pixels_for_pixel_box_logical_side(layout_node, paintable_box.box_model().padding, LogicalSide::InlineStart); }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().padding(), LogicalSide::InlineStart);
|
||
case PropertyID::PaddingLeft:
|
||
if (auto maybe_used_value = used_value_for_property([](auto const& paintable_box) { return paintable_box.box_model().padding.left; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_percentage(layout_node.computed_values().padding().left());
|
||
case PropertyID::PaddingRight:
|
||
if (auto maybe_used_value = used_value_for_property([](auto const& paintable_box) { return paintable_box.box_model().padding.right; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_percentage(layout_node.computed_values().padding().right());
|
||
case PropertyID::PaddingTop:
|
||
if (auto maybe_used_value = used_value_for_property([](auto const& paintable_box) { return paintable_box.box_model().padding.top; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_percentage(layout_node.computed_values().padding().top());
|
||
case PropertyID::Width: {
|
||
auto maybe_used_width = used_value_for_property([](auto const& paintable_box) { return paintable_box.content_width(); });
|
||
if (maybe_used_width.has_value())
|
||
return style_value_for_size(Size::make_px(maybe_used_width.release_value()));
|
||
return style_value_for_size(layout_node.computed_values().width());
|
||
}
|
||
|
||
// -> bottom
|
||
// -> left
|
||
// -> inset-block-end
|
||
// -> inset-block-start
|
||
// -> inset-inline-end
|
||
// -> inset-inline-start
|
||
// -> right
|
||
// -> top
|
||
// -> A resolved value special case property like top defined in another specification
|
||
// If the property applies to a positioned element and the resolved value of the display property is not
|
||
// none or contents, and the property is not over-constrained, then the resolved value is the used value.
|
||
// Otherwise the resolved value is the computed value.
|
||
case PropertyID::Bottom: {
|
||
auto& inset = layout_node.computed_values().inset();
|
||
if (auto maybe_used_value = used_value_for_inset(inset.bottom(), inset.top(), [](auto const& paintable_box) { return paintable_box.box_model().inset.bottom; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
|
||
return style_value_for_length_percentage(inset.bottom());
|
||
}
|
||
case PropertyID::InsetBlockEnd:
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().inset(), LogicalSide::BlockEnd);
|
||
case PropertyID::InsetBlockStart:
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().inset(), LogicalSide::BlockStart);
|
||
case PropertyID::InsetInlineEnd:
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().inset(), LogicalSide::InlineEnd);
|
||
case PropertyID::InsetInlineStart:
|
||
return style_value_for_length_box_logical_side(layout_node, layout_node.computed_values().inset(), LogicalSide::InlineStart);
|
||
case PropertyID::Left: {
|
||
auto& inset = layout_node.computed_values().inset();
|
||
if (auto maybe_used_value = used_value_for_inset(inset.left(), inset.right(), [](auto const& paintable_box) { return paintable_box.box_model().inset.left; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
return style_value_for_length_percentage(inset.left());
|
||
}
|
||
case PropertyID::Right: {
|
||
auto& inset = layout_node.computed_values().inset();
|
||
if (auto maybe_used_value = used_value_for_inset(inset.right(), inset.left(), [](auto const& paintable_box) { return paintable_box.box_model().inset.right; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
|
||
return style_value_for_length_percentage(inset.right());
|
||
}
|
||
case PropertyID::Top: {
|
||
auto& inset = layout_node.computed_values().inset();
|
||
if (auto maybe_used_value = used_value_for_inset(inset.top(), inset.bottom(), [](auto const& paintable_box) { return paintable_box.box_model().inset.top; }); maybe_used_value.has_value())
|
||
return LengthStyleValue::create(Length::make_px(maybe_used_value.release_value()));
|
||
|
||
return style_value_for_length_percentage(inset.top());
|
||
}
|
||
|
||
// -> A resolved value special case property defined in another specification
|
||
// As defined in the relevant specification.
|
||
case PropertyID::Transform: {
|
||
auto transformations = layout_node.computed_values().transformations();
|
||
if (transformations.is_empty())
|
||
return CSSKeywordValue::create(Keyword::None);
|
||
|
||
// https://drafts.csswg.org/css-transforms-2/#serialization-of-the-computed-value
|
||
// The transform property is a resolved value special case property. [CSSOM]
|
||
// When the computed value is a <transform-list>, the resolved value is one <matrix()> function or one <matrix3d()> function computed by the following algorithm:
|
||
// 1. Let transform be a 4x4 matrix initialized to the identity matrix.
|
||
// The elements m11, m22, m33 and m44 of transform must be set to 1; all other elements of transform must be set to 0.
|
||
auto transform = FloatMatrix4x4::identity();
|
||
|
||
// 2. Post-multiply all <transform-function>s in <transform-list> to transform.
|
||
VERIFY(layout_node.first_paintable());
|
||
auto const& paintable_box = as<Painting::PaintableBox const>(*layout_node.first_paintable());
|
||
for (auto transformation : transformations) {
|
||
transform = transform * transformation.to_matrix(paintable_box).release_value();
|
||
}
|
||
|
||
// https://drafts.csswg.org/css-transforms-1/#2d-matrix
|
||
auto is_2d_matrix = [](Gfx::FloatMatrix4x4 const& matrix) -> bool {
|
||
// A 3x2 transformation matrix,
|
||
// or a 4x4 matrix where the items m31, m32, m13, m23, m43, m14, m24, m34 are equal to 0
|
||
// and m33, m44 are equal to 1.
|
||
// NB: We only care about 4x4 matrices here.
|
||
// NB: Our elements are 0-indexed not 1-indexed, and in the opposite order.
|
||
if (matrix.elements()[0][2] != 0 // m31
|
||
|| matrix.elements()[1][2] != 0 // m32
|
||
|| matrix.elements()[2][0] != 0 // m13
|
||
|| matrix.elements()[2][1] != 0 // m23
|
||
|| matrix.elements()[2][3] != 0 // m43
|
||
|| matrix.elements()[3][0] != 0 // m14
|
||
|| matrix.elements()[3][1] != 0 // m24
|
||
|| matrix.elements()[3][2] != 0) // m34
|
||
return false;
|
||
|
||
if (matrix.elements()[2][2] != 1 // m33
|
||
|| matrix.elements()[3][3] != 1) // m44
|
||
return false;
|
||
|
||
return true;
|
||
};
|
||
|
||
// 3. Chose between <matrix()> or <matrix3d()> serialization:
|
||
// -> If transform is a 2D matrix
|
||
// Serialize transform to a <matrix()> function.
|
||
if (is_2d_matrix(transform)) {
|
||
StyleValueVector parameters {
|
||
NumberStyleValue::create(transform.elements()[0][0]),
|
||
NumberStyleValue::create(transform.elements()[1][0]),
|
||
NumberStyleValue::create(transform.elements()[0][1]),
|
||
NumberStyleValue::create(transform.elements()[1][1]),
|
||
NumberStyleValue::create(transform.elements()[0][3]),
|
||
NumberStyleValue::create(transform.elements()[1][3]),
|
||
};
|
||
return TransformationStyleValue::create(PropertyID::Transform, TransformFunction::Matrix, move(parameters));
|
||
}
|
||
// -> Otherwise
|
||
// Serialize transform to a <matrix3d()> function.
|
||
else {
|
||
StyleValueVector parameters {
|
||
NumberStyleValue::create(transform.elements()[0][0]),
|
||
NumberStyleValue::create(transform.elements()[1][0]),
|
||
NumberStyleValue::create(transform.elements()[2][0]),
|
||
NumberStyleValue::create(transform.elements()[3][0]),
|
||
NumberStyleValue::create(transform.elements()[0][1]),
|
||
NumberStyleValue::create(transform.elements()[1][1]),
|
||
NumberStyleValue::create(transform.elements()[2][1]),
|
||
NumberStyleValue::create(transform.elements()[3][1]),
|
||
NumberStyleValue::create(transform.elements()[0][2]),
|
||
NumberStyleValue::create(transform.elements()[1][2]),
|
||
NumberStyleValue::create(transform.elements()[2][2]),
|
||
NumberStyleValue::create(transform.elements()[3][2]),
|
||
NumberStyleValue::create(transform.elements()[0][3]),
|
||
NumberStyleValue::create(transform.elements()[1][3]),
|
||
NumberStyleValue::create(transform.elements()[2][3]),
|
||
NumberStyleValue::create(transform.elements()[3][3]),
|
||
};
|
||
return TransformationStyleValue::create(PropertyID::Transform, TransformFunction::Matrix3d, move(parameters));
|
||
}
|
||
}
|
||
|
||
// -> Any other property
|
||
// The resolved value is the computed value.
|
||
// NOTE: This is handled inside the `default` case.
|
||
case PropertyID::WebkitTextFillColor:
|
||
return CSSColorValue::create_from_color(layout_node.computed_values().webkit_text_fill_color(), ColorSyntax::Modern);
|
||
case PropertyID::Invalid:
|
||
return CSSKeywordValue::create(Keyword::Invalid);
|
||
case PropertyID::Custom:
|
||
dbgln_if(LIBWEB_CSS_DEBUG, "Computed style for custom properties was requested (?)");
|
||
return nullptr;
|
||
default:
|
||
// For grid-template-columns and grid-template-rows the resolved value is the used value.
|
||
// https://www.w3.org/TR/css-grid-2/#resolved-track-list-standalone
|
||
if (property_id == PropertyID::GridTemplateColumns) {
|
||
if (layout_node.first_paintable() && layout_node.first_paintable()->is_paintable_box()) {
|
||
auto const& paintable_box = as<Painting::PaintableBox const>(*layout_node.first_paintable());
|
||
if (auto used_values_for_grid_template_columns = paintable_box.used_values_for_grid_template_columns()) {
|
||
return used_values_for_grid_template_columns;
|
||
}
|
||
}
|
||
} else if (property_id == PropertyID::GridTemplateRows) {
|
||
if (layout_node.first_paintable() && layout_node.first_paintable()->is_paintable_box()) {
|
||
auto const& paintable_box = as<Painting::PaintableBox const>(*layout_node.first_paintable());
|
||
if (auto used_values_for_grid_template_rows = paintable_box.used_values_for_grid_template_rows()) {
|
||
return used_values_for_grid_template_rows;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (!property_is_shorthand(property_id))
|
||
return get_computed_value(property_id);
|
||
|
||
// Handle shorthands in a generic way
|
||
auto longhand_ids = longhands_for_shorthand(property_id);
|
||
StyleValueVector longhand_values;
|
||
longhand_values.ensure_capacity(longhand_ids.size());
|
||
for (auto longhand_id : longhand_ids)
|
||
longhand_values.append(style_value_for_computed_property(layout_node, longhand_id).release_nonnull());
|
||
return ShorthandStyleValue::create(property_id, move(longhand_ids), move(longhand_values));
|
||
}
|
||
}
|
||
|
||
// https://drafts.csswg.org/cssom/#dom-cssstyledeclaration-removeproperty
|
||
WebIDL::ExceptionOr<String> CSSStyleProperties::remove_property(StringView property_name)
|
||
{
|
||
// 1. If the readonly flag is set, then throw a NoModificationAllowedError exception.
|
||
if (is_readonly())
|
||
return WebIDL::NoModificationAllowedError::create(realm(), "Cannot remove property: CSSStyleProperties is read-only."_string);
|
||
|
||
auto property_id = property_id_from_string(property_name);
|
||
if (!property_id.has_value())
|
||
return String {};
|
||
|
||
// 2. If property is not a custom property, let property be property converted to ASCII lowercase.
|
||
// NB: We've already converted it to a PropertyID enum value.
|
||
|
||
// 3. Let value be the return value of invoking getPropertyValue() with property as argument.
|
||
auto value = get_property_value(property_name);
|
||
|
||
// 4. Let removed be false.
|
||
bool removed = false;
|
||
|
||
// FIXME: 5. If property is a shorthand property, for each longhand property longhand that property maps to:
|
||
// 1. If longhand is not a property name of a CSS declaration in the declarations, continue.
|
||
// 2. Remove that CSS declaration and let removed be true.
|
||
|
||
// 6. Otherwise, if property is a case-sensitive match for a property name of a CSS declaration in the declarations, remove that CSS declaration and let removed be true.
|
||
if (property_id == PropertyID::Custom) {
|
||
auto custom_name = FlyString::from_utf8_without_validation(property_name.bytes());
|
||
removed = m_custom_properties.remove(custom_name);
|
||
} else {
|
||
removed = m_properties.remove_first_matching([&](auto& entry) { return entry.property_id == property_id; });
|
||
}
|
||
|
||
// 7. If removed is true, Update style attribute for the CSS declaration block.
|
||
if (removed)
|
||
update_style_attribute();
|
||
|
||
// 8. Return value.
|
||
return value;
|
||
}
|
||
|
||
WebIDL::ExceptionOr<String> CSSStyleProperties::remove_property(PropertyID property_name)
|
||
{
|
||
return remove_property(string_from_property_id(property_name));
|
||
}
|
||
|
||
// https://drafts.csswg.org/cssom/#dom-cssstyleproperties-cssfloat
|
||
String CSSStyleProperties::css_float() const
|
||
{
|
||
// The cssFloat attribute, on getting, must return the result of invoking getPropertyValue() with float as argument.
|
||
return get_property_value("float"sv);
|
||
}
|
||
|
||
WebIDL::ExceptionOr<void> CSSStyleProperties::set_css_float(StringView value)
|
||
{
|
||
// On setting, the attribute must invoke setProperty() with float as first argument, as second argument the given value,
|
||
// and no third argument. Any exceptions thrown must be re-thrown.
|
||
return set_property("float"sv, value, ""sv);
|
||
}
|
||
|
||
// https://www.w3.org/TR/cssom/#serialize-a-css-declaration
|
||
static String serialize_a_css_declaration(CSS::PropertyID property, StringView value, Important important)
|
||
{
|
||
StringBuilder builder;
|
||
|
||
// 1. Let s be the empty string.
|
||
// 2. Append property to s.
|
||
builder.append(string_from_property_id(property));
|
||
|
||
// 3. Append ": " (U+003A U+0020) to s.
|
||
builder.append(": "sv);
|
||
|
||
// 4. Append value to s.
|
||
builder.append(value);
|
||
|
||
// 5. If the important flag is set, append " !important" (U+0020 U+0021 U+0069 U+006D U+0070 U+006F U+0072 U+0074 U+0061 U+006E U+0074) to s.
|
||
if (important == Important::Yes)
|
||
builder.append(" !important"sv);
|
||
|
||
// 6. Append ";" (U+003B) to s.
|
||
builder.append(';');
|
||
|
||
// 7. Return s.
|
||
return MUST(builder.to_string());
|
||
}
|
||
|
||
// https://www.w3.org/TR/cssom/#serialize-a-css-declaration-block
|
||
String CSSStyleProperties::serialized() const
|
||
{
|
||
// 1. Let list be an empty array.
|
||
Vector<String> list;
|
||
|
||
// 2. Let already serialized be an empty array.
|
||
HashTable<PropertyID> already_serialized;
|
||
|
||
// NB: The spec treats custom properties the same as any other property, and expects the above loop to handle them.
|
||
// However, our implementation separates them from regular properties, so we need to handle them separately here.
|
||
// FIXME: Is the relative order of custom properties and regular properties supposed to be preserved?
|
||
for (auto& declaration : m_custom_properties) {
|
||
// 1. Let property be declaration’s property name.
|
||
auto const& property = declaration.key;
|
||
|
||
// 2. If property is in already serialized, continue with the steps labeled declaration loop.
|
||
// NB: It is never in already serialized, as there are no shorthands for custom properties.
|
||
|
||
// 3. If property maps to one or more shorthand properties, let shorthands be an array of those shorthand properties, in preferred order.
|
||
// NB: There are no shorthands for custom properties.
|
||
|
||
// 4. Shorthand loop: For each shorthand in shorthands, follow these substeps: ...
|
||
// NB: There are no shorthands for custom properties.
|
||
|
||
// 5. Let value be the result of invoking serialize a CSS value of declaration.
|
||
auto value = declaration.value.value->to_string(Web::CSS::CSSStyleValue::SerializationMode::Normal);
|
||
|
||
// 6. Let serialized declaration be the result of invoking serialize a CSS declaration with property name property, value value,
|
||
// and the important flag set if declaration has its important flag set.
|
||
// NB: We have to inline this here as the actual implementation does not accept custom properties.
|
||
String serialized_declaration = [&] {
|
||
// https://www.w3.org/TR/cssom/#serialize-a-css-declaration
|
||
StringBuilder builder;
|
||
|
||
// 1. Let s be the empty string.
|
||
// 2. Append property to s.
|
||
builder.append(property);
|
||
|
||
// 3. Append ": " (U+003A U+0020) to s.
|
||
builder.append(": "sv);
|
||
|
||
// 4. Append value to s.
|
||
builder.append(value);
|
||
|
||
// 5. If the important flag is set, append " !important" (U+0020 U+0021 U+0069 U+006D U+0070 U+006F U+0072 U+0074 U+0061 U+006E U+0074) to s.
|
||
if (declaration.value.important == Important::Yes)
|
||
builder.append(" !important"sv);
|
||
|
||
// 6. Append ";" (U+003B) to s.
|
||
builder.append(';');
|
||
|
||
// 7. Return s.
|
||
return MUST(builder.to_string());
|
||
}();
|
||
|
||
// 7. Append serialized declaration to list.
|
||
list.append(move(serialized_declaration));
|
||
|
||
// 8. Append property to already serialized.
|
||
// NB: We don't need to do this, as we don't have shorthands for custom properties.
|
||
}
|
||
|
||
// 3. Declaration loop: For each CSS declaration declaration in declaration block’s declarations, follow these substeps:
|
||
for (auto& declaration : m_properties) {
|
||
// 1. Let property be declaration’s property name.
|
||
auto property = declaration.property_id;
|
||
|
||
// 2. If property is in already serialized, continue with the steps labeled declaration loop.
|
||
if (already_serialized.contains(property))
|
||
continue;
|
||
|
||
// FIXME: 3. If property maps to one or more shorthand properties, let shorthands be an array of those shorthand properties, in preferred order.
|
||
|
||
// FIXME: 4. Shorthand loop: For each shorthand in shorthands, follow these substeps: ...
|
||
|
||
// 5. Let value be the result of invoking serialize a CSS value of declaration.
|
||
auto value = declaration.value->to_string(Web::CSS::CSSStyleValue::SerializationMode::Normal);
|
||
|
||
// 6. Let serialized declaration be the result of invoking serialize a CSS declaration with property name property, value value,
|
||
// and the important flag set if declaration has its important flag set.
|
||
auto serialized_declaration = serialize_a_css_declaration(property, move(value), declaration.important);
|
||
|
||
// 7. Append serialized declaration to list.
|
||
list.append(move(serialized_declaration));
|
||
|
||
// 8. Append property to already serialized.
|
||
already_serialized.set(property);
|
||
}
|
||
|
||
// 4. Return list joined with " " (U+0020).
|
||
StringBuilder builder;
|
||
builder.join(' ', list);
|
||
return MUST(builder.to_string());
|
||
}
|
||
|
||
// https://drafts.csswg.org/cssom/#dom-cssstyledeclaration-csstext
|
||
WebIDL::ExceptionOr<void> CSSStyleProperties::set_css_text(StringView css_text)
|
||
{
|
||
// 1. If the readonly flag is set, then throw a NoModificationAllowedError exception.
|
||
if (is_readonly()) {
|
||
return WebIDL::NoModificationAllowedError::create(realm(), "Cannot modify properties: CSSStyleProperties is read-only."_string);
|
||
}
|
||
|
||
// 2. Empty the declarations.
|
||
// 3. Parse the given value and, if the return value is not the empty list, insert the items in the list into the declarations, in specified order.
|
||
set_declarations_from_text(css_text);
|
||
|
||
// 4. Update style attribute for the CSS declaration block.
|
||
update_style_attribute();
|
||
|
||
return {};
|
||
}
|
||
|
||
// https://drafts.csswg.org/cssom/#set-a-css-declaration
|
||
bool CSSStyleProperties::set_a_css_declaration(PropertyID property_id, NonnullRefPtr<CSSStyleValue const> value, Important important)
|
||
{
|
||
VERIFY(!is_computed());
|
||
|
||
// FIXME: Handle logical property groups.
|
||
|
||
for (auto& property : m_properties) {
|
||
if (property.property_id == property_id) {
|
||
if (property.important == important && *property.value == *value)
|
||
return false;
|
||
property.value = move(value);
|
||
property.important = important;
|
||
return true;
|
||
}
|
||
}
|
||
|
||
m_properties.append(StyleProperty {
|
||
.important = important,
|
||
.property_id = property_id,
|
||
.value = move(value),
|
||
});
|
||
return true;
|
||
}
|
||
|
||
void CSSStyleProperties::empty_the_declarations()
|
||
{
|
||
m_properties.clear();
|
||
m_custom_properties.clear();
|
||
}
|
||
|
||
void CSSStyleProperties::set_the_declarations(Vector<StyleProperty> properties, HashMap<FlyString, StyleProperty> custom_properties)
|
||
{
|
||
m_properties = move(properties);
|
||
m_custom_properties = move(custom_properties);
|
||
}
|
||
|
||
void CSSStyleProperties::set_declarations_from_text(StringView css_text)
|
||
{
|
||
empty_the_declarations();
|
||
auto parsing_params = owner_node().has_value()
|
||
? Parser::ParsingParams(owner_node()->element().document())
|
||
: Parser::ParsingParams();
|
||
auto style = parse_css_style_attribute(parsing_params, css_text);
|
||
set_the_declarations(style.properties, style.custom_properties);
|
||
}
|
||
|
||
}
|