/* * Copyright (C) 2005, 2006, 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 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 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. */ #pragma once #include "AXObjectCache.h" #include "EditCommand.h" #include "CSSPropertyNames.h" #include "UndoStep.h" #include namespace WebCore { class EditingStyle; class DataTransfer; class HTMLElement; class StaticRange; class StyledElement; class Text; class AccessibilityUndoReplacedText { public: AccessibilityUndoReplacedText() { } void configureRangeDeletedByReapplyWithStartingSelection(const VisibleSelection&); void configureRangeDeletedByReapplyWithEndingSelection(const VisibleSelection&); void setRangeDeletedByUnapply(const VisiblePositionIndexRange&); void captureTextForUnapply(); void captureTextForReapply(); void postTextStateChangeNotificationForUnapply(AXObjectCache*); void postTextStateChangeNotificationForReapply(AXObjectCache*); private: int indexForVisiblePosition(const VisiblePosition&, RefPtr&) const; String textDeletedByUnapply(); String textDeletedByReapply(); String m_replacedText; VisiblePositionIndexRange m_rangeDeletedByUnapply; VisiblePositionIndexRange m_rangeDeletedByReapply; }; class EditCommandComposition : public UndoStep { public: static Ref create(Document&, const VisibleSelection& startingSelection, const VisibleSelection& endingSelection, EditAction); void unapply() override; void reapply() override; EditAction editingAction() const override { return m_editAction; } void append(SimpleEditCommand*); bool wasCreateLinkCommand() const { return m_editAction == EditActionCreateLink; } const VisibleSelection& startingSelection() const { return m_startingSelection; } const VisibleSelection& endingSelection() const { return m_endingSelection; } void setStartingSelection(const VisibleSelection&); void setEndingSelection(const VisibleSelection&); Element* startingRootEditableElement() const { return m_startingRootEditableElement.get(); } Element* endingRootEditableElement() const { return m_endingRootEditableElement.get(); } void setRangeDeletedByUnapply(const VisiblePositionIndexRange&); #ifndef NDEBUG virtual void getNodesInCommand(HashSet&); #endif private: EditCommandComposition(Document&, const VisibleSelection& startingSelection, const VisibleSelection& endingSelection, EditAction); RefPtr m_document; VisibleSelection m_startingSelection; VisibleSelection m_endingSelection; Vector> m_commands; RefPtr m_startingRootEditableElement; RefPtr m_endingRootEditableElement; AccessibilityUndoReplacedText m_replacedText; EditAction m_editAction; }; class CompositeEditCommand : public EditCommand { public: virtual ~CompositeEditCommand(); void apply(); bool isFirstCommand(EditCommand* command) { return !m_commands.isEmpty() && m_commands.first() == command; } EditCommandComposition* composition() const; EditCommandComposition& ensureComposition(); virtual bool isCreateLinkCommand() const; virtual bool isTypingCommand() const; virtual bool isDictationCommand() const { return false; } virtual bool preservesTypingStyle() const; virtual bool shouldRetainAutocorrectionIndicator() const; virtual void setShouldRetainAutocorrectionIndicator(bool); virtual bool shouldStopCaretBlinking() const { return false; } virtual String inputEventTypeName() const; virtual String inputEventData() const { return { }; } virtual bool isBeforeInputEventCancelable() const { return true; } virtual bool shouldDispatchInputEvents() const { return true; } Vector> targetRangesForBindings() const; virtual RefPtr inputEventDataTransfer() const; protected: explicit CompositeEditCommand(Document&, EditAction = EditActionUnspecified); // If willApplyCommand returns false, we won't proceed with applying the command. virtual bool willApplyCommand(); virtual void didApplyCommand(); virtual Vector> targetRanges() const; // // sugary-sweet convenience functions to help create and apply edit commands in composite commands // void appendNode(PassRefPtr, PassRefPtr parent); void applyCommandToComposite(PassRefPtr); void applyCommandToComposite(PassRefPtr, const VisibleSelection&); void applyStyle(const EditingStyle*, EditAction = EditActionChangeAttributes); void applyStyle(const EditingStyle*, const Position& start, const Position& end, EditAction = EditActionChangeAttributes); void applyStyledElement(PassRefPtr); void removeStyledElement(PassRefPtr); void deleteSelection(bool smartDelete = false, bool mergeBlocksAfterDelete = true, bool replace = false, bool expandForSpecialElements = true, bool sanitizeMarkup = true); void deleteSelection(const VisibleSelection&, bool smartDelete = false, bool mergeBlocksAfterDelete = true, bool replace = false, bool expandForSpecialElements = true, bool sanitizeMarkup = true); virtual void deleteTextFromNode(PassRefPtr, unsigned offset, unsigned count); void inputText(const String&, bool selectInsertedText = false); bool isRemovableBlock(const Node*); void insertNodeAfter(PassRefPtr, PassRefPtr refChild); void insertNodeAt(PassRefPtr, const Position&); void insertNodeAtTabSpanPosition(PassRefPtr, const Position&); void insertNodeBefore(PassRefPtr, PassRefPtr refChild, ShouldAssumeContentIsAlwaysEditable = DoNotAssumeContentIsAlwaysEditable); void insertParagraphSeparator(bool useDefaultParagraphElement = false, bool pasteBlockqutoeIntoUnquotedArea = false); void insertLineBreak(); void insertTextIntoNode(PassRefPtr, unsigned offset, const String& text); void mergeIdenticalElements(PassRefPtr, PassRefPtr); void rebalanceWhitespace(); void rebalanceWhitespaceAt(const Position&); void rebalanceWhitespaceOnTextSubstring(PassRefPtr, int startOffset, int endOffset); void prepareWhitespaceAtPositionForSplit(Position&); bool canRebalance(const Position&) const; bool shouldRebalanceLeadingWhitespaceFor(const String&) const; void removeCSSProperty(PassRefPtr, CSSPropertyID); void removeNodeAttribute(PassRefPtr, const QualifiedName& attribute); void removeChildrenInRange(PassRefPtr, unsigned from, unsigned to); virtual void removeNode(PassRefPtr, ShouldAssumeContentIsAlwaysEditable = DoNotAssumeContentIsAlwaysEditable); HTMLElement* replaceElementWithSpanPreservingChildrenAndAttributes(PassRefPtr); void removeNodePreservingChildren(PassRefPtr, ShouldAssumeContentIsAlwaysEditable = DoNotAssumeContentIsAlwaysEditable); void removeNodeAndPruneAncestors(PassRefPtr); void moveRemainingSiblingsToNewParent(Node*, Node* pastLastNodeToMove, PassRefPtr prpNewParent); void updatePositionForNodeRemovalPreservingChildren(Position&, Node&); void prune(PassRefPtr); void replaceTextInNode(PassRefPtr, unsigned offset, unsigned count, const String& replacementText); Position replaceSelectedTextInNode(const String&); void replaceTextInNodePreservingMarkers(PassRefPtr, unsigned offset, unsigned count, const String& replacementText); Position positionOutsideTabSpan(const Position&); void setNodeAttribute(PassRefPtr, const QualifiedName& attribute, const AtomicString& value); void splitElement(PassRefPtr, PassRefPtr atChild); void splitTextNode(PassRefPtr, unsigned offset); void splitTextNodeContainingElement(PassRefPtr, unsigned offset); void wrapContentsInDummySpan(Element&); void deleteInsignificantText(PassRefPtr, unsigned start, unsigned end); void deleteInsignificantText(const Position& start, const Position& end); void deleteInsignificantTextDownstream(const Position&); RefPtr appendBlockPlaceholder(PassRefPtr); RefPtr insertBlockPlaceholder(const Position&); RefPtr addBlockPlaceholderIfNeeded(Element*); void removePlaceholderAt(const Position&); Ref insertNewDefaultParagraphElementAt(const Position&); RefPtr moveParagraphContentsToNewBlockIfNecessary(const Position&); void pushAnchorElementDown(Element&); void moveParagraph(const VisiblePosition&, const VisiblePosition&, const VisiblePosition&, bool preserveSelection = false, bool preserveStyle = true); void moveParagraphs(const VisiblePosition&, const VisiblePosition&, const VisiblePosition&, bool preserveSelection = false, bool preserveStyle = true); void moveParagraphWithClones(const VisiblePosition& startOfParagraphToMove, const VisiblePosition& endOfParagraphToMove, Element* blockElement, Node* outerNode); void cloneParagraphUnderNewElement(const Position& start, const Position& end, Node* outerNode, Element* blockElement); void cleanupAfterDeletion(VisiblePosition destination = VisiblePosition()); std::optional shouldBreakOutOfEmptyListItem() const; bool breakOutOfEmptyListItem(); bool breakOutOfEmptyMailBlockquotedParagraph(); Position positionAvoidingSpecialElementBoundary(const Position&); RefPtr splitTreeToNode(Node*, Node*, bool splitAncestor = false); Vector> m_commands; private: bool isCompositeEditCommand() const override { return true; } RefPtr m_composition; }; void applyCommand(PassRefPtr); inline CompositeEditCommand* toCompositeEditCommand(EditCommand* command) { ASSERT(command); ASSERT_WITH_SECURITY_IMPLICATION(command->isCompositeEditCommand()); return static_cast(command); } } // namespace WebCore