forked from mirrors/gecko-dev
		
	 a1ee18724c
			
		
	
	
		a1ee18724c
		
	
	
	
	
		
			
			This is a minimal version. The corresponding code cleanup will be done in a follow up. Differential Revision: https://phabricator.services.mozilla.com/D198045
		
			
				
	
	
		
			3482 lines
		
	
	
	
		
			131 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			3482 lines
		
	
	
	
		
			131 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/. */
 | |
| 
 | |
| #include "ContentEventHandler.h"
 | |
| 
 | |
| #include "mozilla/Assertions.h"
 | |
| #include "mozilla/CheckedInt.h"
 | |
| #include "mozilla/ContentIterator.h"
 | |
| #include "mozilla/IMEStateManager.h"
 | |
| #include "mozilla/IntegerRange.h"
 | |
| #include "mozilla/Maybe.h"
 | |
| #include "mozilla/PresShell.h"
 | |
| #include "mozilla/RangeBoundary.h"
 | |
| #include "mozilla/RangeUtils.h"
 | |
| #include "mozilla/SelectionMovementUtils.h"
 | |
| #include "mozilla/TextComposition.h"
 | |
| #include "mozilla/TextEditor.h"
 | |
| #include "mozilla/TextEvents.h"
 | |
| #include "mozilla/dom/Element.h"
 | |
| #include "mozilla/dom/HTMLBRElement.h"
 | |
| #include "mozilla/dom/HTMLUnknownElement.h"
 | |
| #include "mozilla/dom/Selection.h"
 | |
| #include "mozilla/dom/Text.h"
 | |
| #include "nsCaret.h"
 | |
| #include "nsCOMPtr.h"
 | |
| #include "nsContentUtils.h"
 | |
| #include "nsCopySupport.h"
 | |
| #include "nsElementTable.h"
 | |
| #include "nsFocusManager.h"
 | |
| #include "nsFontMetrics.h"
 | |
| #include "nsFrameSelection.h"
 | |
| #include "nsHTMLTags.h"
 | |
| #include "nsIFrame.h"
 | |
| #include "nsLayoutUtils.h"
 | |
| #include "nsPresContext.h"
 | |
| #include "nsQueryObject.h"
 | |
| #include "nsRange.h"
 | |
| #include "nsTextFragment.h"
 | |
| #include "nsTextFrame.h"
 | |
| #include "nsView.h"
 | |
| #include "mozilla/ViewportUtils.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| 
 | |
| // Work around conflicting define in rpcndr.h
 | |
| #if defined(small)
 | |
| #  undef small
 | |
| #endif  // defined(small)
 | |
| 
 | |
| #if defined(XP_WIN) && 0
 | |
| #  define TRANSLATE_NEW_LINES
 | |
| #endif
 | |
| 
 | |
| namespace mozilla {
 | |
| 
 | |
| using namespace dom;
 | |
| using namespace widget;
 | |
| 
 | |
| /******************************************************************/
 | |
| /* ContentEventHandler::SimpleRangeBase                           */
 | |
| /******************************************************************/
 | |
| template <>
 | |
| ContentEventHandler::SimpleRangeBase<
 | |
|     RefPtr<nsINode>, RangeBoundary>::SimpleRangeBase() = default;
 | |
| 
 | |
| template <>
 | |
| ContentEventHandler::SimpleRangeBase<nsINode*,
 | |
|                                      RawRangeBoundary>::SimpleRangeBase()
 | |
|     : mRoot(nullptr) {
 | |
| #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
 | |
|   mAssertNoGC.emplace();
 | |
| #endif  // #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
 | |
| }
 | |
| 
 | |
| template <>
 | |
| template <typename OtherNodeType, typename OtherRangeBoundaryType>
 | |
| ContentEventHandler::SimpleRangeBase<RefPtr<nsINode>, RangeBoundary>::
 | |
|     SimpleRangeBase(
 | |
|         const SimpleRangeBase<OtherNodeType, OtherRangeBoundaryType>& aOther)
 | |
|     : mRoot(aOther.GetRoot()),
 | |
|       mStart{aOther.Start().AsRaw()},
 | |
|       mEnd{aOther.End().AsRaw()}
 | |
| // Don't use the copy constructor of mAssertNoGC
 | |
| {}
 | |
| 
 | |
| template <>
 | |
| template <typename OtherNodeType, typename OtherRangeBoundaryType>
 | |
| ContentEventHandler::SimpleRangeBase<nsINode*, RawRangeBoundary>::
 | |
|     SimpleRangeBase(
 | |
|         const SimpleRangeBase<OtherNodeType, OtherRangeBoundaryType>& aOther)
 | |
|     : mRoot(aOther.GetRoot()),
 | |
|       mStart{aOther.Start().AsRaw()},
 | |
|       mEnd{aOther.End().AsRaw()} {
 | |
| #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
 | |
|   mAssertNoGC.emplace();
 | |
| #endif  // #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
 | |
| }
 | |
| 
 | |
| template <>
 | |
| ContentEventHandler::SimpleRangeBase<RefPtr<nsINode>, RangeBoundary>::
 | |
|     SimpleRangeBase(
 | |
|         SimpleRangeBase<RefPtr<nsINode>, RangeBoundary>&& aOther) noexcept
 | |
|     : mRoot(std::move(aOther.GetRoot())),
 | |
|       mStart(std::move(aOther.mStart)),
 | |
|       mEnd(std::move(aOther.mEnd)) {}
 | |
| 
 | |
| template <>
 | |
| ContentEventHandler::SimpleRangeBase<nsINode*, RawRangeBoundary>::
 | |
|     SimpleRangeBase(
 | |
|         SimpleRangeBase<nsINode*, RawRangeBoundary>&& aOther) noexcept
 | |
|     : mRoot(std::move(aOther.GetRoot())),
 | |
|       mStart(std::move(aOther.mStart)),
 | |
|       mEnd(std::move(aOther.mEnd)) {
 | |
| #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
 | |
|   mAssertNoGC.emplace();
 | |
| #endif  // #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
 | |
| }
 | |
| 
 | |
| #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
 | |
| template <>
 | |
| ContentEventHandler::SimpleRangeBase<
 | |
|     RefPtr<nsINode>, RangeBoundary>::~SimpleRangeBase() = default;
 | |
| 
 | |
| template <>
 | |
| ContentEventHandler::SimpleRangeBase<nsINode*,
 | |
|                                      RawRangeBoundary>::~SimpleRangeBase() {
 | |
|   MOZ_DIAGNOSTIC_ASSERT(!mMutationGuard.Mutated(0));
 | |
| }
 | |
| #endif  // #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| void ContentEventHandler::SimpleRangeBase<
 | |
|     NodeType, RangeBoundaryType>::AssertStartIsBeforeOrEqualToEnd() {
 | |
|   MOZ_ASSERT(
 | |
|       *nsContentUtils::ComparePoints(
 | |
|           mStart.Container(),
 | |
|           *mStart.Offset(
 | |
|               RangeBoundaryType::OffsetFilter::kValidOrInvalidOffsets),
 | |
|           mEnd.Container(),
 | |
|           *mEnd.Offset(
 | |
|               RangeBoundaryType::OffsetFilter::kValidOrInvalidOffsets)) <= 0);
 | |
| }
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| nsresult
 | |
| ContentEventHandler::SimpleRangeBase<NodeType, RangeBoundaryType>::SetStart(
 | |
|     const RawRangeBoundary& aStart) {
 | |
|   nsINode* newRoot = RangeUtils::ComputeRootNode(aStart.Container());
 | |
|   if (!newRoot) {
 | |
|     return NS_ERROR_DOM_INVALID_NODE_TYPE_ERR;
 | |
|   }
 | |
| 
 | |
|   if (!aStart.IsSetAndValid()) {
 | |
|     return NS_ERROR_DOM_INDEX_SIZE_ERR;
 | |
|   }
 | |
| 
 | |
|   // Collapse if not positioned yet, or if positioned in another document.
 | |
|   if (!IsPositioned() || newRoot != mRoot) {
 | |
|     mRoot = newRoot;
 | |
|     mStart.CopyFrom(aStart, RangeBoundaryIsMutationObserved::Yes);
 | |
|     mEnd.CopyFrom(aStart, RangeBoundaryIsMutationObserved::Yes);
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   mStart.CopyFrom(aStart, RangeBoundaryIsMutationObserved::Yes);
 | |
|   AssertStartIsBeforeOrEqualToEnd();
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| nsresult
 | |
| ContentEventHandler::SimpleRangeBase<NodeType, RangeBoundaryType>::SetEnd(
 | |
|     const RawRangeBoundary& aEnd) {
 | |
|   nsINode* newRoot = RangeUtils::ComputeRootNode(aEnd.Container());
 | |
|   if (!newRoot) {
 | |
|     return NS_ERROR_DOM_INVALID_NODE_TYPE_ERR;
 | |
|   }
 | |
| 
 | |
|   if (!aEnd.IsSetAndValid()) {
 | |
|     return NS_ERROR_DOM_INDEX_SIZE_ERR;
 | |
|   }
 | |
| 
 | |
|   // Collapse if not positioned yet, or if positioned in another document.
 | |
|   if (!IsPositioned() || newRoot != mRoot) {
 | |
|     mRoot = newRoot;
 | |
|     mStart.CopyFrom(aEnd, RangeBoundaryIsMutationObserved::Yes);
 | |
|     mEnd.CopyFrom(aEnd, RangeBoundaryIsMutationObserved::Yes);
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   mEnd.CopyFrom(aEnd, RangeBoundaryIsMutationObserved::Yes);
 | |
|   AssertStartIsBeforeOrEqualToEnd();
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| nsresult
 | |
| ContentEventHandler::SimpleRangeBase<NodeType, RangeBoundaryType>::SetEndAfter(
 | |
|     nsINode* aEndContainer) {
 | |
|   return SetEnd(RangeUtils::GetRawRangeBoundaryAfter(aEndContainer));
 | |
| }
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| void ContentEventHandler::SimpleRangeBase<
 | |
|     NodeType, RangeBoundaryType>::SetStartAndEnd(const nsRange* aRange) {
 | |
|   DebugOnly<nsresult> rv =
 | |
|       SetStartAndEnd(aRange->StartRef().AsRaw(), aRange->EndRef().AsRaw());
 | |
|   MOZ_ASSERT(!aRange->IsPositioned() || NS_SUCCEEDED(rv));
 | |
| }
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| nsresult ContentEventHandler::SimpleRangeBase<
 | |
|     NodeType, RangeBoundaryType>::SetStartAndEnd(const RawRangeBoundary& aStart,
 | |
|                                                  const RawRangeBoundary& aEnd) {
 | |
|   nsINode* newStartRoot = RangeUtils::ComputeRootNode(aStart.Container());
 | |
|   if (!newStartRoot) {
 | |
|     return NS_ERROR_DOM_INVALID_NODE_TYPE_ERR;
 | |
|   }
 | |
|   if (!aStart.IsSetAndValid()) {
 | |
|     return NS_ERROR_DOM_INDEX_SIZE_ERR;
 | |
|   }
 | |
| 
 | |
|   if (aStart.Container() == aEnd.Container()) {
 | |
|     if (!aEnd.IsSetAndValid()) {
 | |
|       return NS_ERROR_DOM_INDEX_SIZE_ERR;
 | |
|     }
 | |
|     MOZ_ASSERT(*aStart.Offset(RawRangeBoundary::OffsetFilter::kValidOffsets) <=
 | |
|                *aEnd.Offset(RawRangeBoundary::OffsetFilter::kValidOffsets));
 | |
|     mRoot = newStartRoot;
 | |
|     mStart.CopyFrom(aStart, RangeBoundaryIsMutationObserved::Yes);
 | |
|     mEnd.CopyFrom(aEnd, RangeBoundaryIsMutationObserved::Yes);
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   nsINode* newEndRoot = RangeUtils::ComputeRootNode(aEnd.Container());
 | |
|   if (!newEndRoot) {
 | |
|     return NS_ERROR_DOM_INVALID_NODE_TYPE_ERR;
 | |
|   }
 | |
|   if (!aEnd.IsSetAndValid()) {
 | |
|     return NS_ERROR_DOM_INDEX_SIZE_ERR;
 | |
|   }
 | |
| 
 | |
|   // If they have different root, this should be collapsed at the end point.
 | |
|   if (newStartRoot != newEndRoot) {
 | |
|     mRoot = newEndRoot;
 | |
|     mStart.CopyFrom(aEnd, RangeBoundaryIsMutationObserved::Yes);
 | |
|     mEnd.CopyFrom(aEnd, RangeBoundaryIsMutationObserved::Yes);
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // Otherwise, set the range as specified.
 | |
|   mRoot = newStartRoot;
 | |
|   mStart.CopyFrom(aStart, RangeBoundaryIsMutationObserved::Yes);
 | |
|   mEnd.CopyFrom(aEnd, RangeBoundaryIsMutationObserved::Yes);
 | |
|   AssertStartIsBeforeOrEqualToEnd();
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| nsresult ContentEventHandler::SimpleRangeBase<NodeType, RangeBoundaryType>::
 | |
|     SelectNodeContents(const nsINode* aNodeToSelectContents) {
 | |
|   nsINode* const newRoot =
 | |
|       RangeUtils::ComputeRootNode(const_cast<nsINode*>(aNodeToSelectContents));
 | |
|   if (!newRoot) {
 | |
|     return NS_ERROR_DOM_INVALID_NODE_TYPE_ERR;
 | |
|   }
 | |
|   mRoot = newRoot;
 | |
|   mStart =
 | |
|       RangeBoundaryType(const_cast<nsINode*>(aNodeToSelectContents), nullptr);
 | |
|   mEnd = RangeBoundaryType(const_cast<nsINode*>(aNodeToSelectContents),
 | |
|                            aNodeToSelectContents->GetLastChild());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| /******************************************************************/
 | |
| /* ContentEventHandler                                            */
 | |
| /******************************************************************/
 | |
| 
 | |
| // NOTE
 | |
| //
 | |
| // ContentEventHandler *creates* ranges as following rules:
 | |
| // 1. Start of range:
 | |
| //   1.1. Cases: [textNode or text[Node or textNode[
 | |
| //        When text node is start of a range, start node is the text node and
 | |
| //        start offset is any number between 0 and the length of the text.
 | |
| //   1.2. Case: [<element>:
 | |
| //        When start of an element node is start of a range, start node is
 | |
| //        parent of the element and start offset is the element's index in the
 | |
| //        parent.
 | |
| //   1.3. Case: <element/>[
 | |
| //        When after an empty element node is start of a range, start node is
 | |
| //        parent of the element and start offset is the element's index in the
 | |
| //        parent + 1.
 | |
| //   1.4. Case: <element>[
 | |
| //        When start of a non-empty element is start of a range, start node is
 | |
| //        the element and start offset is 0.
 | |
| //   1.5. Case: <root>[
 | |
| //        When start of a range is 0 and there are no nodes causing text,
 | |
| //        start node is the root node and start offset is 0.
 | |
| //   1.6. Case: [</root>
 | |
| //        When start of a range is out of bounds, start node is the root node
 | |
| //        and start offset is number of the children.
 | |
| // 2. End of range:
 | |
| //   2.1. Cases: ]textNode or text]Node or textNode]
 | |
| //        When a text node is end of a range, end node is the text node and
 | |
| //        end offset is any number between 0 and the length of the text.
 | |
| //   2.2. Case: ]<element>
 | |
| //        When before an element node (meaning before the open tag of the
 | |
| //        element) is end of a range, end node is previous node causing text.
 | |
| //        Note that this case shouldn't be handled directly.  If rule 2.1 and
 | |
| //        2.3 are handled correctly, the loop with ContentIterator shouldn't
 | |
| //        reach the element node since the loop should've finished already at
 | |
| //        handling the last node which caused some text.
 | |
| //   2.3. Case: <element>]
 | |
| //        When a line break is caused before a non-empty element node and it's
 | |
| //        end of a range, end node is the element and end offset is 0.
 | |
| //        (i.e., including open tag of the element)
 | |
| //   2.4. Cases: <element/>]
 | |
| //        When after an empty element node is end of a range, end node is
 | |
| //        parent of the element node and end offset is the element's index in
 | |
| //        the parent + 1.  (i.e., including close tag of the element or empty
 | |
| //        element)
 | |
| //   2.5. Case: ]</root>
 | |
| //        When end of a range is out of bounds, end node is the root node and
 | |
| //        end offset is number of the children.
 | |
| //
 | |
| // ContentEventHandler *treats* ranges as following additional rules:
 | |
| // 1. When the start node is an element node which doesn't have children,
 | |
| //    it includes a line break caused before itself (i.e., includes its open
 | |
| //    tag).  For example, if start position is { <br>, 0 }, the line break
 | |
| //    caused by <br> should be included into the flatten text.
 | |
| // 2. When the end node is an element node which doesn't have children,
 | |
| //    it includes the end (i.e., includes its close tag except empty element).
 | |
| //    Although, currently, any close tags don't cause line break, this also
 | |
| //    includes its open tag.  For example, if end position is { <br>, 0 }, the
 | |
| //    line break caused by the <br> should be included into the flatten text.
 | |
| 
 | |
| ContentEventHandler::ContentEventHandler(nsPresContext* aPresContext)
 | |
|     : mDocument(aPresContext->Document()) {}
 | |
| 
 | |
| nsresult ContentEventHandler::InitBasic(bool aRequireFlush) {
 | |
|   NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE);
 | |
|   if (aRequireFlush) {
 | |
|     // If text frame which has overflowing selection underline is dirty,
 | |
|     // we need to flush the pending reflow here.
 | |
|     mDocument->FlushPendingNotifications(FlushType::Layout);
 | |
|   }
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::InitRootContent(
 | |
|     const Selection& aNormalSelection) {
 | |
|   // Root content should be computed with normal selection because normal
 | |
|   // selection is typically has at least one range but the other selections
 | |
|   // not so.  If there is a range, computing its root is easy, but if
 | |
|   // there are no ranges, we need to use ancestor limit instead.
 | |
|   MOZ_ASSERT(aNormalSelection.Type() == SelectionType::eNormal);
 | |
| 
 | |
|   if (!aNormalSelection.RangeCount()) {
 | |
|     // If there is no selection range, we should compute the selection root
 | |
|     // from ancestor limiter or root content of the document.
 | |
|     mRootElement =
 | |
|         Element::FromNodeOrNull(aNormalSelection.GetAncestorLimiter());
 | |
|     if (!mRootElement) {
 | |
|       mRootElement = mDocument->GetRootElement();
 | |
|       if (NS_WARN_IF(!mRootElement)) {
 | |
|         return NS_ERROR_NOT_AVAILABLE;
 | |
|       }
 | |
|     }
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   RefPtr<const nsRange> range(aNormalSelection.GetRangeAt(0));
 | |
|   if (NS_WARN_IF(!range)) {
 | |
|     return NS_ERROR_UNEXPECTED;
 | |
|   }
 | |
| 
 | |
|   // If there is a selection, we should retrieve the selection root from
 | |
|   // the range since when the window is inactivated, the ancestor limiter
 | |
|   // of selection was cleared by blur event handler of EditorBase but the
 | |
|   // selection range still keeps storing the nodes.  If the active element of
 | |
|   // the deactive window is <input> or <textarea>, we can compute the
 | |
|   // selection root from them.
 | |
|   nsCOMPtr<nsINode> startNode = range->GetStartContainer();
 | |
|   nsINode* endNode = range->GetEndContainer();
 | |
|   if (NS_WARN_IF(!startNode) || NS_WARN_IF(!endNode)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   // See bug 537041 comment 5, the range could have removed node.
 | |
|   if (NS_WARN_IF(startNode->GetComposedDoc() != mDocument)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   NS_ASSERTION(startNode->GetComposedDoc() == endNode->GetComposedDoc(),
 | |
|                "firstNormalSelectionRange crosses the document boundary");
 | |
| 
 | |
|   RefPtr<PresShell> presShell = mDocument->GetPresShell();
 | |
|   mRootElement =
 | |
|       Element::FromNodeOrNull(startNode->GetSelectionRootContent(presShell));
 | |
|   if (NS_WARN_IF(!mRootElement)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::InitCommon(EventMessage aEventMessage,
 | |
|                                          SelectionType aSelectionType,
 | |
|                                          bool aRequireFlush) {
 | |
|   if (mSelection && mSelection->Type() == aSelectionType) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   mSelection = nullptr;
 | |
|   mRootElement = nullptr;
 | |
|   mFirstSelectedSimpleRange.Clear();
 | |
| 
 | |
|   nsresult rv = InitBasic(aRequireFlush);
 | |
|   NS_ENSURE_SUCCESS(rv, rv);
 | |
| 
 | |
|   RefPtr<nsFrameSelection> frameSel;
 | |
|   if (PresShell* presShell = mDocument->GetPresShell()) {
 | |
|     frameSel = presShell->GetLastFocusedFrameSelection();
 | |
|   }
 | |
|   if (NS_WARN_IF(!frameSel)) {
 | |
|     return NS_ERROR_NOT_AVAILABLE;
 | |
|   }
 | |
| 
 | |
|   mSelection = frameSel->GetSelection(aSelectionType);
 | |
|   if (NS_WARN_IF(!mSelection)) {
 | |
|     return NS_ERROR_NOT_AVAILABLE;
 | |
|   }
 | |
| 
 | |
|   RefPtr<Selection> normalSelection;
 | |
|   if (mSelection->Type() == SelectionType::eNormal) {
 | |
|     normalSelection = mSelection;
 | |
|   } else {
 | |
|     normalSelection = frameSel->GetSelection(SelectionType::eNormal);
 | |
|     if (NS_WARN_IF(!normalSelection)) {
 | |
|       return NS_ERROR_NOT_AVAILABLE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   rv = InitRootContent(*normalSelection);
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   if (mSelection->RangeCount()) {
 | |
|     mFirstSelectedSimpleRange.SetStartAndEnd(mSelection->GetRangeAt(0));
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // Even if there are no selection ranges, it's usual case if aSelectionType
 | |
|   // is a special selection or we're handling eQuerySelectedText.
 | |
|   if (aSelectionType != SelectionType::eNormal ||
 | |
|       aEventMessage == eQuerySelectedText) {
 | |
|     MOZ_ASSERT(!mFirstSelectedSimpleRange.IsPositioned());
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // But otherwise, we need to assume that there is a selection range at the
 | |
|   // beginning of the root content if aSelectionType is eNormal.
 | |
|   rv = mFirstSelectedSimpleRange.CollapseTo(RawRangeBoundary(mRootElement, 0u));
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return NS_ERROR_UNEXPECTED;
 | |
|   }
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::Init(WidgetQueryContentEvent* aEvent) {
 | |
|   NS_ASSERTION(aEvent, "aEvent must not be null");
 | |
|   MOZ_ASSERT(aEvent->mMessage == eQuerySelectedText ||
 | |
|              aEvent->mInput.mSelectionType == SelectionType::eNormal);
 | |
| 
 | |
|   if (NS_WARN_IF(!aEvent->mInput.IsValidOffset()) ||
 | |
|       NS_WARN_IF(!aEvent->mInput.IsValidEventMessage(aEvent->mMessage))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   // Note that we should ignore WidgetQueryContentEvent::Input::mSelectionType
 | |
|   // if the event isn't eQuerySelectedText.
 | |
|   SelectionType selectionType = aEvent->mMessage == eQuerySelectedText
 | |
|                                     ? aEvent->mInput.mSelectionType
 | |
|                                     : SelectionType::eNormal;
 | |
|   if (NS_WARN_IF(selectionType == SelectionType::eNone)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   nsresult rv = InitCommon(aEvent->mMessage, selectionType,
 | |
|                            aEvent->AllowFlushingPendingNotifications());
 | |
|   NS_ENSURE_SUCCESS(rv, rv);
 | |
| 
 | |
|   // Be aware, WidgetQueryContentEvent::mInput::mOffset should be made absolute
 | |
|   // offset before sending it to ContentEventHandler because querying selection
 | |
|   // every time may be expensive.  So, if the caller caches selection, it
 | |
|   // should initialize the event with the cached value.
 | |
|   if (aEvent->mInput.mRelativeToInsertionPoint) {
 | |
|     MOZ_ASSERT(selectionType == SelectionType::eNormal);
 | |
|     TextComposition* composition =
 | |
|         IMEStateManager::GetTextCompositionFor(aEvent->mWidget);
 | |
|     if (composition) {
 | |
|       uint32_t compositionStart = composition->NativeOffsetOfStartComposition();
 | |
|       if (NS_WARN_IF(!aEvent->mInput.MakeOffsetAbsolute(compositionStart))) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
|     } else {
 | |
|       LineBreakType lineBreakType = GetLineBreakType(aEvent);
 | |
|       uint32_t selectionStart = 0;
 | |
|       rv = GetStartOffset(mFirstSelectedSimpleRange, &selectionStart,
 | |
|                           lineBreakType);
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
|       if (NS_WARN_IF(!aEvent->mInput.MakeOffsetAbsolute(selectionStart))) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Ideally, we should emplace only when we return succeeded event.
 | |
|   // However, we need to emplace here since it's hard to store the various
 | |
|   // result.  Intead, `HandleQueryContentEvent()` will reset `mReply` if
 | |
|   // corresponding handler returns error.
 | |
|   aEvent->EmplaceReply();
 | |
| 
 | |
|   aEvent->mReply->mContentsRoot = mRootElement.get();
 | |
|   aEvent->mReply->mIsEditableContent =
 | |
|       mRootElement && mRootElement->IsEditable();
 | |
| 
 | |
|   nsRect r;
 | |
|   nsIFrame* frame = nsCaret::GetGeometry(mSelection, &r);
 | |
|   if (!frame) {
 | |
|     frame = mRootElement->GetPrimaryFrame();
 | |
|     if (NS_WARN_IF(!frame)) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
|   }
 | |
|   aEvent->mReply->mFocusedWidget = frame->GetNearestWidget();
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::Init(WidgetSelectionEvent* aEvent) {
 | |
|   NS_ASSERTION(aEvent, "aEvent must not be null");
 | |
| 
 | |
|   nsresult rv = InitCommon(aEvent->mMessage);
 | |
|   NS_ENSURE_SUCCESS(rv, rv);
 | |
| 
 | |
|   aEvent->mSucceeded = false;
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsIContent* ContentEventHandler::GetFocusedContent() {
 | |
|   nsCOMPtr<nsPIDOMWindowOuter> window = mDocument->GetWindow();
 | |
|   nsCOMPtr<nsPIDOMWindowOuter> focusedWindow;
 | |
|   return nsFocusManager::GetFocusedDescendant(
 | |
|       window, nsFocusManager::eIncludeAllDescendants,
 | |
|       getter_AddRefs(focusedWindow));
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::QueryContentRect(
 | |
|     nsIContent* aContent, WidgetQueryContentEvent* aEvent) {
 | |
|   MOZ_ASSERT(aContent, "aContent must not be null");
 | |
| 
 | |
|   nsIFrame* frame = aContent->GetPrimaryFrame();
 | |
|   NS_ENSURE_TRUE(frame, NS_ERROR_FAILURE);
 | |
| 
 | |
|   // get rect for first frame
 | |
|   nsRect resultRect(nsPoint(0, 0), frame->GetRect().Size());
 | |
|   nsresult rv = ConvertToRootRelativeOffset(frame, resultRect);
 | |
|   NS_ENSURE_SUCCESS(rv, rv);
 | |
| 
 | |
|   nsPresContext* presContext = frame->PresContext();
 | |
| 
 | |
|   // account for any additional frames
 | |
|   while ((frame = frame->GetNextContinuation())) {
 | |
|     nsRect frameRect(nsPoint(0, 0), frame->GetRect().Size());
 | |
|     rv = ConvertToRootRelativeOffset(frame, frameRect);
 | |
|     NS_ENSURE_SUCCESS(rv, rv);
 | |
|     resultRect.UnionRect(resultRect, frameRect);
 | |
|   }
 | |
| 
 | |
|   aEvent->mReply->mRect = LayoutDeviceIntRect::FromAppUnitsToOutside(
 | |
|       resultRect, presContext->AppUnitsPerDevPixel());
 | |
|   // Returning empty rect may cause native IME confused, let's make sure to
 | |
|   // return non-empty rect.
 | |
|   EnsureNonEmptyRect(aEvent->mReply->mRect);
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| // Editor places a padding <br> element under its root content if the editor
 | |
| // doesn't have any text. This happens even for single line editors.
 | |
| // When we get text content and when we change the selection,
 | |
| // we don't want to include the padding <br> elements at the end.
 | |
| static bool IsContentBR(const nsIContent& aContent) {
 | |
|   const HTMLBRElement* brElement = HTMLBRElement::FromNode(aContent);
 | |
|   return brElement && !brElement->IsPaddingForEmptyLastLine() &&
 | |
|          !brElement->IsPaddingForEmptyEditor();
 | |
| }
 | |
| 
 | |
| static bool IsPaddingBR(const nsIContent& aContent) {
 | |
|   return aContent.IsHTMLElement(nsGkAtoms::br) && !IsContentBR(aContent);
 | |
| }
 | |
| 
 | |
| static void ConvertToNativeNewlines(nsString& aString) {
 | |
| #if defined(TRANSLATE_NEW_LINES)
 | |
|   aString.ReplaceSubstring(u"\n"_ns, u"\r\n"_ns);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| static void AppendString(nsString& aString, const Text& aTextNode) {
 | |
|   const uint32_t oldXPLength = aString.Length();
 | |
|   aTextNode.TextFragment().AppendTo(aString);
 | |
|   if (aTextNode.HasFlag(NS_MAYBE_MASKED)) {
 | |
|     TextEditor::MaskString(aString, aTextNode, oldXPLength, 0);
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void AppendSubString(nsString& aString, const Text& aTextNode,
 | |
|                             uint32_t aXPOffset, uint32_t aXPLength) {
 | |
|   const uint32_t oldXPLength = aString.Length();
 | |
|   aTextNode.TextFragment().AppendTo(aString, aXPOffset, aXPLength);
 | |
|   if (aTextNode.HasFlag(NS_MAYBE_MASKED)) {
 | |
|     TextEditor::MaskString(aString, aTextNode, oldXPLength, aXPOffset);
 | |
|   }
 | |
| }
 | |
| 
 | |
| #if defined(TRANSLATE_NEW_LINES)
 | |
| template <typename StringType>
 | |
| static uint32_t CountNewlinesInXPLength(const StringType& aString) {
 | |
|   uint32_t count = 0;
 | |
|   const auto* end = aString.EndReading();
 | |
|   for (const auto* iter = aString.BeginReading(); iter < end; ++iter) {
 | |
|     if (*iter == '\n') {
 | |
|       count++;
 | |
|     }
 | |
|   }
 | |
|   return count;
 | |
| }
 | |
| 
 | |
| static uint32_t CountNewlinesInXPLength(const Text& aTextNode,
 | |
|                                         uint32_t aXPLength) {
 | |
|   const nsTextFragment& textFragment = aTextNode.TextFragment();
 | |
|   // For automated tests, we should abort on debug build.
 | |
|   MOZ_ASSERT(aXPLength == UINT32_MAX || aXPLength <= textFragment.GetLength(),
 | |
|              "aXPLength is out-of-bounds");
 | |
|   const uint32_t length = std::min(aXPLength, textFragment.GetLength());
 | |
|   if (!length) {
 | |
|     return 0;
 | |
|   }
 | |
|   if (textFragment.Is2b()) {
 | |
|     nsDependentSubstring str(textFragment.Get2b(), length);
 | |
|     return CountNewlinesInXPLength(str);
 | |
|   }
 | |
|   nsDependentCSubstring str(textFragment.Get1b(), length);
 | |
|   return CountNewlinesInXPLength(str);
 | |
| }
 | |
| 
 | |
| template <typename StringType>
 | |
| static uint32_t CountNewlinesInNativeLength(const StringType& aString,
 | |
|                                             uint32_t aNativeLength) {
 | |
|   MOZ_ASSERT(
 | |
|       (aNativeLength == UINT32_MAX || aNativeLength <= aString.Length() * 2),
 | |
|       "aNativeLength is unexpected value");
 | |
|   uint32_t count = 0;
 | |
|   uint32_t nativeOffset = 0;
 | |
|   const auto* end = aString.EndReading();
 | |
|   for (const auto* iter = aString.BeginReading();
 | |
|        iter < end && nativeOffset < aNativeLength; ++iter, ++nativeOffset) {
 | |
|     if (*iter == '\n') {
 | |
|       count++;
 | |
|       nativeOffset++;
 | |
|     }
 | |
|   }
 | |
|   return count;
 | |
| }
 | |
| 
 | |
| static uint32_t CountNewlinesInNativeLength(const Text& aTextNode,
 | |
|                                             uint32_t aNativeLength) {
 | |
|   const nsTextFragment& textFragment = aTextNode.TextFragment();
 | |
|   const uint32_t xpLength = textFragment.GetLength();
 | |
|   if (!xpLength) {
 | |
|     return 0;
 | |
|   }
 | |
|   if (textFragment.Is2b()) {
 | |
|     nsDependentSubstring str(textFragment.Get2b(), xpLength);
 | |
|     return CountNewlinesInNativeLength(str, aNativeLength);
 | |
|   }
 | |
|   nsDependentCSubstring str(textFragment.Get1b(), xpLength);
 | |
|   return CountNewlinesInNativeLength(str, aNativeLength);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /* static */
 | |
| uint32_t ContentEventHandler::GetNativeTextLength(const Text& aTextNode,
 | |
|                                                   uint32_t aStartOffset,
 | |
|                                                   uint32_t aEndOffset) {
 | |
|   MOZ_ASSERT(aEndOffset >= aStartOffset,
 | |
|              "aEndOffset must be equals or larger than aStartOffset");
 | |
|   if (aStartOffset == aEndOffset) {
 | |
|     return 0;
 | |
|   }
 | |
|   return GetTextLength(aTextNode, LINE_BREAK_TYPE_NATIVE, aEndOffset) -
 | |
|          GetTextLength(aTextNode, LINE_BREAK_TYPE_NATIVE, aStartOffset);
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| uint32_t ContentEventHandler::GetNativeTextLength(const Text& aTextNode,
 | |
|                                                   uint32_t aMaxLength) {
 | |
|   return GetTextLength(aTextNode, LINE_BREAK_TYPE_NATIVE, aMaxLength);
 | |
| }
 | |
| 
 | |
| /* static inline */
 | |
| uint32_t ContentEventHandler::GetBRLength(LineBreakType aLineBreakType) {
 | |
| #if defined(TRANSLATE_NEW_LINES)
 | |
|   // Length of \r\n
 | |
|   return (aLineBreakType == LINE_BREAK_TYPE_NATIVE) ? 2 : 1;
 | |
| #else
 | |
|   return 1;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| uint32_t ContentEventHandler::GetTextLength(const Text& aTextNode,
 | |
|                                             LineBreakType aLineBreakType,
 | |
|                                             uint32_t aMaxLength) {
 | |
|   const uint32_t textLengthDifference =
 | |
| #if defined(TRANSLATE_NEW_LINES)
 | |
|       // On Windows, the length of a native newline ("\r\n") is twice the length
 | |
|       // of the XP newline ("\n"), so XP length is equal to the length of the
 | |
|       // native offset plus the number of newlines encountered in the string.
 | |
|       (aLineBreakType == LINE_BREAK_TYPE_NATIVE)
 | |
|           ? CountNewlinesInXPLength(aTextNode, aMaxLength)
 | |
|           : 0;
 | |
| #else
 | |
|       // On other platforms, the native and XP newlines are the same.
 | |
|       0;
 | |
| #endif
 | |
| 
 | |
|   const uint32_t length =
 | |
|       std::min(aTextNode.TextFragment().GetLength(), aMaxLength);
 | |
|   return length + textLengthDifference;
 | |
| }
 | |
| 
 | |
| static uint32_t ConvertToXPOffset(const Text& aTextNode,
 | |
|                                   uint32_t aNativeOffset) {
 | |
| #if defined(TRANSLATE_NEW_LINES)
 | |
|   // On Windows, the length of a native newline ("\r\n") is twice the length of
 | |
|   // the XP newline ("\n"), so XP offset is equal to the length of the native
 | |
|   // offset minus the number of newlines encountered in the string.
 | |
|   return aNativeOffset - CountNewlinesInNativeLength(aTextNode, aNativeOffset);
 | |
| #else
 | |
|   // On other platforms, the native and XP newlines are the same.
 | |
|   return aNativeOffset;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| uint32_t ContentEventHandler::GetNativeTextLength(const nsAString& aText) {
 | |
|   const uint32_t textLengthDifference =
 | |
| #if defined(TRANSLATE_NEW_LINES)
 | |
|       // On Windows, the length of a native newline ("\r\n") is twice the length
 | |
|       // of the XP newline ("\n"), so XP length is equal to the length of the
 | |
|       // native offset plus the number of newlines encountered in the string.
 | |
|       CountNewlinesInXPLength(aText);
 | |
| #else
 | |
|       // On other platforms, the native and XP newlines are the same.
 | |
|       0;
 | |
| #endif
 | |
|   return aText.Length() + textLengthDifference;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| bool ContentEventHandler::ShouldBreakLineBefore(const nsIContent& aContent,
 | |
|                                                 const Element* aRootElement) {
 | |
|   // We don't need to append linebreak at the start of the root element.
 | |
|   if (&aContent == aRootElement) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // If it's not an HTML element (including other markup language's elements),
 | |
|   // we shouldn't insert like break before that for now.  Becoming this is a
 | |
|   // problem must be edge case.  E.g., when ContentEventHandler is used with
 | |
|   // MathML or SVG elements.
 | |
|   if (!aContent.IsHTMLElement()) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   switch (
 | |
|       nsHTMLTags::CaseSensitiveAtomTagToId(aContent.NodeInfo()->NameAtom())) {
 | |
|     case eHTMLTag_br:
 | |
|       // If the element is <br>, we need to check if the <br> is caused by web
 | |
|       // content.  Otherwise, i.e., it's caused by internal reason of Gecko,
 | |
|       // it shouldn't be exposed as a line break to flatten text.
 | |
|       return IsContentBR(aContent);
 | |
|     case eHTMLTag_a:
 | |
|     case eHTMLTag_abbr:
 | |
|     case eHTMLTag_acronym:
 | |
|     case eHTMLTag_b:
 | |
|     case eHTMLTag_bdi:
 | |
|     case eHTMLTag_bdo:
 | |
|     case eHTMLTag_big:
 | |
|     case eHTMLTag_cite:
 | |
|     case eHTMLTag_code:
 | |
|     case eHTMLTag_data:
 | |
|     case eHTMLTag_del:
 | |
|     case eHTMLTag_dfn:
 | |
|     case eHTMLTag_em:
 | |
|     case eHTMLTag_font:
 | |
|     case eHTMLTag_i:
 | |
|     case eHTMLTag_ins:
 | |
|     case eHTMLTag_kbd:
 | |
|     case eHTMLTag_mark:
 | |
|     case eHTMLTag_s:
 | |
|     case eHTMLTag_samp:
 | |
|     case eHTMLTag_small:
 | |
|     case eHTMLTag_span:
 | |
|     case eHTMLTag_strike:
 | |
|     case eHTMLTag_strong:
 | |
|     case eHTMLTag_sub:
 | |
|     case eHTMLTag_sup:
 | |
|     case eHTMLTag_time:
 | |
|     case eHTMLTag_tt:
 | |
|     case eHTMLTag_u:
 | |
|     case eHTMLTag_var:
 | |
|       // Note that ideally, we should refer the style of the primary frame of
 | |
|       // aContent for deciding if it's an inline.  However, it's difficult
 | |
|       // IMEContentObserver to notify IME of text change caused by style change.
 | |
|       // Therefore, currently, we should check only from the tag for now.
 | |
|       return false;
 | |
|     case eHTMLTag_userdefined:
 | |
|     case eHTMLTag_unknown:
 | |
|       // If the element is unknown element, we shouldn't insert line breaks
 | |
|       // before it since unknown elements should be ignored.
 | |
|       return false;
 | |
|     default:
 | |
|       return true;
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::GenerateFlatTextContent(
 | |
|     const Element* aElement, nsString& aString, LineBreakType aLineBreakType) {
 | |
|   MOZ_ASSERT(aString.IsEmpty());
 | |
| 
 | |
|   UnsafeSimpleRange rawRange;
 | |
|   nsresult rv = rawRange.SelectNodeContents(aElement);
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return rv;
 | |
|   }
 | |
|   return GenerateFlatTextContent(rawRange, aString, aLineBreakType);
 | |
| }
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| nsresult ContentEventHandler::GenerateFlatTextContent(
 | |
|     const SimpleRangeBase<NodeType, RangeBoundaryType>& aSimpleRange,
 | |
|     nsString& aString, LineBreakType aLineBreakType) {
 | |
|   MOZ_ASSERT(aString.IsEmpty());
 | |
| 
 | |
|   if (aSimpleRange.Collapsed()) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   nsINode* startNode = aSimpleRange.GetStartContainer();
 | |
|   nsINode* endNode = aSimpleRange.GetEndContainer();
 | |
|   if (NS_WARN_IF(!startNode) || NS_WARN_IF(!endNode)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   if (startNode == endNode && startNode->IsText()) {
 | |
|     AppendSubString(aString, *startNode->AsText(), aSimpleRange.StartOffset(),
 | |
|                     aSimpleRange.EndOffset() - aSimpleRange.StartOffset());
 | |
|     ConvertToNativeNewlines(aString);
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   UnsafePreContentIterator preOrderIter;
 | |
|   nsresult rv = preOrderIter.Init(aSimpleRange.Start().AsRaw(),
 | |
|                                   aSimpleRange.End().AsRaw());
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return rv;
 | |
|   }
 | |
|   for (; !preOrderIter.IsDone(); preOrderIter.Next()) {
 | |
|     nsINode* node = preOrderIter.GetCurrentNode();
 | |
|     if (NS_WARN_IF(!node)) {
 | |
|       break;
 | |
|     }
 | |
|     if (!node->IsContent()) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     if (const Text* textNode = Text::FromNode(node)) {
 | |
|       if (textNode == startNode) {
 | |
|         AppendSubString(aString, *textNode, aSimpleRange.StartOffset(),
 | |
|                         textNode->TextLength() - aSimpleRange.StartOffset());
 | |
|       } else if (textNode == endNode) {
 | |
|         AppendSubString(aString, *textNode, 0, aSimpleRange.EndOffset());
 | |
|       } else {
 | |
|         AppendString(aString, *textNode);
 | |
|       }
 | |
|     } else if (ShouldBreakLineBefore(*node->AsContent(), mRootElement)) {
 | |
|       aString.Append(char16_t('\n'));
 | |
|     }
 | |
|   }
 | |
|   if (aLineBreakType == LINE_BREAK_TYPE_NATIVE) {
 | |
|     ConvertToNativeNewlines(aString);
 | |
|   }
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| static FontRange* AppendFontRange(nsTArray<FontRange>& aFontRanges,
 | |
|                                   uint32_t aBaseOffset) {
 | |
|   FontRange* fontRange = aFontRanges.AppendElement();
 | |
|   fontRange->mStartOffset = aBaseOffset;
 | |
|   return fontRange;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| uint32_t ContentEventHandler::GetTextLengthInRange(
 | |
|     const Text& aTextNode, uint32_t aXPStartOffset, uint32_t aXPEndOffset,
 | |
|     LineBreakType aLineBreakType) {
 | |
|   return aLineBreakType == LINE_BREAK_TYPE_NATIVE
 | |
|              ? GetNativeTextLength(aTextNode, aXPStartOffset, aXPEndOffset)
 | |
|              : aXPEndOffset - aXPStartOffset;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void ContentEventHandler::AppendFontRanges(FontRangeArray& aFontRanges,
 | |
|                                            const Text& aTextNode,
 | |
|                                            uint32_t aBaseOffset,
 | |
|                                            uint32_t aXPStartOffset,
 | |
|                                            uint32_t aXPEndOffset,
 | |
|                                            LineBreakType aLineBreakType) {
 | |
|   nsIFrame* frame = aTextNode.GetPrimaryFrame();
 | |
|   if (!frame) {
 | |
|     // It is a non-rendered content, create an empty range for it.
 | |
|     AppendFontRange(aFontRanges, aBaseOffset);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   uint32_t baseOffset = aBaseOffset;
 | |
| #ifdef DEBUG
 | |
|   {
 | |
|     nsTextFrame* text = do_QueryFrame(frame);
 | |
|     MOZ_ASSERT(text, "Not a text frame");
 | |
|   }
 | |
| #endif
 | |
|   auto* curr = static_cast<nsTextFrame*>(frame);
 | |
|   while (curr) {
 | |
|     uint32_t frameXPStart = std::max(
 | |
|         static_cast<uint32_t>(curr->GetContentOffset()), aXPStartOffset);
 | |
|     uint32_t frameXPEnd =
 | |
|         std::min(static_cast<uint32_t>(curr->GetContentEnd()), aXPEndOffset);
 | |
|     if (frameXPStart >= frameXPEnd) {
 | |
|       curr = curr->GetNextContinuation();
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     gfxSkipCharsIterator iter = curr->EnsureTextRun(nsTextFrame::eInflated);
 | |
|     gfxTextRun* textRun = curr->GetTextRun(nsTextFrame::eInflated);
 | |
| 
 | |
|     nsTextFrame* next = nullptr;
 | |
|     if (frameXPEnd < aXPEndOffset) {
 | |
|       next = curr->GetNextContinuation();
 | |
|       while (next && next->GetTextRun(nsTextFrame::eInflated) == textRun) {
 | |
|         frameXPEnd = std::min(static_cast<uint32_t>(next->GetContentEnd()),
 | |
|                               aXPEndOffset);
 | |
|         next =
 | |
|             frameXPEnd < aXPEndOffset ? next->GetNextContinuation() : nullptr;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     gfxTextRun::Range skipRange(iter.ConvertOriginalToSkipped(frameXPStart),
 | |
|                                 iter.ConvertOriginalToSkipped(frameXPEnd));
 | |
|     uint32_t lastXPEndOffset = frameXPStart;
 | |
|     for (gfxTextRun::GlyphRunIterator runIter(textRun, skipRange);
 | |
|          !runIter.AtEnd(); runIter.NextRun()) {
 | |
|       gfxFont* font = runIter.GlyphRun()->mFont.get();
 | |
|       uint32_t startXPOffset =
 | |
|           iter.ConvertSkippedToOriginal(runIter.StringStart());
 | |
|       // It is possible that the first glyph run has exceeded the frame,
 | |
|       // because the whole frame is filled by skipped chars.
 | |
|       if (startXPOffset >= frameXPEnd) {
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       if (startXPOffset > lastXPEndOffset) {
 | |
|         // Create range for skipped leading chars.
 | |
|         AppendFontRange(aFontRanges, baseOffset);
 | |
|         baseOffset += GetTextLengthInRange(aTextNode, lastXPEndOffset,
 | |
|                                            startXPOffset, aLineBreakType);
 | |
|       }
 | |
| 
 | |
|       FontRange* fontRange = AppendFontRange(aFontRanges, baseOffset);
 | |
|       fontRange->mFontName.Append(NS_ConvertUTF8toUTF16(font->GetName()));
 | |
| 
 | |
|       ParentLayerToScreenScale2D cumulativeResolution =
 | |
|           ParentLayerToParentLayerScale(
 | |
|               frame->PresShell()->GetCumulativeResolution()) *
 | |
|           nsLayoutUtils::GetTransformToAncestorScaleCrossProcessForFrameMetrics(
 | |
|               frame);
 | |
|       float scale =
 | |
|           std::max(cumulativeResolution.xScale, cumulativeResolution.yScale);
 | |
| 
 | |
|       fontRange->mFontSize = font->GetAdjustedSize() * scale;
 | |
| 
 | |
|       // The converted original offset may exceed the range,
 | |
|       // hence we need to clamp it.
 | |
|       uint32_t endXPOffset = iter.ConvertSkippedToOriginal(runIter.StringEnd());
 | |
|       endXPOffset = std::min(frameXPEnd, endXPOffset);
 | |
|       baseOffset += GetTextLengthInRange(aTextNode, startXPOffset, endXPOffset,
 | |
|                                          aLineBreakType);
 | |
|       lastXPEndOffset = endXPOffset;
 | |
|     }
 | |
|     if (lastXPEndOffset < frameXPEnd) {
 | |
|       // Create range for skipped trailing chars. It also handles case
 | |
|       // that the whole frame contains only skipped chars.
 | |
|       AppendFontRange(aFontRanges, baseOffset);
 | |
|       baseOffset += GetTextLengthInRange(aTextNode, lastXPEndOffset, frameXPEnd,
 | |
|                                          aLineBreakType);
 | |
|     }
 | |
| 
 | |
|     curr = next;
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::GenerateFlatFontRanges(
 | |
|     const UnsafeSimpleRange& aSimpleRange, FontRangeArray& aFontRanges,
 | |
|     uint32_t& aLength, LineBreakType aLineBreakType) {
 | |
|   MOZ_ASSERT(aFontRanges.IsEmpty(), "aRanges must be empty array");
 | |
| 
 | |
|   if (aSimpleRange.Collapsed()) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   nsINode* startNode = aSimpleRange.GetStartContainer();
 | |
|   nsINode* endNode = aSimpleRange.GetEndContainer();
 | |
|   if (NS_WARN_IF(!startNode) || NS_WARN_IF(!endNode)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   // baseOffset is the flattened offset of each content node.
 | |
|   uint32_t baseOffset = 0;
 | |
|   UnsafePreContentIterator preOrderIter;
 | |
|   nsresult rv = preOrderIter.Init(aSimpleRange.Start().AsRaw(),
 | |
|                                   aSimpleRange.End().AsRaw());
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return rv;
 | |
|   }
 | |
|   for (; !preOrderIter.IsDone(); preOrderIter.Next()) {
 | |
|     nsINode* node = preOrderIter.GetCurrentNode();
 | |
|     if (NS_WARN_IF(!node)) {
 | |
|       break;
 | |
|     }
 | |
|     if (!node->IsContent()) {
 | |
|       continue;
 | |
|     }
 | |
|     nsIContent* content = node->AsContent();
 | |
| 
 | |
|     if (const Text* textNode = Text::FromNode(content)) {
 | |
|       const uint32_t startOffset =
 | |
|           textNode != startNode ? 0 : aSimpleRange.StartOffset();
 | |
|       const uint32_t endOffset = textNode != endNode ? textNode->TextLength()
 | |
|                                                      : aSimpleRange.EndOffset();
 | |
|       AppendFontRanges(aFontRanges, *textNode, baseOffset, startOffset,
 | |
|                        endOffset, aLineBreakType);
 | |
|       baseOffset += GetTextLengthInRange(*textNode, startOffset, endOffset,
 | |
|                                          aLineBreakType);
 | |
|     } else if (ShouldBreakLineBefore(*content, mRootElement)) {
 | |
|       if (aFontRanges.IsEmpty()) {
 | |
|         MOZ_ASSERT(baseOffset == 0);
 | |
|         FontRange* fontRange = AppendFontRange(aFontRanges, baseOffset);
 | |
|         if (nsIFrame* frame = content->GetPrimaryFrame()) {
 | |
|           const nsFont& font = frame->GetParent()->StyleFont()->mFont;
 | |
|           const StyleFontFamilyList& fontList = font.family.families;
 | |
|           MOZ_ASSERT(!fontList.list.IsEmpty(), "Empty font family?");
 | |
|           const StyleSingleFontFamily* fontName =
 | |
|               fontList.list.IsEmpty() ? nullptr : &fontList.list.AsSpan()[0];
 | |
|           nsAutoCString name;
 | |
|           if (fontName) {
 | |
|             fontName->AppendToString(name, false);
 | |
|           }
 | |
|           AppendUTF8toUTF16(name, fontRange->mFontName);
 | |
| 
 | |
|           ParentLayerToScreenScale2D cumulativeResolution =
 | |
|               ParentLayerToParentLayerScale(
 | |
|                   frame->PresShell()->GetCumulativeResolution()) *
 | |
|               nsLayoutUtils::
 | |
|                   GetTransformToAncestorScaleCrossProcessForFrameMetrics(frame);
 | |
| 
 | |
|           float scale = std::max(cumulativeResolution.xScale,
 | |
|                                  cumulativeResolution.yScale);
 | |
| 
 | |
|           fontRange->mFontSize = frame->PresContext()->CSSPixelsToDevPixels(
 | |
|               font.size.ToCSSPixels() * scale);
 | |
|         }
 | |
|       }
 | |
|       baseOffset += GetBRLength(aLineBreakType);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   aLength = baseOffset;
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::ExpandToClusterBoundary(
 | |
|     Text& aTextNode, bool aForward, uint32_t* aXPOffset) const {
 | |
|   // XXX This method assumes that the frame boundaries must be cluster
 | |
|   // boundaries. It's false, but no problem now, maybe.
 | |
|   if (*aXPOffset == 0 || *aXPOffset == aTextNode.TextLength()) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   NS_ASSERTION(*aXPOffset <= aTextNode.TextLength(), "offset is out of range.");
 | |
| 
 | |
|   MOZ_DIAGNOSTIC_ASSERT(mDocument->GetPresShell());
 | |
|   CaretAssociationHint hint =
 | |
|       aForward ? CaretAssociationHint::Before : CaretAssociationHint::After;
 | |
|   nsIFrame* frame = SelectionMovementUtils::GetFrameForNodeOffset(
 | |
|       &aTextNode, int32_t(*aXPOffset), hint);
 | |
|   if (frame) {
 | |
|     auto [startOffset, endOffset] = frame->GetOffsets();
 | |
|     if (*aXPOffset == static_cast<uint32_t>(startOffset) ||
 | |
|         *aXPOffset == static_cast<uint32_t>(endOffset)) {
 | |
|       return NS_OK;
 | |
|     }
 | |
|     if (!frame->IsTextFrame()) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
|     nsTextFrame* textFrame = static_cast<nsTextFrame*>(frame);
 | |
|     int32_t newOffsetInFrame = *aXPOffset - startOffset;
 | |
|     newOffsetInFrame += aForward ? -1 : 1;
 | |
|     // PeekOffsetCharacter() should respect cluster but ignore user-select
 | |
|     // style.  If it returns "FOUND", we should use the result.  Otherwise,
 | |
|     // we shouldn't use the result because the offset was moved to reversed
 | |
|     // direction.
 | |
|     nsTextFrame::PeekOffsetCharacterOptions options;
 | |
|     options.mRespectClusters = true;
 | |
|     options.mIgnoreUserStyleAll = true;
 | |
|     if (textFrame->PeekOffsetCharacter(aForward, &newOffsetInFrame, options) ==
 | |
|         nsIFrame::FOUND) {
 | |
|       *aXPOffset = startOffset + newOffsetInFrame;
 | |
|       return NS_OK;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // If the frame isn't available, we only can check surrogate pair...
 | |
|   if (aTextNode.TextFragment().IsLowSurrogateFollowingHighSurrogateAt(
 | |
|           *aXPOffset)) {
 | |
|     *aXPOffset += aForward ? 1 : -1;
 | |
|   }
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| template <typename RangeType, typename TextNodeType>
 | |
| Result<ContentEventHandler::DOMRangeAndAdjustedOffsetInFlattenedTextBase<
 | |
|            RangeType, TextNodeType>,
 | |
|        nsresult>
 | |
| ContentEventHandler::ConvertFlatTextOffsetToDOMRangeBase(
 | |
|     uint32_t aOffset, uint32_t aLength, LineBreakType aLineBreakType,
 | |
|     bool aExpandToClusterBoundaries) {
 | |
|   DOMRangeAndAdjustedOffsetInFlattenedTextBase<RangeType, TextNodeType> result;
 | |
|   result.mAdjustedOffset = aOffset;
 | |
| 
 | |
|   // Special case like <br contenteditable>
 | |
|   if (!mRootElement->HasChildren()) {
 | |
|     nsresult rv = result.mRange.CollapseTo(RawRangeBoundary(mRootElement, 0u));
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return Err(rv);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   UnsafePreContentIterator preOrderIter;
 | |
|   nsresult rv = preOrderIter.Init(mRootElement);
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return Err(rv);
 | |
|   }
 | |
| 
 | |
|   uint32_t offset = 0;
 | |
|   uint32_t endOffset = aOffset + aLength;
 | |
|   bool startSet = false;
 | |
|   for (; !preOrderIter.IsDone(); preOrderIter.Next()) {
 | |
|     nsINode* node = preOrderIter.GetCurrentNode();
 | |
|     if (NS_WARN_IF(!node)) {
 | |
|       break;
 | |
|     }
 | |
|     // FYI: mRootElement shouldn't cause any text. So, we can skip it simply.
 | |
|     if (node == mRootElement || !node->IsContent()) {
 | |
|       continue;
 | |
|     }
 | |
|     nsIContent* const content = node->AsContent();
 | |
|     Text* const contentAsText = Text::FromNode(content);
 | |
| 
 | |
|     if (contentAsText) {
 | |
|       result.mLastTextNode = contentAsText;
 | |
|     }
 | |
| 
 | |
|     uint32_t textLength = contentAsText
 | |
|                               ? GetTextLength(*contentAsText, aLineBreakType)
 | |
|                               : (ShouldBreakLineBefore(*content, mRootElement)
 | |
|                                      ? GetBRLength(aLineBreakType)
 | |
|                                      : 0);
 | |
|     if (!textLength) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     // When the start offset is in between accumulated offset and the last
 | |
|     // offset of the node, the node is the start node of the range.
 | |
|     if (!startSet && aOffset <= offset + textLength) {
 | |
|       nsINode* startNode = nullptr;
 | |
|       Maybe<uint32_t> startNodeOffset;
 | |
|       if (contentAsText) {
 | |
|         // Rule #1.1: [textNode or text[Node or textNode[
 | |
|         uint32_t xpOffset = aOffset - offset;
 | |
|         if (aLineBreakType == LINE_BREAK_TYPE_NATIVE) {
 | |
|           xpOffset = ConvertToXPOffset(*contentAsText, xpOffset);
 | |
|         }
 | |
| 
 | |
|         if (aExpandToClusterBoundaries) {
 | |
|           const uint32_t oldXPOffset = xpOffset;
 | |
|           nsresult rv =
 | |
|               ExpandToClusterBoundary(*contentAsText, false, &xpOffset);
 | |
|           if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|             return Err(rv);
 | |
|           }
 | |
|           // This is correct since a cluster shouldn't include line break.
 | |
|           result.mAdjustedOffset -= (oldXPOffset - xpOffset);
 | |
|         }
 | |
|         startNode = contentAsText;
 | |
|         startNodeOffset = Some(xpOffset);
 | |
|       } else if (aOffset < offset + textLength) {
 | |
|         // Rule #1.2 [<element>
 | |
|         startNode = content->GetParent();
 | |
|         if (NS_WARN_IF(!startNode)) {
 | |
|           return Err(NS_ERROR_FAILURE);
 | |
|         }
 | |
|         startNodeOffset = startNode->ComputeIndexOf(content);
 | |
|         if (NS_WARN_IF(startNodeOffset.isNothing())) {
 | |
|           // The content is being removed from the parent!
 | |
|           return Err(NS_ERROR_FAILURE);
 | |
|         }
 | |
|       } else if (!content->HasChildren()) {
 | |
|         // Rule #1.3: <element/>[
 | |
|         startNode = content->GetParent();
 | |
|         if (NS_WARN_IF(!startNode)) {
 | |
|           return Err(NS_ERROR_FAILURE);
 | |
|         }
 | |
|         startNodeOffset = startNode->ComputeIndexOf(content);
 | |
|         if (NS_WARN_IF(startNodeOffset.isNothing())) {
 | |
|           // The content is being removed from the parent!
 | |
|           return Err(NS_ERROR_FAILURE);
 | |
|         }
 | |
|         MOZ_ASSERT(*startNodeOffset != UINT32_MAX);
 | |
|         ++(*startNodeOffset);
 | |
|       } else {
 | |
|         // Rule #1.4: <element>[
 | |
|         startNode = content;
 | |
|         startNodeOffset = Some(0);
 | |
|       }
 | |
|       NS_ASSERTION(startNode, "startNode must not be nullptr");
 | |
|       MOZ_ASSERT(startNodeOffset.isSome(),
 | |
|                  "startNodeOffset must not be Nothing");
 | |
|       rv = result.mRange.SetStart(startNode, *startNodeOffset);
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return Err(rv);
 | |
|       }
 | |
|       startSet = true;
 | |
| 
 | |
|       if (!aLength) {
 | |
|         rv = result.mRange.SetEnd(startNode, *startNodeOffset);
 | |
|         if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|           return Err(rv);
 | |
|         }
 | |
|         return result;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // When the end offset is in the content, the node is the end node of the
 | |
|     // range.
 | |
|     if (endOffset <= offset + textLength) {
 | |
|       MOZ_ASSERT(startSet, "The start of the range should've been set already");
 | |
|       if (contentAsText) {
 | |
|         // Rule #2.1: ]textNode or text]Node or textNode]
 | |
|         uint32_t xpOffset = endOffset - offset;
 | |
|         if (aLineBreakType == LINE_BREAK_TYPE_NATIVE) {
 | |
|           const uint32_t xpOffsetCurrent =
 | |
|               ConvertToXPOffset(*contentAsText, xpOffset);
 | |
|           if (xpOffset && GetBRLength(aLineBreakType) > 1) {
 | |
|             MOZ_ASSERT(GetBRLength(aLineBreakType) == 2);
 | |
|             const uint32_t xpOffsetPre =
 | |
|                 ConvertToXPOffset(*contentAsText, xpOffset - 1);
 | |
|             // If previous character's XP offset is same as current character's,
 | |
|             // it means that the end offset is between \r and \n.  So, the
 | |
|             // range end should be after the \n.
 | |
|             if (xpOffsetPre == xpOffsetCurrent) {
 | |
|               xpOffset = xpOffsetCurrent + 1;
 | |
|             } else {
 | |
|               xpOffset = xpOffsetCurrent;
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|         if (aExpandToClusterBoundaries) {
 | |
|           nsresult rv =
 | |
|               ExpandToClusterBoundary(*contentAsText, true, &xpOffset);
 | |
|           if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|             return Err(rv);
 | |
|           }
 | |
|         }
 | |
|         NS_ASSERTION(xpOffset <= INT32_MAX, "The end node offset is too large");
 | |
|         nsresult rv = result.mRange.SetEnd(contentAsText, xpOffset);
 | |
|         if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|           return Err(rv);
 | |
|         }
 | |
|         return result;
 | |
|       }
 | |
| 
 | |
|       if (endOffset == offset) {
 | |
|         // Rule #2.2: ]<element>
 | |
|         // NOTE: Please don't crash on release builds because it must be
 | |
|         //       overreaction but we shouldn't allow this bug when some
 | |
|         //       automated tests find this.
 | |
|         MOZ_ASSERT(false,
 | |
|                    "This case should've already been handled at "
 | |
|                    "the last node which caused some text");
 | |
|         return Err(NS_ERROR_FAILURE);
 | |
|       }
 | |
| 
 | |
|       if (content->HasChildren() &&
 | |
|           ShouldBreakLineBefore(*content, mRootElement)) {
 | |
|         // Rule #2.3: </element>]
 | |
|         rv = result.mRange.SetEnd(content, 0);
 | |
|         if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|           return Err(rv);
 | |
|         }
 | |
|         return result;
 | |
|       }
 | |
| 
 | |
|       // Rule #2.4: <element/>]
 | |
|       nsINode* endNode = content->GetParent();
 | |
|       if (NS_WARN_IF(!endNode)) {
 | |
|         return Err(NS_ERROR_FAILURE);
 | |
|       }
 | |
|       const Maybe<uint32_t> indexInParent = endNode->ComputeIndexOf(content);
 | |
|       if (NS_WARN_IF(indexInParent.isNothing())) {
 | |
|         // The content is being removed from the parent!
 | |
|         return Err(NS_ERROR_FAILURE);
 | |
|       }
 | |
|       MOZ_ASSERT(*indexInParent != UINT32_MAX);
 | |
|       rv = result.mRange.SetEnd(endNode, *indexInParent + 1);
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return Err(rv);
 | |
|       }
 | |
|       return result;
 | |
|     }
 | |
| 
 | |
|     offset += textLength;
 | |
|   }
 | |
| 
 | |
|   if (!startSet) {
 | |
|     if (!offset) {
 | |
|       // Rule #1.5: <root>[</root>
 | |
|       // When there are no nodes causing text, the start of the DOM range
 | |
|       // should be start of the root node since clicking on such editor (e.g.,
 | |
|       // <div contenteditable><span></span></div>) sets caret to the start of
 | |
|       // the editor (i.e., before <span> in the example).
 | |
|       rv = result.mRange.SetStart(mRootElement, 0);
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return Err(rv);
 | |
|       }
 | |
|       if (!aLength) {
 | |
|         rv = result.mRange.SetEnd(mRootElement, 0);
 | |
|         if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|           return Err(rv);
 | |
|         }
 | |
|         return result;
 | |
|       }
 | |
|     } else {
 | |
|       // Rule #1.5: [</root>
 | |
|       rv = result.mRange.SetStart(mRootElement, mRootElement->GetChildCount());
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return result;
 | |
|       }
 | |
|     }
 | |
|     result.mAdjustedOffset = offset;
 | |
|   }
 | |
|   // Rule #2.5: ]</root>
 | |
|   rv = result.mRange.SetEnd(mRootElement, mRootElement->GetChildCount());
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return Err(rv);
 | |
|   }
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| LineBreakType ContentEventHandler::GetLineBreakType(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   return GetLineBreakType(aEvent->mUseNativeLineBreak);
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| LineBreakType ContentEventHandler::GetLineBreakType(
 | |
|     WidgetSelectionEvent* aEvent) {
 | |
|   return GetLineBreakType(aEvent->mUseNativeLineBreak);
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| LineBreakType ContentEventHandler::GetLineBreakType(bool aUseNativeLineBreak) {
 | |
|   return aUseNativeLineBreak ? LINE_BREAK_TYPE_NATIVE : LINE_BREAK_TYPE_XP;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::HandleQueryContentEvent(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   nsresult rv = NS_ERROR_NOT_IMPLEMENTED;
 | |
|   switch (aEvent->mMessage) {
 | |
|     case eQuerySelectedText:
 | |
|       rv = OnQuerySelectedText(aEvent);
 | |
|       break;
 | |
|     case eQueryTextContent:
 | |
|       rv = OnQueryTextContent(aEvent);
 | |
|       break;
 | |
|     case eQueryCaretRect:
 | |
|       rv = OnQueryCaretRect(aEvent);
 | |
|       break;
 | |
|     case eQueryTextRect:
 | |
|       rv = OnQueryTextRect(aEvent);
 | |
|       break;
 | |
|     case eQueryTextRectArray:
 | |
|       rv = OnQueryTextRectArray(aEvent);
 | |
|       break;
 | |
|     case eQueryEditorRect:
 | |
|       rv = OnQueryEditorRect(aEvent);
 | |
|       break;
 | |
|     case eQueryContentState:
 | |
|       rv = OnQueryContentState(aEvent);
 | |
|       break;
 | |
|     case eQuerySelectionAsTransferable:
 | |
|       rv = OnQuerySelectionAsTransferable(aEvent);
 | |
|       break;
 | |
|     case eQueryCharacterAtPoint:
 | |
|       rv = OnQueryCharacterAtPoint(aEvent);
 | |
|       break;
 | |
|     case eQueryDOMWidgetHittest:
 | |
|       rv = OnQueryDOMWidgetHittest(aEvent);
 | |
|       break;
 | |
|     default:
 | |
|       break;
 | |
|   }
 | |
|   if (NS_FAILED(rv)) {
 | |
|     aEvent->mReply.reset();  // Mark the query failed.
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| // Similar to nsFrameSelection::GetFrameForNodeOffset,
 | |
| // but this is more flexible for OnQueryTextRect to use
 | |
| static Result<nsIFrame*, nsresult> GetFrameForTextRect(const nsINode* aNode,
 | |
|                                                        int32_t aNodeOffset,
 | |
|                                                        bool aHint) {
 | |
|   const nsIContent* content = nsIContent::FromNodeOrNull(aNode);
 | |
|   if (NS_WARN_IF(!content)) {
 | |
|     return Err(NS_ERROR_UNEXPECTED);
 | |
|   }
 | |
|   nsIFrame* frame = content->GetPrimaryFrame();
 | |
|   // The node may be invisible, e.g., `display: none`, invisible text node
 | |
|   // around block elements, etc.  Therefore, don't warn when we don't find
 | |
|   // a primary frame.
 | |
|   if (!frame) {
 | |
|     return nullptr;
 | |
|   }
 | |
|   int32_t childNodeOffset = 0;
 | |
|   nsIFrame* returnFrame = nullptr;
 | |
|   nsresult rv = frame->GetChildFrameContainingOffset(
 | |
|       aNodeOffset, aHint, &childNodeOffset, &returnFrame);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return Err(rv);
 | |
|   }
 | |
|   return returnFrame;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQuerySelectedText(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   nsresult rv = Init(aEvent);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->mReply->mOffsetAndData.isNothing());
 | |
| 
 | |
|   if (!mFirstSelectedSimpleRange.IsPositioned()) {
 | |
|     MOZ_ASSERT(aEvent->mReply->mOffsetAndData.isNothing());
 | |
|     MOZ_ASSERT_IF(mSelection, !mSelection->RangeCount());
 | |
|     // This is special case that `mReply` is emplaced, but mOffsetAndData is
 | |
|     // not emplaced but treated as succeeded because of no selection ranges
 | |
|     // is a usual case.
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   const UnsafeSimpleRange firstSelectedSimpleRange(mFirstSelectedSimpleRange);
 | |
|   nsINode* const startNode = firstSelectedSimpleRange.GetStartContainer();
 | |
|   nsINode* const endNode = firstSelectedSimpleRange.GetEndContainer();
 | |
| 
 | |
|   // Make sure the selection is within the root content range.
 | |
|   if (!startNode->IsInclusiveDescendantOf(mRootElement) ||
 | |
|       !endNode->IsInclusiveDescendantOf(mRootElement)) {
 | |
|     return NS_ERROR_NOT_AVAILABLE;
 | |
|   }
 | |
| 
 | |
|   LineBreakType lineBreakType = GetLineBreakType(aEvent);
 | |
|   uint32_t startOffset = 0;
 | |
|   if (NS_WARN_IF(NS_FAILED(GetStartOffset(firstSelectedSimpleRange,
 | |
|                                           &startOffset, lineBreakType)))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   const RawRangeBoundary anchorRef = mSelection->RangeCount() > 0
 | |
|                                          ? mSelection->AnchorRef().AsRaw()
 | |
|                                          : firstSelectedSimpleRange.Start();
 | |
|   const RawRangeBoundary focusRef = mSelection->RangeCount() > 0
 | |
|                                         ? mSelection->FocusRef().AsRaw()
 | |
|                                         : firstSelectedSimpleRange.End();
 | |
|   if (NS_WARN_IF(!anchorRef.IsSet()) || NS_WARN_IF(!focusRef.IsSet())) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   if (mSelection->RangeCount()) {
 | |
|     // If there is only one selection range, the anchor/focus node and offset
 | |
|     // are the information of the range.  Therefore, we have the direction
 | |
|     // information.
 | |
|     if (mSelection->RangeCount() == 1) {
 | |
|       // The selection's points should always be comparable, independent of the
 | |
|       // selection (see nsISelectionController.idl).
 | |
|       Maybe<int32_t> compare =
 | |
|           nsContentUtils::ComparePoints(anchorRef, focusRef);
 | |
|       if (compare.isNothing()) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
| 
 | |
|       aEvent->mReply->mReversed = compare.value() > 0;
 | |
|     }
 | |
|     // However, if there are 2 or more selection ranges, we have no information
 | |
|     // of that.
 | |
|     else {
 | |
|       aEvent->mReply->mReversed = false;
 | |
|     }
 | |
| 
 | |
|     nsString selectedString;
 | |
|     if (!firstSelectedSimpleRange.Collapsed() &&
 | |
|         NS_WARN_IF(NS_FAILED(GenerateFlatTextContent(
 | |
|             firstSelectedSimpleRange, selectedString, lineBreakType)))) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
|     aEvent->mReply->mOffsetAndData.emplace(startOffset, selectedString,
 | |
|                                            OffsetAndDataFor::SelectedString);
 | |
|   } else {
 | |
|     NS_ASSERTION(anchorRef == focusRef,
 | |
|                  "When mSelection doesn't have selection, "
 | |
|                  "mFirstSelectedRawRange must be collapsed");
 | |
| 
 | |
|     aEvent->mReply->mReversed = false;
 | |
|     aEvent->mReply->mOffsetAndData.emplace(startOffset, EmptyString(),
 | |
|                                            OffsetAndDataFor::SelectedString);
 | |
|   }
 | |
| 
 | |
|   Result<nsIFrame*, nsresult> frameForTextRectOrError = GetFrameForTextRect(
 | |
|       focusRef.Container(),
 | |
|       focusRef.Offset(RawRangeBoundary::OffsetFilter::kValidOffsets).valueOr(0),
 | |
|       true);
 | |
|   if (NS_WARN_IF(frameForTextRectOrError.isErr()) ||
 | |
|       !frameForTextRectOrError.inspect()) {
 | |
|     aEvent->mReply->mWritingMode = WritingMode();
 | |
|   } else {
 | |
|     aEvent->mReply->mWritingMode =
 | |
|         frameForTextRectOrError.inspect()->GetWritingMode();
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQueryTextContent(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   nsresult rv = Init(aEvent);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->mReply->mOffsetAndData.isNothing());
 | |
| 
 | |
|   LineBreakType lineBreakType = GetLineBreakType(aEvent);
 | |
| 
 | |
|   Result<UnsafeDOMRangeAndAdjustedOffsetInFlattenedText, nsresult>
 | |
|       domRangeAndAdjustedOffsetOrError = ConvertFlatTextOffsetToUnsafeDOMRange(
 | |
|           aEvent->mInput.mOffset, aEvent->mInput.mLength, lineBreakType, false);
 | |
|   if (MOZ_UNLIKELY(domRangeAndAdjustedOffsetOrError.isErr())) {
 | |
|     NS_WARNING(
 | |
|         "ContentEventHandler::ConvertFlatTextOffsetToDOMRangeBase() failed");
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
|   const UnsafeDOMRangeAndAdjustedOffsetInFlattenedText
 | |
|       domRangeAndAdjustedOffset = domRangeAndAdjustedOffsetOrError.unwrap();
 | |
| 
 | |
|   nsString textInRange;
 | |
|   if (NS_WARN_IF(NS_FAILED(GenerateFlatTextContent(
 | |
|           domRangeAndAdjustedOffset.mRange, textInRange, lineBreakType)))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   aEvent->mReply->mOffsetAndData.emplace(
 | |
|       domRangeAndAdjustedOffset.mAdjustedOffset, textInRange,
 | |
|       OffsetAndDataFor::EditorString);
 | |
| 
 | |
|   if (aEvent->mWithFontRanges) {
 | |
|     uint32_t fontRangeLength;
 | |
|     if (NS_WARN_IF(NS_FAILED(GenerateFlatFontRanges(
 | |
|             domRangeAndAdjustedOffset.mRange, aEvent->mReply->mFontRanges,
 | |
|             fontRangeLength, lineBreakType)))) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
| 
 | |
|     MOZ_ASSERT(fontRangeLength == aEvent->mReply->DataLength(),
 | |
|                "Font ranges doesn't match the string");
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| void ContentEventHandler::EnsureNonEmptyRect(nsRect& aRect) const {
 | |
|   // See the comment in ContentEventHandler.h why this doesn't set them to
 | |
|   // one device pixel.
 | |
|   aRect.height = std::max(1, aRect.height);
 | |
|   aRect.width = std::max(1, aRect.width);
 | |
| }
 | |
| 
 | |
| void ContentEventHandler::EnsureNonEmptyRect(LayoutDeviceIntRect& aRect) const {
 | |
|   aRect.height = std::max(1, aRect.height);
 | |
|   aRect.width = std::max(1, aRect.width);
 | |
| }
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| ContentEventHandler::FrameAndNodeOffset
 | |
| ContentEventHandler::GetFirstFrameInRangeForTextRect(
 | |
|     const SimpleRangeBase<NodeType, RangeBoundaryType>& aSimpleRange) {
 | |
|   RawNodePosition nodePosition;
 | |
|   UnsafePreContentIterator preOrderIter;
 | |
|   nsresult rv = preOrderIter.Init(aSimpleRange.Start().AsRaw(),
 | |
|                                   aSimpleRange.End().AsRaw());
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return FrameAndNodeOffset();
 | |
|   }
 | |
|   for (; !preOrderIter.IsDone(); preOrderIter.Next()) {
 | |
|     nsINode* node = preOrderIter.GetCurrentNode();
 | |
|     if (NS_WARN_IF(!node)) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     auto* content = nsIContent::FromNode(node);
 | |
|     if (MOZ_UNLIKELY(!content)) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     // If the node is invisible (e.g., the node is or is in an invisible node or
 | |
|     // it's a white-space only text node around a block boundary), we should
 | |
|     // ignore it.
 | |
|     if (!content->GetPrimaryFrame()) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     if (auto* textNode = Text::FromNode(content)) {
 | |
|       // If the range starts at the end of a text node, we need to find
 | |
|       // next node which causes text.
 | |
|       const uint32_t offsetInNode = textNode == aSimpleRange.GetStartContainer()
 | |
|                                         ? aSimpleRange.StartOffset()
 | |
|                                         : 0u;
 | |
|       if (offsetInNode < textNode->TextDataLength()) {
 | |
|         nodePosition = {textNode, offsetInNode};
 | |
|         break;
 | |
|       }
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     // If the element node causes a line break before it, it's the first
 | |
|     // node causing text.
 | |
|     if (ShouldBreakLineBefore(*content, mRootElement) ||
 | |
|         IsPaddingBR(*content)) {
 | |
|       nodePosition = {content, 0u};
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!nodePosition.IsSetAndValid()) {
 | |
|     return FrameAndNodeOffset();
 | |
|   }
 | |
| 
 | |
|   Result<nsIFrame*, nsresult> firstFrameOrError = GetFrameForTextRect(
 | |
|       nodePosition.Container(),
 | |
|       *nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets), true);
 | |
|   if (NS_WARN_IF(firstFrameOrError.isErr()) || !firstFrameOrError.inspect()) {
 | |
|     return FrameAndNodeOffset();
 | |
|   }
 | |
|   return FrameAndNodeOffset(
 | |
|       firstFrameOrError.inspect(),
 | |
|       *nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets));
 | |
| }
 | |
| 
 | |
| template <typename NodeType, typename RangeBoundaryType>
 | |
| ContentEventHandler::FrameAndNodeOffset
 | |
| ContentEventHandler::GetLastFrameInRangeForTextRect(
 | |
|     const SimpleRangeBase<NodeType, RangeBoundaryType>& aSimpleRange) {
 | |
|   RawNodePosition nodePosition;
 | |
|   UnsafePreContentIterator preOrderIter;
 | |
|   nsresult rv = preOrderIter.Init(aSimpleRange.Start().AsRaw(),
 | |
|                                   aSimpleRange.End().AsRaw());
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return FrameAndNodeOffset();
 | |
|   }
 | |
| 
 | |
|   const RangeBoundaryType& endPoint = aSimpleRange.End();
 | |
|   MOZ_ASSERT(endPoint.IsSetAndValid());
 | |
|   // If the end point is start of a text node or specified by its parent and
 | |
|   // index, the node shouldn't be included into the range.  For example,
 | |
|   // with this case, |<p>abc[<br>]def</p>|, the range ends at 3rd children of
 | |
|   // <p> (see the range creation rules, "2.4. Cases: <element/>]"). This causes
 | |
|   // following frames:
 | |
|   // +----+-----+
 | |
|   // | abc|[<br>|
 | |
|   // +----+-----+
 | |
|   // +----+
 | |
|   // |]def|
 | |
|   // +----+
 | |
|   // So, if this method includes the 2nd text frame's rect to its result, the
 | |
|   // caller will return too tall rect which includes 2 lines in this case isn't
 | |
|   // expected by native IME  (e.g., popup of IME will be positioned at bottom
 | |
|   // of "d" instead of right-bottom of "c").  Therefore, this method shouldn't
 | |
|   // include the last frame when its content isn't really in aSimpleRange.
 | |
|   nsINode* nextNodeOfRangeEnd = nullptr;
 | |
|   if (endPoint.Container()->IsText()) {
 | |
|     // Don't set nextNodeOfRangeEnd to the start node of aSimpleRange because if
 | |
|     // the container of the end is same as start node of the range, the text
 | |
|     // node shouldn't be next of range end even if the offset is 0.  This
 | |
|     // could occur with empty text node.
 | |
|     if (endPoint.IsStartOfContainer() &&
 | |
|         aSimpleRange.GetStartContainer() != endPoint.Container()) {
 | |
|       nextNodeOfRangeEnd = endPoint.Container();
 | |
|     }
 | |
|   } else if (endPoint.IsSetAndValid()) {
 | |
|     nextNodeOfRangeEnd = endPoint.GetChildAtOffset();
 | |
|   }
 | |
| 
 | |
|   for (preOrderIter.Last(); !preOrderIter.IsDone(); preOrderIter.Prev()) {
 | |
|     nsINode* node = preOrderIter.GetCurrentNode();
 | |
|     if (NS_WARN_IF(!node)) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if (node == nextNodeOfRangeEnd) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     auto* content = nsIContent::FromNode(node);
 | |
|     if (MOZ_UNLIKELY(!content)) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     // If the node is invisible (e.g., the node is or is in an invisible node or
 | |
|     // it's a white-space only text node around a block boundary), we should
 | |
|     // ignore it.
 | |
|     if (!content->GetPrimaryFrame()) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     if (auto* textNode = Text::FromNode(node)) {
 | |
|       nodePosition = {textNode, textNode == aSimpleRange.GetEndContainer()
 | |
|                                     ? aSimpleRange.EndOffset()
 | |
|                                     : textNode->TextDataLength()};
 | |
| 
 | |
|       // If the text node is empty or the last node of the range but the index
 | |
|       // is 0, we should store current position but continue looking for
 | |
|       // previous node (If there are no nodes before it, we should use current
 | |
|       // node position for returning its frame).
 | |
|       if (*nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets) ==
 | |
|           0) {
 | |
|         continue;
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if (ShouldBreakLineBefore(*content, mRootElement) ||
 | |
|         IsPaddingBR(*content)) {
 | |
|       nodePosition = {content, 0u};
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!nodePosition.IsSet()) {
 | |
|     return FrameAndNodeOffset();
 | |
|   }
 | |
| 
 | |
|   Result<nsIFrame*, nsresult> lastFrameOrError = GetFrameForTextRect(
 | |
|       nodePosition.Container(),
 | |
|       *nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets), true);
 | |
|   if (NS_WARN_IF(lastFrameOrError.isErr()) || !lastFrameOrError.inspect()) {
 | |
|     return FrameAndNodeOffset();
 | |
|   }
 | |
| 
 | |
|   // If the last frame is a text frame, we need to check if the range actually
 | |
|   // includes at least one character in the range.  Therefore, if it's not a
 | |
|   // text frame, we need to do nothing anymore.
 | |
|   if (!lastFrameOrError.inspect()->IsTextFrame()) {
 | |
|     return FrameAndNodeOffset(
 | |
|         lastFrameOrError.inspect(),
 | |
|         *nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets));
 | |
|   }
 | |
| 
 | |
|   int32_t start = lastFrameOrError.inspect()->GetOffsets().first;
 | |
| 
 | |
|   // If the start offset in the node is same as the computed offset in the
 | |
|   // node and it's not 0, the frame shouldn't be added to the text rect.  So,
 | |
|   // this should return previous text frame and its last offset if there is
 | |
|   // at least one text frame.
 | |
|   if (*nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets) &&
 | |
|       *nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets) ==
 | |
|           static_cast<uint32_t>(start)) {
 | |
|     const uint32_t newNodePositionOffset =
 | |
|         *nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets);
 | |
|     MOZ_ASSERT(newNodePositionOffset != 0);
 | |
|     nodePosition = {nodePosition.Container(), newNodePositionOffset - 1u};
 | |
|     lastFrameOrError = GetFrameForTextRect(
 | |
|         nodePosition.Container(),
 | |
|         *nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets),
 | |
|         true);
 | |
|     if (NS_WARN_IF(lastFrameOrError.isErr()) || !lastFrameOrError.inspect()) {
 | |
|       return FrameAndNodeOffset();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FrameAndNodeOffset(
 | |
|       lastFrameOrError.inspect(),
 | |
|       *nodePosition.Offset(RawNodePosition::OffsetFilter::kValidOffsets));
 | |
| }
 | |
| 
 | |
| ContentEventHandler::FrameRelativeRect
 | |
| ContentEventHandler::GetLineBreakerRectBefore(nsIFrame* aFrame) {
 | |
|   // Note that this method should be called only with an element's frame whose
 | |
|   // open tag causes a line break or moz-<br> for computing empty last line's
 | |
|   // rect.
 | |
|   MOZ_ASSERT(aFrame->GetContent());
 | |
|   MOZ_ASSERT(ShouldBreakLineBefore(*aFrame->GetContent(), mRootElement) ||
 | |
|              IsPaddingBR(*aFrame->GetContent()));
 | |
| 
 | |
|   nsIFrame* frameForFontMetrics = aFrame;
 | |
| 
 | |
|   // If it's not a <br> frame, this method computes the line breaker's rect
 | |
|   // outside the frame.  Therefore, we need to compute with parent frame's
 | |
|   // font metrics in such case.
 | |
|   if (!aFrame->IsBrFrame() && aFrame->GetParent()) {
 | |
|     frameForFontMetrics = aFrame->GetParent();
 | |
|   }
 | |
| 
 | |
|   // Note that <br> element's rect is decided with line-height but we need
 | |
|   // a rect only with font height.  Additionally, <br> frame's width and
 | |
|   // height are 0 in quirks mode if it's not an empty line.  So, we cannot
 | |
|   // use frame rect information even if it's a <br> frame.
 | |
| 
 | |
|   RefPtr<nsFontMetrics> fontMetrics =
 | |
|       nsLayoutUtils::GetInflatedFontMetricsForFrame(frameForFontMetrics);
 | |
|   if (NS_WARN_IF(!fontMetrics)) {
 | |
|     return FrameRelativeRect();
 | |
|   }
 | |
| 
 | |
|   const WritingMode kWritingMode = frameForFontMetrics->GetWritingMode();
 | |
| 
 | |
|   auto caretBlockAxisMetrics =
 | |
|       aFrame->GetCaretBlockAxisMetrics(kWritingMode, *fontMetrics);
 | |
|   nscoord inlineOffset = 0;
 | |
| 
 | |
|   // If aFrame isn't a <br> frame, caret should be at outside of it because
 | |
|   // the line break is before its open tag.  For example, case of
 | |
|   // |<div><p>some text</p></div>|, caret is before <p> element and in <div>
 | |
|   // element, the caret should be left of top-left corner of <p> element like:
 | |
|   //
 | |
|   // +-<div>-------------------  <div>'s border box
 | |
|   // | I +-<p>-----------------  <p>'s border box
 | |
|   // | I |
 | |
|   // | I |
 | |
|   // |   |
 | |
|   //   ^- caret
 | |
|   //
 | |
|   // However, this is a hack for unusual scenario.  This hack shouldn't be
 | |
|   // used as far as possible.
 | |
|   if (!aFrame->IsBrFrame()) {
 | |
|     if (kWritingMode.IsVertical() && !kWritingMode.IsLineInverted()) {
 | |
|       // above of top-right corner of aFrame.
 | |
|       caretBlockAxisMetrics.mOffset =
 | |
|           aFrame->GetRect().XMost() - caretBlockAxisMetrics.mExtent;
 | |
|     } else {
 | |
|       // above (For vertical) or left (For horizontal) of top-left corner of
 | |
|       // aFrame.
 | |
|       caretBlockAxisMetrics.mOffset = 0;
 | |
|     }
 | |
|     inlineOffset = -aFrame->PresContext()->AppUnitsPerDevPixel();
 | |
|   }
 | |
|   FrameRelativeRect result(aFrame);
 | |
|   if (kWritingMode.IsVertical()) {
 | |
|     result.mRect.x = caretBlockAxisMetrics.mOffset;
 | |
|     result.mRect.y = inlineOffset;
 | |
|     result.mRect.width = caretBlockAxisMetrics.mExtent;
 | |
|   } else {
 | |
|     result.mRect.x = inlineOffset;
 | |
|     result.mRect.y = caretBlockAxisMetrics.mOffset;
 | |
|     result.mRect.height = caretBlockAxisMetrics.mExtent;
 | |
|   }
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| ContentEventHandler::FrameRelativeRect
 | |
| ContentEventHandler::GuessLineBreakerRectAfter(const Text& aTextNode) {
 | |
|   FrameRelativeRect result;
 | |
|   const int32_t length = static_cast<int32_t>(aTextNode.TextLength());
 | |
|   if (NS_WARN_IF(length < 0)) {
 | |
|     return result;
 | |
|   }
 | |
|   // Get the last nsTextFrame which is caused by aTextNode.  Note that
 | |
|   // a text node can cause multiple text frames, e.g., the text is too long
 | |
|   // and wrapped by its parent block or the text has line breakers and its
 | |
|   // white-space property respects the line breakers (e.g., |pre|).
 | |
|   Result<nsIFrame*, nsresult> lastTextFrameOrError =
 | |
|       GetFrameForTextRect(&aTextNode, length, true);
 | |
|   if (NS_WARN_IF(lastTextFrameOrError.isErr()) ||
 | |
|       !lastTextFrameOrError.inspect()) {
 | |
|     return result;
 | |
|   }
 | |
|   const nsRect kLastTextFrameRect = lastTextFrameOrError.inspect()->GetRect();
 | |
|   if (lastTextFrameOrError.inspect()->GetWritingMode().IsVertical()) {
 | |
|     // Below of the last text frame.
 | |
|     result.mRect.SetRect(0, kLastTextFrameRect.height, kLastTextFrameRect.width,
 | |
|                          0);
 | |
|   } else {
 | |
|     // Right of the last text frame (not bidi-aware).
 | |
|     result.mRect.SetRect(kLastTextFrameRect.width, 0, 0,
 | |
|                          kLastTextFrameRect.height);
 | |
|   }
 | |
|   result.mBaseFrame = lastTextFrameOrError.unwrap();
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| ContentEventHandler::FrameRelativeRect
 | |
| ContentEventHandler::GuessFirstCaretRectIn(nsIFrame* aFrame) {
 | |
|   const WritingMode kWritingMode = aFrame->GetWritingMode();
 | |
|   nsPresContext* presContext = aFrame->PresContext();
 | |
| 
 | |
|   // Computes the font height, but if it's not available, we should use
 | |
|   // default font size of Firefox.  The default font size in default settings
 | |
|   // is 16px.
 | |
|   RefPtr<nsFontMetrics> fontMetrics =
 | |
|       nsLayoutUtils::GetInflatedFontMetricsForFrame(aFrame);
 | |
|   const nscoord kMaxHeight = fontMetrics
 | |
|                                  ? fontMetrics->MaxHeight()
 | |
|                                  : 16 * presContext->AppUnitsPerDevPixel();
 | |
| 
 | |
|   nsRect caretRect;
 | |
|   const nsRect kContentRect = aFrame->GetContentRect() - aFrame->GetPosition();
 | |
|   caretRect.y = kContentRect.y;
 | |
|   if (!kWritingMode.IsVertical()) {
 | |
|     if (kWritingMode.IsBidiLTR()) {
 | |
|       caretRect.x = kContentRect.x;
 | |
|     } else {
 | |
|       // Move 1px left for the space of caret itself.
 | |
|       const nscoord kOnePixel = presContext->AppUnitsPerDevPixel();
 | |
|       caretRect.x = kContentRect.XMost() - kOnePixel;
 | |
|     }
 | |
|     caretRect.height = kMaxHeight;
 | |
|     // However, don't add kOnePixel here because it may cause 2px width at
 | |
|     // aligning the edge to device pixels.
 | |
|     caretRect.width = 1;
 | |
|   } else {
 | |
|     if (kWritingMode.IsVerticalLR()) {
 | |
|       caretRect.x = kContentRect.x;
 | |
|     } else {
 | |
|       caretRect.x = kContentRect.XMost() - kMaxHeight;
 | |
|     }
 | |
|     caretRect.width = kMaxHeight;
 | |
|     // Don't add app units for a device pixel because it may cause 2px height
 | |
|     // at aligning the edge to device pixels.
 | |
|     caretRect.height = 1;
 | |
|   }
 | |
|   return FrameRelativeRect(caretRect, aFrame);
 | |
| }
 | |
| 
 | |
| //  static
 | |
| LayoutDeviceIntRect ContentEventHandler::GetCaretRectBefore(
 | |
|     const LayoutDeviceIntRect& aCharRect, const WritingMode& aWritingMode) {
 | |
|   LayoutDeviceIntRect caretRectBefore(aCharRect);
 | |
|   if (aWritingMode.IsVertical()) {
 | |
|     caretRectBefore.height = 1;
 | |
|   } else {
 | |
|     // TODO: Make here bidi-aware.
 | |
|     caretRectBefore.width = 1;
 | |
|   }
 | |
|   return caretRectBefore;
 | |
| }
 | |
| 
 | |
| //  static
 | |
| nsRect ContentEventHandler::GetCaretRectBefore(
 | |
|     const nsRect& aCharRect, const WritingMode& aWritingMode) {
 | |
|   nsRect caretRectBefore(aCharRect);
 | |
|   if (aWritingMode.IsVertical()) {
 | |
|     // For making the height 1 device pixel after aligning the rect edges to
 | |
|     // device pixels, don't add one device pixel in app units here.
 | |
|     caretRectBefore.height = 1;
 | |
|   } else {
 | |
|     // TODO: Make here bidi-aware.
 | |
|     // For making the width 1 device pixel after aligning the rect edges to
 | |
|     // device pixels, don't add one device pixel in app units here.
 | |
|     caretRectBefore.width = 1;
 | |
|   }
 | |
|   return caretRectBefore;
 | |
| }
 | |
| 
 | |
| //  static
 | |
| LayoutDeviceIntRect ContentEventHandler::GetCaretRectAfter(
 | |
|     const LayoutDeviceIntRect& aCharRect, const WritingMode& aWritingMode) {
 | |
|   LayoutDeviceIntRect caretRectAfter(aCharRect);
 | |
|   if (aWritingMode.IsVertical()) {
 | |
|     caretRectAfter.y = aCharRect.YMost() + 1;
 | |
|     caretRectAfter.height = 1;
 | |
|   } else {
 | |
|     // TODO: Make here bidi-aware.
 | |
|     caretRectAfter.x = aCharRect.XMost() + 1;
 | |
|     caretRectAfter.width = 1;
 | |
|   }
 | |
|   return caretRectAfter;
 | |
| }
 | |
| 
 | |
| //  static
 | |
| nsRect ContentEventHandler::GetCaretRectAfter(nsPresContext& aPresContext,
 | |
|                                               const nsRect& aCharRect,
 | |
|                                               const WritingMode& aWritingMode) {
 | |
|   nsRect caretRectAfter(aCharRect);
 | |
|   const nscoord onePixel = aPresContext.AppUnitsPerDevPixel();
 | |
|   if (aWritingMode.IsVertical()) {
 | |
|     caretRectAfter.y = aCharRect.YMost() + onePixel;
 | |
|     // For making the height 1 device pixel after aligning the rect edges to
 | |
|     // device pixels, don't add one device pixel in app units here.
 | |
|     caretRectAfter.height = 1;
 | |
|   } else {
 | |
|     // TODO: Make here bidi-aware.
 | |
|     caretRectAfter.x = aCharRect.XMost() + onePixel;
 | |
|     // For making the width 1 device pixel after aligning the rect edges to
 | |
|     // device pixels, don't add one device pixel in app units here.
 | |
|     caretRectAfter.width = 1;
 | |
|   }
 | |
|   return caretRectAfter;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQueryTextRectArray(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   nsresult rv = Init(aEvent);
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->mReply->mOffsetAndData.isNothing());
 | |
| 
 | |
|   LineBreakType lineBreakType = GetLineBreakType(aEvent);
 | |
|   const uint32_t kBRLength = GetBRLength(lineBreakType);
 | |
| 
 | |
|   WritingMode lastVisibleFrameWritingMode;
 | |
|   LayoutDeviceIntRect rect;
 | |
|   uint32_t offset = aEvent->mInput.mOffset;
 | |
|   const uint32_t kEndOffset = aEvent->mInput.EndOffset();
 | |
|   bool wasLineBreaker = false;
 | |
|   // lastCharRect stores the last charRect value (see below for the detail of
 | |
|   // charRect).
 | |
|   nsRect lastCharRect;
 | |
|   // lastFrame is base frame of lastCharRect.
 | |
|   // TODO: We should look for this if the first text is not visible.  However,
 | |
|   //       users cannot put caret invisible text and users cannot type in it
 | |
|   //       at least only with user's operations.  Therefore, we don't need to
 | |
|   //       fix this immediately.
 | |
|   nsIFrame* lastFrame = nullptr;
 | |
|   nsAutoString flattenedAllText;
 | |
|   flattenedAllText.SetIsVoid(true);
 | |
|   while (offset < kEndOffset) {
 | |
|     Result<DOMRangeAndAdjustedOffsetInFlattenedText, nsresult>
 | |
|         domRangeAndAdjustedOffsetOrError =
 | |
|             ConvertFlatTextOffsetToDOMRange(offset, 1, lineBreakType, true);
 | |
|     if (MOZ_UNLIKELY(domRangeAndAdjustedOffsetOrError.isErr())) {
 | |
|       NS_WARNING(
 | |
|           "ContentEventHandler::ConvertFlatTextOffsetToDOMRangeBase() failed");
 | |
|       return domRangeAndAdjustedOffsetOrError.unwrapErr();
 | |
|     }
 | |
|     const DOMRangeAndAdjustedOffsetInFlattenedText domRangeAndAdjustedOffset =
 | |
|         domRangeAndAdjustedOffsetOrError.unwrap();
 | |
| 
 | |
|     // TODO: When we crossed parent block boundary now, we should fill pending
 | |
|     //       character rects with caret rect after the last visible character
 | |
|     //       rect.
 | |
| 
 | |
|     // If the range is collapsed, offset has already reached the end of the
 | |
|     // contents.
 | |
|     if (domRangeAndAdjustedOffset.mRange.Collapsed()) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     // Get the first frame which causes some text after the offset.
 | |
|     FrameAndNodeOffset firstFrame =
 | |
|         GetFirstFrameInRangeForTextRect(domRangeAndAdjustedOffset.mRange);
 | |
| 
 | |
|     // If GetFirstFrameInRangeForTextRect() does not return valid frame, that
 | |
|     // means that the offset reached the end of contents or there is no visible
 | |
|     // frame in the range generating flattened text.
 | |
|     if (!firstFrame.IsValid()) {
 | |
|       if (flattenedAllText.IsVoid()) {
 | |
|         flattenedAllText.SetIsVoid(false);
 | |
|         if (NS_WARN_IF(NS_FAILED(GenerateFlatTextContent(
 | |
|                 mRootElement, flattenedAllText, lineBreakType)))) {
 | |
|           NS_WARNING("ContentEventHandler::GenerateFlatTextContent() failed");
 | |
|           return NS_ERROR_FAILURE;
 | |
|         }
 | |
|       }
 | |
|       // If we've reached end of the root, append caret rect at the end of
 | |
|       // the root later.
 | |
|       if (offset >= flattenedAllText.Length()) {
 | |
|         break;
 | |
|       }
 | |
|       // Otherwise, we're in an invisible node. If the node is followed by a
 | |
|       // block boundary causing a line break, we can use the boundary.
 | |
|       // Otherwise, if the node follows a block boundary of a parent block, we
 | |
|       // can use caret rect at previous visible frame causing flattened text.
 | |
|       const uint32_t remainingLengthInCurrentRange = [&]() {
 | |
|         if (domRangeAndAdjustedOffset.mLastTextNode) {
 | |
|           if (domRangeAndAdjustedOffset.RangeStartsFromLastTextNode()) {
 | |
|             if (!domRangeAndAdjustedOffset.RangeStartsFromEndOfContainer()) {
 | |
|               return domRangeAndAdjustedOffset.mLastTextNode->TextDataLength() -
 | |
|                      domRangeAndAdjustedOffset.mRange.StartOffset();
 | |
|             }
 | |
|             return 0u;
 | |
|           }
 | |
|           // Must be there are not nodes which may cause generating text.
 | |
|           // Therefore, we can skip all nodes before the last found text node
 | |
|           // and all text in the last text node.
 | |
|           return domRangeAndAdjustedOffset.mLastTextNode->TextDataLength();
 | |
|         }
 | |
|         if (domRangeAndAdjustedOffset.RangeStartsFromContent() &&
 | |
|             ShouldBreakLineBefore(
 | |
|                 *domRangeAndAdjustedOffset.mRange.GetStartContainer()
 | |
|                      ->AsContent(),
 | |
|                 mRootElement)) {
 | |
|           if (kBRLength != 1u && offset - aEvent->mInput.mOffset < kBRLength) {
 | |
|             // Don't return kBRLength if start position is less than the length
 | |
|             // of a line-break because the offset may be between CRLF on
 | |
|             // Windows.  In the case, we will be again here and gets same
 | |
|             // result and we need to pay the penalty only once.  Therefore, we
 | |
|             // can keep going without complicated check.
 | |
|             return 1u;
 | |
|           }
 | |
|           return kBRLength;
 | |
|         }
 | |
|         return 0u;
 | |
|       }();
 | |
|       offset += std::max(1u, remainingLengthInCurrentRange);
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     nsIContent* firstContent = firstFrame.mFrame->GetContent();
 | |
|     if (NS_WARN_IF(!firstContent)) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
| 
 | |
|     bool startsBetweenLineBreaker = false;
 | |
|     nsAutoString chars;
 | |
|     lastVisibleFrameWritingMode = firstFrame->GetWritingMode();
 | |
| 
 | |
|     nsIFrame* baseFrame = firstFrame;
 | |
|     // charRect should have each character rect or line breaker rect relative
 | |
|     // to the base frame.
 | |
|     AutoTArray<nsRect, 16> charRects;
 | |
| 
 | |
|     // If the first frame is a text frame, the result should be computed with
 | |
|     // the frame's API.
 | |
|     if (firstFrame->IsTextFrame()) {
 | |
|       rv = firstFrame->GetCharacterRectsInRange(firstFrame.mOffsetInNode,
 | |
|                                                 kEndOffset - offset, charRects);
 | |
|       if (NS_WARN_IF(NS_FAILED(rv)) || NS_WARN_IF(charRects.IsEmpty())) {
 | |
|         return rv;
 | |
|       }
 | |
|       // Assign the characters whose rects are computed by the call of
 | |
|       // nsTextFrame::GetCharacterRectsInRange().
 | |
|       AppendSubString(chars, *firstContent->AsText(), firstFrame.mOffsetInNode,
 | |
|                       charRects.Length());
 | |
|       if (NS_WARN_IF(chars.Length() != charRects.Length())) {
 | |
|         return NS_ERROR_UNEXPECTED;
 | |
|       }
 | |
|       if (kBRLength > 1 && chars[0] == '\n' &&
 | |
|           offset == aEvent->mInput.mOffset && offset) {
 | |
|         // If start of range starting from previous offset of query range is
 | |
|         // same as the start of query range, the query range starts from
 | |
|         // between a line breaker (i.e., the range starts between "\r" and
 | |
|         // "\n").
 | |
|         Result<UnsafeDOMRangeAndAdjustedOffsetInFlattenedText, nsresult>
 | |
|             domRangeAndAdjustedOffsetOrError =
 | |
|                 ConvertFlatTextOffsetToUnsafeDOMRange(
 | |
|                     aEvent->mInput.mOffset - 1, 1, lineBreakType, true);
 | |
|         if (MOZ_UNLIKELY(domRangeAndAdjustedOffsetOrError.isErr())) {
 | |
|           NS_WARNING(
 | |
|               "ContentEventHandler::ConvertFlatTextOffsetToDOMRangeBase() "
 | |
|               "failed");
 | |
|           return domRangeAndAdjustedOffsetOrError.unwrapErr();
 | |
|         }
 | |
|         const UnsafeDOMRangeAndAdjustedOffsetInFlattenedText
 | |
|             domRangeAndAdjustedOffsetOfPreviousChar =
 | |
|                 domRangeAndAdjustedOffsetOrError.unwrap();
 | |
|         startsBetweenLineBreaker =
 | |
|             domRangeAndAdjustedOffset.mRange.GetStartContainer() ==
 | |
|                 domRangeAndAdjustedOffsetOfPreviousChar.mRange
 | |
|                     .GetStartContainer() &&
 | |
|             domRangeAndAdjustedOffset.mRange.StartOffset() ==
 | |
|                 domRangeAndAdjustedOffsetOfPreviousChar.mRange.StartOffset();
 | |
|       }
 | |
|     }
 | |
|     // Other contents should cause a line breaker rect before it.
 | |
|     // Note that moz-<br> element does not cause any text, however,
 | |
|     // it represents empty line at the last of current block.  Therefore,
 | |
|     // we need to compute its rect too.
 | |
|     else if (ShouldBreakLineBefore(*firstContent, mRootElement) ||
 | |
|              IsPaddingBR(*firstContent)) {
 | |
|       nsRect brRect;
 | |
|       // If the frame is not a <br> frame, we need to compute the caret rect
 | |
|       // with last character's rect before firstContent if there is.
 | |
|       // For example, if caret is after "c" of |<p>abc</p><p>def</p>|, IME may
 | |
|       // query a line breaker's rect after "c".  Then, if we compute it only
 | |
|       // with the 2nd <p>'s block frame, the result will be:
 | |
|       //  +-<p>--------------------------------+
 | |
|       //  |abc                                 |
 | |
|       //  +------------------------------------+
 | |
|       //
 | |
|       // I+-<p>--------------------------------+
 | |
|       //  |def                                 |
 | |
|       //  +------------------------------------+
 | |
|       // However, users expect popup windows of IME should be positioned at
 | |
|       // right-bottom of "c" like this:
 | |
|       //  +-<p>--------------------------------+
 | |
|       //  |abcI                                |
 | |
|       //  +------------------------------------+
 | |
|       //
 | |
|       //  +-<p>--------------------------------+
 | |
|       //  |def                                 |
 | |
|       //  +------------------------------------+
 | |
|       // Therefore, if the first frame isn't a <br> frame and there is a text
 | |
|       // node before the first node in the queried range, we should compute the
 | |
|       // first rect with the previous character's rect.
 | |
|       // If we already compute a character's rect in the queried range, we can
 | |
|       // compute it with the cached last character's rect.  (However, don't
 | |
|       // use this path if it's a <br> frame because trusting <br> frame's rect
 | |
|       // is better than guessing the rect from the previous character.)
 | |
|       if (!firstFrame->IsBrFrame() && !aEvent->mReply->mRectArray.IsEmpty()) {
 | |
|         baseFrame = lastFrame;
 | |
|         brRect = lastCharRect;
 | |
|         if (!wasLineBreaker) {
 | |
|           brRect = GetCaretRectAfter(*baseFrame->PresContext(), brRect,
 | |
|                                      lastVisibleFrameWritingMode);
 | |
|         }
 | |
|       }
 | |
|       // If it's not a <br> frame and it's the first character rect at the
 | |
|       // queried range, we need the previous character rect of the start of
 | |
|       // the queried range if there is a visible text node.
 | |
|       else if (!firstFrame->IsBrFrame() &&
 | |
|                domRangeAndAdjustedOffset.mLastTextNode &&
 | |
|                domRangeAndAdjustedOffset.mLastTextNode->GetPrimaryFrame()) {
 | |
|         FrameRelativeRect brRectRelativeToLastTextFrame =
 | |
|             GuessLineBreakerRectAfter(*domRangeAndAdjustedOffset.mLastTextNode);
 | |
|         if (NS_WARN_IF(!brRectRelativeToLastTextFrame.IsValid())) {
 | |
|           return NS_ERROR_FAILURE;
 | |
|         }
 | |
|         // Look for the last text frame for the last text node.
 | |
|         nsIFrame* primaryFrame =
 | |
|             domRangeAndAdjustedOffset.mLastTextNode->GetPrimaryFrame();
 | |
|         if (NS_WARN_IF(!primaryFrame)) {
 | |
|           return NS_ERROR_FAILURE;
 | |
|         }
 | |
|         baseFrame = primaryFrame->LastContinuation();
 | |
|         if (NS_WARN_IF(!baseFrame)) {
 | |
|           return NS_ERROR_FAILURE;
 | |
|         }
 | |
|         brRect = brRectRelativeToLastTextFrame.RectRelativeTo(baseFrame);
 | |
|       }
 | |
|       // Otherwise, we need to compute the line breaker's rect only with the
 | |
|       // first frame's rect.  But this may be unexpected.  For example,
 | |
|       // |<div contenteditable>[<p>]abc</p></div>|.  In this case, caret is
 | |
|       // before "a", therefore, users expect the rect left of "a".  However,
 | |
|       // we don't have enough information about the next character here and
 | |
|       // this isn't usual case (e.g., IME typically tries to query the rect
 | |
|       // of "a" or caret rect for computing its popup position).  Therefore,
 | |
|       // we shouldn't do more complicated hack here unless we'll get some bug
 | |
|       // reports actually.
 | |
|       else {
 | |
|         FrameRelativeRect relativeBRRect = GetLineBreakerRectBefore(firstFrame);
 | |
|         brRect = relativeBRRect.RectRelativeTo(firstFrame);
 | |
|       }
 | |
|       charRects.AppendElement(brRect);
 | |
|       chars.AssignLiteral("\n");
 | |
|       if (kBRLength > 1 && offset == aEvent->mInput.mOffset && offset) {
 | |
|         // If the first frame for the previous offset of the query range and
 | |
|         // the first frame for the start of query range are same, that means
 | |
|         // the start offset is between the first line breaker (i.e., the range
 | |
|         // starts between "\r" and "\n").
 | |
|         Result<UnsafeDOMRangeAndAdjustedOffsetInFlattenedText, nsresult>
 | |
|             domRangeAndAdjustedOffsetOrError =
 | |
|                 ConvertFlatTextOffsetToUnsafeDOMRange(
 | |
|                     aEvent->mInput.mOffset - 1, 1, lineBreakType, true);
 | |
|         if (MOZ_UNLIKELY(domRangeAndAdjustedOffsetOrError.isErr())) {
 | |
|           NS_WARNING(
 | |
|               "ContentEventHandler::ConvertFlatTextOffsetToDOMRangeBase() "
 | |
|               "failed");
 | |
|           return NS_ERROR_UNEXPECTED;
 | |
|         }
 | |
|         const UnsafeDOMRangeAndAdjustedOffsetInFlattenedText
 | |
|             domRangeAndAdjustedOffset =
 | |
|                 domRangeAndAdjustedOffsetOrError.unwrap();
 | |
|         FrameAndNodeOffset frameForPrevious =
 | |
|             GetFirstFrameInRangeForTextRect(domRangeAndAdjustedOffset.mRange);
 | |
|         startsBetweenLineBreaker = frameForPrevious.mFrame == firstFrame.mFrame;
 | |
|       }
 | |
|     } else {
 | |
|       NS_WARNING(
 | |
|           "The frame is neither a text frame nor a frame whose content "
 | |
|           "causes a line break");
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
| 
 | |
|     for (size_t i = 0; i < charRects.Length() && offset < kEndOffset; i++) {
 | |
|       nsRect charRect = charRects[i];
 | |
|       // Store lastCharRect before applying CSS transform because it may be
 | |
|       // used for computing a line breaker rect.  Then, the computed line
 | |
|       // breaker rect will be applied CSS transform again.  Therefore,
 | |
|       // the value of lastCharRect should be raw rect value relative to the
 | |
|       // base frame.
 | |
|       lastCharRect = charRect;
 | |
|       lastFrame = baseFrame;
 | |
|       rv = ConvertToRootRelativeOffset(baseFrame, charRect);
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return rv;
 | |
|       }
 | |
| 
 | |
|       nsPresContext* presContext = baseFrame->PresContext();
 | |
|       rect = LayoutDeviceIntRect::FromAppUnitsToOutside(
 | |
|           charRect, presContext->AppUnitsPerDevPixel());
 | |
|       if (nsPresContext* rootContext =
 | |
|               presContext->GetInProcessRootContentDocumentPresContext()) {
 | |
|         rect = RoundedOut(ViewportUtils::DocumentRelativeLayoutToVisual(
 | |
|             rect, rootContext->PresShell()));
 | |
|       }
 | |
|       // Returning empty rect may cause native IME confused, let's make sure to
 | |
|       // return non-empty rect.
 | |
|       EnsureNonEmptyRect(rect);
 | |
| 
 | |
|       // If we found some invisible characters followed by current visible
 | |
|       // character, make their rects same as caret rect before the first visible
 | |
|       // character because IME may want to put their UI next to the rect of the
 | |
|       // invisible character for next input.
 | |
|       // Note that chars do not contain the invisible characters.
 | |
|       if (i == 0u && MOZ_LIKELY(offset > aEvent->mInput.mOffset)) {
 | |
|         const uint32_t offsetInRange =
 | |
|             offset - CheckedInt<uint32_t>(aEvent->mInput.mOffset).value();
 | |
|         if (offsetInRange > aEvent->mReply->mRectArray.Length()) {
 | |
|           LayoutDeviceIntRect caretRectBefore =
 | |
|               GetCaretRectBefore(rect, lastVisibleFrameWritingMode);
 | |
|           for ([[maybe_unused]] uint32_t index : IntegerRange<uint32_t>(
 | |
|                    offsetInRange - aEvent->mReply->mRectArray.Length())) {
 | |
|             aEvent->mReply->mRectArray.AppendElement(caretRectBefore);
 | |
|           }
 | |
|           MOZ_ASSERT(aEvent->mReply->mRectArray.Length() == offsetInRange);
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       aEvent->mReply->mRectArray.AppendElement(rect);
 | |
|       offset++;
 | |
| 
 | |
|       // If it's not a line breaker or the line breaker length is same as
 | |
|       // XP line breaker's, we need to do nothing for current character.
 | |
|       wasLineBreaker = chars[i] == '\n';
 | |
|       if (!wasLineBreaker || kBRLength == 1) {
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       MOZ_ASSERT(kBRLength == 2);
 | |
| 
 | |
|       // If it's already reached the end of query range, we don't need to do
 | |
|       // anymore.
 | |
|       if (offset == kEndOffset) {
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       // If the query range starts from between a line breaker, i.e., it starts
 | |
|       // between "\r" and "\n", the appended rect was for the "\n".  Therefore,
 | |
|       // we don't need to append same rect anymore for current "\r\n".
 | |
|       if (startsBetweenLineBreaker) {
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       // The appended rect was for "\r" of "\r\n".  Therefore, we need to
 | |
|       // append same rect for "\n" too because querying rect of "\r" and "\n"
 | |
|       // should return same rect.  E.g., IME may query previous character's
 | |
|       // rect of first character of a line.
 | |
|       aEvent->mReply->mRectArray.AppendElement(rect);
 | |
|       offset++;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // If we've not handled some invisible character rects, fill them as caret
 | |
|   // rect after the last visible character.
 | |
|   if (!aEvent->mReply->mRectArray.IsEmpty()) {
 | |
|     const uint32_t offsetInRange =
 | |
|         offset - CheckedInt<uint32_t>(aEvent->mInput.mOffset).value();
 | |
|     if (offsetInRange > aEvent->mReply->mRectArray.Length()) {
 | |
|       LayoutDeviceIntRect caretRectAfter =
 | |
|           GetCaretRectAfter(aEvent->mReply->mRectArray.LastElement(),
 | |
|                             lastVisibleFrameWritingMode);
 | |
|       for ([[maybe_unused]] uint32_t index : IntegerRange<uint32_t>(
 | |
|                offsetInRange - aEvent->mReply->mRectArray.Length())) {
 | |
|         aEvent->mReply->mRectArray.AppendElement(caretRectAfter);
 | |
|       }
 | |
|       MOZ_ASSERT(aEvent->mReply->mRectArray.Length() == offsetInRange);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // If the query range is longer than actual content length, we should append
 | |
|   // caret rect at the end of the content as the last character rect because
 | |
|   // native IME may want to query character rect at the end of contents for
 | |
|   // deciding the position of a popup window (e.g., suggest window for next
 | |
|   // word).  Note that when this method hasn't appended character rects, it
 | |
|   // means that the offset is too large or the query range is collapsed.
 | |
|   if (offset < kEndOffset || aEvent->mReply->mRectArray.IsEmpty()) {
 | |
|     // If we've already retrieved some character rects before current offset,
 | |
|     // we can guess the last rect from the last character's rect unless it's a
 | |
|     // line breaker.  (If it's a line breaker, the caret rect is in next line.)
 | |
|     if (!aEvent->mReply->mRectArray.IsEmpty() && !wasLineBreaker) {
 | |
|       rect = GetCaretRectAfter(aEvent->mReply->mRectArray.LastElement(),
 | |
|                                lastVisibleFrameWritingMode);
 | |
|       aEvent->mReply->mRectArray.AppendElement(rect);
 | |
|     } else {
 | |
|       // Note that don't use eQueryCaretRect here because if caret is at the
 | |
|       // end of the content, it returns actual caret rect instead of computing
 | |
|       // the rect itself.  It means that the result depends on caret position.
 | |
|       // So, we shouldn't use it for consistency result in automated tests.
 | |
|       WidgetQueryContentEvent queryTextRectEvent(eQueryTextRect, *aEvent);
 | |
|       WidgetQueryContentEvent::Options options(*aEvent);
 | |
|       queryTextRectEvent.InitForQueryTextRect(offset, 1, options);
 | |
|       if (NS_WARN_IF(NS_FAILED(OnQueryTextRect(&queryTextRectEvent))) ||
 | |
|           NS_WARN_IF(queryTextRectEvent.Failed())) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
|       if (queryTextRectEvent.mReply->mWritingMode.IsVertical()) {
 | |
|         queryTextRectEvent.mReply->mRect.height = 1;
 | |
|       } else {
 | |
|         queryTextRectEvent.mReply->mRect.width = 1;
 | |
|       }
 | |
|       aEvent->mReply->mRectArray.AppendElement(
 | |
|           queryTextRectEvent.mReply->mRect);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQueryTextRect(WidgetQueryContentEvent* aEvent) {
 | |
|   // If mLength is 0 (this may be caused by bug of native IME), we should
 | |
|   // redirect this event to OnQueryCaretRect().
 | |
|   if (!aEvent->mInput.mLength) {
 | |
|     return OnQueryCaretRect(aEvent);
 | |
|   }
 | |
| 
 | |
|   nsresult rv = Init(aEvent);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->mReply->mOffsetAndData.isNothing());
 | |
| 
 | |
|   LineBreakType lineBreakType = GetLineBreakType(aEvent);
 | |
|   Result<DOMRangeAndAdjustedOffsetInFlattenedText, nsresult>
 | |
|       domRangeAndAdjustedOffsetOrError = ConvertFlatTextOffsetToDOMRange(
 | |
|           aEvent->mInput.mOffset, aEvent->mInput.mLength, lineBreakType, true);
 | |
|   if (MOZ_UNLIKELY(domRangeAndAdjustedOffsetOrError.isErr())) {
 | |
|     NS_WARNING("ContentEventHandler::ConvertFlatTextOffsetToDOMRange() failed");
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
|   DOMRangeAndAdjustedOffsetInFlattenedText domRangeAndAdjustedOffset =
 | |
|       domRangeAndAdjustedOffsetOrError.unwrap();
 | |
|   nsString string;
 | |
|   if (NS_WARN_IF(NS_FAILED(GenerateFlatTextContent(
 | |
|           domRangeAndAdjustedOffset.mRange, string, lineBreakType)))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
|   aEvent->mReply->mOffsetAndData.emplace(
 | |
|       domRangeAndAdjustedOffset.mAdjustedOffset, string,
 | |
|       OffsetAndDataFor::EditorString);
 | |
| 
 | |
|   // used to iterate over all contents and their frames
 | |
|   PostContentIterator postOrderIter;
 | |
|   rv = postOrderIter.Init(domRangeAndAdjustedOffset.mRange.Start().AsRaw(),
 | |
|                           domRangeAndAdjustedOffset.mRange.End().AsRaw());
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   // Get the first frame which causes some text after the offset.
 | |
|   FrameAndNodeOffset firstFrame =
 | |
|       GetFirstFrameInRangeForTextRect(domRangeAndAdjustedOffset.mRange);
 | |
| 
 | |
|   // If GetFirstFrameInRangeForTextRect() does not return valid frame, that
 | |
|   // means that there are no visible frames having text or the offset reached
 | |
|   // the end of contents.
 | |
|   if (!firstFrame.IsValid()) {
 | |
|     nsAutoString allText;
 | |
|     rv = GenerateFlatTextContent(mRootElement, allText, lineBreakType);
 | |
|     // If the offset doesn't reach the end of contents but there is no frames
 | |
|     // for the node, that means that current offset's node is hidden by CSS or
 | |
|     // something.  Ideally, we should handle it with the last visible text
 | |
|     // node's last character's rect, but it's not usual cases in actual web
 | |
|     // services.  Therefore, currently, we should make this case fail.
 | |
|     if (NS_WARN_IF(NS_FAILED(rv)) ||
 | |
|         static_cast<uint32_t>(aEvent->mInput.mOffset) < allText.Length()) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
| 
 | |
|     // Look for the last frame which should be included text rects.
 | |
|     rv = domRangeAndAdjustedOffset.mRange.SelectNodeContents(mRootElement);
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return NS_ERROR_UNEXPECTED;
 | |
|     }
 | |
|     nsRect rect;
 | |
|     FrameAndNodeOffset lastFrame =
 | |
|         GetLastFrameInRangeForTextRect(domRangeAndAdjustedOffset.mRange);
 | |
|     // If there is at least one frame which can be used for computing a rect
 | |
|     // for a character or a line breaker, we should use it for guessing the
 | |
|     // caret rect at the end of the contents.
 | |
|     nsPresContext* presContext;
 | |
|     if (lastFrame) {
 | |
|       presContext = lastFrame->PresContext();
 | |
|       if (NS_WARN_IF(!lastFrame->GetContent())) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
|       FrameRelativeRect relativeRect;
 | |
|       // If there is a <br> frame at the end, it represents an empty line at
 | |
|       // the end with moz-<br> or content <br> in a block level element.
 | |
|       if (lastFrame->IsBrFrame()) {
 | |
|         relativeRect = GetLineBreakerRectBefore(lastFrame);
 | |
|       }
 | |
|       // If there is a text frame at the end, use its information.
 | |
|       else if (lastFrame->IsTextFrame()) {
 | |
|         const Text* textNode = Text::FromNode(lastFrame->GetContent());
 | |
|         MOZ_ASSERT(textNode);
 | |
|         if (textNode) {
 | |
|           relativeRect = GuessLineBreakerRectAfter(*textNode);
 | |
|         }
 | |
|       }
 | |
|       // If there is an empty frame which is neither a text frame nor a <br>
 | |
|       // frame at the end, guess caret rect in it.
 | |
|       else {
 | |
|         relativeRect = GuessFirstCaretRectIn(lastFrame);
 | |
|       }
 | |
|       if (NS_WARN_IF(!relativeRect.IsValid())) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
|       rect = relativeRect.RectRelativeTo(lastFrame);
 | |
|       rv = ConvertToRootRelativeOffset(lastFrame, rect);
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return rv;
 | |
|       }
 | |
|       aEvent->mReply->mWritingMode = lastFrame->GetWritingMode();
 | |
|     }
 | |
|     // Otherwise, if there are no contents in mRootElement, guess caret rect in
 | |
|     // its frame (with its font height and content box).
 | |
|     else {
 | |
|       nsIFrame* rootContentFrame = mRootElement->GetPrimaryFrame();
 | |
|       if (NS_WARN_IF(!rootContentFrame)) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
|       presContext = rootContentFrame->PresContext();
 | |
|       FrameRelativeRect relativeRect = GuessFirstCaretRectIn(rootContentFrame);
 | |
|       if (NS_WARN_IF(!relativeRect.IsValid())) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
|       rect = relativeRect.RectRelativeTo(rootContentFrame);
 | |
|       rv = ConvertToRootRelativeOffset(rootContentFrame, rect);
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return rv;
 | |
|       }
 | |
|       aEvent->mReply->mWritingMode = rootContentFrame->GetWritingMode();
 | |
|     }
 | |
|     aEvent->mReply->mRect = LayoutDeviceIntRect::FromAppUnitsToOutside(
 | |
|         rect, presContext->AppUnitsPerDevPixel());
 | |
|     if (nsPresContext* rootContext =
 | |
|             presContext->GetInProcessRootContentDocumentPresContext()) {
 | |
|       aEvent->mReply->mRect =
 | |
|           RoundedOut(ViewportUtils::DocumentRelativeLayoutToVisual(
 | |
|               aEvent->mReply->mRect, rootContext->PresShell()));
 | |
|     }
 | |
|     EnsureNonEmptyRect(aEvent->mReply->mRect);
 | |
| 
 | |
|     MOZ_ASSERT(aEvent->Succeeded());
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   nsRect rect, frameRect;
 | |
|   nsPoint ptOffset;
 | |
| 
 | |
|   // If the first frame is a text frame, the result should be computed with
 | |
|   // the frame's rect but not including the rect before start point of the
 | |
|   // queried range.
 | |
|   if (firstFrame->IsTextFrame()) {
 | |
|     rect.SetRect(nsPoint(0, 0), firstFrame->GetRect().Size());
 | |
|     rv = ConvertToRootRelativeOffset(firstFrame, rect);
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return rv;
 | |
|     }
 | |
|     frameRect = rect;
 | |
|     // Exclude the rect before start point of the queried range.
 | |
|     firstFrame->GetPointFromOffset(firstFrame.mOffsetInNode, &ptOffset);
 | |
|     if (firstFrame->GetWritingMode().IsVertical()) {
 | |
|       rect.y += ptOffset.y;
 | |
|       rect.height -= ptOffset.y;
 | |
|     } else {
 | |
|       rect.x += ptOffset.x;
 | |
|       rect.width -= ptOffset.x;
 | |
|     }
 | |
|   }
 | |
|   // If first frame causes a line breaker but it's not a <br> frame, we cannot
 | |
|   // compute proper rect only with the frame because typically caret is at
 | |
|   // right of the last character of it.  For example, if caret is after "c" of
 | |
|   // |<p>abc</p><p>def</p>|, IME may query a line breaker's rect after "c".
 | |
|   // Then, if we compute it only with the 2nd <p>'s block frame, the result
 | |
|   // will be:
 | |
|   //  +-<p>--------------------------------+
 | |
|   //  |abc                                 |
 | |
|   //  +------------------------------------+
 | |
|   //
 | |
|   // I+-<p>--------------------------------+
 | |
|   //  |def                                 |
 | |
|   //  +------------------------------------+
 | |
|   // However, users expect popup windows of IME should be positioned at
 | |
|   // right-bottom of "c" like this:
 | |
|   //  +-<p>--------------------------------+
 | |
|   //  |abcI                                |
 | |
|   //  +------------------------------------+
 | |
|   //
 | |
|   //  +-<p>--------------------------------+
 | |
|   //  |def                                 |
 | |
|   //  +------------------------------------+
 | |
|   // Therefore, if the first frame isn't a <br> frame and there is a visible
 | |
|   // text node before the first node in the queried range, we should compute the
 | |
|   // first rect with the previous character's rect.
 | |
|   else if (!firstFrame->IsBrFrame() &&
 | |
|            domRangeAndAdjustedOffset.mLastTextNode &&
 | |
|            domRangeAndAdjustedOffset.mLastTextNode->GetPrimaryFrame()) {
 | |
|     FrameRelativeRect brRectAfterLastChar =
 | |
|         GuessLineBreakerRectAfter(*domRangeAndAdjustedOffset.mLastTextNode);
 | |
|     if (NS_WARN_IF(!brRectAfterLastChar.IsValid())) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
|     rect = brRectAfterLastChar.mRect;
 | |
|     rv = ConvertToRootRelativeOffset(brRectAfterLastChar.mBaseFrame, rect);
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return rv;
 | |
|     }
 | |
|     frameRect = rect;
 | |
|   }
 | |
|   // Otherwise, we need to compute the line breaker's rect only with the
 | |
|   // first frame's rect.  But this may be unexpected.  For example,
 | |
|   // |<div contenteditable>[<p>]abc</p></div>|.  In this case, caret is before
 | |
|   // "a", therefore, users expect the rect left of "a".  However, we don't
 | |
|   // have enough information about the next character here and this isn't
 | |
|   // usual case (e.g., IME typically tries to query the rect of "a" or caret
 | |
|   // rect for computing its popup position).  Therefore, we shouldn't do
 | |
|   // more complicated hack here unless we'll get some bug reports actually.
 | |
|   else {
 | |
|     FrameRelativeRect relativeRect = GetLineBreakerRectBefore(firstFrame);
 | |
|     if (NS_WARN_IF(!relativeRect.IsValid())) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
|     rect = relativeRect.RectRelativeTo(firstFrame);
 | |
|     rv = ConvertToRootRelativeOffset(firstFrame, rect);
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return rv;
 | |
|     }
 | |
|     frameRect = rect;
 | |
|   }
 | |
|   // UnionRect() requires non-empty rect.  So, let's make sure to get non-emtpy
 | |
|   // rect from the first frame.
 | |
|   EnsureNonEmptyRect(rect);
 | |
| 
 | |
|   // Get the last frame which causes some text in the range.
 | |
|   FrameAndNodeOffset lastFrame =
 | |
|       GetLastFrameInRangeForTextRect(domRangeAndAdjustedOffset.mRange);
 | |
|   if (NS_WARN_IF(!lastFrame.IsValid())) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   // iterate over all covered frames
 | |
|   for (nsIFrame* frame = firstFrame; frame != lastFrame;) {
 | |
|     frame = frame->GetNextContinuation();
 | |
|     if (!frame) {
 | |
|       do {
 | |
|         postOrderIter.Next();
 | |
|         nsINode* node = postOrderIter.GetCurrentNode();
 | |
|         if (!node) {
 | |
|           break;
 | |
|         }
 | |
|         if (!node->IsContent()) {
 | |
|           continue;
 | |
|         }
 | |
|         nsIFrame* primaryFrame = node->AsContent()->GetPrimaryFrame();
 | |
|         // The node may be hidden by CSS.
 | |
|         if (!primaryFrame) {
 | |
|           continue;
 | |
|         }
 | |
|         // We should take only text frame's rect and br frame's rect.  We can
 | |
|         // always use frame rect of text frame and GetLineBreakerRectBefore()
 | |
|         // can return exactly correct rect only for <br> frame for now.  On the
 | |
|         // other hand, GetLineBreakRectBefore() returns guessed caret rect for
 | |
|         // the other frames.  We shouldn't include such odd rect to the result.
 | |
|         if (primaryFrame->IsTextFrame() || primaryFrame->IsBrFrame()) {
 | |
|           frame = primaryFrame;
 | |
|         }
 | |
|       } while (!frame && !postOrderIter.IsDone());
 | |
|       if (!frame) {
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|     if (frame->IsTextFrame()) {
 | |
|       frameRect.SetRect(nsPoint(0, 0), frame->GetRect().Size());
 | |
|     } else {
 | |
|       MOZ_ASSERT(frame->IsBrFrame());
 | |
|       FrameRelativeRect relativeRect = GetLineBreakerRectBefore(frame);
 | |
|       if (NS_WARN_IF(!relativeRect.IsValid())) {
 | |
|         return NS_ERROR_FAILURE;
 | |
|       }
 | |
|       frameRect = relativeRect.RectRelativeTo(frame);
 | |
|     }
 | |
|     rv = ConvertToRootRelativeOffset(frame, frameRect);
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return rv;
 | |
|     }
 | |
|     // UnionRect() requires non-empty rect.  So, let's make sure to get
 | |
|     // non-emtpy rect from the frame.
 | |
|     EnsureNonEmptyRect(frameRect);
 | |
|     if (frame != lastFrame) {
 | |
|       // not last frame, so just add rect to previous result
 | |
|       rect.UnionRect(rect, frameRect);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Get the ending frame rect.
 | |
|   // FYI: If first frame and last frame are same, frameRect is already set
 | |
|   //      to the rect excluding the text before the query range.
 | |
|   if (firstFrame.mFrame != lastFrame.mFrame) {
 | |
|     frameRect.SetRect(nsPoint(0, 0), lastFrame->GetRect().Size());
 | |
|     rv = ConvertToRootRelativeOffset(lastFrame, frameRect);
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return rv;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Shrink the last frame for cutting off the text after the query range.
 | |
|   if (lastFrame->IsTextFrame()) {
 | |
|     lastFrame->GetPointFromOffset(lastFrame.mOffsetInNode, &ptOffset);
 | |
|     if (lastFrame->GetWritingMode().IsVertical()) {
 | |
|       frameRect.height -= lastFrame->GetRect().height - ptOffset.y;
 | |
|     } else {
 | |
|       frameRect.width -= lastFrame->GetRect().width - ptOffset.x;
 | |
|     }
 | |
|     // UnionRect() requires non-empty rect.  So, let's make sure to get
 | |
|     // non-empty rect from the last frame.
 | |
|     EnsureNonEmptyRect(frameRect);
 | |
| 
 | |
|     if (firstFrame.mFrame == lastFrame.mFrame) {
 | |
|       rect.IntersectRect(rect, frameRect);
 | |
|     } else {
 | |
|       rect.UnionRect(rect, frameRect);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   nsPresContext* presContext = lastFrame->PresContext();
 | |
|   aEvent->mReply->mRect = LayoutDeviceIntRect::FromAppUnitsToOutside(
 | |
|       rect, presContext->AppUnitsPerDevPixel());
 | |
|   if (nsPresContext* rootContext =
 | |
|           presContext->GetInProcessRootContentDocumentPresContext()) {
 | |
|     aEvent->mReply->mRect =
 | |
|         RoundedOut(ViewportUtils::DocumentRelativeLayoutToVisual(
 | |
|             aEvent->mReply->mRect, rootContext->PresShell()));
 | |
|   }
 | |
|   // Returning empty rect may cause native IME confused, let's make sure to
 | |
|   // return non-empty rect.
 | |
|   EnsureNonEmptyRect(aEvent->mReply->mRect);
 | |
|   aEvent->mReply->mWritingMode = lastFrame->GetWritingMode();
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQueryEditorRect(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   nsresult rv = Init(aEvent);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   if (NS_WARN_IF(NS_FAILED(QueryContentRect(mRootElement, aEvent)))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQueryCaretRect(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   nsresult rv = Init(aEvent);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   // When the selection is collapsed and the queried offset is current caret
 | |
|   // position, we should return the "real" caret rect.
 | |
|   if (mSelection->IsCollapsed()) {
 | |
|     nsRect caretRect;
 | |
|     nsIFrame* caretFrame = nsCaret::GetGeometry(mSelection, &caretRect);
 | |
|     if (caretFrame) {
 | |
|       uint32_t offset;
 | |
|       rv = GetStartOffset(mFirstSelectedSimpleRange, &offset,
 | |
|                           GetLineBreakType(aEvent));
 | |
|       NS_ENSURE_SUCCESS(rv, rv);
 | |
|       if (offset == aEvent->mInput.mOffset) {
 | |
|         rv = ConvertToRootRelativeOffset(caretFrame, caretRect);
 | |
|         NS_ENSURE_SUCCESS(rv, rv);
 | |
|         nsPresContext* presContext = caretFrame->PresContext();
 | |
|         aEvent->mReply->mRect = LayoutDeviceIntRect::FromAppUnitsToOutside(
 | |
|             caretRect, presContext->AppUnitsPerDevPixel());
 | |
|         if (nsPresContext* rootContext =
 | |
|                 presContext->GetInProcessRootContentDocumentPresContext()) {
 | |
|           aEvent->mReply->mRect =
 | |
|               RoundedOut(ViewportUtils::DocumentRelativeLayoutToVisual(
 | |
|                   aEvent->mReply->mRect, rootContext->PresShell()));
 | |
|         }
 | |
|         // Returning empty rect may cause native IME confused, let's make sure
 | |
|         // to return non-empty rect.
 | |
|         EnsureNonEmptyRect(aEvent->mReply->mRect);
 | |
|         aEvent->mReply->mWritingMode = caretFrame->GetWritingMode();
 | |
|         aEvent->mReply->mOffsetAndData.emplace(
 | |
|             aEvent->mInput.mOffset, EmptyString(),
 | |
|             OffsetAndDataFor::SelectedString);
 | |
| 
 | |
|         MOZ_ASSERT(aEvent->Succeeded());
 | |
|         return NS_OK;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Otherwise, we should guess the caret rect from the character's rect.
 | |
|   WidgetQueryContentEvent queryTextRectEvent(eQueryTextRect, *aEvent);
 | |
|   WidgetQueryContentEvent::Options options(*aEvent);
 | |
|   queryTextRectEvent.InitForQueryTextRect(aEvent->mInput.mOffset, 1, options);
 | |
|   if (NS_WARN_IF(NS_FAILED(OnQueryTextRect(&queryTextRectEvent))) ||
 | |
|       NS_WARN_IF(queryTextRectEvent.Failed())) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
|   queryTextRectEvent.mReply->TruncateData();
 | |
|   aEvent->mReply->mOffsetAndData =
 | |
|       std::move(queryTextRectEvent.mReply->mOffsetAndData);
 | |
|   aEvent->mReply->mWritingMode =
 | |
|       std::move(queryTextRectEvent.mReply->mWritingMode);
 | |
|   aEvent->mReply->mRect = GetCaretRectBefore(queryTextRectEvent.mReply->mRect,
 | |
|                                              aEvent->mReply->mWritingMode);
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQueryContentState(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   if (NS_FAILED(Init(aEvent))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
|   MOZ_ASSERT(aEvent->mReply.isSome());
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQuerySelectionAsTransferable(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   nsresult rv = Init(aEvent);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->mReply.isSome());
 | |
| 
 | |
|   if (mSelection->IsCollapsed()) {
 | |
|     MOZ_ASSERT(!aEvent->mReply->mTransferable);
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   if (NS_WARN_IF(NS_FAILED(nsCopySupport::GetTransferableForSelection(
 | |
|           mSelection, mDocument,
 | |
|           getter_AddRefs(aEvent->mReply->mTransferable))))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQueryCharacterAtPoint(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   nsresult rv = Init(aEvent);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->mReply->mOffsetAndData.isNothing());
 | |
|   MOZ_ASSERT(aEvent->mReply->mTentativeCaretOffset.isNothing());
 | |
| 
 | |
|   PresShell* presShell = mDocument->GetPresShell();
 | |
|   NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE);
 | |
|   nsIFrame* rootFrame = presShell->GetRootFrame();
 | |
|   NS_ENSURE_TRUE(rootFrame, NS_ERROR_FAILURE);
 | |
|   nsIWidget* rootWidget = rootFrame->GetNearestWidget();
 | |
|   NS_ENSURE_TRUE(rootWidget, NS_ERROR_FAILURE);
 | |
| 
 | |
|   // The root frame's widget might be different, e.g., the event was fired on
 | |
|   // a popup but the rootFrame is the document root.
 | |
|   if (rootWidget != aEvent->mWidget) {
 | |
|     MOZ_ASSERT(aEvent->mWidget, "The event must have the widget");
 | |
|     nsView* view = nsView::GetViewFor(aEvent->mWidget);
 | |
|     NS_ENSURE_TRUE(view, NS_ERROR_FAILURE);
 | |
|     rootFrame = view->GetFrame();
 | |
|     NS_ENSURE_TRUE(rootFrame, NS_ERROR_FAILURE);
 | |
|     rootWidget = rootFrame->GetNearestWidget();
 | |
|     NS_ENSURE_TRUE(rootWidget, NS_ERROR_FAILURE);
 | |
|   }
 | |
| 
 | |
|   WidgetQueryContentEvent queryCharAtPointOnRootWidgetEvent(
 | |
|       true, eQueryCharacterAtPoint, rootWidget);
 | |
|   queryCharAtPointOnRootWidgetEvent.mUseNativeLineBreak =
 | |
|       aEvent->mUseNativeLineBreak;
 | |
|   queryCharAtPointOnRootWidgetEvent.mRefPoint = aEvent->mRefPoint;
 | |
|   if (rootWidget != aEvent->mWidget) {
 | |
|     queryCharAtPointOnRootWidgetEvent.mRefPoint +=
 | |
|         aEvent->mWidget->WidgetToScreenOffset() -
 | |
|         rootWidget->WidgetToScreenOffset();
 | |
|   }
 | |
|   nsPoint ptInRoot = nsLayoutUtils::GetEventCoordinatesRelativeTo(
 | |
|       &queryCharAtPointOnRootWidgetEvent, RelativeTo{rootFrame});
 | |
| 
 | |
|   nsIFrame* targetFrame =
 | |
|       nsLayoutUtils::GetFrameForPoint(RelativeTo{rootFrame}, ptInRoot);
 | |
|   if (!targetFrame || !targetFrame->GetContent() ||
 | |
|       !targetFrame->GetContent()->IsInclusiveDescendantOf(mRootElement)) {
 | |
|     // There is no character at the point.
 | |
|     MOZ_ASSERT(aEvent->Succeeded());
 | |
|     return NS_OK;
 | |
|   }
 | |
|   nsPoint ptInTarget = ptInRoot + rootFrame->GetOffsetToCrossDoc(targetFrame);
 | |
|   int32_t rootAPD = rootFrame->PresContext()->AppUnitsPerDevPixel();
 | |
|   int32_t targetAPD = targetFrame->PresContext()->AppUnitsPerDevPixel();
 | |
|   ptInTarget = ptInTarget.ScaleToOtherAppUnits(rootAPD, targetAPD);
 | |
| 
 | |
|   nsIFrame::ContentOffsets tentativeCaretOffsets =
 | |
|       targetFrame->GetContentOffsetsFromPoint(ptInTarget);
 | |
|   if (!tentativeCaretOffsets.content ||
 | |
|       !tentativeCaretOffsets.content->IsInclusiveDescendantOf(mRootElement)) {
 | |
|     // There is no character nor tentative caret point at the point.
 | |
|     MOZ_ASSERT(aEvent->Succeeded());
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   uint32_t tentativeCaretOffset = 0;
 | |
|   if (NS_WARN_IF(NS_FAILED(GetFlatTextLengthInRange(
 | |
|           RawNodePosition(mRootElement, 0u),
 | |
|           RawNodePosition(tentativeCaretOffsets), mRootElement,
 | |
|           &tentativeCaretOffset, GetLineBreakType(aEvent))))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   aEvent->mReply->mTentativeCaretOffset.emplace(tentativeCaretOffset);
 | |
|   if (!targetFrame->IsTextFrame()) {
 | |
|     // There is no character at the point but there is tentative caret point.
 | |
|     MOZ_ASSERT(aEvent->Succeeded());
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   nsTextFrame* textframe = static_cast<nsTextFrame*>(targetFrame);
 | |
|   nsIFrame::ContentOffsets contentOffsets =
 | |
|       textframe->GetCharacterOffsetAtFramePoint(ptInTarget);
 | |
|   NS_ENSURE_TRUE(contentOffsets.content, NS_ERROR_FAILURE);
 | |
|   uint32_t offset = 0;
 | |
|   if (NS_WARN_IF(NS_FAILED(GetFlatTextLengthInRange(
 | |
|           RawNodePosition(mRootElement, 0u), RawNodePosition(contentOffsets),
 | |
|           mRootElement, &offset, GetLineBreakType(aEvent))))) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   WidgetQueryContentEvent queryTextRectEvent(true, eQueryTextRect,
 | |
|                                              aEvent->mWidget);
 | |
|   WidgetQueryContentEvent::Options options(*aEvent);
 | |
|   queryTextRectEvent.InitForQueryTextRect(offset, 1, options);
 | |
|   if (NS_WARN_IF(NS_FAILED(OnQueryTextRect(&queryTextRectEvent))) ||
 | |
|       NS_WARN_IF(queryTextRectEvent.Failed())) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   aEvent->mReply->mOffsetAndData =
 | |
|       std::move(queryTextRectEvent.mReply->mOffsetAndData);
 | |
|   aEvent->mReply->mRect = queryTextRectEvent.mReply->mRect;
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnQueryDOMWidgetHittest(
 | |
|     WidgetQueryContentEvent* aEvent) {
 | |
|   NS_ASSERTION(aEvent, "aEvent must not be null");
 | |
| 
 | |
|   nsresult rv = InitBasic();
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   aEvent->mReply->mWidgetIsHit = false;
 | |
| 
 | |
|   NS_ENSURE_TRUE(aEvent->mWidget, NS_ERROR_FAILURE);
 | |
| 
 | |
|   PresShell* presShell = mDocument->GetPresShell();
 | |
|   NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE);
 | |
|   nsIFrame* docFrame = presShell->GetRootFrame();
 | |
|   NS_ENSURE_TRUE(docFrame, NS_ERROR_FAILURE);
 | |
| 
 | |
|   LayoutDeviceIntPoint eventLoc =
 | |
|       aEvent->mRefPoint + aEvent->mWidget->WidgetToScreenOffset();
 | |
|   CSSIntRect docFrameRect = docFrame->GetScreenRect();
 | |
|   CSSIntPoint eventLocCSS(
 | |
|       docFrame->PresContext()->DevPixelsToIntCSSPixels(eventLoc.x) -
 | |
|           docFrameRect.x,
 | |
|       docFrame->PresContext()->DevPixelsToIntCSSPixels(eventLoc.y) -
 | |
|           docFrameRect.y);
 | |
| 
 | |
|   if (Element* contentUnderMouse = mDocument->ElementFromPointHelper(
 | |
|           eventLocCSS.x, eventLocCSS.y, false, false, ViewportType::Visual)) {
 | |
|     if (nsIFrame* targetFrame = contentUnderMouse->GetPrimaryFrame()) {
 | |
|       if (aEvent->mWidget == targetFrame->GetNearestWidget()) {
 | |
|         aEvent->mReply->mWidgetIsHit = true;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(aEvent->Succeeded());
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| nsresult ContentEventHandler::GetFlatTextLengthInRange(
 | |
|     const RawNodePosition& aStartPosition, const RawNodePosition& aEndPosition,
 | |
|     const Element* aRootElement, uint32_t* aLength,
 | |
|     LineBreakType aLineBreakType, bool aIsRemovingNode /* = false */) {
 | |
|   if (NS_WARN_IF(!aRootElement) || NS_WARN_IF(!aStartPosition.IsSet()) ||
 | |
|       NS_WARN_IF(!aEndPosition.IsSet()) || NS_WARN_IF(!aLength)) {
 | |
|     return NS_ERROR_INVALID_ARG;
 | |
|   }
 | |
| 
 | |
|   if (aStartPosition == aEndPosition) {
 | |
|     *aLength = 0;
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   UnsafePreContentIterator preOrderIter;
 | |
| 
 | |
|   // Working with ContentIterator, we may need to adjust the end position for
 | |
|   // including it forcibly.
 | |
|   RawNodePosition endPosition(aEndPosition);
 | |
| 
 | |
|   // This may be called for retrieving the text of removed nodes.  Even in this
 | |
|   // case, the node thinks it's still in the tree because UnbindFromTree() will
 | |
|   // be called after here.  However, the node was already removed from the
 | |
|   // array of children of its parent.  So, be careful to handle this case.
 | |
|   if (aIsRemovingNode) {
 | |
|     DebugOnly<nsIContent*> parent = aStartPosition.Container()->GetParent();
 | |
|     MOZ_ASSERT(
 | |
|         parent &&
 | |
|             parent->ComputeIndexOf(aStartPosition.Container()).isNothing(),
 | |
|         "At removing the node, the node shouldn't be in the array of children "
 | |
|         "of its parent");
 | |
|     MOZ_ASSERT(aStartPosition.Container() == endPosition.Container(),
 | |
|                "At removing the node, start and end node should be same");
 | |
|     MOZ_ASSERT(*aStartPosition.Offset(
 | |
|                    RawNodePosition::OffsetFilter::kValidOrInvalidOffsets) == 0,
 | |
|                "When the node is being removed, the start offset should be 0");
 | |
|     MOZ_ASSERT(
 | |
|         static_cast<uint32_t>(*endPosition.Offset(
 | |
|             RawNodePosition::OffsetFilter::kValidOrInvalidOffsets)) ==
 | |
|             endPosition.Container()->GetChildCount(),
 | |
|         "When the node is being removed, the end offset should be child count");
 | |
|     nsresult rv = preOrderIter.Init(aStartPosition.Container());
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return rv;
 | |
|     }
 | |
|   } else {
 | |
|     SimpleRange prevSimpleRange;
 | |
|     nsresult rv = prevSimpleRange.SetStart(aStartPosition.AsRaw());
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return rv;
 | |
|     }
 | |
| 
 | |
|     // When the end position is immediately after non-root element's open tag,
 | |
|     // we need to include a line break caused by the open tag.
 | |
|     if (endPosition.Container() != aRootElement &&
 | |
|         endPosition.IsImmediatelyAfterOpenTag()) {
 | |
|       if (endPosition.Container()->HasChildren()) {
 | |
|         // When the end node has some children, move the end position to before
 | |
|         // the open tag of its first child.
 | |
|         nsINode* firstChild = endPosition.Container()->GetFirstChild();
 | |
|         if (NS_WARN_IF(!firstChild)) {
 | |
|           return NS_ERROR_FAILURE;
 | |
|         }
 | |
|         endPosition = RawNodePositionBefore(firstChild, 0u);
 | |
|       } else {
 | |
|         // When the end node is empty, move the end position after the node.
 | |
|         nsIContent* parentContent = endPosition.Container()->GetParent();
 | |
|         if (NS_WARN_IF(!parentContent)) {
 | |
|           return NS_ERROR_FAILURE;
 | |
|         }
 | |
|         Maybe<uint32_t> indexInParent =
 | |
|             parentContent->ComputeIndexOf(endPosition.Container());
 | |
|         if (MOZ_UNLIKELY(NS_WARN_IF(indexInParent.isNothing()))) {
 | |
|           return NS_ERROR_FAILURE;
 | |
|         }
 | |
|         MOZ_ASSERT(*indexInParent != UINT32_MAX);
 | |
|         endPosition = RawNodePositionBefore(parentContent, *indexInParent + 1u);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (endPosition.IsSetAndValid()) {
 | |
|       // Offset is within node's length; set end of range to that offset
 | |
|       rv = prevSimpleRange.SetEnd(endPosition.AsRaw());
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return rv;
 | |
|       }
 | |
|       rv = preOrderIter.Init(prevSimpleRange.Start().AsRaw(),
 | |
|                              prevSimpleRange.End().AsRaw());
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return rv;
 | |
|       }
 | |
|     } else if (endPosition.Container() != aRootElement) {
 | |
|       // Offset is past node's length; set end of range to end of node
 | |
|       rv = prevSimpleRange.SetEndAfter(endPosition.Container());
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return rv;
 | |
|       }
 | |
|       rv = preOrderIter.Init(prevSimpleRange.Start().AsRaw(),
 | |
|                              prevSimpleRange.End().AsRaw());
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return rv;
 | |
|       }
 | |
|     } else {
 | |
|       // Offset is past the root node; set end of range to end of root node
 | |
|       rv = preOrderIter.Init(const_cast<Element*>(aRootElement));
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         return rv;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   *aLength = 0;
 | |
|   for (; !preOrderIter.IsDone(); preOrderIter.Next()) {
 | |
|     nsINode* node = preOrderIter.GetCurrentNode();
 | |
|     if (NS_WARN_IF(!node)) {
 | |
|       break;
 | |
|     }
 | |
|     if (!node->IsContent()) {
 | |
|       continue;
 | |
|     }
 | |
|     nsIContent* content = node->AsContent();
 | |
| 
 | |
|     if (const Text* textNode = Text::FromNode(content)) {
 | |
|       // Note: our range always starts from offset 0
 | |
|       if (node == endPosition.Container()) {
 | |
|         // NOTE: We should have an offset here, as endPosition.Container() is a
 | |
|         // nsINode::eTEXT, which always has an offset.
 | |
|         *aLength += GetTextLength(
 | |
|             *textNode, aLineBreakType,
 | |
|             *endPosition.Offset(
 | |
|                 RawNodePosition::OffsetFilter::kValidOrInvalidOffsets));
 | |
|       } else {
 | |
|         *aLength += GetTextLength(*textNode, aLineBreakType);
 | |
|       }
 | |
|     } else if (ShouldBreakLineBefore(*content, aRootElement)) {
 | |
|       // If the start position is start of this node but doesn't include the
 | |
|       // open tag, don't append the line break length.
 | |
|       if (node == aStartPosition.Container() &&
 | |
|           !aStartPosition.IsBeforeOpenTag()) {
 | |
|         continue;
 | |
|       }
 | |
|       // If the end position is before the open tag, don't append the line
 | |
|       // break length.
 | |
|       if (node == endPosition.Container() && endPosition.IsBeforeOpenTag()) {
 | |
|         continue;
 | |
|       }
 | |
|       *aLength += GetBRLength(aLineBreakType);
 | |
|     }
 | |
|   }
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| template <typename SimpleRangeType>
 | |
| nsresult ContentEventHandler::GetStartOffset(
 | |
|     const SimpleRangeType& aSimpleRange, uint32_t* aOffset,
 | |
|     LineBreakType aLineBreakType) {
 | |
|   // To match the "no skip start" hack in ContentIterator::Init, when range
 | |
|   // offset is 0 and the range node is not a container, we have to assume the
 | |
|   // range _includes_ the node, which means the start offset should _not_
 | |
|   // include the node.
 | |
|   //
 | |
|   // For example, for this content: <br>abc, and range (<br>, 0)-("abc", 1), the
 | |
|   // range includes the linebreak from <br>, so the start offset should _not_
 | |
|   // include <br>, and the start offset should be 0.
 | |
|   //
 | |
|   // However, for this content: <p/>abc, and range (<p>, 0)-("abc", 1), the
 | |
|   // range does _not_ include the linebreak from <p> because <p> is a container,
 | |
|   // so the start offset _should_ include <p>, and the start offset should be 1.
 | |
| 
 | |
|   nsINode* startNode = aSimpleRange.GetStartContainer();
 | |
|   bool startIsContainer = true;
 | |
|   if (startNode->IsHTMLElement()) {
 | |
|     nsAtom* name = startNode->NodeInfo()->NameAtom();
 | |
|     startIsContainer =
 | |
|         nsHTMLElement::IsContainer(nsHTMLTags::AtomTagToId(name));
 | |
|   }
 | |
|   RawNodePosition startPos(startNode, aSimpleRange.StartOffset());
 | |
|   startPos.mAfterOpenTag = startIsContainer;
 | |
|   return GetFlatTextLengthInRange(RawNodePosition(mRootElement, 0u), startPos,
 | |
|                                   mRootElement, aOffset, aLineBreakType);
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::AdjustCollapsedRangeMaybeIntoTextNode(
 | |
|     SimpleRange& aSimpleRange) {
 | |
|   MOZ_ASSERT(aSimpleRange.Collapsed());
 | |
| 
 | |
|   if (!aSimpleRange.Collapsed()) {
 | |
|     return NS_ERROR_INVALID_ARG;
 | |
|   }
 | |
| 
 | |
|   const RangeBoundary& startPoint = aSimpleRange.Start();
 | |
|   if (NS_WARN_IF(!startPoint.IsSet())) {
 | |
|     return NS_ERROR_INVALID_ARG;
 | |
|   }
 | |
| 
 | |
|   // If the node does not have children like a text node, we don't need to
 | |
|   // modify aSimpleRange.
 | |
|   if (!startPoint.Container()->HasChildren()) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // If the container is not a text node but it has a text node at the offset,
 | |
|   // we should adjust the range into the text node.
 | |
|   // NOTE: This is emulating similar situation of EditorBase.
 | |
|   if (startPoint.IsStartOfContainer()) {
 | |
|     // If the range is the start of the container, adjusted the range to the
 | |
|     // start of the first child.
 | |
|     if (!startPoint.Container()->GetFirstChild()->IsText()) {
 | |
|       return NS_OK;
 | |
|     }
 | |
|     nsresult rv = aSimpleRange.CollapseTo(
 | |
|         RawRangeBoundary(startPoint.Container()->GetFirstChild(), 0u));
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       return rv;
 | |
|     }
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   if (!startPoint.IsSetAndValid()) {
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   // If start of the range is next to a child node, adjust the range to the
 | |
|   // end of the previous child (i.e., startPoint.Ref()).
 | |
|   if (!startPoint.Ref()->IsText()) {
 | |
|     return NS_OK;
 | |
|   }
 | |
|   nsresult rv = aSimpleRange.CollapseTo(
 | |
|       RawRangeBoundary(startPoint.Ref(), startPoint.Ref()->Length()));
 | |
|   if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|     return rv;
 | |
|   }
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::ConvertToRootRelativeOffset(nsIFrame* aFrame,
 | |
|                                                           nsRect& aRect) {
 | |
|   NS_ASSERTION(aFrame, "aFrame must not be null");
 | |
| 
 | |
|   nsPresContext* thisPC = aFrame->PresContext();
 | |
|   nsPresContext* rootPC = thisPC->GetRootPresContext();
 | |
|   if (NS_WARN_IF(!rootPC)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
|   nsIFrame* rootFrame = rootPC->PresShell()->GetRootFrame();
 | |
|   if (NS_WARN_IF(!rootFrame)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   aRect = nsLayoutUtils::TransformFrameRectToAncestor(aFrame, aRect, rootFrame);
 | |
| 
 | |
|   // TransformFrameRectToAncestor returned the rect in the ancestor's appUnits,
 | |
|   // but we want it in aFrame's units (in case of different full-zoom factors),
 | |
|   // so convert back.
 | |
|   aRect = aRect.ScaleToOtherAppUnitsRoundOut(rootPC->AppUnitsPerDevPixel(),
 | |
|                                              thisPC->AppUnitsPerDevPixel());
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| static void AdjustRangeForSelection(const Element* aRootElement,
 | |
|                                     nsINode** aNode,
 | |
|                                     Maybe<uint32_t>* aNodeOffset) {
 | |
|   nsINode* node = *aNode;
 | |
|   Maybe<uint32_t> nodeOffset = *aNodeOffset;
 | |
|   if (aRootElement == node || NS_WARN_IF(!node->GetParent()) ||
 | |
|       !node->IsText()) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // When the offset is at the end of the text node, set it to after the
 | |
|   // text node, to make sure the caret is drawn on a new line when the last
 | |
|   // character of the text node is '\n' in <textarea>.
 | |
|   const uint32_t textLength = node->AsContent()->TextLength();
 | |
|   MOZ_ASSERT(nodeOffset.isNothing() || *nodeOffset <= textLength,
 | |
|              "Offset is past length of text node");
 | |
|   if (nodeOffset.isNothing() || *nodeOffset != textLength) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Element* rootParentElement = aRootElement->GetParentElement();
 | |
|   if (NS_WARN_IF(!rootParentElement)) {
 | |
|     return;
 | |
|   }
 | |
|   // If the root node is not an anonymous div of <textarea>, we don't need to
 | |
|   // do this hack.  If you did this, ContentEventHandler couldn't distinguish
 | |
|   // if the range includes open tag of the next node in some cases, e.g.,
 | |
|   // textNode]<p></p> vs. textNode<p>]</p>
 | |
|   if (!rootParentElement->IsHTMLElement(nsGkAtoms::textarea)) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // If the node is being removed from its parent, it holds the ex-parent,
 | |
|   // but the parent have already removed the child from its child chain.
 | |
|   // Therefore `ComputeIndexOf` may fail, but I don't want to make Beta/Nightly
 | |
|   // crash at accessing `Maybe::operator*` so that here checks `isSome`, but
 | |
|   // crashing only in debug builds may help to debug something complicated
 | |
|   // situation, therefore, `MOZ_ASSERT` is put here.
 | |
|   *aNode = node->GetParent();
 | |
|   Maybe<uint32_t> index = (*aNode)->ComputeIndexOf(node);
 | |
|   MOZ_ASSERT(index.isSome());
 | |
|   if (index.isSome()) {
 | |
|     MOZ_ASSERT(*index != UINT32_MAX);
 | |
|     *aNodeOffset = Some(*index + 1u);
 | |
|   } else {
 | |
|     *aNodeOffset = Some(0u);
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsresult ContentEventHandler::OnSelectionEvent(WidgetSelectionEvent* aEvent) {
 | |
|   aEvent->mSucceeded = false;
 | |
| 
 | |
|   // Get selection to manipulate
 | |
|   // XXX why do we need to get them from ISM? This method should work fine
 | |
|   //     without ISM.
 | |
|   nsresult rv = IMEStateManager::GetFocusSelectionAndRootElement(
 | |
|       getter_AddRefs(mSelection), getter_AddRefs(mRootElement));
 | |
|   if (rv != NS_ERROR_NOT_AVAILABLE) {
 | |
|     NS_ENSURE_SUCCESS(rv, rv);
 | |
|   } else {
 | |
|     rv = Init(aEvent);
 | |
|     NS_ENSURE_SUCCESS(rv, rv);
 | |
|   }
 | |
| 
 | |
|   // Get range from offset and length
 | |
|   nsINode* startNode = nullptr;
 | |
|   nsINode* endNode = nullptr;
 | |
|   Maybe<uint32_t> startNodeOffset;
 | |
|   Maybe<uint32_t> endNodeOffset;
 | |
|   {
 | |
|     Result<UnsafeDOMRangeAndAdjustedOffsetInFlattenedText, nsresult>
 | |
|         domRangeAndAdjustedOffsetOrError =
 | |
|             ConvertFlatTextOffsetToUnsafeDOMRange(
 | |
|                 aEvent->mOffset, aEvent->mLength, GetLineBreakType(aEvent),
 | |
|                 aEvent->mExpandToClusterBoundary);
 | |
|     if (MOZ_UNLIKELY(domRangeAndAdjustedOffsetOrError.isErr())) {
 | |
|       NS_WARNING(
 | |
|           "ContentEventHandler::ConvertFlatTextOffsetToDOMRangeBase() failed");
 | |
|       return domRangeAndAdjustedOffsetOrError.unwrapErr();
 | |
|     }
 | |
|     const UnsafeDOMRangeAndAdjustedOffsetInFlattenedText
 | |
|         domRangeAndAdjustedOffset = domRangeAndAdjustedOffsetOrError.unwrap();
 | |
|     startNode = domRangeAndAdjustedOffset.mRange.GetStartContainer();
 | |
|     endNode = domRangeAndAdjustedOffset.mRange.GetEndContainer();
 | |
|     startNodeOffset = Some(domRangeAndAdjustedOffset.mRange.StartOffset());
 | |
|     endNodeOffset = Some(domRangeAndAdjustedOffset.mRange.EndOffset());
 | |
|     AdjustRangeForSelection(mRootElement, &startNode, &startNodeOffset);
 | |
|     AdjustRangeForSelection(mRootElement, &endNode, &endNodeOffset);
 | |
|     if (NS_WARN_IF(!startNode) || NS_WARN_IF(!endNode) ||
 | |
|         NS_WARN_IF(startNodeOffset.isNothing()) ||
 | |
|         NS_WARN_IF(endNodeOffset.isNothing())) {
 | |
|       return NS_ERROR_UNEXPECTED;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (aEvent->mReversed) {
 | |
|     nsCOMPtr<nsINode> startNodeStrong(startNode);
 | |
|     nsCOMPtr<nsINode> endNodeStrong(endNode);
 | |
|     ErrorResult error;
 | |
|     MOZ_KnownLive(mSelection)
 | |
|         ->SetBaseAndExtentInLimiter(*endNodeStrong, *endNodeOffset,
 | |
|                                     *startNodeStrong, *startNodeOffset, error);
 | |
|     if (NS_WARN_IF(error.Failed())) {
 | |
|       return error.StealNSResult();
 | |
|     }
 | |
|   } else {
 | |
|     nsCOMPtr<nsINode> startNodeStrong(startNode);
 | |
|     nsCOMPtr<nsINode> endNodeStrong(endNode);
 | |
|     ErrorResult error;
 | |
|     MOZ_KnownLive(mSelection)
 | |
|         ->SetBaseAndExtentInLimiter(*startNodeStrong, *startNodeOffset,
 | |
|                                     *endNodeStrong, *endNodeOffset, error);
 | |
|     if (NS_WARN_IF(error.Failed())) {
 | |
|       return error.StealNSResult();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // `ContentEventHandler` is a `MOZ_STACK_CLASS`, so `mSelection` is known to
 | |
|   // be alive.
 | |
|   MOZ_KnownLive(mSelection)
 | |
|       ->ScrollIntoView(nsISelectionController::SELECTION_FOCUS_REGION,
 | |
|                        ScrollAxis(), ScrollAxis(), 0);
 | |
|   aEvent->mSucceeded = true;
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsRect ContentEventHandler::FrameRelativeRect::RectRelativeTo(
 | |
|     nsIFrame* aDestFrame) const {
 | |
|   if (!mBaseFrame || NS_WARN_IF(!aDestFrame)) {
 | |
|     return nsRect();
 | |
|   }
 | |
| 
 | |
|   if (NS_WARN_IF(aDestFrame->PresContext() != mBaseFrame->PresContext())) {
 | |
|     return nsRect();
 | |
|   }
 | |
| 
 | |
|   if (aDestFrame == mBaseFrame) {
 | |
|     return mRect;
 | |
|   }
 | |
| 
 | |
|   nsIFrame* rootFrame = mBaseFrame->PresShell()->GetRootFrame();
 | |
|   nsRect baseFrameRectInRootFrame = nsLayoutUtils::TransformFrameRectToAncestor(
 | |
|       mBaseFrame, nsRect(), rootFrame);
 | |
|   nsRect destFrameRectInRootFrame = nsLayoutUtils::TransformFrameRectToAncestor(
 | |
|       aDestFrame, nsRect(), rootFrame);
 | |
|   nsPoint difference =
 | |
|       destFrameRectInRootFrame.TopLeft() - baseFrameRectInRootFrame.TopLeft();
 | |
|   return mRect - difference;
 | |
| }
 | |
| 
 | |
| }  // namespace mozilla
 |