From 4759ad6517a3b82f80ef706060b2cfb047ad8037 Mon Sep 17 00:00:00 2001 From: Fredia Huya-Kouadio Date: Thu, 20 Oct 2022 10:35:37 -0700 Subject: [PATCH] Fix the logic used to route `InputEventScreenDrag` events to `Control` nodes. --- scene/main/viewport.cpp | 22 +++++++++++++++++----- scene/main/viewport.h | 1 + 2 files changed, 18 insertions(+), 5 deletions(-) diff --git a/scene/main/viewport.cpp b/scene/main/viewport.cpp index 5b19ca81e..a0c991f76 100644 --- a/scene/main/viewport.cpp +++ b/scene/main/viewport.cpp @@ -2213,9 +2213,11 @@ void Viewport::_gui_input_event(Ref p_event) { Ref touch_event = p_event; if (touch_event.is_valid()) { Size2 pos = touch_event->get_position(); + const int touch_index = touch_event->get_index(); if (touch_event->is_pressed()) { Control *over = _gui_find_control(pos); if (over) { + gui.touch_focus[touch_index] = over->get_instance_id(); if (!gui.modal_stack.empty()) { Control *top = gui.modal_stack.back()->get(); if (over != top && !top->is_a_parent_of(over)) { @@ -2235,14 +2237,22 @@ void Viewport::_gui_input_event(Ref p_event) { set_input_as_handled(); return; } - } else if (touch_event->get_index() == 0 && gui.last_mouse_focus) { - if (gui.last_mouse_focus->can_process()) { + } else { + ObjectID control_id = gui.touch_focus[touch_index]; + Control *over = Object::cast_to(ObjectDB::get_instance(control_id)); + if (over && over->can_process()) { touch_event = touch_event->xformed_by(Transform2D()); //make a copy - touch_event->set_position(gui.focus_inv_xform.xform(pos)); + if (over == gui.last_mouse_focus) { + pos = gui.focus_inv_xform.xform(pos); + } else { + pos = over->get_global_transform_with_canvas().affine_inverse().xform(pos); + } + touch_event->set_position(pos); - _gui_call_input(gui.last_mouse_focus, touch_event); + _gui_call_input(over, touch_event); } set_input_as_handled(); + gui.touch_focus.erase(touch_index); return; } } @@ -2274,7 +2284,9 @@ void Viewport::_gui_input_event(Ref p_event) { Ref drag_event = p_event; if (drag_event.is_valid()) { - Control *over = gui.mouse_focus; + const int drag_event_index = drag_event->get_index(); + ObjectID control_id = gui.touch_focus[drag_event_index]; + Control *over = Object::cast_to(ObjectDB::get_instance(control_id)); if (!over) { over = _gui_find_control(drag_event->get_position()); } diff --git a/scene/main/viewport.h b/scene/main/viewport.h index 1697d83ac..d711c753b 100644 --- a/scene/main/viewport.h +++ b/scene/main/viewport.h @@ -448,6 +448,7 @@ private: // info used when this is a window bool key_event_accepted; + Map touch_focus; Control *mouse_focus; Control *last_mouse_focus; Control *mouse_click_grabber;