mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 00:12:44 +00:00 
			
		
		
		
	 70a4678d77
			
		
	
	
		70a4678d77
		
	
	
	
	
		
			
			Document now tracks one "inspected" node, set by set_inspected_node() which is called by Browser's DOM inspector view on_selection callback.
		
			
				
	
	
		
			294 lines
		
	
	
	
		
			7.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			294 lines
		
	
	
	
		
			7.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include <AK/FileSystemPath.h>
 | |
| #include <AK/StringBuilder.h>
 | |
| #include <LibCore/CTimer.h>
 | |
| #include <LibHTML/CSS/StyleResolver.h>
 | |
| #include <LibHTML/DOM/Document.h>
 | |
| #include <LibHTML/DOM/DocumentType.h>
 | |
| #include <LibHTML/DOM/Element.h>
 | |
| #include <LibHTML/DOM/ElementFactory.h>
 | |
| #include <LibHTML/DOM/HTMLBodyElement.h>
 | |
| #include <LibHTML/DOM/HTMLHeadElement.h>
 | |
| #include <LibHTML/DOM/HTMLHtmlElement.h>
 | |
| #include <LibHTML/DOM/HTMLTitleElement.h>
 | |
| #include <LibHTML/Frame.h>
 | |
| #include <LibHTML/Layout/LayoutDocument.h>
 | |
| #include <LibHTML/Layout/LayoutTreeBuilder.h>
 | |
| #include <stdio.h>
 | |
| 
 | |
| Document::Document()
 | |
|     : ParentNode(*this, NodeType::DOCUMENT_NODE)
 | |
|     , m_style_resolver(make<StyleResolver>(*this))
 | |
| {
 | |
|     m_style_update_timer = CTimer::construct();
 | |
|     m_style_update_timer->set_single_shot(true);
 | |
|     m_style_update_timer->set_interval(0);
 | |
|     m_style_update_timer->on_timeout = [this] {
 | |
|         update_style();
 | |
|     };
 | |
| }
 | |
| 
 | |
| Document::~Document()
 | |
| {
 | |
| }
 | |
| 
 | |
| void Document::schedule_style_update()
 | |
| {
 | |
|     if (m_style_update_timer->is_active())
 | |
|         return;
 | |
|     m_style_update_timer->start();
 | |
| }
 | |
| 
 | |
| bool Document::is_child_allowed(const Node& node) const
 | |
