LibWeb/CSS: Store @property initial value as a style value

For now, it's the "raw" UnresolvedStyleValue. Later, we'll need to try
to parse it into a "real" style value using the property's syntax.
This commit is contained in:
Sam Atkins 2025-04-07 11:54:51 +01:00
parent 9292b769ef
commit a80408fea6
Notes: github-actions[bot] 2025-04-08 09:00:42 +00:00
4 changed files with 24 additions and 19 deletions

View file

@ -13,12 +13,12 @@ namespace Web::CSS {
GC_DEFINE_ALLOCATOR(CSSPropertyRule); GC_DEFINE_ALLOCATOR(CSSPropertyRule);
GC::Ref<CSSPropertyRule> CSSPropertyRule::create(JS::Realm& realm, FlyString name, FlyString syntax, bool inherits, Optional<String> initial_value) GC::Ref<CSSPropertyRule> CSSPropertyRule::create(JS::Realm& realm, FlyString name, FlyString syntax, bool inherits, RefPtr<CSSStyleValue> initial_value)
{ {
return realm.create<CSSPropertyRule>(realm, move(name), move(syntax), inherits, move(initial_value)); return realm.create<CSSPropertyRule>(realm, move(name), move(syntax), inherits, move(initial_value));
} }
CSSPropertyRule::CSSPropertyRule(JS::Realm& realm, FlyString name, FlyString syntax, bool inherits, Optional<String> initial_value) CSSPropertyRule::CSSPropertyRule(JS::Realm& realm, FlyString name, FlyString syntax, bool inherits, RefPtr<CSSStyleValue> initial_value)
: CSSRule(realm, Type::Property) : CSSRule(realm, Type::Property)
, m_name(move(name)) , m_name(move(name))
, m_syntax(move(syntax)) , m_syntax(move(syntax))
@ -27,6 +27,13 @@ CSSPropertyRule::CSSPropertyRule(JS::Realm& realm, FlyString name, FlyString syn
{ {
} }
Optional<String> CSSPropertyRule::initial_value() const
{
if (m_initial_value)
return m_initial_value->to_string(CSSStyleValue::SerializationMode::Normal);
return {};
}
void CSSPropertyRule::initialize(JS::Realm& realm) void CSSPropertyRule::initialize(JS::Realm& realm)
{ {
Base::initialize(realm); Base::initialize(realm);
@ -63,8 +70,8 @@ String CSSPropertyRule::serialized() const
// 8. If the rules initial-value is present, follow these substeps: // 8. If the rules initial-value is present, follow these substeps:
if (initial_value().has_value()) { if (initial_value().has_value()) {
// 1. The string "initial-value:". // 1. The string "initial-value:".
// 2. The result of performing serialize a CSS value in the rules initial-value followed by a single SEMICOLON (U+003B), followed by a SPACE (U+0020). // 2. The result of performing serialize a CSS value in the rules initial-value followed by a single SEMICOLON
// FIXME: Follow the spec for serializing the value whenever we actually have a CSS value here. // (U+003B), followed by a SPACE (U+0020).
builder.appendff("initial-value: {}; ", initial_value()); builder.appendff("initial-value: {}; ", initial_value());
} }
// 9. A single RIGHT CURLY BRACKET (U+007D). // 9. A single RIGHT CURLY BRACKET (U+007D).

View file

@ -21,17 +21,17 @@ class CSSPropertyRule final : public CSSRule {
GC_DECLARE_ALLOCATOR(CSSPropertyRule); GC_DECLARE_ALLOCATOR(CSSPropertyRule);
public: public:
static GC::Ref<CSSPropertyRule> create(JS::Realm&, FlyString name, FlyString syntax, bool inherits, Optional<String> initial_value); static GC::Ref<CSSPropertyRule> create(JS::Realm&, FlyString name, FlyString syntax, bool inherits, RefPtr<CSSStyleValue> initial_value);
virtual ~CSSPropertyRule() = default; virtual ~CSSPropertyRule() = default;
FlyString const& name() const { return m_name; } FlyString const& name() const { return m_name; }
FlyString const& syntax() const { return m_syntax; } FlyString const& syntax() const { return m_syntax; }
bool inherits() const { return m_inherits; } bool inherits() const { return m_inherits; }
Optional<String> initial_value() const { return m_initial_value; } Optional<String> initial_value() const;
private: private:
CSSPropertyRule(JS::Realm&, FlyString name, FlyString syntax, bool inherits, Optional<String> initial_value); CSSPropertyRule(JS::Realm&, FlyString name, FlyString syntax, bool inherits, RefPtr<CSSStyleValue> initial_value);
virtual void initialize(JS::Realm&) override; virtual void initialize(JS::Realm&) override;
virtual String serialized() const override; virtual String serialized() const override;
@ -39,8 +39,7 @@ private:
FlyString m_name; FlyString m_name;
FlyString m_syntax; FlyString m_syntax;
bool m_inherits; bool m_inherits;
// FIXME: This should hold an actual CSS value, matching the syntax RefPtr<CSSStyleValue> m_initial_value;
Optional<String> m_initial_value;
}; };
template<> template<>

View file

@ -65,13 +65,11 @@ Parser::ParseErrorOr<NonnullRefPtr<CSSStyleValue>> Parser::parse_descriptor_valu
return parse_comma_separated_value_list(tokens, [this](auto& tokens) -> RefPtr<CSSStyleValue> { return parse_comma_separated_value_list(tokens, [this](auto& tokens) -> RefPtr<CSSStyleValue> {
return parse_font_source_value(tokens); return parse_font_source_value(tokens);
}); });
case DescriptorMetadata::ValueType::OptionalDeclarationValue: { case DescriptorMetadata::ValueType::OptionalDeclarationValue:
// FIXME: This is for an @property's initial value. Figure out what this should actually do once we need it. // `component_values` already has what we want. Just skip through its tokens so code below knows we consumed them.
StringBuilder initial_value_sb;
while (tokens.has_next_token()) while (tokens.has_next_token())
initial_value_sb.append(tokens.consume_a_token().to_string()); tokens.discard_a_token();
return StringStyleValue::create(initial_value_sb.to_fly_string_without_validation()); return UnresolvedStyleValue::create(move(component_values), false, {});
}
case DescriptorMetadata::ValueType::PositivePercentage: case DescriptorMetadata::ValueType::PositivePercentage:
if (auto percentage_value = parse_percentage_value(tokens)) { if (auto percentage_value = parse_percentage_value(tokens)) {
if (percentage_value->is_percentage()) { if (percentage_value->is_percentage()) {

View file

@ -535,7 +535,7 @@ GC::Ptr<CSSPropertyRule> Parser::convert_to_property_rule(AtRule const& rule)
Optional<FlyString> syntax_maybe; Optional<FlyString> syntax_maybe;
Optional<bool> inherits_maybe; Optional<bool> inherits_maybe;
Optional<String> initial_value_maybe; RefPtr<CSSStyleValue> initial_value_maybe;
rule.for_each_as_declaration_list([&](auto& declaration) { rule.for_each_as_declaration_list([&](auto& declaration) {
if (auto descriptor = convert_to_descriptor(AtRuleID::Property, declaration); descriptor.has_value()) { if (auto descriptor = convert_to_descriptor(AtRuleID::Property, declaration); descriptor.has_value()) {
@ -558,15 +558,16 @@ GC::Ptr<CSSPropertyRule> Parser::convert_to_property_rule(AtRule const& rule)
return; return;
} }
if (descriptor->descriptor_id == DescriptorID::InitialValue) { if (descriptor->descriptor_id == DescriptorID::InitialValue) {
if (descriptor->value->is_string()) initial_value_maybe = *descriptor->value;
initial_value_maybe = descriptor->value->as_string().string_value().to_string();
return; return;
} }
} }
}); });
// TODO: Parse the initial value using the syntax, if it's provided.
if (syntax_maybe.has_value() && inherits_maybe.has_value()) { if (syntax_maybe.has_value() && inherits_maybe.has_value()) {
return CSSPropertyRule::create(realm(), name, syntax_maybe.value(), inherits_maybe.value(), std::move(initial_value_maybe)); return CSSPropertyRule::create(realm(), name, syntax_maybe.value(), inherits_maybe.value(), move(initial_value_maybe));
} }
return {}; return {};
} }