diff --git a/doc/classes/SubViewportContainer.xml b/doc/classes/SubViewportContainer.xml index 2f666f5089b..08e7ca23f74 100644 --- a/doc/classes/SubViewportContainer.xml +++ b/doc/classes/SubViewportContainer.xml @@ -11,6 +11,7 @@ + If [code]true[/code], the sub-viewport will be automatically resized to the control's size. [b]Note:[/b] If [code]true[/code], this will prohibit changing [member SubViewport.size] of its children manually. diff --git a/doc/classes/Viewport.xml b/doc/classes/Viewport.xml index c0c53146fe3..9a5e7ed6f64 100644 --- a/doc/classes/Viewport.xml +++ b/doc/classes/Viewport.xml @@ -160,8 +160,8 @@ - [method Node._input] - [method Control._gui_input] for [Control] nodes - [method Node._shortcut_input] - - [method Node._unhandled_input] - [method Node._unhandled_key_input] + - [method Node._unhandled_input] If an earlier method marks the input as handled via [method set_input_as_handled], any later method in this list will not be called. If none of the methods handle the event and [member physics_object_picking] is [code]true[/code], the event is used for physics object picking. @@ -183,8 +183,8 @@ While this method serves a similar purpose as [method Input.parse_input_event], it does not remap the specified [param event] based on project settings like [member ProjectSettings.input_devices/pointing/emulate_touch_from_mouse]. Calling this method will propagate calls to child nodes for following methods in the given order: - [method Node._shortcut_input] - - [method Node._unhandled_input] - [method Node._unhandled_key_input] + - [method Node._unhandled_input] If an earlier method marks the input as handled via [method set_input_as_handled], any later method in this list will not be called. If none of the methods handle the event and [member physics_object_picking] is [code]true[/code], the event is used for physics object picking. [b]Note:[/b] This method doesn't propagate input events to embedded [Window]s or [SubViewport]s. diff --git a/scene/gui/subviewport_container.cpp b/scene/gui/subviewport_container.cpp index 91058374869..8142fc204af 100644 --- a/scene/gui/subviewport_container.cpp +++ b/scene/gui/subviewport_container.cpp @@ -154,6 +154,18 @@ void SubViewportContainer::_notification(int p_what) { case NOTIFICATION_MOUSE_EXIT: { _notify_viewports(NOTIFICATION_VP_MOUSE_EXIT); } break; + + case NOTIFICATION_FOCUS_ENTER: { + // If focused, send InputEvent to the SubViewport before the Gui-Input stage. + set_process_input(true); + set_process_unhandled_input(false); + } break; + + case NOTIFICATION_FOCUS_EXIT: { + // A different Control has focus and should receive Gui-Input before the InputEvent is sent to the SubViewport. + set_process_input(false); + set_process_unhandled_input(true); + } break; } } @@ -168,6 +180,14 @@ void SubViewportContainer::_notify_viewports(int p_notification) { } void SubViewportContainer::input(const Ref &p_event) { + _propagate_nonpositional_event(p_event); +} + +void SubViewportContainer::unhandled_input(const Ref &p_event) { + _propagate_nonpositional_event(p_event); +} + +void SubViewportContainer::_propagate_nonpositional_event(const Ref &p_event) { ERR_FAIL_COND(p_event.is_null()); if (Engine::get_singleton()->is_editor_hint()) { @@ -262,5 +282,6 @@ void SubViewportContainer::_bind_methods() { } SubViewportContainer::SubViewportContainer() { - set_process_input(true); + set_process_unhandled_input(true); + set_focus_mode(FOCUS_CLICK); } diff --git a/scene/gui/subviewport_container.h b/scene/gui/subviewport_container.h index 8e5f5d157db..3c6cd09d666 100644 --- a/scene/gui/subviewport_container.h +++ b/scene/gui/subviewport_container.h @@ -41,6 +41,7 @@ class SubViewportContainer : public Container { void _notify_viewports(int p_notification); bool _is_propagated_in_gui_input(const Ref &p_event); void _send_event_to_viewports(const Ref &p_event); + void _propagate_nonpositional_event(const Ref &p_event); protected: void _notification(int p_what); @@ -54,6 +55,7 @@ public: bool is_stretch_enabled() const; virtual void input(const Ref &p_event) override; + virtual void unhandled_input(const Ref &p_event) override; virtual void gui_input(const Ref &p_event) override; void set_stretch_shrink(int p_shrink); int get_stretch_shrink() const; diff --git a/scene/main/viewport.cpp b/scene/main/viewport.cpp index 72676264837..1cde463df45 100644 --- a/scene/main/viewport.cpp +++ b/scene/main/viewport.cpp @@ -3026,18 +3026,18 @@ void Viewport::_push_unhandled_input_internal(const Ref &p_event) { get_tree()->_call_input_pause(shortcut_input_group, SceneTree::CALL_INPUT_TYPE_SHORTCUT_INPUT, p_event, this); } - // Unhandled Input. - if (!is_input_handled()) { - ERR_FAIL_COND(!is_inside_tree()); - get_tree()->_call_input_pause(unhandled_input_group, SceneTree::CALL_INPUT_TYPE_UNHANDLED_INPUT, p_event, this); - } - // Unhandled key Input - Used for performance reasons - This is called a lot less than _unhandled_input since it ignores MouseMotion, and to handle Unicode input with Alt / Ctrl modifiers after handling shortcuts. if (!is_input_handled() && (Object::cast_to(*p_event) != nullptr)) { ERR_FAIL_COND(!is_inside_tree()); get_tree()->_call_input_pause(unhandled_key_input_group, SceneTree::CALL_INPUT_TYPE_UNHANDLED_KEY_INPUT, p_event, this); } + // Unhandled Input. + if (!is_input_handled()) { + ERR_FAIL_COND(!is_inside_tree()); + get_tree()->_call_input_pause(unhandled_input_group, SceneTree::CALL_INPUT_TYPE_UNHANDLED_INPUT, p_event, this); + } + if (physics_object_picking && !is_input_handled()) { if (Input::get_singleton()->get_mouse_mode() != Input::MOUSE_MODE_CAPTURED && (Object::cast_to(*p_event) ||