mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2024-11-07 20:31:04 +03:00
LibWeb: Port the CSS namespace to IDL
This commit is contained in:
parent
020c2b59c4
commit
2d45e1fca5
Notes:
sideshowbarker
2024-07-16 23:51:07 +09:00
Author: https://github.com/trflynn89 Commit: https://github.com/SerenityOS/serenity/commit/2d45e1fca5 Pull-request: https://github.com/SerenityOS/serenity/pull/17870 Reviewed-by: https://github.com/linusg
@ -1,86 +0,0 @@
|
||||
/*
|
||||
* Copyright (c) 2021-2023, Sam Atkins <atkinssj@serenityos.org>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
||||
#include <LibJS/Runtime/ErrorTypes.h>
|
||||
#include <LibJS/Runtime/GlobalObject.h>
|
||||
#include <LibJS/Runtime/VM.h>
|
||||
#include <LibJS/Runtime/Value.h>
|
||||
#include <LibWeb/Bindings/CSSNamespace.h>
|
||||
#include <LibWeb/CSS/Parser/Parser.h>
|
||||
|
||||
namespace Web::Bindings {
|
||||
|
||||
CSSNamespace::CSSNamespace(JS::Realm& realm)
|
||||
: JS::Object(ConstructWithPrototypeTag::Tag, *realm.intrinsics().object_prototype())
|
||||
{
|
||||
}
|
||||
|
||||
JS::ThrowCompletionOr<void> CSSNamespace::initialize(JS::Realm& realm)
|
||||
{
|
||||
MUST_OR_THROW_OOM(Object::initialize(realm));
|
||||
u8 attr = JS::Attribute::Enumerable;
|
||||
define_native_function(realm, "escape", escape, 1, attr);
|
||||
define_native_function(realm, "supports", supports, 2, attr);
|
||||
|
||||
return {};
|
||||
}
|
||||
|
||||
// https://www.w3.org/TR/cssom-1/#dom-css-escape
|
||||
JS_DEFINE_NATIVE_FUNCTION(CSSNamespace::escape)
|
||||
{
|
||||
if (!vm.argument_count())
|
||||
return vm.throw_completion<JS::TypeError>(JS::ErrorType::BadArgCountAtLeastOne, "CSS.escape");
|
||||
|
||||
auto identifier = TRY(vm.argument(0).to_string(vm));
|
||||
return JS::PrimitiveString::create(vm, TRY_OR_THROW_OOM(vm, Web::CSS::serialize_an_identifier(identifier)));
|
||||
}
|
||||
|
||||
// https://www.w3.org/TR/css-conditional-3/#dom-css-supports
|
||||
JS_DEFINE_NATIVE_FUNCTION(CSSNamespace::supports)
|
||||
{
|
||||
auto& realm = *vm.current_realm();
|
||||
|
||||
if (!vm.argument_count())
|
||||
return vm.throw_completion<JS::TypeError>(JS::ErrorType::BadArgCountAtLeastOne, "CSS.supports");
|
||||
|
||||
if (vm.argument_count() >= 2) {
|
||||
// When the supports(property, value) method is invoked with two arguments property and value:
|
||||
auto property_name = TRY(vm.argument(0).to_deprecated_string(vm));
|
||||
|
||||
// If property is an ASCII case-insensitive match for any defined CSS property that the UA supports,
|
||||
// and value successfully parses according to that property’s grammar, return true.
|
||||
auto property = CSS::property_id_from_string(property_name);
|
||||
if (property != CSS::PropertyID::Invalid) {
|
||||
auto value_string = TRY(vm.argument(1).to_deprecated_string(vm));
|
||||
if (parse_css_value(CSS::Parser::ParsingContext { realm }, value_string, property))
|
||||
return JS::Value(true);
|
||||
}
|
||||
// Otherwise, if property is a custom property name string, return true.
|
||||
// FIXME: This check is not enough to make sure this is a valid custom property name, but it's close enough.
|
||||
else if (property_name.starts_with("--"sv) && property_name.length() >= 3) {
|
||||
return JS::Value(true);
|
||||
}
|
||||
|
||||
// Otherwise, return false.
|
||||
return JS::Value(false);
|
||||
} else {
|
||||
// When the supports(conditionText) method is invoked with a single conditionText argument:
|
||||
auto supports_text = TRY(vm.argument(0).to_deprecated_string(vm));
|
||||
|
||||
// If conditionText, parsed and evaluated as a <supports-condition>, would return true, return true.
|
||||
if (auto supports = parse_css_supports(CSS::Parser::ParsingContext { realm }, supports_text); supports && supports->matches())
|
||||
return JS::Value(true);
|
||||
|
||||
// Otherwise, If conditionText, wrapped in parentheses and then parsed and evaluated as a <supports-condition>, would return true, return true.
|
||||
if (auto supports = parse_css_supports(CSS::Parser::ParsingContext { realm }, DeprecatedString::formatted("({})", supports_text)); supports && supports->matches())
|
||||
return JS::Value(true);
|
||||
|
||||
// Otherwise, return false.
|
||||
return JS::Value(false);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -1,28 +0,0 @@
|
||||
/*
|
||||
* Copyright (c) 2021, Sam Atkins <atkinssj@serenityos.org>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <LibJS/Runtime/Object.h>
|
||||
#include <LibWeb/CSS/Serialize.h>
|
||||
|
||||
namespace Web::Bindings {
|
||||
|
||||
// The `CSS` namespace object in IDL. https://www.w3.org/TR/cssom-1/#namespacedef-css
|
||||
class CSSNamespace final : public JS::Object {
|
||||
JS_OBJECT(CSSNamespace, JS::Object)
|
||||
|
||||
public:
|
||||
explicit CSSNamespace(JS::Realm&);
|
||||
virtual JS::ThrowCompletionOr<void> initialize(JS::Realm&) override;
|
||||
virtual ~CSSNamespace() override = default;
|
||||
|
||||
private:
|
||||
JS_DECLARE_NATIVE_FUNCTION(escape);
|
||||
JS_DECLARE_NATIVE_FUNCTION(supports);
|
||||
};
|
||||
|
||||
}
|
@ -5,7 +5,6 @@ set(SOURCES
|
||||
ARIA/ARIAMixin.idl
|
||||
ARIA/Roles.cpp
|
||||
Bindings/AudioConstructor.cpp
|
||||
Bindings/CSSNamespace.cpp
|
||||
Bindings/HostDefined.cpp
|
||||
Bindings/ImageConstructor.cpp
|
||||
Bindings/Intrinsics.cpp
|
||||
@ -18,6 +17,7 @@ set(SOURCES
|
||||
Crypto/SubtleCrypto.cpp
|
||||
CSS/Angle.cpp
|
||||
CSS/Clip.cpp
|
||||
CSS/CSS.cpp
|
||||
CSS/CSSConditionRule.cpp
|
||||
CSS/CSSGroupingRule.cpp
|
||||
CSS/CSSImportRule.cpp
|
||||
|
64
Userland/Libraries/LibWeb/CSS/CSS.cpp
Normal file
64
Userland/Libraries/LibWeb/CSS/CSS.cpp
Normal file
@ -0,0 +1,64 @@
|
||||
/*
|
||||
* Copyright (c) 2021-2023, Sam Atkins <atkinssj@serenityos.org>
|
||||
* Copyright (c) 2023, Tim Flynn <trflynn89@serenityos.org>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
||||
#include <LibJS/Runtime/VM.h>
|
||||
#include <LibWeb/CSS/CSS.h>
|
||||
#include <LibWeb/CSS/Parser/Parser.h>
|
||||
#include <LibWeb/CSS/PropertyID.h>
|
||||
#include <LibWeb/CSS/Serialize.h>
|
||||
|
||||
namespace Web::CSS {
|
||||
|
||||
// https://www.w3.org/TR/cssom-1/#dom-css-escape
|
||||
WebIDL::ExceptionOr<String> escape(JS::VM& vm, StringView identifier)
|
||||
{
|
||||
// The escape(ident) operation must return the result of invoking serialize an identifier of ident.
|
||||
return TRY_OR_THROW_OOM(vm, serialize_an_identifier(identifier));
|
||||
}
|
||||
|
||||
// https://www.w3.org/TR/css-conditional-3/#dom-css-supports
|
||||
bool supports(JS::VM& vm, StringView property, StringView value)
|
||||
{
|
||||
auto& realm = *vm.current_realm();
|
||||
|
||||
// 1. If property is an ASCII case-insensitive match for any defined CSS property that the UA supports,
|
||||
// and value successfully parses according to that property’s grammar, return true.
|
||||
if (auto property_id = property_id_from_string(property); property_id != PropertyID::Invalid) {
|
||||
if (parse_css_value(Parser::ParsingContext { realm }, value, property_id))
|
||||
return true;
|
||||
}
|
||||
|
||||
// 2. Otherwise, if property is a custom property name string, return true.
|
||||
// FIXME: This check is not enough to make sure this is a valid custom property name, but it's close enough.
|
||||
else if (property.starts_with("--"sv) && property.length() >= 3) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// 3. Otherwise, return false.
|
||||
return false;
|
||||
}
|
||||
|
||||
// https://www.w3.org/TR/css-conditional-3/#dom-css-supports
|
||||
WebIDL::ExceptionOr<bool> supports(JS::VM& vm, StringView condition_text)
|
||||
{
|
||||
auto& realm = *vm.current_realm();
|
||||
|
||||
// 1. If conditionText, parsed and evaluated as a <supports-condition>, would return true, return true.
|
||||
if (auto supports = parse_css_supports(Parser::ParsingContext { realm }, condition_text); supports && supports->matches())
|
||||
return true;
|
||||
|
||||
// 2. Otherwise, If conditionText, wrapped in parentheses and then parsed and evaluated as a <supports-condition>, would return true, return true.
|
||||
auto wrapped_condition_text = TRY_OR_THROW_OOM(vm, String::formatted("({})", condition_text));
|
||||
|
||||
if (auto supports = parse_css_supports(Parser::ParsingContext { realm }, wrapped_condition_text); supports && supports->matches())
|
||||
return true;
|
||||
|
||||
// 3. Otherwise, return false.
|
||||
return false;
|
||||
}
|
||||
|
||||
}
|
23
Userland/Libraries/LibWeb/CSS/CSS.h
Normal file
23
Userland/Libraries/LibWeb/CSS/CSS.h
Normal file
@ -0,0 +1,23 @@
|
||||
/*
|
||||
* Copyright (c) 2021, Sam Atkins <atkinssj@serenityos.org>
|
||||
* Copyright (c) 2023, Tim Flynn <trflynn89@serenityos.org>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <AK/String.h>
|
||||
#include <AK/StringView.h>
|
||||
#include <LibJS/Forward.h>
|
||||
#include <LibWeb/WebIDL/ExceptionOr.h>
|
||||
|
||||
// https://www.w3.org/TR/cssom-1/#namespacedef-css
|
||||
namespace Web::CSS {
|
||||
|
||||
WebIDL::ExceptionOr<String> escape(JS::VM&, StringView identifier);
|
||||
|
||||
bool supports(JS::VM&, StringView property, StringView value);
|
||||
WebIDL::ExceptionOr<bool> supports(JS::VM&, StringView condition_text);
|
||||
|
||||
}
|
8
Userland/Libraries/LibWeb/CSS/CSS.idl
Normal file
8
Userland/Libraries/LibWeb/CSS/CSS.idl
Normal file
@ -0,0 +1,8 @@
|
||||
// https://www.w3.org/TR/cssom-1/#namespacedef-css
|
||||
[Exposed=Window]
|
||||
namespace CSS {
|
||||
CSSOMString escape(CSSOMString ident);
|
||||
|
||||
boolean supports(CSSOMString property, CSSOMString value);
|
||||
boolean supports(CSSOMString conditionText);
|
||||
};
|
@ -19,7 +19,6 @@
|
||||
#include <LibJS/Runtime/NativeFunction.h>
|
||||
#include <LibJS/Runtime/Shape.h>
|
||||
#include <LibTextCodec/Decoder.h>
|
||||
#include <LibWeb/Bindings/CSSNamespace.h>
|
||||
#include <LibWeb/Bindings/ExceptionOrUtils.h>
|
||||
#include <LibWeb/Bindings/WindowExposedInterfaces.h>
|
||||
#include <LibWeb/Bindings/WindowPrototype.h>
|
||||
@ -753,7 +752,6 @@ WebIDL::ExceptionOr<void> Window::initialize_web_interfaces(Badge<WindowEnvironm
|
||||
// 1. Let id be namespace’s identifier.
|
||||
// 3. Let namespaceObject be the result of creating a namespace object for namespace in realm.
|
||||
// 3. Perform CreateMethodProperty(target, id, namespaceObject).
|
||||
create_method_property("CSS", MUST_OR_THROW_OOM(heap().allocate<Bindings::CSSNamespace>(realm, realm)));
|
||||
create_method_property("WebAssembly", MUST_OR_THROW_OOM(heap().allocate<Bindings::WebAssemblyObject>(realm, realm)));
|
||||
|
||||
return {};
|
||||
|
@ -8,6 +8,7 @@ libweb_js_bindings(CSS/CSSFontFaceRule)
|
||||
libweb_js_bindings(CSS/CSSGroupingRule)
|
||||
libweb_js_bindings(CSS/CSSImportRule)
|
||||
libweb_js_bindings(CSS/CSSMediaRule)
|
||||
libweb_js_bindings(CSS/CSS NAMESPACE)
|
||||
libweb_js_bindings(CSS/CSSRule)
|
||||
libweb_js_bindings(CSS/CSSRuleList)
|
||||
libweb_js_bindings(CSS/CSSStyleDeclaration)
|
||||
|
Loading…
Reference in New Issue
Block a user