mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 06:12:06 +00:00 
			
		
		
		
	 ca538b6cee
			
		
	
	
		ca538b6cee
		
	
	
	
	
		
			
			You can now register a GWidget subclass with REGISTER_GWIDGET(class) and it will be available for factory construction through the new GWidgetClassRegistration interface. To obtain a GWidgetClassRegistration for a given class name, you call GWidgetClassRegistration::find(class_name). You can also iterate over all the registered classes using GWCR::for_each(callback). This will be very useful for implementing a proper GUI designer, and also in the future for things like script bindings. NOTE: All of the registrations are done in GWidget.cpp at the moment since I ran into trouble with the fricken linker pruning the global constructors this mechanism relies on. :^)
		
			
				
	
	
		
			169 lines
		
	
	
	
		
			5.2 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			169 lines
		
	
	
	
		
			5.2 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable file
		
	
	
	
	
| #include <LibDraw/StylePainter.h>
 | |
| #include <LibGUI/GPainter.h>
 | |
| #include <LibGUI/GSlider.h>
 | |
| 
 | |
| GSlider::GSlider(GWidget* parent)
 | |
|     : GSlider(Orientation::Horizontal, parent)
 | |
| {
 | |
| }
 | |
| 
 | |
| GSlider::GSlider(Orientation orientation, GWidget* parent)
 | |
|     : GWidget(parent)
 | |
|     , m_orientation(orientation)
 | |
| {
 | |
| }
 | |
| 
 | |
| GSlider::~GSlider()
 | |
| {
 | |
| }
 | |
| 
 | |
| void GSlider::set_range(int min, int max)
 | |
| {
 | |
|     ASSERT(min <= max);
 | |
|     if (m_min == min && m_max == max)
 | |
|         return;
 | |
|     m_min = min;
 | |
|     m_max = max;
 | |
| 
 | |
|     if (m_value > max)
 | |
|         m_value = max;
 | |
|     if (m_value < min)
 | |
|         m_value = min;
 | |
|     update();
 | |
| }
 | |
| 
 | |
| void GSlider::set_value(int value)
 | |
| {
 | |
|     if (value > m_max)
 | |
|         value = m_max;
 | |
|     if (value < m_min)
 | |
|         value = m_min;
 | |
|     if (m_value == value)
 | |
|         return;
 | |
|     m_value = value;
 | |
|     update();
 | |
| 
 | |
|     if (on_value_changed)
 | |
|         on_value_changed(m_value);
 | |
| }
 | |
| 
 | |
| void GSlider::paint_event(GPaintEvent& event)
 | |
| {
 | |
|     GPainter painter(*this);
 | |
|     painter.add_clip_rect(event.rect());
 | |
| 
 | |
|     Rect track_rect;
 | |
| 
 | |
|     if (orientation() == Orientation::Horizontal) {
 | |
|         track_rect = { inner_rect().x(), 0, inner_rect().width(), track_size() };
 | |
|         track_rect.center_vertically_within(inner_rect());
 | |
|     } else {
 | |
|         track_rect = { 0, inner_rect().y(), track_size(), inner_rect().height() };
 | |
|         track_rect.center_horizontally_within(inner_rect());
 | |
|     }
 | |
| 
 | |
|     StylePainter::paint_frame(painter, track_rect, FrameShape::Panel, FrameShadow::Sunken, 1);
 | |
|     StylePainter::paint_button(painter, knob_rect(), ButtonStyle::Normal, false, m_knob_hovered);
 | |
| }
 | |
| 
 | |
| Rect GSlider::knob_rect() const
 | |
