/* * Copyright (c) 2020, the SerenityOS developers. * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include namespace Web::DOM { JS_DEFINE_ALLOCATOR(ShadowRoot); ShadowRoot::ShadowRoot(Document& document, Element& host, Bindings::ShadowRootMode mode) : DocumentFragment(document) , m_mode(mode) { document.register_shadow_root({}, *this); set_host(&host); } void ShadowRoot::finalize() { Base::finalize(); document().unregister_shadow_root({}, *this); } void ShadowRoot::initialize(JS::Realm& realm) { Base::initialize(realm); WEB_SET_PROTOTYPE_FOR_INTERFACE(ShadowRoot); } // https://dom.spec.whatwg.org/#dom-shadowroot-onslotchange void ShadowRoot::set_onslotchange(WebIDL::CallbackType* event_handler) { set_event_handler_attribute(HTML::EventNames::slotchange, event_handler); } // https://dom.spec.whatwg.org/#dom-shadowroot-onslotchange WebIDL::CallbackType* ShadowRoot::onslotchange() { return event_handler_attribute(HTML::EventNames::slotchange); } // https://dom.spec.whatwg.org/#ref-for-get-the-parent%E2%91%A6 EventTarget* ShadowRoot::get_parent(Event const& event) { if (!event.composed()) { auto& events_first_invocation_target = verify_cast(*event.path().first().invocation_target); if (&events_first_invocation_target.root() == this) return nullptr; } return host(); } // https://w3c.github.io/DOM-Parsing/#dom-innerhtml-innerhtml WebIDL::ExceptionOr ShadowRoot::inner_html() const { return serialize_fragment(DOMParsing::RequireWellFormed::Yes); } // https://w3c.github.io/DOM-Parsing/#dom-innerhtml-innerhtml WebIDL::ExceptionOr ShadowRoot::set_inner_html(StringView markup) { TRY(DOMParsing::inner_html_setter(*this, markup)); set_needs_style_update(true); return {}; } CSS::StyleSheetList& ShadowRoot::style_sheets() { if (!m_style_sheets) m_style_sheets = CSS::StyleSheetList::create(document()); return *m_style_sheets; } CSS::StyleSheetList const& ShadowRoot::style_sheets() const { return const_cast(this)->style_sheets(); } void ShadowRoot::visit_edges(Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(m_style_sheets); visitor.visit(m_adopted_style_sheets); } JS::NonnullGCPtr ShadowRoot::adopted_style_sheets() const { if (!m_adopted_style_sheets) m_adopted_style_sheets = create_adopted_style_sheets_list(const_cast(document())); return *m_adopted_style_sheets; } WebIDL::ExceptionOr ShadowRoot::set_adopted_style_sheets(JS::Value new_value) { if (!m_adopted_style_sheets) m_adopted_style_sheets = create_adopted_style_sheets_list(const_cast(document())); m_adopted_style_sheets->clear(); auto iterator_record = TRY(get_iterator(vm(), new_value, JS::IteratorHint::Sync)); while (true) { auto next = TRY(iterator_step_value(vm(), iterator_record)); if (!next.has_value()) break; TRY(m_adopted_style_sheets->append(*next)); } return {}; } void ShadowRoot::for_each_css_style_sheet(Function&& callback) const { for (auto& style_sheet : style_sheets().sheets()) callback(*style_sheet); if (m_adopted_style_sheets) { m_adopted_style_sheets->for_each([&](auto& style_sheet) { callback(style_sheet); }); } } }