| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2022-01-31 13:07:22 -05:00
										 |  |  |  * Copyright (c) 2021, Tim Flynn <trflynn89@serenityos.org> | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <LibWeb/DOM/Document.h>
 | 
					
						
							|  |  |  | #include <LibWeb/DOM/Element.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Layout/Label.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Layout/LabelableNode.h>
 | 
					
						
							| 
									
										
										
										
											2021-04-04 12:17:24 -04:00
										 |  |  | #include <LibWeb/Layout/TextNode.h>
 | 
					
						
							| 
									
										
										
										
											2023-02-25 11:04:29 +01:00
										 |  |  | #include <LibWeb/Layout/Viewport.h>
 | 
					
						
							| 
									
										
										
										
											2022-03-10 22:46:35 +01:00
										 |  |  | #include <LibWeb/Painting/LabelablePaintable.h>
 | 
					
						
							| 
									
										
										
										
											2024-06-02 19:00:42 +02:00
										 |  |  | #include <LibWeb/UIEvents/MouseButton.h>
 | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Web::Layout { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-06 10:16:04 -07:00
										 |  |  | JS_DEFINE_ALLOCATOR(Label); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-26 17:42:27 +02:00
										 |  |  | Label::Label(DOM::Document& document, HTML::HTMLLabelElement* element, CSS::StyleProperties style) | 
					
						
							| 
									
										
										
										
											2021-10-06 20:02:41 +02:00
										 |  |  |     : BlockContainer(document, element, move(style)) | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 13:21:51 -06:00
										 |  |  | Label::~Label() = default; | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 17:35:53 +00:00
										 |  |  | void Label::handle_mousedown_on_label(Badge<Painting::TextPaintable>, CSSPixelPoint, unsigned button) | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-02 19:00:42 +02:00
										 |  |  |     if (button != UIEvents::MouseButton::Primary) | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 22:10:08 +01:00
										 |  |  |     if (auto control = dom_node().control(); control && is<Painting::LabelablePaintable>(control->paintable())) { | 
					
						
							| 
									
										
										
										
											2024-03-23 12:03:21 +01:00
										 |  |  |         auto& labelable_paintable = verify_cast<Painting::LabelablePaintable>(*control->paintable()); | 
					
						
							|  |  |  |         labelable_paintable.handle_associated_label_mousedown({}); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     m_tracking_mouse = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 17:35:53 +00:00
										 |  |  | void Label::handle_mouseup_on_label(Badge<Painting::TextPaintable>, CSSPixelPoint position, unsigned button) | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-02 19:00:42 +02:00
										 |  |  |     if (!m_tracking_mouse || button != UIEvents::MouseButton::Primary) | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 22:10:08 +01:00
										 |  |  |     if (auto control = dom_node().control(); control && is<Painting::LabelablePaintable>(control->paintable())) { | 
					
						
							| 
									
										
										
										
											2023-04-20 16:00:42 +01:00
										 |  |  |         bool is_inside_control = control->paintable_box()->absolute_rect().contains(position); | 
					
						
							| 
									
										
										
										
											2023-04-20 16:01:38 +01:00
										 |  |  |         bool is_inside_label = paintable_box()->absolute_rect().contains(position); | 
					
						
							| 
									
										
										
										
											2024-03-23 12:03:21 +01:00
										 |  |  |         if (is_inside_control || is_inside_label) { | 
					
						
							|  |  |  |             auto& labelable_paintable = verify_cast<Painting::LabelablePaintable>(*control->paintable()); | 
					
						
							|  |  |  |             labelable_paintable.handle_associated_label_mouseup({}); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_tracking_mouse = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 17:35:53 +00:00
										 |  |  | void Label::handle_mousemove_on_label(Badge<Painting::TextPaintable>, CSSPixelPoint position, unsigned) | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!m_tracking_mouse) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 22:10:08 +01:00
										 |  |  |     if (auto control = dom_node().control(); control && is<Painting::LabelablePaintable>(control->paintable())) { | 
					
						
							| 
									
										
										
										
											2023-04-20 16:00:42 +01:00
										 |  |  |         bool is_inside_control = control->paintable_box()->absolute_rect().contains(position); | 
					
						
							| 
									
										
										
										
											2023-04-20 16:01:38 +01:00
										 |  |  |         bool is_inside_label = paintable_box()->absolute_rect().contains(position); | 
					
						
							| 
									
										
										
										
											2024-03-23 12:03:21 +01:00
										 |  |  |         auto& labelable_paintable = verify_cast<Painting::LabelablePaintable>(*control->paintable()); | 
					
						
							|  |  |  |         labelable_paintable.handle_associated_label_mousemove({}, is_inside_control || is_inside_label); | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 17:35:53 +00:00
										 |  |  | bool Label::is_inside_associated_label(LabelableNode const& control, CSSPixelPoint position) | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     if (auto* label = label_for_control_node(control); label) | 
					
						
							| 
									
										
										
										
											2023-04-20 16:01:38 +01:00
										 |  |  |         return label->paintable_box()->absolute_rect().contains(position); | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-10 22:46:35 +01:00
										 |  |  | bool Label::is_associated_label_hovered(LabelableNode const& control) | 
					
						
							| 
									
										
										
										
											2021-04-04 12:17:24 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     if (auto* label = label_for_control_node(control); label) { | 
					
						
							|  |  |  |         if (label->document().hovered_node() == &label->dom_node()) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (auto* child = label->first_child_of_type<TextNode>(); child) | 
					
						
							|  |  |  |             return label->document().hovered_node() == &child->dom_node(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 09:55:13 -05:00
										 |  |  | // https://html.spec.whatwg.org/multipage/forms.html#labeled-control
 | 
					
						
							| 
									
										
										
										
											2022-03-10 22:46:35 +01:00
										 |  |  | Label const* Label::label_for_control_node(LabelableNode const& control) | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!control.document().layout_node()) | 
					
						
							| 
									
										
										
										
											2022-02-16 09:55:13 -05:00
										 |  |  |         return nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // The for attribute may be specified to indicate a form control with which the caption is to be associated.
 | 
					
						
							|  |  |  |     // If the attribute is specified, the attribute's value must be the ID of a labelable element in the
 | 
					
						
							|  |  |  |     // same tree as the label element. If the attribute is specified and there is an element in the tree
 | 
					
						
							|  |  |  |     // whose ID is equal to the value of the for attribute, and the first such element in tree order is
 | 
					
						
							|  |  |  |     // a labelable element, then that element is the label element's labeled control.
 | 
					
						
							| 
									
										
										
										
											2024-01-13 20:12:25 +13:00
										 |  |  |     if (auto const& id = control.dom_node().id(); id.has_value() && !id->is_empty()) { | 
					
						
							| 
									
										
										
										
											2022-03-10 22:46:35 +01:00
										 |  |  |         Label const* label = nullptr; | 
					
						
							| 
									
										
										
										
											2022-02-16 09:55:13 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         control.document().layout_node()->for_each_in_inclusive_subtree_of_type<Label>([&](auto& node) { | 
					
						
							|  |  |  |             if (node.dom_node().for_() == id) { | 
					
						
							|  |  |  |                 label = &node; | 
					
						
							| 
									
										
										
										
											2024-05-04 14:47:04 +01:00
										 |  |  |                 return TraversalDecision::Break; | 
					
						
							| 
									
										
										
										
											2022-02-16 09:55:13 -05:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-05-04 14:47:04 +01:00
										 |  |  |             return TraversalDecision::Continue; | 
					
						
							| 
									
										
										
										
											2022-02-16 09:55:13 -05:00
										 |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (label) | 
					
						
							|  |  |  |             return label; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If the for attribute is not specified, but the label element has a labelable element descendant,
 | 
					
						
							|  |  |  |     // then the first such descendant in tree order is the label element's labeled control.
 | 
					
						
							|  |  |  |     return control.first_ancestor_of_type<Label>(); | 
					
						
							| 
									
										
										
										
											2021-04-03 21:32:16 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |