Range.cpp 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687
  1. /*
  2. * Copyright (c) 2020, the SerenityOS developers.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibWeb/DOM/Document.h>
  7. #include <LibWeb/DOM/Node.h>
  8. #include <LibWeb/DOM/Range.h>
  9. #include <LibWeb/DOM/Window.h>
  10. namespace Web::DOM {
  11. NonnullRefPtr<Range> Range::create(Window& window)
  12. {
  13. return Range::create(window.associated_document());
  14. }
  15. NonnullRefPtr<Range> Range::create(Document& document)
  16. {
  17. return adopt_ref(*new Range(document));
  18. }
  19. NonnullRefPtr<Range> Range::create(Node& start_container, size_t start_offset, Node& end_container, size_t end_offset)
  20. {
  21. return adopt_ref(*new Range(start_container, start_offset, end_container, end_offset));
  22. }
  23. NonnullRefPtr<Range> Range::create_with_global_object(Bindings::WindowObject& window)
  24. {
  25. return Range::create(window.impl());
  26. }
  27. Range::Range(Document& document)
  28. : Range(document, 0, document, 0)
  29. {
  30. }
  31. Range::Range(Node& start_container, size_t start_offset, Node& end_container, size_t end_offset)
  32. : m_start_container(start_container)
  33. , m_start_offset(start_offset)
  34. , m_end_container(end_container)
  35. , m_end_offset(end_offset)
  36. {
  37. }
  38. NonnullRefPtr<Range> Range::clone_range() const
  39. {
  40. return adopt_ref(*new Range(const_cast<Node&>(*m_start_container), m_start_offset, const_cast<Node&>(*m_end_container), m_end_offset));
  41. }
  42. NonnullRefPtr<Range> Range::inverted() const
  43. {
  44. return adopt_ref(*new Range(const_cast<Node&>(*m_end_container), m_end_offset, const_cast<Node&>(*m_start_container), m_start_offset));
  45. }
  46. NonnullRefPtr<Range> Range::normalized() const
  47. {
  48. if (m_start_container.ptr() == m_end_container.ptr()) {
  49. if (m_start_offset <= m_end_offset)
  50. return clone_range();
  51. return inverted();
  52. }
  53. if (m_start_container->is_before(m_end_container))
  54. return clone_range();
  55. return inverted();
  56. }
  57. // https://dom.spec.whatwg.org/#dom-range-commonancestorcontainer
  58. NonnullRefPtr<Node> Range::common_ancestor_container() const
  59. {
  60. // 1. Let container be start node.
  61. auto container = m_start_container;
  62. // 2. While container is not an inclusive ancestor of end node, let container be container’s parent.
  63. while (!container->is_inclusive_ancestor_of(m_end_container)) {
  64. VERIFY(container->parent());
  65. container = *container->parent();
  66. }
  67. // 3. Return container.
  68. return container;
  69. }
  70. }