Slider.cpp 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/Assertions.h>
  7. #include <AK/StdLibExtras.h>
  8. #include <LibGUI/Painter.h>
  9. #include <LibGUI/Slider.h>
  10. #include <LibGfx/Palette.h>
  11. #include <LibGfx/StylePainter.h>
  12. REGISTER_WIDGET(GUI, HorizontalSlider)
  13. REGISTER_WIDGET(GUI, Slider)
  14. REGISTER_WIDGET(GUI, VerticalSlider)
  15. namespace GUI {
  16. Slider::Slider(Orientation orientation)
  17. : AbstractSlider(orientation)
  18. {
  19. REGISTER_ENUM_PROPERTY("knob_size_mode", knob_size_mode, set_knob_size_mode, KnobSizeMode,
  20. { KnobSizeMode::Fixed, "Fixed" },
  21. { KnobSizeMode::Proportional, "Proportional" });
  22. }
  23. Slider::~Slider()
  24. {
  25. }
  26. void Slider::paint_event(PaintEvent& event)
  27. {
  28. Painter painter(*this);
  29. painter.add_clip_rect(event.rect());
  30. Gfx::IntRect track_rect;
  31. if (orientation() == Orientation::Horizontal) {
  32. track_rect = { inner_rect().x(), 0, inner_rect().width(), track_size() };
  33. track_rect.center_vertically_within(inner_rect());
  34. } else {
  35. track_rect = { 0, inner_rect().y(), track_size(), inner_rect().height() };
  36. track_rect.center_horizontally_within(inner_rect());
  37. }
  38. Gfx::StylePainter::paint_frame(painter, track_rect, palette(), Gfx::FrameShape::Panel, Gfx::FrameShadow::Sunken, 1);
  39. if (is_enabled())
  40. Gfx::StylePainter::paint_button(painter, knob_rect(), palette(), Gfx::ButtonStyle::Normal, false, m_knob_hovered);
  41. else
  42. Gfx::StylePainter::paint_button(painter, knob_rect(), palette(), Gfx::ButtonStyle::Normal, true, m_knob_hovered);
  43. }
  44. Gfx::IntRect Slider::knob_rect() const
  45. {
  46. auto inner_rect = this->inner_rect();
  47. Gfx::IntRect rect;
  48. rect.set_secondary_offset_for_orientation(orientation(), 0);
  49. rect.set_secondary_size_for_orientation(orientation(), knob_secondary_size());
  50. if (knob_size_mode() == KnobSizeMode::Fixed) {
  51. if (max() - min()) {
  52. float scale = (float)inner_rect.primary_size_for_orientation(orientation()) / (float)(max() - min());
  53. rect.set_primary_offset_for_orientation(orientation(), inner_rect.primary_offset_for_orientation(orientation()) + ((int)((value() - min()) * scale)) - (knob_fixed_primary_size() / 2));
  54. } else
  55. rect.set_primary_size_for_orientation(orientation(), 0);
  56. rect.set_primary_size_for_orientation(orientation(), knob_fixed_primary_size());
  57. } else {
  58. float scale = (float)inner_rect.primary_size_for_orientation(orientation()) / (float)(max() - min() + 1);
  59. rect.set_primary_offset_for_orientation(orientation(), inner_rect.primary_offset_for_orientation(orientation()) + ((int)((value() - min()) * scale)));
  60. if (max() - min())
  61. rect.set_primary_size_for_orientation(orientation(), ::max((int)(scale), knob_fixed_primary_size()));
  62. else
  63. rect.set_primary_size_for_orientation(orientation(), inner_rect.primary_size_for_orientation(orientation()));
  64. }
  65. if (orientation() == Orientation::Horizontal)
  66. rect.center_vertically_within(inner_rect);
  67. else
  68. rect.center_horizontally_within(inner_rect);
  69. return rect;
  70. }
  71. void Slider::mousedown_event(MouseEvent& event)
  72. {
  73. if (event.button() == MouseButton::Left) {
  74. if (knob_rect().contains(event.position())) {
  75. m_dragging = true;
  76. m_drag_origin = event.position();
  77. m_drag_origin_value = value();
  78. return;
  79. }
  80. const auto mouse_offset = event.position().primary_offset_for_orientation(orientation());
  81. if (jump_to_cursor()) {
  82. float normalized_mouse_offset = 0.0f;
  83. if (orientation() == Orientation::Vertical) {
  84. normalized_mouse_offset = static_cast<float>(mouse_offset) / static_cast<float>(height());
  85. } else {
  86. normalized_mouse_offset = static_cast<float>(mouse_offset) / static_cast<float>(width());
  87. }
  88. int new_value = static_cast<int>(min() + ((max() - min()) * normalized_mouse_offset));
  89. set_value(new_value);
  90. } else {
  91. auto knob_first_edge = knob_rect().first_edge_for_orientation(orientation());
  92. auto knob_last_edge = knob_rect().last_edge_for_orientation(orientation());
  93. if (mouse_offset > knob_last_edge)
  94. set_value(value() + page_step());
  95. else if (mouse_offset < knob_first_edge)
  96. set_value(value() - page_step());
  97. }
  98. }
  99. return Widget::mousedown_event(event);
  100. }
  101. void Slider::mousemove_event(MouseEvent& event)
  102. {
  103. set_knob_hovered(knob_rect().contains(event.position()));
  104. if (m_dragging) {
  105. float delta = event.position().primary_offset_for_orientation(orientation()) - m_drag_origin.primary_offset_for_orientation(orientation());
  106. float scrubbable_range = inner_rect().primary_size_for_orientation(orientation());
  107. float value_steps_per_scrubbed_pixel = (max() - min()) / scrubbable_range;
  108. float new_value = m_drag_origin_value + (value_steps_per_scrubbed_pixel * delta);
  109. set_value((int)new_value);
  110. return;
  111. }
  112. return Widget::mousemove_event(event);
  113. }
  114. void Slider::mouseup_event(MouseEvent& event)
  115. {
  116. if (event.button() == MouseButton::Left) {
  117. m_dragging = false;
  118. return;
  119. }
  120. return Widget::mouseup_event(event);
  121. }
  122. void Slider::mousewheel_event(MouseEvent& event)
  123. {
  124. auto acceleration_modifier = step();
  125. auto wheel_delta = event.wheel_delta();
  126. if (event.modifiers() == KeyModifier::Mod_Ctrl)
  127. acceleration_modifier *= 6;
  128. if (knob_size_mode() == KnobSizeMode::Proportional)
  129. wheel_delta /= abs(wheel_delta);
  130. if (orientation() == Orientation::Horizontal)
  131. set_value(value() - wheel_delta * acceleration_modifier);
  132. else
  133. set_value(value() + wheel_delta * acceleration_modifier);
  134. Widget::mousewheel_event(event);
  135. }
  136. void Slider::leave_event(Core::Event& event)
  137. {
  138. if (!is_enabled())
  139. return;
  140. set_knob_hovered(false);
  141. Widget::leave_event(event);
  142. }
  143. void Slider::change_event(Event& event)
  144. {
  145. if (event.type() == Event::Type::EnabledChange) {
  146. if (!is_enabled())
  147. m_dragging = false;
  148. }
  149. Widget::change_event(event);
  150. }
  151. void Slider::set_knob_hovered(bool hovered)
  152. {
  153. if (m_knob_hovered == hovered)
  154. return;
  155. m_knob_hovered = hovered;
  156. update(knob_rect());
  157. }
  158. }