mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-25 18:34:14 +00:00 
			
		
		
		
	 96f34d26c9
			
		
	
	
		96f34d26c9
		
	
	
	
	
		
			
			Use a zero-timer to schedule a style update after invalidating style on any node. Nodes now have a needs_style_update flag which helps us batch and coalesce the work. We also start style updates at the root and work our way through the document, updating any node that has the needs_style_update flag set. This is slower than what we were doing before, but far more correct. There is a ton of room for improvement here. :^)
		
			
				
	
	
		
			78 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			78 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include <AK/StringBuilder.h>
 | |
| #include <LibHTML/CSS/StyleResolver.h>
 | |
| #include <LibHTML/DOM/Element.h>
 | |
| #include <LibHTML/DOM/HTMLAnchorElement.h>
 | |
| #include <LibHTML/DOM/Node.h>
 | |
| #include <LibHTML/Layout/LayoutBlock.h>
 | |
| #include <LibHTML/Layout/LayoutDocument.h>
 | |
| #include <LibHTML/Layout/LayoutInline.h>
 | |
| #include <LibHTML/Layout/LayoutNode.h>
 | |
| #include <LibHTML/Layout/LayoutText.h>
 | |
| 
 | |
| Node::Node(Document& document, NodeType type)
 | |
|     : m_document(document)
 | |
|     , m_type(type)
 | |
| {
 | |
| }
 | |
| 
 | |
| Node::~Node()
 | |
| {
 | |
| }
 | |
| 
 | |
| const HTMLAnchorElement* Node::enclosing_link_element() const
 | |
| {
 | |
|     return first_ancestor_of_type<HTMLAnchorElement>();
 | |
| }
 | |
| 
 | |
| const HTMLElement* Node::enclosing_html_element() const
 | |
| {
 | |
|     return first_ancestor_of_type<HTMLElement>();
 | |
| }
 | |
| 
 | |
| String Node::text_content() const
 | |
| {
 | |
|     Vector<String> strings;
 | |
|     StringBuilder builder;
 | |
|     for (auto* child = first_child(); child; child = child->next_sibling()) {
 | |
|         auto text = child->text_content();
 | |
|         if (!text.is_empty()) {
 | |
|             builder.append(child->text_content());
 | |
|             builder.append(' ');
 | |
|         }
 | |
|     }
 | |
|     if (builder.length() > 1)
 | |
|         builder.trim(1);
 | |
|     return builder.to_string();
 | |
| }
 | |
| 
 | |
| const Element* Node::next_element_sibling() const
 | |
| {
 | |
|     for (auto* node = next_sibling(); node; node = node->next_sibling()) {
 | |
|         if (node->is_element())
 | |
|             return static_cast<const Element*>(node);
 | |
|     }
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| const Element* Node::previous_element_sibling() const
 | |
| {
 | |
|     for (auto* node = previous_sibling(); node; node = node->previous_sibling()) {
 | |
|         if (node->is_element())
 | |
|             return static_cast<const Element*>(node);
 | |
|     }
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| RefPtr<LayoutNode> Node::create_layout_node(const StyleProperties*) const
 | |
| {
 | |
|     return nullptr;
 | |
| }
 | |
| 
 | |
| void Node::invalidate_style()
 | |
| {
 | |
|     for_each_in_subtree([&](auto& node) {
 | |
|         if (is<Element>(node))
 | |
|             node.set_needs_style_update(true);
 | |
|     });
 | |
|     document().schedule_style_update();
 | |
| }
 |