| {
 | |
|     auto inner_rect = this->inner_rect();
 | |
|     Rect rect;
 | |
|     rect.set_secondary_offset_for_orientation(orientation(), 0);
 | |
|     rect.set_secondary_size_for_orientation(orientation(), knob_secondary_size());
 | |
| 
 | |
|     if (knob_size_mode() == KnobSizeMode::Fixed) {
 | |
|         if (m_max - m_min) {
 | |
|             float scale = (float)inner_rect.primary_size_for_orientation(orientation()) / (float)(m_max - m_min);
 | |
|             rect.set_primary_offset_for_orientation(orientation(), inner_rect.primary_offset_for_orientation(orientation()) + ((int)(m_value * scale)) - (knob_fixed_primary_size() / 2));
 | |
|         } else
 | |
|             rect.set_primary_size_for_orientation(orientation(), 0);
 | |
|         rect.set_primary_size_for_orientation(orientation(), knob_fixed_primary_size());
 | |
|     } else {
 | |
|         float scale = (float)inner_rect.primary_size_for_orientation(orientation()) / (float)(m_max - m_min + 1);
 | |
|         rect.set_primary_offset_for_orientation(orientation(), inner_rect.primary_offset_for_orientation(orientation()) + ((int)(m_value * scale)));
 | |
|         if (m_max - m_min)
 | |
|             rect.set_primary_size_for_orientation(orientation(), ::max((int)(scale), knob_fixed_primary_size()));
 | |
|         else
 | |
|             rect.set_primary_size_for_orientation(orientation(), inner_rect.primary_size_for_orientation(orientation()));
 | |
|     }
 | |
|     if (orientation() == Orientation::Horizontal)
 | |
|         rect.center_vertically_within(inner_rect);
 | |
|     else
 | |
|         rect.center_horizontally_within(inner_rect);
 | |
|     return rect;
 | |
| }
 | |
| 
 | |
| void GSlider::mousedown_event(GMouseEvent& event)
 | |
| {
 | |
|     if (!is_enabled())
 | |
|         return;
 | |
|     if (event.button() == GMouseButton::Left) {
 | |
|         if (knob_rect().contains(event.position())) {
 | |
|             m_dragging = true;
 | |
|             m_drag_origin = event.position();
 | |
|             m_drag_origin_value = m_value;
 | |
|             return;
 | |
|         } else {
 | |
|             if (event.position().primary_offset_for_orientation(orientation()) > knob_rect().last_edge_for_orientation(orientation()))
 | |
|                 set_value(m_value + 1);
 | |
|             else if (event.position().primary_offset_for_orientation(orientation()) < knob_rect().first_edge_for_orientation(orientation()))
 | |
|                 set_value(m_value - 1);
 | |
|         }
 | |
|     }
 | |
|     return GWidget::mousedown_event(event);
 | |
| }
 | |
| 
 | |
| void GSlider::mousemove_event(GMouseEvent& event)
 | |
| {
 | |
|     if (!is_enabled())
 | |
|         return;
 | |
|     set_knob_hovered(knob_rect().contains(event.position()));
 | |
|     if (m_dragging) {
 | |
|         float delta = event.position().primary_offset_for_orientation(orientation()) - m_drag_origin.primary_offset_for_orientation(orientation());
 | |
|         float scrubbable_range = inner_rect().primary_size_for_orientation(orientation());
 | |
|         float value_steps_per_scrubbed_pixel = (m_max - m_min) / scrubbable_range;
 | |
|         float new_value = m_drag_origin_value + (value_steps_per_scrubbed_pixel * delta);
 | |
|         set_value((int)new_value);
 | |
|         return;
 | |
|     }
 | |
|     return GWidget::mousemove_event(event);
 | |
| }
 | |
| 
 | |
| void GSlider::mouseup_event(GMouseEvent& event)
 | |
| {
 | |
|     if (!is_enabled())
 | |
|         return;
 | |
|     if (event.button() == GMouseButton::Left) {
 | |
|         m_dragging = false;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     return GWidget::mouseup_event(event);
 | |
| }
 | |
| 
 | |
| void GSlider::leave_event(CEvent& event)
 | |
| {
 | |
|     if (!is_enabled())
 | |
|         return;
 | |
|     set_knob_hovered(false);
 | |
|     GWidget::leave_event(event);
 | |
| }
 | |
| 
 | |
| void GSlider::change_event(GEvent& event)
 | |
| {
 | |
|     if (event.type() == GEvent::Type::EnabledChange) {
 | |
|         if (!is_enabled())
 | |
|             m_dragging = false;
 | |
|     }
 | |
|     GWidget::change_event(event);
 | |
| }
 | |
| 
 | |
| void GSlider::set_knob_hovered(bool hovered)
 | |
| {
 | |
|     if (m_knob_hovered == hovered)
 | |
|         return;
 | |
|     m_knob_hovered = hovered;
 | |
|     update(knob_rect());
 | |
| }
 |