| 
									
										
										
										
											2020-01-18 09:38:21 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org> | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  * modification, are permitted provided that the following conditions are met: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 1. Redistributions of source code must retain the above copyright notice, this | 
					
						
							|  |  |  |  *    list of conditions and the following disclaimer. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 2. Redistributions in binary form must reproduce the above copyright notice, | 
					
						
							|  |  |  |  *    this list of conditions and the following disclaimer in the documentation | 
					
						
							|  |  |  |  *    and/or other materials provided with the distribution. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
					
						
							|  |  |  |  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
					
						
							|  |  |  |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
					
						
							|  |  |  |  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | 
					
						
							|  |  |  |  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
					
						
							|  |  |  |  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
					
						
							|  |  |  |  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
					
						
							|  |  |  |  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
					
						
							|  |  |  |  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
					
						
							|  |  |  |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-11 09:46:53 +00:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2021-02-03 22:47:50 +01:00
										 |  |  | #include <LibJS/Interpreter.h>
 | 
					
						
							|  |  |  | #include <LibJS/Parser.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/ScriptFunction.h>
 | 
					
						
							| 
									
										
										
										
											2021-02-20 00:43:08 +01:00
										 |  |  | #include <LibWeb/DOM/DOMException.h>
 | 
					
						
							| 
									
										
										
										
											2020-11-11 09:46:53 +00:00
										 |  |  | #include <LibWeb/DOM/Document.h>
 | 
					
						
							| 
									
										
										
										
											2021-02-03 22:47:50 +01:00
										 |  |  | #include <LibWeb/DOM/EventListener.h>
 | 
					
						
							| 
									
										
										
										
											2021-02-20 00:43:08 +01:00
										 |  |  | #include <LibWeb/DOM/ExceptionOr.h>
 | 
					
						
							| 
									
										
										
										
											2021-02-03 22:47:50 +01:00
										 |  |  | #include <LibWeb/HTML/EventHandler.h>
 | 
					
						
							| 
									
										
										
										
											2020-11-21 21:53:18 +00:00
										 |  |  | #include <LibWeb/HTML/HTMLAnchorElement.h>
 | 
					
						
							| 
									
										
										
										
											2020-07-26 15:08:16 +02:00
										 |  |  | #include <LibWeb/HTML/HTMLElement.h>
 | 
					
						
							| 
									
										
										
										
											2021-01-01 18:55:47 +01:00
										 |  |  | #include <LibWeb/Layout/BreakNode.h>
 | 
					
						
							| 
									
										
										
										
											2020-11-22 15:53:01 +01:00
										 |  |  | #include <LibWeb/Layout/TextNode.h>
 | 
					
						
							| 
									
										
										
										
											2021-02-03 22:47:50 +01:00
										 |  |  | #include <LibWeb/UIEvents/EventNames.h>
 | 
					
						
							| 
									
										
										
										
											2019-09-29 11:59:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-28 18:20:36 +02:00
										 |  |  | namespace Web::HTML { | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 11:20:15 +01:00
										 |  |  | HTMLElement::HTMLElement(DOM::Document& document, QualifiedName qualified_name) | 
					
						
							|  |  |  |     : Element(document, move(qualified_name)) | 
					
						
							| 
									
										
										
										
											2019-09-29 11:59:38 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HTMLElement::~HTMLElement() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-03 02:57:28 +01:00
										 |  |  | HTMLElement::ContentEditableState HTMLElement::content_editable_state() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto contenteditable = attribute(HTML::AttributeNames::contenteditable); | 
					
						
							|  |  |  |     // "true" and the empty string map to the "true" state.
 | 
					
						
							|  |  |  |     if ((!contenteditable.is_null() && contenteditable.is_empty()) || contenteditable.equals_ignoring_case("true")) | 
					
						
							|  |  |  |         return ContentEditableState::True; | 
					
						
							|  |  |  |     // "false" maps to the "false" state.
 | 
					
						
							|  |  |  |     if (contenteditable.equals_ignoring_case("false")) | 
					
						
							|  |  |  |         return ContentEditableState::False; | 
					
						
							|  |  |  |     // "inherit", an invalid value, and a missing value all map to the "inherit" state.
 | 
					
						
							|  |  |  |     return ContentEditableState::Inherit; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool HTMLElement::is_editable() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (content_editable_state()) { | 
					
						
							|  |  |  |     case ContentEditableState::True: | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     case ContentEditableState::False: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     case ContentEditableState::Inherit: | 
					
						
							|  |  |  |         return parent() && parent()->is_editable(); | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2021-02-23 20:42:32 +01:00
										 |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-08-03 02:57:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String HTMLElement::content_editable() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (content_editable_state()) { | 
					
						
							|  |  |  |     case ContentEditableState::True: | 
					
						
							|  |  |  |         return "true"; | 
					
						
							|  |  |  |     case ContentEditableState::False: | 
					
						
							|  |  |  |         return "false"; | 
					
						
							|  |  |  |     case ContentEditableState::Inherit: | 
					
						
							|  |  |  |         return "inherit"; | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2021-02-23 20:42:32 +01:00
										 |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-08-03 02:57:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-20 00:43:08 +01:00
										 |  |  | // https://html.spec.whatwg.org/multipage/interaction.html#contenteditable
 | 
					
						
							|  |  |  | DOM::ExceptionOr<void> HTMLElement::set_content_editable(const String& content_editable) | 
					
						
							| 
									
										
										
										
											2020-08-03 02:57:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (content_editable.equals_ignoring_case("inherit")) { | 
					
						
							|  |  |  |         remove_attribute(HTML::AttributeNames::contenteditable); | 
					
						
							| 
									
										
										
										
											2021-02-20 00:43:08 +01:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-08-03 02:57:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (content_editable.equals_ignoring_case("true")) { | 
					
						
							|  |  |  |         set_attribute(HTML::AttributeNames::contenteditable, "true"); | 
					
						
							| 
									
										
										
										
											2021-02-20 00:43:08 +01:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-08-03 02:57:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (content_editable.equals_ignoring_case("false")) { | 
					
						
							|  |  |  |         set_attribute(HTML::AttributeNames::contenteditable, "false"); | 
					
						
							| 
									
										
										
										
											2021-02-20 00:43:08 +01:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-08-03 02:57:28 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-20 00:43:08 +01:00
										 |  |  |     return DOM::SyntaxError::create("Invalid contentEditable value, must be 'true', 'false', or 'inherit'"); | 
					
						
							| 
									
										
										
										
											2020-08-03 02:57:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-11 09:46:53 +00:00
										 |  |  | void HTMLElement::set_inner_text(StringView text) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     remove_all_children(); | 
					
						
							|  |  |  |     append_child(document().create_text_node(text)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     set_needs_style_update(true); | 
					
						
							|  |  |  |     document().invalidate_layout(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String HTMLElement::inner_text() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     StringBuilder builder; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // innerText for element being rendered takes visibility into account, so force a layout and then walk the layout tree.
 | 
					
						
							| 
									
										
										
										
											2020-12-14 10:39:39 +01:00
										 |  |  |     document().update_layout(); | 
					
						
							| 
									
										
										
										
											2020-11-11 09:46:53 +00:00
										 |  |  |     if (!layout_node()) | 
					
						
							|  |  |  |         return text_content(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 15:53:01 +01:00
										 |  |  |     Function<void(const Layout::Node&)> recurse = [&](auto& node) { | 
					
						
							| 
									
										
										
										
											2020-11-11 09:46:53 +00:00
										 |  |  |         for (auto* child = node.first_child(); child; child = child->next_sibling()) { | 
					
						
							| 
									
										
										
										
											2021-01-01 18:55:47 +01:00
										 |  |  |             if (is<Layout::TextNode>(child)) | 
					
						
							| 
									
										
										
										
											2020-11-22 15:53:01 +01:00
										 |  |  |                 builder.append(downcast<Layout::TextNode>(*child).text_for_rendering()); | 
					
						
							| 
									
										
										
										
											2021-01-01 18:55:47 +01:00
										 |  |  |             if (is<Layout::BreakNode>(child)) | 
					
						
							| 
									
										
										
										
											2020-11-11 09:46:53 +00:00
										 |  |  |                 builder.append('\n'); | 
					
						
							|  |  |  |             recurse(*child); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     recurse(*layout_node()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return builder.to_string(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-21 21:53:18 +00:00
										 |  |  | bool HTMLElement::cannot_navigate() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // FIXME: Return true if element's node document is not fully active
 | 
					
						
							|  |  |  |     return !is<HTML::HTMLAnchorElement>(this) && !is_connected(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-03 22:47:50 +01:00
										 |  |  | void HTMLElement::parse_attribute(const FlyString& name, const String& value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Element::parse_attribute(name, value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #undef __ENUMERATE
 | 
					
						
							|  |  |  | #define __ENUMERATE(attribute_name, event_name)                          \
 | 
					
						
							|  |  |  |     if (name == HTML::AttributeNames::attribute_name) {                  \ | 
					
						
							|  |  |  |         set_event_handler_attribute(event_name, EventHandler { value }); \ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ENUMERATE_GLOBAL_EVENT_HANDLERS(__ENUMERATE) | 
					
						
							|  |  |  | #undef __ENUMERATE
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  | } |