2025-08-14 16:43:12 +01:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2025, Sam Atkins <sam@ladybird.org>
|
|
|
|
|
*
|
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "CSSUnparsedValue.h"
|
|
|
|
|
#include <LibWeb/Bindings/CSSUnparsedValuePrototype.h>
|
|
|
|
|
#include <LibWeb/Bindings/Intrinsics.h>
|
|
|
|
|
#include <LibWeb/CSS/CSSVariableReferenceValue.h>
|
2025-10-03 13:53:10 +01:00
|
|
|
|
#include <LibWeb/CSS/Parser/Parser.h>
|
|
|
|
|
#include <LibWeb/CSS/StyleValues/UnresolvedStyleValue.h>
|
2025-08-14 16:43:12 +01:00
|
|
|
|
#include <LibWeb/WebIDL/ExceptionOr.h>
|
|
|
|
|
|
|
|
|
|
namespace Web::CSS {
|
|
|
|
|
|
|
|
|
|
GC_DEFINE_ALLOCATOR(CSSUnparsedValue);
|
|
|
|
|
|
|
|
|
|
GC::Ref<CSSUnparsedValue> CSSUnparsedValue::create(JS::Realm& realm, Vector<GCRootCSSUnparsedSegment> value)
|
|
|
|
|
{
|
|
|
|
|
// NB: Convert our GC::Roots into GC::Refs.
|
|
|
|
|
Vector<CSSUnparsedSegment> converted_value;
|
|
|
|
|
for (auto const& variant : value) {
|
|
|
|
|
variant.visit(
|
|
|
|
|
[&](GC::Root<CSSVariableReferenceValue> const& it) { converted_value.append(GC::Ref { *it }); },
|
|
|
|
|
[&](String const& it) { converted_value.append(it); });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return realm.create<CSSUnparsedValue>(realm, move(converted_value));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// https://drafts.css-houdini.org/css-typed-om-1/#dom-cssunparsedvalue-cssunparsedvalue
|
|
|
|
|
WebIDL::ExceptionOr<GC::Ref<CSSUnparsedValue>> CSSUnparsedValue::construct_impl(JS::Realm& realm, Vector<GCRootCSSUnparsedSegment> value)
|
|
|
|
|
{
|
|
|
|
|
// AD-HOC: There is no spec for this, see https://github.com/w3c/css-houdini-drafts/issues/1146
|
|
|
|
|
|
|
|
|
|
return CSSUnparsedValue::create(realm, move(value));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CSSUnparsedValue::CSSUnparsedValue(JS::Realm& realm, Vector<CSSUnparsedSegment> value)
|
|
|
|
|
: CSSStyleValue(realm)
|
|
|
|
|
, m_tokens(move(value))
|
|
|
|
|
{
|
|
|
|
|
m_legacy_platform_object_flags = LegacyPlatformObjectFlags {
|
|
|
|
|
.supports_indexed_properties = true,
|
|
|
|
|
.has_indexed_property_setter = true,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CSSUnparsedValue::~CSSUnparsedValue() = default;
|
|
|
|
|
|
|
|
|
|
void CSSUnparsedValue::initialize(JS::Realm& realm)
|
|
|
|
|
{
|
|
|
|
|
WEB_SET_PROTOTYPE_FOR_INTERFACE(CSSUnparsedValue);
|
|
|
|
|
Base::initialize(realm);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CSSUnparsedValue::visit_edges(Visitor& visitor)
|
|
|
|
|
{
|
|
|
|
|
Base::visit_edges(visitor);
|
|
|
|
|
for (auto const& token : m_tokens) {
|
|
|
|
|
if (auto* variable = token.get_pointer<GC::Ref<CSSVariableReferenceValue>>()) {
|
|
|
|
|
visitor.visit(*variable);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// https://drafts.css-houdini.org/css-typed-om-1/#dom-cssunparsedvalue-length
|
|
|
|
|
WebIDL::UnsignedLong CSSUnparsedValue::length() const
|
|
|
|
|
{
|
|
|
|
|
// The length attribute returns the size of the [[tokens]] internal slot.
|
|
|
|
|
return m_tokens.size();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// https://drafts.css-houdini.org/css-typed-om-1/#ref-for-dfn-determine-the-value-of-an-indexed-property
|
|
|
|
|
Optional<JS::Value> CSSUnparsedValue::item_value(size_t index) const
|
|
|
|
|
{
|
|
|
|
|
// To determine the value of an indexed property of a CSSUnparsedValue this and an index n, let tokens be this’s
|
|
|
|
|
// [[tokens]] internal slot, and return tokens[n].
|
|
|
|
|
if (index >= m_tokens.size())
|
|
|
|
|
return {};
|
|
|
|
|
auto value = m_tokens[index];
|
|
|
|
|
return value.visit(
|
|
|
|
|
[&](GC::Ref<CSSVariableReferenceValue> const& variable) -> JS::Value { return variable; },
|
|
|
|
|
[&](String const& string) -> JS::Value { return JS::PrimitiveString::create(vm(), string); });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static WebIDL::ExceptionOr<CSSUnparsedSegment> unparsed_segment_from_js_value(JS::VM& vm, JS::Value& value)
|
|
|
|
|
{
|
|
|
|
|
if (value.is_object()) {
|
|
|
|
|
if (auto* variable_reference = as_if<CSSVariableReferenceValue>(value.as_object())) {
|
|
|
|
|
return GC::Ref { *variable_reference };
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return TRY(value.to_string(vm));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// https://drafts.css-houdini.org/css-typed-om-1/#ref-for-dfn-set-the-value-of-an-existing-indexed-property
|
|
|
|
|
WebIDL::ExceptionOr<void> CSSUnparsedValue::set_value_of_existing_indexed_property(u32 n, JS::Value value)
|
|
|
|
|
{
|
|
|
|
|
// To set the value of an existing indexed property of a CSSUnparsedValue this, an index n, and a value new value,
|
|
|
|
|
// let tokens be this’s [[tokens]] internal slot, and set tokens[n] to new value.
|
|
|
|
|
m_tokens[n] = TRY(unparsed_segment_from_js_value(vm(), value));
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// https://drafts.css-houdini.org/css-typed-om-1/#ref-for-dfn-set-the-value-of-a-new-indexed-property
|
|
|
|
|
WebIDL::ExceptionOr<void> CSSUnparsedValue::set_value_of_new_indexed_property(u32 n, JS::Value value)
|
|
|
|
|
{
|
|
|
|
|
// To set the value of a new indexed property of a CSSUnparsedValue this, an index n, and a value new value,
|
|
|
|
|
// let tokens be this’s [[tokens]] internal slot. If n is not equal to the size of tokens, throw a RangeError.
|
|
|
|
|
// Otherwise, append new value to tokens.
|
|
|
|
|
if (n != m_tokens.size())
|
|
|
|
|
return WebIDL::SimpleException { WebIDL::SimpleExceptionType::RangeError, "Index out of range"sv };
|
|
|
|
|
|
|
|
|
|
m_tokens.append(TRY(unparsed_segment_from_js_value(vm(), value)));
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-03 13:46:29 +01:00
|
|
|
|
bool CSSUnparsedValue::contains_unparsed_value(CSSUnparsedValue const& needle) const
|
|
|
|
|
{
|
|
|
|
|
for (auto const& unparsed_segment : m_tokens) {
|
|
|
|
|
if (auto const* variable_reference = unparsed_segment.get_pointer<GC::Ref<CSSVariableReferenceValue>>()) {
|
|
|
|
|
auto fallback = (*variable_reference)->fallback();
|
|
|
|
|
if (!fallback)
|
|
|
|
|
continue;
|
|
|
|
|
if (fallback.ptr() == &needle)
|
|
|
|
|
return true;
|
|
|
|
|
if (fallback->contains_unparsed_value(needle))
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-14 16:43:12 +01:00
|
|
|
|
// https://drafts.css-houdini.org/css-typed-om-1/#serialize-a-cssunparsedvalue
|
2025-09-16 15:06:38 +01:00
|
|
|
|
WebIDL::ExceptionOr<String> CSSUnparsedValue::to_string() const
|
2025-08-14 16:43:12 +01:00
|
|
|
|
{
|
2025-10-03 13:46:29 +01:00
|
|
|
|
// AD-HOC: It's possible for one of the m_tokens to contain this in its fallback slot, or a similar situation with
|
|
|
|
|
// more levels of nesting. To avoid crashing, do a scan for that first and return the empty string.
|
|
|
|
|
// Spec issue: https://github.com/w3c/css-houdini-drafts/issues/1158
|
|
|
|
|
if (contains_unparsed_value(*this))
|
|
|
|
|
return ""_string;
|
|
|
|
|
|
2025-08-14 16:43:12 +01:00
|
|
|
|
// To serialize a CSSUnparsedValue this:
|
|
|
|
|
// 1. Let s initially be the empty string.
|
|
|
|
|
StringBuilder s;
|
|
|
|
|
|
|
|
|
|
// 2. For each item in this’s [[tokens]] internal slot:
|
|
|
|
|
for (auto const& item : m_tokens) {
|
|
|
|
|
// FIXME: In order to match the expected test behaviour, this should insert comments, with the same rules as
|
|
|
|
|
// serialize_a_series_of_component_values(). See https://github.com/w3c/css-houdini-drafts/issues/1148
|
2025-09-16 15:06:38 +01:00
|
|
|
|
TRY(item.visit(
|
2025-08-14 16:43:12 +01:00
|
|
|
|
// 1. If item is a USVString, append it to s.
|
2025-09-16 15:06:38 +01:00
|
|
|
|
[&](String const& string) -> WebIDL::ExceptionOr<void> {
|
2025-08-14 16:43:12 +01:00
|
|
|
|
s.append(string);
|
2025-09-16 15:06:38 +01:00
|
|
|
|
return {};
|
2025-08-14 16:43:12 +01:00
|
|
|
|
},
|
|
|
|
|
// 2. Otherwise, item is a CSSVariableReferenceValue. Serialize it, then append the result to s.
|
2025-09-16 15:06:38 +01:00
|
|
|
|
[&](GC::Ref<CSSVariableReferenceValue> const& variable) -> WebIDL::ExceptionOr<void> {
|
|
|
|
|
s.append(TRY(variable->to_string()));
|
|
|
|
|
return {};
|
|
|
|
|
}));
|
2025-08-14 16:43:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 3. Return s.
|
|
|
|
|
return s.to_string_without_validation();
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-03 13:53:10 +01:00
|
|
|
|
// https://drafts.css-houdini.org/css-typed-om-1/#create-an-internal-representation
|
|
|
|
|
WebIDL::ExceptionOr<NonnullRefPtr<StyleValue const>> CSSUnparsedValue::create_an_internal_representation(PropertyNameAndID const&) const
|
|
|
|
|
{
|
|
|
|
|
// If value is a CSSStyleValue subclass,
|
|
|
|
|
// If value does not match the grammar of a list-valued property iteration of property, throw a TypeError.
|
|
|
|
|
//
|
|
|
|
|
// If any component of property’s CSS grammar has a limited numeric range, and the corresponding part of value
|
|
|
|
|
// is a CSSUnitValue that is outside of that range, replace that value with the result of wrapping it in a
|
|
|
|
|
// fresh CSSMathSum whose values internal slot contains only that part of value.
|
|
|
|
|
//
|
|
|
|
|
// Return the value.
|
|
|
|
|
|
|
|
|
|
// https://drafts.css-houdini.org/css-typed-om-1/#cssstylevalue-match-a-grammar
|
|
|
|
|
// A CSSUnparsedValue matches any grammar.
|
|
|
|
|
|
|
|
|
|
// NB: CSSUnparsedValue stores a list of strings, each of which may contain any number of tokens. So the simplest
|
|
|
|
|
// way to convert it to ComponentValues is to serialize and then parse it.
|
|
|
|
|
auto string = TRY(to_string());
|
|
|
|
|
auto parser = Parser::Parser::create(Parser::ParsingParams {}, string);
|
|
|
|
|
auto component_values = parser.parse_as_list_of_component_values();
|
|
|
|
|
return UnresolvedStyleValue::create(move(component_values));
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-14 16:43:12 +01:00
|
|
|
|
}
|