blob: a2b2002c61294ae95987587a6b76ba5df42212c8 [file] [log] [blame]
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +00001/*
2 * Copyright (C) 2019 Apple Inc. All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
6 * are met:
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 *
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23 * THE POSSIBILITY OF SUCH DAMAGE.
24 */
25
26#include "config.h"
27#include "TextManipulationController.h"
28
wenson_hsieh@apple.come2c4adf2020-04-27 21:12:52 +000029#include "AccessibilityObject.h"
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +000030#include "CharacterData.h"
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +000031#include "Editing.h"
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +000032#include "ElementAncestorIterator.h"
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +000033#include "EventLoop.h"
sihui_liu@apple.com5a2c1d32020-04-23 17:23:54 +000034#include "HTMLBRElement.h"
don.olmstead@sony.comf5e4de72020-03-12 03:19:09 +000035#include "HTMLElement.h"
wenson_hsieh@apple.com5f24d542020-05-02 02:26:23 +000036#include "HTMLInputElement.h"
rniwa@webkit.org640b0732020-03-08 03:52:06 +000037#include "HTMLNames.h"
sihui_liu@apple.com5a2c1d32020-04-23 17:23:54 +000038#include "HTMLParserIdioms.h"
wenson_hsieh@apple.comf0ad7db2020-06-03 21:08:21 +000039#include "InputTypeNames.h"
rniwa@webkit.org900f5602019-12-21 02:09:50 +000040#include "NodeTraversal.h"
41#include "PseudoElement.h"
42#include "Range.h"
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +000043#include "ScriptDisallowedScope.h"
rniwa@webkit.org900f5602019-12-21 02:09:50 +000044#include "Text.h"
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +000045#include "TextIterator.h"
46#include "VisibleUnits.h"
47
48namespace WebCore {
49
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +000050inline bool TextManipulationController::ExclusionRule::match(const Element& element) const
51{
52 return switchOn(rule, [&element] (ElementRule rule) {
53 return rule.localName == element.localName();
54 }, [&element] (AttributeRule rule) {
55 return equalIgnoringASCIICase(element.getAttribute(rule.name), rule.value);
commit-queue@webkit.orgd8730752019-12-04 18:36:15 +000056 }, [&element] (ClassRule rule) {
57 return element.hasClass() && element.classNames().contains(rule.className);
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +000058 });
59}
60
61class ExclusionRuleMatcher {
62public:
63 using ExclusionRule = TextManipulationController::ExclusionRule;
64 using Type = TextManipulationController::ExclusionRule::Type;
65
66 ExclusionRuleMatcher(const Vector<ExclusionRule>& rules)
67 : m_rules(rules)
68 { }
69
70 bool isExcluded(Node* node)
71 {
72 if (!node)
73 return false;
74
75 RefPtr<Element> startingElement = is<Element>(*node) ? downcast<Element>(node) : node->parentElement();
76 if (!startingElement)
77 return false;
78
79 Type type = Type::Include;
80 RefPtr<Element> matchingElement;
darin@apple.com25c0c842020-02-23 04:56:03 +000081 for (auto& element : lineageOfType<Element>(*startingElement)) {
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +000082 if (auto typeOrNullopt = typeForElement(element)) {
83 type = *typeOrNullopt;
84 matchingElement = &element;
85 break;
86 }
87 }
88
darin@apple.com25c0c842020-02-23 04:56:03 +000089 for (auto& element : lineageOfType<Element>(*startingElement)) {
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +000090 m_cache.set(element, type);
91 if (&element == matchingElement)
92 break;
93 }
94
95 return type == Type::Exclude;
96 }
97
98 Optional<Type> typeForElement(Element& element)
99 {
100 auto it = m_cache.find(element);
101 if (it != m_cache.end())
102 return it->value;
103
104 for (auto& rule : m_rules) {
105 if (rule.match(element))
106 return rule.type;
107 }
108
109 return WTF::nullopt;
110 }
111
112private:
113 const Vector<ExclusionRule>& m_rules;
114 HashMap<Ref<Element>, ExclusionRule::Type> m_cache;
115};
116
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000117TextManipulationController::TextManipulationController(Document& document)
118 : m_document(makeWeakPtr(document))
119{
120}
121
rniwa@webkit.org95386982020-02-28 03:37:22 +0000122bool TextManipulationController::isInManipulatedElement(Element& element)
123{
124 if (!m_manipulatedElements.capacity())
125 return false; // Fast path for startObservingParagraphs.
126 for (auto& ancestorOrSelf : lineageOfType<Element>(element)) {
127 if (m_manipulatedElements.contains(ancestorOrSelf))
128 return true;
129 }
130 return false;
131}
132
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +0000133void TextManipulationController::startObservingParagraphs(ManipulationItemCallback&& callback, Vector<ExclusionRule>&& exclusionRules)
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000134{
135 auto document = makeRefPtr(m_document.get());
136 if (!document)
137 return;
138
139 m_callback = WTFMove(callback);
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +0000140 m_exclusionRules = WTFMove(exclusionRules);
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000141
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000142 observeParagraphs(firstPositionInNode(m_document.get()), lastPositionInNode(m_document.get()));
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000143 flushPendingItemsForCallback();
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000144}
145
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000146static bool isInPrivateUseArea(UChar character)
147{
148 return 0xE000 <= character && character <= 0xF8FF;
149}
150
151static bool isTokenDelimiter(UChar character)
152{
153 return isHTMLLineBreak(character) || isInPrivateUseArea(character);
154}
155
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000156class ParagraphContentIterator {
157public:
158 ParagraphContentIterator(const Position& start, const Position& end)
wenson_hsieh@apple.comd0ed2bd2020-04-24 00:07:25 +0000159 : m_iterator({ *makeBoundaryPoint(start), *makeBoundaryPoint(end) }, TextIteratorIgnoresStyleVisibility)
darin@apple.comaf46d1b2020-03-11 04:07:19 +0000160 , m_iteratorNode(m_iterator.atEnd() ? nullptr : createLiveRange(m_iterator.range())->firstNode())
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000161 , m_node(start.firstNode())
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000162 , m_pastEndNode(end.firstNode())
163 {
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000164 if (shouldAdvanceIteratorPastCurrentNode())
165 advanceIteratorNodeAndUpdateText();
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000166 }
167
168 void advance()
169 {
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000170 m_text = WTF::nullopt;
171 advanceNode();
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000172
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000173 if (shouldAdvanceIteratorPastCurrentNode())
174 advanceIteratorNodeAndUpdateText();
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000175 }
176
177 struct CurrentContent {
178 RefPtr<Node> node;
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000179 Vector<String> text;
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000180 bool isTextContent { false };
181 bool isReplacedContent { false };
182 };
183
184 CurrentContent currentContent()
185 {
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000186 CurrentContent content = { m_node.copyRef(), m_text ? m_text.value() : Vector<String> { }, !!m_text };
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000187 if (content.node) {
188 if (auto* renderer = content.node->renderer()) {
189 if (renderer->isRenderReplaced()) {
190 content.isTextContent = false;
191 content.isReplacedContent = true;
192 }
193 }
194 }
195 return content;
196 }
197
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000198 bool atEnd() const { return !m_text && m_iterator.atEnd() && m_node == m_pastEndNode; }
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000199
200private:
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000201 bool shouldAdvanceIteratorPastCurrentNode() const { return !m_iterator.atEnd() && m_iteratorNode == m_node; }
202
203 void advanceNode()
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000204 {
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000205 if (m_node == m_pastEndNode)
sihui_liu@apple.com5a2c1d32020-04-23 17:23:54 +0000206 return;
207
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000208 m_node = NodeTraversal::next(*m_node);
209 if (!m_node)
210 m_node = m_pastEndNode;
211 }
212
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000213 void appendToText(Vector<String>& text, StringBuilder& stringBuilder)
214 {
215 if (!stringBuilder.isEmpty()) {
216 text.append(stringBuilder.toString());
217 stringBuilder.clear();
218 }
219 }
220
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000221 void advanceIteratorNodeAndUpdateText()
222 {
223 ASSERT(shouldAdvanceIteratorPastCurrentNode());
224
225 StringBuilder stringBuilder;
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000226 Vector<String> text;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000227 while (shouldAdvanceIteratorPastCurrentNode()) {
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000228 if (!m_iterator.node()) {
229 auto iteratorText = m_iterator.text();
230 bool containsDelimiter = false;
231 for (unsigned index = 0; index < iteratorText.length() && !containsDelimiter; ++index)
232 containsDelimiter = isTokenDelimiter(iteratorText[index]);
233
234 if (containsDelimiter) {
235 appendToText(text, stringBuilder);
236 text.append({ });
237 }
238 } else
239 stringBuilder.append(m_iterator.text());
240
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000241 m_iterator.advance();
242 m_iteratorNode = m_iterator.atEnd() ? nullptr : createLiveRange(m_iterator.range())->firstNode();
243 }
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000244 appendToText(text, stringBuilder);
245 m_text = text;
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000246 }
247
248 TextIterator m_iterator;
249 RefPtr<Node> m_iteratorNode;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000250 RefPtr<Node> m_node;
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000251 RefPtr<Node> m_pastEndNode;
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000252 Optional<Vector<String>> m_text;
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000253};
254
wenson_hsieh@apple.comf0ad7db2020-06-03 21:08:21 +0000255static bool shouldExtractValueForTextManipulation(const HTMLInputElement& input)
256{
aperez@igalia.comcb5f46712020-06-05 15:47:41 +0000257 if (input.isSearchField() || equalIgnoringASCIICase(input.attributeWithoutSynchronization(HTMLNames::typeAttr), InputTypeNames::text()))
wenson_hsieh@apple.comf0ad7db2020-06-03 21:08:21 +0000258 return !input.lastChangeWasUserEdit();
259
260 return input.isTextButton();
261}
262
263static bool isAttributeForTextManipulation(const QualifiedName& nameToCheck)
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000264{
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000265 using namespace HTMLNames;
266 static const QualifiedName* const attributeNames[] = {
267 &titleAttr.get(),
268 &altAttr.get(),
269 &placeholderAttr.get(),
270 &aria_labelAttr.get(),
271 &aria_placeholderAttr.get(),
272 &aria_roledescriptionAttr.get(),
273 &aria_valuetextAttr.get(),
274 };
275 for (auto& entry : attributeNames) {
276 if (*entry == nameToCheck)
277 return true;
278 }
279 return false;
280}
281
wenson_hsieh@apple.com2d16b752020-04-20 20:19:54 +0000282static bool canPerformTextManipulationByReplacingEntireTextContent(const Element& element)
283{
284 return element.hasTagName(HTMLNames::titleTag) || element.hasTagName(HTMLNames::optionTag);
285}
286
wenson_hsieh@apple.comeeb352d2020-04-29 05:16:23 +0000287static Optional<TextManipulationController::ManipulationTokenInfo> tokenInfo(Node* node)
288{
289 if (!node)
290 return WTF::nullopt;
291
292 TextManipulationController::ManipulationTokenInfo result;
293 result.documentURL = node->document().url();
294 if (auto element = is<Element>(node) ? makeRefPtr(downcast<Element>(*node)) : makeRefPtr(node->parentElement())) {
295 result.tagName = element->tagName();
296 if (element->hasAttributeWithoutSynchronization(HTMLNames::roleAttr))
297 result.roleAttribute = element->attributeWithoutSynchronization(HTMLNames::roleAttr);
298 }
299 return result;
300}
301
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000302static bool isEnclosingItemBoundaryElement(const Element& element)
303{
304 auto* renderer = element.renderer();
305 if (!renderer)
306 return false;
307
308 auto role = [](const Element& element) -> AccessibilityRole {
309 return AccessibilityObject::ariaRoleToWebCoreRole(element.attributeWithoutSynchronization(HTMLNames::roleAttr));
310 };
311
312 if (element.hasTagName(HTMLNames::buttonTag) || role(element) == AccessibilityRole::Button)
313 return true;
314
315 if (element.hasTagName(HTMLNames::liTag) || element.hasTagName(HTMLNames::aTag)) {
316 auto displayType = renderer->style().display();
317 if (displayType == DisplayType::Block || displayType == DisplayType::InlineBlock)
318 return true;
319
320 for (auto parent = makeRefPtr(element.parentElement()); parent; parent = parent->parentElement()) {
321 if (parent->hasTagName(HTMLNames::navTag) || role(*parent) == AccessibilityRole::LandmarkNavigation)
322 return true;
323 }
324 }
325
326 return false;
327}
328
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000329TextManipulationController::ManipulationUnit TextManipulationController::createUnit(const Vector<String>& text, Node& textNode)
wenson_hsieh@apple.com600d5b62020-06-05 19:59:00 +0000330{
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000331 ManipulationUnit unit = { textNode, { } };
332 for (auto& textEntry : text) {
333 if (!textEntry.isNull())
334 parse(unit, textEntry, textNode);
335 else {
336 if (unit.tokens.isEmpty())
337 unit.firstTokenContainsDelimiter = true;
338 unit.lastTokenContainsDelimiter = true;
339 }
340 }
341 return unit;
wenson_hsieh@apple.com600d5b62020-06-05 19:59:00 +0000342}
343
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000344void TextManipulationController::parse(ManipulationUnit& unit, const String& text, Node& textNode)
wenson_hsieh@apple.com600d5b62020-06-05 19:59:00 +0000345{
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000346 ExclusionRuleMatcher exclusionRuleMatcher(m_exclusionRules);
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000347 bool isNodeExcluded = exclusionRuleMatcher.isExcluded(&textNode);
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000348 size_t positionOfLastNonHTMLSpace = WTF::notFound;
349 size_t startPositionOfCurrentToken = 0;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000350 size_t index = 0;
351 for (; index < text.length(); ++index) {
352 auto character = text[index];
wenson_hsieh@apple.com600d5b62020-06-05 19:59:00 +0000353 if (isTokenDelimiter(character)) {
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000354 if (positionOfLastNonHTMLSpace != WTF::notFound && startPositionOfCurrentToken <= positionOfLastNonHTMLSpace) {
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000355 auto stringForToken = text.substring(startPositionOfCurrentToken, positionOfLastNonHTMLSpace + 1 - startPositionOfCurrentToken);
356 unit.tokens.append(ManipulationToken { m_tokenIdentifier.generate(), stringForToken, tokenInfo(&textNode), isNodeExcluded });
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000357 startPositionOfCurrentToken = positionOfLastNonHTMLSpace + 1;
358 }
359
wenson_hsieh@apple.com600d5b62020-06-05 19:59:00 +0000360 while (index < text.length() && (isHTMLSpace(text[index]) || isInPrivateUseArea(text[index])))
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000361 ++index;
362
363 --index;
364
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000365 auto stringForToken = text.substring(startPositionOfCurrentToken, index + 1 - startPositionOfCurrentToken);
366 if (unit.tokens.isEmpty() && !unit.firstTokenContainsDelimiter)
367 unit.firstTokenContainsDelimiter = true;
368 unit.tokens.append(ManipulationToken { m_tokenIdentifier.generate(), stringForToken, tokenInfo(&textNode), true });
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000369 startPositionOfCurrentToken = index + 1;
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000370 unit.lastTokenContainsDelimiter = true;
wenson_hsieh@apple.com600d5b62020-06-05 19:59:00 +0000371 } else if (isNotHTMLSpace(character)) {
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000372 if (!isNodeExcluded)
373 unit.areAllTokensExcluded = false;
wenson_hsieh@apple.com600d5b62020-06-05 19:59:00 +0000374 positionOfLastNonHTMLSpace = index;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000375 }
376 }
377
378 if (startPositionOfCurrentToken < text.length()) {
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000379 auto stringForToken = text.substring(startPositionOfCurrentToken, index + 1 - startPositionOfCurrentToken);
380 unit.tokens.append(ManipulationToken { m_tokenIdentifier.generate(), stringForToken, tokenInfo(&textNode), isNodeExcluded });
381 unit.lastTokenContainsDelimiter = false;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000382 }
sihui_liu@apple.com33d45222020-06-05 06:13:30 +0000383}
384
385void TextManipulationController::addItemIfPossible(Vector<ManipulationUnit>&& units)
386{
387 if (units.isEmpty())
388 return;
389
390 size_t index = 0;
391 size_t end = units.size();
392 while (index < units.size() && units[index].areAllTokensExcluded)
393 ++index;
394
395 while (end > 0 && units[end - 1].areAllTokensExcluded)
396 --end;
397
398 if (index == end)
399 return;
400
401 auto startPosition = firstPositionInOrBeforeNode(units.first().node.ptr());
402 auto endPosition = positionAfterNode(units.last().node.ptr());
403 Vector<ManipulationToken> tokens;
404 for (; index < end; ++index)
405 tokens.appendVector(WTFMove(units[index].tokens));
406
407 addItem(ManipulationItemData { startPosition, endPosition, nullptr, nullQName(), WTFMove(tokens) });
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000408}
409
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000410void TextManipulationController::observeParagraphs(const Position& start, const Position& end)
411{
wenson_hsieh@apple.comc626f5c2020-04-08 23:59:31 +0000412 if (start.isNull() || end.isNull())
413 return;
414
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000415 auto document = makeRefPtr(start.document());
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000416 ASSERT(document);
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000417 // TextIterator's constructor may have updated the layout and executed arbitrary scripts.
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000418 if (document != start.document() || document != end.document())
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000419 return;
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000420
sihui_liu@apple.com33d45222020-06-05 06:13:30 +0000421 Vector<ManipulationUnit> unitsInCurrentParagraph;
wenson_hsieh@apple.comf9a72332020-04-24 23:18:05 +0000422 RefPtr<Element> enclosingItemBoundaryElement;
sihui_liu@apple.com33d45222020-06-05 06:13:30 +0000423 ParagraphContentIterator iterator { start, end };
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000424 for (; !iterator.atEnd(); iterator.advance()) {
425 auto content = iterator.currentContent();
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000426 auto* contentNode = content.node.get();
427 ASSERT(contentNode);
wenson_hsieh@apple.comf9a72332020-04-24 23:18:05 +0000428
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000429 if (enclosingItemBoundaryElement && !enclosingItemBoundaryElement->contains(contentNode)) {
sihui_liu@apple.com33d45222020-06-05 06:13:30 +0000430 addItemIfPossible(std::exchange(unitsInCurrentParagraph, { }));
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000431 enclosingItemBoundaryElement = nullptr;
432 }
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000433
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000434 if (RefPtr<Element> currentElementAncestor = is<Element>(*contentNode) ? downcast<Element>(contentNode) : contentNode->parentOrShadowHostElement()) {
435 // We can exit early here because scheduleObservationUpdate calls this function on each paragraph separately.
436 if (isInManipulatedElement(*currentElementAncestor))
437 return;
438 }
439
440 if (is<Element>(*contentNode)) {
441 auto& currentElement = downcast<Element>(*contentNode);
442 if (!content.isTextContent && canPerformTextManipulationByReplacingEntireTextContent(currentElement))
443 addItem(ManipulationItemData { Position(), Position(), makeWeakPtr(currentElement), nullQName(), { ManipulationToken { m_tokenIdentifier.generate(), currentElement.textContent(), tokenInfo(&currentElement) } } });
444
445 if (currentElement.hasAttributes()) {
446 for (auto& attribute : currentElement.attributesIterator()) {
wenson_hsieh@apple.comf0ad7db2020-06-03 21:08:21 +0000447 if (isAttributeForTextManipulation(attribute.name()))
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000448 addItem(ManipulationItemData { Position(), Position(), makeWeakPtr(currentElement), attribute.name(), { ManipulationToken { m_tokenIdentifier.generate(), attribute.value(), tokenInfo(&currentElement) } } });
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000449 }
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000450 }
wenson_hsieh@apple.comf0ad7db2020-06-03 21:08:21 +0000451
452 if (is<HTMLInputElement>(currentElement)) {
453 auto& input = downcast<HTMLInputElement>(currentElement);
454 if (shouldExtractValueForTextManipulation(input))
aperez@igalia.comcb5f46712020-06-05 15:47:41 +0000455 addItem(ManipulationItemData { { }, { }, makeWeakPtr(currentElement), HTMLNames::valueAttr, { ManipulationToken { m_tokenIdentifier.generate(), input.value(), tokenInfo(&currentElement) } } });
wenson_hsieh@apple.comf0ad7db2020-06-03 21:08:21 +0000456 }
457
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000458 if (!enclosingItemBoundaryElement && isEnclosingItemBoundaryElement(currentElement))
459 enclosingItemBoundaryElement = &currentElement;
rniwa@webkit.org95386982020-02-28 03:37:22 +0000460 }
461
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000462 if (content.isReplacedContent) {
sihui_liu@apple.com33d45222020-06-05 06:13:30 +0000463 if (!unitsInCurrentParagraph.isEmpty())
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000464 unitsInCurrentParagraph.append(ManipulationUnit { *contentNode, { ManipulationToken { m_tokenIdentifier.generate(), "[]", tokenInfo(content.node.get()), true } } });
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000465 continue;
466 }
467
468 if (!content.isTextContent)
469 continue;
470
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000471 auto currentUnit = createUnit(content.text, *contentNode);
472 if (currentUnit.firstTokenContainsDelimiter)
sihui_liu@apple.com33d45222020-06-05 06:13:30 +0000473 addItemIfPossible(std::exchange(unitsInCurrentParagraph, { }));
sihui_liu@apple.com5a2c1d32020-04-23 17:23:54 +0000474
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000475 if (unitsInCurrentParagraph.isEmpty() && currentUnit.areAllTokensExcluded)
476 continue;
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000477
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000478 bool currentUnitEndsWithDelimiter = currentUnit.lastTokenContainsDelimiter;
479 unitsInCurrentParagraph.append(WTFMove(currentUnit));
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000480
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000481 if (currentUnitEndsWithDelimiter)
sihui_liu@apple.com33d45222020-06-05 06:13:30 +0000482 addItemIfPossible(std::exchange(unitsInCurrentParagraph, { }));
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000483 }
484
sihui_liu@apple.com33d45222020-06-05 06:13:30 +0000485 addItemIfPossible(std::exchange(unitsInCurrentParagraph, { }));
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000486}
487
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000488void TextManipulationController::didCreateRendererForElement(Element& element)
489{
rniwa@webkit.org95386982020-02-28 03:37:22 +0000490 if (isInManipulatedElement(element))
wenson_hsieh@apple.com54632b72020-01-10 00:40:14 +0000491 return;
492
rniwa@webkit.org95386982020-02-28 03:37:22 +0000493 if (m_elementsWithNewRenderer.computesEmpty())
wenson_hsieh@apple.comd0ed2bd2020-04-24 00:07:25 +0000494 scheduleObservationUpdate();
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000495
496 if (is<PseudoElement>(element)) {
497 if (auto* host = downcast<PseudoElement>(element).hostElement())
rniwa@webkit.org95386982020-02-28 03:37:22 +0000498 m_elementsWithNewRenderer.add(*host);
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000499 } else
rniwa@webkit.org95386982020-02-28 03:37:22 +0000500 m_elementsWithNewRenderer.add(element);
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000501}
502
503using PositionTuple = std::tuple<RefPtr<Node>, unsigned, unsigned>;
504static const PositionTuple makePositionTuple(const Position& position)
505{
aperez@igalia.comcb5f46712020-06-05 15:47:41 +0000506 return { position.anchorNode(), static_cast<unsigned>(position.anchorType()), position.anchorType() == Position::PositionIsOffsetInAnchor ? position.offsetInContainerNode() : 0 };
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000507}
508
509static const std::pair<PositionTuple, PositionTuple> makeHashablePositionRange(const VisiblePosition& start, const VisiblePosition& end)
510{
511 return { makePositionTuple(start.deepEquivalent()), makePositionTuple(end.deepEquivalent()) };
512}
513
wenson_hsieh@apple.comd0ed2bd2020-04-24 00:07:25 +0000514void TextManipulationController::scheduleObservationUpdate()
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000515{
516 if (!m_document)
517 return;
518
519 m_document->eventLoop().queueTask(TaskSource::InternalAsyncTask, [weakThis = makeWeakPtr(*this)] {
520 auto* controller = weakThis.get();
521 if (!controller)
522 return;
523
524 HashSet<Ref<Element>> mutatedElements;
rniwa@webkit.org95386982020-02-28 03:37:22 +0000525 for (auto& weakElement : controller->m_elementsWithNewRenderer)
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000526 mutatedElements.add(weakElement);
rniwa@webkit.org95386982020-02-28 03:37:22 +0000527 controller->m_elementsWithNewRenderer.clear();
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000528
529 HashSet<Ref<Element>> filteredElements;
530 for (auto& element : mutatedElements) {
531 auto* parentElement = element->parentElement();
532 if (!parentElement || !mutatedElements.contains(parentElement))
533 filteredElements.add(element.copyRef());
534 }
535 mutatedElements.clear();
536
537 HashSet<std::pair<PositionTuple, PositionTuple>> paragraphSets;
538 for (auto& element : filteredElements) {
539 auto start = startOfParagraph(firstPositionInOrBeforeNode(element.ptr()));
540 auto end = endOfParagraph(lastPositionInOrAfterNode(element.ptr()));
541
wenson_hsieh@apple.com6b6c8fb2020-04-09 20:16:38 +0000542 if (start.isNull() || end.isNull())
543 continue;
544
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000545 auto key = makeHashablePositionRange(start, end);
546 if (!paragraphSets.add(key).isNewEntry)
547 continue;
548
549 auto* controller = weakThis.get();
550 if (!controller)
551 return; // Finding the start/end of paragraph may have updated layout & executed arbitrary scripts.
552
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000553 controller->observeParagraphs(start.deepEquivalent(), end.deepEquivalent());
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000554 }
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000555 controller->flushPendingItemsForCallback();
rniwa@webkit.orgc46d21d2019-12-16 22:36:55 +0000556 });
557}
558
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000559void TextManipulationController::addItem(ManipulationItemData&& itemData)
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000560{
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000561 const unsigned itemCallbackBatchingSize = 128;
562
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000563 ASSERT(m_document);
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000564 ASSERT(!itemData.tokens.isEmpty());
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000565 auto newID = m_itemIdentifier.generate();
566 m_pendingItemsForCallback.append(ManipulationItem {
567 newID,
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000568 itemData.tokens.map([](auto& token) { return token; })
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000569 });
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000570 m_items.add(newID, WTFMove(itemData));
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000571
572 if (m_pendingItemsForCallback.size() >= itemCallbackBatchingSize)
573 flushPendingItemsForCallback();
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000574}
575
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000576void TextManipulationController::flushPendingItemsForCallback()
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000577{
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000578 m_callback(*m_document, m_pendingItemsForCallback);
579 m_pendingItemsForCallback.clear();
580}
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000581
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000582auto TextManipulationController::completeManipulation(const Vector<WebCore::TextManipulationController::ManipulationItem>& completionItems) -> Vector<ManipulationFailure>
583{
584 Vector<ManipulationFailure> failures;
585 for (unsigned i = 0; i < completionItems.size(); ++i) {
586 auto& itemToComplete = completionItems[i];
587 auto identifier = itemToComplete.identifier;
588 if (!identifier) {
589 failures.append(ManipulationFailure { identifier, i, ManipulationFailureType::InvalidItem });
590 continue;
591 }
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000592
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000593 auto itemDataIterator = m_items.find(identifier);
594 if (itemDataIterator == m_items.end()) {
595 failures.append(ManipulationFailure { identifier, i, ManipulationFailureType::InvalidItem });
596 continue;
597 }
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000598
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000599 ManipulationItemData itemData;
600 std::exchange(itemData, itemDataIterator->value);
601 m_items.remove(itemDataIterator);
602
603 auto failureOrNullopt = replace(itemData, itemToComplete.tokens);
604 if (failureOrNullopt)
605 failures.append(ManipulationFailure { identifier, i, *failureOrNullopt });
606 }
607 return failures;
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000608}
609
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000610struct TokenExchangeData {
611 RefPtr<Node> node;
612 String originalContent;
613 bool isExcluded { false };
614 bool isConsumed { false };
615};
616
617struct ReplacementData {
618 Ref<Node> originalNode;
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +0000619 String newData;
620};
621
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000622Vector<Ref<Node>> TextManipulationController::getPath(Node* ancestor, Node* node)
623{
624 Vector<Ref<Node>> path;
625 RefPtr<ContainerNode> containerNode = is<ContainerNode>(*node) ? &downcast<ContainerNode>(*node) : node->parentNode();
626 for (; containerNode && containerNode != ancestor; containerNode = containerNode->parentNode())
627 path.append(*containerNode);
628 path.reverse();
629 return path;
630}
631
632void TextManipulationController::updateInsertions(Vector<NodeEntry>& lastTopDownPath, const Vector<Ref<Node>>& currentTopDownPath, Node* currentNode, HashSet<Ref<Node>>& insertedNodes, Vector<NodeInsertion>& insertions)
633{
634 size_t i =0;
635 while (i < lastTopDownPath.size() && i < currentTopDownPath.size() && lastTopDownPath[i].first.ptr() == currentTopDownPath[i].ptr())
636 ++i;
637
638 if (i != lastTopDownPath.size() || i != currentTopDownPath.size()) {
639 if (i < lastTopDownPath.size())
640 lastTopDownPath.shrink(i);
641
642 for (;i < currentTopDownPath.size(); ++i) {
643 Ref<Node> node = currentTopDownPath[i];
644 if (!insertedNodes.add(node.copyRef()).isNewEntry) {
645 auto clonedNode = node->cloneNodeInternal(node->document(), Node::CloningOperation::OnlySelf);
646 if (auto* data = node->eventTargetData())
647 data->eventListenerMap.copyEventListenersNotCreatedFromMarkupToTarget(clonedNode.ptr());
648 node = WTFMove(clonedNode);
649 }
650 insertions.append(NodeInsertion { lastTopDownPath.size() ? lastTopDownPath.last().second.ptr() : nullptr, node.copyRef() });
651 lastTopDownPath.append({ currentTopDownPath[i].copyRef(), WTFMove(node) });
652 }
653 }
654
655 if (currentNode)
656 insertions.append(NodeInsertion { lastTopDownPath.size() ? lastTopDownPath.last().second.ptr() : nullptr, *currentNode });
657}
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000658
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000659auto TextManipulationController::replace(const ManipulationItemData& item, const Vector<ManipulationToken>& replacementTokens) -> Optional<ManipulationFailureType>
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000660{
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000661 if (item.start.isOrphan() || item.end.isOrphan())
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000662 return ManipulationFailureType::ContentChanged;
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000663
darin@apple.comd2f1a002020-03-16 23:02:10 +0000664 if (item.start.isNull() || item.end.isNull()) {
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000665 RELEASE_ASSERT(item.tokens.size() == 1);
666 auto element = makeRefPtr(item.element.get());
667 if (!element)
668 return ManipulationFailureType::ContentChanged;
sihui_liu@apple.com33fbdee2020-06-03 19:04:28 +0000669 if (replacementTokens.size() > 1 && !canPerformTextManipulationByReplacingEntireTextContent(*element) && item.attributeName == nullQName())
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000670 return ManipulationFailureType::InvalidToken;
wenson_hsieh@apple.com2d16b752020-04-20 20:19:54 +0000671 auto expectedTokenIdentifier = item.tokens[0].identifier;
672 StringBuilder newValue;
673 for (size_t i = 0; i < replacementTokens.size(); ++i) {
674 if (replacementTokens[i].identifier != expectedTokenIdentifier)
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000675 return ManipulationFailureType::InvalidToken;
wenson_hsieh@apple.com2d16b752020-04-20 20:19:54 +0000676 if (i)
677 newValue.append(' ');
678 newValue.append(replacementTokens[i].content);
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000679 }
680 if (item.attributeName == nullQName())
wenson_hsieh@apple.com2d16b752020-04-20 20:19:54 +0000681 element->setTextContent(newValue.toString());
aperez@igalia.comcb5f46712020-06-05 15:47:41 +0000682 else if (item.attributeName == HTMLNames::valueAttr && is<HTMLInputElement>(*element))
wenson_hsieh@apple.comf0ad7db2020-06-03 21:08:21 +0000683 downcast<HTMLInputElement>(*element).setValue(newValue.toString());
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000684 else
wenson_hsieh@apple.com2d16b752020-04-20 20:19:54 +0000685 element->setAttribute(item.attributeName, newValue.toString());
rniwa@webkit.org640b0732020-03-08 03:52:06 +0000686 return WTF::nullopt;
687 }
688
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000689 size_t currentTokenIndex = 0;
690 HashMap<TokenIdentifier, TokenExchangeData> tokenExchangeMap;
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000691 RefPtr<Node> commonAncestor;
rniwa@webkit.orgbf66d0a2020-03-12 23:29:31 +0000692 RefPtr<Node> firstContentNode;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000693 RefPtr<Node> lastChildOfCommonAncestorInRange;
rniwa@webkit.orgbf66d0a2020-03-12 23:29:31 +0000694 HashSet<Ref<Node>> nodesToRemove;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000695 ParagraphContentIterator iterator { item.start, item.end };
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000696 for (; !iterator.atEnd(); iterator.advance()) {
697 auto content = iterator.currentContent();
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000698 ASSERT(content.node);
699
700 lastChildOfCommonAncestorInRange = content.node;
701 nodesToRemove.add(*content.node);
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000702
703 if (!content.isReplacedContent && !content.isTextContent)
704 continue;
705
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000706 Vector<ManipulationToken> tokensInCurrentNode;
sihui_liu@apple.com5f91c3b2020-06-04 23:58:32 +0000707 if (content.isReplacedContent) {
708 if (currentTokenIndex >= item.tokens.size())
709 return ManipulationFailureType::ContentChanged;
710
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000711 tokensInCurrentNode.append(item.tokens[currentTokenIndex]);
sihui_liu@apple.com5f91c3b2020-06-04 23:58:32 +0000712 } else
sihui_liu@apple.com0e39ec02020-06-09 06:54:20 +0000713 tokensInCurrentNode = createUnit(content.text, *content.node).tokens;
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000714
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000715 bool isNodeIncluded = WTF::anyOf(tokensInCurrentNode, [] (auto& token) {
716 return !token.isExcluded;
717 });
718 for (auto& token : tokensInCurrentNode) {
sihui_liu@apple.com5f91c3b2020-06-04 23:58:32 +0000719 if (currentTokenIndex >= item.tokens.size())
sihui_liu@apple.com5a2c1d32020-04-23 17:23:54 +0000720 return ManipulationFailureType::ContentChanged;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000721
722 auto& currentToken = item.tokens[currentTokenIndex++];
723 if (!content.isReplacedContent && currentToken.content != token.content)
724 return ManipulationFailureType::ContentChanged;
725
726 tokenExchangeMap.set(currentToken.identifier, TokenExchangeData { content.node.copyRef(), currentToken.content, !isNodeIncluded });
sihui_liu@apple.com5a2c1d32020-04-23 17:23:54 +0000727 }
728
rniwa@webkit.orgbf66d0a2020-03-12 23:29:31 +0000729 if (!firstContentNode)
730 firstContentNode = content.node;
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000731
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000732 auto parentNode = content.node->parentNode();
733 if (!commonAncestor)
734 commonAncestor = parentNode;
735 else if (!parentNode->isDescendantOf(commonAncestor.get())) {
736 commonAncestor = commonInclusiveAncestor(*commonAncestor, *parentNode);
737 ASSERT(commonAncestor);
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000738 }
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000739 }
740
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000741 while (lastChildOfCommonAncestorInRange && lastChildOfCommonAncestorInRange->parentNode() != commonAncestor)
742 lastChildOfCommonAncestorInRange = lastChildOfCommonAncestorInRange->parentNode();
743
rniwa@webkit.orgbf66d0a2020-03-12 23:29:31 +0000744 for (auto node = commonAncestor; node; node = node->parentNode())
745 nodesToRemove.remove(*node);
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000746
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000747 HashSet<Ref<Node>> reusedOriginalNodes;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000748 Vector<NodeEntry> lastTopDownPath;
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000749 Vector<NodeInsertion> insertions;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000750 for (size_t index = 0; index < replacementTokens.size(); ++index) {
751 auto& replacementToken = replacementTokens[index];
752 auto it = tokenExchangeMap.find(replacementToken.identifier);
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000753 if (it == tokenExchangeMap.end())
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000754 return ManipulationFailureType::InvalidToken;
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000755
756 auto& exchangeData = it->value;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000757 auto* originalNode = exchangeData.node.get();
758 ASSERT(originalNode);
759 auto replacementText = replacementToken.content;
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000760
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000761 RefPtr<Node> replacementNode;
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000762 if (exchangeData.isExcluded) {
763 if (exchangeData.isConsumed)
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000764 return ManipulationFailureType::ExclusionViolation;
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000765 exchangeData.isConsumed = true;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000766
767 if (!replacementToken.content.isNull() && replacementToken.content != exchangeData.originalContent)
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000768 return ManipulationFailureType::ExclusionViolation;
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000769
770 replacementNode = originalNode;
771 for (RefPtr<Node> descendentNode = NodeTraversal::next(*originalNode, originalNode); descendentNode; descendentNode = NodeTraversal::next(*descendentNode, originalNode))
772 nodesToRemove.remove(*descendentNode);
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000773 } else
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000774 replacementNode = Text::create(commonAncestor->document(), replacementText);
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000775
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000776 auto topDownPath = getPath(commonAncestor.get(), originalNode);
777 updateInsertions(lastTopDownPath, topDownPath, replacementNode.get(), reusedOriginalNodes, insertions);
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000778 }
sihui_liu@apple.com12ab7a72020-06-01 23:54:34 +0000779
780 auto end = lastChildOfCommonAncestorInRange ? positionInParentAfterNode(lastChildOfCommonAncestorInRange.get()) : positionAfterNode(commonAncestor.get());
781 RefPtr<Node> node = item.end.firstNode();
782 RefPtr<Node> endNode = end.firstNode();
783 if (node && node != endNode) {
784 auto topDownPath = getPath(commonAncestor.get(), node->parentNode());
785 updateInsertions(lastTopDownPath, topDownPath, nullptr, reusedOriginalNodes, insertions);
786 }
787 while (node != endNode) {
788 Ref<Node> parentNode = *node->parentNode();
789 while (!lastTopDownPath.isEmpty() && lastTopDownPath.last().first.ptr() != parentNode.ptr())
790 lastTopDownPath.removeLast();
791
792 insertions.append(NodeInsertion { lastTopDownPath.size() ? lastTopDownPath.last().second.ptr() : nullptr, *node });
793 lastTopDownPath.append({ *node, *node });
794 node = NodeTraversal::next(*node);
795 }
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000796
rniwa@webkit.orgbf66d0a2020-03-12 23:29:31 +0000797 Position insertionPoint = positionBeforeNode(firstContentNode.get()).parentAnchoredEquivalent();
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000798 while (insertionPoint.containerNode() != commonAncestor)
799 insertionPoint = positionInParentBeforeNode(insertionPoint.containerNode());
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000800
801 for (auto& node : nodesToRemove)
802 node->remove();
803
804 for (auto& insertion : insertions) {
rniwa@webkit.orgcffb7892020-03-07 01:14:23 +0000805 if (!insertion.parentIfDifferentFromCommonAncestor) {
806 insertionPoint.containerNode()->insertBefore(insertion.child, insertionPoint.computeNodeAfterPosition());
807 insertionPoint = positionInParentAfterNode(insertion.child.ptr());
808 } else
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000809 insertion.parentIfDifferentFromCommonAncestor->appendChild(insertion.child);
wenson_hsieh@apple.com54632b72020-01-10 00:40:14 +0000810 if (is<Element>(insertion.child.get()))
rniwa@webkit.org95386982020-02-28 03:37:22 +0000811 m_manipulatedElements.add(downcast<Element>(insertion.child.get()));
rniwa@webkit.org900f5602019-12-21 02:09:50 +0000812 }
rniwa@webkit.org7afdb0b2019-10-25 20:41:15 +0000813
rniwa@webkit.orge44b91352020-03-04 05:22:14 +0000814 return WTF::nullopt;
rniwa@webkit.orge9da7a82019-10-25 00:08:13 +0000815}
816
817} // namespace WebCore