| /* |
| * 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 "Document.h" |
| #include "Editor.h" |
| #include "Element.h" |
| #include "EventNames.h" |
| #include "Frame.h" |
| #include "NodeTraversal.h" |
| #include "htmlediting.h" |
| |
| namespace WebCore { |
| |
| EditCommand::EditCommand(Document& document, EditAction editingAction) |
| : m_document(document) |
| , m_parent(0) |
| , m_editingAction(editingAction) |
| { |
| ASSERT(document.frame()); |
| setStartingSelection(m_document->frame()->selection().selection()); |
| setEndingSelection(m_startingSelection); |
| } |
| |
| EditCommand::EditCommand(Document& document, const VisibleSelection& startingSelection, const VisibleSelection& endingSelection) |
| : m_document(document) |
| , m_parent(0) |
| { |
| ASSERT(document.frame()); |
| setStartingSelection(startingSelection); |
| setEndingSelection(endingSelection); |
| } |
| |
| EditCommand::~EditCommand() |
| { |
| } |
| |
| Frame& EditCommand::frame() |
| { |
| ASSERT(document().frame()); |
| return *document().frame(); |
| } |
| |
| EditAction EditCommand::editingAction() const |
| { |
| return m_editingAction; |
| } |
| |
| static inline EditCommandComposition* compositionIfPossible(EditCommand* command) |
| { |
| if (!command->isCompositeEditCommand()) |
| return 0; |
| return toCompositeEditCommand(command)->composition(); |
| } |
| |
| void EditCommand::setStartingSelection(const VisibleSelection& s) |
| { |
| for (EditCommand* cmd = this; ; cmd = cmd->m_parent) { |
| if (EditCommandComposition* composition = compositionIfPossible(cmd)) { |
| ASSERT(cmd->isTopLevelCommand()); |
| composition->setStartingSelection(s); |
| } |
| cmd->m_startingSelection = s; |
| if (!cmd->m_parent || cmd->m_parent->isFirstCommand(cmd)) |
| break; |
| } |
| } |
| |
| void EditCommand::setEndingSelection(const VisibleSelection &s) |
| { |
| for (EditCommand* cmd = this; cmd; cmd = cmd->m_parent) { |
| if (EditCommandComposition* composition = compositionIfPossible(cmd)) { |
| ASSERT(cmd->isTopLevelCommand()); |
| composition->setEndingSelection(s); |
| } |
| cmd->m_endingSelection = s; |
| } |
| } |
| |
| void EditCommand::setParent(CompositeEditCommand* parent) |
| { |
| ASSERT((parent && !m_parent) || (!parent && m_parent)); |
| ASSERT(!parent || !isCompositeEditCommand() || !toCompositeEditCommand(this)->composition()); |
| m_parent = parent; |
| if (parent) { |
| m_startingSelection = parent->m_endingSelection; |
| m_endingSelection = parent->m_endingSelection; |
| } |
| } |
| |
| AXTextEditType EditCommand::applyEditType() const |
| { |
| switch (editingAction()) { |
| case EditActionCut: |
| return AXTextEditTypeCut; |
| case EditActionDelete: |
| return AXTextEditTypeDelete; |
| case EditActionDictation: |
| return AXTextEditTypeDictation; |
| case EditActionInsert: |
| return AXTextEditTypeInsert; |
| case EditActionPaste: |
| return AXTextEditTypePaste; |
| case EditActionTyping: |
| return AXTextEditTypeTyping; |
| case EditActionSetColor: |
| case EditActionSetBackgroundColor: |
| case EditActionTurnOffKerning: |
| case EditActionTightenKerning: |
| case EditActionLoosenKerning: |
| case EditActionUseStandardKerning: |
| case EditActionTurnOffLigatures: |
| case EditActionUseStandardLigatures: |
| case EditActionUseAllLigatures: |
| case EditActionRaiseBaseline: |
| case EditActionLowerBaseline: |
| case EditActionSetTraditionalCharacterShape: |
| case EditActionSetFont: |
| case EditActionChangeAttributes: |
| case EditActionAlignLeft: |
| case EditActionAlignRight: |
| case EditActionCenter: |
| case EditActionJustify: |
| case EditActionSetWritingDirection: |
| case EditActionSubscript: |
| case EditActionSuperscript: |
| case EditActionUnderline: |
| case EditActionOutline: |
| case EditActionUnscript: |
| case EditActionBold: |
| case EditActionItalics: |
| case EditActionFormatBlock: |
| case EditActionIndent: |
| case EditActionOutdent: |
| return AXTextEditTypeAttributesChange; |
| // Include default case for unhandled EditAction cases. |
| default: |
| break; |
| } |
| return AXTextEditTypeUnknown; |
| } |
| |
| AXTextEditType EditCommand::unapplyEditType() const |
| { |
| switch (applyEditType()) { |
| case AXTextEditTypeUnknown: |
| return AXTextEditTypeUnknown; |
| case AXTextEditTypeDelete: |
| case AXTextEditTypeCut: |
| return AXTextEditTypeInsert; |
| case AXTextEditTypeInsert: |
| case AXTextEditTypeTyping: |
| case AXTextEditTypeDictation: |
| case AXTextEditTypePaste: |
| return AXTextEditTypeDelete; |
| case AXTextEditTypeAttributesChange: |
| return AXTextEditTypeAttributesChange; |
| } |
| return AXTextEditTypeUnknown; |
| } |
| |
| bool EditCommand::shouldPostAccessibilityNotification() const |
| { |
| return AXObjectCache::accessibilityEnabled() && editingAction() != EditActionUnspecified; |
| } |
| |
| SimpleEditCommand::SimpleEditCommand(Document& document, EditAction editingAction) |
| : EditCommand(document, editingAction) |
| { |
| } |
| |
| void SimpleEditCommand::doReapply() |
| { |
| doApply(); |
| } |
| |
| #ifndef NDEBUG |
| void SimpleEditCommand::addNodeAndDescendants(Node* startNode, HashSet<Node*>& nodes) |
| { |
| for (Node* node = startNode; node; node = NodeTraversal::next(*node, startNode)) |
| nodes.add(node); |
| } |
| #endif |
| |
| void SimpleEditCommand::notifyAccessibilityForTextChange(Node* node, AXTextEditType type, const String& text, const VisiblePosition& position) |
| { |
| if (!AXObjectCache::accessibilityEnabled()) |
| return; |
| AXObjectCache* cache = document().existingAXObjectCache(); |
| if (!cache) |
| return; |
| cache->postTextStateChangeNotification(node, type, text, position); |
| } |
| |
| } // namespace WebCore |