From 4237a6501f8bd4cf768ab7866518fda58596e5c0 Mon Sep 17 00:00:00 2001 From: Maxim Kulkin Date: Wed, 24 Aug 2022 10:57:17 -0400 Subject: [PATCH] Add GraphNode 'selected' and 'unselected' signals, simplify GraphEdit Backport of the fix from master to 3.x --- doc/classes/GraphNode.xml | 10 +++++ scene/gui/graph_edit.cpp | 82 +++++++++++++++++++-------------------- scene/gui/graph_edit.h | 2 + scene/gui/graph_node.cpp | 7 ++++ 4 files changed, 60 insertions(+), 41 deletions(-) diff --git a/doc/classes/GraphNode.xml b/doc/classes/GraphNode.xml index dbf90d52d..0fd4c52f8 100644 --- a/doc/classes/GraphNode.xml +++ b/doc/classes/GraphNode.xml @@ -245,12 +245,22 @@ Emitted when the GraphNode is requested to be resized. Happens on dragging the resizer handle (see [member resizable]). + + + Emitted when the GraphNode is selected. + + Emitted when any GraphNode's slot is updated. + + + Emitted when the GraphNode is unselected. + + diff --git a/scene/gui/graph_edit.cpp b/scene/gui/graph_edit.cpp index 3a76bde90..f306b8739 100644 --- a/scene/gui/graph_edit.cpp +++ b/scene/gui/graph_edit.cpp @@ -380,7 +380,20 @@ void GraphEdit::_graph_node_raised(Node *p_gn) { move_child(connections_layer, first_not_comment); top_layer->raise(); - emit_signal("node_selected", p_gn); +} + +void GraphEdit::_graph_node_selected(Node *p_gn) { + GraphNode *gn = Object::cast_to(p_gn); + ERR_FAIL_COND(!gn); + + emit_signal("node_selected", gn); +} + +void GraphEdit::_graph_node_unselected(Node *p_gn) { + GraphNode *gn = Object::cast_to(p_gn); + ERR_FAIL_COND(!gn); + + emit_signal("node_unselected", gn); } void GraphEdit::_graph_node_moved(Node *p_gn) { @@ -410,6 +423,8 @@ void GraphEdit::add_child_notify(Node *p_child) { if (gn) { gn->set_scale(Vector2(zoom, zoom)); gn->connect("offset_changed", this, "_graph_node_moved", varray(gn)); + gn->connect("selected", this, "_graph_node_selected", varray(gn)); + gn->connect("unselected", this, "_graph_node_unselected", varray(gn)); gn->connect("slot_updated", this, "_graph_node_slot_updated", varray(gn)); gn->connect("raise_request", this, "_graph_node_raised", varray(gn)); gn->connect("item_rect_changed", connections_layer, "update"); @@ -436,6 +451,8 @@ void GraphEdit::remove_child_notify(Node *p_child) { GraphNode *gn = Object::cast_to(p_child); if (gn) { gn->disconnect("offset_changed", this, "_graph_node_moved"); + gn->disconnect("selected", this, "_graph_node_selected"); + gn->disconnect("unselected", this, "_graph_node_unselected"); gn->disconnect("slot_updated", this, "_graph_node_slot_updated"); gn->disconnect("raise_request", this, "_graph_node_raised"); @@ -1140,20 +1157,9 @@ void GraphEdit::_gui_input(const Ref &p_ev) { bool in_box = r.intersects(box_selecting_rect); if (in_box) { - if (!gn->is_selected() && box_selection_mode_additive) { - emit_signal("node_selected", gn); - } else if (gn->is_selected() && !box_selection_mode_additive) { - emit_signal("node_unselected", gn); - } gn->set_selected(box_selection_mode_additive); } else { - bool select = (previous_selected.find(gn) != nullptr); - if (gn->is_selected() && !select) { - emit_signal("node_unselected", gn); - } else if (!gn->is_selected() && select) { - emit_signal("node_selected", gn); - } - gn->set_selected(select); + gn->set_selected(previous_selected.find(gn) != nullptr); } } @@ -1172,13 +1178,7 @@ void GraphEdit::_gui_input(const Ref &p_ev) { continue; } - bool select = (previous_selected.find(gn) != nullptr); - if (gn->is_selected() && !select) { - emit_signal("node_unselected", gn); - } else if (!gn->is_selected() && select) { - emit_signal("node_selected", gn); - } - gn->set_selected(select); + gn->set_selected(previous_selected.find(gn) != nullptr); } top_layer->update(); minimap->update(); @@ -1203,7 +1203,6 @@ void GraphEdit::_gui_input(const Ref &p_ev) { Rect2 r = gn->get_rect(); r.size *= zoom; if (r.has_point(b->get_position())) { - emit_signal("node_unselected", gn); gn->set_selected(false); } } @@ -1235,18 +1234,21 @@ void GraphEdit::_gui_input(const Ref &p_ev) { if (b->get_button_index() == BUTTON_LEFT && b->is_pressed()) { GraphNode *gn = nullptr; + // Find node which was clicked on. for (int i = get_child_count() - 1; i >= 0; i--) { GraphNode *gn_selected = Object::cast_to(get_child(i)); - if (gn_selected) { - if (gn_selected->is_resizing()) { - continue; - } + if (!gn_selected) { + continue; + } - if (gn_selected->has_point((b->get_position() - gn_selected->get_position()) / zoom)) { - gn = gn_selected; - break; - } + if (gn_selected->is_resizing()) { + continue; + } + + if (gn_selected->has_point((b->get_position() - gn_selected->get_position()) / zoom)) { + gn = gn_selected; + break; } } @@ -1255,22 +1257,18 @@ void GraphEdit::_gui_input(const Ref &p_ev) { return; } + // Left-clicked on a node, select it. dragging = true; drag_accum = Vector2(); just_selected = !gn->is_selected(); if (!gn->is_selected() && !Input::get_singleton()->is_key_pressed(KEY_CONTROL)) { for (int i = 0; i < get_child_count(); i++) { GraphNode *o_gn = Object::cast_to(get_child(i)); - if (o_gn) { - if (o_gn == gn) { - o_gn->set_selected(true); - } else { - if (o_gn->is_selected()) { - emit_signal("node_unselected", o_gn); - } - o_gn->set_selected(false); - } + if (!o_gn) { + continue; } + + o_gn->set_selected(o_gn == gn); } } @@ -1293,6 +1291,7 @@ void GraphEdit::_gui_input(const Ref &p_ev) { return; } + // Left-clicked on empty space, start box select. box_selecting = true; box_selecting_from = b->get_position(); if (b->get_control()) { @@ -1325,9 +1324,7 @@ void GraphEdit::_gui_input(const Ref &p_ev) { if (!gn2) { continue; } - if (gn2->is_selected()) { - emit_signal("node_unselected", gn2); - } + gn2->set_selected(false); } } @@ -1335,6 +1332,7 @@ void GraphEdit::_gui_input(const Ref &p_ev) { } if (b->get_button_index() == BUTTON_LEFT && !b->is_pressed() && box_selecting) { + // Box selection ended. Nodes were selected during mouse movement. box_selecting = false; box_selecting_rect = Rect2(); previous_selected.clear(); @@ -1718,6 +1716,8 @@ void GraphEdit::_bind_methods() { ClassDB::bind_method(D_METHOD("is_right_disconnects_enabled"), &GraphEdit::is_right_disconnects_enabled); ClassDB::bind_method(D_METHOD("_graph_node_moved"), &GraphEdit::_graph_node_moved); + ClassDB::bind_method(D_METHOD("_graph_node_selected"), &GraphEdit::_graph_node_selected); + ClassDB::bind_method(D_METHOD("_graph_node_unselected"), &GraphEdit::_graph_node_unselected); ClassDB::bind_method(D_METHOD("_graph_node_raised"), &GraphEdit::_graph_node_raised); ClassDB::bind_method(D_METHOD("_graph_node_slot_updated"), &GraphEdit::_graph_node_slot_updated); diff --git a/scene/gui/graph_edit.h b/scene/gui/graph_edit.h index 8e267976f..846cc0144 100644 --- a/scene/gui/graph_edit.h +++ b/scene/gui/graph_edit.h @@ -169,6 +169,8 @@ private: void _draw_cos_line(CanvasItem *p_where, const Vector2 &p_from, const Vector2 &p_to, const Color &p_color, const Color &p_to_color, float p_width = 2.0, float p_bezier_ratio = 1.0); + void _graph_node_selected(Node *p_gn); + void _graph_node_unselected(Node *p_gn); void _graph_node_raised(Node *p_gn); void _graph_node_moved(Node *p_gn); void _graph_node_slot_updated(int p_index, Node *p_gn); diff --git a/scene/gui/graph_node.cpp b/scene/gui/graph_node.cpp index e04ccf44b..4563979dd 100644 --- a/scene/gui/graph_node.cpp +++ b/scene/gui/graph_node.cpp @@ -587,7 +587,12 @@ Vector2 GraphNode::get_offset() const { } void GraphNode::set_selected(bool p_selected) { + if (selected == p_selected) { + return; + } + selected = p_selected; + emit_signal(p_selected ? "selected" : "unselected"); update(); } @@ -874,6 +879,8 @@ void GraphNode::_bind_methods() { ADD_PROPERTY(PropertyInfo(Variant::INT, "overlay", PROPERTY_HINT_ENUM, "Disabled,Breakpoint,Position"), "set_overlay", "get_overlay"); ADD_SIGNAL(MethodInfo("offset_changed")); + ADD_SIGNAL(MethodInfo("selected")); + ADD_SIGNAL(MethodInfo("unselected")); ADD_SIGNAL(MethodInfo("slot_updated", PropertyInfo(Variant::INT, "idx"))); ADD_SIGNAL(MethodInfo("dragged", PropertyInfo(Variant::VECTOR2, "from"), PropertyInfo(Variant::VECTOR2, "to"))); ADD_SIGNAL(MethodInfo("raise_request"));