| {
 | |
|     switch (node.type()) {
 | |
|     case NodeType::DOCUMENT_NODE:
 | |
|     case NodeType::TEXT_NODE:
 | |
|         return false;
 | |
|     case NodeType::COMMENT_NODE:
 | |
|         return true;
 | |
|     case NodeType::DOCUMENT_TYPE_NODE:
 | |
|         return !first_child_of_type<DocumentType>();
 | |
|     case NodeType::ELEMENT_NODE:
 | |
|         return !first_child_of_type<Element>();
 | |
|     default:
 | |
|         return false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Document::fixup()
 | |
| {
 | |
|     if (!first_child() || !is<DocumentType>(*first_child()))
 | |
|         prepend_child(adopt(*new DocumentType(*this)));
 | |
| 
 | |
|     if (is<HTMLHtmlElement>(first_child()->next_sibling()))
 | |
|         return;
 | |
| 
 | |
|     auto body = create_element(*this, "body");
 | |
|     auto html = create_element(*this, "html");
 | |
|     html->append_child(body);
 | |
|     this->donate_all_children_to(body);
 | |
|     this->append_child(html);
 | |
| }
 | |
| 
 | |
| const HTMLHtmlElement* Document::document_element() const
 | |
| {
 | |
|     return first_child_of_type<HTMLHtmlElement>();
 | |
| }
 | |
| 
 | |
| const HTMLHeadElement* Document::head() const
 | |
| {
 | |
|     auto* html = document_element();
 | |
|     if (!html)
 | |
|         return nullptr;
 | |
|     return html->first_child_of_type<HTMLHeadElement>();
 | |
| }
 | |
| 
 | |
| const HTMLBodyElement* Document::body() const
 | |
| {
 | |
|     auto* html = document_element();
 | |
|     if (!html)
 | |
|         return nullptr;
 | |
|     return html->first_child_of_type<HTMLBodyElement>();
 | |
| }
 | |
| 
 | |
| String Document::title() const
 | |
| {
 | |
|     auto* head_element = head();
 | |
|     if (!head_element)
 | |
|         return {};
 | |
| 
 | |
|     auto* title_element = head_element->first_child_of_type<HTMLTitleElement>();
 | |
|     if (!title_element)
 | |
|         return {};
 | |
| 
 | |
|     return title_element->text_content();
 | |
| }
 | |
| 
 | |
| void Document::attach_to_frame(Badge<Frame>, Frame& frame)
 | |
| {
 | |
|     m_frame = frame.make_weak_ptr();
 | |
|     layout();
 | |
| }
 | |
| 
 | |
| void Document::detach_from_frame(Badge<Frame>, Frame&)
 | |
| {
 | |
|     m_layout_root = nullptr;
 | |
|     m_frame = nullptr;
 | |
| }
 | |
| 
 | |
| Color Document::background_color() const
 | |
| {
 | |
|     auto* body_element = body();
 | |
|     if (!body_element)
 | |
|         return Color::White;
 | |
| 
 | |
|     auto* body_layout_node = body_element->layout_node();
 | |
|     if (!body_layout_node)
 | |
|         return Color::White;
 | |
| 
 | |
|     auto background_color = body_layout_node->style().property(CSS::PropertyID::BackgroundColor);
 | |
|     if (!background_color.has_value() || !background_color.value()->is_color())
 | |
|         return Color::White;
 | |
| 
 | |
|     return background_color.value()->to_color(*this);
 | |
| }
 | |
| 
 | |
| RefPtr<GraphicsBitmap> Document::background_image() const
 | |
| {
 | |
|     auto* body_element = body();
 | |
|     if (!body_element)
 | |
|         return {};
 | |
| 
 | |
|     auto* body_layout_node = body_element->layout_node();
 | |
|     if (!body_layout_node)
 | |
|         return {};
 | |
| 
 | |
|     auto background_image = body_layout_node->style().property(CSS::PropertyID::BackgroundImage);
 | |
|     if (!background_image.has_value() || !background_image.value()->is_image())
 | |
|         return {};
 | |
| 
 | |
|     auto& image_value = static_cast<const ImageStyleValue&>(*background_image.value());
 | |
|     if (!image_value.bitmap())
 | |
|         return {};
 | |
| 
 | |
|     return *image_value.bitmap();
 | |
| }
 | |
| 
 | |
| URL Document::complete_url(const String& string) const
 | |
| {
 | |
|     URL url(string);
 | |
|     if (url.is_valid())
 | |
|         return url;
 | |
| 
 | |
|     FileSystemPath fspath(m_url.path());
 | |
|     StringBuilder builder;
 | |
|     builder.append('/');
 | |
| 
 | |
|     bool document_url_ends_in_slash = m_url.path()[m_url.path().length() - 1] == '/';
 | |
| 
 | |
|     for (int i = 0; i < fspath.parts().size(); ++i) {
 | |
|         if (i == fspath.parts().size() - 1 && !document_url_ends_in_slash)
 | |
|             break;
 | |
|         builder.append(fspath.parts()[i]);
 | |
|         builder.append('/');
 | |
|     }
 | |
|     builder.append(string);
 | |
|     auto built = builder.to_string();
 | |
|     fspath = FileSystemPath(built);
 | |
| 
 | |
|     url = m_url;
 | |
|     url.set_path(fspath.string());
 | |
|     return url;
 | |
| }
 | |
| 
 | |
| void Document::force_layout()
 | |
| {
 | |
|     m_layout_root = nullptr;
 | |
|     layout();
 | |
| }
 | |
| 
 | |
| void Document::layout()
 | |
| {
 | |
|     if (!m_layout_root) {
 | |
|         LayoutTreeBuilder tree_builder;
 | |
|         m_layout_root = tree_builder.build(*this);
 | |
|     }
 | |
|     m_layout_root->layout();
 | |
|     m_layout_root->set_needs_display();
 | |
| }
 | |
| 
 | |
| void Document::update_style()
 | |
| {
 | |
|     for_each_in_subtree([&](Node& node) {
 | |
|         if (node.needs_style_update())
 | |
|             to<Element>(node).recompute_style();
 | |
|         return IterationDecision::Continue;
 | |
|     });
 | |
|     update_layout();
 | |
| }
 | |
| 
 | |
| void Document::update_layout()
 | |
| {
 | |
|     layout();
 | |
|     if (on_layout_updated)
 | |
|         on_layout_updated();
 | |
| }
 | |
| 
 | |
| RefPtr<LayoutNode> Document::create_layout_node(const StyleProperties*) const
 | |
| {
 | |
|     return adopt(*new LayoutDocument(*this, StyleProperties::create()));
 | |
| }
 | |
| 
 | |
| void Document::set_link_color(Color color)
 | |
| {
 | |
|     m_link_color = color;
 | |
| }
 | |
| 
 | |
| void Document::set_active_link_color(Color color)
 | |
| {
 | |
|     m_active_link_color = color;
 | |
| }
 | |
| 
 | |
| void Document::set_visited_link_color(Color color)
 | |
| {
 | |
|     m_visited_link_color = color;
 | |
| }
 | |
| 
 | |
| const LayoutDocument* Document::layout_node() const
 | |
| {
 | |
|     return static_cast<const LayoutDocument*>(Node::layout_node());
 | |
| }
 | |
| 
 | |
| LayoutDocument* Document::layout_node()
 | |
| {
 | |
|     return static_cast<LayoutDocument*>(Node::layout_node());
 | |
| }
 | |
| 
 | |
| void Document::set_inspected_node(Node* node)
 | |
| {
 | |
|     if (m_inspected_node == node)
 | |
|         return;
 | |
| 
 | |
|     if (m_inspected_node && m_inspected_node->layout_node())
 | |
|         m_inspected_node->layout_node()->set_needs_display();
 | |
| 
 | |
|     m_inspected_node = node;
 | |
| 
 | |
|     if (m_inspected_node && m_inspected_node->layout_node())
 | |
|         m_inspected_node->layout_node()->set_needs_display();
 | |
| }
 | |
| 
 | |
| void Document::set_hovered_node(Node* node)
 | |
| {
 | |
|     if (m_hovered_node == node)
 | |
|         return;
 | |
| 
 | |
|     RefPtr<Node> old_hovered_node = move(m_hovered_node);
 | |
|     m_hovered_node = node;
 | |
| 
 | |
|     invalidate_style();
 | |
| }
 | |
| 
 | |
| const Element* Document::get_element_by_id(const String& id) const
 | |
| {
 | |
|     const Element* element = nullptr;
 | |
|     for_each_in_subtree([&](auto& node) {
 | |
|         if (is<Element>(node) && to<Element>(node).attribute("id") == id) {
 | |
|             element = &to<Element>(node);
 | |
|             return IterationDecision::Break;
 | |
|         }
 | |
|         return IterationDecision::Continue;
 | |
|     });
 | |
|     return element;
 | |
| }
 | |
| 
 | |
| Vector<const Element*> Document::get_elements_by_name(const String& name) const
 | |
| {
 | |
|     Vector<const Element*> elements;
 | |
|     for_each_in_subtree([&](auto& node) {
 | |
|         if (is<Element>(node) && to<Element>(node).attribute("name") == name)
 | |
|             elements.append(&to<Element>(node));
 | |
|         return IterationDecision::Continue;
 | |
|     });
 | |
|     return elements;
 | |
| }
 |