| /* |
| * Copyright (C) 2005, 2006, 2007, 2013 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. |
| */ |
| |
| #include "config.h" |
| #include "EditCommand.h" |
| |
| #include "AXObjectCache.h" |
| #include "CompositeEditCommand.h" |
| #include "DocumentInlines.h" |
| #include "Editing.h" |
| #include "Editor.h" |
| #include "Element.h" |
| #include "HTMLTextFormControlElement.h" |
| #include "NodeTraversal.h" |
| |
| namespace WebCore { |
| |
| ASCIILiteral inputTypeNameForEditingAction(EditAction action) |
| { |
| switch (action) { |
| case EditAction::Justify: |
| return "formatJustifyFull"_s; |
| case EditAction::AlignLeft: |
| return "formatJustifyLeft"_s; |
| case EditAction::AlignRight: |
| return "formatJustifyRight"_s; |
| case EditAction::Center: |
| return "formatJustifyCenter"_s; |
| case EditAction::Subscript: |
| return "formatSubscript"_s; |
| case EditAction::Superscript: |
| return "formatSuperscript"_s; |
| case EditAction::Underline: |
| return "formatUnderline"_s; |
| case EditAction::StrikeThrough: |
| return "formatStrikeThrough"_s; |
| case EditAction::SetColor: |
| return "formatFontColor"_s; |
| case EditAction::DeleteByDrag: |
| return "deleteByDrag"_s; |
| case EditAction::Cut: |
| return "deleteByCut"_s; |
| case EditAction::Bold: |
| return "formatBold"_s; |
| case EditAction::Italics: |
| return "formatItalic"_s; |
| case EditAction::Paste: |
| return "insertFromPaste"_s; |
| case EditAction::Delete: |
| case EditAction::TypingDeleteSelection: |
| return "deleteContent"_s; |
| case EditAction::TypingDeleteBackward: |
| return "deleteContentBackward"_s; |
| case EditAction::TypingDeleteForward: |
| return "deleteContentForward"_s; |
| case EditAction::TypingDeleteWordBackward: |
| return "deleteWordBackward"_s; |
| case EditAction::TypingDeleteWordForward: |
| return "deleteWordForward"_s; |
| case EditAction::TypingDeleteLineBackward: |
| return "deleteHardLineBackward"_s; |
| case EditAction::TypingDeleteLineForward: |
| return "deleteHardLineForward"_s; |
| case EditAction::TypingDeletePendingComposition: |
| return "deleteCompositionText"_s; |
| case EditAction::TypingDeleteFinalComposition: |
| return "deleteByComposition"_s; |
| case EditAction::Insert: |
| case EditAction::TypingInsertText: |
| return "insertText"_s; |
| case EditAction::InsertReplacement: |
| return "insertReplacementText"_s; |
| case EditAction::InsertFromDrop: |
| return "insertFromDrop"_s; |
| case EditAction::TypingInsertLineBreak: |
| return "insertLineBreak"_s; |
| case EditAction::TypingInsertParagraph: |
| return "insertParagraph"_s; |
| case EditAction::InsertOrderedList: |
| return "insertOrderedList"_s; |
| case EditAction::InsertUnorderedList: |
| return "insertUnorderedList"_s; |
| case EditAction::TypingInsertPendingComposition: |
| return "insertCompositionText"_s; |
| case EditAction::TypingInsertFinalComposition: |
| return "insertFromComposition"_s; |
| case EditAction::Indent: |
| return "formatIndent"_s; |
| case EditAction::Outdent: |
| return "formatOutdent"_s; |
| case EditAction::SetInlineWritingDirection: |
| return "formatSetInlineTextDirection"_s; |
| case EditAction::SetBlockWritingDirection: |
| return "formatSetBlockTextDirection"_s; |
| case EditAction::CreateLink: |
| return "insertLink"_s; |
| default: |
| return ""_s; |
| } |
| } |
| |
| EditCommand::EditCommand(Document& document, EditAction editingAction) |
| : m_document { document } |
| , m_startingSelection { m_document->selection().selection() } |
| , m_endingSelection { m_startingSelection } |
| , m_editingAction { editingAction } |
| { |
| } |
| |
| EditCommand::EditCommand(Document& document, const VisibleSelection& startingSelection, const VisibleSelection& endingSelection) |
| : m_document { document } |
| , m_startingSelection { startingSelection } |
| , m_endingSelection { endingSelection } |
| { |
| } |
| |
| EditCommand::~EditCommand() = default; |
| |
| EditAction EditCommand::editingAction() const |
| { |
| return m_editingAction; |
| } |
| |
| static RefPtr<EditCommandComposition> compositionIfPossible(EditCommand& command) |
| { |
| if (!command.isCompositeEditCommand()) |
| return nullptr; |
| return static_cast<CompositeEditCommand&>(command).composition(); |
| } |
| |
| bool EditCommand::isEditingTextAreaOrTextInput() const |
| { |
| return enclosingTextFormControl(m_document->selection().selection().start()); |
| } |
| |
| void EditCommand::setStartingSelection(const VisibleSelection& selection) |
| { |
| for (RefPtr command = this; ; command = command->m_parent.get()) { |
| if (auto composition = compositionIfPossible(*command)) |
| composition->setStartingSelection(selection); |
| command->m_startingSelection = selection; |
| if (!command->m_parent || command->m_parent->isFirstCommand(command.get())) |
| break; |
| } |
| } |
| |
| void EditCommand::setEndingSelection(const VisibleSelection& selection) |
| { |
| for (RefPtr command = this; command; command = command->m_parent.get()) { |
| if (auto composition = compositionIfPossible(*command)) |
| composition->setEndingSelection(selection); |
| command->m_endingSelection = selection; |
| } |
| } |
| |
| void EditCommand::setParent(CompositeEditCommand* parent) |
| { |
| ASSERT((parent && !m_parent) || (!parent && m_parent)); |
| m_parent = parent; |
| if (parent) { |
| m_startingSelection = parent->m_endingSelection; |
| m_endingSelection = parent->m_endingSelection; |
| } |
| } |
| |
| void EditCommand::postTextStateChangeNotification(AXTextEditType type, const String& text) |
| { |
| if (!AXObjectCache::accessibilityEnabled()) |
| return; |
| postTextStateChangeNotification(type, text, m_document->selection().selection().start()); |
| } |
| |
| void EditCommand::postTextStateChangeNotification(AXTextEditType type, const String& text, const VisiblePosition& position) |
| { |
| if (!AXObjectCache::accessibilityEnabled()) |
| return; |
| if (!text.length()) |
| return; |
| auto* cache = document().existingAXObjectCache(); |
| if (!cache) |
| return; |
| RefPtr node { highestEditableRoot(position.deepEquivalent(), HasEditableAXRole) }; |
| cache->postTextStateChangeNotification(node.get(), type, text, position); |
| } |
| |
| SimpleEditCommand::SimpleEditCommand(Document& document, EditAction editingAction) |
| : EditCommand(document, editingAction) |
| { |
| } |
| |
| void SimpleEditCommand::doReapply() |
| { |
| doApply(); |
| } |
| |
| #ifndef NDEBUG |
| void SimpleEditCommand::addNodeAndDescendants(Node* startNode, HashSet<Ref<Node>>& nodes) |
| { |
| for (Node* node = startNode; node; node = NodeTraversal::next(*node, startNode)) |
| nodes.add(*node); |
| } |
| #endif |
| |
| } // namespace WebCore |