2020-06-13 22:24:49 +02:00
|
|
|
/*
|
2024-10-04 13:19:50 +02:00
|
|
|
* Copyright (c) 2020, Andreas Kling <andreas@ladybird.org>
|
2020-06-13 22:24:49 +02:00
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-06-13 22:24:49 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
|
|
#include <LibCore/Forward.h>
|
|
|
|
|
#include <LibGfx/Forward.h>
|
2022-12-12 12:20:02 +01:00
|
|
|
#include <LibWeb/HTML/NavigableContainer.h>
|
2023-05-12 07:17:01 +02:00
|
|
|
#include <LibWeb/Layout/ImageProvider.h>
|
2020-06-13 22:24:49 +02:00
|
|
|
|
2020-07-28 18:20:36 +02:00
|
|
|
namespace Web::HTML {
|
2020-06-13 22:24:49 +02:00
|
|
|
|
2022-03-23 08:24:11 -04:00
|
|
|
class HTMLObjectElement final
|
2022-12-12 12:20:02 +01:00
|
|
|
: public NavigableContainer
|
2023-05-12 07:17:01 +02:00
|
|
|
, public Layout::ImageProvider {
|
2022-12-12 12:20:02 +01:00
|
|
|
WEB_PLATFORM_OBJECT(HTMLObjectElement, NavigableContainer)
|
2024-11-15 04:01:23 +13:00
|
|
|
GC_DECLARE_ALLOCATOR(HTMLObjectElement);
|
2022-03-23 20:15:56 -04:00
|
|
|
|
|
|
|
|
enum class Representation {
|
|
|
|
|
Unknown,
|
|
|
|
|
Image,
|
2024-12-07 08:49:29 -05:00
|
|
|
ContentNavigable,
|
2022-03-23 20:15:56 -04:00
|
|
|
Children,
|
|
|
|
|
};
|
2022-03-23 18:55:54 -04:00
|
|
|
|
2020-06-13 22:24:49 +02:00
|
|
|
public:
|
|
|
|
|
virtual ~HTMLObjectElement() override;
|
|
|
|
|
|
2025-07-22 00:58:28 +02:00
|
|
|
virtual void form_associated_element_attribute_changed(FlyString const& name, Optional<String> const& old_value, Optional<String> const& value, Optional<FlyString> const& namespace_) override;
|
2024-04-17 13:56:55 +02:00
|
|
|
virtual void form_associated_element_was_removed(DOM::Node*) override;
|
2020-06-13 22:24:49 +02:00
|
|
|
|
2023-09-03 15:39:01 +12:00
|
|
|
String data() const;
|
2025-10-31 12:30:47 +00:00
|
|
|
void set_data(String const& data);
|
2022-03-23 08:24:11 -04:00
|
|
|
|
2024-01-16 19:04:45 +01:00
|
|
|
String type() const { return get_attribute_value(HTML::AttributeNames::type); }
|
2020-06-13 22:24:49 +02:00
|
|
|
|
2026-02-15 03:14:09 +00:00
|
|
|
// ^FormAssociatedElement
|
|
|
|
|
virtual bool is_form_associated_element() const override { return true; }
|
|
|
|
|
|
2022-03-01 21:03:30 +00:00
|
|
|
// ^FormAssociatedElement
|
|
|
|
|
// https://html.spec.whatwg.org/multipage/forms.html#category-listed
|
|
|
|
|
virtual bool is_listed() const override { return true; }
|
|
|
|
|
|
LibWeb: Map document element focus to the viewport
Apply the focusing steps' get-the-focusable-area mapping before
rejecting a non-focusable target. This preserves documentElement.focus()
by mapping the non-focusable document element to the Document viewport.
Also map rendered navigable containers with content navigables to their
active document, while leaving hidden containers unfocused. Preserve
Window focus events for child document viewports reached through iframe
focus, while still suppressing the top-level viewport surrogate events.
Treat rendered object elements as focusable through their default
non-null tabindex, even when they show fallback or image content instead
of a child navigable.
Keep the spec focus-chain common-tail handling intact for viewport
focus. The Document object is only our surrogate for the viewport, so
designate viewport focus from the new focus target without dispatching
Window focus/focusin events for that top-level surrogate.
Pass that viewport surrogate as the fallback target for fragment
scrolling and NavigateEvent focus reset, so unfocusable body or fragment
targets still clear stale element focus.
Cover documentElement.focus() in both the activeElement and focus-chain
tests, including a tabindex document element that remains focused as an
element. Cover object focus with and without a child navigable, hidden
object focus attempts, iframe focus events, hidden iframe focus, and
blurring a focused iframe after it becomes hidden. Also cover viewport
fallback for intercepted navigation focus reset and fragment scrolling
to an unfocusable target.
2026-05-20 15:52:18 +02:00
|
|
|
// ^EventTarget
|
|
|
|
|
virtual bool is_focusable() const override
|
|
|
|
|
{
|
|
|
|
|
return meets_focusable_area_rendering_requirements();
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-18 14:11:55 +02:00
|
|
|
virtual void visit_edges(Cell::Visitor&) override;
|
2026-06-10 14:58:06 -05:00
|
|
|
virtual void adopted_from(DOM::Document&) override;
|
2023-08-18 14:11:55 +02:00
|
|
|
|
2020-06-13 22:24:49 +02:00
|
|
|
private:
|
2022-08-28 13:42:07 +02:00
|
|
|
HTMLObjectElement(DOM::Document&, DOM::QualifiedName);
|
|
|
|
|
|
2023-12-24 14:41:33 +01:00
|
|
|
virtual bool is_html_object_element() const override { return true; }
|
|
|
|
|
|
2023-08-07 08:41:28 +02:00
|
|
|
virtual void initialize(JS::Realm&) override;
|
2023-01-10 06:28:20 -05:00
|
|
|
|
2024-12-23 17:51:10 +01:00
|
|
|
virtual bool is_presentational_hint(FlyString const&) const override;
|
2026-04-30 10:44:26 +01:00
|
|
|
virtual void apply_presentational_hints(Vector<CSS::StyleProperty>&) const override;
|
2024-10-01 17:26:01 +01:00
|
|
|
|
LibWeb: Make layout nodes refcounted
Move the layout tree from GC allocation to refcounted ownership so
removed layout and paint subtrees are destroyed synchronously instead
of waiting for the next GC sweep. This dramatically reduces GC memory
usage peaks after layout tree churn and makes it easier for memory use
to fall back after large document updates.
Update layout factories, tree traversal, SVG layout node creation,
paintable back-pointers, and pseudo-element layout links to use RefPtr
ownership.
Make display: contents follow the same shape as Blink and WebKit: the
element itself does not create a layout node, and its children are
flattened into the nearest layout parent. Wrap direct non-whitespace
text in an anonymous inline node when the boxless element contributes
inherited style to that text.
Use an internal inline wrapper for display: contents pseudo-elements
so generated content can still participate in layout, painting, hit
testing, and pseudo-element queries. Keep CSSOM reporting the computed
display value from the pseudo style, not the internal wrapper.
Remove the retained out-of-tree layout node list and its testing hook,
since the flattened model does not need a side owner for boxless
elements. Add coverage for inherited text style, dynamic insertion
order, pseudo-element hit testing, and computed style queries.
2026-06-07 17:50:33 +02:00
|
|
|
virtual RefPtr<Layout::Node> create_layout_node(CSS::ComputedProperties const&) override;
|
2024-12-20 11:32:17 +01:00
|
|
|
virtual void adjust_computed_style(CSS::ComputedProperties&) override;
|
2020-06-13 22:24:49 +02:00
|
|
|
|
2022-03-24 11:14:19 -04:00
|
|
|
bool has_ancestor_media_element_or_object_element_not_showing_fallback_content() const;
|
|
|
|
|
|
2022-03-23 08:24:11 -04:00
|
|
|
void queue_element_task_to_run_object_representation_steps();
|
2024-12-07 08:49:29 -05:00
|
|
|
void run_object_representation_handler_steps(Fetch::Infrastructure::Response const&, MimeSniff::MimeType const&, ReadonlyBytes);
|
2022-03-24 11:09:45 -04:00
|
|
|
void run_object_representation_completed_steps(Representation);
|
2022-03-23 08:24:11 -04:00
|
|
|
void run_object_representation_fallback_steps();
|
|
|
|
|
|
2023-06-11 16:02:37 +02:00
|
|
|
void load_image();
|
2022-03-24 11:09:45 -04:00
|
|
|
void update_layout_and_child_objects(Representation);
|
2022-03-23 08:24:11 -04:00
|
|
|
|
2024-12-07 08:49:29 -05:00
|
|
|
void resource_did_load(Fetch::Infrastructure::Response const&, ReadonlyBytes);
|
|
|
|
|
void resource_did_fail();
|
2022-03-23 08:24:11 -04:00
|
|
|
|
2022-11-05 03:58:14 +00:00
|
|
|
// ^DOM::Element
|
|
|
|
|
virtual i32 default_tab_index_value() const override;
|
|
|
|
|
|
2023-05-12 07:17:01 +02:00
|
|
|
// ^Layout::ImageProvider
|
2024-02-18 20:10:37 -05:00
|
|
|
virtual bool is_image_available() const override;
|
2023-05-20 16:27:31 +02:00
|
|
|
virtual Optional<CSSPixels> intrinsic_width() const override;
|
|
|
|
|
virtual Optional<CSSPixels> intrinsic_height() const override;
|
2023-09-03 17:33:58 -05:00
|
|
|
virtual Optional<CSSPixelFraction> intrinsic_aspect_ratio() const override;
|
2026-05-07 14:39:50 +02:00
|
|
|
virtual Optional<Gfx::DecodedImageFrame> current_image_frame_sized(Gfx::IntSize) const override;
|
2023-05-12 07:17:01 +02:00
|
|
|
virtual void set_visible_in_viewport(bool) override;
|
2025-07-26 14:46:23 +02:00
|
|
|
virtual GC::Ptr<DOM::Element const> to_html_element() const override { return *this; }
|
2025-11-04 20:51:46 +01:00
|
|
|
virtual size_t current_frame_index() const override { return 0; }
|
|
|
|
|
virtual GC::Ptr<DecodedImageData> decoded_image_data() const override { return image_data(); }
|
2023-05-12 07:17:01 +02:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
GC::Ptr<DecodedImageData> image_data() const;
|
2023-06-11 16:02:37 +02:00
|
|
|
|
2024-12-07 08:49:29 -05:00
|
|
|
Representation m_representation { Representation::Unknown };
|
|
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
GC::Ptr<SharedResourceRequest> m_resource_request;
|
2024-11-26 15:29:07 +01:00
|
|
|
|
2024-12-11 13:18:21 -05:00
|
|
|
GC::Ptr<DOM::DocumentObserver> m_document_observer;
|
|
|
|
|
|
2025-05-22 12:16:11 -04:00
|
|
|
Vector<DOM::DocumentLoadEventDelayer> m_document_load_event_delayer_for_object_representation_task;
|
|
|
|
|
Vector<DOM::DocumentLoadEventDelayer> m_document_load_event_delayer_for_resource_load;
|
2020-06-13 22:24:49 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
}
|
2023-12-24 14:41:33 +01:00
|
|
|
|
|
|
|
|
namespace Web::DOM {
|
2025-05-13 07:06:33 -04:00
|
|
|
|
2023-12-24 14:41:33 +01:00
|
|
|
template<>
|
|
|
|
|
inline bool Node::fast_is<HTML::HTMLObjectElement>() const { return is_html_object_element(); }
|
2025-05-13 07:06:33 -04:00
|
|
|
|
2023-12-24 14:41:33 +01:00
|
|
|
}
|