blob: dae1d1bfa80d993364ac64683018c9c1fe64749f [file] [log] [blame]
/*
* Copyright (C) 2005, 2006, 2007 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(new ApplyStyleCommand(document(), style, editingAction));
}
void CompositeEditCommand::applyStyle(CSSStyleDeclaration* style, const Position& start, const Position& end, EditAction editingAction)
{
applyCommandToComposite(new ApplyStyleCommand(document(), style, start, end, editingAction));
}
void CompositeEditCommand::applyStyledElement(Element* element)
{
applyCommandToComposite(new ApplyStyleCommand(element, false));
}
void CompositeEditCommand::removeStyledElement(Element* element)
{
applyCommandToComposite(new ApplyStyleCommand(element, true));
}
void CompositeEditCommand::insertParagraphSeparator(bool useDefaultParagraphElement)
{
applyCommandToComposite(new InsertParagraphSeparatorCommand(document(), useDefaultParagraphElement));
}
void CompositeEditCommand::insertLineBreak()
{
applyCommandToComposite(new InsertLineBreakCommand(document()));
}
void CompositeEditCommand::insertNodeBefore(Node* insertChild, Node* refChild)
{
ASSERT(!refChild->hasTagName(bodyTag));
applyCommandToComposite(new InsertNodeBeforeCommand(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(new AppendNodeCommand(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(new RemoveNodeCommand(removeChild));
}
void CompositeEditCommand::removeNodePreservingChildren(Node* removeChild)
{
applyCommandToComposite(new RemoveNodePreservingChildrenCommand(removeChild));
}
void CompositeEditCommand::removeNodeAndPruneAncestors(Node* node)
{
RefPtr<Node> parent = node->parentNode();
removeNode(node);
prune(parent);
}
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(new SplitTextNodeCommand(text, offset));
}
void CompositeEditCommand::splitElement(Element* element, Node* atChild)
{
applyCommandToComposite(new SplitElementCommand(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(new MergeIdenticalElementsCommand(first, second));
}
void CompositeEditCommand::wrapContentsInDummySpan(Element* element)
{
applyCommandToComposite(new WrapContentsInDummySpanCommand(element));
}
void CompositeEditCommand::splitTextNodeContainingElement(Text *text, int offset)
{
applyCommandToComposite(new SplitTextNodeContainingElementCommand(text, offset));
}
void CompositeEditCommand::joinTextNodes(Text *text1, Text *text2)
{
applyCommandToComposite(new JoinTextNodesCommand(text1, text2));
}
void CompositeEditCommand::inputText(const String &text, bool selectInsertedText)
{
int offset = 0;
int length = text.length();
RefPtr<Range> startRange = new Range(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 = new InsertTextCommand(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(new InsertIntoTextNodeCommand(node, offset, text));
}
void CompositeEditCommand::deleteTextFromNode(Text *node, int offset, int count)
{
applyCommandToComposite(new DeleteFromTextNodeCommand(node, offset, count));
}
void CompositeEditCommand::replaceTextInNode(Text *node, int offset, int count, const String &replacementText)
{
applyCommandToComposite(new DeleteFromTextNodeCommand(node, offset, count));
applyCommandToComposite(new InsertIntoTextNodeCommand(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(new DeleteSelectionCommand(document(), smartDelete, mergeBlocksAfterDelete, replace, expandForSpecialElements));
}
void CompositeEditCommand::deleteSelection(const Selection &selection, bool smartDelete, bool mergeBlocksAfterDelete, bool replace, bool expandForSpecialElements)
{
if (selection.isRange())
applyCommandToComposite(new DeleteSelectionCommand(selection, smartDelete, mergeBlocksAfterDelete, replace, expandForSpecialElements));
}
void CompositeEditCommand::removeCSSProperty(CSSStyleDeclaration *decl, int property)
{
applyCommandToComposite(new RemoveCSSPropertyCommand(document(), decl, property));
}
void CompositeEditCommand::removeNodeAttribute(Element* element, const QualifiedName& attribute)
{
if (element->getAttribute(attribute).isNull())
return;
applyCommandToComposite(new RemoveNodeAttributeCommand(element, attribute));
}
void CompositeEditCommand::setNodeAttribute(Element* element, const QualifiedName& attribute, const String &value)
{
applyCommandToComposite(new SetNodeAttributeCommand(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);
}
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.get();
}
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.get();
}
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);
}
}
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 = new Range(document(), rangeCompliantEquivalent(startOfParagraphToMove.deepEquivalent()), rangeCompliantEquivalent(visibleStart.deepEquivalent()));
startIndex = TextIterator::rangeLength(startRange.get(), true);
}
endIndex = 0;
if (endInParagraph) {
RefPtr<Range> endRange = new Range(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 = new Range(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;
// 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))
deleteTextFromNode(static_cast<Text*>(node), 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(new Range(document(), Position(document(), 0), rangeCompliantEquivalent(destination.deepEquivalent())));
destinationIndex = TextIterator::rangeLength(startToDestinationRange.get(), true);
setEndingSelection(destination);
applyCommandToComposite(new ReplaceSelectionCommand(document(), fragment.get(), true, false, !preserveStyle, false, true));
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();
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));
CSSComputedStyleDeclaration endingStyle(endingSelection().start().node());
endingStyle.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.
Node* CompositeEditCommand::splitTreeToNode(Node* start, Node* end, bool splitAncestor)
{
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(new SplitElementCommand(static_cast<Element*>(node->parent()), node));
}
if (splitAncestor)
return splitTreeToNode(end, end->parent());
return node;
}
PassRefPtr<Element> createBlockPlaceholderElement(Document* document)
{
ExceptionCode ec = 0;
RefPtr<Element> breakNode = document->createElementNS(xhtmlNamespaceURI, "br", ec);
ASSERT(ec == 0);
return breakNode.release();
}
} // namespace WebCore