summaryrefslogtreecommitdiffstats
path: root/scene/main
diff options
context:
space:
mode:
authorMister Puma <MisterPuma80@gmail.com>2024-10-18 12:52:23 -0700
committerSpartan322 <Megacake1234@gmail.com>2024-10-22 11:51:27 -0400
commitf98dc56dd77716f3ed6ed759f245c8520e22b1f0 (patch)
tree3db8f49a86751f5514756181315ffca3edc474a7 /scene/main
parent9a49cab65c8e8ed8ac5f0613d96ddde403d12728 (diff)
downloadredot-engine-f98dc56dd77716f3ed6ed759f245c8520e22b1f0.tar.gz
Optimize `Node.find_children`
(cherry picked from commit blazium-engine/blazium@474bfd82dc585948b957380cea4c0d328d8ac375)
Diffstat (limited to 'scene/main')
-rw-r--r--scene/main/node.cpp65
1 files changed, 47 insertions, 18 deletions
diff --git a/scene/main/node.cpp b/scene/main/node.cpp
index 06f912243a..06740381af 100644
--- a/scene/main/node.cpp
+++ b/scene/main/node.cpp
@@ -1906,24 +1906,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;
}
@@ -1932,12 +1960,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) {