forked from mirrors/gecko-dev
		
	 81f5e14057
			
		
	
	
		81f5e14057
		
	
	
	
	
		
			
			This patch implements the `::target-text` pseudo element. Similarly to the Custom Highlight API, this is done implementing a new Selection type. Differential Revision: https://phabricator.services.mozilla.com/D195687
		
			
				
	
	
		
			3121 lines
		
	
	
	
		
			110 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			3121 lines
		
	
	
	
		
			110 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | |
| /* vim: set ts=8 sts=2 et sw=2 tw=80: */
 | |
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| /*
 | |
|  * Implementation of nsFrameSelection
 | |
|  */
 | |
| 
 | |
| #include "nsFrameSelection.h"
 | |
| 
 | |
| #include "ErrorList.h"
 | |
| #include "mozilla/intl/BidiEmbeddingLevel.h"
 | |
| #include "mozilla/Attributes.h"
 | |
| #include "mozilla/AutoRestore.h"
 | |
| #include "mozilla/BasePrincipal.h"
 | |
| #include "mozilla/HTMLEditor.h"
 | |
| #include "mozilla/IntegerRange.h"
 | |
| #include "mozilla/Logging.h"
 | |
| #include "mozilla/PresShell.h"
 | |
| #include "mozilla/ScrollTypes.h"
 | |
| #include "mozilla/StaticAnalysisFunctions.h"
 | |
| #include "mozilla/StaticPrefs_bidi.h"
 | |
| #include "mozilla/StaticPrefs_dom.h"
 | |
| #include "mozilla/StaticPrefs_layout.h"
 | |
| #include "mozilla/Unused.h"
 | |
| 
 | |
| #include "nsCOMPtr.h"
 | |
| #include "nsDebug.h"
 | |
| #include "nsFrameTraversal.h"
 | |
| #include "nsString.h"
 | |
| #include "nsISelectionListener.h"
 | |
| #include "nsDeviceContext.h"
 | |
| #include "nsIContent.h"
 | |
| #include "nsRange.h"
 | |
| #include "nsITableCellLayout.h"
 | |
| #include "nsTArray.h"
 | |
| #include "nsTableWrapperFrame.h"
 | |
| #include "nsTableCellFrame.h"
 | |
| #include "nsIScrollableFrame.h"
 | |
| #include "nsCCUncollectableMarker.h"
 | |
| #include "nsTextFragment.h"
 | |
| #include <algorithm>
 | |
| #include "nsContentUtils.h"
 | |
| #include "nsCSSFrameConstructor.h"
 | |
| 
 | |
| #include "nsGkAtoms.h"
 | |
| #include "nsLayoutUtils.h"
 | |
| #include "nsBidiPresUtils.h"
 | |
| #include "nsTextFrame.h"
 | |
| 
 | |
| #include "nsThreadUtils.h"
 | |
| #include "mozilla/Preferences.h"
 | |
| 
 | |
| #include "mozilla/PresShell.h"
 | |
| #include "nsPresContext.h"
 | |
| #include "nsCaret.h"
 | |
| 
 | |
| #include "mozilla/MouseEvents.h"
 | |
| #include "mozilla/TextEvents.h"
 | |
| 
 | |
| // notifications
 | |
| #include "mozilla/dom/Document.h"
 | |
| 
 | |
| #include "nsISelectionController.h"  //for the enums
 | |
| #include "nsCopySupport.h"
 | |
| #include "nsIClipboard.h"
 | |
| #include "nsIFrameInlines.h"
 | |
| 
 | |
| #include "nsError.h"
 | |
| #include "mozilla/AutoCopyListener.h"
 | |
| #include "mozilla/dom/Element.h"
 | |
| #include "mozilla/dom/Highlight.h"
 | |
| #include "mozilla/dom/Selection.h"
 | |
| #include "mozilla/dom/ShadowRoot.h"
 | |
| #include "mozilla/dom/StaticRange.h"
 | |
| #include "mozilla/dom/Text.h"
 | |
| #include "mozilla/ErrorResult.h"
 | |
| #include "mozilla/dom/SelectionBinding.h"
 | |
| #include "mozilla/AsyncEventDispatcher.h"
 | |
| #include "mozilla/Telemetry.h"
 | |
| 
 | |
| #include "nsFocusManager.h"
 | |
| #include "nsPIDOMWindow.h"
 | |
| 
 | |
| #include "SelectionMovementUtils.h"
 | |
| 
 | |
| using namespace mozilla;
 | |
| using namespace mozilla::dom;
 | |
| 
 | |
| static LazyLogModule sFrameSelectionLog("FrameSelection");
 | |
| 
 | |
| // #define DEBUG_TABLE 1
 | |
| 
 | |
| /**
 | |
|  * Add cells to the selection inside of the given cells range.
 | |
|  *
 | |
|  * @param  aTable             [in] HTML table element
 | |
|  * @param  aStartRowIndex     [in] row index where the cells range starts
 | |
|  * @param  aStartColumnIndex  [in] column index where the cells range starts
 | |
|  * @param  aEndRowIndex       [in] row index where the cells range ends
 | |
|  * @param  aEndColumnIndex    [in] column index where the cells range ends
 | |
|  */
 | |
| static nsresult AddCellsToSelection(const nsIContent* aTableContent,
 | |
|                                     int32_t aStartRowIndex,
 | |
|                                     int32_t aStartColumnIndex,
 | |
|                                     int32_t aEndRowIndex,
 | |
|                                     int32_t aEndColumnIndex,
 | |
|                                     Selection& aNormalSelection);
 | |
| 
 | |
| static nsAtom* GetTag(nsINode* aNode);
 | |
| 
 | |
| static nsINode* GetClosestInclusiveTableCellAncestor(nsINode* aDomNode);
 | |
| MOZ_CAN_RUN_SCRIPT_BOUNDARY static nsresult CreateAndAddRange(
 | |
|     nsINode* aContainer, int32_t aOffset, Selection& aNormalSelection);
 | |
| static nsresult SelectCellElement(nsIContent* aCellElement,
 | |
|                                   Selection& aNormalSelection);
 | |
| 
 | |
| #ifdef XP_MACOSX
 | |
| static nsresult UpdateSelectionCacheOnRepaintSelection(Selection* aSel);
 | |
| #endif  // XP_MACOSX
 | |
| 
 | |
| #ifdef PRINT_RANGE
 | |
| static void printRange(nsRange* aDomRange);
 | |
| #  define DEBUG_OUT_RANGE(x) printRange(x)
 | |
| #else
 | |
| #  define DEBUG_OUT_RANGE(x)
 | |
| #endif  // PRINT_RANGE
 | |
| 
 | |
| /******************************************************************************
 | |
|  * mozilla::PeekOffsetStruct
 | |
|  ******************************************************************************/
 | |
| 
 | |
| // #define DEBUG_SELECTION // uncomment for printf describing every collapse and
 | |
| //  extend. #define DEBUG_NAVIGATION
 | |
| 
 | |
| // #define DEBUG_TABLE_SELECTION 1
 | |
| 
 | |
| namespace mozilla {
 | |
| 
 | |
| PeekOffsetStruct::PeekOffsetStruct(nsSelectionAmount aAmount,
 | |
|                                    nsDirection aDirection, int32_t aStartOffset,
 | |
|                                    nsPoint aDesiredCaretPos,
 | |
|                                    const PeekOffsetOptions aOptions,
 | |
|                                    EWordMovementType aWordMovementType)
 | |
|     : mAmount(aAmount),
 | |
|       mDirection(aDirection),
 | |
|       mStartOffset(aStartOffset),
 | |
|       mDesiredCaretPos(aDesiredCaretPos),
 | |
|       mWordMovementType(aWordMovementType),
 | |
|       mOptions(aOptions),
 | |
|       mResultFrame(nullptr),
 | |
|       mContentOffset(0),
 | |
|       mAttach(CaretAssociationHint::Before) {}
 | |
| 
 | |
| }  // namespace mozilla
 | |
| 
 | |
| // Array which contains index of each SelectionType in
 | |
| // Selection::mDOMSelections. For avoiding using if nor switch to retrieve the
 | |
| // index, this needs to have -1 for SelectionTypes which won't be created its
 | |
| // Selection instance.
 | |
| static const int8_t kIndexOfSelections[] = {
 | |
|     -1,  // SelectionType::eInvalid
 | |
|     -1,  // SelectionType::eNone
 | |
|     0,   // SelectionType::eNormal
 | |
|     1,   // SelectionType::eSpellCheck
 | |
|     2,   // SelectionType::eIMERawClause
 | |
|     3,   // SelectionType::eIMESelectedRawClause
 | |
|     4,   // SelectionType::eIMEConvertedClause
 | |
|     5,   // SelectionType::eIMESelectedClause
 | |
|     6,   // SelectionType::eAccessibility
 | |
|     7,   // SelectionType::eFind
 | |
|     8,   // SelectionType::eURLSecondary
 | |
|     9,   // SelectionType::eURLStrikeout
 | |
|     10,  // SelectionType::eTargetText
 | |
|     -1,  // SelectionType::eHighlight
 | |
| };
 | |
| 
 | |
| inline int8_t GetIndexFromSelectionType(SelectionType aSelectionType) {
 | |
|   // The enum value of eInvalid is -1 and the others are sequential value
 | |
|   // starting from 0.  Therefore, |SelectionType + 1| is the index of
 | |
|   // kIndexOfSelections.
 | |
|   return kIndexOfSelections[static_cast<int8_t>(aSelectionType) + 1];
 | |
| }
 | |
| 
 | |
| /*
 | |
| The limiter is used specifically for the text areas and textfields
 | |
| In that case it is the DIV tag that is anonymously created for the text
 | |
| areas/fields.  Text nodes and BR nodes fall beneath it.  In the case of a
 | |
| BR node the limiter will be the parent and the offset will point before or
 | |
| after the BR node.  In the case of the text node the parent content is
 | |
| the text node itself and the offset will be the exact character position.
 | |
| The offset is not important to check for validity.  Simply look at the
 | |
| passed in content.  If it equals the limiter then the selection point is valid.
 | |
| If its parent it the limiter then the point is also valid.  In the case of
 | |
| NO limiter all points are valid since you are in a topmost iframe. (browser
 | |
| or composer)
 | |
| */
 | |
| bool nsFrameSelection::IsValidSelectionPoint(nsINode* aNode) const {
 | |
|   if (!aNode) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   nsIContent* limiter = GetLimiter();
 | |
|   if (limiter && limiter != aNode && limiter != aNode->GetParent()) {
 | |
|     // if newfocus == the limiter. that's ok. but if not there and not parent
 | |
|     // bad
 | |
|     return false;  // not in the right content. tLimiter said so
 | |
|   }
 | |
| 
 | |
|   limiter = GetAncestorLimiter();
 | |
|   return !limiter || aNode->IsInclusiveDescendantOf(limiter);
 | |
| }
 | |
| 
 | |
