2020-12-06 19:51:55 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2020, the SerenityOS developers.
|
2022-01-30 23:35:51 +00:00
|
|
|
* Copyright (c) 2022, Luke Wilde <lukew@serenityos.org>
|
2024-10-04 13:19:50 +02:00
|
|
|
* Copyright (c) 2022-2023, Andreas Kling <andreas@ladybird.org>
|
2025-01-09 10:20:07 +01:00
|
|
|
* Copyright (c) 2024-2025, Jelle Raaijmakers <jelle@ladybird.org>
|
2020-12-06 19:51:55 +01:00
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-12-06 19:51:55 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2022-01-30 23:35:51 +00:00
|
|
|
#include <LibWeb/DOM/AbstractRange.h>
|
2025-01-09 10:20:07 +01:00
|
|
|
#include <LibWeb/DOM/Node.h>
|
2023-01-11 19:48:53 +01:00
|
|
|
#include <LibWeb/Selection/Selection.h>
|
2024-01-04 09:14:41 +13:00
|
|
|
#include <LibWeb/WebIDL/Types.h>
|
2020-12-06 19:51:55 +01:00
|
|
|
|
|
|
|
namespace Web::DOM {
|
|
|
|
|
2022-10-11 16:10:32 +02:00
|
|
|
enum class RelativeBoundaryPointPosition {
|
|
|
|
Equal,
|
|
|
|
Before,
|
|
|
|
After,
|
|
|
|
};
|
|
|
|
|
|
|
|
// https://dom.spec.whatwg.org/#concept-range-bp-position
|
2024-12-18 13:30:19 +01:00
|
|
|
RelativeBoundaryPointPosition position_of_boundary_point_relative_to_other_boundary_point(BoundaryPoint a, BoundaryPoint b);
|
2022-10-11 16:10:32 +02:00
|
|
|
|
2022-01-30 23:35:51 +00:00
|
|
|
class Range final : public AbstractRange {
|
2022-08-28 13:42:07 +02:00
|
|
|
WEB_PLATFORM_OBJECT(Range, AbstractRange);
|
2024-11-15 04:01:23 +13:00
|
|
|
GC_DECLARE_ALLOCATOR(Range);
|
2022-08-09 01:06:47 +02:00
|
|
|
|
2020-12-06 19:51:55 +01:00
|
|
|
public:
|
2024-11-15 04:01:23 +13:00
|
|
|
[[nodiscard]] static GC::Ref<Range> create(Document&);
|
|
|
|
[[nodiscard]] static GC::Ref<Range> create(HTML::Window&);
|
2024-11-30 10:32:32 +01:00
|
|
|
[[nodiscard]] static GC::Ref<Range> create(GC::Ref<Node> start_container, WebIDL::UnsignedLong start_offset, GC::Ref<Node> end_container, WebIDL::UnsignedLong end_offset);
|
2024-11-15 04:01:23 +13:00
|
|
|
static WebIDL::ExceptionOr<GC::Ref<Range>> construct_impl(JS::Realm&);
|
2022-01-30 23:35:51 +00:00
|
|
|
|
2022-08-09 01:06:47 +02:00
|
|
|
virtual ~Range() override;
|
2020-12-06 19:51:55 +01:00
|
|
|
|
2024-11-30 10:32:32 +01:00
|
|
|
WebIDL::ExceptionOr<void> set_start(GC::Ref<Node> node, WebIDL::UnsignedLong offset);
|
|
|
|
WebIDL::ExceptionOr<void> set_end(GC::Ref<Node> node, WebIDL::UnsignedLong offset);
|
|
|
|
WebIDL::ExceptionOr<void> set_start_before(GC::Ref<Node> node);
|
|
|
|
WebIDL::ExceptionOr<void> set_start_after(GC::Ref<Node> node);
|
|
|
|
WebIDL::ExceptionOr<void> set_end_before(GC::Ref<Node> node);
|
|
|
|
WebIDL::ExceptionOr<void> set_end_after(GC::Ref<Node> node);
|
|
|
|
WebIDL::ExceptionOr<void> select_node(GC::Ref<Node> node);
|
2022-01-31 18:42:36 +00:00
|
|
|
void collapse(bool to_start);
|
2024-11-30 10:32:32 +01:00
|
|
|
WebIDL::ExceptionOr<void> select_node_contents(GC::Ref<Node>);
|
2020-12-06 19:51:55 +01:00
|
|
|
|
2024-03-12 13:08:10 +01:00
|
|
|
void increase_start_offset(Badge<Node>, WebIDL::UnsignedLong);
|
|
|
|
void increase_end_offset(Badge<Node>, WebIDL::UnsignedLong);
|
|
|
|
void decrease_start_offset(Badge<Node>, WebIDL::UnsignedLong);
|
|
|
|
void decrease_end_offset(Badge<Node>, WebIDL::UnsignedLong);
|
|
|
|
|
2022-01-31 18:13:15 +00:00
|
|
|
// https://dom.spec.whatwg.org/#dom-range-start_to_start
|
2024-01-04 09:14:41 +13:00
|
|
|
enum HowToCompareBoundaryPoints : WebIDL::UnsignedShort {
|
2022-01-31 18:13:15 +00:00
|
|
|
START_TO_START = 0,
|
|
|
|
START_TO_END = 1,
|
|
|
|
END_TO_END = 2,
|
|
|
|
END_TO_START = 3,
|
|
|
|
};
|
|
|
|
|
2024-01-04 09:14:41 +13:00
|
|
|
WebIDL::ExceptionOr<WebIDL::Short> compare_boundary_points(WebIDL::UnsignedShort how, Range const& source_range) const;
|
2022-01-31 18:13:15 +00:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
GC::Ref<Range> inverted() const;
|
|
|
|
GC::Ref<Range> normalized() const;
|
|
|
|
GC::Ref<Range> clone_range() const;
|
2020-12-06 19:51:55 +01:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
GC::Ref<Node> common_ancestor_container() const;
|
2022-02-25 20:45:03 +01:00
|
|
|
|
2022-01-31 18:58:08 +00:00
|
|
|
// https://dom.spec.whatwg.org/#dom-range-detach
|
|
|
|
void detach() const
|
|
|
|
{
|
|
|
|
// The detach() method steps are to do nothing.
|
|
|
|
// Note: Its functionality (disabling a Range object) was removed, but the method itself is preserved for compatibility.
|
|
|
|
}
|
|
|
|
|
2024-11-30 10:32:32 +01:00
|
|
|
bool intersects_node(GC::Ref<Node>) const;
|
|
|
|
WebIDL::ExceptionOr<bool> is_point_in_range(GC::Ref<Node>, WebIDL::UnsignedLong offset) const;
|
|
|
|
WebIDL::ExceptionOr<WebIDL::Short> compare_point(GC::Ref<Node>, WebIDL::UnsignedLong offset) const;
|
2022-01-31 19:10:12 +00:00
|
|
|
|
2022-09-25 17:03:42 +01:00
|
|
|
WebIDL::ExceptionOr<void> delete_contents();
|
2024-11-15 04:01:23 +13:00
|
|
|
WebIDL::ExceptionOr<GC::Ref<DocumentFragment>> extract_contents();
|
|
|
|
WebIDL::ExceptionOr<GC::Ref<DocumentFragment>> clone_contents();
|
2022-03-21 18:06:28 +01:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
WebIDL::ExceptionOr<void> insert_node(GC::Ref<Node>);
|
|
|
|
WebIDL::ExceptionOr<void> surround_contents(GC::Ref<Node> new_parent);
|
2022-03-21 18:58:00 +01:00
|
|
|
|
2023-08-26 16:46:33 +12:00
|
|
|
String to_string() const;
|
2022-03-21 16:29:19 +01:00
|
|
|
|
2022-03-21 20:05:25 +01:00
|
|
|
static HashTable<Range*>& live_ranges();
|
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
GC::Ref<Geometry::DOMRectList> get_client_rects();
|
|
|
|
GC::Ref<Geometry::DOMRect> get_bounding_client_rect();
|
2022-12-09 18:50:42 +00:00
|
|
|
|
2024-11-30 10:32:32 +01:00
|
|
|
bool contains_node(GC::Ref<Node>) const;
|
2023-01-11 19:47:38 +01:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
void set_associated_selection(Badge<Selection::Selection>, GC::Ptr<Selection::Selection>);
|
2023-01-11 19:48:53 +01:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
WebIDL::ExceptionOr<GC::Ref<DocumentFragment>> create_contextual_fragment(String const& fragment);
|
2023-03-10 14:56:44 +01:00
|
|
|
|
2025-01-09 10:20:07 +01:00
|
|
|
template<typename Callback>
|
|
|
|
void for_each_contained(Callback callback) const
|
|
|
|
{
|
|
|
|
return const_cast<Range*>(this)->for_each_contained(move(callback));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename Callback>
|
|
|
|
void for_each_contained(Callback callback)
|
|
|
|
{
|
|
|
|
GC::Ptr<Node> end = m_end_container;
|
|
|
|
while (end && !end->next_sibling())
|
|
|
|
end = end->parent();
|
|
|
|
if (end)
|
|
|
|
end = end->next_sibling();
|
|
|
|
|
|
|
|
for (GC::Ptr<Node> node = m_start_container; node && node != end; node = node->next_in_pre_order()) {
|
|
|
|
if (contains_node(*node)) {
|
|
|
|
if (callback(*node) == IterationDecision::Break)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-06 19:51:55 +01:00
|
|
|
private:
|
2022-09-25 16:15:49 -06:00
|
|
|
explicit Range(Document&);
|
2024-11-30 10:32:32 +01:00
|
|
|
Range(GC::Ref<Node> start_container, WebIDL::UnsignedLong start_offset, GC::Ref<Node> end_container, WebIDL::UnsignedLong end_offset);
|
2022-09-25 16:15:49 -06:00
|
|
|
|
2023-08-07 08:41:28 +02:00
|
|
|
virtual void initialize(JS::Realm&) override;
|
2023-01-11 19:48:53 +01:00
|
|
|
virtual void visit_edges(Cell::Visitor&) override;
|
2023-01-10 06:28:20 -05:00
|
|
|
|
2024-11-30 10:32:32 +01:00
|
|
|
GC::Ref<Node> root() const;
|
2022-01-31 18:05:54 +00:00
|
|
|
|
2023-01-11 19:48:53 +01:00
|
|
|
void update_associated_selection();
|
|
|
|
|
2022-01-31 18:05:54 +00:00
|
|
|
enum class StartOrEnd {
|
|
|
|
Start,
|
|
|
|
End,
|
|
|
|
};
|
|
|
|
|
2024-11-30 10:32:32 +01:00
|
|
|
WebIDL::ExceptionOr<void> set_start_or_end(GC::Ref<Node> node, u32 offset, StartOrEnd start_or_end);
|
|
|
|
WebIDL::ExceptionOr<void> select(GC::Ref<Node> node);
|
2022-03-21 18:06:28 +01:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
WebIDL::ExceptionOr<GC::Ref<DocumentFragment>> extract();
|
|
|
|
WebIDL::ExceptionOr<GC::Ref<DocumentFragment>> clone_the_contents();
|
|
|
|
WebIDL::ExceptionOr<void> insert(GC::Ref<Node>);
|
2022-03-21 18:06:28 +01:00
|
|
|
|
2024-11-30 10:32:32 +01:00
|
|
|
bool partially_contains_node(GC::Ref<Node>) const;
|
2023-01-11 19:48:53 +01:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
GC::Ptr<Selection::Selection> m_associated_selection;
|
2020-12-06 19:51:55 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|