CursorTool.cpp 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. #include "CursorTool.h"
  2. #include "FormEditorWidget.h"
  3. #include "FormWidget.h"
  4. #include "WidgetTreeModel.h"
  5. #include <AK/LogStream.h>
  6. void CursorTool::on_mousedown(GMouseEvent& event)
  7. {
  8. dbg() << "CursorTool::on_mousedown";
  9. auto& form_widget = m_editor.form_widget();
  10. auto result = form_widget.hit_test(event.position(), GWidget::ShouldRespectGreediness::No);
  11. if (event.button() == GMouseButton::Left) {
  12. if (result.widget && result.widget != &form_widget) {
  13. if (event.modifiers() & Mod_Ctrl) {
  14. m_editor.selection().toggle(*result.widget);
  15. } else if (!event.modifiers()) {
  16. if (!m_editor.selection().contains(*result.widget)) {
  17. dbg() << "Selection didn't contain " << *result.widget << ", making it the only selected one";
  18. m_editor.selection().set(*result.widget);
  19. }
  20. m_drag_origin = event.position();
  21. m_positions_before_drag.clear();
  22. m_editor.selection().for_each([&](auto& widget) {
  23. m_positions_before_drag.set(&widget, widget.relative_position());
  24. return IterationDecision::Continue;
  25. });
  26. }
  27. } else {
  28. m_editor.selection().clear();
  29. m_rubber_banding = true;
  30. m_rubber_band_origin = event.position();
  31. m_rubber_band_position = event.position();
  32. form_widget.update();
  33. }
  34. // FIXME: Do we need to update any part of the FormEditorWidget outside the FormWidget?
  35. form_widget.update();
  36. }
  37. }
  38. void CursorTool::on_mouseup(GMouseEvent& event)
  39. {
  40. dbg() << "CursorTool::on_mouseup";
  41. if (event.button() == GMouseButton::Left) {
  42. auto& form_widget = m_editor.form_widget();
  43. auto result = form_widget.hit_test(event.position(), GWidget::ShouldRespectGreediness::No);
  44. if (!m_dragging && !(event.modifiers() & Mod_Ctrl)) {
  45. if (result.widget && result.widget != &form_widget) {
  46. m_editor.selection().set(*result.widget);
  47. // FIXME: Do we need to update any part of the FormEditorWidget outside the FormWidget?
  48. form_widget.update();
  49. }
  50. }
  51. m_dragging = false;
  52. m_rubber_banding = false;
  53. form_widget.update();
  54. }
  55. }
  56. void CursorTool::on_mousemove(GMouseEvent& event)
  57. {
  58. dbg() << "CursorTool::on_mousemove";
  59. auto& form_widget = m_editor.form_widget();
  60. if (m_rubber_banding) {
  61. set_rubber_band_position(event.position());
  62. return;
  63. }
  64. if (!m_dragging && event.buttons() & GMouseButton::Left && event.position() != m_drag_origin) {
  65. auto result = form_widget.hit_test(event.position(), GWidget::ShouldRespectGreediness::No);
  66. if (result.widget && result.widget != &form_widget) {
  67. if (!m_editor.selection().contains(*result.widget)) {
  68. m_editor.selection().set(*result.widget);
  69. // FIXME: Do we need to update any part of the FormEditorWidget outside the FormWidget?
  70. form_widget.update();
  71. }
  72. }
  73. m_dragging = true;
  74. }
  75. if (m_dragging) {
  76. auto movement_delta = event.position() - m_drag_origin;
  77. m_editor.selection().for_each([&](auto& widget) {
  78. auto new_rect = widget.relative_rect();
  79. new_rect.set_location(m_positions_before_drag.get(&widget).value_or({}).translated(movement_delta));
  80. new_rect.set_x(new_rect.x() - (new_rect.x() % m_editor.form_widget().grid_size()));
  81. new_rect.set_y(new_rect.y() - (new_rect.y() % m_editor.form_widget().grid_size()));
  82. widget.set_relative_rect(new_rect);
  83. return IterationDecision::Continue;
  84. });
  85. m_editor.model().update();
  86. return;
  87. }
  88. }
  89. void CursorTool::on_keydown(GKeyEvent& event)
  90. {
  91. dbg() << "CursorTool::on_keydown";
  92. auto move_selected_widgets_by = [this](int x, int y) {
  93. m_editor.selection().for_each([&](auto& widget) {
  94. widget.move_by(x, y);
  95. return IterationDecision::Continue;
  96. });
  97. };
  98. if (event.modifiers() == 0) {
  99. switch (event.key()) {
  100. case Key_Down:
  101. move_selected_widgets_by(0, m_editor.form_widget().grid_size());
  102. break;
  103. case Key_Up:
  104. move_selected_widgets_by(0, -m_editor.form_widget().grid_size());
  105. break;
  106. case Key_Left:
  107. move_selected_widgets_by(-m_editor.form_widget().grid_size(), 0);
  108. break;
  109. case Key_Right:
  110. move_selected_widgets_by(m_editor.form_widget().grid_size(), 0);
  111. break;
  112. }
  113. }
  114. }
  115. void CursorTool::set_rubber_band_position(const Point& position)
  116. {
  117. if (m_rubber_band_position == position)
  118. return;
  119. m_rubber_band_position = position;
  120. auto rubber_band_rect = this->rubber_band_rect();
  121. m_editor.selection().clear();
  122. m_editor.form_widget().for_each_child_widget([&](auto& child) {
  123. if (child.relative_rect().intersects(rubber_band_rect))
  124. m_editor.selection().add(child);
  125. return IterationDecision::Continue;
  126. });
  127. m_editor.form_widget().update();
  128. }
  129. Rect CursorTool::rubber_band_rect() const
  130. {
  131. if (!m_rubber_banding)
  132. return {};
  133. return Rect::from_two_points(m_rubber_band_origin, m_rubber_band_position);
  134. }
  135. void CursorTool::on_second_paint(GPainter& painter, GPaintEvent&)
  136. {
  137. if (!m_rubber_banding)
  138. return;
  139. auto rect = rubber_band_rect();
  140. painter.fill_rect(rect, Color(244, 202, 158, 60));
  141. painter.draw_rect(rect, Color(110, 34, 9));
  142. }