| namespace mozilla {
 | |
| struct MOZ_RAII AutoPrepareFocusRange {
 | |
|   AutoPrepareFocusRange(Selection* aSelection,
 | |
|                         const bool aMultiRangeSelection) {
 | |
|     MOZ_ASSERT(aSelection);
 | |
|     MOZ_ASSERT(aSelection->GetType() == SelectionType::eNormal);
 | |
| 
 | |
|     if (aSelection->mStyledRanges.mRanges.Length() <= 1) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (aSelection->mFrameSelection->IsUserSelectionReason()) {
 | |
|       mUserSelect.emplace(aSelection);
 | |
|     }
 | |
| 
 | |
|     nsTArray<StyledRange>& ranges = aSelection->mStyledRanges.mRanges;
 | |
|     if (!aSelection->mUserInitiated || aMultiRangeSelection) {
 | |
|       // Scripted command or the user is starting a new explicit multi-range
 | |
|       // selection.
 | |
|       for (StyledRange& entry : ranges) {
 | |
|         MOZ_ASSERT(entry.mRange->IsDynamicRange());
 | |
|         entry.mRange->AsDynamicRange()->SetIsGenerated(false);
 | |
|       }
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (!IsAnchorRelativeOperation(
 | |
|             aSelection->mFrameSelection->mSelectionChangeReasons)) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // This operation is against the anchor but our current mAnchorFocusRange
 | |
|     // represents the focus in a multi-range selection.  The anchor from a user
 | |
|     // perspective is the most distant generated range on the opposite side.
 | |
|     // Find that range and make it the mAnchorFocusRange.
 | |
|     nsRange* const newAnchorFocusRange =
 | |
|         FindGeneratedRangeMostDistantFromAnchor(*aSelection);
 | |
| 
 | |
|     if (!newAnchorFocusRange) {
 | |
|       // There are no generated ranges - that's fine.
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Setup the new mAnchorFocusRange and mark the old one as generated.
 | |
|     if (aSelection->mAnchorFocusRange) {
 | |
|       aSelection->mAnchorFocusRange->SetIsGenerated(true);
 | |
|     }
 | |
| 
 | |
|     newAnchorFocusRange->SetIsGenerated(false);
 | |
|     aSelection->mAnchorFocusRange = newAnchorFocusRange;
 | |
| 
 | |
|     RemoveGeneratedRanges(*aSelection);
 | |
| 
 | |
|     if (aSelection->mFrameSelection) {
 | |
|       aSelection->mFrameSelection->InvalidateDesiredCaretPos();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   static nsRange* FindGeneratedRangeMostDistantFromAnchor(
 | |
|       const Selection& aSelection) {
 | |
|     const nsTArray<StyledRange>& ranges = aSelection.mStyledRanges.mRanges;
 | |
|     const size_t len = ranges.Length();
 | |
|     nsRange* result{nullptr};
 | |
|     if (aSelection.GetDirection() == eDirNext) {
 | |
|       for (size_t i = 0; i < len; ++i) {
 | |
|         // This function is only called for selections with type == eNormal.
 | |
|         // (see MOZ_ASSERT in constructor).
 | |
|         // Therefore, all ranges must be dynamic.
 | |
|         if (ranges[i].mRange->AsDynamicRange()->IsGenerated()) {
 | |
|           result = ranges[i].mRange->AsDynamicRange();
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|     } else {
 | |
|       size_t i = len;
 | |
|       while (i--) {
 | |
|         if (ranges[i].mRange->AsDynamicRange()->IsGenerated()) {
 | |
|           result = ranges[i].mRange->AsDynamicRange();
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return result;
 | |
|   }
 | |
| 
 | |
|   static void RemoveGeneratedRanges(Selection& aSelection) {
 | |
|     RefPtr<nsPresContext> presContext = aSelection.GetPresContext();
 | |
|     nsTArray<StyledRange>& ranges = aSelection.mStyledRanges.mRanges;
 | |
|     size_t i = ranges.Length();
 | |
|     while (i--) {
 | |
|       // This function is only called for selections with type == eNormal.
 | |
|       // (see MOZ_ASSERT in constructor).
 | |
|       // Therefore, all ranges must be dynamic.
 | |
|       if (!ranges[i].mRange->IsDynamicRange()) {
 | |
|         continue;
 | |
|       }
 | |
|       nsRange* range = ranges[i].mRange->AsDynamicRange();
 | |
|       if (range->IsGenerated()) {
 | |
|         range->UnregisterSelection(aSelection);
 | |
|         aSelection.SelectFrames(presContext, *range, false);
 | |
|         ranges.RemoveElementAt(i);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * @aParam aSelectionChangeReasons can be multiple of the reasons defined in
 | |
|              nsISelectionListener.idl.
 | |
|    */
 | |
|   static bool IsAnchorRelativeOperation(const int16_t aSelectionChangeReasons) {
 | |
|     return aSelectionChangeReasons &
 | |
|            (nsISelectionListener::DRAG_REASON |
 | |
|             nsISelectionListener::MOUSEDOWN_REASON |
 | |
|             nsISelectionListener::MOUSEUP_REASON |
 | |
|             nsISelectionListener::COLLAPSETOSTART_REASON);
 | |
|   }
 | |
| 
 | |
|   Maybe<Selection::AutoUserInitiated> mUserSelect;
 | |
| };
 | |
| 
 | |
| }  // namespace mozilla
 | |
| 
 | |
| ////////////BEGIN nsFrameSelection methods
 | |
| 
 | |
| template Result<RefPtr<nsRange>, nsresult>
 | |
| nsFrameSelection::CreateRangeExtendedToSomewhere(
 | |
|     nsDirection aDirection, const nsSelectionAmount aAmount,
 | |
|     CaretMovementStyle aMovementStyle);
 | |
| template Result<RefPtr<StaticRange>, nsresult>
 | |
| nsFrameSelection::CreateRangeExtendedToSomewhere(
 | |
|     nsDirection aDirection, const nsSelectionAmount aAmount,
 | |
|     CaretMovementStyle aMovementStyle);
 | |
| 
 | |
| nsFrameSelection::nsFrameSelection(PresShell* aPresShell, nsIContent* aLimiter,
 | |
|                                    const bool aAccessibleCaretEnabled) {
 | |
|   for (size_t i = 0; i < ArrayLength(mDomSelections); i++) {
 | |
|     mDomSelections[i] = new Selection(kPresentSelectionTypes[i], this);
 | |
|   }
 | |
| 
 | |
| #ifdef XP_MACOSX
 | |
|   // On macOS, cache the current selection to send to service menu of macOS.
 | |
|   bool enableAutoCopy = true;
 | |
|   AutoCopyListener::Init(nsIClipboard::kSelectionCache);
 | |
| #else   // #ifdef XP_MACOSX
 | |
|   // Check to see if the auto-copy pref is enabled and make the normal
 | |
|   // Selection notifies auto-copy listener of its changes.
 | |
|   bool enableAutoCopy = AutoCopyListener::IsPrefEnabled();
 | |
|   if (enableAutoCopy) {
 | |
|     AutoCopyListener::Init(nsIClipboard::kSelectionClipboard);
 | |
|   }
 | |
| #endif  // #ifdef XP_MACOSX #else
 | |
| 
 | |
|   if (enableAutoCopy) {
 | |
|     int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|     if (mDomSelections[index]) {
 | |
|       mDomSelections[index]->NotifyAutoCopy();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   mPresShell = aPresShell;
 | |
|   mDragState = false;
 | |
|   mLimiters.mLimiter = aLimiter;
 | |
| 
 | |
|   // This should only ever be initialized on the main thread, so we are OK here.
 | |
|   MOZ_ASSERT(NS_IsMainThread());
 | |
| 
 | |
|   int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
| 
 | |
|   mAccessibleCaretEnabled = aAccessibleCaretEnabled;
 | |
|   if (mAccessibleCaretEnabled) {
 | |
|     mDomSelections[index]->MaybeNotifyAccessibleCaretEventHub(aPresShell);
 | |
|   }
 | |
| 
 | |
|   if (mDomSelections[index]) {
 | |
|     mDomSelections[index]->EnableSelectionChangeEvent();
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsFrameSelection::~nsFrameSelection() = default;
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_CLASS(nsFrameSelection)
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(nsFrameSelection)
 | |
|   for (size_t i = 0; i < ArrayLength(tmp->mDomSelections); ++i) {
 | |
|     tmp->mDomSelections[i] = nullptr;
 | |
|   }
 | |
|   tmp->mHighlightSelections.Clear();
 | |
| 
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(
 | |
|       mTableSelection.mClosestInclusiveTableCellAncestor)
 | |
|   tmp->mTableSelection.mMode = TableSelectionMode::None;
 | |
|   tmp->mTableSelection.mDragSelectingCells = false;
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mTableSelection.mStartSelectedCell)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mTableSelection.mEndSelectedCell)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mTableSelection.mAppendStartSelectedCell)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mTableSelection.mUnselectCellOnMouseUp)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mMaintainedRange.mRange)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mLimiters.mLimiter)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mLimiters.mAncestorLimiter)
 | |
| NS_IMPL_CYCLE_COLLECTION_UNLINK_END
 | |
| NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(nsFrameSelection)
 | |
|   if (tmp->mPresShell && tmp->mPresShell->GetDocument() &&
 | |
|       nsCCUncollectableMarker::InGeneration(
 | |
|           cb, tmp->mPresShell->GetDocument()->GetMarkedCCGeneration())) {
 | |
|     return NS_SUCCESS_INTERRUPTED_TRAVERSE;
 | |
|   }
 | |
|   for (size_t i = 0; i < ArrayLength(tmp->mDomSelections); ++i) {
 | |
|     NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mDomSelections[i])
 | |
|   }
 | |
| 
 | |
|   for (const auto& value : tmp->mHighlightSelections) {
 | |
|     CycleCollectionNoteChild(cb, value.second().get(),
 | |
|                              "mHighlightSelections[]");
 | |
|   }
 | |
| 
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(
 | |
|       mTableSelection.mClosestInclusiveTableCellAncestor)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTableSelection.mStartSelectedCell)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTableSelection.mEndSelectedCell)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTableSelection.mAppendStartSelectedCell)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTableSelection.mUnselectCellOnMouseUp)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mMaintainedRange.mRange)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mLimiters.mLimiter)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mLimiters.mAncestorLimiter)
 | |
| NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
 | |
| 
 | |
| bool nsFrameSelection::Caret::IsVisualMovement(
 | |
|     bool aContinueSelection, CaretMovementStyle aMovementStyle) const {
 | |
|   int32_t movementFlag = StaticPrefs::bidi_edit_caret_movement_style();
 | |
|   return aMovementStyle == eVisual ||
 | |
|          (aMovementStyle == eUsePrefStyle &&
 | |
|           (movementFlag == 1 || (movementFlag == 2 && !aContinueSelection)));
 | |
| }
 | |
| 
 | |
| // Get the x (or y, in vertical writing mode) position requested
 | |
| // by the Key Handling for line-up/down
 | |
| nsresult nsFrameSelection::DesiredCaretPos::FetchPos(
 | |
|     nsPoint& aDesiredCaretPos, const PresShell& aPresShell,
 | |
|     Selection& aNormalSelection) const {
 | |
|   MOZ_ASSERT(aNormalSelection.GetType() == SelectionType::eNormal);
 | |
| 
 | |
|   if (mIsSet) {
 | |
|     aDesiredCaretPos = mValue;
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   RefPtr<nsCaret> caret = aPresShell.GetCaret();
 | |
|   if (!caret) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   caret->SetSelection(&aNormalSelection);
 | |
| 
 | |
|   nsRect coord;
 | |
|   nsIFrame* caretFrame = caret->GetGeometry(&coord);
 | |
|   if (!caretFrame) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
|   nsPoint viewOffset(0, 0);
 | |
|   nsView* view = nullptr;
 | |
|   caretFrame->GetOffsetFromView(viewOffset, &view);
 | |
|   if (view) {
 | |
|     coord += viewOffset;
 | |
|   }
 | |
|   aDesiredCaretPos = coord.TopLeft();
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::InvalidateDesiredCaretPos()  // do not listen to
 | |
|                                                     // mDesiredCaretPos.mValue;
 | |
|                                                     // you must get another.
 | |
| {
 | |
|   mDesiredCaretPos.Invalidate();
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::DesiredCaretPos::Invalidate() { mIsSet = false; }
 | |
| 
 | |
| void nsFrameSelection::DesiredCaretPos::Set(const nsPoint& aPos) {
 | |
|   mValue = aPos;
 | |
|   mIsSet = true;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::ConstrainFrameAndPointToAnchorSubtree(
 | |
|     nsIFrame* aFrame, const nsPoint& aPoint, nsIFrame** aRetFrame,
 | |
|     nsPoint& aRetPoint) const {
 | |
|   //
 | |
|   // The whole point of this method is to return a frame and point that
 | |
|   // that lie within the same valid subtree as the anchor node's frame,
 | |
|   // for use with the method GetContentAndOffsetsFromPoint().
 | |
|   //
 | |
|   // A valid subtree is defined to be one where all the content nodes in
 | |
|   // the tree have a valid parent-child relationship.
 | |
|   //
 | |
|   // If the anchor frame and aFrame are in the same subtree, aFrame will
 | |
|   // be returned in aRetFrame. If they are in different subtrees, we
 | |
|   // return the frame for the root of the subtree.
 | |
|   //
 | |
| 
 | |
|   if (!aFrame || !aRetFrame) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   *aRetFrame = aFrame;
 | |
|   aRetPoint = aPoint;
 | |
| 
 | |
|   //
 | |
|   // Get the frame and content for the selection's anchor point!
 | |
|   //
 | |
| 
 | |
|   int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   if (!mDomSelections[index]) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   nsCOMPtr<nsIContent> anchorContent =
 | |
|       do_QueryInterface(mDomSelections[index]->GetAnchorNode());
 | |
|   if (!anchorContent) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Now find the root of the subtree containing the anchor's content.
 | |
|   //
 | |
| 
 | |
|   NS_ENSURE_STATE(mPresShell);
 | |
|   RefPtr<PresShell> presShell = mPresShell;
 | |
|   nsIContent* anchorRoot =
 | |
|       anchorContent
 | |
|           ->GetSelectionRootContent(
 | |
|               presShell,
 | |
|               StaticPrefs::dom_shadowdom_selection_across_boundary_enabled() /* aAllowCrossShadowBoundary */);
 | |
|   NS_ENSURE_TRUE(anchorRoot, NS_ERROR_UNEXPECTED);
 | |
| 
 | |
|   //
 | |
|   // Now find the root of the subtree containing aFrame's content.
 | |
|   //
 | |
| 
 | |
|   nsCOMPtr<nsIContent> content = aFrame->GetContent();
 | |
| 
 | |
|   if (content) {
 | |
|     nsIContent* contentRoot =
 | |
|         content->GetSelectionRootContent(
 | |
|             presShell, StaticPrefs::
 | |
|                            dom_shadowdom_selection_across_boundary_enabled() /* aAllowCrossShadowBoundary */);
 | |
|     NS_ENSURE_TRUE(contentRoot, NS_ERROR_UNEXPECTED);
 | |
| 
 | |
|     if (anchorRoot == contentRoot) {
 | |
|       // If the aFrame's content isn't the capturing content, it should be
 | |
|       // a descendant.  At this time, we can return simply.
 | |
|       nsIContent* capturedContent = PresShell::GetCapturingContent();
 | |
|       if (capturedContent != content) {
 | |
|         return NS_OK;
 | |
|       }
 | |
| 
 | |
|       // Find the frame under the mouse cursor with the root frame.
 | |
|       // At this time, don't use the anchor's frame because it may not have
 | |
|       // fixed positioned frames.
 | |
|       nsIFrame* rootFrame = presShell->GetRootFrame();
 | |
|       nsPoint ptInRoot = aPoint + aFrame->GetOffsetTo(rootFrame);
 | |
|       nsIFrame* cursorFrame =
 | |
|           nsLayoutUtils::GetFrameForPoint(RelativeTo{rootFrame}, ptInRoot);
 | |
| 
 | |
|       // If the mouse cursor in on a frame which is descendant of same
 | |
|       // selection root, we can expand the selection to the frame.
 | |
|       if (cursorFrame && cursorFrame->PresShell() == presShell) {
 | |
|         nsCOMPtr<nsIContent> cursorContent = cursorFrame->GetContent();
 | |
|         NS_ENSURE_TRUE(cursorContent, NS_ERROR_FAILURE);
 | |
|         nsIContent* cursorContentRoot = cursorContent->GetSelectionRootContent(
 | |
|             presShell, StaticPrefs::
 | |
|                            dom_shadowdom_selection_across_boundary_enabled() /* aAllowCrossShadowBoundary */);
 | |
|         NS_ENSURE_TRUE(cursorContentRoot, NS_ERROR_UNEXPECTED);
 | |
|         if (cursorContentRoot == anchorRoot) {
 | |
|           *aRetFrame = cursorFrame;
 | |
|           aRetPoint = aPoint + aFrame->GetOffsetTo(cursorFrame);
 | |
|           return NS_OK;
 | |
|         }
 | |
|       }
 | |
|       // Otherwise, e.g., the cursor isn't on any frames (e.g., the mouse
 | |
|       // cursor is out of the window), we should use the frame of the anchor
 | |
|       // root.
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // When we can't find a frame which is under the mouse cursor and has a same
 | |
|   // selection root as the anchor node's, we should return the selection root
 | |
|   // frame.
 | |
|   //
 | |
| 
 | |
|   *aRetFrame = anchorRoot->GetPrimaryFrame();
 | |
| 
 | |
|   if (!*aRetFrame) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Now make sure that aRetPoint is converted to the same coordinate
 | |
|   // system used by aRetFrame.
 | |
|   //
 | |
| 
 | |
|   aRetPoint = aPoint + aFrame->GetOffsetTo(*aRetFrame);
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::SetCaretBidiLevelAndMaybeSchedulePaint(
 | |
|     mozilla::intl::BidiEmbeddingLevel aLevel) {
 | |
|   // If the current level is undefined, we have just inserted new text.
 | |
|   // In this case, we don't want to reset the keyboard language
 | |
|   mCaret.mBidiLevel = aLevel;
 | |
| 
 | |
|   RefPtr<nsCaret> caret;
 | |
|   if (mPresShell && (caret = mPresShell->GetCaret())) {
 | |
|     caret->SchedulePaint();
 | |
|   }
 | |
| }
 | |
| 
 | |
| mozilla::intl::BidiEmbeddingLevel nsFrameSelection::GetCaretBidiLevel() const {
 | |
|   return mCaret.mBidiLevel;
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::UndefineCaretBidiLevel() {
 | |
|   mCaret.mBidiLevel = mozilla::intl::BidiEmbeddingLevel(mCaret.mBidiLevel |
 | |
|                                                         BIDI_LEVEL_UNDEFINED);
 | |
| }
 | |
| 
 | |
| #ifdef PRINT_RANGE
 | |
| void printRange(nsRange* aDomRange) {
 | |
|   if (!aDomRange) {
 | |
|     printf("NULL Range\n");
 | |
|   }
 | |
|   nsINode* startNode = aDomRange->GetStartContainer();
 | |
|   nsINode* endNode = aDomRange->GetEndContainer();
 | |
|   int32_t startOffset = aDomRange->StartOffset();
 | |
|   int32_t endOffset = aDomRange->EndOffset();
 | |
| 
 | |
|   printf("range: 0x%lx\t start: 0x%lx %ld, \t end: 0x%lx,%ld\n",
 | |
|          (unsigned long)aDomRange, (unsigned long)startNode, (long)startOffset,
 | |
|          (unsigned long)endNode, (long)endOffset);
 | |
| }
 | |
| #endif /* PRINT_RANGE */
 | |
| 
 | |
| static nsAtom* GetTag(nsINode* aNode) {
 | |
|   nsCOMPtr<nsIContent> content = do_QueryInterface(aNode);
 | |
|   if (!content) {
 | |
|     MOZ_ASSERT_UNREACHABLE("bad node passed to GetTag()");
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   return content->NodeInfo()->NameAtom();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * https://dom.spec.whatwg.org/#concept-tree-inclusive-ancestor.
 | |
|  */
 | |
| static nsINode* GetClosestInclusiveTableCellAncestor(nsINode* aDomNode) {
 | |
|   if (!aDomNode) return nullptr;
 | |
|   nsINode* current = aDomNode;
 | |
|   // Start with current node and look for a table cell
 | |
|   while (current) {
 | |
|     nsAtom* tag = GetTag(current);
 | |
|     if (tag == nsGkAtoms::td || tag == nsGkAtoms::th) return current;
 | |
|     current = current->GetParent();
 | |
|   }
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| static nsDirection GetCaretDirection(const nsIFrame& aFrame,
 | |
|                                      nsDirection aDirection,
 | |
|                                      bool aVisualMovement) {
 | |
|   const mozilla::intl::BidiDirection paragraphDirection =
 | |
|       nsBidiPresUtils::ParagraphDirection(&aFrame);
 | |
|   return (aVisualMovement &&
 | |
|           paragraphDirection == mozilla::intl::BidiDirection::RTL)
 | |
|              ? nsDirection(1 - aDirection)
 | |
|              : aDirection;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::MoveCaret(nsDirection aDirection,
 | |
|                                      bool aContinueSelection,
 | |
|                                      const nsSelectionAmount aAmount,
 | |
|                                      CaretMovementStyle aMovementStyle) {
 | |
|   NS_ENSURE_STATE(mPresShell);
 | |
|   // Flush out layout, since we need it to be up to date to do caret
 | |
|   // positioning.
 | |
|   OwningNonNull<PresShell> presShell(*mPresShell);
 | |
|   presShell->FlushPendingNotifications(FlushType::Layout);
 | |
| 
 | |
|   if (!mPresShell) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   nsPresContext* context = mPresShell->GetPresContext();
 | |
|   if (!context) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   const int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   const RefPtr<Selection> sel = mDomSelections[index];
 | |
|   if (!sel) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   int32_t scrollFlags = Selection::SCROLL_FOR_CARET_MOVE;
 | |
|   if (sel->IsEditorSelection()) {
 | |
|     // If caret moves in editor, it should cause scrolling even if it's in
 | |
|     // overflow: hidden;.
 | |
|     scrollFlags |= Selection::SCROLL_OVERFLOW_HIDDEN;
 | |
|   }
 | |
| 
 | |
|   const bool doCollapse = [&] {
 | |
|     if (sel->IsCollapsed() || aContinueSelection) {
 | |
|       return false;
 | |
|     }
 | |
|     if (aAmount > eSelectLine) {
 | |
|       return false;
 | |
|     }
 | |
|     int32_t caretStyle = StaticPrefs::layout_selection_caret_style();
 | |
|     return caretStyle == 2 || (caretStyle == 0 && aAmount != eSelectLine);
 | |
|   }();
 | |
| 
 | |
|   if (doCollapse) {
 | |
|     if (aDirection == eDirPrevious) {
 | |
|       SetChangeReasons(nsISelectionListener::COLLAPSETOSTART_REASON);
 | |
|       mCaret.mHint = CaretAssociationHint::After;
 | |
|     } else {
 | |
|       SetChangeReasons(nsISelectionListener::COLLAPSETOEND_REASON);
 | |
|       mCaret.mHint = CaretAssociationHint::Before;
 | |
|     }
 | |
|   } else {
 | |
|     SetChangeReasons(nsISelectionListener::KEYPRESS_REASON);
 | |
|   }
 | |
| 
 | |
|   mCaretMoveAmount = aAmount;
 | |
| 
 | |
|   AutoPrepareFocusRange prep(sel, false);
 | |
| 
 | |
|   // we must keep this around and revalidate it when its just UP/DOWN
 | |
|   nsPoint desiredPos(0, 0);
 | |
| 
 | |
|   if (aAmount == eSelectLine) {
 | |
|     nsresult result = mDesiredCaretPos.FetchPos(desiredPos, *mPresShell, *sel);
 | |
|     if (NS_FAILED(result)) {
 | |
|       return result;
 | |
|     }
 | |
|     mDesiredCaretPos.Set(desiredPos);
 | |
|   }
 | |
| 
 | |
|   bool visualMovement =
 | |
|       mCaret.IsVisualMovement(aContinueSelection, aMovementStyle);
 | |
|   const PrimaryFrameData frameForFocus =
 | |
|       sel->GetPrimaryFrameForCaretAtFocusNode(visualMovement);
 | |
|   if (!frameForFocus.mFrame) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
|   if (visualMovement) {
 | |
|     // FYI: This was done during a call of GetPrimaryFrameForCaretAtFocusNode.
 | |
|     // Therefore, this may not be intended by the original author.
 | |
|     SetHint(frameForFocus.mHint);
 | |
|   }
 | |
| 
 | |
|   Result<bool, nsresult> isIntraLineCaretMove =
 | |
|       SelectionMovementUtils::IsIntraLineCaretMove(aAmount);
 | |
|   nsDirection direction{aDirection};
 | |
|   if (isIntraLineCaretMove.isErr()) {
 | |
|     return isIntraLineCaretMove.unwrapErr();
 | |
|   }
 | |
|   if (isIntraLineCaretMove.inspect()) {
 | |
|     // Forget old caret position for moving caret to different line since
 | |
|     // caret position may be changed.
 | |
|     mDesiredCaretPos.Invalidate();
 | |
|     direction =
 | |
|         GetCaretDirection(*frameForFocus.mFrame, aDirection, visualMovement);
 | |
|   }
 | |
| 
 | |
|   if (doCollapse) {
 | |
|     const nsRange* anchorFocusRange = sel->GetAnchorFocusRange();
 | |
|     if (anchorFocusRange) {
 | |
|       RefPtr<nsINode> node;
 | |
|       uint32_t offset;
 | |
|       if (visualMovement &&
 | |
|           nsBidiPresUtils::IsReversedDirectionFrame(frameForFocus.mFrame)) {
 | |
|         direction = nsDirection(1 - direction);
 | |
|       }
 | |
|       if (direction == eDirPrevious) {
 | |
|         node = anchorFocusRange->GetStartContainer();
 | |
|         offset = anchorFocusRange->StartOffset();
 | |
|       } else {
 | |
|         node = anchorFocusRange->GetEndContainer();
 | |
|         offset = anchorFocusRange->EndOffset();
 | |
|       }
 | |
|       sel->CollapseInLimiter(node, offset);
 | |
|     }
 | |
|     sel->ScrollIntoView(nsISelectionController::SELECTION_FOCUS_REGION,
 | |
|                         ScrollAxis(), ScrollAxis(), scrollFlags);
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   CaretAssociationHint tHint(
 | |
|       mCaret.mHint);  // temporary variable so we dont set
 | |
|                       // mCaret.mHint until it is necessary
 | |
| 
 | |
|   Result<PeekOffsetStruct, nsresult> result = PeekOffsetForCaretMove(
 | |
|       direction, aContinueSelection, aAmount, aMovementStyle, desiredPos);
 | |
|   nsresult rv;
 | |
|   if (result.isOk() && result.inspect().mResultContent) {
 | |
|     const PeekOffsetStruct& pos = result.inspect();
 | |
|     nsIFrame* theFrame;
 | |
|     int32_t frameStart, frameEnd;
 | |
| 
 | |
|     if (aAmount <= eSelectWordNoSpace) {
 | |
|       // For left/right, PeekOffset() sets pos.mResultFrame correctly, but does
 | |
|       // not set pos.mAttachForward, so determine the hint here based on the
 | |
|       // result frame and offset: If we're at the end of a text frame, set the
 | |
|       // hint to ASSOCIATE_BEFORE to indicate that we want the caret displayed
 | |
|       // at the end of this frame, not at the beginning of the next one.
 | |
|       theFrame = pos.mResultFrame;
 | |
|       std::tie(frameStart, frameEnd) = theFrame->GetOffsets();
 | |
|       if (frameEnd == pos.mContentOffset && !(frameStart == 0 && frameEnd == 0))
 | |
|         tHint = CaretAssociationHint::Before;
 | |
|       else
 | |
|         tHint = CaretAssociationHint::After;
 | |
|     } else {
 | |
|       // For up/down and home/end, pos.mResultFrame might not be set correctly,
 | |
|       // or not at all. In these cases, get the frame based on the content and
 | |
|       // hint returned by PeekOffset().
 | |
|       tHint = pos.mAttach;
 | |
|       theFrame = SelectionMovementUtils::GetFrameForNodeOffset(
 | |
|           pos.mResultContent, pos.mContentOffset, tHint);
 | |
|       if (!theFrame) return NS_ERROR_FAILURE;
 | |
| 
 | |
|       std::tie(frameStart, frameEnd) = theFrame->GetOffsets();
 | |
|     }
 | |
| 
 | |
|     if (context->BidiEnabled()) {
 | |
|       switch (aAmount) {
 | |
|         case eSelectBeginLine:
 | |
|         case eSelectEndLine: {
 | |
|           // In Bidi contexts, PeekOffset calculates pos.mContentOffset
 | |
|           // differently depending on whether the movement is visual or logical.
 | |
|           // For visual movement, pos.mContentOffset depends on the direction-
 | |
|           // ality of the first/last frame on the line (theFrame), and the caret
 | |
|           // directionality must correspond.
 | |
|           FrameBidiData bidiData = theFrame->GetBidiData();
 | |
|           SetCaretBidiLevelAndMaybeSchedulePaint(
 | |
|               visualMovement ? bidiData.embeddingLevel : bidiData.baseLevel);
 | |
|           break;
 | |
|         }
 | |
|         default:
 | |
|           // If the current position is not a frame boundary, it's enough just
 | |
|           // to take the Bidi level of the current frame
 | |
|           if ((pos.mContentOffset != frameStart &&
 | |
|                pos.mContentOffset != frameEnd) ||
 | |
|               eSelectLine == aAmount) {
 | |
|             SetCaretBidiLevelAndMaybeSchedulePaint(
 | |
|                 theFrame->GetEmbeddingLevel());
 | |
|           } else {
 | |
|             BidiLevelFromMove(mPresShell, pos.mResultContent,
 | |
|                               pos.mContentOffset, aAmount, tHint);
 | |
|           }
 | |
|       }
 | |
|     }
 | |
|     // "pos" is on the stack, so pos.mResultContent has stack lifetime, so using
 | |
|     // MOZ_KnownLive is ok.
 | |
|     const FocusMode focusMode = aContinueSelection
 | |
|                                     ? FocusMode::kExtendSelection
 | |
|                                     : FocusMode::kCollapseToNewPoint;
 | |
|     rv = TakeFocus(MOZ_KnownLive(*pos.mResultContent), pos.mContentOffset,
 | |
|                    pos.mContentOffset, tHint, focusMode);
 | |
|   } else if (aAmount <= eSelectWordNoSpace && direction == eDirNext &&
 | |
|              !aContinueSelection) {
 | |
|     // Collapse selection if PeekOffset failed, we either
 | |
|     //  1. bumped into the BRFrame, bug 207623
 | |
|     //  2. had select-all in a text input (DIV range), bug 352759.
 | |
|     bool isBRFrame = frameForFocus.mFrame->IsBrFrame();
 | |
|     RefPtr<nsINode> node = sel->GetFocusNode();
 | |
|     sel->CollapseInLimiter(node, sel->FocusOffset());
 | |
|     // Note: 'frameForFocus.mFrame' might be dead here.
 | |
|     if (!isBRFrame) {
 | |
|       mCaret.mHint = CaretAssociationHint::Before;  // We're now at the end of
 | |
|                                                     // the frame to the left.
 | |
|     }
 | |
|     rv = NS_OK;
 | |
|   } else {
 | |
|     rv = result.isErr() ? result.unwrapErr() : NS_OK;
 | |
|   }
 | |
|   if (NS_SUCCEEDED(rv)) {
 | |
|     rv = sel->ScrollIntoView(nsISelectionController::SELECTION_FOCUS_REGION,
 | |
|                              ScrollAxis(), ScrollAxis(), scrollFlags);
 | |
|   }
 | |
| 
 | |
|   return rv;
 | |
| }
 | |
| 
 | |
| Result<PeekOffsetStruct, nsresult> nsFrameSelection::PeekOffsetForCaretMove(
 | |
|     nsDirection aDirection, bool aContinueSelection,
 | |
|     const nsSelectionAmount aAmount, CaretMovementStyle aMovementStyle,
 | |
|     const nsPoint& aDesiredCaretPos) const {
 | |
|   if (!mPresShell) {
 | |
|     return Err(NS_ERROR_NULL_POINTER);
 | |
|   }
 | |
| 
 | |
|   Selection* selection =
 | |
|       mDomSelections[GetIndexFromSelectionType(SelectionType::eNormal)];
 | |
|   if (!selection) {
 | |
|     return Err(NS_ERROR_NULL_POINTER);
 | |
|   }
 | |
| 
 | |
|   nsIContent* content = nsIContent::FromNodeOrNull(selection->GetFocusNode());
 | |
|   if (!content) {
 | |
|     return Err(NS_ERROR_FAILURE);
 | |
|   }
 | |
|   MOZ_ASSERT(mPresShell->GetDocument() == content->GetComposedDoc());
 | |
| 
 | |
|   const bool visualMovement =
 | |
|       mCaret.IsVisualMovement(aContinueSelection, aMovementStyle);
 | |
| 
 | |
|   PeekOffsetOptions options;
 | |
|   // set data using mLimiters.mLimiter to stop on scroll views.  If we have a
 | |
|   // limiter then we stop peeking when we hit scrollable views.  If no limiter
 | |
|   // then just let it go ahead
 | |
|   if (mLimiters.mLimiter) {
 | |
|     options += PeekOffsetOption::StopAtScroller;
 | |
|   }
 | |
|   if (visualMovement) {
 | |
|     options += PeekOffsetOption::Visual;
 | |
|   }
 | |
|   if (aContinueSelection) {
 | |
|     options += PeekOffsetOption::Extend;
 | |
|   }
 | |
|   if (selection->IsEditorSelection()) {
 | |
|     options += PeekOffsetOption::ForceEditableRegion;
 | |
|   }
 | |
| 
 | |
|   return SelectionMovementUtils::PeekOffsetForCaretMove(
 | |
|       content, selection->FocusOffset(), aDirection, GetHint(),
 | |
|       GetCaretBidiLevel(), aAmount, aDesiredCaretPos, options);
 | |
| }
 | |
| 
 | |
| nsPrevNextBidiLevels nsFrameSelection::GetPrevNextBidiLevels(
 | |
|     nsIContent* aNode, uint32_t aContentOffset, bool aJumpLines) const {
 | |
|   return SelectionMovementUtils::GetPrevNextBidiLevels(
 | |
|       aNode, aContentOffset, mCaret.mHint, aJumpLines);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::MaintainSelection(nsSelectionAmount aAmount) {
 | |
|   int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   if (!mDomSelections[index]) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   mMaintainedRange.MaintainAnchorFocusRange(*mDomSelections[index], aAmount);
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::BidiLevelFromMove(PresShell* aPresShell,
 | |
|                                          nsIContent* aNode,
 | |
|                                          uint32_t aContentOffset,
 | |
|                                          nsSelectionAmount aAmount,
 | |
|                                          CaretAssociationHint aHint) {
 | |
|   switch (aAmount) {
 | |
|     // Movement within the line: the new cursor Bidi level is the level of the
 | |
|     // last character moved over
 | |
|     case eSelectCharacter:
 | |
|     case eSelectCluster:
 | |
|     case eSelectWord:
 | |
|     case eSelectWordNoSpace:
 | |
|     case eSelectBeginLine:
 | |
|     case eSelectEndLine:
 | |
|     case eSelectNoAmount: {
 | |
|       nsPrevNextBidiLevels levels =
 | |
|           SelectionMovementUtils::GetPrevNextBidiLevels(aNode, aContentOffset,
 | |
|                                                         aHint, false);
 | |
| 
 | |
|       SetCaretBidiLevelAndMaybeSchedulePaint(
 | |
|           aHint == CaretAssociationHint::Before ? levels.mLevelBefore
 | |
|                                                 : levels.mLevelAfter);
 | |
|       break;
 | |
|     }
 | |
|       /*
 | |
|     // Up and Down: the new cursor Bidi level is the smaller of the two
 | |
|     surrounding characters case eSelectLine: case eSelectParagraph:
 | |
|       GetPrevNextBidiLevels(aContext, aNode, aContentOffset, &firstFrame,
 | |
|     &secondFrame, &firstLevel, &secondLevel);
 | |
|       aPresShell->SetCaretBidiLevelAndMaybeSchedulePaint(std::min(firstLevel,
 | |
|     secondLevel)); break;
 | |
|       */
 | |
| 
 | |
|     default:
 | |
|       UndefineCaretBidiLevel();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::BidiLevelFromClick(nsIContent* aNode,
 | |
|                                           uint32_t aContentOffset) {
 | |
|   nsIFrame* clickInFrame = nullptr;
 | |
|   clickInFrame = SelectionMovementUtils::GetFrameForNodeOffset(
 | |
|       aNode, aContentOffset, mCaret.mHint);
 | |
|   if (!clickInFrame) return;
 | |
| 
 | |
|   SetCaretBidiLevelAndMaybeSchedulePaint(clickInFrame->GetEmbeddingLevel());
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::MaintainedRange::AdjustNormalSelection(
 | |
|     const nsIContent* aContent, const int32_t aOffset,
 | |
|     Selection& aNormalSelection) const {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   if (!mRange || !aContent) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsINode* rangeStartNode = mRange->GetStartContainer();
 | |
|   nsINode* rangeEndNode = mRange->GetEndContainer();
 | |
|   const uint32_t rangeStartOffset = mRange->StartOffset();
 | |
|   const uint32_t rangeEndOffset = mRange->EndOffset();
 | |
| 
 | |
|   NS_ASSERTION(aOffset >= 0, "aOffset should not be negative");
 | |
|   const Maybe<int32_t> relToStart =
 | |
|       nsContentUtils::ComparePoints_AllowNegativeOffsets(
 | |
|           rangeStartNode, rangeStartOffset, aContent, aOffset);
 | |
|   if (NS_WARN_IF(!relToStart)) {
 | |
|     // Potentially handle this properly when Selection across Shadow DOM
 | |
|     // boundary is implemented
 | |
|     // (https://bugzilla.mozilla.org/show_bug.cgi?id=1607497).
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   const Maybe<int32_t> relToEnd =
 | |
|       nsContentUtils::ComparePoints_AllowNegativeOffsets(
 | |
|           rangeEndNode, rangeEndOffset, aContent, aOffset);
 | |
|   if (NS_WARN_IF(!relToEnd)) {
 | |
|     // Potentially handle this properly when Selection across Shadow DOM
 | |
|     // boundary is implemented
 | |
|     // (https://bugzilla.mozilla.org/show_bug.cgi?id=1607497).
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // If aContent/aOffset is inside (or at the edge of) the maintained
 | |
|   // selection, or if it is on the "anchor" side of the maintained selection,
 | |
|   // we need to do something.
 | |
|   if ((*relToStart <= 0 && *relToEnd >= 0) ||
 | |
|       (*relToStart > 0 && aNormalSelection.GetDirection() == eDirNext) ||
 | |
|       (*relToEnd < 0 && aNormalSelection.GetDirection() == eDirPrevious)) {
 | |
|     // Set the current range to the maintained range.
 | |
|     aNormalSelection.ReplaceAnchorFocusRange(mRange);
 | |
|     // Set the direction of the selection so that the anchor will be on the
 | |
|     // far side of the maintained selection, relative to aContent/aOffset.
 | |
|     aNormalSelection.SetDirection(*relToStart > 0 ? eDirPrevious : eDirNext);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::MaintainedRange::AdjustContentOffsets(
 | |
|     nsIFrame::ContentOffsets& aOffsets, StopAtScroller aStopAtScroller) const {
 | |
|   // Adjust offsets according to maintained amount
 | |
|   if (mRange && mAmount != eSelectNoAmount) {
 | |
|     nsINode* rangenode = mRange->GetStartContainer();
 | |
|     int32_t rangeOffset = mRange->StartOffset();
 | |
|     const Maybe<int32_t> relativePosition = nsContentUtils::ComparePoints(
 | |
|         rangenode, rangeOffset, aOffsets.content, aOffsets.offset);
 | |
|     if (NS_WARN_IF(!relativePosition)) {
 | |
|       // Potentially handle this properly when Selection across Shadow DOM
 | |
|       // boundary is implemented
 | |
|       // (https://bugzilla.mozilla.org/show_bug.cgi?id=1607497).
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     nsDirection direction = *relativePosition > 0 ? eDirPrevious : eDirNext;
 | |
|     nsSelectionAmount amount = mAmount;
 | |
|     if (amount == eSelectBeginLine && direction == eDirNext) {
 | |
|       amount = eSelectEndLine;
 | |
|     }
 | |
| 
 | |
|     uint32_t offset;
 | |
|     nsIFrame* frame = SelectionMovementUtils::GetFrameForNodeOffset(
 | |
|         aOffsets.content, aOffsets.offset, CaretAssociationHint::After,
 | |
|         &offset);
 | |
| 
 | |
|     PeekOffsetOptions peekOffsetOptions{};
 | |
|     if (aStopAtScroller == StopAtScroller::Yes) {
 | |
|       peekOffsetOptions += PeekOffsetOption::StopAtScroller;
 | |
|     }
 | |
|     if (frame && amount == eSelectWord && direction == eDirPrevious) {
 | |
|       // To avoid selecting the previous word when at start of word,
 | |
|       // first move one character forward.
 | |
|       PeekOffsetStruct charPos(eSelectCharacter, eDirNext,
 | |
|                                static_cast<int32_t>(offset), nsPoint(0, 0),
 | |
|                                peekOffsetOptions);
 | |
|       if (NS_SUCCEEDED(frame->PeekOffset(&charPos))) {
 | |
|         frame = charPos.mResultFrame;
 | |
|         offset = charPos.mContentOffset;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     PeekOffsetStruct pos(amount, direction, static_cast<int32_t>(offset),
 | |
|                          nsPoint(0, 0), peekOffsetOptions);
 | |
|     if (frame && NS_SUCCEEDED(frame->PeekOffset(&pos)) && pos.mResultContent) {
 | |
|       aOffsets.content = pos.mResultContent;
 | |
|       aOffsets.offset = pos.mContentOffset;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::MaintainedRange::MaintainAnchorFocusRange(
 | |
|     const Selection& aNormalSelection, const nsSelectionAmount aAmount) {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   mAmount = aAmount;
 | |
| 
 | |
|   const nsRange* anchorFocusRange = aNormalSelection.GetAnchorFocusRange();
 | |
|   if (anchorFocusRange && aAmount != eSelectNoAmount) {
 | |
|     mRange = anchorFocusRange->CloneRange();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   mRange = nullptr;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::HandleClick(nsIContent* aNewFocus,
 | |
|                                        uint32_t aContentOffset,
 | |
|                                        uint32_t aContentEndOffset,
 | |
|                                        const FocusMode aFocusMode,
 | |
|                                        CaretAssociationHint aHint) {
 | |
|   if (!aNewFocus) return NS_ERROR_INVALID_ARG;
 | |
| 
 | |
|   if (MOZ_LOG_TEST(sFrameSelectionLog, LogLevel::Debug)) {
 | |
|     const int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|     MOZ_LOG(sFrameSelectionLog, LogLevel::Debug,
 | |
|             ("%s: selection=%p, new focus=%p, offsets=(%u,%u), focus mode=%i",
 | |
|              __FUNCTION__,
 | |
|              mDomSelections[index] ? mDomSelections[index].get() : nullptr,
 | |
|              aNewFocus, aContentOffset, aContentEndOffset,
 | |
|              static_cast<int>(aFocusMode)));
 | |
|   }
 | |
| 
 | |
|   mDesiredCaretPos.Invalidate();
 | |
| 
 | |
|   if (aFocusMode != FocusMode::kExtendSelection) {
 | |
|     mMaintainedRange.mRange = nullptr;
 | |
|     if (!IsValidSelectionPoint(aNewFocus)) {
 | |
|       mLimiters.mAncestorLimiter = nullptr;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Don't take focus when dragging off of a table
 | |
|   if (!mTableSelection.mDragSelectingCells) {
 | |
|     BidiLevelFromClick(aNewFocus, aContentOffset);
 | |
|     SetChangeReasons(nsISelectionListener::MOUSEDOWN_REASON +
 | |
|                      nsISelectionListener::DRAG_REASON);
 | |
| 
 | |
|     const int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|     RefPtr<Selection> selection = mDomSelections[index];
 | |
|     MOZ_ASSERT(selection);
 | |
| 
 | |
|     if (aFocusMode == FocusMode::kExtendSelection) {
 | |
|       mMaintainedRange.AdjustNormalSelection(aNewFocus, aContentOffset,
 | |
|                                              *selection);
 | |
|     }
 | |
| 
 | |
|     AutoPrepareFocusRange prep(selection,
 | |
|                                aFocusMode == FocusMode::kMultiRangeSelection);
 | |
|     return TakeFocus(*aNewFocus, aContentOffset, aContentEndOffset, aHint,
 | |
|                      aFocusMode);
 | |
|   }
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::HandleDrag(nsIFrame* aFrame, const nsPoint& aPoint) {
 | |
|   if (!aFrame || !mPresShell) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsresult result;
 | |
|   nsIFrame* newFrame = 0;
 | |
|   nsPoint newPoint;
 | |
| 
 | |
|   result = ConstrainFrameAndPointToAnchorSubtree(aFrame, aPoint, &newFrame,
 | |
|                                                  newPoint);
 | |
|   if (NS_FAILED(result)) return;
 | |
|   if (!newFrame) return;
 | |
| 
 | |
|   nsIFrame::ContentOffsets offsets =
 | |
|       newFrame->GetContentOffsetsFromPoint(newPoint);
 | |
|   if (!offsets.content) return;
 | |
| 
 | |
|   const int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   RefPtr<Selection> selection = mDomSelections[index];
 | |
|   if (newFrame->IsSelected() && selection) {
 | |
|     // `MOZ_KnownLive` required because of
 | |
|     // https://bugzilla.mozilla.org/show_bug.cgi?id=1636889.
 | |
|     mMaintainedRange.AdjustNormalSelection(MOZ_KnownLive(offsets.content),
 | |
|                                            offsets.offset, *selection);
 | |
|   }
 | |
| 
 | |
|   mMaintainedRange.AdjustContentOffsets(
 | |
|       offsets, mLimiters.mLimiter ? MaintainedRange::StopAtScroller::Yes
 | |
|                                   : MaintainedRange::StopAtScroller::No);
 | |
| 
 | |
|   // TODO: no click has happened, rename `HandleClick`.
 | |
|   HandleClick(MOZ_KnownLive(offsets.content) /* bug 1636889 */, offsets.offset,
 | |
|               offsets.offset, FocusMode::kExtendSelection, offsets.associate);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::StartAutoScrollTimer(nsIFrame* aFrame,
 | |
|                                                 const nsPoint& aPoint,
 | |
|                                                 uint32_t aDelay) {
 | |
|   int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   if (!mDomSelections[index]) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   RefPtr<Selection> selection = mDomSelections[index];
 | |
|   return selection->StartAutoScrollTimer(aFrame, aPoint, aDelay);
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::StopAutoScrollTimer() {
 | |
|   int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   if (!mDomSelections[index]) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   mDomSelections[index]->StopAutoScrollTimer();
 | |
| }
 | |
| 
 | |
| // static
 | |
| nsINode* nsFrameSelection::TableSelection::IsContentInActivelyEditableTableCell(
 | |
|     nsPresContext* aContext, nsIContent* aContent) {
 | |
|   if (!aContext) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   RefPtr<HTMLEditor> htmlEditor = nsContentUtils::GetHTMLEditor(aContext);
 | |
|   if (!htmlEditor) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   nsINode* inclusiveTableCellAncestor =
 | |
|       GetClosestInclusiveTableCellAncestor(aContent);
 | |
|   if (!inclusiveTableCellAncestor) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   const Element* editingHost = htmlEditor->ComputeEditingHost();
 | |
|   if (!editingHost) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   const bool editableCell =
 | |
|       inclusiveTableCellAncestor->IsInclusiveDescendantOf(editingHost);
 | |
|   return editableCell ? inclusiveTableCellAncestor : nullptr;
 | |
| }
 | |
| 
 | |
| namespace {
 | |
| struct ParentAndOffset {
 | |
|   explicit ParentAndOffset(const nsINode& aNode)
 | |
|       : mParent{aNode.GetParent()},
 | |
|         mOffset{mParent ? mParent->ComputeIndexOf_Deprecated(&aNode) : 0} {}
 | |
| 
 | |
|   nsINode* mParent;
 | |
| 
 | |
|   // 0, if there's no parent.
 | |
|   int32_t mOffset;
 | |
| };
 | |
| 
 | |
| }  // namespace
 | |
| /**
 | |
| hard to go from nodes to frames, easy the other way!
 | |
|  */
 | |
| nsresult nsFrameSelection::TakeFocus(nsIContent& aNewFocus,
 | |
|                                      uint32_t aContentOffset,
 | |
|                                      uint32_t aContentEndOffset,
 | |
|                                      CaretAssociationHint aHint,
 | |
|                                      const FocusMode aFocusMode) {
 | |
|   NS_ENSURE_STATE(mPresShell);
 | |
| 
 | |
|   if (!IsValidSelectionPoint(&aNewFocus)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   MOZ_LOG(sFrameSelectionLog, LogLevel::Verbose,
 | |
|           ("%s: new focus=%p, offsets=(%u, %u), hint=%i, focusMode=%i",
 | |
|            __FUNCTION__, &aNewFocus, aContentOffset, aContentEndOffset,
 | |
|            static_cast<int>(aHint), static_cast<int>(aFocusMode)));
 | |
| 
 | |
|   mPresShell->FrameSelectionWillTakeFocus(*this);
 | |
| 
 | |
|   // Clear all table selection data
 | |
|   mTableSelection.mMode = TableSelectionMode::None;
 | |
|   mTableSelection.mDragSelectingCells = false;
 | |
|   mTableSelection.mStartSelectedCell = nullptr;
 | |
|   mTableSelection.mEndSelectedCell = nullptr;
 | |
|   mTableSelection.mAppendStartSelectedCell = nullptr;
 | |
|   mCaret.mHint = aHint;
 | |
| 
 | |
|   int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   if (!mDomSelections[index]) return NS_ERROR_NULL_POINTER;
 | |
| 
 | |
|   Maybe<Selection::AutoUserInitiated> userSelect;
 | |
|   if (IsUserSelectionReason()) {
 | |
|     userSelect.emplace(mDomSelections[index]);
 | |
|   }
 | |
| 
 | |
|   // traverse through document and unselect crap here
 | |
|   switch (aFocusMode) {
 | |
|     case FocusMode::kCollapseToNewPoint:
 | |
|       [[fallthrough]];
 | |
|     case FocusMode::kMultiRangeSelection: {
 | |
|       // single click? setting cursor down
 | |
|       const Batching saveBatching =
 | |
|           mBatching;  // hack to use the collapse code.
 | |
|       mBatching.mCounter = 1;
 | |
| 
 | |
|       RefPtr<Selection> selection = mDomSelections[index];
 | |
| 
 | |
|       if (aFocusMode == FocusMode::kMultiRangeSelection) {
 | |
|         // Remove existing collapsed ranges as there's no point in having
 | |
|         // non-anchor/focus collapsed ranges.
 | |
|         selection->RemoveCollapsedRanges();
 | |
| 
 | |
|         ErrorResult error;
 | |
|         RefPtr<nsRange> newRange = nsRange::Create(
 | |
|             &aNewFocus, aContentOffset, &aNewFocus, aContentOffset, error);
 | |
|         if (NS_WARN_IF(error.Failed())) {
 | |
|           return error.StealNSResult();
 | |
|         }
 | |
|         MOZ_ASSERT(newRange);
 | |
|         selection->AddRangeAndSelectFramesAndNotifyListeners(*newRange,
 | |
|                                                              IgnoreErrors());
 | |
|       } else {
 | |
|         bool oldDesiredPosSet =
 | |
|             mDesiredCaretPos.mIsSet;  // need to keep old desired
 | |
|                                       // position if it was set.
 | |
|         selection->CollapseInLimiter(&aNewFocus, aContentOffset);
 | |
|         mDesiredCaretPos.mIsSet =
 | |
|             oldDesiredPosSet;  // now reset desired pos back.
 | |
|       }
 | |
| 
 | |
|       mBatching = saveBatching;
 | |
| 
 | |
|       if (aContentEndOffset != aContentOffset) {
 | |
|         selection->Extend(&aNewFocus, aContentEndOffset);
 | |
|       }
 | |
| 
 | |
|       // find out if we are inside a table. if so, find out which one and which
 | |
|       // cell once we do that, the next time we get a takefocus, check the
 | |
|       // parent tree. if we are no longer inside same table ,cell then switch to
 | |
|       // table selection mode. BUT only do this in an editor
 | |
| 
 | |
|       NS_ENSURE_STATE(mPresShell);
 | |
|       RefPtr<nsPresContext> context = mPresShell->GetPresContext();
 | |
|       mTableSelection.mClosestInclusiveTableCellAncestor = nullptr;
 | |
|       if (nsINode* inclusiveTableCellAncestor =
 | |
|               TableSelection::IsContentInActivelyEditableTableCell(
 | |
|                   context, &aNewFocus)) {
 | |
|         mTableSelection.mClosestInclusiveTableCellAncestor =
 | |
|             inclusiveTableCellAncestor;
 | |
|         MOZ_LOG(sFrameSelectionLog, LogLevel::Debug,
 | |
|                 ("%s: Collapsing into new cell", __FUNCTION__));
 | |
|       }
 | |
| 
 | |
|       break;
 | |
|     }
 | |
|     case FocusMode::kExtendSelection: {
 | |
|       // Now update the range list:
 | |
|       nsINode* inclusiveTableCellAncestor =
 | |
|           GetClosestInclusiveTableCellAncestor(&aNewFocus);
 | |
|       if (mTableSelection.mClosestInclusiveTableCellAncestor &&
 | |
|           inclusiveTableCellAncestor &&
 | |
|           inclusiveTableCellAncestor !=
 | |
|               mTableSelection
 | |
|                   .mClosestInclusiveTableCellAncestor)  // switch to cell
 | |
|                                                         // selection mode
 | |
|       {
 | |
|         MOZ_LOG(sFrameSelectionLog, LogLevel::Debug,
 | |
|                 ("%s: moving into new cell", __FUNCTION__));
 | |
| 
 | |
|         WidgetMouseEvent event(false, eVoidEvent, nullptr,
 | |
|                                WidgetMouseEvent::eReal);
 | |
| 
 | |
|         // Start selecting in the cell we were in before
 | |
|         ParentAndOffset parentAndOffset{
 | |
|             *mTableSelection.mClosestInclusiveTableCellAncestor};
 | |
|         if (parentAndOffset.mParent) {
 | |
|           const nsresult result = HandleTableSelection(
 | |
|               parentAndOffset.mParent, parentAndOffset.mOffset,
 | |
|               TableSelectionMode::Cell, &event);
 | |
|           if (NS_WARN_IF(NS_FAILED(result))) {
 | |
|             return result;
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         // Find the parent of this new cell and extend selection to it
 | |
|         parentAndOffset = ParentAndOffset{*inclusiveTableCellAncestor};
 | |
| 
 | |
|         // XXXX We need to REALLY get the current key shift state
 | |
|         //  (we'd need to add event listener -- let's not bother for now)
 | |
|         event.mModifiers &= ~MODIFIER_SHIFT;  // aContinueSelection;
 | |
|         if (parentAndOffset.mParent) {
 | |
|           mTableSelection.mClosestInclusiveTableCellAncestor =
 | |
|               inclusiveTableCellAncestor;
 | |
|           // Continue selection into next cell
 | |
|           const nsresult result = HandleTableSelection(
 | |
|               parentAndOffset.mParent, parentAndOffset.mOffset,
 | |
|               TableSelectionMode::Cell, &event);
 | |
|           if (NS_WARN_IF(NS_FAILED(result))) {
 | |
|             return result;
 | |
|           }
 | |
|         }
 | |
|       } else {
 | |
|         RefPtr<Selection> selection = mDomSelections[index];
 | |
|         // XXXX Problem: Shift+click in browser is appending text selection to
 | |
|         // selected table!!!
 | |
|         //   is this the place to erase selected cells ?????
 | |
|         uint32_t offset =
 | |
|             (selection->GetDirection() == eDirNext &&
 | |
|              aContentEndOffset > aContentOffset)  // didn't go far enough
 | |
|                 ? aContentEndOffset  // this will only redraw the diff
 | |
|                 : aContentOffset;
 | |
|         selection->Extend(&aNewFocus, offset);
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Don't notify selection listeners if batching is on:
 | |
|   if (IsBatching()) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // Be aware, the Selection instance may be destroyed after this call.
 | |
|   return NotifySelectionListeners(SelectionType::eNormal);
 | |
| }
 | |
| 
 | |
| UniquePtr<SelectionDetails> nsFrameSelection::LookUpSelection(
 | |
|     nsIContent* aContent, int32_t aContentOffset, int32_t aContentLength,
 | |
|     bool aSlowCheck) const {
 | |
|   if (!aContent || !mPresShell) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   // TODO: Layout should use `uint32_t` for handling offset in DOM nodes
 | |
|   //       (for example: bug 1735262)
 | |
|   MOZ_ASSERT(aContentOffset >= 0);
 | |
|   MOZ_ASSERT(aContentLength >= 0);
 | |
|   if (MOZ_UNLIKELY(aContentOffset < 0) || MOZ_UNLIKELY(aContentLength < 0)) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   UniquePtr<SelectionDetails> details;
 | |
| 
 | |
|   for (size_t j = 0; j < ArrayLength(mDomSelections); j++) {
 | |
|     if (mDomSelections[j]) {
 | |
|       details = mDomSelections[j]->LookUpSelection(
 | |
|           aContent, static_cast<uint32_t>(aContentOffset),
 | |
|           static_cast<uint32_t>(aContentLength), std::move(details),
 | |
|           kPresentSelectionTypes[j], aSlowCheck);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // This may seem counter intuitive at first. Highlight selections need to be
 | |
|   // iterated from back to front:
 | |
|   //
 | |
|   //  - `mHighlightSelections` is ordered by insertion, i.e. if two or more
 | |
|   //  highlights overlap, the latest must take precedence.
 | |
|   //  - however, the `LookupSelection()` algorithm reverses the order by setting
 | |
|   //    the current `details` as `mNext`.
 | |
|   for (const auto& iter : Reversed(mHighlightSelections)) {
 | |
|     details = iter.second()->LookUpSelection(
 | |
|         aContent, static_cast<uint32_t>(aContentOffset),
 | |
|         static_cast<uint32_t>(aContentLength), std::move(details),
 | |
|         SelectionType::eHighlight, aSlowCheck);
 | |
|   }
 | |
| 
 | |
|   return details;
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::SetDragState(bool aState) {
 | |
|   if (mDragState == aState) return;
 | |
| 
 | |
|   mDragState = aState;
 | |
| 
 | |
|   if (!mDragState) {
 | |
|     mTableSelection.mDragSelectingCells = false;
 | |
|     // Notify that reason is mouse up.
 | |
|     SetChangeReasons(nsISelectionListener::MOUSEUP_REASON);
 | |
| 
 | |
|     // flag is set to NotApplicable in `Selection::NotifySelectionListeners`.
 | |
|     // since this function call is part of click event, this would immediately
 | |
|     // reset the flag, rendering it useless.
 | |
|     AutoRestore<ClickSelectionType> restoreClickSelectionType(
 | |
|         mClickSelectionType);
 | |
|     // Be aware, the Selection instance may be destroyed after this call.
 | |
|     NotifySelectionListeners(SelectionType::eNormal);
 | |
|   }
 | |
| }
 | |
| 
 | |
| Selection* nsFrameSelection::GetSelection(SelectionType aSelectionType) const {
 | |
|   int8_t index = GetIndexFromSelectionType(aSelectionType);
 | |
|   if (index < 0) return nullptr;
 | |
| 
 | |
|   return mDomSelections[index];
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::AddHighlightSelection(
 | |
|     nsAtom* aHighlightName, mozilla::dom::Highlight& aHighlight) {
 | |
|   RefPtr<Selection> selection =
 | |
|       aHighlight.CreateHighlightSelection(aHighlightName, this);
 | |
|   if (auto iter =
 | |
|           std::find_if(mHighlightSelections.begin(), mHighlightSelections.end(),
 | |
|                        [&aHighlightName](auto const& aElm) {
 | |
|                          return aElm.first() == aHighlightName;
 | |
|                        });
 | |
|       iter != mHighlightSelections.end()) {
 | |
|     iter->second() = std::move(selection);
 | |
|   } else {
 | |
|     mHighlightSelections.AppendElement(
 | |
|         CompactPair<RefPtr<nsAtom>, RefPtr<Selection>>(aHighlightName,
 | |
|                                                        std::move(selection)));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::RemoveHighlightSelection(nsAtom* aHighlightName) {
 | |
|   if (auto iter =
 | |
|           std::find_if(mHighlightSelections.begin(), mHighlightSelections.end(),
 | |
|                        [&aHighlightName](auto const& aElm) {
 | |
|                          return aElm.first() == aHighlightName;
 | |
|                        });
 | |
|       iter != mHighlightSelections.end()) {
 | |
|     RefPtr<Selection> selection = iter->second();
 | |
|     selection->RemoveAllRanges(IgnoreErrors());
 | |
|     mHighlightSelections.RemoveElementAt(iter);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::AddHighlightSelectionRange(
 | |
|     nsAtom* aHighlightName, mozilla::dom::Highlight& aHighlight,
 | |
|     mozilla::dom::AbstractRange& aRange) {
 | |
|   if (auto iter =
 | |
|           std::find_if(mHighlightSelections.begin(), mHighlightSelections.end(),
 | |
|                        [&aHighlightName](auto const& aElm) {
 | |
|                          return aElm.first() == aHighlightName;
 | |
|                        });
 | |
|       iter != mHighlightSelections.end()) {
 | |
|     RefPtr<Selection> selection = iter->second();
 | |
|     selection->AddHighlightRangeAndSelectFramesAndNotifyListeners(aRange);
 | |
|   } else {
 | |
|     // if the selection does not exist yet, add all of its ranges and exit.
 | |
|     RefPtr<Selection> selection =
 | |
|         aHighlight.CreateHighlightSelection(aHighlightName, this);
 | |
|     mHighlightSelections.AppendElement(
 | |
|         CompactPair<RefPtr<nsAtom>, RefPtr<Selection>>(aHighlightName,
 | |
|                                                        std::move(selection)));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::RemoveHighlightSelectionRange(
 | |
|     nsAtom* aHighlightName, mozilla::dom::AbstractRange& aRange) {
 | |
|   if (auto iter =
 | |
|           std::find_if(mHighlightSelections.begin(), mHighlightSelections.end(),
 | |
|                        [&aHighlightName](auto const& aElm) {
 | |
|                          return aElm.first() == aHighlightName;
 | |
|                        });
 | |
|       iter != mHighlightSelections.end()) {
 | |
|     // NOLINTNEXTLINE(performance-unnecessary-copy-initialization)
 | |
|     RefPtr<Selection> selection = iter->second();
 | |
|     selection->RemoveRangeAndUnselectFramesAndNotifyListeners(aRange,
 | |
|                                                               IgnoreErrors());
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::ScrollSelectionIntoView(SelectionType aSelectionType,
 | |
|                                                    SelectionRegion aRegion,
 | |
|                                                    int16_t aFlags) const {
 | |
|   int8_t index = GetIndexFromSelectionType(aSelectionType);
 | |
|   if (index < 0) return NS_ERROR_INVALID_ARG;
 | |
| 
 | |
|   if (!mDomSelections[index]) return NS_ERROR_NULL_POINTER;
 | |
| 
 | |
|   ScrollAxis verticalScroll = ScrollAxis();
 | |
|   int32_t flags = Selection::SCROLL_DO_FLUSH;
 | |
|   if (aFlags & nsISelectionController::SCROLL_SYNCHRONOUS) {
 | |
|     flags |= Selection::SCROLL_SYNCHRONOUS;
 | |
|   } else if (aFlags & nsISelectionController::SCROLL_FIRST_ANCESTOR_ONLY) {
 | |
|     flags |= Selection::SCROLL_FIRST_ANCESTOR_ONLY;
 | |
|   }
 | |
|   if (aFlags & nsISelectionController::SCROLL_OVERFLOW_HIDDEN) {
 | |
|     flags |= Selection::SCROLL_OVERFLOW_HIDDEN;
 | |
|   }
 | |
|   if (aFlags & nsISelectionController::SCROLL_CENTER_VERTICALLY) {
 | |
|     verticalScroll =
 | |
|         ScrollAxis(WhereToScroll::Center, WhenToScroll::IfNotFullyVisible);
 | |
|   }
 | |
|   if (aFlags & nsISelectionController::SCROLL_FOR_CARET_MOVE) {
 | |
|     flags |= Selection::SCROLL_FOR_CARET_MOVE;
 | |
|   }
 | |
| 
 | |
|   // After ScrollSelectionIntoView(), the pending notifications might be
 | |
|   // flushed and PresShell/PresContext/Frames may be dead. See bug 418470.
 | |
|   RefPtr<Selection> sel = mDomSelections[index];
 | |
|   return sel->ScrollIntoView(aRegion, verticalScroll, ScrollAxis(), flags);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::RepaintSelection(SelectionType aSelectionType) {
 | |
|   int8_t index = GetIndexFromSelectionType(aSelectionType);
 | |
|   if (index < 0) return NS_ERROR_INVALID_ARG;
 | |
|   if (!mDomSelections[index]) return NS_ERROR_NULL_POINTER;
 | |
|   NS_ENSURE_STATE(mPresShell);
 | |
| 
 | |
| // On macOS, update the selection cache to the new active selection
 | |
| // aka the current selection.
 | |
| #ifdef XP_MACOSX
 | |
|   // Check that we're in the an active window and, if this is Web content,
 | |
|   // in the frontmost tab.
 | |
|   Document* doc = mPresShell->GetDocument();
 | |
|   if (doc && IsInActiveTab(doc) && aSelectionType == SelectionType::eNormal) {
 | |
|     UpdateSelectionCacheOnRepaintSelection(mDomSelections[index]);
 | |
|   }
 | |
| #endif
 | |
|   return mDomSelections[index]->Repaint(mPresShell->GetPresContext());
 | |
| }
 | |
| 
 | |
| nsIFrame* nsFrameSelection::GetFrameToPageSelect() const {
 | |
|   if (NS_WARN_IF(!mPresShell)) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   nsIFrame* rootFrameToSelect;
 | |
|   if (mLimiters.mLimiter) {
 | |
|     rootFrameToSelect = mLimiters.mLimiter->GetPrimaryFrame();
 | |
|     if (NS_WARN_IF(!rootFrameToSelect)) {
 | |
|       return nullptr;
 | |
|     }
 | |
|   } else if (mLimiters.mAncestorLimiter) {
 | |
|     rootFrameToSelect = mLimiters.mAncestorLimiter->GetPrimaryFrame();
 | |
|     if (NS_WARN_IF(!rootFrameToSelect)) {
 | |
|       return nullptr;
 | |
|     }
 | |
|   } else {
 | |
|     rootFrameToSelect = mPresShell->GetRootScrollFrame();
 | |
|     if (NS_WARN_IF(!rootFrameToSelect)) {
 | |
|       return nullptr;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   nsCOMPtr<nsIContent> contentToSelect = mPresShell->GetContentForScrolling();
 | |
|   if (contentToSelect) {
 | |
|     // If there is selected content, look for nearest and vertical scrollable
 | |
|     // parent under the root frame.
 | |
|     for (nsIFrame* frame = contentToSelect->GetPrimaryFrame();
 | |
|          frame && frame != rootFrameToSelect; frame = frame->GetParent()) {
 | |
|       nsIScrollableFrame* scrollableFrame = do_QueryFrame(frame);
 | |
|       if (!scrollableFrame) {
 | |
|         continue;
 | |
|       }
 | |
|       ScrollStyles scrollStyles = scrollableFrame->GetScrollStyles();
 | |
|       if (scrollStyles.mVertical == StyleOverflow::Hidden) {
 | |
|         continue;
 | |
|       }
 | |
|       layers::ScrollDirections directions =
 | |
|           scrollableFrame->GetAvailableScrollingDirections();
 | |
|       if (directions.contains(layers::ScrollDirection::eVertical)) {
 | |
|         // If there is sub scrollable frame, let's use its page size to select.
 | |
|         return frame;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   // Otherwise, i.e., there is no scrollable frame or only the root frame is
 | |
|   // scrollable, let's return the root frame because Shift + PageUp/PageDown
 | |
|   // should expand the selection in the root content even if it's not
 | |
|   // scrollable.
 | |
|   return rootFrameToSelect;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::PageMove(bool aForward, bool aExtend,
 | |
|                                     nsIFrame* aFrame,
 | |
|                                     SelectionIntoView aSelectionIntoView) {
 | |
|   MOZ_ASSERT(aFrame);
 | |
| 
 | |
|   // expected behavior for PageMove is to scroll AND move the caret
 | |
|   // and remain relative position of the caret in view. see Bug 4302.
 | |
| 
 | |
|   // Get the scrollable frame.  If aFrame is not scrollable, this is nullptr.
 | |
|   nsIScrollableFrame* scrollableFrame = aFrame->GetScrollTargetFrame();
 | |
|   // Get the scrolled frame.  If aFrame is not scrollable, this is aFrame
 | |
|   // itself.
 | |
|   nsIFrame* scrolledFrame =
 | |
|       scrollableFrame ? scrollableFrame->GetScrolledFrame() : aFrame;
 | |
|   if (!scrolledFrame) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // find out where the caret is.
 | |
|   // we should know mDesiredCaretPos.mValue value of nsFrameSelection, but I
 | |
|   // havent seen that behavior in other windows applications yet.
 | |
|   RefPtr<Selection> selection = GetSelection(SelectionType::eNormal);
 | |
|   if (!selection) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   nsRect caretPos;
 | |
|   nsIFrame* caretFrame = nsCaret::GetGeometry(selection, &caretPos);
 | |
|   if (!caretFrame) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // If the scrolled frame is outside of current selection limiter,
 | |
|   // we need to scroll the frame but keep moving selection in the limiter.
 | |
|   nsIFrame* frameToClick = scrolledFrame;
 | |
|   if (!IsValidSelectionPoint(scrolledFrame->GetContent())) {
 | |
|     frameToClick = GetFrameToPageSelect();
 | |
|     if (NS_WARN_IF(!frameToClick)) {
 | |
|       return NS_OK;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (scrollableFrame) {
 | |
|     // If there is a scrollable frame, adjust pseudo-click position with page
 | |
|     // scroll amount.
 | |
|     // XXX This may scroll more than one page if ScrollSelectionIntoView is
 | |
|     //     called later because caret may not fully visible.  E.g., if
 | |
|     //     clicking line will be visible only half height with scrolling
 | |
|     //     the frame, ScrollSelectionIntoView additionally scrolls to show
 | |
|     //     the caret entirely.
 | |
|     if (aForward) {
 | |
|       caretPos.y += scrollableFrame->GetPageScrollAmount().height;
 | |
|     } else {
 | |
|       caretPos.y -= scrollableFrame->GetPageScrollAmount().height;
 | |
|     }
 | |
|   } else {
 | |
|     // Otherwise, adjust pseudo-click position with the frame size.
 | |
|     if (aForward) {
 | |
|       caretPos.y += frameToClick->GetSize().height;
 | |
|     } else {
 | |
|       caretPos.y -= frameToClick->GetSize().height;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   caretPos += caretFrame->GetOffsetTo(frameToClick);
 | |
| 
 | |
|   // get a content at desired location
 | |
|   nsPoint desiredPoint;
 | |
|   desiredPoint.x = caretPos.x;
 | |
|   desiredPoint.y = caretPos.y + caretPos.height / 2;
 | |
|   nsIFrame::ContentOffsets offsets =
 | |
|       frameToClick->GetContentOffsetsFromPoint(desiredPoint);
 | |
| 
 | |
|   if (!offsets.content) {
 | |
|     // XXX Do we need to handle ScrollSelectionIntoView in this case?
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // First, place the caret.
 | |
|   bool selectionChanged;
 | |
|   {
 | |
|     // We don't want any script to run until we check whether selection is
 | |
|     // modified by HandleClick.
 | |
|     SelectionBatcher ensureNoSelectionChangeNotifications(selection,
 | |
|                                                           __FUNCTION__);
 | |
| 
 | |
|     RangeBoundary oldAnchor = selection->AnchorRef();
 | |
|     RangeBoundary oldFocus = selection->FocusRef();
 | |
| 
 | |
|     const FocusMode focusMode =
 | |
|         aExtend ? FocusMode::kExtendSelection : FocusMode::kCollapseToNewPoint;
 | |
|     HandleClick(MOZ_KnownLive(offsets.content) /* bug 1636889 */,
 | |
|                 offsets.offset, offsets.offset, focusMode,
 | |
|                 CaretAssociationHint::After);
 | |
| 
 | |
|     selectionChanged = selection->AnchorRef() != oldAnchor ||
 | |
|                        selection->FocusRef() != oldFocus;
 | |
|   }
 | |
| 
 | |
|   bool doScrollSelectionIntoView = !(
 | |
|       aSelectionIntoView == SelectionIntoView::IfChanged && !selectionChanged);
 | |
| 
 | |
|   // Then, scroll the given frame one page.
 | |
|   if (scrollableFrame) {
 | |
|     // If we'll call ScrollSelectionIntoView later and selection wasn't
 | |
|     // changed and we scroll outside of selection limiter, we shouldn't use
 | |
|     // smooth scroll here because nsIScrollableFrame uses normal runnable,
 | |
|     // but ScrollSelectionIntoView uses early runner and it cancels the
 | |
|     // pending smooth scroll.  Therefore, if we used smooth scroll in such
 | |
|     // case, ScrollSelectionIntoView would scroll to show caret instead of
 | |
|     // page scroll of an element outside selection limiter.
 | |
|     ScrollMode scrollMode = doScrollSelectionIntoView && !selectionChanged &&
 | |
|                                     scrolledFrame != frameToClick
 | |
|                                 ? ScrollMode::Instant
 | |
|                                 : ScrollMode::Smooth;
 | |
|     scrollableFrame->ScrollBy(nsIntPoint(0, aForward ? 1 : -1),
 | |
|                               ScrollUnit::PAGES, scrollMode);
 | |
|   }
 | |
| 
 | |
|   // Finally, scroll selection into view if requested.
 | |
|   if (!doScrollSelectionIntoView) {
 | |
|     return NS_OK;
 | |
|   }
 | |
|   return ScrollSelectionIntoView(
 | |
|       SelectionType::eNormal, nsISelectionController::SELECTION_FOCUS_REGION,
 | |
|       nsISelectionController::SCROLL_SYNCHRONOUS |
 | |
|           nsISelectionController::SCROLL_FOR_CARET_MOVE);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::PhysicalMove(int16_t aDirection, int16_t aAmount,
 | |
|                                         bool aExtend) {
 | |
|   NS_ENSURE_STATE(mPresShell);
 | |
|   // Flush out layout, since we need it to be up to date to do caret
 | |
|   // positioning.
 | |
|   OwningNonNull<PresShell> presShell(*mPresShell);
 | |
|   presShell->FlushPendingNotifications(FlushType::Layout);
 | |
| 
 | |
|   if (!mPresShell) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // Check that parameters are safe
 | |
|   if (aDirection < 0 || aDirection > 3 || aAmount < 0 || aAmount > 1) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   nsPresContext* context = mPresShell->GetPresContext();
 | |
|   if (!context) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   RefPtr<Selection> sel = mDomSelections[index];
 | |
|   if (!sel) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   // Map the abstract movement amounts (0-1) to direction-specific
 | |
|   // selection units.
 | |
|   static const nsSelectionAmount inlineAmount[] = {eSelectCluster, eSelectWord};
 | |
|   static const nsSelectionAmount blockPrevAmount[] = {eSelectLine,
 | |
|                                                       eSelectBeginLine};
 | |
|   static const nsSelectionAmount blockNextAmount[] = {eSelectLine,
 | |
|                                                       eSelectEndLine};
 | |
| 
 | |
|   struct PhysicalToLogicalMapping {
 | |
|     nsDirection direction;
 | |
|     const nsSelectionAmount* amounts;
 | |
|   };
 | |
|   static const PhysicalToLogicalMapping verticalLR[4] = {
 | |
|       {eDirPrevious, blockPrevAmount},  // left
 | |
|       {eDirNext, blockNextAmount},      // right
 | |
|       {eDirPrevious, inlineAmount},     // up
 | |
|       {eDirNext, inlineAmount}          // down
 | |
|   };
 | |
|   static const PhysicalToLogicalMapping verticalRL[4] = {
 | |
|       {eDirNext, blockNextAmount},
 | |
|       {eDirPrevious, blockPrevAmount},
 | |
|       {eDirPrevious, inlineAmount},
 | |
|       {eDirNext, inlineAmount}};
 | |
|   static const PhysicalToLogicalMapping horizontal[4] = {
 | |
|       {eDirPrevious, inlineAmount},
 | |
|       {eDirNext, inlineAmount},
 | |
|       {eDirPrevious, blockPrevAmount},
 | |
|       {eDirNext, blockNextAmount}};
 | |
| 
 | |
|   WritingMode wm;
 | |
|   const PrimaryFrameData frameForFocus =
 | |
|       sel->GetPrimaryFrameForCaretAtFocusNode(true);
 | |
|   if (frameForFocus.mFrame) {
 | |
|     // FYI: Setting the caret association hint was done during a call of
 | |
|     // GetPrimaryFrameForCaretAtFocusNode.  Therefore, this may not be intended
 | |
|     // by the original author.
 | |
|     sel->GetFrameSelection()->SetHint(frameForFocus.mHint);
 | |
| 
 | |
|     if (!frameForFocus.mFrame->Style()->IsTextCombined()) {
 | |
|       wm = frameForFocus.mFrame->GetWritingMode();
 | |
|     } else {
 | |
|       // Using different direction for horizontal-in-vertical would
 | |
|       // make it hard to navigate via keyboard. Inherit the moving
 | |
|       // direction from its parent.
 | |
|       MOZ_ASSERT(frameForFocus.mFrame->IsTextFrame());
 | |
|       wm = frameForFocus.mFrame->GetParent()->GetWritingMode();
 | |
|       MOZ_ASSERT(wm.IsVertical(),
 | |
|                  "Text combined "
 | |
|                  "can only appear in vertical text");
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const PhysicalToLogicalMapping& mapping =
 | |
|       wm.IsVertical()
 | |
|           ? wm.IsVerticalLR() ? verticalLR[aDirection] : verticalRL[aDirection]
 | |
|           : horizontal[aDirection];
 | |
| 
 | |
|   nsresult rv =
 | |
|       MoveCaret(mapping.direction, aExtend, mapping.amounts[aAmount], eVisual);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     // If we tried to do a line move, but couldn't move in the given direction,
 | |
|     // then we'll "promote" this to a line-edge move instead.
 | |
|     if (mapping.amounts[aAmount] == eSelectLine) {
 | |
|       rv = MoveCaret(mapping.direction, aExtend, mapping.amounts[aAmount + 1],
 | |
|                      eVisual);
 | |
|     }
 | |
|     // And if it was a next-word move that failed (which can happen when
 | |
|     // eat_space_to_next_word is true, see bug 1153237), then just move forward
 | |
|     // to the line-edge.
 | |
|     else if (mapping.amounts[aAmount] == eSelectWord &&
 | |
|              mapping.direction == eDirNext) {
 | |
|       rv = MoveCaret(eDirNext, aExtend, eSelectEndLine, eVisual);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return rv;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::CharacterMove(bool aForward, bool aExtend) {
 | |
|   return MoveCaret(aForward ? eDirNext : eDirPrevious, aExtend, eSelectCluster,
 | |
|                    eUsePrefStyle);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::WordMove(bool aForward, bool aExtend) {
 | |
|   return MoveCaret(aForward ? eDirNext : eDirPrevious, aExtend, eSelectWord,
 | |
|                    eUsePrefStyle);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::LineMove(bool aForward, bool aExtend) {
 | |
|   return MoveCaret(aForward ? eDirNext : eDirPrevious, aExtend, eSelectLine,
 | |
|                    eUsePrefStyle);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::IntraLineMove(bool aForward, bool aExtend) {
 | |
|   if (aForward) {
 | |
|     return MoveCaret(eDirNext, aExtend, eSelectEndLine, eLogical);
 | |
|   } else {
 | |
|     return MoveCaret(eDirPrevious, aExtend, eSelectBeginLine, eLogical);
 | |
|   }
 | |
| }
 | |
| 
 | |
| template <typename RangeType>
 | |
| Result<RefPtr<RangeType>, nsresult>
 | |
| nsFrameSelection::CreateRangeExtendedToSomewhere(
 | |
|     nsDirection aDirection, const nsSelectionAmount aAmount,
 | |
|     CaretMovementStyle aMovementStyle) {
 | |
|   MOZ_ASSERT(aDirection == eDirNext || aDirection == eDirPrevious);
 | |
|   MOZ_ASSERT(aAmount == eSelectCharacter || aAmount == eSelectCluster ||
 | |
|              aAmount == eSelectWord || aAmount == eSelectBeginLine ||
 | |
|              aAmount == eSelectEndLine);
 | |
|   MOZ_ASSERT(aMovementStyle == eLogical || aMovementStyle == eVisual ||
 | |
|              aMovementStyle == eUsePrefStyle);
 | |
| 
 | |
|   if (!mPresShell) {
 | |
|     return Err(NS_ERROR_UNEXPECTED);
 | |
|   }
 | |
|   OwningNonNull<PresShell> presShell(*mPresShell);
 | |
|   presShell->FlushPendingNotifications(FlushType::Layout);
 | |
|   if (!mPresShell) {
 | |
|     return Err(NS_ERROR_FAILURE);
 | |
|   }
 | |
|   Selection* selection =
 | |
|       mDomSelections[GetIndexFromSelectionType(SelectionType::eNormal)];
 | |
|   if (!selection || selection->RangeCount() != 1) {
 | |
|     return Err(NS_ERROR_FAILURE);
 | |
|   }
 | |
|   RefPtr<const nsRange> firstRange = selection->GetRangeAt(0);
 | |
|   if (!firstRange || !firstRange->IsPositioned()) {
 | |
|     return Err(NS_ERROR_FAILURE);
 | |
|   }
 | |
|   Result<PeekOffsetStruct, nsresult> result = PeekOffsetForCaretMove(
 | |
|       aDirection, true, aAmount, aMovementStyle, nsPoint(0, 0));
 | |
|   if (result.isErr()) {
 | |
|     return Err(NS_ERROR_FAILURE);
 | |
|   }
 | |
|   const PeekOffsetStruct& pos = result.inspect();
 | |
|   RefPtr<RangeType> range;
 | |
|   if (NS_WARN_IF(!pos.mResultContent)) {
 | |
|     return range;
 | |
|   }
 | |
|   if (aDirection == eDirPrevious) {
 | |
|     range = RangeType::Create(
 | |
|         RawRangeBoundary(pos.mResultContent, pos.mContentOffset),
 | |
|         firstRange->EndRef(), IgnoreErrors());
 | |
|   } else {
 | |
|     range = RangeType::Create(
 | |
|         firstRange->StartRef(),
 | |
|         RawRangeBoundary(pos.mResultContent, pos.mContentOffset),
 | |
|         IgnoreErrors());
 | |
|   }
 | |
|   return range;
 | |
| }
 | |
| 
 | |
| //////////END FRAMESELECTION
 | |
| 
 | |
| LazyLogModule gBatchLog("SelectionBatch");
 | |
| 
 | |
| void nsFrameSelection::StartBatchChanges(const char* aRequesterFuncName) {
 | |
|   MOZ_LOG(gBatchLog, LogLevel::Info,
 | |
|           ("%p%snsFrameSelection::StartBatchChanges(%s)", this,
 | |
|            std::string((mBatching.mCounter + 1) * 2, ' ').c_str(),
 | |
|            aRequesterFuncName));
 | |
|   mBatching.mCounter++;
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::EndBatchChanges(const char* aRequesterFuncName,
 | |
|                                        int16_t aReasons) {
 | |
|   MOZ_LOG(gBatchLog, LogLevel::Info,
 | |
|           ("%p%snsFrameSelection::EndBatchChanges  (%s, %s)", this,
 | |
|            std::string(mBatching.mCounter * 2, ' ').c_str(), aRequesterFuncName,
 | |
|            SelectionChangeReasonsToCString(aReasons).get()));
 | |
|   MOZ_ASSERT(mBatching.mCounter > 0, "Bad mBatching.mCounter");
 | |
|   mBatching.mCounter--;
 | |
| 
 | |
|   if (mBatching.mCounter == 0 && mBatching.mChangesDuringBatching) {
 | |
|     AddChangeReasons(aReasons);
 | |
|     mCaretMoveAmount = eSelectNoAmount;
 | |
|     mBatching.mChangesDuringBatching = false;
 | |
|     // Be aware, the Selection instance may be destroyed after this call.
 | |
|     NotifySelectionListeners(SelectionType::eNormal);
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::NotifySelectionListeners(
 | |
|     SelectionType aSelectionType) {
 | |
|   int8_t index = GetIndexFromSelectionType(aSelectionType);
 | |
|   if (index >= 0 && mDomSelections[index]) {
 | |
|     RefPtr<Selection> selection = mDomSelections[index];
 | |
|     selection->NotifySelectionListeners();
 | |
|     mCaretMoveAmount = eSelectNoAmount;
 | |
|     return NS_OK;
 | |
|   }
 | |
|   return NS_ERROR_FAILURE;
 | |
| }
 | |
| 
 | |
| // Start of Table Selection methods
 | |
| 
 | |
| static bool IsCell(nsIContent* aContent) {
 | |
|   return aContent->IsAnyOfHTMLElements(nsGkAtoms::td, nsGkAtoms::th);
 | |
| }
 | |
| 
 | |
| // static
 | |
| nsITableCellLayout* nsFrameSelection::GetCellLayout(
 | |
|     const nsIContent* aCellContent) {
 | |
|   nsITableCellLayout* cellLayoutObject =
 | |
|       do_QueryFrame(aCellContent->GetPrimaryFrame());
 | |
|   return cellLayoutObject;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::ClearNormalSelection() {
 | |
|   int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   RefPtr<Selection> selection = mDomSelections[index];
 | |
|   if (!selection) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   ErrorResult err;
 | |
|   selection->RemoveAllRanges(err);
 | |
|   return err.StealNSResult();
 | |
| }
 | |
| 
 | |
| static nsIContent* GetFirstSelectedContent(const nsRange* aRange) {
 | |
|   if (!aRange) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aRange->GetStartContainer(), "Must have start parent!");
 | |
|   MOZ_ASSERT(aRange->GetStartContainer()->IsElement(), "Unexpected parent");
 | |
| 
 | |
|   return aRange->GetChildAtStartOffset();
 | |
| }
 | |
| 
 | |
| // Table selection support.
 | |
| // TODO: Separate table methods into a separate nsITableSelection interface
 | |
| nsresult nsFrameSelection::HandleTableSelection(nsINode* aParentContent,
 | |
|                                                 int32_t aContentOffset,
 | |
|                                                 TableSelectionMode aTarget,
 | |
|                                                 WidgetMouseEvent* aMouseEvent) {
 | |
|   const int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   RefPtr<Selection> selection = mDomSelections[index];
 | |
|   if (!selection) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   return mTableSelection.HandleSelection(aParentContent, aContentOffset,
 | |
|                                          aTarget, aMouseEvent, mDragState,
 | |
|                                          *selection);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::TableSelection::HandleSelection(
 | |
|     nsINode* aParentContent, int32_t aContentOffset, TableSelectionMode aTarget,
 | |
|     WidgetMouseEvent* aMouseEvent, bool aDragState,
 | |
|     Selection& aNormalSelection) {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   NS_ENSURE_TRUE(aParentContent, NS_ERROR_NULL_POINTER);
 | |
|   NS_ENSURE_TRUE(aMouseEvent, NS_ERROR_NULL_POINTER);
 | |
| 
 | |
|   if (aDragState && mDragSelectingCells &&
 | |
|       aTarget == TableSelectionMode::Table) {
 | |
|     // We were selecting cells and user drags mouse in table border or inbetween
 | |
|     // cells,
 | |
|     //  just do nothing
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   RefPtr<nsIContent> childContent =
 | |
|       aParentContent->GetChildAt_Deprecated(aContentOffset);
 | |
| 
 | |
|   // When doing table selection, always set the direction to next so
 | |
|   // we can be sure that anchorNode's offset always points to the
 | |
|   // selected cell
 | |
|   aNormalSelection.SetDirection(eDirNext);
 | |
| 
 | |
|   // Stack-class to wrap all table selection changes in
 | |
|   //  BeginBatchChanges() / EndBatchChanges()
 | |
|   SelectionBatcher selectionBatcher(&aNormalSelection, __FUNCTION__);
 | |
| 
 | |
|   if (aDragState && mDragSelectingCells) {
 | |
|     return HandleDragSelecting(aTarget, childContent, aMouseEvent,
 | |
|                                aNormalSelection);
 | |
|   }
 | |
| 
 | |
|   return HandleMouseUpOrDown(aTarget, aDragState, childContent, aParentContent,
 | |
|                              aContentOffset, aMouseEvent, aNormalSelection);
 | |
| }
 | |
| 
 | |
| class nsFrameSelection::TableSelection::RowAndColumnRelation {
 | |
|  public:
 | |
|   static Result<RowAndColumnRelation, nsresult> Create(
 | |
|       const nsIContent* aFirst, const nsIContent* aSecond) {
 | |
|     RowAndColumnRelation result;
 | |
| 
 | |
|     nsresult errorResult =
 | |
|         GetCellIndexes(aFirst, result.mFirst.mRow, result.mFirst.mColumn);
 | |
|     if (NS_FAILED(errorResult)) {
 | |
|       return Err(errorResult);
 | |
|     }
 | |
| 
 | |
|     errorResult =
 | |
|         GetCellIndexes(aSecond, result.mSecond.mRow, result.mSecond.mColumn);
 | |
|     if (NS_FAILED(errorResult)) {
 | |
|       return Err(errorResult);
 | |
|     }
 | |
| 
 | |
|     return result;
 | |
|   }
 | |
| 
 | |
|   bool IsSameColumn() const { return mFirst.mColumn == mSecond.mColumn; }
 | |
| 
 | |
|   bool IsSameRow() const { return mFirst.mRow == mSecond.mRow; }
 | |
| 
 | |
|  private:
 | |
|   RowAndColumnRelation() = default;
 | |
| 
 | |
|   struct RowAndColumn {
 | |
|     int32_t mRow = 0;
 | |
|     int32_t mColumn = 0;
 | |
|   };
 | |
| 
 | |
|   RowAndColumn mFirst;
 | |
|   RowAndColumn mSecond;
 | |
| };
 | |
| 
 | |
| nsresult nsFrameSelection::TableSelection::HandleDragSelecting(
 | |
|     TableSelectionMode aTarget, nsIContent* aChildContent,
 | |
|     const WidgetMouseEvent* aMouseEvent, Selection& aNormalSelection) {
 | |
|   // We are drag-selecting
 | |
|   if (aTarget != TableSelectionMode::Table) {
 | |
|     // If dragging in the same cell as last event, do nothing
 | |
|     if (mEndSelectedCell == aChildContent) {
 | |
|       return NS_OK;
 | |
|     }
 | |
| 
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|     printf(
 | |
|         " mStartSelectedCell = %p, "
 | |
|         "mEndSelectedCell = %p, aChildContent = %p "
 | |
|         "\n",
 | |
|         mStartSelectedCell.get(), mEndSelectedCell.get(), aChildContent);
 | |
| #endif
 | |
|     // aTarget can be any "cell mode",
 | |
|     //  so we can easily drag-select rows and columns
 | |
|     // Once we are in row or column mode,
 | |
|     //  we can drift into any cell to stay in that mode
 | |
|     //  even if aTarget = TableSelectionMode::Cell
 | |
| 
 | |
|     if (mMode == TableSelectionMode::Row ||
 | |
|         mMode == TableSelectionMode::Column) {
 | |
|       if (mEndSelectedCell) {
 | |
|         Result<RowAndColumnRelation, nsresult> rowAndColumnRelation =
 | |
|             RowAndColumnRelation::Create(mEndSelectedCell, aChildContent);
 | |
| 
 | |
|         if (rowAndColumnRelation.isErr()) {
 | |
|           return rowAndColumnRelation.unwrapErr();
 | |
|         }
 | |
| 
 | |
|         if ((mMode == TableSelectionMode::Row &&
 | |
|              rowAndColumnRelation.inspect().IsSameRow()) ||
 | |
|             (mMode == TableSelectionMode::Column &&
 | |
|              rowAndColumnRelation.inspect().IsSameColumn())) {
 | |
|           return NS_OK;
 | |
|         }
 | |
|       }
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|       printf(" Dragged into a new column or row\n");
 | |
| #endif
 | |
|       // Continue dragging row or column selection
 | |
| 
 | |
|       return SelectRowOrColumn(aChildContent, aNormalSelection);
 | |
|     }
 | |
|     if (mMode == TableSelectionMode::Cell) {
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|       printf("HandleTableSelection: Dragged into a new cell\n");
 | |
| #endif
 | |
|       // Trick for quick selection of rows and columns
 | |
|       // Hold down shift, then start selecting in one direction
 | |
|       // If next cell dragged into is in same row, select entire row,
 | |
|       //   if next cell is in same column, select entire column
 | |
|       if (mStartSelectedCell && aMouseEvent->IsShift()) {
 | |
|         Result<RowAndColumnRelation, nsresult> rowAndColumnRelation =
 | |
|             RowAndColumnRelation::Create(mStartSelectedCell, aChildContent);
 | |
|         if (rowAndColumnRelation.isErr()) {
 | |
|           return rowAndColumnRelation.unwrapErr();
 | |
|         }
 | |
| 
 | |
|         if (rowAndColumnRelation.inspect().IsSameRow() ||
 | |
|             rowAndColumnRelation.inspect().IsSameColumn()) {
 | |
|           // Force new selection block
 | |
|           mStartSelectedCell = nullptr;
 | |
|           aNormalSelection.RemoveAllRanges(IgnoreErrors());
 | |
| 
 | |
|           if (rowAndColumnRelation.inspect().IsSameRow()) {
 | |
|             mMode = TableSelectionMode::Row;
 | |
|           } else {
 | |
|             mMode = TableSelectionMode::Column;
 | |
|           }
 | |
| 
 | |
|           return SelectRowOrColumn(aChildContent, aNormalSelection);
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // Reselect block of cells to new end location
 | |
|       return SelectBlockOfCells(mStartSelectedCell, aChildContent,
 | |
|                                 aNormalSelection);
 | |
|     }
 | |
|   }
 | |
|   // Do nothing if dragging in table, but outside a cell
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::TableSelection::HandleMouseUpOrDown(
 | |
|     TableSelectionMode aTarget, bool aDragState, nsIContent* aChildContent,
 | |
|     nsINode* aParentContent, int32_t aContentOffset,
 | |
|     const WidgetMouseEvent* aMouseEvent, Selection& aNormalSelection) {
 | |
|   nsresult result = NS_OK;
 | |
|   // Not dragging  -- mouse event is down or up
 | |
|   if (aDragState) {
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|     printf("HandleTableSelection: Mouse down event\n");
 | |
| #endif
 | |
|     // Clear cell we stored in mouse-down
 | |
|     mUnselectCellOnMouseUp = nullptr;
 | |
| 
 | |
|     if (aTarget == TableSelectionMode::Cell) {
 | |
|       bool isSelected = false;
 | |
| 
 | |
|       // Check if we have other selected cells
 | |
|       nsIContent* previousCellNode =
 | |
|           GetFirstSelectedContent(GetFirstCellRange(aNormalSelection));
 | |
|       if (previousCellNode) {
 | |
|         // We have at least 1 other selected cell
 | |
| 
 | |
|         // Check if new cell is already selected
 | |
|         nsIFrame* cellFrame = aChildContent->GetPrimaryFrame();
 | |
|         if (!cellFrame) {
 | |
|           return NS_ERROR_NULL_POINTER;
 | |
|         }
 | |
|         isSelected = cellFrame->IsSelected();
 | |
|       } else {
 | |
|         // No cells selected -- remove non-cell selection
 | |
|         aNormalSelection.RemoveAllRanges(IgnoreErrors());
 | |
|       }
 | |
|       mDragSelectingCells = true;  // Signal to start drag-cell-selection
 | |
|       mMode = aTarget;
 | |
|       // Set start for new drag-selection block (not appended)
 | |
|       mStartSelectedCell = aChildContent;
 | |
|       // The initial block end is same as the start
 | |
|       mEndSelectedCell = aChildContent;
 | |
| 
 | |
|       if (isSelected) {
 | |
|         // Remember this cell to (possibly) unselect it on mouseup
 | |
|         mUnselectCellOnMouseUp = aChildContent;
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|         printf(
 | |
|             "HandleTableSelection: Saving "
 | |
|             "mUnselectCellOnMouseUp\n");
 | |
| #endif
 | |
|       } else {
 | |
|         // Select an unselected cell
 | |
|         // but first remove existing selection if not in same table
 | |
|         if (previousCellNode &&
 | |
|             !IsInSameTable(previousCellNode, aChildContent)) {
 | |
|           aNormalSelection.RemoveAllRanges(IgnoreErrors());
 | |
|           // Reset selection mode that is cleared in RemoveAllRanges
 | |
|           mMode = aTarget;
 | |
|         }
 | |
| 
 | |
|         return ::SelectCellElement(aChildContent, aNormalSelection);
 | |
|       }
 | |
| 
 | |
|       return NS_OK;
 | |
|     }
 | |
|     if (aTarget == TableSelectionMode::Table) {
 | |
|       // TODO: We currently select entire table when clicked between cells,
 | |
|       //  should we restrict to only around border?
 | |
|       //  *** How do we get location data for cell and click?
 | |
|       mDragSelectingCells = false;
 | |
|       mStartSelectedCell = nullptr;
 | |
|       mEndSelectedCell = nullptr;
 | |
| 
 | |
|       // Remove existing selection and select the table
 | |
|       aNormalSelection.RemoveAllRanges(IgnoreErrors());
 | |
|       return CreateAndAddRange(aParentContent, aContentOffset,
 | |
|                                aNormalSelection);
 | |
|     }
 | |
|     if (aTarget == TableSelectionMode::Row ||
 | |
|         aTarget == TableSelectionMode::Column) {
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|       printf("aTarget == %d\n", aTarget);
 | |
| #endif
 | |
| 
 | |
|       // Start drag-selecting mode so multiple rows/cols can be selected
 | |
|       // Note: Currently, nsIFrame::GetDataForTableSelection
 | |
|       //       will never call us for row or column selection on mouse down
 | |
|       mDragSelectingCells = true;
 | |
| 
 | |
|       // Force new selection block
 | |
|       mStartSelectedCell = nullptr;
 | |
|       aNormalSelection.RemoveAllRanges(IgnoreErrors());
 | |
|       // Always do this AFTER RemoveAllRanges
 | |
|       mMode = aTarget;
 | |
| 
 | |
|       return SelectRowOrColumn(aChildContent, aNormalSelection);
 | |
|     }
 | |
|   } else {
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|     printf(
 | |
|         "HandleTableSelection: Mouse UP event. "
 | |
|         "mDragSelectingCells=%d, "
 | |
|         "mStartSelectedCell=%p\n",
 | |
|         mDragSelectingCells, mStartSelectedCell.get());
 | |
| #endif
 | |
|     // First check if we are extending a block selection
 | |
|     const uint32_t rangeCount = aNormalSelection.RangeCount();
 | |
| 
 | |
|     if (rangeCount > 0 && aMouseEvent->IsShift() && mAppendStartSelectedCell &&
 | |
|         mAppendStartSelectedCell != aChildContent) {
 | |
|       // Shift key is down: append a block selection
 | |
|       mDragSelectingCells = false;
 | |
| 
 | |
|       return SelectBlockOfCells(mAppendStartSelectedCell, aChildContent,
 | |
|                                 aNormalSelection);
 | |
|     }
 | |
| 
 | |
|     if (mDragSelectingCells) {
 | |
|       mAppendStartSelectedCell = mStartSelectedCell;
 | |
|     }
 | |
| 
 | |
|     mDragSelectingCells = false;
 | |
|     mStartSelectedCell = nullptr;
 | |
|     mEndSelectedCell = nullptr;
 | |
| 
 | |
|     // Any other mouseup actions require that Ctrl or Cmd key is pressed
 | |
|     //  else stop table selection mode
 | |
|     bool doMouseUpAction = false;
 | |
| #ifdef XP_MACOSX
 | |
|     doMouseUpAction = aMouseEvent->IsMeta();
 | |
| #else
 | |
|     doMouseUpAction = aMouseEvent->IsControl();
 | |
| #endif
 | |
|     if (!doMouseUpAction) {
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|       printf(
 | |
|           "HandleTableSelection: Ending cell selection on mouseup: "
 | |
|           "mAppendStartSelectedCell=%p\n",
 | |
|           mAppendStartSelectedCell.get());
 | |
| #endif
 | |
|       return NS_OK;
 | |
|     }
 | |
|     // Unselect a cell only if it wasn't
 | |
|     //  just selected on mousedown
 | |
|     if (aChildContent == mUnselectCellOnMouseUp) {
 | |
|       // Scan ranges to find the cell to unselect (the selection range to
 | |
|       // remove)
 | |
|       // XXXbz it's really weird that this lives outside the loop, so once we
 | |
|       // find one we keep looking at it even if we find no more cells...
 | |
|       nsINode* previousCellParent = nullptr;
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|       printf(
 | |
|           "HandleTableSelection: Unselecting "
 | |
|           "mUnselectCellOnMouseUp; "
 | |
|           "rangeCount=%d\n",
 | |
|           rangeCount);
 | |
| #endif
 | |
|       for (const uint32_t i : IntegerRange(rangeCount)) {
 | |
|         MOZ_ASSERT(aNormalSelection.RangeCount() == rangeCount);
 | |
|         // Strong reference, because sometimes we want to remove
 | |
|         // this range, and then we might be the only owner.
 | |
|         RefPtr<nsRange> range = aNormalSelection.GetRangeAt(i);
 | |
|         if (MOZ_UNLIKELY(!range)) {
 | |
|           return NS_ERROR_NULL_POINTER;
 | |
|         }
 | |
| 
 | |
|         nsINode* container = range->GetStartContainer();
 | |
|         if (!container) {
 | |
|           return NS_ERROR_NULL_POINTER;
 | |
|         }
 | |
| 
 | |
|         int32_t offset = range->StartOffset();
 | |
|         // Be sure previous selection is a table cell
 | |
|         nsIContent* child = range->GetChildAtStartOffset();
 | |
|         if (child && IsCell(child)) {
 | |
|           previousCellParent = container;
 | |
|         }
 | |
| 
 | |
|         // We're done if we didn't find parent of a previously-selected cell
 | |
|         if (!previousCellParent) {
 | |
|           break;
 | |
|         }
 | |
| 
 | |
|         if (previousCellParent == aParentContent && offset == aContentOffset) {
 | |
|           // Cell is already selected
 | |
|           if (rangeCount == 1) {
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|             printf("HandleTableSelection: Unselecting single selected cell\n");
 | |
| #endif
 | |
|             // This was the only cell selected.
 | |
|             // Collapse to "normal" selection inside the cell
 | |
|             mStartSelectedCell = nullptr;
 | |
|             mEndSelectedCell = nullptr;
 | |
|             mAppendStartSelectedCell = nullptr;
 | |
|             // TODO: We need a "Collapse to just before deepest child" routine
 | |
|             // Even better, should we collapse to just after the LAST deepest
 | |
|             // child
 | |
|             //  (i.e., at the end of the cell's contents)?
 | |
|             return aNormalSelection.CollapseInLimiter(aChildContent, 0);
 | |
|           }
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|           printf(
 | |
|               "HandleTableSelection: Removing cell from multi-cell "
 | |
|               "selection\n");
 | |
| #endif
 | |
|           // Unselecting the start of previous block
 | |
|           // XXX What do we use now!
 | |
|           if (aChildContent == mAppendStartSelectedCell) {
 | |
|             mAppendStartSelectedCell = nullptr;
 | |
|           }
 | |
| 
 | |
|           // Deselect cell by removing its range from selection
 | |
|           ErrorResult err;
 | |
|           aNormalSelection.RemoveRangeAndUnselectFramesAndNotifyListeners(
 | |
|               *range, err);
 | |
|           return err.StealNSResult();
 | |
|         }
 | |
|       }
 | |
|       mUnselectCellOnMouseUp = nullptr;
 | |
|     }
 | |
|   }
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::TableSelection::SelectBlockOfCells(
 | |
|     nsIContent* aStartCell, nsIContent* aEndCell, Selection& aNormalSelection) {
 | |
|   NS_ENSURE_TRUE(aStartCell, NS_ERROR_NULL_POINTER);
 | |
|   NS_ENSURE_TRUE(aEndCell, NS_ERROR_NULL_POINTER);
 | |
|   mEndSelectedCell = aEndCell;
 | |
| 
 | |
|   nsresult result = NS_OK;
 | |
| 
 | |
|   // If new end cell is in a different table, do nothing
 | |
|   const RefPtr<const nsIContent> table = IsInSameTable(aStartCell, aEndCell);
 | |
|   if (!table) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // Get starting and ending cells' location in the cellmap
 | |
|   int32_t startRowIndex, startColIndex, endRowIndex, endColIndex;
 | |
|   result = GetCellIndexes(aStartCell, startRowIndex, startColIndex);
 | |
|   if (NS_FAILED(result)) return result;
 | |
|   result = GetCellIndexes(aEndCell, endRowIndex, endColIndex);
 | |
|   if (NS_FAILED(result)) return result;
 | |
| 
 | |
|   if (mDragSelectingCells) {
 | |
|     // Drag selecting: remove selected cells outside of new block limits
 | |
|     // TODO: `UnselectCells`'s return value shouldn't be ignored.
 | |
|     UnselectCells(table, startRowIndex, startColIndex, endRowIndex, endColIndex,
 | |
|                   true, aNormalSelection);
 | |
|   }
 | |
| 
 | |
|   // Note that we select block in the direction of user's mouse dragging,
 | |
|   //  which means start cell may be after the end cell in either row or column
 | |
|   return AddCellsToSelection(table, startRowIndex, startColIndex, endRowIndex,
 | |
|                              endColIndex, aNormalSelection);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::TableSelection::UnselectCells(
 | |
|     const nsIContent* aTableContent, int32_t aStartRowIndex,
 | |
|     int32_t aStartColumnIndex, int32_t aEndRowIndex, int32_t aEndColumnIndex,
 | |
|     bool aRemoveOutsideOfCellRange, mozilla::dom::Selection& aNormalSelection) {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   nsTableWrapperFrame* tableFrame =
 | |
|       do_QueryFrame(aTableContent->GetPrimaryFrame());
 | |
|   if (!tableFrame) return NS_ERROR_FAILURE;
 | |
| 
 | |
|   int32_t minRowIndex = std::min(aStartRowIndex, aEndRowIndex);
 | |
|   int32_t maxRowIndex = std::max(aStartRowIndex, aEndRowIndex);
 | |
|   int32_t minColIndex = std::min(aStartColumnIndex, aEndColumnIndex);
 | |
|   int32_t maxColIndex = std::max(aStartColumnIndex, aEndColumnIndex);
 | |
| 
 | |
|   // Strong reference because we sometimes remove the range
 | |
|   RefPtr<nsRange> range = GetFirstCellRange(aNormalSelection);
 | |
|   nsIContent* cellNode = GetFirstSelectedContent(range);
 | |
|   MOZ_ASSERT(!range || cellNode, "Must have cellNode if had a range");
 | |
| 
 | |
|   int32_t curRowIndex, curColIndex;
 | |
|   while (cellNode) {
 | |
|     nsresult result = GetCellIndexes(cellNode, curRowIndex, curColIndex);
 | |
|     if (NS_FAILED(result)) return result;
 | |
| 
 | |
| #ifdef DEBUG_TABLE_SELECTION
 | |
|     if (!range) printf("RemoveCellsToSelection -- range is null\n");
 | |
| #endif
 | |
| 
 | |
|     if (range) {
 | |
|       if (aRemoveOutsideOfCellRange) {
 | |
|         if (curRowIndex < minRowIndex || curRowIndex > maxRowIndex ||
 | |
|             curColIndex < minColIndex || curColIndex > maxColIndex) {
 | |
|           aNormalSelection.RemoveRangeAndUnselectFramesAndNotifyListeners(
 | |
|               *range, IgnoreErrors());
 | |
|           // Since we've removed the range, decrement pointer to next range
 | |
|           mSelectedCellIndex--;
 | |
|         }
 | |
| 
 | |
|       } else {
 | |
|         // Remove cell from selection if it belongs to the given cells range or
 | |
|         // it is spanned onto the cells range.
 | |
|         nsTableCellFrame* cellFrame =
 | |
|             tableFrame->GetCellFrameAt(curRowIndex, curColIndex);
 | |
| 
 | |
|         uint32_t origRowIndex = cellFrame->RowIndex();
 | |
|         uint32_t origColIndex = cellFrame->ColIndex();
 | |
|         uint32_t actualRowSpan =
 | |
|             tableFrame->GetEffectiveRowSpanAt(origRowIndex, origColIndex);
 | |
|         uint32_t actualColSpan =
 | |
|             tableFrame->GetEffectiveColSpanAt(curRowIndex, curColIndex);
 | |
|         if (origRowIndex <= static_cast<uint32_t>(maxRowIndex) &&
 | |
|             maxRowIndex >= 0 &&
 | |
|             origRowIndex + actualRowSpan - 1 >=
 | |
|                 static_cast<uint32_t>(minRowIndex) &&
 | |
|             origColIndex <= static_cast<uint32_t>(maxColIndex) &&
 | |
|             maxColIndex >= 0 &&
 | |
|             origColIndex + actualColSpan - 1 >=
 | |
|                 static_cast<uint32_t>(minColIndex)) {
 | |
|           aNormalSelection.RemoveRangeAndUnselectFramesAndNotifyListeners(
 | |
|               *range, IgnoreErrors());
 | |
|           // Since we've removed the range, decrement pointer to next range
 | |
|           mSelectedCellIndex--;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     range = GetNextCellRange(aNormalSelection);
 | |
|     cellNode = GetFirstSelectedContent(range);
 | |
|     MOZ_ASSERT(!range || cellNode, "Must have cellNode if had a range");
 | |
|   }
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult SelectCellElement(nsIContent* aCellElement,
 | |
|                            Selection& aNormalSelection) {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   nsIContent* parent = aCellElement->GetParent();
 | |
| 
 | |
|   // Get child offset
 | |
|   const int32_t offset = parent->ComputeIndexOf_Deprecated(aCellElement);
 | |
| 
 | |
|   return CreateAndAddRange(parent, offset, aNormalSelection);
 | |
| }
 | |
| 
 | |
| static nsresult AddCellsToSelection(const nsIContent* aTableContent,
 | |
|                                     int32_t aStartRowIndex,
 | |
|                                     int32_t aStartColumnIndex,
 | |
|                                     int32_t aEndRowIndex,
 | |
|                                     int32_t aEndColumnIndex,
 | |
|                                     Selection& aNormalSelection) {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   nsTableWrapperFrame* tableFrame =
 | |
|       do_QueryFrame(aTableContent->GetPrimaryFrame());
 | |
|   if (!tableFrame) {  // Check that |table| is a table.
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   nsresult result = NS_OK;
 | |
|   uint32_t row = aStartRowIndex;
 | |
|   while (true) {
 | |
|     uint32_t col = aStartColumnIndex;
 | |
|     while (true) {
 | |
|       nsTableCellFrame* cellFrame = tableFrame->GetCellFrameAt(row, col);
 | |
| 
 | |
|       // Skip cells that are spanned from previous locations or are already
 | |
|       // selected
 | |
|       if (cellFrame) {
 | |
|         uint32_t origRow = cellFrame->RowIndex();
 | |
|         uint32_t origCol = cellFrame->ColIndex();
 | |
|         if (origRow == row && origCol == col && !cellFrame->IsSelected()) {
 | |
|           result = SelectCellElement(cellFrame->GetContent(), aNormalSelection);
 | |
|           if (NS_FAILED(result)) {
 | |
|             return result;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       // Done when we reach end column
 | |
|       if (col == static_cast<uint32_t>(aEndColumnIndex)) {
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       if (aStartColumnIndex < aEndColumnIndex) {
 | |
|         col++;
 | |
|       } else {
 | |
|         col--;
 | |
|       }
 | |
|     }
 | |
|     if (row == static_cast<uint32_t>(aEndRowIndex)) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if (aStartRowIndex < aEndRowIndex) {
 | |
|       row++;
 | |
|     } else {
 | |
|       row--;
 | |
|     }
 | |
|   }
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::RemoveCellsFromSelection(nsIContent* aTable,
 | |
|                                                     int32_t aStartRowIndex,
 | |
|                                                     int32_t aStartColumnIndex,
 | |
|                                                     int32_t aEndRowIndex,
 | |
|                                                     int32_t aEndColumnIndex) {
 | |
|   const int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   const RefPtr<mozilla::dom::Selection> selection = mDomSelections[index];
 | |
|   if (!selection) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   return mTableSelection.UnselectCells(aTable, aStartRowIndex,
 | |
|                                        aStartColumnIndex, aEndRowIndex,
 | |
|                                        aEndColumnIndex, false, *selection);
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::RestrictCellsToSelection(nsIContent* aTable,
 | |
|                                                     int32_t aStartRowIndex,
 | |
|                                                     int32_t aStartColumnIndex,
 | |
|                                                     int32_t aEndRowIndex,
 | |
|                                                     int32_t aEndColumnIndex) {
 | |
|   const int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   const RefPtr<mozilla::dom::Selection> selection = mDomSelections[index];
 | |
|   if (!selection) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   return mTableSelection.UnselectCells(aTable, aStartRowIndex,
 | |
|                                        aStartColumnIndex, aEndRowIndex,
 | |
|                                        aEndColumnIndex, true, *selection);
 | |
| }
 | |
| 
 | |
| Result<nsFrameSelection::TableSelection::FirstAndLastCell, nsresult>
 | |
| nsFrameSelection::TableSelection::FindFirstAndLastCellOfRowOrColumn(
 | |
|     const nsIContent& aCellContent) const {
 | |
|   const nsIContent* table = GetParentTable(&aCellContent);
 | |
|   if (!table) {
 | |
|     return Err(NS_ERROR_NULL_POINTER);
 | |
|   }
 | |
| 
 | |
|   // Get table and cell layout interfaces to access
 | |
|   // cell data based on cellmap location
 | |
|   // Frames are not ref counted, so don't use an nsCOMPtr
 | |
|   nsTableWrapperFrame* tableFrame = do_QueryFrame(table->GetPrimaryFrame());
 | |
|   if (!tableFrame) {
 | |
|     return Err(NS_ERROR_FAILURE);
 | |
|   }
 | |
|   nsITableCellLayout* cellLayout = GetCellLayout(&aCellContent);
 | |
|   if (!cellLayout) {
 | |
|     return Err(NS_ERROR_FAILURE);
 | |
|   }
 | |
| 
 | |
|   // Get location of target cell:
 | |
|   int32_t rowIndex, colIndex;
 | |
|   nsresult result = cellLayout->GetCellIndexes(rowIndex, colIndex);
 | |
|   if (NS_FAILED(result)) {
 | |
|     return Err(result);
 | |
|   }
 | |
| 
 | |
|   // Be sure we start at proper beginning
 | |
|   // (This allows us to select row or col given ANY cell!)
 | |
|   if (mMode == TableSelectionMode::Row) {
 | |
|     colIndex = 0;
 | |
|   }
 | |
|   if (mMode == TableSelectionMode::Column) {
 | |
|     rowIndex = 0;
 | |
|   }
 | |
| 
 | |
|   FirstAndLastCell firstAndLastCell;
 | |
|   while (true) {
 | |
|     // Loop through all cells in column or row to find first and last
 | |
|     nsCOMPtr<nsIContent> curCellContent =
 | |
|         tableFrame->GetCellAt(rowIndex, colIndex);
 | |
|     if (!curCellContent) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if (!firstAndLastCell.mFirst) {
 | |
|       firstAndLastCell.mFirst = curCellContent;
 | |
|     }
 | |
| 
 | |
|     firstAndLastCell.mLast = std::move(curCellContent);
 | |
| 
 | |
|     // Move to next cell in cellmap, skipping spanned locations
 | |
|     if (mMode == TableSelectionMode::Row) {
 | |
|       colIndex += tableFrame->GetEffectiveRowSpanAt(rowIndex, colIndex);
 | |
|     } else {
 | |
|       rowIndex += tableFrame->GetEffectiveRowSpanAt(rowIndex, colIndex);
 | |
|     }
 | |
|   }
 | |
|   return firstAndLastCell;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::TableSelection::SelectRowOrColumn(
 | |
|     nsIContent* aCellContent, Selection& aNormalSelection) {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   if (!aCellContent) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   Result<FirstAndLastCell, nsresult> firstAndLastCell =
 | |
|       FindFirstAndLastCellOfRowOrColumn(*aCellContent);
 | |
|   if (firstAndLastCell.isErr()) {
 | |
|     return firstAndLastCell.unwrapErr();
 | |
|   }
 | |
| 
 | |
|   // Use SelectBlockOfCells:
 | |
|   // This will replace existing selection,
 | |
|   //  but allow unselecting by dragging out of selected region
 | |
|   if (firstAndLastCell.inspect().mFirst && firstAndLastCell.inspect().mLast) {
 | |
|     nsresult rv{NS_OK};
 | |
| 
 | |
|     if (!mStartSelectedCell) {
 | |
|       // We are starting a new block, so select the first cell
 | |
|       rv = ::SelectCellElement(firstAndLastCell.inspect().mFirst,
 | |
|                                aNormalSelection);
 | |
|       if (NS_FAILED(rv)) {
 | |
|         return rv;
 | |
|       }
 | |
|       mStartSelectedCell = firstAndLastCell.inspect().mFirst;
 | |
|     }
 | |
| 
 | |
|     rv = SelectBlockOfCells(mStartSelectedCell,
 | |
|                             firstAndLastCell.inspect().mLast, aNormalSelection);
 | |
| 
 | |
|     // This gets set to the cell at end of row/col,
 | |
|     //   but we need it to be the cell under cursor
 | |
|     mEndSelectedCell = aCellContent;
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
| #if 0
 | |
| // This is a more efficient strategy that appends row to current selection,
 | |
| //  but doesn't allow dragging OFF of an existing selection to unselect!
 | |
|   do {
 | |
|     // Loop through all cells in column or row
 | |
|     result = tableLayout->GetCellDataAt(rowIndex, colIndex,
 | |
|                                         getter_AddRefs(cellElement),
 | |
|                                         curRowIndex, curColIndex,
 | |
|                                         rowSpan, colSpan,
 | |
|                                         actualRowSpan, actualColSpan,
 | |
|                                         isSelected);
 | |
|     if (NS_FAILED(result)) return result;
 | |
|     // We're done when cell is not found
 | |
|     if (!cellElement) break;
 | |
| 
 | |
| 
 | |
|     // Check spans else we infinitely loop
 | |
|     NS_ASSERTION(actualColSpan, "actualColSpan is 0!");
 | |
|     NS_ASSERTION(actualRowSpan, "actualRowSpan is 0!");
 | |
| 
 | |
|     // Skip cells that are already selected or span from outside our region
 | |
|     if (!isSelected && rowIndex == curRowIndex && colIndex == curColIndex)
 | |
|     {
 | |
|       result = SelectCellElement(cellElement);
 | |
|       if (NS_FAILED(result)) return result;
 | |
|     }
 | |
|     // Move to next row or column in cellmap, skipping spanned locations
 | |
|     if (mMode == TableSelectionMode::Row)
 | |
|       colIndex += actualColSpan;
 | |
|     else
 | |
|       rowIndex += actualRowSpan;
 | |
|   }
 | |
|   while (cellElement);
 | |
| #endif
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| // static
 | |
| nsIContent* nsFrameSelection::GetFirstCellNodeInRange(const nsRange* aRange) {
 | |
|   if (!aRange) return nullptr;
 | |
| 
 | |
|   nsIContent* childContent = aRange->GetChildAtStartOffset();
 | |
|   if (!childContent) return nullptr;
 | |
|   // Don't return node if not a cell
 | |
|   if (!IsCell(childContent)) return nullptr;
 | |
| 
 | |
|   return childContent;
 | |
| }
 | |
| 
 | |
| nsRange* nsFrameSelection::TableSelection::GetFirstCellRange(
 | |
|     const mozilla::dom::Selection& aNormalSelection) {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   nsRange* firstRange = aNormalSelection.GetRangeAt(0);
 | |
|   if (!GetFirstCellNodeInRange(firstRange)) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   // Setup for next cell
 | |
|   mSelectedCellIndex = 1;
 | |
| 
 | |
|   return firstRange;
 | |
| }
 | |
| 
 | |
| nsRange* nsFrameSelection::TableSelection::GetNextCellRange(
 | |
|     const mozilla::dom::Selection& aNormalSelection) {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   nsRange* range =
 | |
|       aNormalSelection.GetRangeAt(AssertedCast<uint32_t>(mSelectedCellIndex));
 | |
| 
 | |
|   // Get first node in next range of selection - test if it's a cell
 | |
|   if (!GetFirstCellNodeInRange(range)) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   // Setup for next cell
 | |
|   mSelectedCellIndex++;
 | |
| 
 | |
|   return range;
 | |
| }
 | |
| 
 | |
| // static
 | |
| nsresult nsFrameSelection::GetCellIndexes(const nsIContent* aCell,
 | |
|                                           int32_t& aRowIndex,
 | |
|                                           int32_t& aColIndex) {
 | |
|   if (!aCell) return NS_ERROR_NULL_POINTER;
 | |
| 
 | |
|   aColIndex = 0;  // initialize out params
 | |
|   aRowIndex = 0;
 | |
| 
 | |
|   nsITableCellLayout* cellLayoutObject = GetCellLayout(aCell);
 | |
|   if (!cellLayoutObject) return NS_ERROR_FAILURE;
 | |
|   return cellLayoutObject->GetCellIndexes(aRowIndex, aColIndex);
 | |
| }
 | |
| 
 | |
| // static
 | |
| nsIContent* nsFrameSelection::IsInSameTable(const nsIContent* aContent1,
 | |
|                                             const nsIContent* aContent2) {
 | |
|   if (!aContent1 || !aContent2) return nullptr;
 | |
| 
 | |
|   nsIContent* tableNode1 = GetParentTable(aContent1);
 | |
|   nsIContent* tableNode2 = GetParentTable(aContent2);
 | |
| 
 | |
|   // Must be in the same table.  Note that we want to return false for
 | |
|   // the test if both tables are null.
 | |
|   return (tableNode1 == tableNode2) ? tableNode1 : nullptr;
 | |
| }
 | |
| 
 | |
| // static
 | |
| nsIContent* nsFrameSelection::GetParentTable(const nsIContent* aCell) {
 | |
|   if (!aCell) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   for (nsIContent* parent = aCell->GetParent(); parent;
 | |
|        parent = parent->GetParent()) {
 | |
|     if (parent->IsHTMLElement(nsGkAtoms::table)) {
 | |
|       return parent;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| nsresult nsFrameSelection::SelectCellElement(nsIContent* aCellElement) {
 | |
|   const int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|   const RefPtr<Selection> selection = mDomSelections[index];
 | |
|   if (!selection) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   return ::SelectCellElement(aCellElement, *selection);
 | |
| }
 | |
| 
 | |
| nsresult CreateAndAddRange(nsINode* aContainer, int32_t aOffset,
 | |
|                            Selection& aNormalSelection) {
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   if (!aContainer) {
 | |
|     return NS_ERROR_NULL_POINTER;
 | |
|   }
 | |
| 
 | |
|   // Set range around child at given offset
 | |
|   ErrorResult error;
 | |
|   RefPtr<nsRange> range =
 | |
|       nsRange::Create(aContainer, aOffset, aContainer, aOffset + 1, error);
 | |
|   if (NS_WARN_IF(error.Failed())) {
 | |
|     return error.StealNSResult();
 | |
|   }
 | |
|   MOZ_ASSERT(range);
 | |
| 
 | |
|   ErrorResult err;
 | |
|   aNormalSelection.AddRangeAndSelectFramesAndNotifyListeners(*range, err);
 | |
|   return err.StealNSResult();
 | |
| }
 | |
| 
 | |
| // End of Table Selection
 | |
| 
 | |
| void nsFrameSelection::SetAncestorLimiter(nsIContent* aLimiter) {
 | |
|   if (mLimiters.mAncestorLimiter != aLimiter) {
 | |
|     mLimiters.mAncestorLimiter = aLimiter;
 | |
|     int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|     if (!mDomSelections[index]) return;
 | |
| 
 | |
|     if (!IsValidSelectionPoint(mDomSelections[index]->GetFocusNode())) {
 | |
|       ClearNormalSelection();
 | |
|       if (mLimiters.mAncestorLimiter) {
 | |
|         SetChangeReasons(nsISelectionListener::NO_REASON);
 | |
|         nsCOMPtr<nsIContent> limiter(mLimiters.mAncestorLimiter);
 | |
|         const nsresult rv =
 | |
|             TakeFocus(*limiter, 0, 0, CaretAssociationHint::Before,
 | |
|                       FocusMode::kCollapseToNewPoint);
 | |
|         Unused << NS_WARN_IF(NS_FAILED(rv));
 | |
|         // TODO: in case of failure, propagate it to the callers.
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::SetDelayedCaretData(WidgetMouseEvent* aMouseEvent) {
 | |
|   if (aMouseEvent) {
 | |
|     mDelayedMouseEvent.mIsValid = true;
 | |
|     mDelayedMouseEvent.mIsShift = aMouseEvent->IsShift();
 | |
|     mDelayedMouseEvent.mClickCount = aMouseEvent->mClickCount;
 | |
|   } else {
 | |
|     mDelayedMouseEvent.mIsValid = false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void nsFrameSelection::DisconnectFromPresShell() {
 | |
|   if (mAccessibleCaretEnabled) {
 | |
|     int8_t index = GetIndexFromSelectionType(SelectionType::eNormal);
 | |
|     mDomSelections[index]->StopNotifyingAccessibleCaretEventHub();
 | |
|   }
 | |
| 
 | |
|   StopAutoScrollTimer();
 | |
|   for (size_t i = 0; i < ArrayLength(mDomSelections); i++) {
 | |
|     mDomSelections[i]->Clear(nullptr);
 | |
|   }
 | |
|   mPresShell = nullptr;
 | |
| }
 | |
| 
 | |
| #ifdef XP_MACOSX
 | |
| /**
 | |
|  * See Bug 1288453.
 | |
|  *
 | |
|  * Update the selection cache on repaint to handle when a pre-existing
 | |
|  * selection becomes active aka the current selection.
 | |
|  *
 | |
|  * 1. Change the current selection by click n dragging another selection.
 | |
|  *   - Make a selection on content page. Make a selection in a text editor.
 | |
|  *   - You can click n drag the content selection to make it active again.
 | |
|  * 2. Change the current selection when switching to a tab with a selection.
 | |
|  *   - Make selection in tab.
 | |
|  *   - Switching tabs will make its respective selection active.
 | |
|  *
 | |
|  * Therefore, we only update the selection cache on a repaint
 | |
|  * if the current selection being repainted is not an empty selection.
 | |
|  *
 | |
|  * If the current selection is empty. The current selection cache
 | |
|  * would be cleared by AutoCopyListener::OnSelectionChange().
 | |
|  */
 | |
| static nsresult UpdateSelectionCacheOnRepaintSelection(Selection* aSel) {
 | |
|   PresShell* presShell = aSel->GetPresShell();
 | |
|   if (!presShell) {
 | |
|     return NS_OK;
 | |
|   }
 | |
|   nsCOMPtr<Document> aDoc = presShell->GetDocument();
 | |
| 
 | |
|   if (aDoc && aSel && !aSel->IsCollapsed()) {
 | |
|     return nsCopySupport::EncodeDocumentWithContextAndPutToClipboard(
 | |
|         aSel, aDoc, nsIClipboard::kSelectionCache, false);
 | |
|   }
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| #endif  // XP_MACOSX
 | |
| 
 | |
| // mozilla::AutoCopyListener
 | |
| 
 | |
| int16_t AutoCopyListener::sClipboardID = -1;
 | |
| 
 | |
| /*
 | |
|  * What we do now:
 | |
|  * On every selection change, we copy to the clipboard anew, creating a
 | |
|  * HTML buffer, a transferable, an nsISupportsString and
 | |
|  * a huge mess every time.  This is basically what
 | |
|  * nsCopySupport::EncodeDocumentWithContextAndPutToClipboard() does to move the
 | |
|  * selection into the clipboard for Edit->Copy.
 | |
|  *
 | |
|  * What we should do, to make our end of the deal faster:
 | |
|  * Create a singleton transferable with our own magic converter.  When selection
 | |
|  * changes (use a quick cache to detect ``real'' changes), we put the new
 | |
|  * Selection in the transferable.  Our magic converter will take care of
 | |
|  * transferable->whatever-other-format when the time comes to actually
 | |
|  * hand over the clipboard contents.
 | |
|  *
 | |
|  * Other issues:
 | |
|  * - which X clipboard should we populate?
 | |
|  * - should we use a different one than Edit->Copy, so that inadvertant
 | |
|  *   selections (or simple clicks, which currently cause a selection
 | |
|  *   notification, regardless of if they're in the document which currently has
 | |
|  *   selection!) don't lose the contents of the ``application''?  Or should we
 | |
|  *   just put some intelligence in the ``is this a real selection?'' code to
 | |
|  *   protect our selection against clicks in other documents that don't create
 | |
|  *   selections?
 | |
|  * - maybe we should just never clear the X clipboard?  That would make this
 | |
|  *   problem just go away, which is very tempting.
 | |
|  *
 | |
|  * On macOS,
 | |
|  * nsIClipboard::kSelectionCache is the flag for current selection cache.
 | |
|  * Set the current selection cache on the parent process in
 | |
|  * widget cocoa nsClipboard whenever selection changes.
 | |
|  */
 | |
| 
 | |
| // static
 | |
| void AutoCopyListener::OnSelectionChange(Document* aDocument,
 | |
|                                          Selection& aSelection,
 | |
|                                          int16_t aReason) {
 | |
|   MOZ_ASSERT(IsValidClipboardID(sClipboardID));
 | |
| 
 | |
|   // For now, we should prevent any updates caused by a call of Selection API.
 | |
|   // We should allow this in some cases later, though. See the valid usage in
 | |
|   // bug 1567160.
 | |
|   if (aReason & nsISelectionListener::JS_REASON) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (sClipboardID == nsIClipboard::kSelectionCache) {
 | |
|     // Do nothing if this isn't in the active window and,
 | |
|     // in the case of Web content, in the frontmost tab.
 | |
|     if (!aDocument || !IsInActiveTab(aDocument)) {
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static const int16_t kResasonsToHandle =
 | |
|       nsISelectionListener::MOUSEUP_REASON |
 | |
|       nsISelectionListener::SELECTALL_REASON |
 | |
|       nsISelectionListener::KEYPRESS_REASON;
 | |
|   if (!(aReason & kResasonsToHandle)) {
 | |
|     return;  // Don't care if we are still dragging.
 | |
|   }
 | |
| 
 | |
|   if (!aDocument || aSelection.IsCollapsed()) {
 | |
| #ifdef DEBUG_CLIPBOARD
 | |
|     fprintf(stderr, "CLIPBOARD: no selection/collapsed selection\n");
 | |
| #endif
 | |
|     if (sClipboardID != nsIClipboard::kSelectionCache) {
 | |
|       // XXX Should we clear X clipboard?
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // If on macOS, clear the current selection transferable cached
 | |
|     // on the parent process (nsClipboard) when the selection is empty.
 | |
|     DebugOnly<nsresult> rv = nsCopySupport::ClearSelectionCache();
 | |
|     NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
 | |
|                          "nsCopySupport::ClearSelectionCache() failed");
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   DebugOnly<nsresult> rv =
 | |
|       nsCopySupport::EncodeDocumentWithContextAndPutToClipboard(
 | |
|           &aSelection, aDocument, sClipboardID, false);
 | |
|   NS_WARNING_ASSERTION(
 | |
|       NS_SUCCEEDED(rv),
 | |
|       "nsCopySupport::EncodeDocumentWithContextAndPutToClipboard() failed");
 | |
| }
 |