/* * Copyright (c) 2023, Luke Wilde * Copyright (c) 2025, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include namespace Web::HTML { struct ElementDefinitionOptions { Optional extends; }; // https://html.spec.whatwg.org/multipage/custom-elements.html#customelementregistry class WEB_API CustomElementRegistry : public Bindings::PlatformObject { WEB_PLATFORM_OBJECT(CustomElementRegistry, Bindings::PlatformObject); GC_DECLARE_ALLOCATOR(CustomElementRegistry); public: [[nodiscard]] static GC::Ref construct_impl(JS::Realm&); virtual ~CustomElementRegistry() override; JS::ThrowCompletionOr define(String const& name, WebIDL::CallbackType* constructor, ElementDefinitionOptions options); Variant, Empty> get(String const& name) const; Optional get_name(GC::Root const& constructor) const; WebIDL::ExceptionOr> when_defined(String const& name); void upgrade(GC::Ref root) const; WebIDL::ExceptionOr initialize_for_bindings(GC::Ref root); bool is_scoped() const { return m_is_scoped; } void append_scoped_document(GC::Ref); GC::Ptr get_definition_with_name_and_local_name(String const& name, String const& local_name) const; GC::Ptr get_definition_from_new_target(JS::FunctionObject const& new_target) const; private: CustomElementRegistry(JS::Realm&); virtual void initialize(JS::Realm&) override; virtual void visit_edges(Visitor&) override; // https://html.spec.whatwg.org/multipage/custom-elements.html#is-scoped // Every CustomElementRegistry has an is scoped, a boolean, initially false. bool m_is_scoped { false }; // https://html.spec.whatwg.org/multipage/custom-elements.html#scoped-document-set // Every CustomElementRegistry has a scoped document set, a set of Document objects, initially « ». // NB: This is a "weak set", see https://github.com/whatwg/html/issues/12092#issuecomment-3769252677 GC::WeakHashSet m_scoped_documents; // https://html.spec.whatwg.org/multipage/custom-elements.html#custom-element-definition-set // Every CustomElementRegistry has a custom element definition set, a set of custom element definitions, // initially « ». Lookup of items in this set uses their name, local name, or constructor. Vector> m_custom_element_definitions; // https://html.spec.whatwg.org/multipage/custom-elements.html#element-definition-is-running // Every CustomElementRegistry also has an element definition is running boolean which is used to prevent reentrant // invocations of element definition. It is initially false. bool m_element_definition_is_running { false }; // https://html.spec.whatwg.org/multipage/custom-elements.html#when-defined-promise-map // Every CustomElementRegistry also has a when-defined promise map, mapping valid custom element names to promises. // It is used to implement the whenDefined() method. OrderedHashMap> m_when_defined_promise_map; }; GC::Ptr look_up_a_custom_element_registry(DOM::Node const&); GC::Ptr look_up_a_custom_element_definition(GC::Ptr registry, Optional const& namespace_, FlyString const& local_name, Optional const& is); bool is_a_global_custom_element_registry(GC::Ptr); }