view_panner.cpp 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. /*************************************************************************/
  2. /* view_panner.cpp */
  3. /*************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /*************************************************************************/
  8. /* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
  9. /* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
  10. /* */
  11. /* Permission is hereby granted, free of charge, to any person obtaining */
  12. /* a copy of this software and associated documentation files (the */
  13. /* "Software"), to deal in the Software without restriction, including */
  14. /* without limitation the rights to use, copy, modify, merge, publish, */
  15. /* distribute, sublicense, and/or sell copies of the Software, and to */
  16. /* permit persons to whom the Software is furnished to do so, subject to */
  17. /* the following conditions: */
  18. /* */
  19. /* The above copyright notice and this permission notice shall be */
  20. /* included in all copies or substantial portions of the Software. */
  21. /* */
  22. /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
  23. /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
  24. /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
  25. /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
  26. /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
  27. /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
  28. /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
  29. /*************************************************************************/
  30. #include "view_panner.h"
  31. #include "core/input/input.h"
  32. #include "core/input/shortcut.h"
  33. #include "core/os/keyboard.h"
  34. bool ViewPanner::gui_input(const Ref<InputEvent> &p_event, Rect2 p_canvas_rect) {
  35. Ref<InputEventMouseButton> mb = p_event;
  36. if (mb.is_valid()) {
  37. Vector2 scroll_vec = Vector2((mb->get_button_index() == MouseButton::WHEEL_RIGHT) - (mb->get_button_index() == MouseButton::WHEEL_LEFT), (mb->get_button_index() == MouseButton::WHEEL_DOWN) - (mb->get_button_index() == MouseButton::WHEEL_UP));
  38. if (scroll_vec != Vector2()) {
  39. if (control_scheme == SCROLL_PANS) {
  40. if (mb->is_ctrl_pressed()) {
  41. scroll_vec.y *= mb->get_factor();
  42. callback_helper(zoom_callback, varray(scroll_vec, mb->get_position(), mb->is_alt_pressed()));
  43. return true;
  44. } else {
  45. Vector2 panning;
  46. if (mb->is_shift_pressed()) {
  47. panning.x += mb->get_factor() * scroll_vec.y;
  48. panning.y += mb->get_factor() * scroll_vec.x;
  49. } else {
  50. panning.y += mb->get_factor() * scroll_vec.y;
  51. panning.x += mb->get_factor() * scroll_vec.x;
  52. }
  53. callback_helper(scroll_callback, varray(panning, mb->is_alt_pressed()));
  54. return true;
  55. }
  56. } else {
  57. if (mb->is_ctrl_pressed()) {
  58. Vector2 panning;
  59. if (mb->is_shift_pressed()) {
  60. panning.x += mb->get_factor() * scroll_vec.y;
  61. panning.y += mb->get_factor() * scroll_vec.x;
  62. } else {
  63. panning.y += mb->get_factor() * scroll_vec.y;
  64. panning.x += mb->get_factor() * scroll_vec.x;
  65. }
  66. callback_helper(scroll_callback, varray(panning, mb->is_alt_pressed()));
  67. return true;
  68. } else if (!mb->is_shift_pressed()) {
  69. scroll_vec.y *= mb->get_factor();
  70. callback_helper(zoom_callback, varray(scroll_vec, mb->get_position(), mb->is_alt_pressed()));
  71. return true;
  72. }
  73. }
  74. }
  75. // Alt is not used for button presses, so ignore it.
  76. if (mb->is_alt_pressed()) {
  77. return false;
  78. }
  79. bool is_drag_event = mb->get_button_index() == MouseButton::MIDDLE ||
  80. (enable_rmb && mb->get_button_index() == MouseButton::RIGHT) ||
  81. (!simple_panning_enabled && mb->get_button_index() == MouseButton::LEFT && is_panning()) ||
  82. (force_drag && mb->get_button_index() == MouseButton::LEFT);
  83. if (is_drag_event) {
  84. if (mb->is_pressed()) {
  85. is_dragging = true;
  86. } else {
  87. is_dragging = false;
  88. }
  89. return mb->get_button_index() != MouseButton::LEFT || mb->is_pressed(); // Don't consume LMB release events (it fixes some selection problems).
  90. }
  91. }
  92. Ref<InputEventMouseMotion> mm = p_event;
  93. if (mm.is_valid()) {
  94. if (is_dragging) {
  95. if (p_canvas_rect != Rect2()) {
  96. callback_helper(pan_callback, varray(Input::get_singleton()->warp_mouse_motion(mm, p_canvas_rect)));
  97. } else {
  98. callback_helper(pan_callback, varray(mm->get_relative()));
  99. }
  100. return true;
  101. }
  102. }
  103. Ref<InputEventKey> k = p_event;
  104. if (k.is_valid()) {
  105. if (pan_view_shortcut.is_valid() && pan_view_shortcut->matches_event(k)) {
  106. pan_key_pressed = k->is_pressed();
  107. if (simple_panning_enabled || (Input::get_singleton()->get_mouse_button_mask() & MouseButton::LEFT) != MouseButton::NONE) {
  108. is_dragging = pan_key_pressed;
  109. }
  110. return true;
  111. }
  112. }
  113. return false;
  114. }
  115. void ViewPanner::release_pan_key() {
  116. pan_key_pressed = false;
  117. is_dragging = false;
  118. }
  119. void ViewPanner::callback_helper(Callable p_callback, Vector<Variant> p_args) {
  120. const Variant **argptr = (const Variant **)alloca(sizeof(Variant *) * p_args.size());
  121. for (int i = 0; i < p_args.size(); i++) {
  122. argptr[i] = &p_args[i];
  123. }
  124. Variant result;
  125. Callable::CallError ce;
  126. p_callback.callp(argptr, p_args.size(), result, ce);
  127. }
  128. void ViewPanner::set_callbacks(Callable p_scroll_callback, Callable p_pan_callback, Callable p_zoom_callback) {
  129. scroll_callback = p_scroll_callback;
  130. pan_callback = p_pan_callback;
  131. zoom_callback = p_zoom_callback;
  132. }
  133. void ViewPanner::set_control_scheme(ControlScheme p_scheme) {
  134. control_scheme = p_scheme;
  135. }
  136. void ViewPanner::set_enable_rmb(bool p_enable) {
  137. enable_rmb = p_enable;
  138. }
  139. void ViewPanner::set_pan_shortcut(Ref<Shortcut> p_shortcut) {
  140. pan_view_shortcut = p_shortcut;
  141. pan_key_pressed = false;
  142. }
  143. void ViewPanner::set_simple_panning_enabled(bool p_enabled) {
  144. simple_panning_enabled = p_enabled;
  145. }
  146. void ViewPanner::setup(ControlScheme p_scheme, Ref<Shortcut> p_shortcut, bool p_simple_panning) {
  147. set_control_scheme(p_scheme);
  148. set_pan_shortcut(p_shortcut);
  149. set_simple_panning_enabled(p_simple_panning);
  150. }
  151. bool ViewPanner::is_panning() const {
  152. return is_dragging || pan_key_pressed;
  153. }
  154. void ViewPanner::set_force_drag(bool p_force) {
  155. force_drag = p_force;
  156. }
  157. ViewPanner::ViewPanner() {
  158. Array inputs;
  159. inputs.append(InputEventKey::create_reference(Key::SPACE));
  160. pan_view_shortcut.instantiate();
  161. pan_view_shortcut->set_events(inputs);
  162. }