123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596 |
- /*
- * Copyright (c) 2021, Luke Wilde <lukew@serenityos.org>
- * Copyright (c) 2022, Andreas Kling <andreas@ladybird.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include <LibGC/Heap.h>
- #include <LibJS/Runtime/Error.h>
- #include <LibWeb/DOM/LiveNodeList.h>
- #include <LibWeb/DOM/Node.h>
- namespace Web::DOM {
- GC_DEFINE_ALLOCATOR(LiveNodeList);
- GC::Ref<NodeList> LiveNodeList::create(JS::Realm& realm, Node const& root, Scope scope, Function<bool(Node const&)> filter)
- {
- return realm.create<LiveNodeList>(realm, root, scope, move(filter));
- }
- LiveNodeList::LiveNodeList(JS::Realm& realm, Node const& root, Scope scope, Function<bool(Node const&)> filter)
- : NodeList(realm)
- , m_root(root)
- , m_filter(move(filter))
- , m_scope(scope)
- {
- }
- LiveNodeList::~LiveNodeList() = default;
- void LiveNodeList::visit_edges(Cell::Visitor& visitor)
- {
- Base::visit_edges(visitor);
- visitor.visit(m_root);
- }
- GC::RootVector<Node*> LiveNodeList::collection() const
- {
- GC::RootVector<Node*> nodes(heap());
- if (m_scope == Scope::Descendants) {
- m_root->for_each_in_subtree([&](auto& node) {
- if (m_filter(node))
- nodes.append(const_cast<Node*>(&node));
- return TraversalDecision::Continue;
- });
- } else {
- m_root->for_each_child([&](auto& node) {
- if (m_filter(node))
- nodes.append(const_cast<Node*>(&node));
- return IterationDecision::Continue;
- });
- }
- return nodes;
- }
- Node* LiveNodeList::first_matching(Function<bool(Node const&)> const& filter) const
- {
- Node* matched_node = nullptr;
- if (m_scope == Scope::Descendants) {
- m_root->for_each_in_subtree([&](auto& node) {
- if (m_filter(node) && filter(node)) {
- matched_node = const_cast<Node*>(&node);
- return TraversalDecision::Break;
- }
- return TraversalDecision::Continue;
- });
- } else {
- m_root->for_each_child([&](auto& node) {
- if (m_filter(node) && filter(node)) {
- matched_node = const_cast<Node*>(&node);
- return IterationDecision::Break;
- }
- return IterationDecision::Continue;
- });
- }
- return matched_node;
- }
- // https://dom.spec.whatwg.org/#dom-nodelist-length
- u32 LiveNodeList::length() const
- {
- return collection().size();
- }
- // https://dom.spec.whatwg.org/#dom-nodelist-item
- Node const* LiveNodeList::item(u32 index) const
- {
- // The item(index) method must return the indexth node in the collection. If there is no indexth node in the collection, then the method must return null.
- auto nodes = collection();
- if (index >= nodes.size())
- return nullptr;
- return nodes[index];
- }
- }
|