/* * Copyright (c) 2018-2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include namespace Web::DOM { class ParentNode : public Node { WEB_PLATFORM_OBJECT(ParentNode, Node); JS_DECLARE_ALLOCATOR(ParentNode); public: template void for_each_child(F) const; template void for_each_child(F); JS::GCPtr first_element_child(); JS::GCPtr last_element_child(); u32 child_element_count() const; WebIDL::ExceptionOr> query_selector(StringView); WebIDL::ExceptionOr> query_selector_all(StringView); JS::NonnullGCPtr children(); JS::NonnullGCPtr get_elements_by_tag_name(FlyString const&); JS::NonnullGCPtr get_elements_by_tag_name_ns(Optional, FlyString const&); WebIDL::ExceptionOr prepend(Vector, String>> const& nodes); WebIDL::ExceptionOr append(Vector, String>> const& nodes); WebIDL::ExceptionOr replace_children(Vector, String>> const& nodes); protected: ParentNode(JS::Realm& realm, Document& document, NodeType type) : Node(realm, document, type) { } ParentNode(Document& document, NodeType type) : Node(document, type) { } virtual void visit_edges(Cell::Visitor&) override; private: JS::GCPtr m_children; }; template<> inline bool Node::fast_is() const { return is_parent_node(); } template inline U* Node::shadow_including_first_ancestor_of_type() { for (auto* ancestor = parent_or_shadow_host(); ancestor; ancestor = ancestor->parent_or_shadow_host()) { if (is(*ancestor)) return &verify_cast(*ancestor); } return nullptr; } template inline void ParentNode::for_each_child(Callback callback) const { for (auto* node = first_child(); node; node = node->next_sibling()) { if (callback(*node) == IterationDecision::Break) return; } } template inline void ParentNode::for_each_child(Callback callback) { for (auto* node = first_child(); node; node = node->next_sibling()) { if (callback(*node) == IterationDecision::Break) return; } } }