| 
									
										
										
										
											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-03-25 18:53:20 +01:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:32:51 +01:00
										 |  |  | #include <LibWeb/CSS/Length.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-25 18:53:20 +01:00
										 |  |  | #include <LibWeb/CSS/PropertyID.h>
 | 
					
						
							|  |  |  | #include <LibWeb/CSS/StyleResolver.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:32:51 +01:00
										 |  |  | #include <LibWeb/DOM/Document.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-25 18:53:20 +01:00
										 |  |  | #include <LibWeb/DOM/DocumentFragment.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:32:51 +01:00
										 |  |  | #include <LibWeb/DOM/Element.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-25 18:53:20 +01:00
										 |  |  | #include <LibWeb/DOM/Text.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Dump.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:32:51 +01:00
										 |  |  | #include <LibWeb/Layout/LayoutBlock.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Layout/LayoutInline.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Layout/LayoutListItem.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Layout/LayoutTable.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Layout/LayoutTableCell.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Layout/LayoutTableRow.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-09 21:08:15 +02:00
										 |  |  | #include <LibWeb/Layout/LayoutTableRowGroup.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:32:51 +01:00
										 |  |  | #include <LibWeb/Layout/LayoutTreeBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-25 18:53:20 +01:00
										 |  |  | #include <LibWeb/Parser/HTMLParser.h>
 | 
					
						
							| 
									
										
										
										
											2019-06-15 18:55:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  | namespace Web { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-22 19:10:43 +01:00
										 |  |  | Element::Element(Document& document, const FlyString& tag_name) | 
					
						
							| 
									
										
										
										
											2019-09-29 11:43:07 +02:00
										 |  |  |     : ParentNode(document, NodeType::ELEMENT_NODE) | 
					
						
							| 
									
										
										
										
											2019-06-15 18:55:47 +02:00
										 |  |  |     , m_tag_name(tag_name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Element::~Element() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-22 13:10:04 +01:00
										 |  |  | Attribute* Element::find_attribute(const FlyString& name) | 
					
						
							| 
									
										
										
										
											2019-06-15 21:08:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     for (auto& attribute : m_attributes) { | 
					
						
							|  |  |  |         if (attribute.name() == name) | 
					
						
							|  |  |  |             return &attribute; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-22 13:10:04 +01:00
										 |  |  | const Attribute* Element::find_attribute(const FlyString& name) const | 
					
						
							| 
									
										
										
										
											2019-06-15 21:08:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     for (auto& attribute : m_attributes) { | 
					
						
							|  |  |  |         if (attribute.name() == name) | 
					
						
							|  |  |  |             return &attribute; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-22 13:10:04 +01:00
										 |  |  | String Element::attribute(const FlyString& name) const | 
					
						
							| 
									
										
										
										
											2019-06-15 21:08:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (auto* attribute = find_attribute(name)) | 
					
						
							|  |  |  |         return attribute->value(); | 
					
						
							| 
									
										
										
										
											2019-10-05 22:27:52 +02:00
										 |  |  |     return {}; | 
					
						
							| 
									
										
										
										
											2019-06-15 21:08:36 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-22 13:10:04 +01:00
										 |  |  | void Element::set_attribute(const FlyString& name, const String& value) | 
					
						
							| 
									
										
										
										
											2019-06-15 21:08:36 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (auto* attribute = find_attribute(name)) | 
					
						
							|  |  |  |         attribute->set_value(value); | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2019-08-01 16:50:15 +02:00
										 |  |  |         m_attributes.empend(name, value); | 
					
						
							| 
									
										
										
										
											2019-10-06 10:09:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     parse_attribute(name, value); | 
					
						
							| 
									
										
										
										
											2019-06-15 21:08:36 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Element::set_attributes(Vector<Attribute>&& attributes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_attributes = move(attributes); | 
					
						
							| 
									
										
										
										
											2019-10-06 10:09:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (auto& attribute : m_attributes) | 
					
						
							|  |  |  |         parse_attribute(attribute.name(), attribute.value()); | 
					
						
							| 
									
										
										
										
											2019-06-15 21:08:36 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-06-15 22:49:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-26 23:07:19 +02:00
										 |  |  | bool Element::has_class(const FlyString& class_name) const | 
					
						
							| 
									
										
										
										
											2019-06-27 20:40:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-05-26 23:07:19 +02:00
										 |  |  |     for (auto& class_ : m_classes) { | 
					
						
							|  |  |  |         if (class_ == class_name) | 
					
						
							| 
									
										
										
										
											2019-06-27 20:40:21 +02:00
										 |  |  |             return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-10-05 22:27:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-15 15:06:16 +02:00
										 |  |  | RefPtr<LayoutNode> Element::create_layout_node(const StyleProperties* parent_style) const | 
					
						
							| 
									
										
										
										
											2019-10-05 22:27:52 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-10-15 15:06:16 +02:00
										 |  |  |     auto style = document().style_resolver().resolve_style(*this, parent_style); | 
					
						
							| 
									
										
										
										
											2020-01-02 14:52:34 +01:00
										 |  |  |     const_cast<Element&>(*this).m_resolved_style = style; | 
					
						
							| 
									
										
										
										
											2019-10-15 15:06:16 +02:00
										 |  |  |     auto display = style->string_or_fallback(CSS::PropertyID::Display, "inline"); | 
					
						
							| 
									
										
										
										
											2019-10-05 22:27:52 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (display == "none") | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2020-06-05 13:24:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (tag_name() == "noscript" && document().is_scripting_enabled()) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-11 23:16:53 +02:00
										 |  |  |     if (display == "block") | 
					
						
							| 
									
										
										
										
											2019-10-07 09:23:53 +02:00
										 |  |  |         return adopt(*new LayoutBlock(this, move(style))); | 
					
						
							| 
									
										
										
										
											2019-10-05 22:27:52 +02:00
										 |  |  |     if (display == "inline") | 
					
						
							| 
									
										
										
										
											2019-10-07 09:23:53 +02:00
										 |  |  |         return adopt(*new LayoutInline(*this, move(style))); | 
					
						
							| 
									
										
										
										
											2019-10-11 23:16:53 +02:00
										 |  |  |     if (display == "list-item") | 
					
						
							|  |  |  |         return adopt(*new LayoutListItem(*this, move(style))); | 
					
						
							| 
									
										
										
										
											2019-10-17 23:09:41 +02:00
										 |  |  |     if (display == "table") | 
					
						
							|  |  |  |         return adopt(*new LayoutTable(*this, move(style))); | 
					
						
							|  |  |  |     if (display == "table-row") | 
					
						
							|  |  |  |         return adopt(*new LayoutTableRow(*this, move(style))); | 
					
						
							|  |  |  |     if (display == "table-cell") | 
					
						
							|  |  |  |         return adopt(*new LayoutTableCell(*this, move(style))); | 
					
						
							| 
									
										
										
										
											2020-06-09 21:08:15 +02:00
										 |  |  |     if (display == "table-row-group") | 
					
						
							|  |  |  |         return adopt(*new LayoutTableRowGroup(*this, move(style))); | 
					
						
							| 
									
										
										
										
											2020-05-05 16:06:22 +02:00
										 |  |  |     if (display == "inline-block") { | 
					
						
							|  |  |  |         auto inline_block = adopt(*new LayoutBlock(this, move(style))); | 
					
						
							|  |  |  |         inline_block->set_inline(true); | 
					
						
							|  |  |  |         return inline_block; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-10-05 22:27:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-05 13:17:31 +02:00
										 |  |  |     dbg() << "Unknown display type: _" << display << "_"; | 
					
						
							| 
									
										
										
										
											2020-05-28 12:43:46 +02:00
										 |  |  |     return adopt(*new LayoutBlock(this, move(style))); | 
					
						
							| 
									
										
										
										
											2019-10-05 22:27:52 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-10-06 10:09:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-26 23:07:19 +02:00
										 |  |  | void Element::parse_attribute(const FlyString& name, const String& value) | 
					
						
							| 
									
										
										
										
											2019-10-06 10:09:55 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-05-26 23:07:19 +02:00
										 |  |  |     if (name == "class") { | 
					
						
							|  |  |  |         auto new_classes = value.split_view(' '); | 
					
						
							|  |  |  |         m_classes.clear(); | 
					
						
							|  |  |  |         m_classes.ensure_capacity(new_classes.size()); | 
					
						
							|  |  |  |         for (auto& new_class : new_classes) { | 
					
						
							|  |  |  |             m_classes.unchecked_append(new_class); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-10-06 10:09:55 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-10-14 18:32:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum class StyleDifference { | 
					
						
							|  |  |  |     None, | 
					
						
							|  |  |  |     NeedsRepaint, | 
					
						
							|  |  |  |     NeedsRelayout, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static StyleDifference compute_style_difference(const StyleProperties& old_style, const StyleProperties& new_style, const Document& document) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (old_style == new_style) | 
					
						
							|  |  |  |         return StyleDifference::None; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool needs_repaint = false; | 
					
						
							|  |  |  |     bool needs_relayout = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-15 17:52:08 +02:00
										 |  |  |     if (new_style.string_or_fallback(CSS::PropertyID::Display, "block") != old_style.string_or_fallback(CSS::PropertyID::Color, "block")) | 
					
						
							|  |  |  |         needs_relayout = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-14 18:32:02 +02:00
										 |  |  |     if (new_style.color_or_fallback(CSS::PropertyID::Color, document, Color::Black) != old_style.color_or_fallback(CSS::PropertyID::Color, document, Color::Black)) | 
					
						
							|  |  |  |         needs_repaint = true; | 
					
						
							|  |  |  |     else if (new_style.color_or_fallback(CSS::PropertyID::BackgroundColor, document, Color::Black) != old_style.color_or_fallback(CSS::PropertyID::BackgroundColor, document, Color::Black)) | 
					
						
							|  |  |  |         needs_repaint = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (needs_relayout) | 
					
						
							|  |  |  |         return StyleDifference::NeedsRelayout; | 
					
						
							|  |  |  |     if (needs_repaint) | 
					
						
							|  |  |  |         return StyleDifference::NeedsRepaint; | 
					
						
							|  |  |  |     return StyleDifference::None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Element::recompute_style() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-10-19 18:57:02 +02:00
										 |  |  |     set_needs_style_update(false); | 
					
						
							| 
									
										
										
										
											2019-10-14 18:32:02 +02:00
										 |  |  |     ASSERT(parent()); | 
					
						
							|  |  |  |     auto* parent_layout_node = parent()->layout_node(); | 
					
						
							| 
									
										
										
										
											2019-10-19 18:57:02 +02:00
										 |  |  |     if (!parent_layout_node) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-10-14 18:32:02 +02:00
										 |  |  |     ASSERT(parent_layout_node); | 
					
						
							|  |  |  |     auto style = document().style_resolver().resolve_style(*this, &parent_layout_node->style()); | 
					
						
							| 
									
										
										
										
											2020-01-02 14:52:34 +01:00
										 |  |  |     m_resolved_style = style; | 
					
						
							| 
									
										
										
										
											2019-10-19 18:57:02 +02:00
										 |  |  |     if (!layout_node()) { | 
					
						
							|  |  |  |         if (style->string_or_fallback(CSS::PropertyID::Display, "inline") == "none") | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         // We need a new layout tree here!
 | 
					
						
							|  |  |  |         LayoutTreeBuilder tree_builder; | 
					
						
							|  |  |  |         tree_builder.build(*this); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-05 21:47:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Don't bother with style on widgets. NATIVE LOOK & FEEL BABY!
 | 
					
						
							|  |  |  |     if (layout_node()->is_widget()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-14 18:32:02 +02:00
										 |  |  |     auto diff = compute_style_difference(layout_node()->style(), *style, document()); | 
					
						
							|  |  |  |     if (diff == StyleDifference::None) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     layout_node()->set_style(*style); | 
					
						
							|  |  |  |     if (diff == StyleDifference::NeedsRelayout) { | 
					
						
							| 
									
										
										
										
											2020-06-15 17:52:08 +02:00
										 |  |  |         document().force_layout(); | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-10-14 18:32:02 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (diff == StyleDifference::NeedsRepaint) { | 
					
						
							|  |  |  |         layout_node()->set_needs_display(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-01-04 03:09:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-05 17:09:12 +01:00
										 |  |  | NonnullRefPtr<StyleProperties> Element::computed_style() | 
					
						
							| 
									
										
										
										
											2020-01-04 03:09:22 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-01-05 16:54:38 +01:00
										 |  |  |     auto properties = m_resolved_style->clone(); | 
					
						
							| 
									
										
										
										
											2020-01-04 03:09:22 +01:00
										 |  |  |     if (layout_node() && layout_node()->has_style()) { | 
					
						
							|  |  |  |         CSS::PropertyID box_model_metrics[] = { | 
					
						
							|  |  |  |             CSS::PropertyID::MarginTop, | 
					
						
							|  |  |  |             CSS::PropertyID::MarginBottom, | 
					
						
							|  |  |  |             CSS::PropertyID::MarginLeft, | 
					
						
							|  |  |  |             CSS::PropertyID::MarginRight, | 
					
						
							|  |  |  |             CSS::PropertyID::PaddingTop, | 
					
						
							|  |  |  |             CSS::PropertyID::PaddingBottom, | 
					
						
							|  |  |  |             CSS::PropertyID::PaddingLeft, | 
					
						
							|  |  |  |             CSS::PropertyID::PaddingRight, | 
					
						
							| 
									
										
										
										
											2020-01-05 17:04:37 +01:00
										 |  |  |             CSS::PropertyID::BorderTopWidth, | 
					
						
							|  |  |  |             CSS::PropertyID::BorderBottomWidth, | 
					
						
							|  |  |  |             CSS::PropertyID::BorderLeftWidth, | 
					
						
							|  |  |  |             CSS::PropertyID::BorderRightWidth, | 
					
						
							| 
									
										
										
										
											2020-01-04 03:09:22 +01:00
										 |  |  |         }; | 
					
						
							|  |  |  |         for (CSS::PropertyID id : box_model_metrics) { | 
					
						
							|  |  |  |             auto prop = layout_node()->style().property(id); | 
					
						
							| 
									
										
										
										
											2020-03-06 10:26:05 +01:00
										 |  |  |             if (prop.has_value()) | 
					
						
							| 
									
										
										
										
											2020-01-04 03:09:22 +01:00
										 |  |  |                 properties->set_property(id, prop.value()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-01-05 17:38:52 +01:00
										 |  |  |     return properties; | 
					
						
							| 
									
										
										
										
											2020-01-04 03:09:22 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-25 18:53:20 +01:00
										 |  |  | void Element::set_inner_html(StringView markup) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto fragment = parse_html_fragment(document(), markup); | 
					
						
							|  |  |  |     remove_all_children(); | 
					
						
							|  |  |  |     if (!fragment) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     while (RefPtr<Node> child = fragment->first_child()) { | 
					
						
							|  |  |  |         fragment->remove_child(*child); | 
					
						
							|  |  |  |         append_child(*child); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     set_needs_style_update(true); | 
					
						
							|  |  |  |     document().schedule_style_update(); | 
					
						
							|  |  |  |     document().invalidate_layout(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String Element::inner_html() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     StringBuilder builder; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Function<void(const Node&)> recurse = [&](auto& node) { | 
					
						
							|  |  |  |         for (auto* child = node.first_child(); child; child = child->next_sibling()) { | 
					
						
							|  |  |  |             if (child->is_element()) { | 
					
						
							|  |  |  |                 builder.append('<'); | 
					
						
							|  |  |  |                 builder.append(to<Element>(*child).tag_name()); | 
					
						
							|  |  |  |                 builder.append('>'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 recurse(*child); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 builder.append("</"); | 
					
						
							|  |  |  |                 builder.append(to<Element>(*child).tag_name()); | 
					
						
							|  |  |  |                 builder.append('>'); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (child->is_text()) { | 
					
						
							|  |  |  |                 builder.append(to<Text>(*child).data()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     recurse(*this); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return builder.to_string(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 10:27:02 +01:00
										 |  |  | } |