EditingHostManager.cpp 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. /*
  2. * Copyright (c) 2024, Aliaksandr Kalenik <kalenik.aliaksandr@gmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibWeb/DOM/Document.h>
  7. #include <LibWeb/DOM/EditingHostManager.h>
  8. #include <LibWeb/DOM/Range.h>
  9. #include <LibWeb/DOM/Text.h>
  10. #include <LibWeb/Selection/Selection.h>
  11. namespace Web::DOM {
  12. GC_DEFINE_ALLOCATOR(EditingHostManager);
  13. GC::Ref<EditingHostManager> EditingHostManager::create(JS::Realm& realm, GC::Ref<Document> document)
  14. {
  15. return realm.create<EditingHostManager>(document);
  16. }
  17. EditingHostManager::EditingHostManager(GC::Ref<Document> document)
  18. : m_document(document)
  19. {
  20. }
  21. void EditingHostManager::visit_edges(Cell::Visitor& visitor)
  22. {
  23. Base::visit_edges(visitor);
  24. visitor.visit(m_document);
  25. visitor.visit(m_active_contenteditable_element);
  26. }
  27. void EditingHostManager::handle_insert(String const& data)
  28. {
  29. auto selection = m_document->get_selection();
  30. auto selection_range = selection->range();
  31. if (!selection_range) {
  32. return;
  33. }
  34. auto node = selection->anchor_node();
  35. if (!node || !node->is_editable()) {
  36. return;
  37. }
  38. if (!is<DOM::Text>(*node)) {
  39. auto& realm = node->realm();
  40. auto text = realm.create<DOM::Text>(node->document(), data);
  41. MUST(node->append_child(*text));
  42. MUST(selection->collapse(*text, 1));
  43. return;
  44. }
  45. auto& text_node = static_cast<DOM::Text&>(*node);
  46. MUST(selection_range->delete_contents());
  47. MUST(text_node.insert_data(selection->anchor_offset(), data));
  48. VERIFY(selection->is_collapsed());
  49. auto utf16_data = MUST(AK::utf8_to_utf16(data));
  50. Utf16View const utf16_view { utf16_data };
  51. auto length = utf16_view.length_in_code_units();
  52. MUST(selection->collapse(*node, selection->anchor_offset() + length));
  53. text_node.invalidate_style(DOM::StyleInvalidationReason::EditingInsertion);
  54. }
  55. void EditingHostManager::select_all()
  56. {
  57. if (!m_active_contenteditable_element) {
  58. return;
  59. }
  60. auto selection = m_document->get_selection();
  61. if (!selection->anchor_node() || !selection->focus_node()) {
  62. return;
  63. }
  64. MUST(selection->set_base_and_extent(*selection->anchor_node(), 0, *selection->focus_node(), selection->focus_node()->length()));
  65. }
  66. void EditingHostManager::set_selection_anchor(GC::Ref<DOM::Node> anchor_node, size_t anchor_offset)
  67. {
  68. auto selection = m_document->get_selection();
  69. MUST(selection->collapse(*anchor_node, anchor_offset));
  70. m_document->reset_cursor_blink_cycle();
  71. }
  72. void EditingHostManager::set_selection_focus(GC::Ref<DOM::Node> focus_node, size_t focus_offset)
  73. {
  74. if (!m_active_contenteditable_element || !m_active_contenteditable_element->is_ancestor_of(*focus_node))
  75. return;
  76. auto selection = m_document->get_selection();
  77. if (!selection->anchor_node())
  78. return;
  79. MUST(selection->set_base_and_extent(*selection->anchor_node(), selection->anchor_offset(), *focus_node, focus_offset));
  80. m_document->reset_cursor_blink_cycle();
  81. }
  82. void EditingHostManager::move_cursor_to_start(CollapseSelection collapse)
  83. {
  84. auto selection = m_document->get_selection();
  85. auto node = selection->anchor_node();
  86. if (!node || !is<DOM::Text>(*node))
  87. return;
  88. if (collapse == CollapseSelection::Yes) {
  89. MUST(selection->collapse(node, 0));
  90. m_document->reset_cursor_blink_cycle();
  91. return;
  92. }
  93. MUST(selection->set_base_and_extent(*node, selection->anchor_offset(), *node, 0));
  94. }
  95. void EditingHostManager::move_cursor_to_end(CollapseSelection collapse)
  96. {
  97. auto selection = m_document->get_selection();
  98. auto node = selection->anchor_node();
  99. if (!node || !is<DOM::Text>(*node))
  100. return;
  101. if (collapse == CollapseSelection::Yes) {
  102. m_document->reset_cursor_blink_cycle();
  103. MUST(selection->collapse(node, node->length()));
  104. return;
  105. }
  106. MUST(selection->set_base_and_extent(*node, selection->anchor_offset(), *node, node->length()));
  107. }
  108. void EditingHostManager::increment_cursor_position_offset(CollapseSelection collapse)
  109. {
  110. auto selection = m_document->get_selection();
  111. if (!selection)
  112. return;
  113. selection->move_offset_to_next_character(collapse == CollapseSelection::Yes);
  114. }
  115. void EditingHostManager::decrement_cursor_position_offset(CollapseSelection collapse)
  116. {
  117. auto selection = m_document->get_selection();
  118. if (!selection)
  119. return;
  120. selection->move_offset_to_previous_character(collapse == CollapseSelection::Yes);
  121. }
  122. void EditingHostManager::increment_cursor_position_to_next_word(CollapseSelection collapse)
  123. {
  124. auto selection = m_document->get_selection();
  125. if (!selection)
  126. return;
  127. selection->move_offset_to_next_character(collapse == CollapseSelection::Yes);
  128. }
  129. void EditingHostManager::decrement_cursor_position_to_previous_word(CollapseSelection collapse)
  130. {
  131. auto selection = m_document->get_selection();
  132. if (!selection)
  133. return;
  134. selection->move_offset_to_previous_word(collapse == CollapseSelection::Yes);
  135. }
  136. void EditingHostManager::handle_delete(DeleteDirection direction)
  137. {
  138. auto selection = m_document->get_selection();
  139. auto selection_range = selection->range();
  140. if (!selection_range) {
  141. return;
  142. }
  143. if (selection->is_collapsed()) {
  144. auto node = selection->anchor_node();
  145. if (!node || !is<DOM::Text>(*node)) {
  146. return;
  147. }
  148. auto& text_node = static_cast<DOM::Text&>(*node);
  149. if (direction == DeleteDirection::Backward) {
  150. if (selection->anchor_offset() > 0) {
  151. MUST(text_node.delete_data(selection->anchor_offset() - 1, 1));
  152. text_node.invalidate_style(DOM::StyleInvalidationReason::EditingInsertion);
  153. }
  154. } else {
  155. if (selection->anchor_offset() < text_node.data().bytes_as_string_view().length()) {
  156. MUST(text_node.delete_data(selection->anchor_offset(), 1));
  157. text_node.invalidate_style(DOM::StyleInvalidationReason::EditingInsertion);
  158. }
  159. }
  160. m_document->reset_cursor_blink_cycle();
  161. return;
  162. }
  163. MUST(selection_range->delete_contents());
  164. }
  165. void EditingHostManager::handle_return_key()
  166. {
  167. dbgln("FIXME: Implement EditingHostManager::handle_return_key()");
  168. }
  169. }