Index: Source/core/editing/CompositeEditCommand.cpp |
diff --git a/Source/core/editing/CompositeEditCommand.cpp b/Source/core/editing/CompositeEditCommand.cpp |
deleted file mode 100644 |
index ca322e7fb3e2b3d7d8a5a9b55f7be7ca106b8749..0000000000000000000000000000000000000000 |
--- a/Source/core/editing/CompositeEditCommand.cpp |
+++ /dev/null |
@@ -1,1511 +0,0 @@ |
-/* |
- * Copyright (C) 2005, 2006, 2007, 2008 Apple Inc. All rights reserved. |
- * |
- * Redistribution and use in source and binary forms, with or without |
- * modification, are permitted provided that the following conditions |
- * are met: |
- * 1. Redistributions of source code must retain the above copyright |
- * notice, this list of conditions and the following disclaimer. |
- * 2. Redistributions in binary form must reproduce the above copyright |
- * notice, this list of conditions and the following disclaimer in the |
- * documentation and/or other materials provided with the distribution. |
- * |
- * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY |
- * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
- * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR |
- * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
- * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
- * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
- * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
- * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
- */ |
- |
-#include "config.h" |
-#include "core/editing/CompositeEditCommand.h" |
- |
-#include "bindings/core/v8/ExceptionStatePlaceholder.h" |
-#include "core/HTMLNames.h" |
-#include "core/dom/Document.h" |
-#include "core/dom/DocumentFragment.h" |
-#include "core/dom/ElementTraversal.h" |
-#include "core/dom/NodeTraversal.h" |
-#include "core/dom/Range.h" |
-#include "core/dom/Text.h" |
-#include "core/editing/AppendNodeCommand.h" |
-#include "core/editing/ApplyStyleCommand.h" |
-#include "core/editing/DeleteFromTextNodeCommand.h" |
-#include "core/editing/DeleteSelectionCommand.h" |
-#include "core/editing/EditingUtilities.h" |
-#include "core/editing/Editor.h" |
-#include "core/editing/InsertIntoTextNodeCommand.h" |
-#include "core/editing/InsertLineBreakCommand.h" |
-#include "core/editing/InsertNodeBeforeCommand.h" |
-#include "core/editing/InsertParagraphSeparatorCommand.h" |
-#include "core/editing/MergeIdenticalElementsCommand.h" |
-#include "core/editing/PlainTextRange.h" |
-#include "core/editing/RemoveCSSPropertyCommand.h" |
-#include "core/editing/RemoveNodeCommand.h" |
-#include "core/editing/RemoveNodePreservingChildrenCommand.h" |
-#include "core/editing/ReplaceNodeWithSpanCommand.h" |
-#include "core/editing/ReplaceSelectionCommand.h" |
-#include "core/editing/SetNodeAttributeCommand.h" |
-#include "core/editing/SplitElementCommand.h" |
-#include "core/editing/SplitTextNodeCommand.h" |
-#include "core/editing/SplitTextNodeContainingElementCommand.h" |
-#include "core/editing/VisibleUnits.h" |
-#include "core/editing/WrapContentsInDummySpanCommand.h" |
-#include "core/editing/iterators/TextIterator.h" |
-#include "core/editing/markers/DocumentMarkerController.h" |
-#include "core/editing/serializers/Serialization.h" |
-#include "core/editing/spellcheck/SpellChecker.h" |
-#include "core/events/ScopedEventQueue.h" |
-#include "core/frame/LocalFrame.h" |
-#include "core/html/HTMLBRElement.h" |
-#include "core/html/HTMLDivElement.h" |
-#include "core/html/HTMLElement.h" |
-#include "core/html/HTMLLIElement.h" |
-#include "core/html/HTMLQuoteElement.h" |
-#include "core/html/HTMLSpanElement.h" |
-#include "core/layout/LayoutBlock.h" |
-#include "core/layout/LayoutListItem.h" |
-#include "core/layout/LayoutText.h" |
-#include "core/layout/line/InlineTextBox.h" |
- |
-namespace blink { |
- |
-using namespace HTMLNames; |
- |
-PassRefPtrWillBeRawPtr<EditCommandComposition> EditCommandComposition::create(Document* document, |
- const VisibleSelection& startingSelection, const VisibleSelection& endingSelection, EditAction editAction) |
-{ |
- return adoptRefWillBeNoop(new EditCommandComposition(document, startingSelection, endingSelection, editAction)); |
-} |
- |
-EditCommandComposition::EditCommandComposition(Document* document, const VisibleSelection& startingSelection, const VisibleSelection& endingSelection, EditAction editAction) |
- : m_document(document) |
- , m_startingSelection(startingSelection) |
- , m_endingSelection(endingSelection) |
- , m_startingRootEditableElement(startingSelection.rootEditableElement()) |
- , m_endingRootEditableElement(endingSelection.rootEditableElement()) |
- , m_editAction(editAction) |
-{ |
-} |
- |
-bool EditCommandComposition::belongsTo(const LocalFrame& frame) const |
-{ |
- ASSERT(m_document); |
- return m_document->frame() == &frame; |
-} |
- |
-void EditCommandComposition::unapply() |
-{ |
- ASSERT(m_document); |
- RefPtrWillBeRawPtr<LocalFrame> frame = m_document->frame(); |
- ASSERT(frame); |
- |
- // Changes to the document may have been made since the last editing operation that require a layout, as in <rdar://problem/5658603>. |
- // Low level operations, like RemoveNodeCommand, don't require a layout because the high level operations that use them perform one |
- // if one is necessary (like for the creation of VisiblePositions). |
- m_document->updateLayoutIgnorePendingStylesheets(); |
- |
- { |
- size_t size = m_commands.size(); |
- for (size_t i = size; i; --i) |
- m_commands[i - 1]->doUnapply(); |
- } |
- |
- frame->editor().unappliedEditing(this); |
-} |
- |
-void EditCommandComposition::reapply() |
-{ |
- ASSERT(m_document); |
- RefPtrWillBeRawPtr<LocalFrame> frame = m_document->frame(); |
- ASSERT(frame); |
- |
- // Changes to the document may have been made since the last editing operation that require a layout, as in <rdar://problem/5658603>. |
- // Low level operations, like RemoveNodeCommand, don't require a layout because the high level operations that use them perform one |
- // if one is necessary (like for the creation of VisiblePositions). |
- m_document->updateLayoutIgnorePendingStylesheets(); |
- |
- { |
- for (const auto& command : m_commands) |
- command->doReapply(); |
- } |
- |
- frame->editor().reappliedEditing(this); |
-} |
- |
-void EditCommandComposition::append(SimpleEditCommand* command) |
-{ |
- m_commands.append(command); |
-} |
- |
-void EditCommandComposition::setStartingSelection(const VisibleSelection& selection) |
-{ |
- m_startingSelection = selection; |
- m_startingRootEditableElement = selection.rootEditableElement(); |
-} |
- |
-void EditCommandComposition::setEndingSelection(const VisibleSelection& selection) |
-{ |
- m_endingSelection = selection; |
- m_endingRootEditableElement = selection.rootEditableElement(); |
-} |
- |
-DEFINE_TRACE(EditCommandComposition) |
-{ |
- visitor->trace(m_document); |
- visitor->trace(m_startingSelection); |
- visitor->trace(m_endingSelection); |
- visitor->trace(m_commands); |
- visitor->trace(m_startingRootEditableElement); |
- visitor->trace(m_endingRootEditableElement); |
- UndoStep::trace(visitor); |
-} |
- |
-CompositeEditCommand::CompositeEditCommand(Document& document) |
- : EditCommand(document) |
-{ |
-} |
- |
-CompositeEditCommand::~CompositeEditCommand() |
-{ |
- ASSERT(isTopLevelCommand() || !m_composition); |
-} |
- |
-void CompositeEditCommand::apply() |
-{ |
- if (!endingSelection().isContentRichlyEditable()) { |
- switch (editingAction()) { |
- case EditActionTyping: |
- case EditActionPaste: |
- case EditActionDrag: |
- case EditActionSetWritingDirection: |
- case EditActionCut: |
- case EditActionUnspecified: |
- break; |
- default: |
- ASSERT_NOT_REACHED(); |
- return; |
- } |
- } |
- ensureComposition(); |
- |
- // Changes to the document may have been made since the last editing operation that require a layout, as in <rdar://problem/5658603>. |
- // Low level operations, like RemoveNodeCommand, don't require a layout because the high level operations that use them perform one |
- // if one is necessary (like for the creation of VisiblePositions). |
- document().updateLayoutIgnorePendingStylesheets(); |
- |
- LocalFrame* frame = document().frame(); |
- ASSERT(frame); |
- { |
- EventQueueScope eventQueueScope; |
- doApply(); |
- } |
- |
- // Only need to call appliedEditing for top-level commands, |
- // and TypingCommands do it on their own (see TypingCommand::typingAddedToOpenCommand). |
- if (!isTypingCommand()) |
- frame->editor().appliedEditing(this); |
- setShouldRetainAutocorrectionIndicator(false); |
-} |
- |
-EditCommandComposition* CompositeEditCommand::ensureComposition() |
-{ |
- CompositeEditCommand* command = this; |
- while (command && command->parent()) |
- command = command->parent(); |
- if (!command->m_composition) |
- command->m_composition = EditCommandComposition::create(&document(), startingSelection(), endingSelection(), editingAction()); |
- return command->m_composition.get(); |
-} |
- |
-bool CompositeEditCommand::preservesTypingStyle() const |
-{ |
- return false; |
-} |
- |
-bool CompositeEditCommand::isTypingCommand() const |
-{ |
- return false; |
-} |
- |
-void CompositeEditCommand::setShouldRetainAutocorrectionIndicator(bool) |
-{ |
-} |
- |
-// |
-// sugary-sweet convenience functions to help create and apply edit commands in composite commands |
-// |
-void CompositeEditCommand::applyCommandToComposite(PassRefPtrWillBeRawPtr<EditCommand> prpCommand) |
-{ |
- RefPtrWillBeRawPtr<EditCommand> command = prpCommand; |
- command->setParent(this); |
- command->doApply(); |
- if (command->isSimpleEditCommand()) { |
- command->setParent(0); |
- ensureComposition()->append(toSimpleEditCommand(command.get())); |
- } |
- m_commands.append(command.release()); |
-} |
- |
-void CompositeEditCommand::applyCommandToComposite(PassRefPtrWillBeRawPtr<CompositeEditCommand> command, const VisibleSelection& selection) |
-{ |
- command->setParent(this); |
- if (!VisibleSelection::InDOMTree::equalSelections(selection, command->endingSelection())) { |
- command->setStartingSelection(selection); |
- command->setEndingSelection(selection); |
- } |
- command->doApply(); |
- m_commands.append(command); |
-} |
- |
-void CompositeEditCommand::applyStyle(const EditingStyle* style, EditAction editingAction) |
-{ |
- applyCommandToComposite(ApplyStyleCommand::create(document(), style, editingAction)); |
-} |
- |
-void CompositeEditCommand::applyStyle(const EditingStyle* style, const Position& start, const Position& end, EditAction editingAction) |
-{ |
- applyCommandToComposite(ApplyStyleCommand::create(document(), style, start, end, editingAction)); |
-} |
- |
-void CompositeEditCommand::applyStyledElement(PassRefPtrWillBeRawPtr<Element> element) |
-{ |
- applyCommandToComposite(ApplyStyleCommand::create(element, false)); |
-} |
- |
-void CompositeEditCommand::removeStyledElement(PassRefPtrWillBeRawPtr<Element> element) |
-{ |
- applyCommandToComposite(ApplyStyleCommand::create(element, true)); |
-} |
- |
-void CompositeEditCommand::insertParagraphSeparator(bool useDefaultParagraphElement, bool pasteBlockqutoeIntoUnquotedArea) |
-{ |
- applyCommandToComposite(InsertParagraphSeparatorCommand::create(document(), useDefaultParagraphElement, pasteBlockqutoeIntoUnquotedArea)); |
-} |
- |
-bool CompositeEditCommand::isRemovableBlock(const Node* node) |
-{ |
- ASSERT(node); |
- if (!isHTMLDivElement(*node)) |
- return false; |
- |
- const HTMLDivElement& element = toHTMLDivElement(*node); |
- ContainerNode* parentNode = element.parentNode(); |
- if (parentNode && parentNode->firstChild() != parentNode->lastChild()) |
- return false; |
- |
- if (!element.hasAttributes()) |
- return true; |
- |
- return false; |
-} |
- |
-void CompositeEditCommand::insertNodeBefore(PassRefPtrWillBeRawPtr<Node> insertChild, PassRefPtrWillBeRawPtr<Node> refChild, ShouldAssumeContentIsAlwaysEditable shouldAssumeContentIsAlwaysEditable) |
-{ |
- ASSERT(!isHTMLBodyElement(*refChild)); |
- applyCommandToComposite(InsertNodeBeforeCommand::create(insertChild, refChild, shouldAssumeContentIsAlwaysEditable)); |
-} |
- |
-void CompositeEditCommand::insertNodeAfter(PassRefPtrWillBeRawPtr<Node> insertChild, PassRefPtrWillBeRawPtr<Node> refChild) |
-{ |
- ASSERT(insertChild); |
- ASSERT(refChild); |
- ASSERT(!isHTMLBodyElement(*refChild)); |
- ContainerNode* parent = refChild->parentNode(); |
- ASSERT(parent); |
- ASSERT(!parent->isShadowRoot()); |
- if (parent->lastChild() == refChild) { |
- appendNode(insertChild, parent); |
- } else { |
- ASSERT(refChild->nextSibling()); |
- insertNodeBefore(insertChild, refChild->nextSibling()); |
- } |
-} |
- |
-void CompositeEditCommand::insertNodeAt(PassRefPtrWillBeRawPtr<Node> insertChild, const Position& editingPosition) |
-{ |
- ASSERT(isEditablePosition(editingPosition, ContentIsEditable, DoNotUpdateStyle)); |
- // For editing positions like [table, 0], insert before the table, |
- // likewise for replaced elements, brs, etc. |
- Position p = editingPosition.parentAnchoredEquivalent(); |
- Node* refChild = p.anchorNode(); |
- int offset = p.offsetInContainerNode(); |
- |
- if (canHaveChildrenForEditing(refChild)) { |
- Node* child = refChild->firstChild(); |
- for (int i = 0; child && i < offset; i++) |
- child = child->nextSibling(); |
- if (child) |
- insertNodeBefore(insertChild, child); |
- else |
- appendNode(insertChild, toContainerNode(refChild)); |
- } else if (caretMinOffset(refChild) >= offset) { |
- insertNodeBefore(insertChild, refChild); |
- } else if (refChild->isTextNode() && caretMaxOffset(refChild) > offset) { |
- splitTextNode(toText(refChild), offset); |
- |
- // Mutation events (bug 22634) from the text node insertion may have removed the refChild |
- if (!refChild->inDocument()) |
- return; |
- insertNodeBefore(insertChild, refChild); |
- } else { |
- insertNodeAfter(insertChild, refChild); |
- } |
-} |
- |
-void CompositeEditCommand::appendNode(PassRefPtrWillBeRawPtr<Node> node, PassRefPtrWillBeRawPtr<ContainerNode> parent) |
-{ |
- // When cloneParagraphUnderNewElement() clones the fallback content |
- // of an OBJECT element, the ASSERT below may fire since the return |
- // value of canHaveChildrenForEditing is not reliable until the layout |
- // object of the OBJECT is created. Hence we ignore this check for OBJECTs. |
- ASSERT(canHaveChildrenForEditing(parent.get()) |
- || (parent->isElementNode() && toElement(parent.get())->tagQName() == objectTag)); |
- applyCommandToComposite(AppendNodeCommand::create(parent, node)); |
-} |
- |
-void CompositeEditCommand::removeChildrenInRange(PassRefPtrWillBeRawPtr<Node> node, unsigned from, unsigned to) |
-{ |
- WillBeHeapVector<RefPtrWillBeMember<Node>> children; |
- Node* child = NodeTraversal::childAt(*node, from); |
- for (unsigned i = from; child && i < to; i++, child = child->nextSibling()) |
- children.append(child); |
- |
- size_t size = children.size(); |
- for (size_t i = 0; i < size; ++i) |
- removeNode(children[i].release()); |
-} |
- |
-void CompositeEditCommand::removeNode(PassRefPtrWillBeRawPtr<Node> node, ShouldAssumeContentIsAlwaysEditable shouldAssumeContentIsAlwaysEditable) |
-{ |
- if (!node || !node->nonShadowBoundaryParentNode()) |
- return; |
- applyCommandToComposite(RemoveNodeCommand::create(node, shouldAssumeContentIsAlwaysEditable)); |
-} |
- |
-void CompositeEditCommand::removeNodePreservingChildren(PassRefPtrWillBeRawPtr<Node> node, ShouldAssumeContentIsAlwaysEditable shouldAssumeContentIsAlwaysEditable) |
-{ |
- applyCommandToComposite(RemoveNodePreservingChildrenCommand::create(node, shouldAssumeContentIsAlwaysEditable)); |
-} |
- |
-void CompositeEditCommand::removeNodeAndPruneAncestors(PassRefPtrWillBeRawPtr<Node> node, Node* excludeNode) |
-{ |
- ASSERT(node.get() != excludeNode); |
- RefPtrWillBeRawPtr<ContainerNode> parent = node->parentNode(); |
- removeNode(node); |
- prune(parent.release(), excludeNode); |
-} |
- |
-void CompositeEditCommand::moveRemainingSiblingsToNewParent(Node* node, Node* pastLastNodeToMove, PassRefPtrWillBeRawPtr<Element> prpNewParent) |
-{ |
- NodeVector nodesToRemove; |
- RefPtrWillBeRawPtr<Element> newParent = prpNewParent; |
- |
- for (; node && node != pastLastNodeToMove; node = node->nextSibling()) |
- nodesToRemove.append(node); |
- |
- for (unsigned i = 0; i < nodesToRemove.size(); i++) { |
- removeNode(nodesToRemove[i]); |
- appendNode(nodesToRemove[i], newParent); |
- } |
-} |
- |
-void CompositeEditCommand::updatePositionForNodeRemovalPreservingChildren(Position& position, Node& node) |
-{ |
- int offset = position.isOffsetInAnchor() ? position.offsetInContainerNode() : 0; |
- updatePositionForNodeRemoval(position, node); |
- if (offset == 0) |
- return; |
- position = Position(position.computeContainerNode(), offset); |
-} |
- |
-HTMLSpanElement* CompositeEditCommand::replaceElementWithSpanPreservingChildrenAndAttributes(PassRefPtrWillBeRawPtr<HTMLElement> node) |
-{ |
- // It would also be possible to implement all of ReplaceNodeWithSpanCommand |
- // as a series of existing smaller edit commands. Someone who wanted to |
- // reduce the number of edit commands could do so here. |
- RefPtrWillBeRawPtr<ReplaceNodeWithSpanCommand> command = ReplaceNodeWithSpanCommand::create(node); |
- applyCommandToComposite(command); |
- // Returning a raw pointer here is OK because the command is retained by |
- // applyCommandToComposite (thus retaining the span), and the span is also |
- // in the DOM tree, and thus alive whie it has a parent. |
- ASSERT(command->spanElement()->inDocument()); |
- return command->spanElement(); |
-} |
- |
-void CompositeEditCommand::prune(PassRefPtrWillBeRawPtr<Node> node, Node* excludeNode) |
-{ |
- if (RefPtrWillBeRawPtr<Node> highestNodeToRemove = highestNodeToRemoveInPruning(node.get(), excludeNode)) |
- removeNode(highestNodeToRemove.release()); |
-} |
- |
-void CompositeEditCommand::splitTextNode(PassRefPtrWillBeRawPtr<Text> node, unsigned offset) |
-{ |
- applyCommandToComposite(SplitTextNodeCommand::create(node, offset)); |
-} |
- |
-void CompositeEditCommand::splitElement(PassRefPtrWillBeRawPtr<Element> element, PassRefPtrWillBeRawPtr<Node> atChild) |
-{ |
- applyCommandToComposite(SplitElementCommand::create(element, atChild)); |
-} |
- |
-void CompositeEditCommand::mergeIdenticalElements(PassRefPtrWillBeRawPtr<Element> prpFirst, PassRefPtrWillBeRawPtr<Element> prpSecond) |
-{ |
- RefPtrWillBeRawPtr<Element> first = prpFirst; |
- RefPtrWillBeRawPtr<Element> second = prpSecond; |
- ASSERT(!first->isDescendantOf(second.get()) && second != first); |
- if (first->nextSibling() != second) { |
- removeNode(second); |
- insertNodeAfter(second, first); |
- } |
- applyCommandToComposite(MergeIdenticalElementsCommand::create(first, second)); |
-} |
- |
-void CompositeEditCommand::wrapContentsInDummySpan(PassRefPtrWillBeRawPtr<Element> element) |
-{ |
- applyCommandToComposite(WrapContentsInDummySpanCommand::create(element)); |
-} |
- |
-void CompositeEditCommand::splitTextNodeContainingElement(PassRefPtrWillBeRawPtr<Text> text, unsigned offset) |
-{ |
- applyCommandToComposite(SplitTextNodeContainingElementCommand::create(text, offset)); |
-} |
- |
-void CompositeEditCommand::insertTextIntoNode(PassRefPtrWillBeRawPtr<Text> node, unsigned offset, const String& text) |
-{ |
- if (!text.isEmpty()) |
- applyCommandToComposite(InsertIntoTextNodeCommand::create(node, offset, text)); |
-} |
- |
-void CompositeEditCommand::deleteTextFromNode(PassRefPtrWillBeRawPtr<Text> node, unsigned offset, unsigned count) |
-{ |
- applyCommandToComposite(DeleteFromTextNodeCommand::create(node, offset, count)); |
-} |
- |
-void CompositeEditCommand::replaceTextInNode(PassRefPtrWillBeRawPtr<Text> prpNode, unsigned offset, unsigned count, const String& replacementText) |
-{ |
- RefPtrWillBeRawPtr<Text> node(prpNode); |
- applyCommandToComposite(DeleteFromTextNodeCommand::create(node, offset, count)); |
- if (!replacementText.isEmpty()) |
- applyCommandToComposite(InsertIntoTextNodeCommand::create(node, offset, replacementText)); |
-} |
- |
-Position CompositeEditCommand::replaceSelectedTextInNode(const String& text) |
-{ |
- Position start = endingSelection().start(); |
- Position end = endingSelection().end(); |
- if (start.computeContainerNode() != end.computeContainerNode() || !start.computeContainerNode()->isTextNode() || isTabHTMLSpanElementTextNode(start.computeContainerNode())) |
- return Position(); |
- |
- RefPtrWillBeRawPtr<Text> textNode = toText(start.computeContainerNode()); |
- replaceTextInNode(textNode, start.offsetInContainerNode(), end.offsetInContainerNode() - start.offsetInContainerNode(), text); |
- |
- return Position(textNode.release(), start.offsetInContainerNode() + text.length()); |
-} |
- |
-static void copyMarkerTypesAndDescriptions(const DocumentMarkerVector& markerPointers, Vector<DocumentMarker::MarkerType>& types, Vector<String>& descriptions) |
-{ |
- size_t arraySize = markerPointers.size(); |
- types.reserveCapacity(arraySize); |
- descriptions.reserveCapacity(arraySize); |
- for (const auto& markerPointer : markerPointers) { |
- types.append(markerPointer->type()); |
- descriptions.append(markerPointer->description()); |
- } |
-} |
- |
-void CompositeEditCommand::replaceTextInNodePreservingMarkers(PassRefPtrWillBeRawPtr<Text> prpNode, unsigned offset, unsigned count, const String& replacementText) |
-{ |
- RefPtrWillBeRawPtr<Text> node(prpNode); |
- DocumentMarkerController& markerController = document().markers(); |
- Vector<DocumentMarker::MarkerType> types; |
- Vector<String> descriptions; |
- copyMarkerTypesAndDescriptions(markerController.markersInRange(EphemeralRange(Position(node.get(), offset), Position(node.get(), offset + count)), DocumentMarker::AllMarkers()), types, descriptions); |
- replaceTextInNode(node, offset, count, replacementText); |
- Position startPosition(node.get(), offset); |
- Position endPosition(node.get(), offset + replacementText.length()); |
- ASSERT(types.size() == descriptions.size()); |
- for (size_t i = 0; i < types.size(); ++i) |
- markerController.addMarker(startPosition, endPosition, types[i], descriptions[i]); |
-} |
- |
-Position CompositeEditCommand::positionOutsideTabSpan(const Position& pos) |
-{ |
- if (!isTabHTMLSpanElementTextNode(pos.anchorNode())) |
- return pos; |
- |
- switch (pos.anchorType()) { |
- case PositionAnchorType::BeforeChildren: |
- case PositionAnchorType::AfterChildren: |
- ASSERT_NOT_REACHED(); |
- return pos; |
- case PositionAnchorType::OffsetInAnchor: |
- break; |
- case PositionAnchorType::BeforeAnchor: |
- return positionInParentBeforeNode(*pos.anchorNode()); |
- case PositionAnchorType::AfterAnchor: |
- return positionInParentAfterNode(*pos.anchorNode()); |
- } |
- |
- HTMLSpanElement* tabSpan = tabSpanElement(pos.computeContainerNode()); |
- ASSERT(tabSpan); |
- |
- if (pos.offsetInContainerNode() <= caretMinOffset(pos.computeContainerNode())) |
- return positionInParentBeforeNode(*tabSpan); |
- |
- if (pos.offsetInContainerNode() >= caretMaxOffset(pos.computeContainerNode())) |
- return positionInParentAfterNode(*tabSpan); |
- |
- splitTextNodeContainingElement(toText(pos.computeContainerNode()), pos.offsetInContainerNode()); |
- return positionInParentBeforeNode(*tabSpan); |
-} |
- |
-void CompositeEditCommand::insertNodeAtTabSpanPosition(PassRefPtrWillBeRawPtr<Node> node, const Position& pos) |
-{ |
- // insert node before, after, or at split of tab span |
- insertNodeAt(node, positionOutsideTabSpan(pos)); |
-} |
- |
-void CompositeEditCommand::deleteSelection(bool smartDelete, bool mergeBlocksAfterDelete, bool expandForSpecialElements, bool sanitizeMarkup) |
-{ |
- if (endingSelection().isRange()) |
- applyCommandToComposite(DeleteSelectionCommand::create(document(), smartDelete, mergeBlocksAfterDelete, expandForSpecialElements, sanitizeMarkup)); |
-} |
- |
-void CompositeEditCommand::deleteSelection(const VisibleSelection &selection, bool smartDelete, bool mergeBlocksAfterDelete, bool expandForSpecialElements, bool sanitizeMarkup) |
-{ |
- if (selection.isRange()) |
- applyCommandToComposite(DeleteSelectionCommand::create(selection, smartDelete, mergeBlocksAfterDelete, expandForSpecialElements, sanitizeMarkup)); |
-} |
- |
-void CompositeEditCommand::removeCSSProperty(PassRefPtrWillBeRawPtr<Element> element, CSSPropertyID property) |
-{ |
- applyCommandToComposite(RemoveCSSPropertyCommand::create(document(), element, property)); |
-} |
- |
-void CompositeEditCommand::removeElementAttribute(PassRefPtrWillBeRawPtr<Element> element, const QualifiedName& attribute) |
-{ |
- setNodeAttribute(element, attribute, AtomicString()); |
-} |
- |
-void CompositeEditCommand::setNodeAttribute(PassRefPtrWillBeRawPtr<Element> element, const QualifiedName& attribute, const AtomicString& value) |
-{ |
- applyCommandToComposite(SetNodeAttributeCommand::create(element, attribute, value)); |
-} |
- |
-static inline bool containsOnlyWhitespace(const String& text) |
-{ |
- for (unsigned i = 0; i < text.length(); ++i) { |
- if (!isWhitespace(text[i])) |
- return false; |
- } |
- |
- return true; |
-} |
- |
-bool CompositeEditCommand::shouldRebalanceLeadingWhitespaceFor(const String& text) const |
-{ |
- return containsOnlyWhitespace(text); |
-} |
- |
-bool CompositeEditCommand::canRebalance(const Position& position) const |
-{ |
- Node* node = position.computeContainerNode(); |
- if (!position.isOffsetInAnchor() || !node || !node->isTextNode()) |
- return false; |
- |
- Text* textNode = toText(node); |
- if (textNode->length() == 0) |
- return false; |
- |
- LayoutText* layoutText = textNode->layoutObject(); |
- if (layoutText && !layoutText->style()->collapseWhiteSpace()) |
- return false; |
- |
- return true; |
-} |
- |
-// FIXME: Doesn't go into text nodes that contribute adjacent text (siblings, cousins, etc). |
-void CompositeEditCommand::rebalanceWhitespaceAt(const Position& position) |
-{ |
- Node* node = position.computeContainerNode(); |
- if (!canRebalance(position)) |
- return; |
- |
- // If the rebalance is for the single offset, and neither text[offset] nor text[offset - 1] are some form of whitespace, do nothing. |
- int offset = position.computeOffsetInContainerNode(); |
- String text = toText(node)->data(); |
- if (!isWhitespace(text[offset])) { |
- offset--; |
- if (offset < 0 || !isWhitespace(text[offset])) |
- return; |
- } |
- |
- rebalanceWhitespaceOnTextSubstring(toText(node), position.offsetInContainerNode(), position.offsetInContainerNode()); |
-} |
- |
-void CompositeEditCommand::rebalanceWhitespaceOnTextSubstring(PassRefPtrWillBeRawPtr<Text> prpTextNode, int startOffset, int endOffset) |
-{ |
- RefPtrWillBeRawPtr<Text> textNode = prpTextNode; |
- |
- String text = textNode->data(); |
- ASSERT(!text.isEmpty()); |
- |
- // Set upstream and downstream to define the extent of the whitespace surrounding text[offset]. |
- int upstream = startOffset; |
- while (upstream > 0 && isWhitespace(text[upstream - 1])) |
- upstream--; |
- |
- int downstream = endOffset; |
- while ((unsigned)downstream < text.length() && isWhitespace(text[downstream])) |
- downstream++; |
- |
- int length = downstream - upstream; |
- if (!length) |
- return; |
- |
- VisiblePosition visibleUpstreamPos(Position(textNode, upstream)); |
- VisiblePosition visibleDownstreamPos(Position(textNode, downstream)); |
- |
- String string = text.substring(upstream, length); |
- String rebalancedString = stringWithRebalancedWhitespace(string, |
- // FIXME: Because of the problem mentioned at the top of this function, we |
- // must also use nbsps at the start/end of the string because this function |
- // doesn't get all surrounding whitespace, just the whitespace in the |
- // current text node. |
- isStartOfParagraph(visibleUpstreamPos) || upstream == 0, |
- isEndOfParagraph(visibleDownstreamPos) || (unsigned)downstream == text.length()); |
- |
- if (string != rebalancedString) |
- replaceTextInNodePreservingMarkers(textNode.release(), upstream, length, rebalancedString); |
-} |
- |
-void CompositeEditCommand::prepareWhitespaceAtPositionForSplit(Position& position) |
-{ |
- Node* node = position.anchorNode(); |
- if (!node || !node->isTextNode()) |
- return; |
- Text* textNode = toText(node); |
- |
- if (textNode->length() == 0) |
- return; |
- LayoutText* layoutText = textNode->layoutObject(); |
- if (layoutText && !layoutText->style()->collapseWhiteSpace()) |
- return; |
- |
- // Delete collapsed whitespace so that inserting nbsps doesn't uncollapse it. |
- Position upstreamPos = position.upstream(); |
- deleteInsignificantText(upstreamPos, position.downstream()); |
- position = upstreamPos.downstream(); |
- |
- VisiblePosition visiblePos(position); |
- VisiblePosition previousVisiblePos(visiblePos.previous()); |
- replaceCollapsibleWhitespaceWithNonBreakingSpaceIfNeeded(previousVisiblePos); |
- replaceCollapsibleWhitespaceWithNonBreakingSpaceIfNeeded(visiblePos); |
-} |
- |
-void CompositeEditCommand::replaceCollapsibleWhitespaceWithNonBreakingSpaceIfNeeded(const VisiblePosition& visiblePosition) |
-{ |
- if (!isCollapsibleWhitespace(visiblePosition.characterAfter())) |
- return; |
- Position pos = visiblePosition.deepEquivalent().downstream(); |
- if (!pos.computeContainerNode() || !pos.computeContainerNode()->isTextNode()) |
- return; |
- replaceTextInNodePreservingMarkers(toText(pos.computeContainerNode()), pos.offsetInContainerNode(), 1, nonBreakingSpaceString()); |
-} |
- |
-void CompositeEditCommand::rebalanceWhitespace() |
-{ |
- VisibleSelection selection = endingSelection(); |
- if (selection.isNone()) |
- return; |
- |
- rebalanceWhitespaceAt(selection.start()); |
- if (selection.isRange()) |
- rebalanceWhitespaceAt(selection.end()); |
-} |
- |
-void CompositeEditCommand::deleteInsignificantText(PassRefPtrWillBeRawPtr<Text> textNode, unsigned start, unsigned end) |
-{ |
- if (!textNode || start >= end) |
- return; |
- |
- document().updateLayout(); |
- |
- LayoutText* textLayoutObject = textNode->layoutObject(); |
- if (!textLayoutObject) |
- return; |
- |
- Vector<InlineTextBox*> sortedTextBoxes; |
- size_t sortedTextBoxesPosition = 0; |
- |
- for (InlineTextBox* textBox = textLayoutObject->firstTextBox(); textBox; textBox = textBox->nextTextBox()) |
- sortedTextBoxes.append(textBox); |
- |
- // If there is mixed directionality text, the boxes can be out of order, |
- // (like Arabic with embedded LTR), so sort them first. |
- if (textLayoutObject->containsReversedText()) |
- std::sort(sortedTextBoxes.begin(), sortedTextBoxes.end(), InlineTextBox::compareByStart); |
- InlineTextBox* box = sortedTextBoxes.isEmpty() ? 0 : sortedTextBoxes[sortedTextBoxesPosition]; |
- |
- if (!box) { |
- // whole text node is empty |
- removeNode(textNode); |
- return; |
- } |
- |
- unsigned length = textNode->length(); |
- if (start >= length || end > length) |
- return; |
- |
- unsigned removed = 0; |
- InlineTextBox* prevBox = nullptr; |
- String str; |
- |
- // This loop structure works to process all gaps preceding a box, |
- // and also will look at the gap after the last box. |
- while (prevBox || box) { |
- unsigned gapStart = prevBox ? prevBox->start() + prevBox->len() : 0; |
- if (end < gapStart) { |
- // No more chance for any intersections |
- break; |
- } |
- |
- unsigned gapEnd = box ? box->start() : length; |
- bool indicesIntersect = start <= gapEnd && end >= gapStart; |
- int gapLen = gapEnd - gapStart; |
- if (indicesIntersect && gapLen > 0) { |
- gapStart = std::max(gapStart, start); |
- if (str.isNull()) |
- str = textNode->data().substring(start, end - start); |
- // remove text in the gap |
- str.remove(gapStart - start - removed, gapLen); |
- removed += gapLen; |
- } |
- |
- prevBox = box; |
- if (box) { |
- if (++sortedTextBoxesPosition < sortedTextBoxes.size()) |
- box = sortedTextBoxes[sortedTextBoxesPosition]; |
- else |
- box = 0; |
- } |
- } |
- |
- if (!str.isNull()) { |
- // Replace the text between start and end with our pruned version. |
- if (!str.isEmpty()) { |
- replaceTextInNode(textNode, start, end - start, str); |
- } else { |
- // Assert that we are not going to delete all of the text in the node. |
- // If we were, that should have been done above with the call to |
- // removeNode and return. |
- ASSERT(start > 0 || end - start < textNode->length()); |
- deleteTextFromNode(textNode, start, end - start); |
- } |
- } |
-} |
- |
-void CompositeEditCommand::deleteInsignificantText(const Position& start, const Position& end) |
-{ |
- if (start.isNull() || end.isNull()) |
- return; |
- |
- if (comparePositions(start, end) >= 0) |
- return; |
- |
- WillBeHeapVector<RefPtrWillBeMember<Text>> nodes; |
- for (Node& node : NodeTraversal::startsAt(start.anchorNode())) { |
- if (node.isTextNode()) |
- nodes.append(toText(&node)); |
- if (&node == end.anchorNode()) |
- break; |
- } |
- |
- for (const auto& node : nodes) { |
- Text* textNode = node.get(); |
- int startOffset = textNode == start.anchorNode() ? start.computeOffsetInContainerNode() : 0; |
- int endOffset = textNode == end.anchorNode() ? end.computeOffsetInContainerNode() : static_cast<int>(textNode->length()); |
- deleteInsignificantText(textNode, startOffset, endOffset); |
- } |
-} |
- |
-void CompositeEditCommand::deleteInsignificantTextDownstream(const Position& pos) |
-{ |
- Position end = VisiblePosition(pos, VP_DEFAULT_AFFINITY).next().deepEquivalent().downstream(); |
- deleteInsignificantText(pos, end); |
-} |
- |
-PassRefPtrWillBeRawPtr<HTMLBRElement> CompositeEditCommand::appendBlockPlaceholder(PassRefPtrWillBeRawPtr<Element> container) |
-{ |
- if (!container) |
- return nullptr; |
- |
- document().updateLayoutIgnorePendingStylesheets(); |
- |
- // Should assert isLayoutBlockFlow || isInlineFlow when deletion improves. See 4244964. |
- ASSERT(container->layoutObject()); |
- |
- RefPtrWillBeRawPtr<HTMLBRElement> placeholder = createBlockPlaceholderElement(document()); |
- appendNode(placeholder, container); |
- return placeholder.release(); |
-} |
- |
-PassRefPtrWillBeRawPtr<HTMLBRElement> CompositeEditCommand::insertBlockPlaceholder(const Position& pos) |
-{ |
- if (pos.isNull()) |
- return nullptr; |
- |
- // Should assert isLayoutBlockFlow || isInlineFlow when deletion improves. See 4244964. |
- ASSERT(pos.anchorNode()->layoutObject()); |
- |
- RefPtrWillBeRawPtr<HTMLBRElement> placeholder = createBlockPlaceholderElement(document()); |
- insertNodeAt(placeholder, pos); |
- return placeholder.release(); |
-} |
- |
-PassRefPtrWillBeRawPtr<HTMLBRElement> CompositeEditCommand::addBlockPlaceholderIfNeeded(Element* container) |
-{ |
- if (!container) |
- return nullptr; |
- |
- document().updateLayoutIgnorePendingStylesheets(); |
- |
- LayoutObject* layoutObject = container->layoutObject(); |
- if (!layoutObject || !layoutObject->isLayoutBlockFlow()) |
- return nullptr; |
- |
- // append the placeholder to make sure it follows |
- // any unrendered blocks |
- LayoutBlockFlow* block = toLayoutBlockFlow(layoutObject); |
- if (block->size().height() == 0 || (block->isListItem() && toLayoutListItem(block)->isEmpty())) |
- return appendBlockPlaceholder(container); |
- |
- return nullptr; |
-} |
- |
-// Assumes that the position is at a placeholder and does the removal without much checking. |
-void CompositeEditCommand::removePlaceholderAt(const Position& p) |
-{ |
- ASSERT(lineBreakExistsAtPosition(p)); |
- |
- // We are certain that the position is at a line break, but it may be a br or a preserved newline. |
- if (isHTMLBRElement(*p.anchorNode())) { |
- removeNode(p.anchorNode()); |
- return; |
- } |
- |
- deleteTextFromNode(toText(p.anchorNode()), p.offsetInContainerNode(), 1); |
-} |
- |
-PassRefPtrWillBeRawPtr<HTMLElement> CompositeEditCommand::insertNewDefaultParagraphElementAt(const Position& position) |
-{ |
- RefPtrWillBeRawPtr<HTMLElement> paragraphElement = createDefaultParagraphElement(document()); |
- paragraphElement->appendChild(createBreakElement(document())); |
- insertNodeAt(paragraphElement, position); |
- return paragraphElement.release(); |
-} |
- |
-// If the paragraph is not entirely within it's own block, create one and move the paragraph into |
-// it, and return that block. Otherwise return 0. |
-PassRefPtrWillBeRawPtr<HTMLElement> CompositeEditCommand::moveParagraphContentsToNewBlockIfNecessary(const Position& pos) |
-{ |
- ASSERT(isEditablePosition(pos, ContentIsEditable, DoNotUpdateStyle)); |
- |
- // It's strange that this function is responsible for verifying that pos has not been invalidated |
- // by an earlier call to this function. The caller, applyBlockStyle, should do this. |
- VisiblePosition visiblePos(pos, VP_DEFAULT_AFFINITY); |
- VisiblePosition visibleParagraphStart(startOfParagraph(visiblePos)); |
- VisiblePosition visibleParagraphEnd = endOfParagraph(visiblePos); |
- VisiblePosition next = visibleParagraphEnd.next(); |
- VisiblePosition visibleEnd = next.isNotNull() ? next : visibleParagraphEnd; |
- |
- Position upstreamStart = visibleParagraphStart.deepEquivalent().upstream(); |
- Position upstreamEnd = visibleEnd.deepEquivalent().upstream(); |
- |
- // If there are no VisiblePositions in the same block as pos then |
- // upstreamStart will be outside the paragraph |
- if (comparePositions(pos, upstreamStart) < 0) |
- return nullptr; |
- |
- // Perform some checks to see if we need to perform work in this function. |
- if (isBlock(upstreamStart.anchorNode())) { |
- // If the block is the root editable element, always move content to a new block, |
- // since it is illegal to modify attributes on the root editable element for editing. |
- if (upstreamStart.anchorNode() == editableRootForPosition(upstreamStart)) { |
- // If the block is the root editable element and it contains no visible content, create a new |
- // block but don't try and move content into it, since there's nothing for moveParagraphs to move. |
- if (!Position::hasRenderedNonAnonymousDescendantsWithHeight(upstreamStart.anchorNode()->layoutObject())) |
- return insertNewDefaultParagraphElementAt(upstreamStart); |
- } else if (isBlock(upstreamEnd.anchorNode())) { |
- if (!upstreamEnd.anchorNode()->isDescendantOf(upstreamStart.anchorNode())) { |
- // If the paragraph end is a descendant of paragraph start, then we need to run |
- // the rest of this function. If not, we can bail here. |
- return nullptr; |
- } |
- } else if (enclosingBlock(upstreamEnd.anchorNode()) != upstreamStart.anchorNode()) { |
- // It should be an ancestor of the paragraph start. |
- // We can bail as we have a full block to work with. |
- return nullptr; |
- } else if (isEndOfEditableOrNonEditableContent(visibleEnd)) { |
- // At the end of the editable region. We can bail here as well. |
- return nullptr; |
- } |
- } |
- |
- if (visibleParagraphEnd.isNull()) |
- return nullptr; |
- |
- RefPtrWillBeRawPtr<HTMLElement> newBlock = insertNewDefaultParagraphElementAt(upstreamStart); |
- |
- bool endWasBr = isHTMLBRElement(*visibleParagraphEnd.deepEquivalent().anchorNode()); |
- |
- // Inserting default paragraph element can change visible position. We |
- // should update visible positions before use them. |
- visiblePos = VisiblePosition(pos, VP_DEFAULT_AFFINITY); |
- visibleParagraphStart = VisiblePosition(startOfParagraph(visiblePos)); |
- visibleParagraphEnd = VisiblePosition(endOfParagraph(visiblePos)); |
- moveParagraphs(visibleParagraphStart, visibleParagraphEnd, VisiblePosition(firstPositionInNode(newBlock.get()))); |
- |
- if (newBlock->lastChild() && isHTMLBRElement(*newBlock->lastChild()) && !endWasBr) |
- removeNode(newBlock->lastChild()); |
- |
- return newBlock.release(); |
-} |
- |
-void CompositeEditCommand::pushAnchorElementDown(Element* anchorNode) |
-{ |
- if (!anchorNode) |
- return; |
- |
- ASSERT(anchorNode->isLink()); |
- |
- setEndingSelection(VisibleSelection::selectionFromContentsOfNode(anchorNode)); |
- applyStyledElement(anchorNode); |
- // Clones of anchorNode have been pushed down, now remove it. |
- if (anchorNode->inDocument()) |
- removeNodePreservingChildren(anchorNode); |
-} |
- |
-// Clone the paragraph between start and end under blockElement, |
-// preserving the hierarchy up to outerNode. |
- |
-void CompositeEditCommand::cloneParagraphUnderNewElement(const Position& start, const Position& end, Node* passedOuterNode, Element* blockElement) |
-{ |
- ASSERT(comparePositions(start, end) <= 0); |
- ASSERT(passedOuterNode); |
- ASSERT(blockElement); |
- |
- // First we clone the outerNode |
- RefPtrWillBeRawPtr<Node> lastNode = nullptr; |
- RefPtrWillBeRawPtr<Node> outerNode = passedOuterNode; |
- |
- if (outerNode->isRootEditableElement()) { |
- lastNode = blockElement; |
- } else { |
- lastNode = outerNode->cloneNode(isRenderedHTMLTableElement(outerNode.get())); |
- appendNode(lastNode, blockElement); |
- } |
- |
- if (start.anchorNode() != outerNode && lastNode->isElementNode() && start.anchorNode()->isDescendantOf(outerNode.get())) { |
- WillBeHeapVector<RefPtrWillBeMember<Node>> ancestors; |
- |
- // Insert each node from innerNode to outerNode (excluded) in a list. |
- for (Node* n = start.anchorNode(); n && n != outerNode; n = n->parentNode()) |
- ancestors.append(n); |
- |
- // Clone every node between start.anchorNode() and outerBlock. |
- |
- for (size_t i = ancestors.size(); i != 0; --i) { |
- Node* item = ancestors[i - 1].get(); |
- RefPtrWillBeRawPtr<Node> child = item->cloneNode(isRenderedHTMLTableElement(item)); |
- appendNode(child, toElement(lastNode)); |
- lastNode = child.release(); |
- } |
- } |
- |
- // Scripts specified in javascript protocol may remove |outerNode| |
- // during insertion, e.g. <iframe src="javascript:..."> |
- if (!outerNode->inDocument()) |
- return; |
- |
- // Handle the case of paragraphs with more than one node, |
- // cloning all the siblings until end.anchorNode() is reached. |
- |
- if (start.anchorNode() != end.anchorNode() && !start.anchorNode()->isDescendantOf(end.anchorNode())) { |
- // If end is not a descendant of outerNode we need to |
- // find the first common ancestor to increase the scope |
- // of our nextSibling traversal. |
- while (outerNode && !end.anchorNode()->isDescendantOf(outerNode.get())) { |
- outerNode = outerNode->parentNode(); |
- } |
- |
- if (!outerNode) |
- return; |
- |
- RefPtrWillBeRawPtr<Node> startNode = start.anchorNode(); |
- for (RefPtrWillBeRawPtr<Node> node = NodeTraversal::nextSkippingChildren(*startNode, outerNode.get()); node; node = NodeTraversal::nextSkippingChildren(*node, outerNode.get())) { |
- // Move lastNode up in the tree as much as node was moved up in the |
- // tree by NodeTraversal::nextSkippingChildren, so that the relative depth between |
- // node and the original start node is maintained in the clone. |
- while (startNode && lastNode && startNode->parentNode() != node->parentNode()) { |
- startNode = startNode->parentNode(); |
- lastNode = lastNode->parentNode(); |
- } |
- |
- if (!lastNode || !lastNode->parentNode()) |
- return; |
- |
- RefPtrWillBeRawPtr<Node> clonedNode = node->cloneNode(true); |
- insertNodeAfter(clonedNode, lastNode); |
- lastNode = clonedNode.release(); |
- if (node == end.anchorNode() || end.anchorNode()->isDescendantOf(node.get())) |
- break; |
- } |
- } |
-} |
- |
- |
-// There are bugs in deletion when it removes a fully selected table/list. |
-// It expands and removes the entire table/list, but will let content |
-// before and after the table/list collapse onto one line. |
-// Deleting a paragraph will leave a placeholder. Remove it (and prune |
-// empty or unrendered parents). |
- |
-void CompositeEditCommand::cleanupAfterDeletion(VisiblePosition destination) |
-{ |
- VisiblePosition caretAfterDelete = endingSelection().visibleStart(); |
- Node* destinationNode = destination.deepEquivalent().anchorNode(); |
- if (caretAfterDelete.deepEquivalent() != destination.deepEquivalent() && isStartOfParagraph(caretAfterDelete) && isEndOfParagraph(caretAfterDelete)) { |
- // Note: We want the rightmost candidate. |
- Position position = caretAfterDelete.deepEquivalent().downstream(); |
- Node* node = position.anchorNode(); |
- |
- // Bail if we'd remove an ancestor of our destination. |
- if (destinationNode && destinationNode->isDescendantOf(node)) |
- return; |
- |
- // Normally deletion will leave a br as a placeholder. |
- if (isHTMLBRElement(*node)) { |
- removeNodeAndPruneAncestors(node, destinationNode); |
- |
- // If the selection to move was empty and in an empty block that |
- // doesn't require a placeholder to prop itself open (like a bordered |
- // div or an li), remove it during the move (the list removal code |
- // expects this behavior). |
- } else if (isBlock(node)) { |
- // If caret position after deletion and destination position coincides, |
- // node should not be removed. |
- if (!rendersInDifferentPosition(position, destination.deepEquivalent())) { |
- prune(node, destinationNode); |
- return; |
- } |
- removeNodeAndPruneAncestors(node, destinationNode); |
- } else if (lineBreakExistsAtPosition(position)) { |
- // There is a preserved '\n' at caretAfterDelete. |
- // We can safely assume this is a text node. |
- Text* textNode = toText(node); |
- if (textNode->length() == 1) |
- removeNodeAndPruneAncestors(node, destinationNode); |
- else |
- deleteTextFromNode(textNode, position.computeOffsetInContainerNode(), 1); |
- } |
- } |
-} |
- |
-// This is a version of moveParagraph that preserves style by keeping the original markup |
-// It is currently used only by IndentOutdentCommand but it is meant to be used in the |
-// future by several other commands such as InsertList and the align commands. |
-// The blockElement parameter is the element to move the paragraph to, |
-// outerNode is the top element of the paragraph hierarchy. |
- |
-void CompositeEditCommand::moveParagraphWithClones(const VisiblePosition& startOfParagraphToMove, const VisiblePosition& endOfParagraphToMove, HTMLElement* blockElement, Node* outerNode) |
-{ |
- ASSERT(outerNode); |
- ASSERT(blockElement); |
- |
- VisiblePosition beforeParagraph = startOfParagraphToMove.previous(); |
- VisiblePosition afterParagraph(endOfParagraphToMove.next()); |
- |
- // We upstream() the end and downstream() the start so that we don't include collapsed whitespace in the move. |
- // When we paste a fragment, spaces after the end and before the start are treated as though they were rendered. |
- Position start = startOfParagraphToMove.deepEquivalent().downstream(); |
- Position end = startOfParagraphToMove.deepEquivalent() == endOfParagraphToMove.deepEquivalent() ? start : endOfParagraphToMove.deepEquivalent().upstream(); |
- if (comparePositions(start, end) > 0) |
- end = start; |
- |
- cloneParagraphUnderNewElement(start, end, outerNode, blockElement); |
- |
- setEndingSelection(VisibleSelection(start, end, DOWNSTREAM)); |
- deleteSelection(false, false, false); |
- |
- // There are bugs in deletion when it removes a fully selected table/list. |
- // It expands and removes the entire table/list, but will let content |
- // before and after the table/list collapse onto one line. |
- |
- cleanupAfterDeletion(); |
- |
- // Add a br if pruning an empty block level element caused a collapse. For example: |
- // foo^ |
- // <div>bar</div> |
- // baz |
- // Imagine moving 'bar' to ^. 'bar' will be deleted and its div pruned. That would |
- // cause 'baz' to collapse onto the line with 'foobar' unless we insert a br. |
- // Must recononicalize these two VisiblePositions after the pruning above. |
- beforeParagraph = VisiblePosition(beforeParagraph.deepEquivalent()); |
- afterParagraph = VisiblePosition(afterParagraph.deepEquivalent()); |
- |
- if (beforeParagraph.isNotNull() && !isRenderedTableElement(beforeParagraph.deepEquivalent().anchorNode()) |
- && ((!isEndOfParagraph(beforeParagraph) && !isStartOfParagraph(beforeParagraph)) || beforeParagraph.deepEquivalent() == afterParagraph.deepEquivalent())) { |
- // FIXME: Trim text between beforeParagraph and afterParagraph if they aren't equal. |
- insertNodeAt(createBreakElement(document()), beforeParagraph.deepEquivalent()); |
- } |
-} |
- |
-void CompositeEditCommand::moveParagraph(const VisiblePosition& startOfParagraphToMove, const VisiblePosition& endOfParagraphToMove, const VisiblePosition& destination, bool preserveSelection, bool preserveStyle, Node* constrainingAncestor) |
-{ |
- ASSERT(isStartOfParagraph(startOfParagraphToMove)); |
- ASSERT(isEndOfParagraph(endOfParagraphToMove)); |
- moveParagraphs(startOfParagraphToMove, endOfParagraphToMove, destination, preserveSelection, preserveStyle, constrainingAncestor); |
-} |
- |
-void CompositeEditCommand::moveParagraphs(const VisiblePosition& startOfParagraphToMove, const VisiblePosition& endOfParagraphToMove, const VisiblePosition& destination, bool preserveSelection, bool preserveStyle, Node* constrainingAncestor) |
-{ |
- if (startOfParagraphToMove.deepEquivalent() == destination.deepEquivalent() || startOfParagraphToMove.isNull()) |
- return; |
- |
- int startIndex = -1; |
- int endIndex = -1; |
- int destinationIndex = -1; |
- bool originalIsDirectional = endingSelection().isDirectional(); |
- if (preserveSelection && !endingSelection().isNone()) { |
- VisiblePosition visibleStart = endingSelection().visibleStart(); |
- VisiblePosition visibleEnd = endingSelection().visibleEnd(); |
- |
- bool startAfterParagraph = comparePositions(visibleStart, endOfParagraphToMove) > 0; |
- bool endBeforeParagraph = comparePositions(visibleEnd, startOfParagraphToMove) < 0; |
- |
- if (!startAfterParagraph && !endBeforeParagraph) { |
- bool startInParagraph = comparePositions(visibleStart, startOfParagraphToMove) >= 0; |
- bool endInParagraph = comparePositions(visibleEnd, endOfParagraphToMove) <= 0; |
- |
- startIndex = 0; |
- if (startInParagraph) |
- startIndex = TextIterator::rangeLength(startOfParagraphToMove.toParentAnchoredPosition(), visibleStart.toParentAnchoredPosition(), true); |
- |
- endIndex = 0; |
- if (endInParagraph) |
- endIndex = TextIterator::rangeLength(startOfParagraphToMove.toParentAnchoredPosition(), visibleEnd.toParentAnchoredPosition(), true); |
- } |
- } |
- |
- VisiblePosition beforeParagraph = startOfParagraphToMove.previous(CannotCrossEditingBoundary); |
- VisiblePosition afterParagraph(endOfParagraphToMove.next(CannotCrossEditingBoundary)); |
- |
- // We upstream() the end and downstream() the start so that we don't include collapsed whitespace in the move. |
- // When we paste a fragment, spaces after the end and before the start are treated as though they were rendered. |
- Position start = startOfParagraphToMove.deepEquivalent().downstream(); |
- Position end = endOfParagraphToMove.deepEquivalent().upstream(); |
- |
- // FIXME: This is an inefficient way to preserve style on nodes in the paragraph to move. It |
- // shouldn't matter though, since moved paragraphs will usually be quite small. |
- RefPtrWillBeRawPtr<DocumentFragment> fragment = startOfParagraphToMove.deepEquivalent() != endOfParagraphToMove.deepEquivalent() ? |
- createFragmentFromMarkup(document(), createMarkup(start.parentAnchoredEquivalent(), end.parentAnchoredEquivalent(), DoNotAnnotateForInterchange, ConvertBlocksToInlines::Convert, DoNotResolveURLs, constrainingAncestor), "") : nullptr; |
- |
- // A non-empty paragraph's style is moved when we copy and move it. We don't move |
- // anything if we're given an empty paragraph, but an empty paragraph can have style |
- // too, <div><b><br></b></div> for example. Save it so that we can preserve it later. |
- RefPtrWillBeRawPtr<EditingStyle> styleInEmptyParagraph = nullptr; |
- if (startOfParagraphToMove.deepEquivalent() == endOfParagraphToMove.deepEquivalent() && preserveStyle) { |
- styleInEmptyParagraph = EditingStyle::create(startOfParagraphToMove.deepEquivalent()); |
- styleInEmptyParagraph->mergeTypingStyle(&document()); |
- // The moved paragraph should assume the block style of the destination. |
- styleInEmptyParagraph->removeBlockProperties(); |
- } |
- |
- // FIXME (5098931): We should add a new insert action "WebViewInsertActionMoved" and call shouldInsertFragment here. |
- |
- setEndingSelection(VisibleSelection(start, end, DOWNSTREAM)); |
- document().frame()->spellChecker().clearMisspellingsAndBadGrammar(endingSelection()); |
- deleteSelection(false, false, false); |
- |
- ASSERT(destination.deepEquivalent().inDocument()); |
- cleanupAfterDeletion(destination); |
- ASSERT(destination.deepEquivalent().inDocument()); |
- |
- // Add a br if pruning an empty block level element caused a collapse. For example: |
- // foo^ |
- // <div>bar</div> |
- // baz |
- // Imagine moving 'bar' to ^. 'bar' will be deleted and its div pruned. That would |
- // cause 'baz' to collapse onto the line with 'foobar' unless we insert a br. |
- // Must recononicalize these two VisiblePositions after the pruning above. |
- beforeParagraph = VisiblePosition(beforeParagraph.deepEquivalent()); |
- afterParagraph = VisiblePosition(afterParagraph.deepEquivalent()); |
- if (beforeParagraph.isNotNull() && (!isEndOfParagraph(beforeParagraph) || beforeParagraph.deepEquivalent() == afterParagraph.deepEquivalent())) { |
- // FIXME: Trim text between beforeParagraph and afterParagraph if they aren't equal. |
- insertNodeAt(createBreakElement(document()), beforeParagraph.deepEquivalent()); |
- // Need an updateLayout here in case inserting the br has split a text node. |
- document().updateLayoutIgnorePendingStylesheets(); |
- } |
- |
- destinationIndex = TextIterator::rangeLength(firstPositionInNode(document().documentElement()), destination.toParentAnchoredPosition(), true); |
- |
- setEndingSelection(VisibleSelection(destination, originalIsDirectional)); |
- ASSERT(endingSelection().isCaretOrRange()); |
- ReplaceSelectionCommand::CommandOptions options = ReplaceSelectionCommand::SelectReplacement | ReplaceSelectionCommand::MovingParagraph; |
- if (!preserveStyle) |
- options |= ReplaceSelectionCommand::MatchStyle; |
- applyCommandToComposite(ReplaceSelectionCommand::create(document(), fragment, options)); |
- |
- document().frame()->spellChecker().markMisspellingsAndBadGrammar(endingSelection()); |
- |
- // If the selection is in an empty paragraph, restore styles from the old empty paragraph to the new empty paragraph. |
- bool selectionIsEmptyParagraph = endingSelection().isCaret() && isStartOfParagraph(endingSelection().visibleStart()) && isEndOfParagraph(endingSelection().visibleStart()); |
- if (styleInEmptyParagraph && selectionIsEmptyParagraph) |
- applyStyle(styleInEmptyParagraph.get()); |
- |
- if (!preserveSelection || startIndex == -1) |
- return; |
- Element* documentElement = document().documentElement(); |
- if (!documentElement) |
- return; |
- // Fragment creation (using createMarkup) incorrectly uses regular spaces |
- // instead of nbsps for some spaces that were rendered (11475), which causes |
- // spaces to be collapsed during the move operation. This results in a call |
- // to rangeFromLocationAndLength with a location past the end of the |
- // document (which will return null). |
- EphemeralRange startRange = PlainTextRange(destinationIndex + startIndex).createRangeForSelection(*documentElement); |
- if (startRange.isNull()) |
- return; |
- EphemeralRange endRange = PlainTextRange(destinationIndex + endIndex).createRangeForSelection(*documentElement); |
- if (endRange.isNull()) |
- return; |
- setEndingSelection(VisibleSelection(startRange.startPosition(), endRange.startPosition(), DOWNSTREAM, originalIsDirectional)); |
-} |
- |
-// FIXME: Send an appropriate shouldDeleteRange call. |
-bool CompositeEditCommand::breakOutOfEmptyListItem() |
-{ |
- RefPtrWillBeRawPtr<Node> emptyListItem = enclosingEmptyListItem(endingSelection().visibleStart()); |
- if (!emptyListItem) |
- return false; |
- |
- RefPtrWillBeRawPtr<EditingStyle> style = EditingStyle::create(endingSelection().start()); |
- style->mergeTypingStyle(&document()); |
- |
- RefPtrWillBeRawPtr<ContainerNode> listNode = emptyListItem->parentNode(); |
- // FIXME: Can't we do something better when the immediate parent wasn't a list node? |
- if (!listNode |
- || (!isHTMLUListElement(*listNode) && !isHTMLOListElement(*listNode)) |
- || !listNode->hasEditableStyle() |
- || listNode == emptyListItem->rootEditableElement()) |
- return false; |
- |
- RefPtrWillBeRawPtr<HTMLElement> newBlock = nullptr; |
- if (ContainerNode* blockEnclosingList = listNode->parentNode()) { |
- if (isHTMLLIElement(*blockEnclosingList)) { // listNode is inside another list item |
- if (visiblePositionAfterNode(*blockEnclosingList).deepEquivalent() == visiblePositionAfterNode(*listNode).deepEquivalent()) { |
- // If listNode appears at the end of the outer list item, then move listNode outside of this list item |
- // e.g. <ul><li>hello <ul><li><br></li></ul> </li></ul> should become <ul><li>hello</li> <ul><li><br></li></ul> </ul> after this section |
- // If listNode does NOT appear at the end, then we should consider it as a regular paragraph. |
- // e.g. <ul><li> <ul><li><br></li></ul> hello</li></ul> should become <ul><li> <div><br></div> hello</li></ul> at the end |
- splitElement(toElement(blockEnclosingList), listNode); |
- removeNodePreservingChildren(listNode->parentNode()); |
- newBlock = createListItemElement(document()); |
- } |
- // If listNode does NOT appear at the end of the outer list item, then behave as if in a regular paragraph. |
- } else if (isHTMLOListElement(*blockEnclosingList) || isHTMLUListElement(*blockEnclosingList)) { |
- newBlock = createListItemElement(document()); |
- } |
- } |
- if (!newBlock) |
- newBlock = createDefaultParagraphElement(document()); |
- |
- RefPtrWillBeRawPtr<Node> previousListNode = emptyListItem->isElementNode() ? ElementTraversal::previousSibling(*emptyListItem): emptyListItem->previousSibling(); |
- RefPtrWillBeRawPtr<Node> nextListNode = emptyListItem->isElementNode() ? ElementTraversal::nextSibling(*emptyListItem): emptyListItem->nextSibling(); |
- if (isListItem(nextListNode.get()) || isHTMLListElement(nextListNode.get())) { |
- // If emptyListItem follows another list item or nested list, split the list node. |
- if (isListItem(previousListNode.get()) || isHTMLListElement(previousListNode.get())) |
- splitElement(toElement(listNode), emptyListItem); |
- |
- // If emptyListItem is followed by other list item or nested list, then insert newBlock before the list node. |
- // Because we have splitted the element, emptyListItem is the first element in the list node. |
- // i.e. insert newBlock before ul or ol whose first element is emptyListItem |
- insertNodeBefore(newBlock, listNode); |
- removeNode(emptyListItem); |
- } else { |
- // When emptyListItem does not follow any list item or nested list, insert newBlock after the enclosing list node. |
- // Remove the enclosing node if emptyListItem is the only child; otherwise just remove emptyListItem. |
- insertNodeAfter(newBlock, listNode); |
- removeNode(isListItem(previousListNode.get()) || isHTMLListElement(previousListNode.get()) ? emptyListItem.get() : listNode.get()); |
- } |
- |
- appendBlockPlaceholder(newBlock); |
- setEndingSelection(VisibleSelection(firstPositionInNode(newBlock.get()), DOWNSTREAM, endingSelection().isDirectional())); |
- |
- style->prepareToApplyAt(endingSelection().start()); |
- if (!style->isEmpty()) |
- applyStyle(style.get()); |
- |
- return true; |
-} |
- |
-// If the caret is in an empty quoted paragraph, and either there is nothing before that |
-// paragraph, or what is before is unquoted, and the user presses delete, unquote that paragraph. |
-bool CompositeEditCommand::breakOutOfEmptyMailBlockquotedParagraph() |
-{ |
- if (!endingSelection().isCaret()) |
- return false; |
- |
- VisiblePosition caret(endingSelection().visibleStart()); |
- HTMLQuoteElement* highestBlockquote = toHTMLQuoteElement(highestEnclosingNodeOfType(caret.deepEquivalent(), &isMailHTMLBlockquoteElement)); |
- if (!highestBlockquote) |
- return false; |
- |
- if (!isStartOfParagraph(caret) || !isEndOfParagraph(caret)) |
- return false; |
- |
- VisiblePosition previous(caret.previous(CannotCrossEditingBoundary)); |
- // Only move forward if there's nothing before the caret, or if there's unquoted content before it. |
- if (enclosingNodeOfType(previous.deepEquivalent(), &isMailHTMLBlockquoteElement)) |
- return false; |
- |
- RefPtrWillBeRawPtr<HTMLBRElement> br = createBreakElement(document()); |
- // We want to replace this quoted paragraph with an unquoted one, so insert a br |
- // to hold the caret before the highest blockquote. |
- insertNodeBefore(br, highestBlockquote); |
- VisiblePosition atBR(positionBeforeNode(br.get())); |
- // If the br we inserted collapsed, for example foo<br><blockquote>...</blockquote>, insert |
- // a second one. |
- if (!isStartOfParagraph(atBR)) |
- insertNodeBefore(createBreakElement(document()), br); |
- setEndingSelection(VisibleSelection(atBR, endingSelection().isDirectional())); |
- |
- // If this is an empty paragraph there must be a line break here. |
- if (!lineBreakExistsAtVisiblePosition(caret)) |
- return false; |
- |
- Position caretPos(caret.deepEquivalent().downstream()); |
- // A line break is either a br or a preserved newline. |
- ASSERT(isHTMLBRElement(caretPos.anchorNode()) || (caretPos.anchorNode()->isTextNode() && caretPos.anchorNode()->layoutObject()->style()->preserveNewline())); |
- |
- if (isHTMLBRElement(*caretPos.anchorNode())) { |
- removeNodeAndPruneAncestors(caretPos.anchorNode()); |
- } else if (caretPos.anchorNode()->isTextNode()) { |
- ASSERT(caretPos.computeOffsetInContainerNode() == 0); |
- Text* textNode = toText(caretPos.anchorNode()); |
- ContainerNode* parentNode = textNode->parentNode(); |
- // The preserved newline must be the first thing in the node, since otherwise the previous |
- // paragraph would be quoted, and we verified that it wasn't above. |
- deleteTextFromNode(textNode, 0, 1); |
- prune(parentNode); |
- } |
- |
- return true; |
-} |
- |
-// Operations use this function to avoid inserting content into an anchor when at the start or the end of |
-// that anchor, as in NSTextView. |
-// FIXME: This is only an approximation of NSTextViews insertion behavior, which varies depending on how |
-// the caret was made. |
-Position CompositeEditCommand::positionAvoidingSpecialElementBoundary(const Position& original) |
-{ |
- if (original.isNull()) |
- return original; |
- |
- VisiblePosition visiblePos(original); |
- Element* enclosingAnchor = enclosingAnchorElement(original); |
- Position result = original; |
- |
- if (!enclosingAnchor) |
- return result; |
- |
- // Don't avoid block level anchors, because that would insert content into the wrong paragraph. |
- if (enclosingAnchor && !isBlock(enclosingAnchor)) { |
- VisiblePosition firstInAnchor(firstPositionInNode(enclosingAnchor)); |
- VisiblePosition lastInAnchor(lastPositionInNode(enclosingAnchor)); |
- // If visually just after the anchor, insert *inside* the anchor unless it's the last |
- // VisiblePosition in the document, to match NSTextView. |
- if (visiblePos.deepEquivalent() == lastInAnchor.deepEquivalent()) { |
- // Make sure anchors are pushed down before avoiding them so that we don't |
- // also avoid structural elements like lists and blocks (5142012). |
- if (original.anchorNode() != enclosingAnchor && original.anchorNode()->parentNode() != enclosingAnchor) { |
- pushAnchorElementDown(enclosingAnchor); |
- enclosingAnchor = enclosingAnchorElement(original); |
- if (!enclosingAnchor) |
- return original; |
- } |
- // Don't insert outside an anchor if doing so would skip over a line break. It would |
- // probably be safe to move the line break so that we could still avoid the anchor here. |
- Position downstream(visiblePos.deepEquivalent().downstream()); |
- if (lineBreakExistsAtVisiblePosition(visiblePos) && downstream.anchorNode()->isDescendantOf(enclosingAnchor)) |
- return original; |
- |
- result = positionInParentAfterNode(*enclosingAnchor); |
- } |
- // If visually just before an anchor, insert *outside* the anchor unless it's the first |
- // VisiblePosition in a paragraph, to match NSTextView. |
- if (visiblePos.deepEquivalent() == firstInAnchor.deepEquivalent()) { |
- // Make sure anchors are pushed down before avoiding them so that we don't |
- // also avoid structural elements like lists and blocks (5142012). |
- if (original.anchorNode() != enclosingAnchor && original.anchorNode()->parentNode() != enclosingAnchor) { |
- pushAnchorElementDown(enclosingAnchor); |
- enclosingAnchor = enclosingAnchorElement(original); |
- } |
- if (!enclosingAnchor) |
- return original; |
- |
- result = positionInParentBeforeNode(*enclosingAnchor); |
- } |
- } |
- |
- if (result.isNull() || !editableRootForPosition(result)) |
- result = original; |
- |
- return result; |
-} |
- |
-// Splits the tree parent by parent until we reach the specified ancestor. We use VisiblePositions |
-// to determine if the split is necessary. Returns the last split node. |
-PassRefPtrWillBeRawPtr<Node> CompositeEditCommand::splitTreeToNode(Node* start, Node* end, bool shouldSplitAncestor) |
-{ |
- ASSERT(start); |
- ASSERT(end); |
- ASSERT(start != end); |
- |
- if (shouldSplitAncestor && end->parentNode()) |
- end = end->parentNode(); |
- if (!start->isDescendantOf(end)) |
- return end; |
- |
- RefPtrWillBeRawPtr<Node> endNode = end; |
- RefPtrWillBeRawPtr<Node> node = nullptr; |
- for (node = start; node->parentNode() != endNode; node = node->parentNode()) { |
- Element* parentElement = node->parentElement(); |
- if (!parentElement) |
- break; |
- // Do not split a node when doing so introduces an empty node. |
- VisiblePosition positionInParent(firstPositionInNode(parentElement)); |
- VisiblePosition positionInNode(firstPositionInOrBeforeNode(node.get())); |
- if (positionInParent.deepEquivalent() != positionInNode.deepEquivalent()) |
- splitElement(parentElement, node); |
- } |
- |
- return node.release(); |
-} |
- |
-DEFINE_TRACE(CompositeEditCommand) |
-{ |
- visitor->trace(m_commands); |
- visitor->trace(m_composition); |
- EditCommand::trace(visitor); |
-} |
- |
-} // namespace blink |