/* * Copyright (C) 2004, 2008, 2009, 2010 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 "SelectionController.h" #include "CharacterData.h" #include "DeleteSelectionCommand.h" #include "Document.h" #include "Editor.h" #include "EditorClient.h" #include "Element.h" #include "EventHandler.h" #include "ExceptionCode.h" #include "FloatQuad.h" #include "FocusController.h" #include "Frame.h" #include "FrameTree.h" #include "FrameView.h" #include "GraphicsContext.h" #include "HTMLFormElement.h" #include "HTMLFrameElementBase.h" #include "HTMLInputElement.h" #include "HTMLNames.h" #include "HitTestRequest.h" #include "HitTestResult.h" #include "Page.h" #include "Range.h" #include "RenderLayer.h" #include "RenderTextControl.h" #include "RenderTheme.h" #include "RenderView.h" #include "RenderWidget.h" #include "SecureTextInput.h" #include "Settings.h" #include "TextIterator.h" #include "TypingCommand.h" #include "htmlediting.h" #include "visible_units.h" #ifdef ANDROID_ALLOW_TURNING_OFF_CARET #include "WebViewCore.h" #endif #include #include #define EDIT_DEBUG 0 namespace WebCore { using namespace HTMLNames; const int NoXPosForVerticalArrowNavigation = INT_MIN; SelectionController::SelectionController(Frame* frame, bool isDragCaretController) : m_frame(frame) , m_xPosForVerticalArrowNavigation(NoXPosForVerticalArrowNavigation) , m_granularity(CharacterGranularity) , m_caretBlinkTimer(this, &SelectionController::caretBlinkTimerFired) , m_caretRectNeedsUpdate(true) , m_absCaretBoundsDirty(true) , m_isDragCaretController(isDragCaretController) , m_isCaretBlinkingSuspended(false) , m_focused(frame && frame->page() && frame->page()->focusController()->focusedFrame() == frame) , m_caretVisible(isDragCaretController) , m_caretPaint(true) { setIsDirectional(false); } void SelectionController::moveTo(const VisiblePosition &pos, bool userTriggered, CursorAlignOnScroll align) { SetSelectionOptions options = CloseTyping | ClearTypingStyle; if (userTriggered) options |= UserTriggered; setSelection(VisibleSelection(pos.deepEquivalent(), pos.deepEquivalent(), pos.affinity()), options, align); } void SelectionController::moveTo(const VisiblePosition &base, const VisiblePosition &extent, bool userTriggered) { SetSelectionOptions options = CloseTyping | ClearTypingStyle; if (userTriggered) options |= UserTriggered; setSelection(VisibleSelection(base.deepEquivalent(), extent.deepEquivalent(), base.affinity()), options); } void SelectionController::moveTo(const Position &pos, EAffinity affinity, bool userTriggered) { SetSelectionOptions options = CloseTyping | ClearTypingStyle; if (userTriggered) options |= UserTriggered; setSelection(VisibleSelection(pos, affinity), options); } void SelectionController::moveTo(const Range *r, EAffinity affinity, bool userTriggered) { SetSelectionOptions options = CloseTyping | ClearTypingStyle; if (userTriggered) options |= UserTriggered; VisibleSelection selection = r ? VisibleSelection(r->startPosition(), r->endPosition(), affinity) : VisibleSelection(Position(), Position(), affinity); setSelection(selection, options); } void SelectionController::moveTo(const Position &base, const Position &extent, EAffinity affinity, bool userTriggered) { SetSelectionOptions options = CloseTyping | ClearTypingStyle; if (userTriggered) options |= UserTriggered; setSelection(VisibleSelection(base, extent, affinity), options); } void SelectionController::setSelection(const VisibleSelection& s, SetSelectionOptions options, CursorAlignOnScroll align, TextGranularity granularity, DirectionalityPolicy directionalityPolicy) { m_granularity = granularity; bool closeTyping = options & CloseTyping; bool shouldClearTypingStyle = options & ClearTypingStyle; bool userTriggered = options & UserTriggered; setIsDirectional(directionalityPolicy == MakeDirectionalSelection); if (m_isDragCaretController) { invalidateCaretRect(); m_selection = s; m_caretRectNeedsUpdate = true; invalidateCaretRect(); updateCaretRect(); return; } if (!m_frame) { m_selection = s; return; } // : Infinite recursion at SelectionController::setSelection // if document->frame() == m_frame we can get into an infinite loop if (s.base().anchorNode()) { Document* document = s.base().anchorNode()->document(); if (document && document->frame() && document->frame() != m_frame && document != m_frame->document()) { document->frame()->selection()->setSelection(s, options); return; } } if (closeTyping) TypingCommand::closeTyping(m_frame->editor()->lastEditCommand()); if (shouldClearTypingStyle) clearTypingStyle(); if (m_selection == s) { // Even if selection was not changed, selection offsets may have been changed. notifyRendererOfSelectionChange(userTriggered); return; } VisibleSelection oldSelection = m_selection; m_selection = s; m_caretRectNeedsUpdate = true; if (!s.isNone()) setFocusedNodeIfNeeded(); updateAppearance(); // Always clear the x position used for vertical arrow navigation. // It will be restored by the vertical arrow navigation code if necessary. m_xPosForVerticalArrowNavigation = NoXPosForVerticalArrowNavigation; selectFrameElementInParentIfFullySelected(); notifyRendererOfSelectionChange(userTriggered); m_frame->editor()->respondToChangedSelection(oldSelection, options); if (userTriggered) { ScrollAlignment alignment; if (m_frame->editor()->behavior().shouldCenterAlignWhenSelectionIsRevealed()) alignment = (align == AlignCursorOnScrollAlways) ? ScrollAlignment::alignCenterAlways : ScrollAlignment::alignCenterIfNeeded; else alignment = (align == AlignCursorOnScrollAlways) ? ScrollAlignment::alignTopAlways : ScrollAlignment::alignToEdgeIfNeeded; revealSelection(alignment, true); } notifyAccessibilityForSelectionChange(); m_frame->document()->enqueueDocumentEvent(Event::create(eventNames().selectionchangeEvent, false, false)); } static bool removingNodeRemovesPosition(Node* node, const Position& position) { if (!position.anchorNode()) return false; if (position.anchorNode() == node) return true; if (!node->isElementNode()) return false; Element* element = static_cast(node); return element->contains(position.anchorNode()) || element->contains(position.anchorNode()->shadowAncestorNode()); } void SelectionController::nodeWillBeRemoved(Node *node) { if (isNone()) return; // There can't be a selection inside a fragment, so if a fragment's node is being removed, // the selection in the document that created the fragment needs no adjustment. if (node && highestAncestor(node)->nodeType() == Node::DOCUMENT_FRAGMENT_NODE) return; respondToNodeModification(node, removingNodeRemovesPosition(node, m_selection.base()), removingNodeRemovesPosition(node, m_selection.extent()), removingNodeRemovesPosition(node, m_selection.start()), removingNodeRemovesPosition(node, m_selection.end())); } void SelectionController::respondToNodeModification(Node* node, bool baseRemoved, bool extentRemoved, bool startRemoved, bool endRemoved) { bool clearRenderTreeSelection = false; bool clearDOMTreeSelection = false; if (startRemoved || endRemoved) { // FIXME: When endpoints are removed, we should just alter the selection, instead of blowing it away. clearRenderTreeSelection = true; clearDOMTreeSelection = true; } else if (baseRemoved || extentRemoved) { // The base and/or extent are about to be removed, but the start and end aren't. // Change the base and extent to the start and end, but don't re-validate the // selection, since doing so could move the start and end into the node // that is about to be removed. if (m_selection.isBaseFirst()) m_selection.setWithoutValidation(m_selection.start(), m_selection.end()); else m_selection.setWithoutValidation(m_selection.end(), m_selection.start()); } else if (RefPtr range = m_selection.firstRange()) { ExceptionCode ec = 0; Range::CompareResults compareResult = range->compareNode(node, ec); if (!ec && (compareResult == Range::NODE_BEFORE_AND_AFTER || compareResult == Range::NODE_INSIDE)) { // If we did nothing here, when this node's renderer was destroyed, the rect that it // occupied would be invalidated, but, selection gaps that change as a result of // the removal wouldn't be invalidated. // FIXME: Don't do so much unnecessary invalidation. clearRenderTreeSelection = true; } } if (clearRenderTreeSelection) { RefPtr document = m_selection.start().anchorNode()->document(); document->updateStyleIfNeeded(); if (RenderView* view = toRenderView(document->renderer())) view->clearSelection(); } if (clearDOMTreeSelection) setSelection(VisibleSelection(), 0); } enum EndPointType { EndPointIsStart, EndPointIsEnd }; static bool shouldRemovePositionAfterAdoptingTextReplacement(Position& position, EndPointType type, CharacterData* node, unsigned offset, unsigned oldLength, unsigned newLength) { if (!position.anchorNode() || position.anchorNode() != node || position.anchorType() != Position::PositionIsOffsetInAnchor) return false; ASSERT(position.offsetInContainerNode() >= 0); unsigned positionOffset = static_cast(position.offsetInContainerNode()); if (positionOffset > offset && positionOffset < offset + oldLength) return true; // Adjust the offset if the position is after or at the end of the deleted contents (positionOffset >= offset + oldLength) // to avoid having a stale offset except when the position is the end of selection and nothing is deleted, in which case, // adjusting offset results in incorrectly extending the selection until the end of newly inserted contents. if ((positionOffset > offset + oldLength) || (positionOffset == offset + oldLength && (type == EndPointIsStart || oldLength))) position.moveToOffset(positionOffset - oldLength + newLength); return false; } void SelectionController::textWillBeReplaced(CharacterData* node, unsigned offset, unsigned oldLength, unsigned newLength) { // The fragment check is a performance optimization. See http://trac.webkit.org/changeset/30062. if (isNone() || !node || highestAncestor(node)->nodeType() == Node::DOCUMENT_FRAGMENT_NODE) return; Position base = m_selection.base(); Position extent = m_selection.extent(); Position start = m_selection.start(); Position end = m_selection.end(); bool shouldRemoveBase = shouldRemovePositionAfterAdoptingTextReplacement(base, m_selection.isBaseFirst() ? EndPointIsStart : EndPointIsEnd, node, offset, oldLength, newLength); bool shouldRemoveExtent = shouldRemovePositionAfterAdoptingTextReplacement(extent, m_selection.isBaseFirst() ? EndPointIsEnd : EndPointIsStart, node, offset, oldLength, newLength); bool shouldRemoveStart = shouldRemovePositionAfterAdoptingTextReplacement(start, EndPointIsStart, node, offset, oldLength, newLength); bool shouldRemoveEnd = shouldRemovePositionAfterAdoptingTextReplacement(end, EndPointIsEnd, node, offset, oldLength, newLength); if ((base != m_selection.base() || extent != m_selection.extent() || start != m_selection.start() || end != m_selection.end()) && !shouldRemoveStart && !shouldRemoveEnd) { VisibleSelection newSelection; if (!shouldRemoveBase && !shouldRemoveExtent) newSelection.setWithoutValidation(base, extent); else { if (newSelection.isBaseFirst()) newSelection.setWithoutValidation(start, end); else newSelection.setWithoutValidation(end, start); } m_frame->document()->updateLayout(); setSelection(newSelection, 0); return; } respondToNodeModification(node, shouldRemoveBase, shouldRemoveExtent, shouldRemoveStart, shouldRemoveEnd); } void SelectionController::setIsDirectional(bool isDirectional) { m_isDirectional = !m_frame || m_frame->editor()->behavior().shouldConsiderSelectionAsDirectional() || isDirectional; } TextDirection SelectionController::directionOfEnclosingBlock() { return WebCore::directionOfEnclosingBlock(m_selection.extent()); } void SelectionController::willBeModified(EAlteration alter, SelectionDirection direction) { if (alter != AlterationExtend) return; Position start = m_selection.start(); Position end = m_selection.end(); bool baseIsStart = true; if (m_isDirectional) { // Make base and extent match start and end so we extend the user-visible selection. // This only matters for cases where base and extend point to different positions than // start and end (e.g. after a double-click to select a word). if (m_selection.isBaseFirst()) baseIsStart = true; else baseIsStart = false; } else { switch (direction) { case DirectionRight: if (directionOfEnclosingBlock() == LTR) baseIsStart = true; else baseIsStart = false; break; case DirectionForward: baseIsStart = true; break; case DirectionLeft: if (directionOfEnclosingBlock() == LTR) baseIsStart = false; else baseIsStart = true; break; case DirectionBackward: baseIsStart = false; break; } } if (baseIsStart) { m_selection.setBase(start); m_selection.setExtent(end); } else { m_selection.setBase(end); m_selection.setExtent(start); } } VisiblePosition SelectionController::positionForPlatform(bool isGetStart) const { Settings* settings = m_frame ? m_frame->settings() : 0; if (settings && settings->editingBehaviorType() == EditingMacBehavior) return isGetStart ? m_selection.visibleStart() : m_selection.visibleEnd(); // Linux and Windows always extend selections from the extent endpoint. // FIXME: VisibleSelection should be fixed to ensure as an invariant that // base/extent always point to the same nodes as start/end, but which points // to which depends on the value of isBaseFirst. Then this can be changed // to just return m_sel.extent(). return m_selection.isBaseFirst() ? m_selection.visibleEnd() : m_selection.visibleStart(); } VisiblePosition SelectionController::startForPlatform() const { return positionForPlatform(true); } VisiblePosition SelectionController::endForPlatform() const { return positionForPlatform(false); } VisiblePosition SelectionController::modifyExtendingRight(TextGranularity granularity) { VisiblePosition pos(m_selection.extent(), m_selection.affinity()); // The difference between modifyExtendingRight and modifyExtendingForward is: // modifyExtendingForward always extends forward logically. // modifyExtendingRight behaves the same as modifyExtendingForward except for extending character or word, // it extends forward logically if the enclosing block is LTR direction, // but it extends backward logically if the enclosing block is RTL direction. switch (granularity) { case CharacterGranularity: if (directionOfEnclosingBlock() == LTR) pos = pos.next(CannotCrossEditingBoundary); else pos = pos.previous(CannotCrossEditingBoundary); break; case WordGranularity: if (directionOfEnclosingBlock() == LTR) pos = nextWordPosition(pos); else pos = previousWordPosition(pos); break; case LineBoundary: if (directionOfEnclosingBlock() == LTR) pos = modifyExtendingForward(granularity); else pos = modifyExtendingBackward(granularity); break; case SentenceGranularity: case LineGranularity: case ParagraphGranularity: case SentenceBoundary: case ParagraphBoundary: case DocumentBoundary: // FIXME: implement all of the above? pos = modifyExtendingForward(granularity); break; case WebKitVisualWordGranularity: break; } return pos; } VisiblePosition SelectionController::modifyExtendingForward(TextGranularity granularity) { VisiblePosition pos(m_selection.extent(), m_selection.affinity()); switch (granularity) { case CharacterGranularity: pos = pos.next(CannotCrossEditingBoundary); break; case WordGranularity: pos = nextWordPosition(pos); break; case SentenceGranularity: pos = nextSentencePosition(pos); break; case LineGranularity: pos = nextLinePosition(pos, xPosForVerticalArrowNavigation(EXTENT)); break; case ParagraphGranularity: pos = nextParagraphPosition(pos, xPosForVerticalArrowNavigation(EXTENT)); break; case SentenceBoundary: pos = endOfSentence(endForPlatform()); break; case LineBoundary: pos = logicalEndOfLine(endForPlatform()); break; case ParagraphBoundary: pos = endOfParagraph(endForPlatform()); break; case DocumentBoundary: pos = endForPlatform(); if (isEditablePosition(pos.deepEquivalent())) pos = endOfEditableContent(pos); else pos = endOfDocument(pos); break; case WebKitVisualWordGranularity: break; } return pos; } VisiblePosition SelectionController::modifyMovingRight(TextGranularity granularity) { VisiblePosition pos; switch (granularity) { case CharacterGranularity: if (isRange()) { if (directionOfEnclosingBlock() == LTR) pos = VisiblePosition(m_selection.end(), m_selection.affinity()); else pos = VisiblePosition(m_selection.start(), m_selection.affinity()); } else pos = VisiblePosition(m_selection.extent(), m_selection.affinity()).right(true); break; case WordGranularity: case SentenceGranularity: case LineGranularity: case ParagraphGranularity: case SentenceBoundary: case ParagraphBoundary: case DocumentBoundary: // FIXME: Implement all of the above. pos = modifyMovingForward(granularity); break; case LineBoundary: pos = rightBoundaryOfLine(startForPlatform(), directionOfEnclosingBlock()); break; case WebKitVisualWordGranularity: pos = rightWordPosition(VisiblePosition(m_selection.extent(), m_selection.affinity())); break; } return pos; } VisiblePosition SelectionController::modifyMovingForward(TextGranularity granularity) { VisiblePosition pos; // FIXME: Stay in editable content for the less common granularities. switch (granularity) { case CharacterGranularity: if (isRange()) pos = VisiblePosition(m_selection.end(), m_selection.affinity()); else pos = VisiblePosition(m_selection.extent(), m_selection.affinity()).next(CannotCrossEditingBoundary); break; case WordGranularity: pos = nextWordPosition(VisiblePosition(m_selection.extent(), m_selection.affinity())); break; case SentenceGranularity: pos = nextSentencePosition(VisiblePosition(m_selection.extent(), m_selection.affinity())); break; case LineGranularity: { // down-arrowing from a range selection that ends at the start of a line needs // to leave the selection at that line start (no need to call nextLinePosition!) pos = endForPlatform(); if (!isRange() || !isStartOfLine(pos)) pos = nextLinePosition(pos, xPosForVerticalArrowNavigation(START)); break; } case ParagraphGranularity: pos = nextParagraphPosition(endForPlatform(), xPosForVerticalArrowNavigation(START)); break; case SentenceBoundary: pos = endOfSentence(endForPlatform()); break; case LineBoundary: pos = logicalEndOfLine(endForPlatform()); break; case ParagraphBoundary: pos = endOfParagraph(endForPlatform()); break; case DocumentBoundary: pos = endForPlatform(); if (isEditablePosition(pos.deepEquivalent())) pos = endOfEditableContent(pos); else pos = endOfDocument(pos); break; case WebKitVisualWordGranularity: break; } return pos; } VisiblePosition SelectionController::modifyExtendingLeft(TextGranularity granularity) { VisiblePosition pos(m_selection.extent(), m_selection.affinity()); // The difference between modifyExtendingLeft and modifyExtendingBackward is: // modifyExtendingBackward always extends backward logically. // modifyExtendingLeft behaves the same as modifyExtendingBackward except for extending character or word, // it extends backward logically if the enclosing block is LTR direction, // but it extends forward logically if the enclosing block is RTL direction. switch (granularity) { case CharacterGranularity: if (directionOfEnclosingBlock() == LTR) pos = pos.previous(CannotCrossEditingBoundary); else pos = pos.next(CannotCrossEditingBoundary); break; case WordGranularity: if (directionOfEnclosingBlock() == LTR) pos = previousWordPosition(pos); else pos = nextWordPosition(pos); break; case LineBoundary: if (directionOfEnclosingBlock() == LTR) pos = modifyExtendingBackward(granularity); else pos = modifyExtendingForward(granularity); break; case SentenceGranularity: case LineGranularity: case ParagraphGranularity: case SentenceBoundary: case ParagraphBoundary: case DocumentBoundary: pos = modifyExtendingBackward(granularity); break; case WebKitVisualWordGranularity: break; } return pos; } VisiblePosition SelectionController::modifyExtendingBackward(TextGranularity granularity) { VisiblePosition pos(m_selection.extent(), m_selection.affinity()); // Extending a selection backward by word or character from just after a table selects // the table. This "makes sense" from the user perspective, esp. when deleting. // It was done here instead of in VisiblePosition because we want VPs to iterate // over everything. switch (granularity) { case CharacterGranularity: pos = pos.previous(CannotCrossEditingBoundary); break; case WordGranularity: pos = previousWordPosition(pos); break; case SentenceGranularity: pos = previousSentencePosition(pos); break; case LineGranularity: pos = previousLinePosition(pos, xPosForVerticalArrowNavigation(EXTENT)); break; case ParagraphGranularity: pos = previousParagraphPosition(pos, xPosForVerticalArrowNavigation(EXTENT)); break; case SentenceBoundary: pos = startOfSentence(startForPlatform()); break; case LineBoundary: pos = logicalStartOfLine(startForPlatform()); break; case ParagraphBoundary: pos = startOfParagraph(startForPlatform()); break; case DocumentBoundary: pos = startForPlatform(); if (isEditablePosition(pos.deepEquivalent())) pos = startOfEditableContent(pos); else pos = startOfDocument(pos); break; case WebKitVisualWordGranularity: break; } return pos; } VisiblePosition SelectionController::modifyMovingLeft(TextGranularity granularity) { VisiblePosition pos; switch (granularity) { case CharacterGranularity: if (isRange()) if (directionOfEnclosingBlock() == LTR) pos = VisiblePosition(m_selection.start(), m_selection.affinity()); else pos = VisiblePosition(m_selection.end(), m_selection.affinity()); else pos = VisiblePosition(m_selection.extent(), m_selection.affinity()).left(true); break; case WordGranularity: case SentenceGranularity: case LineGranularity: case ParagraphGranularity: case SentenceBoundary: case ParagraphBoundary: case DocumentBoundary: // FIXME: Implement all of the above. pos = modifyMovingBackward(granularity); break; case LineBoundary: pos = leftBoundaryOfLine(startForPlatform(), directionOfEnclosingBlock()); break; case WebKitVisualWordGranularity: pos = leftWordPosition(VisiblePosition(m_selection.extent(), m_selection.affinity())); break; } return pos; } VisiblePosition SelectionController::modifyMovingBackward(TextGranularity granularity) { VisiblePosition pos; switch (granularity) { case CharacterGranularity: if (isRange()) pos = VisiblePosition(m_selection.start(), m_selection.affinity()); else pos = VisiblePosition(m_selection.extent(), m_selection.affinity()).previous(CannotCrossEditingBoundary); break; case WordGranularity: pos = previousWordPosition(VisiblePosition(m_selection.extent(), m_selection.affinity())); break; case SentenceGranularity: pos = previousSentencePosition(VisiblePosition(m_selection.extent(), m_selection.affinity())); break; case LineGranularity: pos = previousLinePosition(startForPlatform(), xPosForVerticalArrowNavigation(START)); break; case ParagraphGranularity: pos = previousParagraphPosition(startForPlatform(), xPosForVerticalArrowNavigation(START)); break; case SentenceBoundary: pos = startOfSentence(startForPlatform()); break; case LineBoundary: pos = logicalStartOfLine(startForPlatform()); break; case ParagraphBoundary: pos = startOfParagraph(startForPlatform()); break; case DocumentBoundary: pos = startForPlatform(); if (isEditablePosition(pos.deepEquivalent())) pos = startOfEditableContent(pos); else pos = startOfDocument(pos); break; case WebKitVisualWordGranularity: break; } return pos; } static bool isBoundary(TextGranularity granularity) { return granularity == LineBoundary || granularity == ParagraphBoundary || granularity == DocumentBoundary; } bool SelectionController::modify(EAlteration alter, SelectionDirection direction, TextGranularity granularity, bool userTriggered) { if (userTriggered) { SelectionController trialSelectionController; trialSelectionController.setSelection(m_selection); trialSelectionController.setIsDirectional(m_isDirectional); trialSelectionController.modify(alter, direction, granularity, false); bool change = shouldChangeSelection(trialSelectionController.selection()); if (!change) return false; } willBeModified(alter, direction); bool wasRange = m_selection.isRange(); Position originalStartPosition = m_selection.start(); VisiblePosition position; switch (direction) { case DirectionRight: if (alter == AlterationMove) position = modifyMovingRight(granularity); else position = modifyExtendingRight(granularity); break; case DirectionForward: if (alter == AlterationExtend) position = modifyExtendingForward(granularity); else position = modifyMovingForward(granularity); break; case DirectionLeft: if (alter == AlterationMove) position = modifyMovingLeft(granularity); else position = modifyExtendingLeft(granularity); break; case DirectionBackward: if (alter == AlterationExtend) position = modifyExtendingBackward(granularity); else position = modifyMovingBackward(granularity); break; } if (position.isNull()) return false; if (isSpatialNavigationEnabled(m_frame)) if (!wasRange && alter == AlterationMove && position == originalStartPosition) return false; // Some of the above operations set an xPosForVerticalArrowNavigation. // Setting a selection will clear it, so save it to possibly restore later. // Note: the START position type is arbitrary because it is unused, it would be // the requested position type if there were no xPosForVerticalArrowNavigation set. int x = xPosForVerticalArrowNavigation(START); switch (alter) { case AlterationMove: moveTo(position, userTriggered); break; case AlterationExtend: // Standard Mac behavior when extending to a boundary is grow the selection rather than leaving the // base in place and moving the extent. Matches NSTextView. if (!m_frame || !m_frame->editor()->behavior().shouldAlwaysGrowSelectionWhenExtendingToBoundary() || m_selection.isCaret() || !isBoundary(granularity)) setExtent(position, userTriggered); else { TextDirection textDirection = directionOfEnclosingBlock(); if (direction == DirectionForward || (textDirection == LTR && direction == DirectionRight) || (textDirection == RTL && direction == DirectionLeft)) setEnd(position, userTriggered); else setStart(position, userTriggered); } break; } if (granularity == LineGranularity || granularity == ParagraphGranularity) m_xPosForVerticalArrowNavigation = x; if (userTriggered) m_granularity = CharacterGranularity; setCaretRectNeedsUpdate(); setIsDirectional(alter == AlterationExtend); return true; } // FIXME: Maybe baseline would be better? static bool absoluteCaretY(const VisiblePosition &c, int &y) { IntRect rect = c.absoluteCaretBounds(); if (rect.isEmpty()) return false; y = rect.y() + rect.height() / 2; return true; } bool SelectionController::modify(EAlteration alter, int verticalDistance, bool userTriggered, CursorAlignOnScroll align) { if (!verticalDistance) return false; if (userTriggered) { SelectionController trialSelectionController; trialSelectionController.setSelection(m_selection); trialSelectionController.setIsDirectional(m_isDirectional); trialSelectionController.modify(alter, verticalDistance, false); bool change = shouldChangeSelection(trialSelectionController.selection()); if (!change) return false; } bool up = verticalDistance < 0; if (up) verticalDistance = -verticalDistance; willBeModified(alter, up ? DirectionBackward : DirectionForward); VisiblePosition pos; int xPos = 0; switch (alter) { case AlterationMove: pos = VisiblePosition(up ? m_selection.start() : m_selection.end(), m_selection.affinity()); xPos = xPosForVerticalArrowNavigation(up ? START : END); m_selection.setAffinity(up ? UPSTREAM : DOWNSTREAM); break; case AlterationExtend: pos = VisiblePosition(m_selection.extent(), m_selection.affinity()); xPos = xPosForVerticalArrowNavigation(EXTENT); m_selection.setAffinity(DOWNSTREAM); break; } int startY; if (!absoluteCaretY(pos, startY)) return false; if (up) startY = -startY; int lastY = startY; VisiblePosition result; VisiblePosition next; for (VisiblePosition p = pos; ; p = next) { next = (up ? previousLinePosition : nextLinePosition)(p, xPos); if (next.isNull() || next == p) break; int nextY; if (!absoluteCaretY(next, nextY)) break; if (up) nextY = -nextY; if (nextY - startY > verticalDistance) break; if (nextY >= lastY) { lastY = nextY; result = next; } } if (result.isNull()) return false; switch (alter) { case AlterationMove: moveTo(result, userTriggered, align); break; case AlterationExtend: setExtent(result, userTriggered); break; } if (userTriggered) m_granularity = CharacterGranularity; setIsDirectional(alter == AlterationExtend); return true; } int SelectionController::xPosForVerticalArrowNavigation(EPositionType type) { int x = 0; if (isNone()) return x; Position pos; switch (type) { case START: pos = m_selection.start(); break; case END: pos = m_selection.end(); break; case BASE: pos = m_selection.base(); break; case EXTENT: pos = m_selection.extent(); break; } Frame* frame = pos.anchorNode()->document()->frame(); if (!frame) return x; if (m_xPosForVerticalArrowNavigation == NoXPosForVerticalArrowNavigation) { VisiblePosition visiblePosition(pos, m_selection.affinity()); // VisiblePosition creation can fail here if a node containing the selection becomes visibility:hidden // after the selection is created and before this function is called. x = visiblePosition.isNotNull() ? visiblePosition.xOffsetForVerticalNavigation() : 0; m_xPosForVerticalArrowNavigation = x; } else x = m_xPosForVerticalArrowNavigation; return x; } void SelectionController::clear() { m_granularity = CharacterGranularity; setSelection(VisibleSelection()); } void SelectionController::setStart(const VisiblePosition &pos, bool userTriggered) { if (m_selection.isBaseFirst()) setBase(pos, userTriggered); else setExtent(pos, userTriggered); } void SelectionController::setEnd(const VisiblePosition &pos, bool userTriggered) { if (m_selection.isBaseFirst()) setExtent(pos, userTriggered); else setBase(pos, userTriggered); } void SelectionController::setBase(const VisiblePosition &pos, bool userTriggered) { SetSelectionOptions options = CloseTyping | ClearTypingStyle; if (userTriggered) options |= UserTriggered; setSelection(VisibleSelection(pos.deepEquivalent(), m_selection.extent(), pos.affinity()), options); } void SelectionController::setExtent(const VisiblePosition &pos, bool userTriggered) { SetSelectionOptions options = CloseTyping | ClearTypingStyle; if (userTriggered) options |= UserTriggered; setSelection(VisibleSelection(m_selection.base(), pos.deepEquivalent(), pos.affinity()), options); } void SelectionController::setBase(const Position &pos, EAffinity affinity, bool userTriggered) { SetSelectionOptions options = CloseTyping | ClearTypingStyle; if (userTriggered) options |= UserTriggered; setSelection(VisibleSelection(pos, m_selection.extent(), affinity), options); } void SelectionController::setExtent(const Position &pos, EAffinity affinity, bool userTriggered) { SetSelectionOptions options = CloseTyping | ClearTypingStyle; if (userTriggered) options |= UserTriggered; setSelection(VisibleSelection(m_selection.base(), pos, affinity), options); } void SelectionController::setCaretRectNeedsUpdate(bool flag) { m_caretRectNeedsUpdate = flag; } void SelectionController::updateCaretRect() { if (isNone() || !m_selection.start().anchorNode()->inDocument() || !m_selection.end().anchorNode()->inDocument()) { m_caretRect = IntRect(); return; } m_selection.start().anchorNode()->document()->updateStyleIfNeeded(); m_caretRect = IntRect(); if (isCaret()) { VisiblePosition pos(m_selection.start(), m_selection.affinity()); if (pos.isNotNull()) { ASSERT(pos.deepEquivalent().deprecatedNode()->renderer()); // First compute a rect local to the renderer at the selection start RenderObject* renderer; IntRect localRect = pos.localCaretRect(renderer); // Get the renderer that will be responsible for painting the caret (which // is either the renderer we just found, or one of its containers) RenderObject* caretPainter = caretRenderer(); // Compute an offset between the renderer and the caretPainter bool unrooted = false; while (renderer != caretPainter) { RenderObject* containerObject = renderer->container(); if (!containerObject) { unrooted = true; break; } localRect.move(renderer->offsetFromContainer(containerObject, localRect.location())); renderer = containerObject; } if (!unrooted) m_caretRect = localRect; m_absCaretBoundsDirty = true; } } m_caretRectNeedsUpdate = false; } RenderObject* SelectionController::caretRenderer() const { Node* node = m_selection.start().deprecatedNode(); if (!node) return 0; RenderObject* renderer = node->renderer(); if (!renderer) return 0; // if caretNode is a block and caret is inside it then caret should be painted by that block bool paintedByBlock = renderer->isBlockFlow() && caretRendersInsideNode(node); return paintedByBlock ? renderer : renderer->containingBlock(); } IntRect SelectionController::localCaretRect() { if (m_caretRectNeedsUpdate) updateCaretRect(); return m_caretRect; } IntRect SelectionController::absoluteBoundsForLocalRect(const IntRect& rect) const { RenderObject* caretPainter = caretRenderer(); if (!caretPainter) return IntRect(); IntRect localRect(rect); if (caretPainter->isBox()) toRenderBox(caretPainter)->flipForWritingMode(localRect); return caretPainter->localToAbsoluteQuad(FloatRect(localRect)).enclosingBoundingBox(); } IntRect SelectionController::absoluteCaretBounds() { recomputeCaretRect(); return m_absCaretBounds; } static IntRect repaintRectForCaret(IntRect caret) { if (caret.isEmpty()) return IntRect(); // Ensure that the dirty rect intersects the block that paints the caret even in the case where // the caret itself is just outside the block. See . caret.inflateX(1); return caret; } IntRect SelectionController::caretRepaintRect() const { return absoluteBoundsForLocalRect(repaintRectForCaret(localCaretRectForPainting())); } bool SelectionController::recomputeCaretRect() { if (!m_caretRectNeedsUpdate) return false; if (!m_frame) return false; FrameView* v = m_frame->document()->view(); if (!v) return false; IntRect oldRect = m_caretRect; IntRect newRect = localCaretRect(); if (oldRect == newRect && !m_absCaretBoundsDirty) return false; IntRect oldAbsCaretBounds = m_absCaretBounds; // FIXME: Rename m_caretRect to m_localCaretRect. m_absCaretBounds = absoluteBoundsForLocalRect(m_caretRect); m_absCaretBoundsDirty = false; if (oldAbsCaretBounds == m_absCaretBounds) return false; IntRect oldAbsoluteCaretRepaintBounds = m_absoluteCaretRepaintBounds; // We believe that we need to inflate the local rect before transforming it to obtain the repaint bounds. m_absoluteCaretRepaintBounds = caretRepaintRect(); #if ENABLE(TEXT_CARET) if (RenderView* view = toRenderView(m_frame->document()->renderer())) { // FIXME: make caret repainting container-aware. view->repaintRectangleInViewAndCompositedLayers(oldAbsoluteCaretRepaintBounds, false); if (shouldRepaintCaret(view)) view->repaintRectangleInViewAndCompositedLayers(m_absoluteCaretRepaintBounds, false); } #endif return true; } bool SelectionController::shouldRepaintCaret(const RenderView* view) const { ASSERT(view); Frame* frame = view->frameView() ? view->frameView()->frame() : 0; // The frame where the selection started. bool caretBrowsing = frame && frame->settings() && frame->settings()->caretBrowsingEnabled(); return (caretBrowsing || isContentEditable()); } void SelectionController::invalidateCaretRect() { if (!isCaret()) return; Document* d = m_selection.start().anchorNode()->document(); // recomputeCaretRect will always return false for the drag caret, // because its m_frame is always 0. bool caretRectChanged = recomputeCaretRect(); // EDIT FIXME: This is an unfortunate hack. // Basically, we can't trust this layout position since we // can't guarantee that the check to see if we are in unrendered // content will work at this point. We may have to wait for // a layout and re-render of the document to happen. So, resetting this // flag will cause another caret layout to happen the first time // that we try to paint the caret after this call. That one will work since // it happens after the document has accounted for any editing // changes which may have been done. // And, we need to leave this layout here so the caret moves right // away after clicking. m_caretRectNeedsUpdate = true; if (!caretRectChanged) { RenderView* view = toRenderView(d->renderer()); if (view && shouldRepaintCaret(view)) view->repaintRectangleInViewAndCompositedLayers(caretRepaintRect(), false); } } void SelectionController::paintCaret(GraphicsContext* context, int tx, int ty, const IntRect& clipRect) { #if ENABLE(TEXT_CARET) if (!m_caretVisible) return; if (!m_caretPaint) return; if (!m_selection.isCaret()) return; IntRect drawingRect = localCaretRectForPainting(); if (caretRenderer() && caretRenderer()->isBox()) toRenderBox(caretRenderer())->flipForWritingMode(drawingRect); drawingRect.move(tx, ty); IntRect caret = intersection(drawingRect, clipRect); if (caret.isEmpty()) return; Color caretColor = Color::black; ColorSpace colorSpace = ColorSpaceDeviceRGB; Element* element = rootEditableElement(); if (element && element->renderer()) { caretColor = element->renderer()->style()->visitedDependentColor(CSSPropertyColor); colorSpace = element->renderer()->style()->colorSpace(); } context->fillRect(caret, caretColor, colorSpace); #else UNUSED_PARAM(context); UNUSED_PARAM(tx); UNUSED_PARAM(ty); UNUSED_PARAM(clipRect); #endif } void SelectionController::debugRenderer(RenderObject *r, bool selected) const { if (r->node()->isElementNode()) { Element* element = static_cast(r->node()); fprintf(stderr, "%s%s\n", selected ? "==> " : " ", element->localName().string().utf8().data()); } else if (r->isText()) { RenderText* textRenderer = toRenderText(r); if (!textRenderer->textLength() || !textRenderer->firstTextBox()) { fprintf(stderr, "%s#text (empty)\n", selected ? "==> " : " "); return; } static const int max = 36; String text = textRenderer->text(); int textLength = text.length(); if (selected) { int offset = 0; if (r->node() == m_selection.start().containerNode()) offset = m_selection.start().computeOffsetInContainerNode(); else if (r->node() == m_selection.end().containerNode()) offset = m_selection.end().computeOffsetInContainerNode(); int pos; InlineTextBox* box = textRenderer->findNextInlineTextBox(offset, pos); text = text.substring(box->start(), box->len()); String show; int mid = max / 2; int caret = 0; // text is shorter than max if (textLength < max) { show = text; caret = pos; } else if (pos - mid < 0) { // too few characters to left show = text.left(max - 3) + "..."; caret = pos; } else if (pos - mid >= 0 && pos + mid <= textLength) { // enough characters on each side show = "..." + text.substring(pos - mid + 3, max - 6) + "..."; caret = mid; } else { // too few characters on right show = "..." + text.right(max - 3); caret = pos - (textLength - show.length()); } show.replace('\n', ' '); show.replace('\r', ' '); fprintf(stderr, "==> #text : \"%s\" at offset %d\n", show.utf8().data(), pos); fprintf(stderr, " "); for (int i = 0; i < caret; i++) fprintf(stderr, " "); fprintf(stderr, "^\n"); } else { if ((int)text.length() > max) text = text.left(max - 3) + "..."; else text = text.left(max); fprintf(stderr, " #text : \"%s\"\n", text.utf8().data()); } } } bool SelectionController::contains(const IntPoint& point) { Document* document = m_frame->document(); // Treat a collapsed selection like no selection. if (!isRange()) return false; if (!document->renderer()) return false; HitTestRequest request(HitTestRequest::ReadOnly | HitTestRequest::Active); HitTestResult result(point); document->renderView()->layer()->hitTest(request, result); Node* innerNode = result.innerNode(); if (!innerNode || !innerNode->renderer()) return false; VisiblePosition visiblePos(innerNode->renderer()->positionForPoint(result.localPoint())); if (visiblePos.isNull()) return false; if (m_selection.visibleStart().isNull() || m_selection.visibleEnd().isNull()) return false; Position start(m_selection.visibleStart().deepEquivalent()); Position end(m_selection.visibleEnd().deepEquivalent()); Position p(visiblePos.deepEquivalent()); return comparePositions(start, p) <= 0 && comparePositions(p, end) <= 0; } // Workaround for the fact that it's hard to delete a frame. // Call this after doing user-triggered selections to make it easy to delete the frame you entirely selected. // Can't do this implicitly as part of every setSelection call because in some contexts it might not be good // for the focus to move to another frame. So instead we call it from places where we are selecting with the // mouse or the keyboard after setting the selection. void SelectionController::selectFrameElementInParentIfFullySelected() { // Find the parent frame; if there is none, then we have nothing to do. Frame* parent = m_frame->tree()->parent(); if (!parent) return; Page* page = m_frame->page(); if (!page) return; // Check if the selection contains the entire frame contents; if not, then there is nothing to do. if (!isRange()) return; if (!isStartOfDocument(selection().visibleStart())) return; if (!isEndOfDocument(selection().visibleEnd())) return; // Get to the