diff options
author | George L. Albany <Megacake1234@gmail.com> | 2024-11-02 07:09:23 +0000 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-11-02 07:09:23 +0000 |
commit | 9043ceac65972c4195dd71f15a8a1cc7aa72985f (patch) | |
tree | e581b9c42dd06f2bede00adbd619a051fc940b38 /scene/main | |
parent | 3302f803aa4ea09b16a8824ca3bea33f3e2b48ba (diff) | |
parent | f98dc56dd77716f3ed6ed759f245c8520e22b1f0 (diff) | |
download | redot-engine-9043ceac65972c4195dd71f15a8a1cc7aa72985f.tar.gz |
Merge pull request #773 from Spartan322/improve/find_children
Optimize `Node.find_children`
Diffstat (limited to 'scene/main')
-rw-r--r-- | scene/main/node.cpp | 65 |
1 files changed, 47 insertions, 18 deletions
diff --git a/scene/main/node.cpp b/scene/main/node.cpp index 8d0a3f831f..fcea7be3de 100644 --- a/scene/main/node.cpp +++ b/scene/main/node.cpp @@ -1931,24 +1931,52 @@ Node *Node::find_child(const String &p_pattern, bool p_recursive, bool p_owned) // Can be recursive or not, and limited to owned nodes. TypedArray<Node> Node::find_children(const String &p_pattern, const String &p_type, bool p_recursive, bool p_owned) const { ERR_THREAD_GUARD_V(TypedArray<Node>()); - TypedArray<Node> ret; - ERR_FAIL_COND_V(p_pattern.is_empty() && p_type.is_empty(), ret); - _update_children_cache(); - Node *const *cptr = data.children_cache.ptr(); - int ccount = data.children_cache.size(); - for (int i = 0; i < ccount; i++) { - if (p_owned && !cptr[i]->data.owner) { - continue; + TypedArray<Node> matches; + ERR_FAIL_COND_V(p_pattern.is_empty() && p_type.is_empty(), matches); + + // Save basic pattern and type info for faster lookup + bool is_pattern_empty = p_pattern.is_empty(); + bool is_type_empty = p_type.is_empty(); + bool is_type_global_class = !is_type_empty && ScriptServer::is_global_class(p_type); + String type_global_path = is_type_global_class ? ScriptServer::get_global_class_path(p_type) : ""; + + LocalVector<Node *> to_search; + to_search.push_back((Node *)this); + bool is_adding_children = true; + while (!to_search.is_empty()) { + // Pop the next entry off the search stack + Node *entry = Object::cast_to<Node>(to_search[0]); + to_search.remove_at(0); + + // Add all the children to the list to search + entry->_update_children_cache(); + if (is_adding_children) { + Node *const *cptr = entry->data.children_cache.ptr(); + int ccount = entry->data.children_cache.size(); + for (int i = 0; i < ccount; i++) { + if (p_owned && !cptr[i]->data.owner) { + continue; + } + + to_search.push_back(cptr[i]); + } + + // Stop further child adding if we don't want recursive + if (!p_recursive) { + is_adding_children = false; + } } - if (p_pattern.is_empty() || cptr[i]->data.name.operator String().match(p_pattern)) { - if (p_type.is_empty() || cptr[i]->is_class(p_type)) { - ret.append(cptr[i]); - } else if (cptr[i]->get_script_instance()) { - Ref<Script> scr = cptr[i]->get_script_instance()->get_script(); + // Check if the entry matches + bool is_pattern_match = is_pattern_empty || entry->data.name.operator String().match(p_pattern); + bool is_type_match = is_type_empty || entry->is_class(p_type); + bool is_script_type_match = false; + if (!is_type_match) { + if (ScriptInstance *script_inst = entry->get_script_instance()) { + Ref<Script> scr = script_inst->get_script(); while (scr.is_valid()) { - if ((ScriptServer::is_global_class(p_type) && ScriptServer::get_global_class_path(p_type) == scr->get_path()) || p_type == scr->get_path()) { - ret.append(cptr[i]); + if ((is_type_global_class && type_global_path == scr->get_path()) || p_type == scr->get_path()) { + is_script_type_match = true; break; } @@ -1957,12 +1985,13 @@ TypedArray<Node> Node::find_children(const String &p_pattern, const String &p_ty } } - if (p_recursive) { - ret.append_array(cptr[i]->find_children(p_pattern, p_type, true, p_owned)); + // Save it if it matches the pattern and at least one type + if (is_pattern_match && (is_type_match || is_script_type_match)) { + matches.append(entry); } } - return ret; + return matches; } void Node::reparent(Node *p_parent, bool p_keep_global_transform) { |