Source code

Revision control

Copy as Markdown

Other Tools

/* -*- 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