| 
									
										
										
										
											2020-05-25 00:25:02 +02:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2022-10-17 10:46:11 +02:00
										 |  |  |  * Copyright (c) 2020-2022, Andreas Kling <kling@serenityos.org> | 
					
						
							| 
									
										
										
										
											2020-05-25 00:25:02 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-05-25 00:25:02 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-24 19:24:36 +02:00
										 |  |  | #include <LibWeb/DOM/Element.h>
 | 
					
						
							| 
									
										
										
										
											2021-09-25 23:15:48 +02:00
										 |  |  | #include <LibWeb/HTML/Parser/HTMLParser.h>
 | 
					
						
							| 
									
										
										
										
											2020-07-28 19:18:23 +02:00
										 |  |  | #include <LibWeb/HTML/Parser/StackOfOpenElements.h>
 | 
					
						
							| 
									
										
										
										
											2020-05-24 19:24:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-28 18:20:36 +02:00
										 |  |  | namespace Web::HTML { | 
					
						
							| 
									
										
										
										
											2020-05-24 19:24:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | static Vector<FlyString> s_base_list { "applet"_fly_string, "caption"_fly_string, "html"_fly_string, "table"_fly_string, "td"_fly_string, "th"_fly_string, "marquee"_fly_string, "object"_fly_string, "template"_fly_string }; | 
					
						
							| 
									
										
										
										
											2020-05-24 22:21:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 13:21:51 -06:00
										 |  |  | StackOfOpenElements::~StackOfOpenElements() = default; | 
					
						
							| 
									
										
										
										
											2020-05-24 19:24:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-17 10:46:11 +02:00
										 |  |  | void StackOfOpenElements::visit_edges(JS::Cell::Visitor& visitor) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (auto& element : m_elements) | 
					
						
							|  |  |  |         visitor.visit(element); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | bool StackOfOpenElements::has_in_scope_impl(FlyString const& tag_name, Vector<FlyString> const& list) const | 
					
						
							| 
									
										
										
										
											2020-05-24 19:24:36 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-04-13 19:21:43 +02:00
										 |  |  |     for (auto const& element : m_elements.in_reverse()) { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (element->local_name() == tag_name) | 
					
						
							| 
									
										
										
										
											2020-05-24 19:24:36 +02:00
										 |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (list.contains_slow(element->local_name())) | 
					
						
							| 
									
										
										
										
											2020-05-24 19:24:36 +02:00
										 |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-23 20:42:32 +01:00
										 |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-05-24 19:24:36 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | bool StackOfOpenElements::has_in_scope(FlyString const& tag_name) const | 
					
						
							| 
									
										
										
										
											2020-05-24 22:21:25 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     return has_in_scope_impl(tag_name, s_base_list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | bool StackOfOpenElements::has_in_scope_impl(const DOM::Element& target_node, Vector<FlyString> const& list) const | 
					
						
							| 
									
										
										
										
											2020-05-27 23:22:42 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-04-13 19:21:43 +02:00
										 |  |  |     for (auto& element : m_elements.in_reverse()) { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (element.ptr() == &target_node) | 
					
						
							| 
									
										
										
										
											2020-05-27 23:22:42 +02:00
										 |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (list.contains_slow(element->local_name())) | 
					
						
							| 
									
										
										
										
											2020-05-27 23:22:42 +02:00
										 |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-23 20:42:32 +01:00
										 |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-05-27 23:22:42 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 19:37:56 +02:00
										 |  |  | bool StackOfOpenElements::has_in_scope(const DOM::Element& target_node) const | 
					
						
							| 
									
										
										
										
											2020-05-27 23:22:42 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     return has_in_scope_impl(target_node, s_base_list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | bool StackOfOpenElements::has_in_button_scope(FlyString const& tag_name) const | 
					
						
							| 
									
										
										
										
											2020-05-24 22:21:25 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     auto list = s_base_list; | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  |     list.append("button"_fly_string); | 
					
						
							| 
									
										
										
										
											2020-05-24 22:21:25 +02:00
										 |  |  |     return has_in_scope_impl(tag_name, list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | bool StackOfOpenElements::has_in_table_scope(FlyString const& tag_name) const | 
					
						
							| 
									
										
										
										
											2020-05-25 20:30:34 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  |     return has_in_scope_impl(tag_name, { "html"_fly_string, "table"_fly_string, "template"_fly_string }); | 
					
						
							| 
									
										
										
										
											2020-05-25 20:30:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | bool StackOfOpenElements::has_in_list_item_scope(FlyString const& tag_name) const | 
					
						
							| 
									
										
										
										
											2020-05-29 22:06:05 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     auto list = s_base_list; | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  |     list.append("ol"_fly_string); | 
					
						
							|  |  |  |     list.append("ul"_fly_string); | 
					
						
							| 
									
										
										
										
											2020-05-29 22:06:05 +02:00
										 |  |  |     return has_in_scope_impl(tag_name, list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-20 21:26:34 +01:00
										 |  |  | // https://html.spec.whatwg.org/multipage/parsing.html#has-an-element-in-select-scope
 | 
					
						
							|  |  |  | // The stack of open elements is said to have a particular element in select scope
 | 
					
						
							|  |  |  | // when it has that element in the specific scope consisting of all element types except the following:
 | 
					
						
							|  |  |  | // - optgroup in the HTML namespace
 | 
					
						
							|  |  |  | // - option in the HTML namespace
 | 
					
						
							|  |  |  | // NOTE: In this case it's "all element types _except_"
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | bool StackOfOpenElements::has_in_select_scope(FlyString const& tag_name) const | 
					
						
							| 
									
										
										
										
											2020-05-30 19:58:52 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-03-20 21:26:34 +01:00
										 |  |  |     // https://html.spec.whatwg.org/multipage/parsing.html#has-an-element-in-the-specific-scope
 | 
					
						
							| 
									
										
										
										
											2022-04-13 19:21:43 +02:00
										 |  |  |     // 1. Initialize node to be the current node (the bottommost node of the stack).
 | 
					
						
							|  |  |  |     for (auto& node : m_elements.in_reverse()) { | 
					
						
							| 
									
										
										
										
											2022-03-20 21:26:34 +01:00
										 |  |  |         // 2. If node is the target node, terminate in a match state.
 | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (node->local_name() == tag_name) | 
					
						
							| 
									
										
										
										
											2022-03-20 21:26:34 +01:00
										 |  |  |             return true; | 
					
						
							|  |  |  |         // 3. Otherwise, if node is one of the element types in list, terminate in a failure state.
 | 
					
						
							|  |  |  |         // NOTE: Here "list" refers to all elements except option and optgroup
 | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (node->local_name() != HTML::TagNames::option && node->local_name() != HTML::TagNames::optgroup) | 
					
						
							| 
									
										
										
										
											2022-03-20 21:26:34 +01:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         // 4. Otherwise, set node to the previous entry in the stack of open elements and return to step 2.
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // [4.] (This will never fail, since the loop will always terminate in the previous step if the top of the stack
 | 
					
						
							|  |  |  |     // — an html element — is reached.)
 | 
					
						
							|  |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-05-30 19:58:52 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 19:37:56 +02:00
										 |  |  | bool StackOfOpenElements::contains(const DOM::Element& element) const | 
					
						
							| 
									
										
										
										
											2020-05-24 22:39:59 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     for (auto& element_on_stack : m_elements) { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (&element == element_on_stack.ptr()) | 
					
						
							| 
									
										
										
										
											2020-05-24 22:39:59 +02:00
										 |  |  |             return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | bool StackOfOpenElements::contains(FlyString const& tag_name) const | 
					
						
							| 
									
										
										
										
											2020-05-30 11:13:57 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     for (auto& element_on_stack : m_elements) { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (element_on_stack->local_name() == tag_name) | 
					
						
							| 
									
										
										
										
											2020-05-30 11:13:57 +02:00
										 |  |  |             return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | void StackOfOpenElements::pop_until_an_element_with_tag_name_has_been_popped(FlyString const& tag_name) | 
					
						
							| 
									
										
										
										
											2020-05-28 18:09:31 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |     while (m_elements.last()->local_name() != tag_name) | 
					
						
							| 
									
										
										
										
											2021-12-02 12:54:34 +00:00
										 |  |  |         (void)pop(); | 
					
						
							|  |  |  |     (void)pop(); | 
					
						
							| 
									
										
										
										
											2020-05-28 18:09:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  | JS::GCPtr<DOM::Element> StackOfOpenElements::topmost_special_node_below(DOM::Element const& formatting_element) | 
					
						
							| 
									
										
										
										
											2020-05-29 22:06:05 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |     JS::GCPtr<DOM::Element> found_element = nullptr; | 
					
						
							| 
									
										
										
										
											2022-04-13 19:21:43 +02:00
										 |  |  |     for (auto& element : m_elements.in_reverse()) { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (element.ptr() == &formatting_element) | 
					
						
							| 
									
										
										
										
											2020-05-29 22:06:05 +02:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2023-11-04 10:19:21 +01:00
										 |  |  |         if (HTMLParser::is_special_tag(element->local_name(), element->namespace_uri())) | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |             found_element = element.ptr(); | 
					
						
							| 
									
										
										
										
											2020-05-29 22:06:05 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |     return found_element.ptr(); | 
					
						
							| 
									
										
										
										
											2020-05-29 22:06:05 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:07:44 +13:00
										 |  |  | StackOfOpenElements::LastElementResult StackOfOpenElements::last_element_with_tag_name(FlyString const& tag_name) | 
					
						
							| 
									
										
										
										
											2020-06-21 17:00:55 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     for (ssize_t i = m_elements.size() - 1; i >= 0; --i) { | 
					
						
							|  |  |  |         auto& element = m_elements[i]; | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (element->local_name() == tag_name) | 
					
						
							|  |  |  |             return { element.ptr(), i }; | 
					
						
							| 
									
										
										
										
											2020-06-21 17:00:55 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-08-19 22:30:33 +01:00
										 |  |  |     return { nullptr, -1 }; | 
					
						
							| 
									
										
										
										
											2020-06-21 17:00:55 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  | JS::GCPtr<DOM::Element> StackOfOpenElements::element_immediately_above(DOM::Element const& target) | 
					
						
							| 
									
										
										
										
											2020-06-21 17:00:55 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     bool found_target = false; | 
					
						
							| 
									
										
										
										
											2022-04-13 19:21:43 +02:00
										 |  |  |     for (auto& element : m_elements.in_reverse()) { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (element.ptr() == &target) { | 
					
						
							| 
									
										
										
										
											2020-06-21 17:00:55 +02:00
										 |  |  |             found_target = true; | 
					
						
							|  |  |  |         } else if (found_target) | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |             return element.ptr(); | 
					
						
							| 
									
										
										
										
											2020-06-21 17:00:55 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  | void StackOfOpenElements::remove(DOM::Element const& element) | 
					
						
							| 
									
										
										
										
											2022-03-20 02:11:42 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |     m_elements.remove_first_matching([&element](auto& other) { | 
					
						
							|  |  |  |         return other.ptr() == &element; | 
					
						
							| 
									
										
										
										
											2022-03-20 02:11:42 +01:00
										 |  |  |     }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  | void StackOfOpenElements::replace(DOM::Element const& to_remove, JS::NonnullGCPtr<DOM::Element> to_add) | 
					
						
							| 
									
										
										
										
											2022-03-20 02:11:42 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     for (size_t i = 0; i < m_elements.size(); i++) { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (m_elements[i].ptr() == &to_remove) { | 
					
						
							| 
									
										
										
										
											2022-03-20 02:11:42 +01:00
										 |  |  |             m_elements.remove(i); | 
					
						
							| 
									
										
										
										
											2022-10-17 10:46:11 +02:00
										 |  |  |             m_elements.insert(i, to_add); | 
					
						
							| 
									
										
										
										
											2022-03-20 02:11:42 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  | void StackOfOpenElements::insert_immediately_below(JS::NonnullGCPtr<DOM::Element> element_to_add, DOM::Element const& target) | 
					
						
							| 
									
										
										
										
											2022-03-20 02:11:42 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     for (size_t i = 0; i < m_elements.size(); i++) { | 
					
						
							| 
									
										
										
										
											2022-08-28 13:42:07 +02:00
										 |  |  |         if (m_elements[i].ptr() == &target) { | 
					
						
							| 
									
										
										
										
											2022-10-17 10:46:11 +02:00
										 |  |  |             m_elements.insert(i + 1, element_to_add); | 
					
						
							| 
									
										
										
										
											2022-03-20 02:11:42 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-24 19:24:36 +02:00
										 |  |  | } |