blob: a1104806bec51fb498ac7bfbd9a00de8a1d1dbc9 [file] [log] [blame]
/*
* Copyright (C) 2005, 2006 Apple Computer, 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 "ReplaceSelectionCommand.h"
#include "ApplyStyleCommand.h"
#include "BeforeTextInsertedEvent.h"
#include "CSSComputedStyleDeclaration.h"
#include "CSSPropertyNames.h"
#include "CSSValueKeywords.h"
#include "Document.h"
#include "DocumentFragment.h"
#include "EditingText.h"
#include "EventNames.h"
#include "Element.h"
#include "Frame.h"
#include "HTMLElement.h"
#include "HTMLInterchange.h"
#include "HTMLInputElement.h"
#include "HTMLNames.h"
#include "SelectionController.h"
#include "SmartReplace.h"
#include "TextIterator.h"
#include "htmlediting.h"
#include "markup.h"
#include "visible_units.h"
namespace WebCore {
using namespace EventNames;
using namespace HTMLNames;
static bool isInterchangeNewlineNode(const Node *node)
{
static String interchangeNewlineClassString(AppleInterchangeNewline);
return node && node->hasTagName(brTag) &&
static_cast<const Element *>(node)->getAttribute(classAttr) == interchangeNewlineClassString;
}
static bool isInterchangeConvertedSpaceSpan(const Node *node)
{
static String convertedSpaceSpanClassString(AppleConvertedSpace);
return node->isHTMLElement() &&
static_cast<const HTMLElement *>(node)->getAttribute(classAttr) == convertedSpaceSpanClassString;
}
ReplacementFragment::ReplacementFragment(Document* document, DocumentFragment* fragment, bool matchStyle, const Selection& selection)
: m_document(document),
m_fragment(fragment),
m_matchStyle(matchStyle),
m_hasInterchangeNewlineAtStart(false),
m_hasInterchangeNewlineAtEnd(false)
{
if (!m_document)
return;
if (!m_fragment)
return;
if (!m_fragment->firstChild())
return;
Element* editableRoot = selection.rootEditableElement();
ASSERT(editableRoot);
if (!editableRoot)
return;
Node* shadowAncestorNode = editableRoot->shadowAncestorNode();
if (!editableRoot->getHTMLEventListener(webkitBeforeTextInsertedEvent) &&
// FIXME: Remove these checks once textareas and textfields actually register an event handler.
!(shadowAncestorNode && shadowAncestorNode->renderer() && shadowAncestorNode->renderer()->isTextField()) &&
!(shadowAncestorNode && shadowAncestorNode->renderer() && shadowAncestorNode->renderer()->isTextArea()) &&
editableRoot->isContentRichlyEditable()) {
removeInterchangeNodes(m_fragment->firstChild());
return;
}
Node* styleNode = selection.base().node();
RefPtr<Node> holder = insertFragmentForTestRendering(styleNode);
RefPtr<Range> range = Selection::selectionFromContentsOfNode(holder.get()).toRange();
String text = plainText(range.get());
// Give the root a chance to change the text.
RefPtr<BeforeTextInsertedEvent> evt = new BeforeTextInsertedEvent(text);
ExceptionCode ec = 0;
editableRoot->dispatchEvent(evt, ec, true);
ASSERT(ec == 0);
if (text != evt->text() || !editableRoot->isContentRichlyEditable()) {
restoreTestRenderingNodesToFragment(holder.get());
removeNode(holder);
m_fragment = createFragmentFromText(selection.toRange().get(), evt->text());
if (!m_fragment->firstChild())
return;
holder = insertFragmentForTestRendering(styleNode);
}
removeInterchangeNodes(holder->firstChild());
removeUnrenderedNodes(holder.get());
restoreTestRenderingNodesToFragment(holder.get());
removeNode(holder);
}
bool ReplacementFragment::isEmpty() const
{
return (!m_fragment || !m_fragment->firstChild()) && !m_hasInterchangeNewlineAtStart && !m_hasInterchangeNewlineAtEnd;
}
Node *ReplacementFragment::firstChild() const
{
return m_fragment ? m_fragment->firstChild() : 0;
}
Node *ReplacementFragment::lastChild() const
{
return m_fragment ? m_fragment->lastChild() : 0;
}
void ReplacementFragment::removeNodePreservingChildren(Node *node)
{
if (!node)
return;
while (RefPtr<Node> n = node->firstChild()) {
removeNode(n);
insertNodeBefore(n.get(), node);
}
removeNode(node);
}
void ReplacementFragment::removeNode(PassRefPtr<Node> node)
{
if (!node)
return;
Node *parent = node->parentNode();
if (!parent)
return;
ExceptionCode ec = 0;
parent->removeChild(node.get(), ec);
ASSERT(ec == 0);
}
void ReplacementFragment::insertNodeBefore(Node *node, Node *refNode)
{
if (!node || !refNode)
return;
Node *parent = refNode->parentNode();
if (!parent)
return;
ExceptionCode ec = 0;
parent->insertBefore(node, refNode, ec);
ASSERT(ec == 0);
}
PassRefPtr<Node> ReplacementFragment::insertFragmentForTestRendering(Node* context)
{
Node* body = m_document->body();
if (!body)
return 0;
RefPtr<StyledElement> holder = static_pointer_cast<StyledElement>(createDefaultParagraphElement(m_document.get()));
ExceptionCode ec = 0;
// Copy the whitespace and user-select style from the context onto this element.
// FIXME: We should examine other style properties to see if they would be appropriate to consider during the test rendering.
Node* n = context;
while (n && !n->isElementNode())
n = n->parentNode();
if (n) {
RefPtr<CSSComputedStyleDeclaration> conFontStyle = new CSSComputedStyleDeclaration(static_cast<Element*>(n));
CSSStyleDeclaration* style = holder->style();
style->setProperty(CSS_PROP_WHITE_SPACE, conFontStyle->getPropertyValue(CSS_PROP_WHITE_SPACE), false, ec);
ASSERT(ec == 0);
style->setProperty(CSS_PROP__WEBKIT_USER_SELECT, conFontStyle->getPropertyValue(CSS_PROP__WEBKIT_USER_SELECT), false, ec);
ASSERT(ec == 0);
}
holder->appendChild(m_fragment, ec);
ASSERT(ec == 0);
body->appendChild(holder.get(), ec);
ASSERT(ec == 0);
m_document->updateLayoutIgnorePendingStylesheets();
return holder.release();
}
void ReplacementFragment::restoreTestRenderingNodesToFragment(Node *holder)
{
if (!holder)
return;
ExceptionCode ec = 0;
while (RefPtr<Node> node = holder->firstChild()) {
holder->removeChild(node.get(), ec);
ASSERT(ec == 0);
m_fragment->appendChild(node.get(), ec);
ASSERT(ec == 0);
}
}
void ReplacementFragment::removeUnrenderedNodes(Node* holder)
{
Vector<Node*> unrendered;
for (Node* node = holder->firstChild(); node; node = node->traverseNextNode(holder))
if (!isNodeRendered(node) && !isTableStructureNode(node))
unrendered.append(node);
size_t n = unrendered.size();
for (size_t i = 0; i < n; ++i)
removeNode(unrendered[i]);
}
void ReplacementFragment::removeInterchangeNodes(Node* startNode)
{
Node* node = startNode;
Node* newlineAtStartNode = 0;
Node* newlineAtEndNode = 0;
while (node) {
Node *next = node->traverseNextNode();
if (isInterchangeNewlineNode(node)) {
if (next || node == startNode) {
m_hasInterchangeNewlineAtStart = true;
newlineAtStartNode = node;
}
else {
m_hasInterchangeNewlineAtEnd = true;
newlineAtEndNode = node;
}
}
else if (isInterchangeConvertedSpaceSpan(node)) {
RefPtr<Node> n = 0;
while ((n = node->firstChild())) {
removeNode(n);
insertNodeBefore(n.get(), node);
}
removeNode(node);
if (n)
next = n->traverseNextNode();
}
node = next;
}
if (newlineAtStartNode)
removeNode(newlineAtStartNode);
if (newlineAtEndNode)
removeNode(newlineAtEndNode);
}
ReplaceSelectionCommand::ReplaceSelectionCommand(Document* document, PassRefPtr<DocumentFragment> fragment,
bool selectReplacement, bool smartReplace, bool matchStyle, bool preventNesting, bool movingParagraph,
EditAction editAction)
: CompositeEditCommand(document),
m_selectReplacement(selectReplacement),
m_smartReplace(smartReplace),
m_matchStyle(matchStyle),
m_documentFragment(fragment),
m_preventNesting(preventNesting),
m_movingParagraph(movingParagraph),
m_editAction(editAction)
{
}
bool ReplaceSelectionCommand::shouldMergeStart(bool selectionStartWasStartOfParagraph, bool fragmentHasInterchangeNewlineAtStart)
{
VisiblePosition startOfInsertedContent(positionAtStartOfInsertedContent());
VisiblePosition prev = startOfInsertedContent.previous(true);
if (prev.isNull())
return false;
return !selectionStartWasStartOfParagraph &&
!fragmentHasInterchangeNewlineAtStart &&
isStartOfParagraph(startOfInsertedContent) &&
!startOfInsertedContent.deepEquivalent().node()->hasTagName(brTag) &&
shouldMerge(startOfInsertedContent, prev);
}
bool ReplaceSelectionCommand::shouldMergeEnd(bool selectionEndWasEndOfParagraph)
{
VisiblePosition endOfInsertedContent(positionAtEndOfInsertedContent());
VisiblePosition next = endOfInsertedContent.next(true);
if (next.isNull())
return false;
return !selectionEndWasEndOfParagraph &&
isEndOfParagraph(endOfInsertedContent) &&
!endOfInsertedContent.deepEquivalent().node()->hasTagName(brTag) &&
shouldMerge(endOfInsertedContent, next);
}
static bool isMailPasteAsQuotationNode(const Node* node)
{
return node && node->hasTagName(blockquoteTag) && node->isElementNode() && static_cast<const Element*>(node)->getAttribute(classAttr) == ApplePasteAsQuotation;
}
// Wrap CompositeEditCommand::removeNodePreservingChildren() so we can update the nodes we track
void ReplaceSelectionCommand::removeNodePreservingChildren(Node* node)
{
if (m_firstNodeInserted == node)
m_firstNodeInserted = node->traverseNextNode();
if (m_lastLeafInserted == node)
m_lastLeafInserted = node->lastChild() ? node->lastChild() : node->traverseNextSibling();
CompositeEditCommand::removeNodePreservingChildren(node);
}
// Wrap CompositeEditCommand::removeNodeAndPruneAncestors() so we can update the nodes we track
void ReplaceSelectionCommand::removeNodeAndPruneAncestors(Node* node)
{
// prepare in case m_firstNodeInserted and/or m_lastLeafInserted get removed
// FIXME: shouldn't m_lastLeafInserted be adjusted using traversePreviousNode()?
Node* afterFirst = m_firstNodeInserted ? m_firstNodeInserted->traverseNextSibling() : 0;
Node* afterLast = m_lastLeafInserted ? m_lastLeafInserted->traverseNextSibling() : 0;
CompositeEditCommand::removeNodeAndPruneAncestors(node);
// adjust m_firstNodeInserted and m_lastLeafInserted since either or both may have been removed
if (m_lastLeafInserted && !m_lastLeafInserted->inDocument())
m_lastLeafInserted = afterLast;
if (m_firstNodeInserted && !m_firstNodeInserted->inDocument())
m_firstNodeInserted = m_lastLeafInserted && m_lastLeafInserted->inDocument() ? afterFirst : 0;
}
bool ReplaceSelectionCommand::shouldMerge(const VisiblePosition& from, const VisiblePosition& to)
{
if (from.isNull() || to.isNull())
return false;
Node* fromNode = from.deepEquivalent().node();
Node* toNode = to.deepEquivalent().node();
Node* fromNodeBlock = enclosingBlock(fromNode);
return !enclosingNodeOfType(from.deepEquivalent(), &isMailPasteAsQuotationNode) &&
fromNodeBlock && (!fromNodeBlock->hasTagName(blockquoteTag) || isMailBlockquote(fromNodeBlock)) &&
enclosingListChild(fromNode) == enclosingListChild(toNode) &&
enclosingTableCell(from.deepEquivalent()) == enclosingTableCell(from.deepEquivalent()) &&
// Don't merge to or from a position before or after a block because it would
// be a no-op and cause infinite recursion.
!isBlock(fromNode) && !isBlock(toNode);
}
// Style rules that match just inserted elements could change their appearance, like
// a div inserted into a document with div { display:inline; }.
void ReplaceSelectionCommand::negateStyleRulesThatAffectAppearance()
{
for (RefPtr<Node> node = m_firstNodeInserted.get(); node; node = node->traverseNextNode()) {
// FIXME: <rdar://problem/5371536> Style rules that match pasted content can change it's appearance
if (isStyleSpan(node.get())) {
HTMLElement* e = static_cast<HTMLElement*>(node.get());
// There are other styles that style rules can give to style spans,
// but these are the two important ones because they'll prevent
// inserted content from appearing in the right paragraph.
// FIXME: Hyatt is concerned that selectively using display:inline will give inconsistent
// results. We already know one issue because td elements ignore their display property
// in quirks mode (which Mail.app is always in). We should look for an alternative.
if (isBlock(e))
e->getInlineStyleDecl()->setProperty(CSS_PROP_DISPLAY, CSS_VAL_INLINE);
if (e->renderer() && e->renderer()->style()->floating() != FNONE)
e->getInlineStyleDecl()->setProperty(CSS_PROP_FLOAT, CSS_VAL_NONE);
}
if (node == m_lastLeafInserted)
break;
}
}
void ReplaceSelectionCommand::removeUnrenderedTextNodesAtEnds()
{
document()->updateLayoutIgnorePendingStylesheets();
if (!m_lastLeafInserted->renderer() &&
m_lastLeafInserted->isTextNode() &&
!enclosingNodeWithTag(Position(m_lastLeafInserted.get(), 0), selectTag) &&
!enclosingNodeWithTag(Position(m_lastLeafInserted.get(), 0), scriptTag)) {
RefPtr<Node> previous = m_firstNodeInserted == m_lastLeafInserted ? 0 : m_lastLeafInserted->traversePreviousNode();
removeNode(m_lastLeafInserted.get());
m_lastLeafInserted = previous;
}
// We don't have to make sure that m_firstNodeInserted isn't inside a select or script element, because
// it is a top level node in the fragment and the user can't insert into those elements.
if (!m_firstNodeInserted->renderer() &&
m_firstNodeInserted->isTextNode()) {
RefPtr<Node> next = m_firstNodeInserted == m_lastLeafInserted ? 0 : m_firstNodeInserted->traverseNextSibling();
removeNode(m_firstNodeInserted.get());
m_firstNodeInserted = next;
}
}
void ReplaceSelectionCommand::removeRedundantStyles(Node* mailBlockquoteEnclosingSelectionStart)
{
// There's usually a top level style span that holds the document's default style, push it down.
Node* node = m_firstNodeInserted.get();
if (isStyleSpan(node) && mailBlockquoteEnclosingSelectionStart) {
// Calculate the document default style.
RefPtr<CSSMutableStyleDeclaration> blockquoteStyle = Position(mailBlockquoteEnclosingSelectionStart, 0).computedStyle()->copyInheritableProperties();
RefPtr<CSSMutableStyleDeclaration> spanStyle = static_cast<HTMLElement*>(node)->inlineStyleDecl();
spanStyle->merge(blockquoteStyle.get());
}
// Compute and save the non-redundant styles for all HTML elements.
// Don't do any mutation here, because that would cause the diffs to trigger layouts.
Vector<RefPtr<CSSMutableStyleDeclaration> > styles;
Vector<RefPtr<HTMLElement> > elements;
for (node = m_firstNodeInserted.get(); node; node = node->traverseNextNode()) {
if (node->isHTMLElement() && isStyleSpan(node)) {
elements.append(static_cast<HTMLElement*>(node));
RefPtr<CSSMutableStyleDeclaration> parentStyle = computedStyle(node->parentNode())->copyInheritableProperties();
RefPtr<CSSMutableStyleDeclaration> style = computedStyle(node)->copyInheritableProperties();
parentStyle->diff(style.get());
// Remove any inherited block properties that are now in the span's style. This cuts out meaningless properties
// and prevents properties from magically affecting blocks later if the style is cloned for a new block element
// during a future editing operation.
style->removeBlockProperties();
styles.append(style.release());
}
if (node == m_lastLeafInserted)
break;
}
size_t count = styles.size();
for (size_t i = 0; i < count; ++i) {
HTMLElement* element = elements[i].get();
// Handle case where the element was already removed by earlier processing.
// It's possible this no longer occurs, but it did happen in an earlier version
// that processed elements in a less-determistic order, and I can't prove it
// does not occur.
if (!element->inDocument())
continue;
// Remove empty style spans.
if (isStyleSpan(element) && !element->hasChildNodes()) {
removeNodeAndPruneAncestors(element);
continue;
}
// Remove redundant style tags and style spans.
CSSMutableStyleDeclaration* style = styles[i].get();
if (style->length() == 0
&& (isStyleSpan(element)
|| element->hasTagName(bTag)
|| element->hasTagName(fontTag)
|| element->hasTagName(iTag)
|| element->hasTagName(uTag))) {
removeNodePreservingChildren(element);
continue;
}
// Clear redundant styles from elements.
CSSMutableStyleDeclaration* inlineStyleDecl = element->inlineStyleDecl();
if (inlineStyleDecl) {
CSSComputedStyleDeclaration::removeComputedInheritablePropertiesFrom(inlineStyleDecl);
inlineStyleDecl->merge(style, true);
setNodeAttribute(element, styleAttr, inlineStyleDecl->cssText());
}
}
}
void ReplaceSelectionCommand::handlePasteAsQuotationNode()
{
Node* node = m_firstNodeInserted.get();
if (isMailPasteAsQuotationNode(node))
static_cast<Element*>(node)->setAttribute(classAttr, "");
}
VisiblePosition ReplaceSelectionCommand::positionAtEndOfInsertedContent()
{
Node* lastNode = m_lastLeafInserted.get();
Node* enclosingSelect = enclosingNodeWithTag(Position(lastNode, 0), selectTag);
if (enclosingSelect)
lastNode = enclosingSelect;
return VisiblePosition(Position(lastNode, maxDeepOffset(lastNode)));
}
VisiblePosition ReplaceSelectionCommand::positionAtStartOfInsertedContent()
{
// Return the inserted content's first VisiblePosition.
return VisiblePosition(nextCandidate(positionBeforeNode(m_firstNodeInserted.get())));
}
void ReplaceSelectionCommand::doApply()
{
Selection selection = endingSelection();
ASSERT(selection.isCaretOrRange());
ASSERT(selection.start().node());
if (selection.isNone() || !selection.start().node())
return;
bool selectionIsPlainText = !selection.isContentRichlyEditable();
Element* currentRoot = selection.rootEditableElement();
ReplacementFragment fragment(document(), m_documentFragment.get(), m_matchStyle, selection);
if (m_matchStyle)
m_insertionStyle = styleAtPosition(selection.start());
VisiblePosition visibleStart = selection.visibleStart();
VisiblePosition visibleEnd = selection.visibleEnd();
bool selectionEndWasEndOfParagraph = isEndOfParagraph(visibleEnd);
bool selectionStartWasStartOfParagraph = isStartOfParagraph(visibleStart);
Node* mailBlockquoteEnclosingSelectionStart = nearestMailBlockquote(visibleStart.deepEquivalent().node());
Node* startBlock = enclosingBlock(visibleStart.deepEquivalent().node());
if (selectionStartWasStartOfParagraph && selectionEndWasEndOfParagraph ||
startBlock == currentRoot ||
startBlock && startBlock->renderer() && startBlock->renderer()->isListItem() ||
selectionIsPlainText)
m_preventNesting = false;
Position insertionPos = selection.start();
if (selection.isRange()) {
// When the end of the selection being pasted into is at the end of a paragraph, and that selection
// spans multiple blocks, not merging may leave an empty line.
// When the start of the selection being pasted into is at the start of a block, not merging
// will leave hanging block(s).
bool mergeBlocksAfterDelete = isEndOfParagraph(visibleEnd) || isStartOfBlock(visibleStart);
// FIXME: We should only expand to include fully selected special elements if we are copying a
// selection and pasting it on top of itself.
deleteSelection(false, mergeBlocksAfterDelete, true, false);
visibleStart = endingSelection().visibleStart();
if (fragment.hasInterchangeNewlineAtStart()) {
if (isEndOfParagraph(visibleStart) && !isStartOfParagraph(visibleStart)) {
if (!isEndOfDocument(visibleStart))
setEndingSelection(visibleStart.next());
} else
insertParagraphSeparator();
}
insertionPos = endingSelection().start();
} else {
ASSERT(selection.isCaret());
if (fragment.hasInterchangeNewlineAtStart()) {
VisiblePosition next = visibleStart.next(true);
if (isEndOfParagraph(visibleStart) && !isStartOfParagraph(visibleStart) && next.isNotNull())
setEndingSelection(next);
else
insertParagraphSeparator();
}
// We split the current paragraph in two to avoid nesting the blocks from the fragment inside the current block.
// For example paste <div>foo</div><div>bar</div><div>baz</div> into <div>x^x</div>, where ^ is the caret.
// As long as the div styles are the same, visually you'd expect: <div>xbar</div><div>bar</div><div>bazx</div>,
// not <div>xbar<div>bar</div><div>bazx</div></div>
if (m_preventNesting && !isEndOfParagraph(visibleStart) && !isStartOfParagraph(visibleStart)) {
insertParagraphSeparator();
setEndingSelection(endingSelection().visibleStart().previous());
}
insertionPos = endingSelection().start();
}
// Inserting content could cause whitespace to collapse, e.g. inserting <div>foo</div> into hello^ world.
prepareWhitespaceAtPositionForSplit(insertionPos);
// NOTE: This would be an incorrect usage of downstream() if downstream() were changed to mean the last position after
// p that maps to the same visible position as p (since in the case where a br is at the end of a block and collapsed
// away, there are positions after the br which map to the same visible position as [br, 0]).
Node* endBR = insertionPos.downstream().node()->hasTagName(brTag) ? insertionPos.downstream().node() : 0;
VisiblePosition originalVisPosBeforeEndBR;
if (endBR)
originalVisPosBeforeEndBR = VisiblePosition(endBR, 0, DOWNSTREAM).previous();
startBlock = enclosingBlock(insertionPos.node());
// Adjust insertionPos to prevent nesting.
if (m_preventNesting && startBlock) {
ASSERT(startBlock != currentRoot);
VisiblePosition visibleInsertionPos(insertionPos);
if (isEndOfBlock(visibleInsertionPos) && !(isStartOfBlock(visibleInsertionPos) && fragment.hasInterchangeNewlineAtEnd()))
insertionPos = positionAfterNode(startBlock);
else if (isStartOfBlock(visibleInsertionPos))
insertionPos = positionBeforeNode(startBlock);
}
// Paste into run of tabs splits the tab span.
insertionPos = positionOutsideTabSpan(insertionPos);
// Paste at start or end of link goes outside of link.
insertionPos = positionAvoidingSpecialElementBoundary(insertionPos);
Frame *frame = document()->frame();
// FIXME: Improve typing style.
// See this bug: <rdar://problem/3769899> Implementation of typing style needs improvement
frame->clearTypingStyle();
setTypingStyle(0);
// Remove the top level style span if its unnecessary before inserting it into the document, its faster.
RefPtr<CSSMutableStyleDeclaration> styleAtInsertionPos = insertionPos.computedStyle()->copyInheritableProperties();
if (isStyleSpan(fragment.firstChild())) {
Node* styleSpan = fragment.firstChild();
String styleText = static_cast<Element*>(styleSpan)->getAttribute(styleAttr);
if (styleText == styleAtInsertionPos->cssText())
fragment.removeNodePreservingChildren(styleSpan);
}
// We're finished if there is nothing to add.
if (fragment.isEmpty() || !fragment.firstChild())
return;
// 1) Insert the content.
// 2) Remove redundant styles and style tags, this inner <b> for example: <b>foo <b>bar</b> baz</b>.
// 3) Merge the start of the added content with the content before the position being pasted into.
// 4) Do one of the following: a) expand the last br if the fragment ends with one and it collapsed,
// b) merge the last paragraph of the incoming fragment with the paragraph that contained the
// end of the selection that was pasted into, or c) handle an interchange newline at the end of the
// incoming fragment.
// 5) Add spaces for smart replace.
// 6) Select the replacement if requested, and match style if requested.
VisiblePosition startOfInsertedContent, endOfInsertedContent;
RefPtr<Node> refNode = fragment.firstChild();
RefPtr<Node> node = refNode->nextSibling();
fragment.removeNode(refNode);
insertNodeAtAndUpdateNodesInserted(refNode.get(), insertionPos);
while (node) {
Node* next = node->nextSibling();
fragment.removeNode(node);
insertNodeAfterAndUpdateNodesInserted(node.get(), refNode.get());
refNode = node;
node = next;
}
removeUnrenderedTextNodesAtEnds();
negateStyleRulesThatAffectAppearance();
removeRedundantStyles(mailBlockquoteEnclosingSelectionStart);
if (!m_firstNodeInserted)
return;
endOfInsertedContent = positionAtEndOfInsertedContent();
startOfInsertedContent = positionAtStartOfInsertedContent();
// We inserted before the startBlock to prevent nesting, and the content before the startBlock wasn't in its own block and
// didn't have a br after it, so the inserted content ended up in the same paragraph.
if (startBlock && insertionPos.node() == startBlock->parentNode() && (unsigned)insertionPos.offset() < startBlock->nodeIndex() && !isStartOfParagraph(startOfInsertedContent))
insertNodeAt(createBreakElement(document()).get(), startOfInsertedContent.deepEquivalent());
Position lastPositionToSelect;
bool interchangeNewlineAtEnd = fragment.hasInterchangeNewlineAtEnd();
if (shouldRemoveEndBR(endBR, originalVisPosBeforeEndBR))
removeNodeAndPruneAncestors(endBR);
if (shouldMergeStart(selectionStartWasStartOfParagraph, fragment.hasInterchangeNewlineAtStart())) {
// Bail to avoid infinite recursion.
if (m_movingParagraph) {
// setting display:inline does not work for td elements in quirks mode
ASSERT(m_firstNodeInserted->hasTagName(tdTag));
return;
}
VisiblePosition destination = startOfInsertedContent.previous();
VisiblePosition startOfParagraphToMove = startOfInsertedContent;
// FIXME: Maintain positions for the start and end of inserted content instead of keeping nodes. The nodes are
// only ever used to create positions where inserted content starts/ends.
moveParagraph(startOfParagraphToMove, endOfParagraph(startOfParagraphToMove), destination);
m_firstNodeInserted = endingSelection().visibleStart().deepEquivalent().downstream().node();
if (!m_lastLeafInserted->inDocument())
m_lastLeafInserted = endingSelection().visibleEnd().deepEquivalent().upstream().node();
}
endOfInsertedContent = positionAtEndOfInsertedContent();
startOfInsertedContent = positionAtStartOfInsertedContent();
if (interchangeNewlineAtEnd) {
VisiblePosition next = endOfInsertedContent.next(true);
if (selectionEndWasEndOfParagraph || !isEndOfParagraph(endOfInsertedContent) || next.isNull()) {
if (!isStartOfParagraph(endOfInsertedContent)) {
setEndingSelection(endOfInsertedContent);
// Use a default paragraph element (a plain div) for the empty paragraph, using the last paragraph
// block's style seems to annoy users.
insertParagraphSeparator(true);
// Select up to the paragraph separator that was added.
lastPositionToSelect = endingSelection().visibleStart().deepEquivalent();
updateNodesInserted(lastPositionToSelect.node());
}
} else {
// Select up to the beginning of the next paragraph.
lastPositionToSelect = next.deepEquivalent().downstream();
}
} else if (shouldMergeEnd(selectionEndWasEndOfParagraph)) {
// Bail to avoid infinite recursion.
if (m_movingParagraph) {
ASSERT_NOT_REACHED();
return;
}
// Merging two paragraphs will destroy the moved one's block styles. Always move forward to preserve
// the block style of the paragraph already in the document, unless the paragraph to move would include the
// what was the start of the selection that was pasted into.
bool mergeForward = !inSameParagraph(startOfInsertedContent, endOfInsertedContent) || isStartOfParagraph(startOfInsertedContent);
VisiblePosition destination = mergeForward ? endOfInsertedContent.next() : endOfInsertedContent;
VisiblePosition startOfParagraphToMove = mergeForward ? startOfParagraph(endOfInsertedContent) : endOfInsertedContent.next();
moveParagraph(startOfParagraphToMove, endOfParagraph(startOfParagraphToMove), destination);
// Merging forward will remove m_lastLeafInserted from the document.
// FIXME: Maintain positions for the start and end of inserted content instead of keeping nodes. The nodes are
// only ever used to create positions where inserted content starts/ends.
if (mergeForward) {
m_lastLeafInserted = destination.previous().deepEquivalent().node();
if (!m_firstNodeInserted->inDocument())
m_firstNodeInserted = endingSelection().visibleStart().deepEquivalent().node();
}
}
handlePasteAsQuotationNode();
endOfInsertedContent = positionAtEndOfInsertedContent();
startOfInsertedContent = positionAtStartOfInsertedContent();
// Add spaces for smart replace.
if (m_smartReplace && currentRoot) {
// Disable smart replace for password fields.
Node* start = currentRoot->shadowAncestorNode();
if (start->hasTagName(inputTag) && static_cast<HTMLInputElement*>(start)->inputType() == HTMLInputElement::PASSWORD)
m_smartReplace = false;
}
if (m_smartReplace) {
bool needsTrailingSpace = !isEndOfParagraph(endOfInsertedContent) &&
!isCharacterSmartReplaceExempt(endOfInsertedContent.characterAfter(), false);
if (needsTrailingSpace) {
RenderObject* renderer = m_lastLeafInserted->renderer();
bool collapseWhiteSpace = !renderer || renderer->style()->collapseWhiteSpace();
Node* endNode = positionAtEndOfInsertedContent().deepEquivalent().upstream().node();
if (endNode->isTextNode()) {
Text* text = static_cast<Text*>(endNode);
insertTextIntoNode(text, text->length(), collapseWhiteSpace ? nonBreakingSpaceString() : " ");
} else {
RefPtr<Node> node = document()->createEditingTextNode(collapseWhiteSpace ? nonBreakingSpaceString() : " ");
insertNodeAfterAndUpdateNodesInserted(node.get(), endNode);
}
}
bool needsLeadingSpace = !isStartOfParagraph(startOfInsertedContent) &&
!isCharacterSmartReplaceExempt(startOfInsertedContent.previous().characterAfter(), true);
if (needsLeadingSpace) {
RenderObject* renderer = m_lastLeafInserted->renderer();
bool collapseWhiteSpace = !renderer || renderer->style()->collapseWhiteSpace();
Node* startNode = positionAtStartOfInsertedContent().deepEquivalent().downstream().node();
if (startNode->isTextNode()) {
Text* text = static_cast<Text*>(startNode);
insertTextIntoNode(text, 0, collapseWhiteSpace ? nonBreakingSpaceString() : " ");
} else {
RefPtr<Node> node = document()->createEditingTextNode(collapseWhiteSpace ? nonBreakingSpaceString() : " ");
// Don't updateNodesInserted. Doing so would set m_lastLeafInserted to be the node containing the
// leading space, but m_lastLeafInserted is supposed to mark the end of pasted content.
insertNodeBefore(node.get(), startNode);
// FIXME: Use positions to track the start/end of inserted content.
m_firstNodeInserted = node;
}
}
}
completeHTMLReplacement(lastPositionToSelect);
}
bool ReplaceSelectionCommand::shouldRemoveEndBR(Node* endBR, const VisiblePosition& originalVisPosBeforeEndBR)
{
if (!endBR || !endBR->inDocument())
return false;
VisiblePosition visiblePos(Position(endBR, 0));
// Don't remove the br if nothing was inserted.
if (visiblePos.previous() == originalVisPosBeforeEndBR)
return false;
// Remove the br if it is collapsed away and so is unnecessary.
if (!document()->inStrictMode() && isEndOfBlock(visiblePos) && !isStartOfParagraph(visiblePos))
return true;
// A br that was originally holding a line open should be displaced by inserted content or turned into a line break.
// A br that was originally acting as a line break should still be acting as a line break, not as a placeholder.
return isStartOfParagraph(visiblePos) && isEndOfParagraph(visiblePos);
}
void ReplaceSelectionCommand::completeHTMLReplacement(const Position &lastPositionToSelect)
{
Position start;
Position end;
// FIXME: This should never not be the case.
if (m_firstNodeInserted && m_firstNodeInserted->inDocument() && m_lastLeafInserted && m_lastLeafInserted->inDocument()) {
start = positionAtStartOfInsertedContent().deepEquivalent();
end = positionAtEndOfInsertedContent().deepEquivalent();
// FIXME (11475): Remove this and require that the creator of the fragment to use nbsps.
rebalanceWhitespaceAt(start);
rebalanceWhitespaceAt(end);
if (m_matchStyle) {
ASSERT(m_insertionStyle);
applyStyle(m_insertionStyle.get(), start, end);
}
if (lastPositionToSelect.isNotNull())
end = lastPositionToSelect;
} else if (lastPositionToSelect.isNotNull())
start = end = lastPositionToSelect;
else
return;
if (m_selectReplacement)
setEndingSelection(Selection(start, end, SEL_DEFAULT_AFFINITY));
else
setEndingSelection(Selection(end, SEL_DEFAULT_AFFINITY));
}
EditAction ReplaceSelectionCommand::editingAction() const
{
return m_editAction;
}
void ReplaceSelectionCommand::insertNodeAfterAndUpdateNodesInserted(Node *insertChild, Node *refChild)
{
insertNodeAfter(insertChild, refChild);
updateNodesInserted(insertChild);
}
void ReplaceSelectionCommand::insertNodeAtAndUpdateNodesInserted(Node *insertChild, const Position& p)
{
insertNodeAt(insertChild, p);
updateNodesInserted(insertChild);
}
void ReplaceSelectionCommand::insertNodeBeforeAndUpdateNodesInserted(Node *insertChild, Node *refChild)
{
insertNodeBefore(insertChild, refChild);
updateNodesInserted(insertChild);
}
void ReplaceSelectionCommand::updateNodesInserted(Node *node)
{
if (!node)
return;
if (!m_firstNodeInserted)
m_firstNodeInserted = node;
if (node == m_lastLeafInserted)
return;
m_lastLeafInserted = node->lastDescendant();
}
} // namespace WebCore