| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  | #include <LibGUI/GAbstractButton.h>
 | 
					
						
							| 
									
										
										
										
											2019-05-24 22:54:37 +02:00
										 |  |  | #include <LibGUI/GPainter.h>
 | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | GAbstractButton::GAbstractButton(GWidget* parent) | 
					
						
							| 
									
										
										
										
											2019-07-13 10:27:19 +02:00
										 |  |  |     : GAbstractButton({}, parent) | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-02 14:58:02 +02:00
										 |  |  | GAbstractButton::GAbstractButton(const StringView& text, GWidget* parent) | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  |     : GWidget(parent) | 
					
						
							|  |  |  |     , m_text(text) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-21 18:13:17 +02:00
										 |  |  |     m_auto_repeat_timer = CTimer::construct(this); | 
					
						
							| 
									
										
										
										
											2019-09-20 15:19:46 +02:00
										 |  |  |     m_auto_repeat_timer->on_timeout = [this] { | 
					
						
							| 
									
										
										
										
											2019-07-13 10:27:19 +02:00
										 |  |  |         click(); | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GAbstractButton::~GAbstractButton() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-02 14:58:02 +02:00
										 |  |  | void GAbstractButton::set_text(const StringView& text) | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (m_text == text) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     m_text = text; | 
					
						
							|  |  |  |     update(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::set_checked(bool checked) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_checked == checked) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     m_checked = checked; | 
					
						
							| 
									
										
										
										
											2019-06-12 05:57:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (is_exclusive() && checked) { | 
					
						
							| 
									
										
										
										
											2019-06-16 21:01:51 +01:00
										 |  |  |         parent_widget()->for_each_child_of_type<GAbstractButton>([&](auto& sibling) { | 
					
						
							| 
									
										
										
										
											2019-08-12 18:48:50 +02:00
										 |  |  |             if (!sibling.is_exclusive() || !sibling.is_checked()) | 
					
						
							| 
									
										
										
										
											2019-06-12 05:57:26 +02:00
										 |  |  |                 return IterationDecision::Continue; | 
					
						
							|  |  |  |             sibling.m_checked = false; | 
					
						
							|  |  |  |             sibling.update(); | 
					
						
							|  |  |  |             if (sibling.on_checked) | 
					
						
							|  |  |  |                 sibling.on_checked(false); | 
					
						
							|  |  |  |             return IterationDecision::Continue; | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         m_checked = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  |     update(); | 
					
						
							| 
									
										
										
										
											2019-05-24 17:11:42 +02:00
										 |  |  |     if (on_checked) | 
					
						
							|  |  |  |         on_checked(checked); | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::set_checkable(bool checkable) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_checkable == checkable) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     m_checkable = checkable; | 
					
						
							|  |  |  |     update(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::mousemove_event(GMouseEvent& event) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool is_over = rect().contains(event.position()); | 
					
						
							|  |  |  |     m_hovered = is_over; | 
					
						
							|  |  |  |     if (event.buttons() & GMouseButton::Left) { | 
					
						
							|  |  |  |         if (is_enabled()) { | 
					
						
							|  |  |  |             bool being_pressed = is_over; | 
					
						
							|  |  |  |             if (being_pressed != m_being_pressed) { | 
					
						
							|  |  |  |                 m_being_pressed = being_pressed; | 
					
						
							| 
									
										
										
										
											2019-07-13 10:27:19 +02:00
										 |  |  |                 if (m_auto_repeat_interval) { | 
					
						
							|  |  |  |                     if (!m_being_pressed) | 
					
						
							| 
									
										
										
										
											2019-09-20 15:19:46 +02:00
										 |  |  |                         m_auto_repeat_timer->stop(); | 
					
						
							| 
									
										
										
										
											2019-07-13 10:27:19 +02:00
										 |  |  |                     else | 
					
						
							| 
									
										
										
										
											2019-09-20 15:19:46 +02:00
										 |  |  |                         m_auto_repeat_timer->start(m_auto_repeat_interval); | 
					
						
							| 
									
										
										
										
											2019-07-13 10:27:19 +02:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  |                 update(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     GWidget::mousemove_event(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::mousedown_event(GMouseEvent& event) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef GABSTRACTBUTTON_DEBUG
 | 
					
						
							|  |  |  |     dbgprintf("GAbstractButton::mouse_down_event: x=%d, y=%d, button=%u\n", event.x(), event.y(), (unsigned)event.button()); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     if (event.button() == GMouseButton::Left) { | 
					
						
							|  |  |  |         if (is_enabled()) { | 
					
						
							|  |  |  |             m_being_pressed = true; | 
					
						
							|  |  |  |             update(); | 
					
						
							| 
									
										
										
										
											2019-07-13 10:27:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (m_auto_repeat_interval) { | 
					
						
							|  |  |  |                 click(); | 
					
						
							| 
									
										
										
										
											2019-09-20 15:19:46 +02:00
										 |  |  |                 m_auto_repeat_timer->start(m_auto_repeat_interval); | 
					
						
							| 
									
										
										
										
											2019-07-13 10:27:19 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     GWidget::mousedown_event(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::mouseup_event(GMouseEvent& event) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef GABSTRACTBUTTON_DEBUG
 | 
					
						
							|  |  |  |     dbgprintf("GAbstractButton::mouse_up_event: x=%d, y=%d, button=%u\n", event.x(), event.y(), (unsigned)event.button()); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     if (event.button() == GMouseButton::Left) { | 
					
						
							| 
									
										
										
										
											2019-09-20 15:19:46 +02:00
										 |  |  |         bool was_auto_repeating = m_auto_repeat_timer->is_active(); | 
					
						
							|  |  |  |         m_auto_repeat_timer->stop(); | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  |         if (is_enabled()) { | 
					
						
							|  |  |  |             bool was_being_pressed = m_being_pressed; | 
					
						
							|  |  |  |             m_being_pressed = false; | 
					
						
							|  |  |  |             update(); | 
					
						
							| 
									
										
										
										
											2019-07-13 10:27:19 +02:00
										 |  |  |             if (was_being_pressed && !was_auto_repeating) | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  |                 click(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     GWidget::mouseup_event(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::enter_event(CEvent&) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_hovered = true; | 
					
						
							|  |  |  |     update(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::leave_event(CEvent&) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_hovered = false; | 
					
						
							|  |  |  |     update(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::keydown_event(GKeyEvent& event) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-20 20:37:31 +02:00
										 |  |  |     if (event.key() == KeyCode::Key_Return) { | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  |         click(); | 
					
						
							| 
									
										
										
										
											2019-09-20 20:37:31 +02:00
										 |  |  |         event.accept(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-24 16:32:20 +02:00
										 |  |  |     GWidget::keydown_event(event); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-24 22:54:37 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::paint_text(GPainter& painter, const Rect& rect, const Font& font, TextAlignment text_alignment) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-25 04:01:22 +02:00
										 |  |  |     auto clipped_rect = rect.intersected(this->rect()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-24 22:54:37 +02:00
										 |  |  |     if (!is_enabled()) { | 
					
						
							| 
									
										
										
										
											2019-05-25 04:01:22 +02:00
										 |  |  |         painter.draw_text(clipped_rect.translated(1, 1), text(), font, text_alignment, Color::White, TextElision::Right); | 
					
						
							|  |  |  |         painter.draw_text(clipped_rect, text(), font, text_alignment, Color::from_rgb(0x808080), TextElision::Right); | 
					
						
							| 
									
										
										
										
											2019-05-24 22:54:37 +02:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (text().is_empty()) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-12-24 02:10:01 +01:00
										 |  |  |     painter.draw_text(clipped_rect, text(), font, text_alignment, SystemColor::ButtonText, TextElision::Right); | 
					
						
							| 
									
										
										
										
											2019-05-25 04:01:22 +02:00
										 |  |  |     if (is_focused()) | 
					
						
							|  |  |  |         painter.draw_rect(clipped_rect.inflated(6, 4), Color(140, 140, 140)); | 
					
						
							| 
									
										
										
										
											2019-05-24 22:54:37 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-25 13:40:57 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GAbstractButton::change_event(GEvent& event) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (event.type() == GEvent::Type::EnabledChange) { | 
					
						
							|  |  |  |         if (!is_enabled()) { | 
					
						
							|  |  |  |             bool was_being_pressed = m_being_pressed; | 
					
						
							|  |  |  |             m_being_pressed = false; | 
					
						
							|  |  |  |             if (was_being_pressed) | 
					
						
							|  |  |  |                 update(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     GWidget::change_event(event); | 
					
						
							|  |  |  | } |