diff options
Diffstat (limited to 'WebCore/editing/CompositeEditCommand.cpp')
-rw-r--r-- | WebCore/editing/CompositeEditCommand.cpp | 960 |
1 files changed, 0 insertions, 960 deletions
diff --git a/WebCore/editing/CompositeEditCommand.cpp b/WebCore/editing/CompositeEditCommand.cpp deleted file mode 100644 index 3700023..0000000 --- a/WebCore/editing/CompositeEditCommand.cpp +++ /dev/null @@ -1,960 +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 "CompositeEditCommand.h" - -#include "AppendNodeCommand.h" -#include "ApplyStyleCommand.h" -#include "CSSComputedStyleDeclaration.h" -#include "CSSMutableStyleDeclaration.h" -#include "CharacterNames.h" -#include "DeleteFromTextNodeCommand.h" -#include "DeleteSelectionCommand.h" -#include "Document.h" -#include "DocumentFragment.h" -#include "EditorInsertAction.h" -#include "Element.h" -#include "HTMLNames.h" -#include "InlineTextBox.h" -#include "InsertIntoTextNodeCommand.h" -#include "InsertLineBreakCommand.h" -#include "InsertNodeBeforeCommand.h" -#include "InsertParagraphSeparatorCommand.h" -#include "InsertTextCommand.h" -#include "JoinTextNodesCommand.h" -#include "MergeIdenticalElementsCommand.h" -#include "Range.h" -#include "RemoveCSSPropertyCommand.h" -#include "RemoveNodeAttributeCommand.h" -#include "RemoveNodeCommand.h" -#include "RemoveNodePreservingChildrenCommand.h" -#include "ReplaceSelectionCommand.h" -#include "SetNodeAttributeCommand.h" -#include "SplitElementCommand.h" -#include "SplitTextNodeCommand.h" -#include "SplitTextNodeContainingElementCommand.h" -#include "Text.h" -#include "TextIterator.h" -#include "WrapContentsInDummySpanCommand.h" -#include "htmlediting.h" -#include "markup.h" -#include "visible_units.h" - -using namespace std; - -namespace WebCore { - -using namespace HTMLNames; - -CompositeEditCommand::CompositeEditCommand(Document *document) - : EditCommand(document) -{ -} - -void CompositeEditCommand::doUnapply() -{ - size_t size = m_commands.size(); - for (size_t i = size; i != 0; --i) - m_commands[i - 1]->unapply(); -} - -void CompositeEditCommand::doReapply() -{ - size_t size = m_commands.size(); - for (size_t i = 0; i != size; ++i) - m_commands[i]->reapply(); -} - -// -// sugary-sweet convenience functions to help create and apply edit commands in composite commands -// -void CompositeEditCommand::applyCommandToComposite(PassRefPtr<EditCommand> cmd) -{ - cmd->setParent(this); - cmd->apply(); - m_commands.append(cmd); -} - -void CompositeEditCommand::applyStyle(CSSStyleDeclaration* style, EditAction editingAction) -{ - applyCommandToComposite(ApplyStyleCommand::create(document(), style, editingAction)); -} - -void CompositeEditCommand::applyStyle(CSSStyleDeclaration* style, const Position& start, const Position& end, EditAction editingAction) -{ - applyCommandToComposite(ApplyStyleCommand::create(document(), style, start, end, editingAction)); -} - -void CompositeEditCommand::applyStyledElement(Element* element) -{ - applyCommandToComposite(ApplyStyleCommand::create(element, false)); -} - -void CompositeEditCommand::removeStyledElement(Element* element) -{ - applyCommandToComposite(ApplyStyleCommand::create(element, true)); -} - -void CompositeEditCommand::insertParagraphSeparator(bool useDefaultParagraphElement) -{ - applyCommandToComposite(InsertParagraphSeparatorCommand::create(document(), useDefaultParagraphElement)); -} - -void CompositeEditCommand::insertLineBreak() -{ - applyCommandToComposite(InsertLineBreakCommand::create(document())); -} - -void CompositeEditCommand::insertNodeBefore(Node* insertChild, Node* refChild) -{ - ASSERT(!refChild->hasTagName(bodyTag)); - applyCommandToComposite(InsertNodeBeforeCommand::create(insertChild, refChild)); -} - -void CompositeEditCommand::insertNodeAfter(Node* insertChild, Node* refChild) -{ - ASSERT(!refChild->hasTagName(bodyTag)); - if (refChild->parentNode()->lastChild() == refChild) - appendNode(insertChild, refChild->parentNode()); - else { - ASSERT(refChild->nextSibling()); - insertNodeBefore(insertChild, refChild->nextSibling()); - } -} - -void CompositeEditCommand::insertNodeAt(Node* insertChild, const Position& editingPosition) -{ - ASSERT(isEditablePosition(editingPosition)); - // For editing positions like [table, 0], insert before the table, - // likewise for replaced elements, brs, etc. - Position p = rangeCompliantEquivalent(editingPosition); - Node* refChild = p.node(); - int offset = p.offset(); - - 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, refChild); - } else if (caretMinOffset(refChild) >= offset) { - insertNodeBefore(insertChild, refChild); - } else if (refChild->isTextNode() && caretMaxOffset(refChild) > offset) { - splitTextNode(static_cast<Text *>(refChild), offset); - insertNodeBefore(insertChild, refChild); - } else { - insertNodeAfter(insertChild, refChild); - } -} - -void CompositeEditCommand::appendNode(Node* newChild, Node* parent) -{ - ASSERT(canHaveChildrenForEditing(parent)); - applyCommandToComposite(AppendNodeCommand::create(parent, newChild)); -} - -void CompositeEditCommand::removeChildrenInRange(Node* node, int from, int to) -{ - Node* nodeToRemove = node->childNode(from); - for (int i = from; i < to; i++) { - ASSERT(nodeToRemove); - Node* next = nodeToRemove->nextSibling(); - removeNode(nodeToRemove); - nodeToRemove = next; - } -} - -void CompositeEditCommand::removeNode(Node* removeChild) -{ - applyCommandToComposite(RemoveNodeCommand::create(removeChild)); -} - -void CompositeEditCommand::removeNodePreservingChildren(Node* removeChild) -{ - applyCommandToComposite(RemoveNodePreservingChildrenCommand::create(removeChild)); -} - -void CompositeEditCommand::removeNodeAndPruneAncestors(Node* node) -{ - RefPtr<Node> parent = node->parentNode(); - removeNode(node); - prune(parent); -} - -static bool hasARenderedDescendant(Node* node) -{ - Node* n = node->firstChild(); - while (n) { - if (n->renderer()) - return true; - n = n->traverseNextNode(node); - } - return false; -} - -void CompositeEditCommand::prune(PassRefPtr<Node> node) -{ - while (node) { - // If you change this rule you may have to add an updateLayout() here. - RenderObject* renderer = node->renderer(); - if (renderer && (!renderer->canHaveChildren() || hasARenderedDescendant(node.get()) || node->rootEditableElement() == node)) - return; - - RefPtr<Node> next = node->parentNode(); - removeNode(node.get()); - node = next; - } -} - -void CompositeEditCommand::splitTextNode(Text *text, int offset) -{ - applyCommandToComposite(SplitTextNodeCommand::create(text, offset)); -} - -void CompositeEditCommand::splitElement(Element* element, Node* atChild) -{ - applyCommandToComposite(SplitElementCommand::create(element, atChild)); -} - -void CompositeEditCommand::mergeIdenticalElements(Element* first, Element* second) -{ - ASSERT(!first->isDescendantOf(second) && second != first); - if (first->nextSibling() != second) { - removeNode(second); - insertNodeAfter(second, first); - } - applyCommandToComposite(MergeIdenticalElementsCommand::create(first, second)); -} - -void CompositeEditCommand::wrapContentsInDummySpan(Element* element) -{ - applyCommandToComposite(WrapContentsInDummySpanCommand::create(element)); -} - -void CompositeEditCommand::splitTextNodeContainingElement(Text *text, int offset) -{ - applyCommandToComposite(SplitTextNodeContainingElementCommand::create(text, offset)); -} - -void CompositeEditCommand::joinTextNodes(Text *text1, Text *text2) -{ - applyCommandToComposite(JoinTextNodesCommand::create(text1, text2)); -} - -void CompositeEditCommand::inputText(const String &text, bool selectInsertedText) -{ - int offset = 0; - int length = text.length(); - RefPtr<Range> startRange = Range::create(document(), Position(document()->documentElement(), 0), endingSelection().start()); - int startIndex = TextIterator::rangeLength(startRange.get()); - int newline; - do { - newline = text.find('\n', offset); - if (newline != offset) { - RefPtr<InsertTextCommand> command = InsertTextCommand::create(document()); - applyCommandToComposite(command); - int substringLength = newline == -1 ? length - offset : newline - offset; - command->input(text.substring(offset, substringLength), false); - } - if (newline != -1) - insertLineBreak(); - - offset = newline + 1; - } while (newline != -1 && offset != length); - - if (selectInsertedText) { - RefPtr<Range> selectedRange = TextIterator::rangeFromLocationAndLength(document()->documentElement(), startIndex, length); - setEndingSelection(Selection(selectedRange.get())); - } -} - -void CompositeEditCommand::insertTextIntoNode(Text *node, int offset, const String &text) -{ - applyCommandToComposite(InsertIntoTextNodeCommand::create(node, offset, text)); -} - -void CompositeEditCommand::deleteTextFromNode(Text *node, int offset, int count) -{ - applyCommandToComposite(DeleteFromTextNodeCommand::create(node, offset, count)); -} - -void CompositeEditCommand::replaceTextInNode(Text *node, int offset, int count, const String &replacementText) -{ - applyCommandToComposite(DeleteFromTextNodeCommand::create(node, offset, count)); - applyCommandToComposite(InsertIntoTextNodeCommand::create(node, offset, replacementText)); -} - -Position CompositeEditCommand::positionOutsideTabSpan(const Position& pos) -{ - if (!isTabSpanTextNode(pos.node())) - return pos; - - Node* tabSpan = tabSpanNode(pos.node()); - - if (pos.offset() <= caretMinOffset(pos.node())) - return positionBeforeNode(tabSpan); - - if (pos.offset() >= caretMaxOffset(pos.node())) - return positionAfterNode(tabSpan); - - splitTextNodeContainingElement(static_cast<Text *>(pos.node()), pos.offset()); - return positionBeforeNode(tabSpan); -} - -void CompositeEditCommand::insertNodeAtTabSpanPosition(Node* node, const Position& pos) -{ - // insert node before, after, or at split of tab span - Position insertPos = positionOutsideTabSpan(pos); - insertNodeAt(node, insertPos); -} - -void CompositeEditCommand::deleteSelection(bool smartDelete, bool mergeBlocksAfterDelete, bool replace, bool expandForSpecialElements) -{ - if (endingSelection().isRange()) - applyCommandToComposite(DeleteSelectionCommand::create(document(), smartDelete, mergeBlocksAfterDelete, replace, expandForSpecialElements)); -} - -void CompositeEditCommand::deleteSelection(const Selection &selection, bool smartDelete, bool mergeBlocksAfterDelete, bool replace, bool expandForSpecialElements) -{ - if (selection.isRange()) - applyCommandToComposite(DeleteSelectionCommand::create(selection, smartDelete, mergeBlocksAfterDelete, replace, expandForSpecialElements)); -} - -void CompositeEditCommand::removeCSSProperty(CSSStyleDeclaration *decl, int property) -{ - applyCommandToComposite(RemoveCSSPropertyCommand::create(document(), decl, property)); -} - -void CompositeEditCommand::removeNodeAttribute(Element* element, const QualifiedName& attribute) -{ - if (element->getAttribute(attribute).isNull()) - return; - applyCommandToComposite(RemoveNodeAttributeCommand::create(element, attribute)); -} - -void CompositeEditCommand::setNodeAttribute(Element* element, const QualifiedName& attribute, const String &value) -{ - applyCommandToComposite(SetNodeAttributeCommand::create(element, attribute, value)); -} - -static inline bool isWhitespace(UChar c) -{ - return c == noBreakSpace || c == ' ' || c == '\n' || c == '\t'; -} - -// FIXME: Doesn't go into text nodes that contribute adjacent text (siblings, cousins, etc). -void CompositeEditCommand::rebalanceWhitespaceAt(const Position& position) -{ - Node* node = position.node(); - if (!node || !node->isTextNode()) - return; - Text* textNode = static_cast<Text*>(node); - - if (textNode->length() == 0) - return; - RenderObject* renderer = textNode->renderer(); - if (renderer && !renderer->style()->collapseWhiteSpace()) - return; - - String text = textNode->data(); - ASSERT(!text.isEmpty()); - - int offset = position.offset(); - // If neither text[offset] nor text[offset - 1] are some form of whitespace, do nothing. - if (!isWhitespace(text[offset])) { - offset--; - if (offset < 0 || !isWhitespace(text[offset])) - return; - } - - // Set upstream and downstream to define the extent of the whitespace surrounding text[offset]. - int upstream = offset; - while (upstream > 0 && isWhitespace(text[upstream - 1])) - upstream--; - - int downstream = offset; - while ((unsigned)downstream + 1 < text.length() && isWhitespace(text[downstream + 1])) - downstream++; - - int length = downstream - upstream + 1; - ASSERT(length > 0); - - VisiblePosition visibleUpstreamPos(Position(position.node(), upstream)); - VisiblePosition visibleDownstreamPos(Position(position.node(), downstream + 1)); - - 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() - 1); - - if (string != rebalancedString) - replaceTextInNode(textNode, upstream, length, rebalancedString); -} - -void CompositeEditCommand::prepareWhitespaceAtPositionForSplit(Position& position) -{ - Node* node = position.node(); - if (!node || !node->isTextNode()) - return; - Text* textNode = static_cast<Text*>(node); - - if (textNode->length() == 0) - return; - RenderObject* renderer = textNode->renderer(); - if (renderer && !renderer->style()->collapseWhiteSpace()) - return; - - // Delete collapsed whitespace so that inserting nbsps doesn't uncollapse it. - Position upstreamPos = position.upstream(); - deleteInsignificantText(position.upstream(), position.downstream()); - position = upstreamPos.downstream(); - - VisiblePosition visiblePos(position); - VisiblePosition previousVisiblePos(visiblePos.previous()); - Position previous(previousVisiblePos.deepEquivalent()); - - if (isCollapsibleWhitespace(previousVisiblePos.characterAfter()) && previous.node()->isTextNode() && !previous.node()->hasTagName(brTag)) - replaceTextInNode(static_cast<Text*>(previous.node()), previous.offset(), 1, nonBreakingSpaceString()); - if (isCollapsibleWhitespace(visiblePos.characterAfter()) && position.node()->isTextNode() && !position.node()->hasTagName(brTag)) - replaceTextInNode(static_cast<Text*>(position.node()), position.offset(), 1, nonBreakingSpaceString()); -} - -void CompositeEditCommand::rebalanceWhitespace() -{ - Selection selection = endingSelection(); - if (selection.isNone()) - return; - - rebalanceWhitespaceAt(selection.start()); - if (selection.isRange()) - rebalanceWhitespaceAt(selection.end()); -} - -void CompositeEditCommand::deleteInsignificantText(Text* textNode, int start, int end) -{ - if (!textNode || !textNode->renderer() || start >= end) - return; - - RenderText* textRenderer = static_cast<RenderText*>(textNode->renderer()); - InlineTextBox* box = textRenderer->firstTextBox(); - if (!box) { - // whole text node is empty - removeNode(textNode); - return; - } - - int length = textNode->length(); - if (start >= length || end > length) - return; - - int removed = 0; - InlineTextBox* prevBox = 0; - 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) { - int gapStart = prevBox ? prevBox->m_start + prevBox->m_len : 0; - if (end < gapStart) - // No more chance for any intersections - break; - - int gapEnd = box ? box->m_start : length; - bool indicesIntersect = start <= gapEnd && end >= gapStart; - int gapLen = gapEnd - gapStart; - if (indicesIntersect && gapLen > 0) { - gapStart = max(gapStart, start); - gapEnd = min(gapEnd, end); - if (str.isNull()) - str = textNode->string()->substring(start, end - start); - // remove text in the gap - str.remove(gapStart - start - removed, gapLen); - removed += gapLen; - } - - prevBox = box; - if (box) - box = box->nextTextBox(); - } - - 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 || (unsigned)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 (Range::compareBoundaryPoints(start, end) >= 0) - return; - - Node* next; - for (Node* node = start.node(); node; node = next) { - next = node->traverseNextNode(); - if (node->isTextNode()) { - Text* textNode = static_cast<Text*>(node); - int startOffset = node == start.node() ? start.offset() : 0; - int endOffset = node == end.node() ? end.offset() : textNode->length(); - deleteInsignificantText(textNode, startOffset, endOffset); - } - if (node == end.node()) - break; - } -} - -void CompositeEditCommand::deleteInsignificantTextDownstream(const Position& pos) -{ - Position end = VisiblePosition(pos, VP_DEFAULT_AFFINITY).next().deepEquivalent().downstream(); - deleteInsignificantText(pos, end); -} - -PassRefPtr<Node> CompositeEditCommand::appendBlockPlaceholder(Node* node) -{ - if (!node) - return 0; - - // Should assert isBlockFlow || isInlineFlow when deletion improves. See 4244964. - ASSERT(node->renderer()); - - RefPtr<Node> placeholder = createBlockPlaceholderElement(document()); - appendNode(placeholder.get(), node); - return placeholder.release(); -} - -PassRefPtr<Node> CompositeEditCommand::insertBlockPlaceholder(const Position& pos) -{ - if (pos.isNull()) - return 0; - - // Should assert isBlockFlow || isInlineFlow when deletion improves. See 4244964. - ASSERT(pos.node()->renderer()); - - RefPtr<Node> placeholder = createBlockPlaceholderElement(document()); - insertNodeAt(placeholder.get(), pos); - return placeholder.release(); -} - -PassRefPtr<Node> CompositeEditCommand::addBlockPlaceholderIfNeeded(Node* node) -{ - if (!node) - return 0; - - updateLayout(); - - RenderObject *renderer = node->renderer(); - if (!renderer || !renderer->isBlockFlow()) - return 0; - - // append the placeholder to make sure it follows - // any unrendered blocks - if (renderer->height() == 0 || (renderer->isListItem() && renderer->isEmpty())) - return appendBlockPlaceholder(node); - - return 0; -} - -// Removes '\n's and brs that will collapse when content is inserted just before them. -// FIXME: We shouldn't really have to remove placeholders, but removing them is a workaround for 9661. -void CompositeEditCommand::removePlaceholderAt(const VisiblePosition& visiblePosition) -{ - if (visiblePosition.isNull()) - return; - - Position p = visiblePosition.deepEquivalent().downstream(); - // If a br or '\n' is at the end of a block and not at the start of a paragraph, - // then it is superfluous, so adding content before a br or '\n' that is at - // the start of a paragraph will render it superfluous. - // FIXME: This doesn't remove placeholders at the end of anonymous blocks. - if (isEndOfBlock(visiblePosition) && isStartOfParagraph(visiblePosition)) { - if (p.node()->hasTagName(brTag) && p.offset() == 0) - removeNode(p.node()); - else if (lineBreakExistsAtPosition(visiblePosition)) - deleteTextFromNode(static_cast<Text*>(p.node()), p.offset(), 1); - } -} - -PassRefPtr<Node> CompositeEditCommand::moveParagraphContentsToNewBlockIfNecessary(const Position& pos) -{ - if (pos.isNull()) - return 0; - - updateLayout(); - - 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 paragraphStart = visibleParagraphStart.deepEquivalent().upstream(); - Position end = visibleEnd.deepEquivalent().upstream(); - - // If there are no VisiblePositions in the same block as pos then - // paragraphStart will be outside the paragraph - if (Range::compareBoundaryPoints(pos, paragraphStart) < 0) - return 0; - - // Perform some checks to see if we need to perform work in this function. - if (isBlock(paragraphStart.node())) { - if (isBlock(end.node())) { - if (!end.node()->isDescendantOf(paragraphStart.node())) { - // 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 0; - } - } - else if (enclosingBlock(end.node()) != paragraphStart.node()) { - // The visibleEnd. It must be an ancestor of the paragraph start. - // We can bail as we have a full block to work with. - ASSERT(paragraphStart.node()->isDescendantOf(enclosingBlock(end.node()))); - return 0; - } - else if (isEndOfDocument(visibleEnd)) { - // At the end of the document. We can bail here as well. - return 0; - } - } - - RefPtr<Node> newBlock = createDefaultParagraphElement(document()); - appendNode(createBreakElement(document()).get(), newBlock.get()); - insertNodeAt(newBlock.get(), paragraphStart); - - moveParagraphs(visibleParagraphStart, visibleParagraphEnd, VisiblePosition(Position(newBlock.get(), 0))); - - return newBlock.get(); -} - -void CompositeEditCommand::pushAnchorElementDown(Node* anchorNode) -{ - if (!anchorNode) - return; - - ASSERT(anchorNode->isLink()); - - setEndingSelection(Selection::selectionFromContentsOfNode(anchorNode)); - applyStyledElement(static_cast<Element*>(anchorNode)); - // Clones of anchorNode have been pushed down, now remove it. - if (anchorNode->inDocument()) - removeNodePreservingChildren(anchorNode); -} - -// We must push partially selected anchors down before creating or removing -// links from a selection to create fully selected chunks that can be removed. -// ApplyStyleCommand doesn't do this for us because styles can be nested. -// Anchors cannot be nested. -void CompositeEditCommand::pushPartiallySelectedAnchorElementsDown() -{ - Selection originalSelection = endingSelection(); - VisiblePosition visibleStart(originalSelection.start()); - VisiblePosition visibleEnd(originalSelection.end()); - - Node* startAnchor = enclosingAnchorElement(originalSelection.start()); - VisiblePosition startOfStartAnchor(Position(startAnchor, 0)); - if (startAnchor && startOfStartAnchor != visibleStart) - pushAnchorElementDown(startAnchor); - - Node* endAnchor = enclosingAnchorElement(originalSelection.end()); - VisiblePosition endOfEndAnchor(Position(endAnchor, 0)); - if (endAnchor && endOfEndAnchor != visibleEnd) - pushAnchorElementDown(endAnchor); - - ASSERT(originalSelection.start().node()->inDocument() && originalSelection.end().node()->inDocument()); - setEndingSelection(originalSelection); -} - -// This moves a paragraph preserving its style. -void CompositeEditCommand::moveParagraph(const VisiblePosition& startOfParagraphToMove, const VisiblePosition& endOfParagraphToMove, const VisiblePosition& destination, bool preserveSelection, bool preserveStyle) -{ - ASSERT(isStartOfParagraph(startOfParagraphToMove)); - ASSERT(isEndOfParagraph(endOfParagraphToMove)); - moveParagraphs(startOfParagraphToMove, endOfParagraphToMove, destination, preserveSelection, preserveStyle); -} - -void CompositeEditCommand::moveParagraphs(const VisiblePosition& startOfParagraphToMove, const VisiblePosition& endOfParagraphToMove, const VisiblePosition& destination, bool preserveSelection, bool preserveStyle) -{ - if (startOfParagraphToMove == destination) - return; - - int startIndex = -1; - int endIndex = -1; - int destinationIndex = -1; - if (preserveSelection && !endingSelection().isNone()) { - VisiblePosition visibleStart = endingSelection().visibleStart(); - VisiblePosition visibleEnd = endingSelection().visibleEnd(); - - bool startAfterParagraph = Range::compareBoundaryPoints(visibleStart.deepEquivalent(), endOfParagraphToMove.deepEquivalent()) > 0; - bool endBeforeParagraph = Range::compareBoundaryPoints(visibleEnd.deepEquivalent(), startOfParagraphToMove.deepEquivalent()) < 0; - - if (!startAfterParagraph && !endBeforeParagraph) { - bool startInParagraph = Range::compareBoundaryPoints(visibleStart.deepEquivalent(), startOfParagraphToMove.deepEquivalent()) >= 0; - bool endInParagraph = Range::compareBoundaryPoints(visibleEnd.deepEquivalent(), endOfParagraphToMove.deepEquivalent()) <= 0; - - startIndex = 0; - if (startInParagraph) { - RefPtr<Range> startRange = Range::create(document(), rangeCompliantEquivalent(startOfParagraphToMove.deepEquivalent()), rangeCompliantEquivalent(visibleStart.deepEquivalent())); - startIndex = TextIterator::rangeLength(startRange.get(), true); - } - - endIndex = 0; - if (endInParagraph) { - RefPtr<Range> endRange = Range::create(document(), rangeCompliantEquivalent(startOfParagraphToMove.deepEquivalent()), rangeCompliantEquivalent(visibleEnd.deepEquivalent())); - endIndex = TextIterator::rangeLength(endRange.get(), true); - } - } - } - - 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 = endOfParagraphToMove.deepEquivalent().upstream(); - - // start and end can't be used directly to create a Range; they are "editing positions" - Position startRangeCompliant = rangeCompliantEquivalent(start); - Position endRangeCompliant = rangeCompliantEquivalent(end); - RefPtr<Range> range = Range::create(document(), startRangeCompliant.node(), startRangeCompliant.offset(), endRangeCompliant.node(), endRangeCompliant.offset()); - - // 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. - RefPtr<DocumentFragment> fragment = startOfParagraphToMove != endOfParagraphToMove ? createFragmentFromMarkup(document(), createMarkup(range.get(), 0, DoNotAnnotateForInterchange, true), "") : 0; - - // 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. - RefPtr<CSSMutableStyleDeclaration> styleInEmptyParagraph; - if (startOfParagraphToMove == endOfParagraphToMove && preserveStyle) { - styleInEmptyParagraph = styleAtPosition(startOfParagraphToMove.deepEquivalent()); - // 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(Selection(start, end, DOWNSTREAM)); - deleteSelection(false, false, false, false); - - ASSERT(destination.deepEquivalent().node()->inDocument()); - - // 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). - VisiblePosition caretAfterDelete = endingSelection().visibleStart(); - if (isStartOfParagraph(caretAfterDelete) && isEndOfParagraph(caretAfterDelete)) { - // Note: We want the rightmost candidate. - Position position = caretAfterDelete.deepEquivalent().downstream(); - Node* node = position.node(); - // Normally deletion will leave a br as a placeholder. - if (node->hasTagName(brTag)) - removeNodeAndPruneAncestors(node); - // 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)) - removeNodeAndPruneAncestors(node); - else if (lineBreakExistsAtPosition(caretAfterDelete)) { - // There is a preserved '\n' at caretAfterDelete. - Text* textNode = static_cast<Text*>(node); - if (textNode->length() == 1) - removeNodeAndPruneAncestors(node); - else - deleteTextFromNode(textNode, position.offset(), 1); - } - } - - // 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 == afterParagraph)) { - // FIXME: Trim text between beforeParagraph and afterParagraph if they aren't equal. - insertNodeAt(createBreakElement(document()).get(), beforeParagraph.deepEquivalent()); - // Need an updateLayout here in case inserting the br has split a text node. - updateLayout(); - } - - RefPtr<Range> startToDestinationRange(Range::create(document(), Position(document(), 0), rangeCompliantEquivalent(destination.deepEquivalent()))); - destinationIndex = TextIterator::rangeLength(startToDestinationRange.get(), true); - - setEndingSelection(destination); - applyCommandToComposite(ReplaceSelectionCommand::create(document(), fragment.get(), true, false, !preserveStyle, false, true)); - // Restore styles from an empty paragraph to the new empty paragraph. - if (styleInEmptyParagraph) - applyStyle(styleInEmptyParagraph.get()); - - if (preserveSelection && startIndex != -1) { - // 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). - RefPtr<Range> start = TextIterator::rangeFromLocationAndLength(document()->documentElement(), destinationIndex + startIndex, 0, true); - RefPtr<Range> end = TextIterator::rangeFromLocationAndLength(document()->documentElement(), destinationIndex + endIndex, 0, true); - if (start && end) - setEndingSelection(Selection(start->startPosition(), end->startPosition(), DOWNSTREAM)); - } -} - -// FIXME: Send an appropriate shouldDeleteRange call. -bool CompositeEditCommand::breakOutOfEmptyListItem() -{ - Node* emptyListItem = enclosingEmptyListItem(endingSelection().visibleStart()); - if (!emptyListItem) - return false; - - RefPtr<CSSMutableStyleDeclaration> style = styleAtPosition(endingSelection().start()); - - Node* listNode = emptyListItem->parentNode(); - - if (!listNode->isContentEditable()) - return false; - - RefPtr<Node> newBlock = isListElement(listNode->parentNode()) ? createListItemElement(document()) : createDefaultParagraphElement(document()); - - if (emptyListItem->renderer()->nextSibling()) { - if (emptyListItem->renderer()->previousSibling()) - splitElement(static_cast<Element*>(listNode), emptyListItem); - insertNodeBefore(newBlock.get(), listNode); - removeNode(emptyListItem); - } else { - insertNodeAfter(newBlock.get(), listNode); - removeNode(emptyListItem->renderer()->previousSibling() ? emptyListItem : listNode); - } - - appendBlockPlaceholder(newBlock.get()); - setEndingSelection(Selection(Position(newBlock.get(), 0), DOWNSTREAM)); - - computedStyle(endingSelection().start().node())->diff(style.get()); - if (style->length() > 0) - applyStyle(style.get()); - - 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, bool alwaysAvoidAnchors) -{ - if (original.isNull()) - return original; - - VisiblePosition visiblePos(original); - Node* enclosingAnchor = enclosingAnchorElement(original); - Position result = original; - // Don't avoid block level anchors, because that would insert content into the wrong paragraph. - if (enclosingAnchor && !isBlock(enclosingAnchor)) { - VisiblePosition firstInAnchor(Position(enclosingAnchor, 0)); - VisiblePosition lastInAnchor(Position(enclosingAnchor, maxDeepOffset(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 == lastInAnchor && (isEndOfDocument(visiblePos) || alwaysAvoidAnchors)) { - // 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.node() != enclosingAnchor && original.node()->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 (lineBreakExistsAtPosition(visiblePos) && downstream.node()->isDescendantOf(enclosingAnchor)) - return original; - - result = positionAfterNode(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 == firstInAnchor && (!isStartOfParagraph(visiblePos) || alwaysAvoidAnchors)) { - // 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.node() != enclosingAnchor && original.node()->parentNode() != enclosingAnchor) { - pushAnchorElementDown(enclosingAnchor); - enclosingAnchor = enclosingAnchorElement(original); - } - result = positionBeforeNode(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. -PassRefPtr<Node> CompositeEditCommand::splitTreeToNode(Node* start, Node* end, bool splitAncestor) -{ - RefPtr<Node> node; - for (node = start; node && node->parent() != end; node = node->parent()) { - VisiblePosition positionInParent(Position(node->parent(), 0), DOWNSTREAM); - VisiblePosition positionInNode(Position(node, 0), DOWNSTREAM); - if (positionInParent != positionInNode) - applyCommandToComposite(SplitElementCommand::create(static_cast<Element*>(node->parent()), node)); - } - if (splitAncestor) - return splitTreeToNode(end, end->parent()); - return node.release(); -} - -PassRefPtr<Element> createBlockPlaceholderElement(Document* document) -{ - ExceptionCode ec = 0; - RefPtr<Element> breakNode = document->createElementNS(xhtmlNamespaceURI, "br", ec); - ASSERT(ec == 0); - return breakNode.release(); -} - -} // namespace WebCore |