Unfortunately, this patch becomes big because this includes multiple utility
methods to normalize white-spaces and touching a lot of places.
In some edge cases, this starts failing when the new white-space normalizer is
enabled.
```
FAIL execCommand("forwarddelete", false, ""): "a [] | b" - assert_equals: Modified text is wrong expected "a " but got "a "
FAIL execCommand("forwarddelete", false, ""): "a [] | b" - assert_equals: Modified text is wrong expected "a " but got "a "
FAIL execCommand("forwarddelete", false, ""): "a []b| c" - assert_equals: Modified text is wrong expected "a " but got "a "
FAIL execCommand("forwarddelete", false, ""): "a [] <span style=white-space:pre;> </span>" - assert_equals: expected "a <span style=\"white-space:pre;\"> </span>" but got "a <span style=\"white-space:pre;\"> </span>"
```
In these failures, we normalize the trailing white-spaces of the preceding
`Text` too, but Chrome does not do that. However, in some cases, they do
in similar cases. Therefore, it's hard to align all cases due to the handling
order difference. Additionally, these failure results are not odd. So, these
new failures are acceptable.
```
FAIL execCommand("forwarddelete", false, ""): "<span>abc[] <span> def</span></span>" - assert_equals: expected "<span>abc<span> def</span></span>" but got "<span>abc<span> def</span></span>"
FAIL execCommand("forwarddelete", false, ""): "<span><span>abc[] </span> def</span>" - assert_equals: expected "<span><span>abc</span> def</span>" but got "<span><span>abc</span> def</span>"
FAIL execCommand("forwarddelete", false, ""): "<span>abc[] </span><span> def</span>" - assert_equals: expected "<span>abc</span><span> def</span>" but got "<span>abc</span><span> def</span>"
FAIL execCommand("forwarddelete", false, ""): "<span>abc[] </span><span> def</span>" - assert_equals: expected "<span>abc </span><span> def</span>" but got "<span>abc </span><span> def</span>"
```
In these failures, we don't normalize the leading white-spaces of the following
`Text`. However, Chrome does so in some other cases. So, it's hard to align
the behavior in these cases too. The new normalizer ensures that the preceding
`Text` ends with a visible char. Therefore, the following `Text` won't starts
with new invisible white-spaces. Therefore, these failures are also acceptable.
```
FAIL execCommand("forwarddelete", false, ""): "<span style=white-space:pre;> [] </span> a" - assert_equals: expected "<span style=\"white-space:pre;\"> </span> a" but got "<span style=\"white-space:pre;\"> </span> a"
```
In this failure, we don't normalize the following `Text` but Chrome does it.
However, I don't think we should do it because this updates the `Text` which
is preformatted. Therefore, this is also acceptable.
So, I think that we can accept all new failures.
Differential Revision: https://phabricator.services.mozilla.com/D239468
3193 lines
126 KiB
C++
3193 lines
126 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* 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 "HTMLEditUtils.h"
|
|
|
|
#include "AutoClonedRangeArray.h" // for AutoClonedRangeArray
|
|
#include "CSSEditUtils.h" // for CSSEditUtils
|
|
#include "EditAction.h" // for EditAction
|
|
#include "EditorBase.h" // for EditorBase, EditorType
|
|
#include "EditorDOMPoint.h" // for EditorDOMPoint, etc.
|
|
#include "EditorForwards.h" // for CollectChildrenOptions
|
|
#include "EditorUtils.h" // for EditorUtils
|
|
#include "HTMLEditHelpers.h" // for EditorInlineStyle
|
|
#include "WSRunScanner.h" // for WSRunScanner
|
|
|
|
#include "mozilla/ArrayUtils.h" // for ArrayLength
|
|
#include "mozilla/Assertions.h" // for MOZ_ASSERT, etc.
|
|
#include "mozilla/Attributes.h"
|
|
#include "mozilla/StaticPrefs_editor.h" // for StaticPrefs::editor_
|
|
#include "mozilla/RangeUtils.h" // for RangeUtils
|
|
#include "mozilla/dom/DocumentInlines.h" // for GetBodyElement()
|
|
#include "mozilla/dom/Element.h" // for Element, nsINode
|
|
#include "mozilla/dom/ElementInlines.h" // for IsContentEditablePlainTextOnly()
|
|
#include "mozilla/dom/HTMLAnchorElement.h"
|
|
#include "mozilla/dom/HTMLBodyElement.h"
|
|
#include "mozilla/dom/HTMLInputElement.h"
|
|
#include "mozilla/ServoCSSParser.h" // for ServoCSSParser
|
|
#include "mozilla/dom/StaticRange.h"
|
|
#include "mozilla/dom/Text.h" // for Text
|
|
|
|
#include "nsAString.h" // for nsAString::IsEmpty
|
|
#include "nsAtom.h" // for nsAtom
|
|
#include "nsAttrValue.h" // nsAttrValue
|
|
#include "nsCaseTreatment.h"
|
|
#include "nsCOMPtr.h" // for nsCOMPtr, operator==, etc.
|
|
#include "nsComputedDOMStyle.h" // for nsComputedDOMStyle
|
|
#include "nsDebug.h" // for NS_ASSERTION, etc.
|
|
#include "nsElementTable.h" // for nsHTMLElement
|
|
#include "nsError.h" // for NS_SUCCEEDED
|
|
#include "nsGkAtoms.h" // for nsGkAtoms, nsGkAtoms::a, etc.
|
|
#include "nsHTMLTags.h"
|
|
#include "nsIContentInlines.h" // for nsIContent::IsInDesignMode(), etc.
|
|
#include "nsIFrameInlines.h" // for nsIFrame::IsFlexOrGridItem()
|
|
#include "nsLiteralString.h" // for NS_LITERAL_STRING
|
|
#include "nsNameSpaceManager.h" // for kNameSpaceID_None
|
|
#include "nsPrintfCString.h" // nsPringfCString
|
|
#include "nsString.h" // for nsAutoString
|
|
#include "nsStyledElement.h"
|
|
#include "nsStyleStruct.h" // for StyleDisplay
|
|
#include "nsStyleUtil.h" // for nsStyleUtil
|
|
#include "nsTextFragment.h" // for nsTextFragment
|
|
#include "nsTextFrame.h" // for nsTextFrame
|
|
|
|
namespace mozilla {
|
|
|
|
using namespace dom;
|
|
using EditorType = EditorBase::EditorType;
|
|
|
|
template nsIContent* HTMLEditUtils::GetPreviousContent(
|
|
const EditorDOMPoint& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck, const Element* aAncestorLimiter);
|
|
template nsIContent* HTMLEditUtils::GetPreviousContent(
|
|
const EditorRawDOMPoint& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck, const Element* aAncestorLimiter);
|
|
template nsIContent* HTMLEditUtils::GetPreviousContent(
|
|
const EditorDOMPointInText& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck, const Element* aAncestorLimiter);
|
|
template nsIContent* HTMLEditUtils::GetPreviousContent(
|
|
const EditorRawDOMPointInText& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck, const Element* aAncestorLimiter);
|
|
template nsIContent* HTMLEditUtils::GetNextContent(
|
|
const EditorDOMPoint& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck, const Element* aAncestorLimiter);
|
|
template nsIContent* HTMLEditUtils::GetNextContent(
|
|
const EditorRawDOMPoint& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck, const Element* aAncestorLimiter);
|
|
template nsIContent* HTMLEditUtils::GetNextContent(
|
|
const EditorDOMPointInText& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck, const Element* aAncestorLimiter);
|
|
template nsIContent* HTMLEditUtils::GetNextContent(
|
|
const EditorRawDOMPointInText& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck, const Element* aAncestorLimiter);
|
|
|
|
template EditorDOMPoint HTMLEditUtils::GetPreviousEditablePoint(
|
|
nsIContent& aContent, const Element* aAncestorLimiter,
|
|
InvisibleWhiteSpaces aInvisibleWhiteSpaces,
|
|
TableBoundary aHowToTreatTableBoundary);
|
|
template EditorRawDOMPoint HTMLEditUtils::GetPreviousEditablePoint(
|
|
nsIContent& aContent, const Element* aAncestorLimiter,
|
|
InvisibleWhiteSpaces aInvisibleWhiteSpaces,
|
|
TableBoundary aHowToTreatTableBoundary);
|
|
template EditorDOMPoint HTMLEditUtils::GetNextEditablePoint(
|
|
nsIContent& aContent, const Element* aAncestorLimiter,
|
|
InvisibleWhiteSpaces aInvisibleWhiteSpaces,
|
|
TableBoundary aHowToTreatTableBoundary);
|
|
template EditorRawDOMPoint HTMLEditUtils::GetNextEditablePoint(
|
|
nsIContent& aContent, const Element* aAncestorLimiter,
|
|
InvisibleWhiteSpaces aInvisibleWhiteSpaces,
|
|
TableBoundary aHowToTreatTableBoundary);
|
|
|
|
template EditorDOMPoint HTMLEditUtils::LineRequiresPaddingLineBreakToBeVisible(
|
|
const EditorDOMPoint& aPoint, const Element& aEditingHost);
|
|
template EditorDOMPoint HTMLEditUtils::LineRequiresPaddingLineBreakToBeVisible(
|
|
const EditorRawDOMPoint& aPoint, const Element& aEditingHost);
|
|
template EditorDOMPoint HTMLEditUtils::LineRequiresPaddingLineBreakToBeVisible(
|
|
const EditorDOMPointInText& aPoint, const Element& aEditingHost);
|
|
template EditorDOMPoint HTMLEditUtils::LineRequiresPaddingLineBreakToBeVisible(
|
|
const EditorRawDOMPointInText& aPoint, const Element& aEditingHost);
|
|
|
|
template nsIContent* HTMLEditUtils::GetContentToPreserveInlineStyles(
|
|
const EditorDOMPoint& aPoint, const Element& aEditingHost);
|
|
template nsIContent* HTMLEditUtils::GetContentToPreserveInlineStyles(
|
|
const EditorRawDOMPoint& aPoint, const Element& aEditingHost);
|
|
|
|
template EditorDOMPoint HTMLEditUtils::GetBetterInsertionPointFor(
|
|
const nsIContent& aContentToInsert, const EditorDOMPoint& aPointToInsert);
|
|
template EditorRawDOMPoint HTMLEditUtils::GetBetterInsertionPointFor(
|
|
const nsIContent& aContentToInsert,
|
|
const EditorRawDOMPoint& aPointToInsert);
|
|
template EditorDOMPoint HTMLEditUtils::GetBetterInsertionPointFor(
|
|
const nsIContent& aContentToInsert,
|
|
const EditorRawDOMPoint& aPointToInsert);
|
|
template EditorRawDOMPoint HTMLEditUtils::GetBetterInsertionPointFor(
|
|
const nsIContent& aContentToInsert, const EditorDOMPoint& aPointToInsert);
|
|
|
|
template EditorDOMPoint HTMLEditUtils::GetBetterCaretPositionToInsertText(
|
|
const EditorDOMPoint& aPoint);
|
|
template EditorDOMPoint HTMLEditUtils::GetBetterCaretPositionToInsertText(
|
|
const EditorRawDOMPoint& aPoint);
|
|
template EditorRawDOMPoint HTMLEditUtils::GetBetterCaretPositionToInsertText(
|
|
const EditorDOMPoint& aPoint);
|
|
template EditorRawDOMPoint HTMLEditUtils::GetBetterCaretPositionToInsertText(
|
|
const EditorRawDOMPoint& aPoint);
|
|
|
|
template Result<EditorDOMPoint, nsresult>
|
|
HTMLEditUtils::ComputePointToPutCaretInElementIfOutside(
|
|
const Element& aElement, const EditorDOMPoint& aCurrentPoint);
|
|
template Result<EditorRawDOMPoint, nsresult>
|
|
HTMLEditUtils::ComputePointToPutCaretInElementIfOutside(
|
|
const Element& aElement, const EditorDOMPoint& aCurrentPoint);
|
|
template Result<EditorDOMPoint, nsresult>
|
|
HTMLEditUtils::ComputePointToPutCaretInElementIfOutside(
|
|
const Element& aElement, const EditorRawDOMPoint& aCurrentPoint);
|
|
template Result<EditorRawDOMPoint, nsresult>
|
|
HTMLEditUtils::ComputePointToPutCaretInElementIfOutside(
|
|
const Element& aElement, const EditorRawDOMPoint& aCurrentPoint);
|
|
|
|
template bool HTMLEditUtils::IsSameCSSColorValue(const nsAString& aColorA,
|
|
const nsAString& aColorB);
|
|
template bool HTMLEditUtils::IsSameCSSColorValue(const nsACString& aColorA,
|
|
const nsACString& aColorB);
|
|
|
|
template Maybe<EditorLineBreak> HTMLEditUtils::GetFollowingUnnecessaryLineBreak(
|
|
const EditorDOMPoint& aPoint);
|
|
template Maybe<EditorLineBreak> HTMLEditUtils::GetFollowingUnnecessaryLineBreak(
|
|
const EditorRawDOMPoint& aPoint);
|
|
template Maybe<EditorLineBreak> HTMLEditUtils::GetFollowingUnnecessaryLineBreak(
|
|
const EditorDOMPointInText& aPoint);
|
|
template Maybe<EditorLineBreak> HTMLEditUtils::GetFollowingUnnecessaryLineBreak(
|
|
const EditorRawDOMPointInText& aPoint);
|
|
template Maybe<EditorRawLineBreak>
|
|
HTMLEditUtils::GetFollowingUnnecessaryLineBreak(const EditorDOMPoint& aPoint);
|
|
template Maybe<EditorRawLineBreak>
|
|
HTMLEditUtils::GetFollowingUnnecessaryLineBreak(
|
|
const EditorRawDOMPoint& aPoint);
|
|
template Maybe<EditorRawLineBreak>
|
|
HTMLEditUtils::GetFollowingUnnecessaryLineBreak(
|
|
const EditorDOMPointInText& aPoint);
|
|
template Maybe<EditorRawLineBreak>
|
|
HTMLEditUtils::GetFollowingUnnecessaryLineBreak(
|
|
const EditorRawDOMPointInText& aPoint);
|
|
|
|
template bool HTMLEditUtils::PointIsImmediatelyBeforeCurrentBlockBoundary(
|
|
const EditorDOMPoint& aPoint,
|
|
IgnoreInvisibleLineBreak aIgnoreInvisibleLineBreak);
|
|
template bool HTMLEditUtils::PointIsImmediatelyBeforeCurrentBlockBoundary(
|
|
const EditorRawDOMPoint& aPoint,
|
|
IgnoreInvisibleLineBreak aIgnoreInvisibleLineBreak);
|
|
template bool HTMLEditUtils::PointIsImmediatelyBeforeCurrentBlockBoundary(
|
|
const EditorDOMPointInText& aPoint,
|
|
IgnoreInvisibleLineBreak aIgnoreInvisibleLineBreak);
|
|
template bool HTMLEditUtils::PointIsImmediatelyBeforeCurrentBlockBoundary(
|
|
const EditorRawDOMPointInText& aPoint,
|
|
IgnoreInvisibleLineBreak aIgnoreInvisibleLineBreak);
|
|
|
|
template Maybe<EditorLineBreak> HTMLEditUtils::GetUnnecessaryLineBreak(
|
|
const Element& aBlockElement, ScanLineBreak aScanLineBreak);
|
|
template Maybe<EditorRawLineBreak> HTMLEditUtils::GetUnnecessaryLineBreak(
|
|
const Element& aBlockElement, ScanLineBreak aScanLineBreak);
|
|
|
|
bool HTMLEditUtils::ElementIsEditableRoot(const Element& aElement) {
|
|
MOZ_ASSERT(!aElement.IsInNativeAnonymousSubtree());
|
|
if (NS_WARN_IF(!aElement.IsEditable()) ||
|
|
NS_WARN_IF(!aElement.IsInComposedDoc())) {
|
|
return false;
|
|
}
|
|
return !aElement.GetParent() || // root element
|
|
!aElement.GetParent()->IsEditable() || // editing host
|
|
aElement.OwnerDoc()->GetBody() == &aElement; // the <body>
|
|
}
|
|
|
|
bool HTMLEditUtils::CanContentsBeJoined(const nsIContent& aLeftContent,
|
|
const nsIContent& aRightContent) {
|
|
if (aLeftContent.NodeInfo()->NameAtom() !=
|
|
aRightContent.NodeInfo()->NameAtom()) {
|
|
return false;
|
|
}
|
|
|
|
if (!aLeftContent.IsElement()) {
|
|
return true; // can join text nodes, etc
|
|
}
|
|
MOZ_ASSERT(aRightContent.IsElement());
|
|
|
|
if (aLeftContent.NodeInfo()->NameAtom() == nsGkAtoms::font) {
|
|
const nsAttrValue* const leftSize =
|
|
aLeftContent.AsElement()->GetParsedAttr(nsGkAtoms::size);
|
|
const nsAttrValue* const rightSize =
|
|
aRightContent.AsElement()->GetParsedAttr(nsGkAtoms::size);
|
|
if (!leftSize ^ !rightSize || (leftSize && !leftSize->Equals(*rightSize))) {
|
|
return false;
|
|
}
|
|
|
|
const nsAttrValue* const leftColor =
|
|
aLeftContent.AsElement()->GetParsedAttr(nsGkAtoms::color);
|
|
const nsAttrValue* const rightColor =
|
|
aRightContent.AsElement()->GetParsedAttr(nsGkAtoms::color);
|
|
if (!leftColor ^ !rightColor ||
|
|
(leftColor && !leftColor->Equals(*rightColor))) {
|
|
return false;
|
|
}
|
|
|
|
const nsAttrValue* const leftFace =
|
|
aLeftContent.AsElement()->GetParsedAttr(nsGkAtoms::face);
|
|
const nsAttrValue* const rightFace =
|
|
aRightContent.AsElement()->GetParsedAttr(nsGkAtoms::face);
|
|
if (!leftFace ^ !rightFace || (leftFace && !leftFace->Equals(*rightFace))) {
|
|
return false;
|
|
}
|
|
}
|
|
nsStyledElement* leftStyledElement =
|
|
nsStyledElement::FromNode(const_cast<nsIContent*>(&aLeftContent));
|
|
if (!leftStyledElement) {
|
|
return false;
|
|
}
|
|
nsStyledElement* rightStyledElement =
|
|
nsStyledElement::FromNode(const_cast<nsIContent*>(&aRightContent));
|
|
if (!rightStyledElement) {
|
|
return false;
|
|
}
|
|
return CSSEditUtils::DoStyledElementsHaveSameStyle(*leftStyledElement,
|
|
*rightStyledElement);
|
|
}
|
|
|
|
static bool IsHTMLBlockElementByDefault(const nsIContent& aContent) {
|
|
if (!aContent.IsHTMLElement()) {
|
|
return false;
|
|
}
|
|
if (aContent.IsHTMLElement(nsGkAtoms::br)) { // shortcut for TextEditor
|
|
MOZ_ASSERT(!nsHTMLElement::IsBlock(
|
|
nsHTMLTags::CaseSensitiveAtomTagToId(nsGkAtoms::br)));
|
|
return false;
|
|
}
|
|
// We want to treat these as block nodes even though nsHTMLElement says
|
|
// they're not.
|
|
if (aContent.IsAnyOfHTMLElements(
|
|
nsGkAtoms::body, nsGkAtoms::head, nsGkAtoms::tbody, nsGkAtoms::thead,
|
|
nsGkAtoms::tfoot, nsGkAtoms::tr, nsGkAtoms::th, nsGkAtoms::td,
|
|
nsGkAtoms::dt, nsGkAtoms::dd)) {
|
|
return true;
|
|
}
|
|
|
|
return nsHTMLElement::IsBlock(
|
|
nsHTMLTags::CaseSensitiveAtomTagToId(aContent.NodeInfo()->NameAtom()));
|
|
}
|
|
|
|
bool HTMLEditUtils::IsBlockElement(const nsIContent& aContent,
|
|
BlockInlineCheck aBlockInlineCheck) {
|
|
MOZ_ASSERT(aBlockInlineCheck != BlockInlineCheck::Unused);
|
|
|
|
if (MOZ_UNLIKELY(!aContent.IsElement())) {
|
|
return false;
|
|
}
|
|
// If it's a <br>, we should always treat it as an inline element because
|
|
// its preceding collapse white-spaces and another <br> works same as usual
|
|
// even if you set its style to `display:block`.
|
|
if (aContent.IsHTMLElement(nsGkAtoms::br)) {
|
|
return false;
|
|
}
|
|
if (aBlockInlineCheck == BlockInlineCheck::UseHTMLDefaultStyle) {
|
|
return IsHTMLBlockElementByDefault(aContent);
|
|
}
|
|
// Let's treat the document element and the body element is a block to avoid
|
|
// complicated things which may be detected by fuzzing.
|
|
if (aContent.OwnerDoc()->GetDocumentElement() == &aContent ||
|
|
(aContent.IsHTMLElement(nsGkAtoms::body) &&
|
|
aContent.OwnerDoc()->GetBodyElement() == &aContent)) {
|
|
return true;
|
|
}
|
|
RefPtr<const ComputedStyle> elementStyle =
|
|
nsComputedDOMStyle::GetComputedStyleNoFlush(aContent.AsElement());
|
|
if (MOZ_UNLIKELY(!elementStyle)) { // If aContent is not in the composed tree
|
|
return IsHTMLBlockElementByDefault(aContent);
|
|
}
|
|
const nsStyleDisplay* styleDisplay = elementStyle->StyleDisplay();
|
|
if (MOZ_UNLIKELY(styleDisplay->mDisplay == StyleDisplay::None)) {
|
|
// Typically, we should not keep handling editing in invisible nodes, but if
|
|
// we reach here, let's fallback to the default style for protecting the
|
|
// structure as far as possible.
|
|
return IsHTMLBlockElementByDefault(aContent);
|
|
}
|
|
// Both Blink and WebKit treat ruby style as a block, see IsEnclosingBlock()
|
|
// in Chromium or isBlock() in WebKit.
|
|
if (styleDisplay->IsRubyDisplayType()) {
|
|
return true;
|
|
}
|
|
// If the outside is not inline, treat it as block.
|
|
if (!styleDisplay->IsInlineOutsideStyle()) {
|
|
return true;
|
|
}
|
|
// If we're checking display-inside, inline-block, etc should be a block too.
|
|
return aBlockInlineCheck == BlockInlineCheck::UseComputedDisplayStyle &&
|
|
styleDisplay->DisplayInside() == StyleDisplayInside::FlowRoot &&
|
|
// Treat widgets as inline since they won't hide collapsible
|
|
// white-spaces around them.
|
|
styleDisplay->EffectiveAppearance() == StyleAppearance::None;
|
|
}
|
|
|
|
bool HTMLEditUtils::IsInlineContent(const nsIContent& aContent,
|
|
BlockInlineCheck aBlockInlineCheck) {
|
|
MOZ_ASSERT(aBlockInlineCheck != BlockInlineCheck::Unused);
|
|
|
|
if (!aContent.IsElement()) {
|
|
return true;
|
|
}
|
|
// If it's a <br>, we should always treat it as an inline element because
|
|
// its preceding collapse white-spaces and another <br> works same as usual
|
|
// even if you set its style to `display:block`.
|
|
if (aContent.IsHTMLElement(nsGkAtoms::br)) {
|
|
return true;
|
|
}
|
|
if (aBlockInlineCheck == BlockInlineCheck::UseHTMLDefaultStyle) {
|
|
return !IsHTMLBlockElementByDefault(aContent);
|
|
}
|
|
// Let's treat the document element and the body element is a block to avoid
|
|
// complicated things which may be detected by fuzzing.
|
|
if (aContent.OwnerDoc()->GetDocumentElement() == &aContent ||
|
|
(aContent.IsHTMLElement(nsGkAtoms::body) &&
|
|
aContent.OwnerDoc()->GetBodyElement() == &aContent)) {
|
|
return false;
|
|
}
|
|
RefPtr<const ComputedStyle> elementStyle =
|
|
nsComputedDOMStyle::GetComputedStyleNoFlush(aContent.AsElement());
|
|
if (MOZ_UNLIKELY(!elementStyle)) { // If aContent is not in the composed tree
|
|
return !IsHTMLBlockElementByDefault(aContent);
|
|
}
|
|
const nsStyleDisplay* styleDisplay = elementStyle->StyleDisplay();
|
|
if (MOZ_UNLIKELY(styleDisplay->mDisplay == StyleDisplay::None)) {
|
|
// Similar to IsBlockElement, let's fallback to refer the default style.
|
|
// Note that if you change here, you may need to check the parent element
|
|
// style if aContent.
|
|
return !IsHTMLBlockElementByDefault(aContent);
|
|
}
|
|
// Different block IsBlockElement, when the display-outside is inline, it's
|
|
// simply an inline element.
|
|
return styleDisplay->IsInlineOutsideStyle() ||
|
|
styleDisplay->IsRubyDisplayType();
|
|
}
|
|
|
|
bool HTMLEditUtils::IsFlexOrGridItem(const Element& aElement) {
|
|
nsIFrame* frame = aElement.GetPrimaryFrame();
|
|
return frame && frame->IsFlexOrGridItem();
|
|
}
|
|
|
|
bool HTMLEditUtils::IsInclusiveAncestorCSSDisplayNone(
|
|
const nsIContent& aContent) {
|
|
if (NS_WARN_IF(!aContent.IsInComposedDoc())) {
|
|
return true;
|
|
}
|
|
for (const Element* element :
|
|
aContent.InclusiveFlatTreeAncestorsOfType<Element>()) {
|
|
RefPtr<const ComputedStyle> elementStyle =
|
|
nsComputedDOMStyle::GetComputedStyleNoFlush(element);
|
|
if (NS_WARN_IF(!elementStyle)) {
|
|
continue;
|
|
}
|
|
const nsStyleDisplay* styleDisplay = elementStyle->StyleDisplay();
|
|
if (MOZ_UNLIKELY(styleDisplay->mDisplay == StyleDisplay::None)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool HTMLEditUtils::IsVisibleElementEvenIfLeafNode(const nsIContent& aContent) {
|
|
if (!aContent.IsElement()) {
|
|
return false;
|
|
}
|
|
// Assume non-HTML element is visible.
|
|
if (!aContent.IsHTMLElement()) {
|
|
return true;
|
|
}
|
|
// XXX Should we return false if the element is display:none?
|
|
if (HTMLEditUtils::IsBlockElement(
|
|
aContent, BlockInlineCheck::UseComputedDisplayStyle)) {
|
|
return true;
|
|
}
|
|
// <br> element may not have a frame, but it always affects surrounding
|
|
// content. Therefore, it should be treated as visible. The others which are
|
|
// checked here are replace elements which provide something visible content.
|
|
if (aContent.IsAnyOfHTMLElements(nsGkAtoms::applet, nsGkAtoms::br,
|
|
nsGkAtoms::iframe, nsGkAtoms::img,
|
|
nsGkAtoms::meter, nsGkAtoms::progress,
|
|
nsGkAtoms::select, nsGkAtoms::textarea)) {
|
|
return true;
|
|
}
|
|
if (const HTMLInputElement* inputElement =
|
|
HTMLInputElement::FromNode(&aContent)) {
|
|
return inputElement->ControlType() != FormControlType::InputHidden;
|
|
}
|
|
// If the element has a primary frame and it's not empty, the element is
|
|
// visible.
|
|
// XXX This method does not guarantee that the layout has already been
|
|
// updated. Therefore, this check might be wrong in the edge cases.
|
|
// However, basically, editor apps should not depend on this path, this
|
|
// is required if last <br> before a block boundary becomes visible because
|
|
// of followed by empty but styled frame like <span style=padding:1px></span>.
|
|
if (aContent.GetPrimaryFrame() &&
|
|
!aContent.GetPrimaryFrame()->GetSize().IsEmpty()) {
|
|
return true;
|
|
}
|
|
// Maybe, empty inline element such as <span>.
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* IsInlineStyle() returns true if aNode is an inline style.
|
|
*/
|
|
bool HTMLEditUtils::IsInlineStyle(nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
return aNode->IsAnyOfHTMLElements(
|
|
nsGkAtoms::b, nsGkAtoms::i, nsGkAtoms::u, nsGkAtoms::tt, nsGkAtoms::s,
|
|
nsGkAtoms::strike, nsGkAtoms::big, nsGkAtoms::small, nsGkAtoms::sub,
|
|
nsGkAtoms::sup, nsGkAtoms::font);
|
|
}
|
|
|
|
bool HTMLEditUtils::IsDisplayOutsideInline(const Element& aElement) {
|
|
RefPtr<const ComputedStyle> elementStyle =
|
|
nsComputedDOMStyle::GetComputedStyleNoFlush(&aElement);
|
|
if (!elementStyle) {
|
|
return false;
|
|
}
|
|
return elementStyle->StyleDisplay()->DisplayOutside() ==
|
|
StyleDisplayOutside::Inline;
|
|
}
|
|
|
|
bool HTMLEditUtils::IsDisplayInsideFlowRoot(const Element& aElement) {
|
|
RefPtr<const ComputedStyle> elementStyle =
|
|
nsComputedDOMStyle::GetComputedStyleNoFlush(&aElement);
|
|
if (!elementStyle) {
|
|
return false;
|
|
}
|
|
return elementStyle->StyleDisplay()->DisplayInside() ==
|
|
StyleDisplayInside::FlowRoot;
|
|
}
|
|
|
|
bool HTMLEditUtils::IsRemovableInlineStyleElement(Element& aElement) {
|
|
if (!aElement.IsHTMLElement()) {
|
|
return false;
|
|
}
|
|
// https://w3c.github.io/editing/execCommand.html#removeformat-candidate
|
|
if (aElement.IsAnyOfHTMLElements(
|
|
nsGkAtoms::abbr, // Chrome ignores, but does not make sense.
|
|
nsGkAtoms::acronym, nsGkAtoms::b,
|
|
nsGkAtoms::bdi, // Chrome ignores, but does not make sense.
|
|
nsGkAtoms::bdo, nsGkAtoms::big, nsGkAtoms::cite, nsGkAtoms::code,
|
|
// nsGkAtoms::del, Chrome ignores, but does not make sense but
|
|
// execCommand unofficial draft excludes this. Spec issue:
|
|
// https://github.com/w3c/editing/issues/192
|
|
nsGkAtoms::dfn, nsGkAtoms::em, nsGkAtoms::font, nsGkAtoms::i,
|
|
nsGkAtoms::ins, nsGkAtoms::kbd,
|
|
nsGkAtoms::mark, // Chrome ignores, but does not make sense.
|
|
nsGkAtoms::nobr, nsGkAtoms::q, nsGkAtoms::s, nsGkAtoms::samp,
|
|
nsGkAtoms::small, nsGkAtoms::span, nsGkAtoms::strike,
|
|
nsGkAtoms::strong, nsGkAtoms::sub, nsGkAtoms::sup, nsGkAtoms::tt,
|
|
nsGkAtoms::u, nsGkAtoms::var)) {
|
|
return true;
|
|
}
|
|
// If it's a <blink> element, we can remove it.
|
|
nsAutoString tagName;
|
|
aElement.GetTagName(tagName);
|
|
return tagName.LowerCaseEqualsASCII("blink");
|
|
}
|
|
|
|
/**
|
|
* IsNodeThatCanOutdent() returns true if aNode is a list, list item or
|
|
* blockquote.
|
|
*/
|
|
bool HTMLEditUtils::IsNodeThatCanOutdent(nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
return aNode->IsAnyOfHTMLElements(nsGkAtoms::ul, nsGkAtoms::ol, nsGkAtoms::dl,
|
|
nsGkAtoms::li, nsGkAtoms::dd, nsGkAtoms::dt,
|
|
nsGkAtoms::blockquote);
|
|
}
|
|
|
|
/**
|
|
* IsHeader() returns true if aNode is an html header.
|
|
*/
|
|
bool HTMLEditUtils::IsHeader(nsINode& aNode) {
|
|
return aNode.IsAnyOfHTMLElements(nsGkAtoms::h1, nsGkAtoms::h2, nsGkAtoms::h3,
|
|
nsGkAtoms::h4, nsGkAtoms::h5, nsGkAtoms::h6);
|
|
}
|
|
|
|
/**
|
|
* IsListItem() returns true if aNode is an html list item.
|
|
*/
|
|
bool HTMLEditUtils::IsListItem(const nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
return aNode->IsAnyOfHTMLElements(nsGkAtoms::li, nsGkAtoms::dd,
|
|
nsGkAtoms::dt);
|
|
}
|
|
|
|
/**
|
|
* IsAnyTableElement() returns true if aNode is an html table, td, tr, ...
|
|
*/
|
|
bool HTMLEditUtils::IsAnyTableElement(const nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
return aNode->IsAnyOfHTMLElements(
|
|
nsGkAtoms::table, nsGkAtoms::tr, nsGkAtoms::td, nsGkAtoms::th,
|
|
nsGkAtoms::thead, nsGkAtoms::tfoot, nsGkAtoms::tbody, nsGkAtoms::caption);
|
|
}
|
|
|
|
/**
|
|
* IsAnyTableElementButNotTable() returns true if aNode is an html td, tr, ...
|
|
* (doesn't include table)
|
|
*/
|
|
bool HTMLEditUtils::IsAnyTableElementButNotTable(nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
return aNode->IsAnyOfHTMLElements(nsGkAtoms::tr, nsGkAtoms::td, nsGkAtoms::th,
|
|
nsGkAtoms::thead, nsGkAtoms::tfoot,
|
|
nsGkAtoms::tbody, nsGkAtoms::caption);
|
|
}
|
|
|
|
/**
|
|
* IsTable() returns true if aNode is an html table.
|
|
*/
|
|
bool HTMLEditUtils::IsTable(nsINode* aNode) {
|
|
return aNode && aNode->IsHTMLElement(nsGkAtoms::table);
|
|
}
|
|
|
|
/**
|
|
* IsTableRow() returns true if aNode is an html tr.
|
|
*/
|
|
bool HTMLEditUtils::IsTableRow(nsINode* aNode) {
|
|
return aNode && aNode->IsHTMLElement(nsGkAtoms::tr);
|
|
}
|
|
|
|
/**
|
|
* IsTableCell() returns true if aNode is an html td or th.
|
|
*/
|
|
bool HTMLEditUtils::IsTableCell(const nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
return aNode->IsAnyOfHTMLElements(nsGkAtoms::td, nsGkAtoms::th);
|
|
}
|
|
|
|
/**
|
|
* IsTableCellOrCaption() returns true if aNode is an html td or th or caption.
|
|
*/
|
|
bool HTMLEditUtils::IsTableCellOrCaption(nsINode& aNode) {
|
|
return aNode.IsAnyOfHTMLElements(nsGkAtoms::td, nsGkAtoms::th,
|
|
nsGkAtoms::caption);
|
|
}
|
|
|
|
/**
|
|
* IsAnyListElement() returns true if aNode is an html list.
|
|
*/
|
|
bool HTMLEditUtils::IsAnyListElement(const nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
return aNode->IsAnyOfHTMLElements(nsGkAtoms::ul, nsGkAtoms::ol,
|
|
nsGkAtoms::dl);
|
|
}
|
|
|
|
/**
|
|
* IsPre() returns true if aNode is an html pre node.
|
|
*/
|
|
bool HTMLEditUtils::IsPre(const nsINode* aNode) {
|
|
return aNode && aNode->IsHTMLElement(nsGkAtoms::pre);
|
|
}
|
|
|
|
/**
|
|
* IsImage() returns true if aNode is an html image node.
|
|
*/
|
|
bool HTMLEditUtils::IsImage(nsINode* aNode) {
|
|
return aNode && aNode->IsHTMLElement(nsGkAtoms::img);
|
|
}
|
|
|
|
bool HTMLEditUtils::IsLink(const nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
|
|
if (!aNode->IsContent()) {
|
|
return false;
|
|
}
|
|
|
|
RefPtr<const dom::HTMLAnchorElement> anchor =
|
|
dom::HTMLAnchorElement::FromNodeOrNull(aNode->AsContent());
|
|
if (!anchor) {
|
|
return false;
|
|
}
|
|
|
|
nsAutoCString tmpText;
|
|
anchor->GetHref(tmpText);
|
|
return !tmpText.IsEmpty();
|
|
}
|
|
|
|
bool HTMLEditUtils::IsNamedAnchor(const nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
if (!aNode->IsHTMLElement(nsGkAtoms::a)) {
|
|
return false;
|
|
}
|
|
|
|
nsAutoString text;
|
|
return aNode->AsElement()->GetAttr(nsGkAtoms::name, text) && !text.IsEmpty();
|
|
}
|
|
|
|
/**
|
|
* IsMozDiv() returns true if aNode is an html div node with |type = _moz|.
|
|
*/
|
|
bool HTMLEditUtils::IsMozDiv(nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
return aNode->IsHTMLElement(nsGkAtoms::div) &&
|
|
aNode->AsElement()->AttrValueIs(kNameSpaceID_None, nsGkAtoms::type,
|
|
u"_moz"_ns, eIgnoreCase);
|
|
}
|
|
|
|
/**
|
|
* IsMailCite() returns true if aNode is an html blockquote with |type=cite|.
|
|
*/
|
|
bool HTMLEditUtils::IsMailCite(const Element& aElement) {
|
|
// don't ask me why, but our html mailcites are id'd by "type=cite"...
|
|
if (aElement.AttrValueIs(kNameSpaceID_None, nsGkAtoms::type, u"cite"_ns,
|
|
eIgnoreCase)) {
|
|
return true;
|
|
}
|
|
|
|
// ... but our plaintext mailcites by "_moz_quote=true". go figure.
|
|
if (aElement.AttrValueIs(kNameSpaceID_None, nsGkAtoms::mozquote, u"true"_ns,
|
|
eIgnoreCase)) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* IsFormWidget() returns true if aNode is a form widget of some kind.
|
|
*/
|
|
bool HTMLEditUtils::IsFormWidget(const nsINode* aNode) {
|
|
MOZ_ASSERT(aNode);
|
|
return aNode->IsAnyOfHTMLElements(nsGkAtoms::textarea, nsGkAtoms::select,
|
|
nsGkAtoms::button, nsGkAtoms::output,
|
|
nsGkAtoms::progress, nsGkAtoms::meter,
|
|
nsGkAtoms::input);
|
|
}
|
|
|
|
bool HTMLEditUtils::SupportsAlignAttr(nsINode& aNode) {
|
|
return aNode.IsAnyOfHTMLElements(
|
|
nsGkAtoms::hr, nsGkAtoms::table, nsGkAtoms::tbody, nsGkAtoms::tfoot,
|
|
nsGkAtoms::thead, nsGkAtoms::tr, nsGkAtoms::td, nsGkAtoms::th,
|
|
nsGkAtoms::div, nsGkAtoms::p, nsGkAtoms::h1, nsGkAtoms::h2, nsGkAtoms::h3,
|
|
nsGkAtoms::h4, nsGkAtoms::h5, nsGkAtoms::h6);
|
|
}
|
|
|
|
bool HTMLEditUtils::IsVisibleTextNode(const Text& aText) {
|
|
if (!aText.TextDataLength()) {
|
|
return false;
|
|
}
|
|
|
|
Maybe<uint32_t> visibleCharOffset =
|
|
HTMLEditUtils::GetInclusiveNextNonCollapsibleCharOffset(
|
|
EditorDOMPointInText(&aText, 0));
|
|
if (visibleCharOffset.isSome()) {
|
|
return true;
|
|
}
|
|
|
|
// Now, all characters in aText is collapsible white-spaces. The node is
|
|
// invisible if next to block boundary.
|
|
return !HTMLEditUtils::GetElementOfImmediateBlockBoundary(
|
|
aText, WalkTreeDirection::Forward) &&
|
|
!HTMLEditUtils::GetElementOfImmediateBlockBoundary(
|
|
aText, WalkTreeDirection::Backward);
|
|
}
|
|
|
|
bool HTMLEditUtils::IsInVisibleTextFrames(nsPresContext* aPresContext,
|
|
const Text& aText) {
|
|
// TODO(dholbert): aPresContext is now unused; maybe we can remove it, here
|
|
// and in IsEmptyNode? We do use it as a signal (implicitly here,
|
|
// more-explicitly in IsEmptyNode) that we are in a "SafeToAskLayout" case...
|
|
// If/when we remove it, we should be sure we're not losing that signal of
|
|
// strictness, since this function here does absolutely need to query layout.
|
|
MOZ_ASSERT(aPresContext);
|
|
|
|
if (!aText.TextDataLength()) {
|
|
return false;
|
|
}
|
|
|
|
nsTextFrame* textFrame = do_QueryFrame(aText.GetPrimaryFrame());
|
|
if (!textFrame) {
|
|
return false;
|
|
}
|
|
|
|
return textFrame->HasVisibleText();
|
|
}
|
|
|
|
template <typename PT, typename CT>
|
|
EditorDOMPoint HTMLEditUtils::LineRequiresPaddingLineBreakToBeVisible(
|
|
const EditorDOMPointBase<PT, CT>& aPoint, const Element& aEditingHost) {
|
|
if (MOZ_UNLIKELY(!aPoint.IsInContentNode())) {
|
|
return EditorDOMPoint();
|
|
}
|
|
MOZ_ASSERT(HTMLEditUtils::NodeIsEditableOrNotInComposedDoc(
|
|
*aPoint.template ContainerAs<nsIContent>()));
|
|
// First, if the container is an element node, get the next deepest point.
|
|
EditorRawDOMPoint point = aPoint.template To<EditorRawDOMPoint>();
|
|
if (point.IsContainerElement()) {
|
|
for (nsIContent* child = point.GetChild(); child;
|
|
child = child->GetFirstChild()) {
|
|
if (child->IsHTMLElement(nsGkAtoms::br)) {
|
|
return EditorDOMPoint();
|
|
}
|
|
if (!HTMLEditUtils::NodeIsEditableOrNotInComposedDoc(*child) ||
|
|
HTMLEditUtils::IsBlockElement(
|
|
*child, BlockInlineCheck::UseComputedDisplayOutsideStyle) ||
|
|
(child->IsElement() && !HTMLEditUtils::IsContainerNode(*child))) {
|
|
break;
|
|
}
|
|
point.Set(child, 0);
|
|
}
|
|
}
|
|
// If the point is in a Text, check the next character in it to skip the
|
|
// expensive check below.
|
|
if (point.IsInTextNode()) {
|
|
if (!point.IsStartOfContainer() &&
|
|
!point.IsPreviousCharCollapsibleASCIISpace()) {
|
|
return EditorDOMPoint(); // Not following collapsible white-space
|
|
}
|
|
if (!point.IsEndOfContainer()) {
|
|
if (!point.IsCharCollapsibleASCIISpace()) {
|
|
return EditorDOMPoint();
|
|
}
|
|
const bool linefeedPreformatted = EditorUtils::IsNewLinePreformatted(
|
|
*point.template ContainerAs<Text>());
|
|
const nsTextFragment& fragment =
|
|
point.template ContainerAs<Text>()->TextFragment();
|
|
for (uint32_t i : IntegerRange(point.Offset(), fragment.GetLength())) {
|
|
const char16_t ch = fragment.CharAt(i);
|
|
if (linefeedPreformatted && ch == HTMLEditUtils::kNewLine) {
|
|
return EditorDOMPoint(); // Followed by a preformatted line break.
|
|
}
|
|
if (!nsCRT::IsAsciiSpace(ch)) {
|
|
return EditorDOMPoint(); // Followed by a visible character.
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const auto AdjustPointToInsertPaddingLineBreak =
|
|
[](EditorDOMPoint& aPointToInsertLineBreak,
|
|
const Element* aParentBlockElement, const Element& aEditingHost) {
|
|
if (MOZ_UNLIKELY(!aPointToInsertLineBreak.IsInContentNode())) {
|
|
aPointToInsertLineBreak.Clear();
|
|
return;
|
|
}
|
|
while (MOZ_UNLIKELY(
|
|
!HTMLEditUtils::CanNodeContain(
|
|
*aPointToInsertLineBreak.GetContainer(), *nsGkAtoms::br) ||
|
|
!HTMLEditUtils::NodeIsEditableOrNotInComposedDoc(
|
|
*aPointToInsertLineBreak.GetContainer()))) {
|
|
if (MOZ_UNLIKELY(aPointToInsertLineBreak.GetContainer() ==
|
|
aParentBlockElement ||
|
|
aPointToInsertLineBreak.GetContainer() ==
|
|
&aEditingHost)) {
|
|
aPointToInsertLineBreak.Clear();
|
|
return;
|
|
}
|
|
aPointToInsertLineBreak.SetAfterContainer();
|
|
if (MOZ_UNLIKELY(!aPointToInsertLineBreak.IsInContentNode())) {
|
|
aPointToInsertLineBreak.Clear();
|
|
return;
|
|
}
|
|
}
|
|
};
|
|
|
|
// If the point is in an empty block, we can skip the expensive check below
|
|
// too.
|
|
const Element* maybeNonEditableBlock =
|
|
HTMLEditUtils::GetInclusiveAncestorElement(
|
|
*point.ContainerAs<nsIContent>(), ClosestBlockElement,
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
if (maybeNonEditableBlock &&
|
|
HTMLEditUtils::IsEmptyNode(
|
|
*maybeNonEditableBlock,
|
|
{EmptyCheckOption::TreatSingleBRElementAsVisible})) {
|
|
EditorDOMPoint pointToInsertLineBreak =
|
|
HTMLEditUtils::GetDeepestEditableEndPointOf<EditorDOMPoint>(
|
|
*maybeNonEditableBlock);
|
|
if (pointToInsertLineBreak.IsInTextNode()) {
|
|
pointToInsertLineBreak.SetAfterContainer();
|
|
}
|
|
AdjustPointToInsertPaddingLineBreak(pointToInsertLineBreak,
|
|
maybeNonEditableBlock, aEditingHost);
|
|
return pointToInsertLineBreak;
|
|
}
|
|
|
|
EditorDOMPoint preferredPaddingLineBreakPoint;
|
|
const bool followedByBlockBoundary = [&]() {
|
|
if (point.GetContainer() == maybeNonEditableBlock &&
|
|
point.IsEndOfContainer()) {
|
|
preferredPaddingLineBreakPoint = point.To<EditorDOMPoint>();
|
|
return true;
|
|
}
|
|
if (point.GetContainer() == &aEditingHost && point.IsEndOfContainer()) {
|
|
return false;
|
|
}
|
|
const WSScanResult nextThing =
|
|
WSRunScanner::ScanInclusiveNextVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::All, point,
|
|
BlockInlineCheck::UseComputedDisplayOutsideStyle);
|
|
if (nextThing.ReachedBlockBoundary()) {
|
|
if (nextThing.ReachedCurrentBlockBoundary()) {
|
|
preferredPaddingLineBreakPoint = point.AfterContainer<EditorDOMPoint>();
|
|
} else {
|
|
preferredPaddingLineBreakPoint = point.To<EditorDOMPoint>();
|
|
}
|
|
// FIXME: Scan an editable point to put a padding <br>.
|
|
if (NS_WARN_IF(!HTMLEditUtils::NodeIsEditableOrNotInComposedDoc(
|
|
*preferredPaddingLineBreakPoint.GetContainer()))) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}();
|
|
if (!followedByBlockBoundary) {
|
|
return EditorDOMPoint();
|
|
}
|
|
const bool followingBlockBoundaryOrCollapsibleWhiteSpace = [&]() {
|
|
if (point.GetContainer() == maybeNonEditableBlock &&
|
|
point.IsStartOfContainer()) {
|
|
return true;
|
|
}
|
|
const WSScanResult previousThing =
|
|
WSRunScanner::ScanPreviousVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::EditableNodes, preferredPaddingLineBreakPoint,
|
|
BlockInlineCheck::UseComputedDisplayOutsideStyle);
|
|
if (previousThing.ContentIsText()) {
|
|
if (MOZ_UNLIKELY(!previousThing.TextPtr()->TextDataLength())) {
|
|
return false;
|
|
}
|
|
auto atLastChar = EditorRawDOMPointInText(
|
|
previousThing.TextPtr(),
|
|
previousThing.TextPtr()->TextDataLength() - 1);
|
|
if (atLastChar.IsCharCollapsibleASCIISpace()) {
|
|
preferredPaddingLineBreakPoint.SetAfter(previousThing.TextPtr());
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
return previousThing.ReachedBlockBoundary();
|
|
}();
|
|
if (!followingBlockBoundaryOrCollapsibleWhiteSpace) {
|
|
return EditorDOMPoint();
|
|
}
|
|
AdjustPointToInsertPaddingLineBreak(preferredPaddingLineBreakPoint,
|
|
maybeNonEditableBlock, aEditingHost);
|
|
return preferredPaddingLineBreakPoint;
|
|
}
|
|
|
|
Element* HTMLEditUtils::GetElementOfImmediateBlockBoundary(
|
|
const nsIContent& aContent, const WalkTreeDirection aDirection) {
|
|
MOZ_ASSERT(aContent.IsHTMLElement(nsGkAtoms::br) || aContent.IsText());
|
|
|
|
// First, we get a block container. This is not designed for reaching
|
|
// no block boundaries in the tree.
|
|
Element* maybeNonEditableAncestorBlock = HTMLEditUtils::GetAncestorElement(
|
|
aContent, HTMLEditUtils::ClosestBlockElement,
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
if (NS_WARN_IF(!maybeNonEditableAncestorBlock)) {
|
|
return nullptr;
|
|
}
|
|
|
|
auto getNextContent = [&aDirection, &maybeNonEditableAncestorBlock](
|
|
const nsIContent& aContent) -> nsIContent* {
|
|
return aDirection == WalkTreeDirection::Forward
|
|
? HTMLEditUtils::GetNextContent(
|
|
aContent,
|
|
{WalkTreeOption::IgnoreDataNodeExceptText,
|
|
WalkTreeOption::StopAtBlockBoundary},
|
|
BlockInlineCheck::UseComputedDisplayStyle,
|
|
maybeNonEditableAncestorBlock)
|
|
: HTMLEditUtils::GetPreviousContent(
|
|
aContent,
|
|
{WalkTreeOption::IgnoreDataNodeExceptText,
|
|
WalkTreeOption::StopAtBlockBoundary},
|
|
BlockInlineCheck::UseComputedDisplayStyle,
|
|
maybeNonEditableAncestorBlock);
|
|
};
|
|
|
|
// Then, scan block element boundary while we don't see visible things.
|
|
const bool isBRElement = aContent.IsHTMLElement(nsGkAtoms::br);
|
|
for (nsIContent* nextContent = getNextContent(aContent); nextContent;
|
|
nextContent = getNextContent(*nextContent)) {
|
|
if (nextContent->IsElement()) {
|
|
// Break is right before a child block, it's not visible
|
|
if (HTMLEditUtils::IsBlockElement(
|
|
*nextContent, BlockInlineCheck::UseComputedDisplayStyle)) {
|
|
return nextContent->AsElement();
|
|
}
|
|
|
|
// XXX How about other non-HTML elements? Assume they are styled as
|
|
// blocks for now.
|
|
if (!nextContent->IsHTMLElement()) {
|
|
return nextContent->AsElement();
|
|
}
|
|
|
|
if (nextContent->IsHTMLElement(nsGkAtoms::br)) {
|
|
// If aContent is a <br> element, another <br> element prevents the
|
|
// block boundary special handling.
|
|
if (isBRElement) {
|
|
return nullptr;
|
|
}
|
|
|
|
MOZ_ASSERT(aContent.IsText());
|
|
// Following <br> element always hides its following block boundary.
|
|
// I.e., white-spaces is at end of the text node is visible.
|
|
if (aDirection == WalkTreeDirection::Forward) {
|
|
return nullptr;
|
|
}
|
|
// Otherwise, if text node follows <br> element, its white-spaces at
|
|
// start of the text node are invisible. In this case, we return
|
|
// the found <br> element.
|
|
return nextContent->AsElement();
|
|
}
|
|
|
|
// If there is a visible content which generates something visible,
|
|
// stop scanning.
|
|
if (HTMLEditUtils::IsVisibleElementEvenIfLeafNode(*nextContent)) {
|
|
return nullptr;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
switch (nextContent->NodeType()) {
|
|
case nsINode::TEXT_NODE:
|
|
case nsINode::CDATA_SECTION_NODE:
|
|
break;
|
|
default:
|
|
continue;
|
|
}
|
|
|
|
Text* textNode = Text::FromNode(nextContent);
|
|
MOZ_ASSERT(textNode);
|
|
if (!textNode->TextLength()) {
|
|
continue; // empty invisible text node, keep scanning next one.
|
|
}
|
|
if (HTMLEditUtils::IsInclusiveAncestorCSSDisplayNone(*textNode)) {
|
|
continue; // Styled as invisible.
|
|
}
|
|
if (!textNode->TextIsOnlyWhitespace()) {
|
|
return nullptr; // found a visible text node.
|
|
}
|
|
const nsTextFragment& textFragment = textNode->TextFragment();
|
|
const bool isWhiteSpacePreformatted =
|
|
EditorUtils::IsWhiteSpacePreformatted(*textNode);
|
|
const bool isNewLinePreformatted =
|
|
EditorUtils::IsNewLinePreformatted(*textNode);
|
|
if (!isWhiteSpacePreformatted && !isNewLinePreformatted) {
|
|
// if the white-space only text node is not preformatted, ignore it.
|
|
continue;
|
|
}
|
|
for (uint32_t i = 0; i < textFragment.GetLength(); i++) {
|
|
if (textFragment.CharAt(i) == HTMLEditUtils::kNewLine) {
|
|
if (isNewLinePreformatted) {
|
|
return nullptr; // found a visible text node.
|
|
}
|
|
continue;
|
|
}
|
|
if (isWhiteSpacePreformatted) {
|
|
return nullptr; // found a visible text node.
|
|
}
|
|
}
|
|
// All white-spaces in the text node is invisible, keep scanning next one.
|
|
}
|
|
|
|
// There is no visible content and reached current block boundary. Then,
|
|
// the <br> element is the last content in the block and invisible.
|
|
// XXX Should we treat it visible if it's the only child of a block?
|
|
return maybeNonEditableAncestorBlock;
|
|
}
|
|
|
|
template <typename PT, typename CT>
|
|
bool HTMLEditUtils::PointIsImmediatelyBeforeCurrentBlockBoundary(
|
|
const EditorDOMPointBase<PT, CT>& aPoint,
|
|
IgnoreInvisibleLineBreak aIgnoreInvisibleLineBreak) {
|
|
MOZ_ASSERT(aPoint.IsSetAndValidInComposedDoc());
|
|
|
|
if (MOZ_UNLIKELY(!aPoint.IsInContentNode())) {
|
|
return false;
|
|
}
|
|
const WSScanResult nextThing =
|
|
WSRunScanner::ScanInclusiveNextVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::EditableNodes, aPoint,
|
|
BlockInlineCheck::UseComputedDisplayOutsideStyle);
|
|
if (nextThing.ReachedCurrentBlockBoundary()) {
|
|
return true;
|
|
}
|
|
if (nextThing.ReachedInvisibleBRElement()) {
|
|
if (aIgnoreInvisibleLineBreak == IgnoreInvisibleLineBreak::No) {
|
|
return false;
|
|
}
|
|
const WSScanResult afterInvisibleBRThing =
|
|
WSRunScanner::ScanInclusiveNextVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::EditableNodes,
|
|
nextThing.PointAfterReachedContent<EditorRawDOMPoint>(),
|
|
BlockInlineCheck::UseComputedDisplayOutsideStyle);
|
|
return afterInvisibleBRThing.ReachedCurrentBlockBoundary();
|
|
}
|
|
if (nextThing.ReachedPreformattedLineBreak()) {
|
|
if (aIgnoreInvisibleLineBreak == IgnoreInvisibleLineBreak::No) {
|
|
return false;
|
|
}
|
|
const WSScanResult afterPreformattedLineBreakThing =
|
|
WSRunScanner::ScanInclusiveNextVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::EditableNodes,
|
|
nextThing.PointAfterReachedContent<EditorRawDOMPoint>(),
|
|
BlockInlineCheck::UseComputedDisplayOutsideStyle);
|
|
return afterPreformattedLineBreakThing.ReachedCurrentBlockBoundary();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
template <typename EditorLineBreakType>
|
|
Maybe<EditorLineBreakType> HTMLEditUtils::GetUnnecessaryLineBreak(
|
|
const Element& aBlockElement, ScanLineBreak aScanLineBreak) {
|
|
auto* lastLineBreakContent = [&]() -> nsIContent* {
|
|
const WalkTreeOptions onlyPrecedingLine{
|
|
WalkTreeOption::StopAtBlockBoundary};
|
|
for (nsIContent* content =
|
|
aScanLineBreak == ScanLineBreak::AtEndOfBlock
|
|
? HTMLEditUtils::GetLastLeafContent(
|
|
aBlockElement, {LeafNodeType::OnlyLeafNode})
|
|
: HTMLEditUtils::GetPreviousContent(
|
|
aBlockElement, onlyPrecedingLine,
|
|
BlockInlineCheck::UseComputedDisplayStyle,
|
|
aBlockElement.GetParentElement());
|
|
content;
|
|
content =
|
|
aScanLineBreak == ScanLineBreak::AtEndOfBlock
|
|
? HTMLEditUtils::GetPreviousLeafContentOrPreviousBlockElement(
|
|
*content, {LeafNodeType::OnlyLeafNode},
|
|
BlockInlineCheck::UseComputedDisplayStyle,
|
|
&aBlockElement)
|
|
: HTMLEditUtils::GetPreviousContent(
|
|
*content, onlyPrecedingLine,
|
|
BlockInlineCheck::UseComputedDisplayStyle,
|
|
aBlockElement.GetParentElement())) {
|
|
// If we're scanning preceding <br> element of aBlockElement, we don't
|
|
// need to look for a line break in another block because the caller
|
|
// needs to handle only preceding <br> element of aBlockElement.
|
|
if (aScanLineBreak == ScanLineBreak::BeforeBlock &&
|
|
HTMLEditUtils::IsBlockElement(
|
|
*content, BlockInlineCheck::UseComputedDisplayStyle)) {
|
|
return nullptr;
|
|
}
|
|
if (Text* textNode = Text::FromNode(content)) {
|
|
if (!textNode->TextLength()) {
|
|
continue; // ignore empty text node
|
|
}
|
|
const nsTextFragment& textFragment = textNode->TextFragment();
|
|
if (EditorUtils::IsNewLinePreformatted(*textNode) &&
|
|
textFragment.CharAt(textFragment.GetLength() - 1u) ==
|
|
HTMLEditUtils::kNewLine) {
|
|
// If the text node ends with a preserved line break, it's unnecessary
|
|
// unless it follows another preformatted line break.
|
|
if (textFragment.GetLength() == 1u) {
|
|
return textNode; // Need to scan previous leaf.
|
|
}
|
|
return textFragment.CharAt(textFragment.GetLength() - 2u) ==
|
|
HTMLEditUtils::kNewLine
|
|
? nullptr
|
|
: textNode;
|
|
}
|
|
if (HTMLEditUtils::IsVisibleTextNode(*textNode)) {
|
|
return nullptr;
|
|
}
|
|
continue;
|
|
}
|
|
if (content->IsCharacterData()) {
|
|
continue; // ignore hidden character data nodes like comment
|
|
}
|
|
if (content->IsHTMLElement(nsGkAtoms::br)) {
|
|
return content;
|
|
}
|
|
if (HTMLEditUtils::IsVisibleElementEvenIfLeafNode(*content)) {
|
|
return nullptr;
|
|
}
|
|
// Otherwise, e.g., empty <b>, we should keep scanning.
|
|
}
|
|
return nullptr;
|
|
}();
|
|
if (!lastLineBreakContent) {
|
|
return Nothing();
|
|
}
|
|
|
|
// If the found node is a text node and contains only one preformatted new
|
|
// line break, we need to keep scanning previous one, but if it has 2 or more
|
|
// characters, we know it has redundant line break.
|
|
Text* const lastLineBreakText = Text::FromNode(lastLineBreakContent);
|
|
if (lastLineBreakText && lastLineBreakText->TextDataLength() != 1u) {
|
|
return Some(EditorLineBreakType::AtLastChar(*lastLineBreakText));
|
|
}
|
|
HTMLBRElement* const lastBRElement =
|
|
lastLineBreakText ? nullptr
|
|
: HTMLBRElement::FromNode(lastLineBreakContent);
|
|
MOZ_ASSERT_IF(!lastLineBreakText, lastBRElement);
|
|
|
|
// Scan previous leaf content, but now, we can stop at child block boundary.
|
|
const Element* blockElement = HTMLEditUtils::GetAncestorElement(
|
|
*lastLineBreakContent, HTMLEditUtils::ClosestBlockElement,
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
for (nsIContent* content =
|
|
HTMLEditUtils::GetPreviousLeafContentOrPreviousBlockElement(
|
|
*lastLineBreakContent, {LeafNodeType::LeafNodeOrChildBlock},
|
|
BlockInlineCheck::UseComputedDisplayStyle, blockElement);
|
|
content;
|
|
content = HTMLEditUtils::GetPreviousLeafContentOrPreviousBlockElement(
|
|
*content, {LeafNodeType::LeafNodeOrChildBlock},
|
|
BlockInlineCheck::UseComputedDisplayStyle, blockElement)) {
|
|
if (HTMLEditUtils::IsBlockElement(
|
|
*content, BlockInlineCheck::UseComputedDisplayStyle) ||
|
|
(content->IsElement() && !content->IsHTMLElement())) {
|
|
// Now, must found <div>...<div>...</div><br></div>
|
|
// ^^^^
|
|
// In this case, the <br> element is necessary to make a following empty
|
|
// line of the inner <div> visible.
|
|
return Nothing();
|
|
}
|
|
if (Text* textNode = Text::FromNode(content)) {
|
|
if (!textNode->TextDataLength()) {
|
|
continue; // ignore empty text node
|
|
}
|
|
const nsTextFragment& textFragment = textNode->TextFragment();
|
|
if (EditorUtils::IsNewLinePreformatted(*textNode) &&
|
|
textFragment.CharAt(textFragment.GetLength() - 1u) ==
|
|
HTMLEditUtils::kNewLine) {
|
|
// So, we are here because the preformatted line break is followed by
|
|
// lastLineBreakContent which is <br> or a text node containing only
|
|
// one. In this case, even if their parents are different,
|
|
// lastLineBreakContent is necessary to make the last line visible.
|
|
return Nothing();
|
|
}
|
|
if (!HTMLEditUtils::IsVisibleTextNode(*textNode)) {
|
|
continue;
|
|
}
|
|
if (EditorUtils::IsWhiteSpacePreformatted(*textNode)) {
|
|
// If the white-space is preserved, neither following <br> nor a
|
|
// preformatted line break is not necessary.
|
|
return Some(lastLineBreakText
|
|
? EditorLineBreakType::AtLastChar(*lastLineBreakText)
|
|
: EditorLineBreakType(*lastBRElement));
|
|
}
|
|
// Otherwise, only if the last character is a collapsible white-space,
|
|
// we need lastLineBreakContent to make the trailing white-space visible.
|
|
switch (textFragment.CharAt(textFragment.GetLength() - 1u)) {
|
|
case HTMLEditUtils::kSpace:
|
|
case HTMLEditUtils::kNewLine:
|
|
case HTMLEditUtils::kCarriageReturn:
|
|
case HTMLEditUtils::kTab:
|
|
return Nothing();
|
|
default:
|
|
return Some(lastLineBreakText
|
|
? EditorLineBreakType::AtLastChar(*lastLineBreakText)
|
|
: EditorLineBreakType(*lastBRElement));
|
|
}
|
|
}
|
|
if (content->IsCharacterData()) {
|
|
continue; // ignore hidden character data nodes like comment
|
|
}
|
|
// If lastLineBreakContent follows a <br> element in same block, it's
|
|
// necessary to make the empty last line visible.
|
|
if (content->IsHTMLElement(nsGkAtoms::br)) {
|
|
return Nothing();
|
|
}
|
|
if (HTMLEditUtils::IsVisibleElementEvenIfLeafNode(*content)) {
|
|
return Some(lastLineBreakText
|
|
? EditorLineBreakType::AtLastChar(*lastLineBreakText)
|
|
: EditorLineBreakType(*lastBRElement));
|
|
}
|
|
// Otherwise, ignore empty inline elements such as <b>.
|
|
}
|
|
// If the block is empty except invisible data nodes and lastLineBreakContent,
|
|
// lastLineBreakContent is necessary to make the block visible.
|
|
return Nothing();
|
|
}
|
|
|
|
template <typename EditorLineBreakType, typename EditorDOMPointType>
|
|
Maybe<EditorLineBreakType> HTMLEditUtils::GetFollowingUnnecessaryLineBreak(
|
|
const EditorDOMPointType& aPoint) {
|
|
MOZ_ASSERT(aPoint.IsSetAndValid());
|
|
MOZ_ASSERT(aPoint.IsInContentNode());
|
|
|
|
const WSScanResult nextThing =
|
|
WSRunScanner::ScanInclusiveNextVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::All, aPoint,
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
if (!nextThing.ReachedBRElement() &&
|
|
!(nextThing.ReachedPreformattedLineBreak() &&
|
|
nextThing.PointAtReachedContent<EditorRawDOMPoint>()
|
|
.IsAtLastContent())) {
|
|
return Nothing(); // no line break next to aPoint
|
|
}
|
|
const WSScanResult nextThingOfLineBreak =
|
|
WSRunScanner::ScanInclusiveNextVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::All,
|
|
nextThing.PointAfterReachedContent<EditorRawDOMPoint>(),
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
const Element* const blockElement =
|
|
nextThingOfLineBreak.ReachedBlockBoundary()
|
|
? nextThingOfLineBreak.ElementPtr()
|
|
: HTMLEditUtils::GetAncestorElement(
|
|
*nextThing.GetContent(), {AncestorType::ClosestBlockElement},
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
if (MOZ_UNLIKELY(!blockElement)) {
|
|
return Nothing();
|
|
}
|
|
Maybe<EditorLineBreakType> unnecessaryLineBreak =
|
|
GetUnnecessaryLineBreak<EditorLineBreakType>(
|
|
*blockElement, nextThingOfLineBreak.ReachedOtherBlockElement()
|
|
? ScanLineBreak::BeforeBlock
|
|
: ScanLineBreak::AtEndOfBlock);
|
|
// If the line break content is different from the found line break
|
|
// immediately after aPoint, it's too far. So, the caller should not touch it.
|
|
if (unnecessaryLineBreak.isSome() &&
|
|
&unnecessaryLineBreak->ContentRef() != nextThing.GetContent()) {
|
|
unnecessaryLineBreak.reset();
|
|
}
|
|
return unnecessaryLineBreak;
|
|
}
|
|
|
|
uint32_t HTMLEditUtils::GetFirstVisibleCharOffset(const Text& aText) {
|
|
const nsTextFragment& textFragment = aText.TextFragment();
|
|
if (!textFragment.GetLength() || !EditorRawDOMPointInText(&aText, 0u)
|
|
.IsCharCollapsibleASCIISpaceOrNBSP()) {
|
|
return 0u;
|
|
}
|
|
const WSScanResult previousThingOfText =
|
|
WSRunScanner::ScanPreviousVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::All, EditorRawDOMPoint(&aText),
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
if (!previousThingOfText.ReachedLineBoundary()) {
|
|
return 0u;
|
|
}
|
|
return HTMLEditUtils::GetInclusiveNextNonCollapsibleCharOffset(aText, 0u)
|
|
.valueOr(textFragment.GetLength());
|
|
}
|
|
|
|
uint32_t HTMLEditUtils::GetOffsetAfterLastVisibleChar(const Text& aText) {
|
|
const nsTextFragment& textFragment = aText.TextFragment();
|
|
if (!textFragment.GetLength()) {
|
|
return 0u;
|
|
}
|
|
if (!EditorRawDOMPointInText::AtLastContentOf(aText)
|
|
.IsCharCollapsibleASCIISpaceOrNBSP()) {
|
|
return textFragment.GetLength();
|
|
}
|
|
const WSScanResult nextThingOfText =
|
|
WSRunScanner::ScanInclusiveNextVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::All, EditorRawDOMPoint::After(aText),
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
if (!nextThingOfText.ReachedLineBoundary()) {
|
|
return textFragment.GetLength();
|
|
}
|
|
const Maybe<uint32_t> lastNonCollapsibleCharOffset =
|
|
HTMLEditUtils::GetPreviousNonCollapsibleCharOffset(
|
|
aText, textFragment.GetLength());
|
|
if (lastNonCollapsibleCharOffset.isNothing()) {
|
|
return 0u;
|
|
}
|
|
if (*lastNonCollapsibleCharOffset == textFragment.GetLength() - 1u) {
|
|
return textFragment.GetLength();
|
|
}
|
|
const uint32_t firstTrailingWhiteSpaceOffset =
|
|
*lastNonCollapsibleCharOffset + 1u;
|
|
MOZ_ASSERT(firstTrailingWhiteSpaceOffset < textFragment.GetLength());
|
|
if (nextThingOfText.ReachedBlockBoundary()) {
|
|
return firstTrailingWhiteSpaceOffset;
|
|
}
|
|
// If followed by <br> or preformatted line break, one white-space is
|
|
// rendered.
|
|
return firstTrailingWhiteSpaceOffset + 1u;
|
|
}
|
|
|
|
uint32_t HTMLEditUtils::GetInvisibleWhiteSpaceCount(
|
|
const Text& aText, uint32_t aOffset /* = 0u */,
|
|
uint32_t aLength /* = UINT32_MAX */) {
|
|
const nsTextFragment& textFragment = aText.TextFragment();
|
|
if (!aLength || textFragment.GetLength() <= aOffset) {
|
|
return 0u;
|
|
}
|
|
const uint32_t endOffset = static_cast<uint32_t>(
|
|
std::min(static_cast<uint64_t>(aOffset) + aLength,
|
|
static_cast<uint64_t>(textFragment.GetLength())));
|
|
const auto firstVisibleOffset = [&]() -> uint32_t {
|
|
// If the white-space sequence follows a preformatted linebreak, ASCII
|
|
// spaces at start are invisible.
|
|
if (aOffset &&
|
|
textFragment.CharAt(aOffset - 1u) == HTMLEditUtils::kNewLine) {
|
|
MOZ_ASSERT(EditorUtils::IsNewLinePreformatted(aText));
|
|
for (const uint32_t offset : IntegerRange(aOffset, endOffset)) {
|
|
if (textFragment.CharAt(offset) == HTMLEditUtils::kNBSP) {
|
|
return offset;
|
|
}
|
|
}
|
|
return endOffset; // all white-spaces are invisible.
|
|
}
|
|
if (aOffset) {
|
|
return aOffset - 1u;
|
|
}
|
|
return HTMLEditUtils::GetFirstVisibleCharOffset(aText);
|
|
}();
|
|
if (firstVisibleOffset >= endOffset) {
|
|
return endOffset - aOffset; // All white-spaces are invisible.
|
|
}
|
|
const auto afterLastVisibleOffset = [&]() -> uint32_t {
|
|
// If the white-spaces are followed by a preformatted line break, ASCII
|
|
// spaces at end are invisible.
|
|
if (endOffset < textFragment.GetLength() &&
|
|
textFragment.CharAt(endOffset) == HTMLEditUtils::kNewLine) {
|
|
MOZ_ASSERT(EditorUtils::IsNewLinePreformatted(aText));
|
|
for (const uint32_t offset : Reversed(IntegerRange(aOffset, endOffset))) {
|
|
if (textFragment.CharAt(offset) == HTMLEditUtils::kNBSP) {
|
|
return offset + 1u;
|
|
}
|
|
}
|
|
return aOffset; // all white-spaces are invisible.
|
|
}
|
|
if (endOffset < textFragment.GetLength() - 1u) {
|
|
return endOffset;
|
|
}
|
|
return HTMLEditUtils::GetOffsetAfterLastVisibleChar(aText);
|
|
}();
|
|
if (aOffset >= afterLastVisibleOffset) {
|
|
return endOffset - aOffset; // All white-spaces are invisible.
|
|
}
|
|
enum class PrevChar { NotChar, Space, NBSP };
|
|
PrevChar prevChar = PrevChar::NotChar;
|
|
uint32_t invisibleChars = 0u;
|
|
for (const uint32_t offset : IntegerRange(aOffset, endOffset)) {
|
|
if (textFragment.CharAt(offset) == HTMLEditUtils::kNBSP) {
|
|
prevChar = PrevChar::NBSP;
|
|
continue;
|
|
}
|
|
MOZ_ASSERT(
|
|
EditorRawDOMPointInText(&aText, offset).IsCharCollapsibleASCIISpace());
|
|
if (offset < firstVisibleOffset || offset >= afterLastVisibleOffset ||
|
|
// white-space after another white-space is invisible
|
|
prevChar == PrevChar::Space) {
|
|
invisibleChars++;
|
|
}
|
|
prevChar = PrevChar::Space;
|
|
}
|
|
return invisibleChars;
|
|
}
|
|
|
|
bool HTMLEditUtils::IsEmptyNode(nsPresContext* aPresContext,
|
|
const nsINode& aNode,
|
|
const EmptyCheckOptions& aOptions /* = {} */,
|
|
bool* aSeenBR /* = nullptr */) {
|
|
MOZ_ASSERT_IF(aOptions.contains(EmptyCheckOption::SafeToAskLayout),
|
|
aPresContext);
|
|
|
|
if (aSeenBR) {
|
|
*aSeenBR = false;
|
|
}
|
|
|
|
if (const Text* text = Text::FromNode(&aNode)) {
|
|
return aOptions.contains(EmptyCheckOption::SafeToAskLayout)
|
|
? !IsInVisibleTextFrames(aPresContext, *text)
|
|
: !IsVisibleTextNode(*text);
|
|
}
|
|
|
|
if (!aNode.IsElement()) {
|
|
return false;
|
|
}
|
|
|
|
if (
|
|
// If it's not a container such as an <hr> or <br>, etc, it should be
|
|
// treated as not empty.
|
|
!IsContainerNode(*aNode.AsContent()) ||
|
|
// If it's a named anchor, we shouldn't treat it as empty because it
|
|
// has special meaning even if invisible.
|
|
IsNamedAnchor(&aNode) ||
|
|
// Form widgets should be treated as not empty because they have special
|
|
// meaning even if invisible.
|
|
IsFormWidget(&aNode)) {
|
|
return false;
|
|
}
|
|
|
|
const auto [isListItem, isTableCell, hasAppearance] =
|
|
[&]() MOZ_NEVER_INLINE_DEBUG -> std::tuple<bool, bool, bool> {
|
|
// Let's stop treating the document element and the <body> as a list item
|
|
// nor a table cell to avoid tricky cases.
|
|
if (aNode.OwnerDoc()->GetDocumentElement() == &aNode ||
|
|
(aNode.IsHTMLElement(nsGkAtoms::body) &&
|
|
aNode.OwnerDoc()->GetBodyElement() == &aNode)) {
|
|
return {false, false, false};
|
|
}
|
|
|
|
RefPtr<const ComputedStyle> elementStyle =
|
|
nsComputedDOMStyle::GetComputedStyleNoFlush(aNode.AsElement());
|
|
// If there is no style information like in a document fragment, let's refer
|
|
// the default style.
|
|
if (MOZ_UNLIKELY(!elementStyle)) {
|
|
return {IsListItem(&aNode), IsTableCell(&aNode), false};
|
|
}
|
|
const nsStyleDisplay* styleDisplay = elementStyle->StyleDisplay();
|
|
if (NS_WARN_IF(!styleDisplay)) {
|
|
return {IsListItem(&aNode), IsTableCell(&aNode), false};
|
|
}
|
|
if (styleDisplay->mDisplay != StyleDisplay::None &&
|
|
styleDisplay->HasAppearance()) {
|
|
return {false, false, true};
|
|
}
|
|
if (styleDisplay->IsListItem()) {
|
|
return {true, false, false};
|
|
}
|
|
if (styleDisplay->mDisplay == StyleDisplay::TableCell) {
|
|
return {false, true, false};
|
|
}
|
|
// The default display of <dt> and <dd> is block. Therefore, we need
|
|
// special handling for them.
|
|
return {styleDisplay->mDisplay == StyleDisplay::Block &&
|
|
aNode.IsAnyOfHTMLElements(nsGkAtoms::dd, nsGkAtoms::dt),
|
|
false, false};
|
|
}();
|
|
|
|
// The web author created native widget without form control elements. Let's
|
|
// treat it as visible.
|
|
if (hasAppearance) {
|
|
return false;
|
|
}
|
|
|
|
if (isListItem &&
|
|
aOptions.contains(EmptyCheckOption::TreatListItemAsVisible)) {
|
|
return false;
|
|
}
|
|
if (isTableCell &&
|
|
aOptions.contains(EmptyCheckOption::TreatTableCellAsVisible)) {
|
|
return false;
|
|
}
|
|
|
|
bool seenBR = aSeenBR && *aSeenBR;
|
|
for (nsIContent* childContent = aNode.GetFirstChild(); childContent;
|
|
childContent = childContent->GetNextSibling()) {
|
|
// Is the child editable and non-empty? if so, return false
|
|
if (aOptions.contains(
|
|
EmptyCheckOption::TreatNonEditableContentAsInvisible) &&
|
|
!EditorUtils::IsEditableContent(*childContent, EditorType::HTML)) {
|
|
continue;
|
|
}
|
|
|
|
if (Text* text = Text::FromNode(childContent)) {
|
|
// break out if we find we aren't empty
|
|
if (aOptions.contains(EmptyCheckOption::SafeToAskLayout)
|
|
? IsInVisibleTextFrames(aPresContext, *text)
|
|
: IsVisibleTextNode(*text)) {
|
|
return false;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
MOZ_ASSERT(childContent != &aNode);
|
|
|
|
if (!aOptions.contains(EmptyCheckOption::TreatSingleBRElementAsVisible) &&
|
|
!seenBR && childContent->IsHTMLElement(nsGkAtoms::br)) {
|
|
// Ignore first <br> element in it if caller wants so because it's
|
|
// typically a padding <br> element of for a parent block.
|
|
seenBR = true;
|
|
if (aSeenBR) {
|
|
*aSeenBR = true;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (aOptions.contains(EmptyCheckOption::TreatBlockAsVisible) &&
|
|
HTMLEditUtils::IsBlockElement(
|
|
*childContent, BlockInlineCheck::UseComputedDisplayOutsideStyle)) {
|
|
return false;
|
|
}
|
|
|
|
// Note: list items or table cells are not considered empty
|
|
// if they contain other lists or tables
|
|
EmptyCheckOptions options(aOptions);
|
|
if (childContent->IsElement() && (isListItem || isTableCell)) {
|
|
options += {EmptyCheckOption::TreatListItemAsVisible,
|
|
EmptyCheckOption::TreatTableCellAsVisible};
|
|
}
|
|
if (!IsEmptyNode(aPresContext, *childContent, options, &seenBR)) {
|
|
if (aSeenBR) {
|
|
*aSeenBR = seenBR;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (aSeenBR) {
|
|
*aSeenBR = seenBR;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool HTMLEditUtils::ShouldInsertLinefeedCharacter(
|
|
const EditorDOMPoint& aPointToInsert, const Element& aEditingHost) {
|
|
MOZ_ASSERT(aPointToInsert.IsSetAndValid());
|
|
|
|
if (!aPointToInsert.IsInContentNode()) {
|
|
return false;
|
|
}
|
|
|
|
// If in contenteditable=plaintext-only, we should use linefeed when it's
|
|
// preformatted.
|
|
if (aEditingHost.IsContentEditablePlainTextOnly()) {
|
|
return EditorUtils::IsNewLinePreformatted(
|
|
*aPointToInsert.ContainerAs<nsIContent>());
|
|
}
|
|
|
|
// closestEditableBlockElement can be nullptr if aEditingHost is an inline
|
|
// element.
|
|
Element* closestEditableBlockElement =
|
|
HTMLEditUtils::GetInclusiveAncestorElement(
|
|
*aPointToInsert.ContainerAs<nsIContent>(),
|
|
HTMLEditUtils::ClosestEditableBlockElement,
|
|
BlockInlineCheck::UseComputedDisplayOutsideStyle);
|
|
|
|
// If and only if the nearest block is the editing host or its parent,
|
|
// and new line character is preformatted, we should insert a linefeed.
|
|
return (!closestEditableBlockElement ||
|
|
closestEditableBlockElement == &aEditingHost) &&
|
|
EditorUtils::IsNewLinePreformatted(
|
|
*aPointToInsert.ContainerAs<nsIContent>());
|
|
}
|
|
|
|
// We use bitmasks to test containment of elements. Elements are marked to be
|
|
// in certain groups by setting the mGroup member of the `ElementInfo` struct
|
|
// to the corresponding GROUP_ values (OR'ed together). Similarly, elements are
|
|
// marked to allow containment of certain groups by setting the
|
|
// mCanContainGroups member of the `ElementInfo` struct to the corresponding
|
|
// GROUP_ values (OR'ed together).
|
|
// Testing containment then simply consists of checking whether the
|
|
// mCanContainGroups bitmask of an element and the mGroup bitmask of a
|
|
// potential child overlap.
|
|
|
|
#define GROUP_NONE 0
|
|
|
|
// body, head, html
|
|
#define GROUP_TOPLEVEL (1 << 1)
|
|
|
|
// base, link, meta, script, style, title
|
|
#define GROUP_HEAD_CONTENT (1 << 2)
|
|
|
|
// b, big, i, s, small, strike, tt, u
|
|
#define GROUP_FONTSTYLE (1 << 3)
|
|
|
|
// abbr, acronym, cite, code, datalist, del, dfn, em, ins, kbd, mark, rb, rp
|
|
// rt, rtc, ruby, samp, strong, var
|
|
#define GROUP_PHRASE (1 << 4)
|
|
|
|
// a, applet, basefont, bdi, bdo, br, font, iframe, img, map, meter, object,
|
|
// output, picture, progress, q, script, span, sub, sup
|
|
#define GROUP_SPECIAL (1 << 5)
|
|
|
|
// button, form, input, label, select, textarea
|
|
#define GROUP_FORMCONTROL (1 << 6)
|
|
|
|
// address, applet, article, aside, blockquote, button, center, del, details,
|
|
// dialog, dir, div, dl, fieldset, figure, footer, form, h1, h2, h3, h4, h5,
|
|
// h6, header, hgroup, hr, iframe, ins, main, map, menu, nav, noframes,
|
|
// noscript, object, ol, p, pre, table, search, section, summary, ul
|
|
#define GROUP_BLOCK (1 << 7)
|
|
|
|
// frame, frameset
|
|
#define GROUP_FRAME (1 << 8)
|
|
|
|
// col, tbody
|
|
#define GROUP_TABLE_CONTENT (1 << 9)
|
|
|
|
// tr
|
|
#define GROUP_TBODY_CONTENT (1 << 10)
|
|
|
|
// td, th
|
|
#define GROUP_TR_CONTENT (1 << 11)
|
|
|
|
// col
|
|
#define GROUP_COLGROUP_CONTENT (1 << 12)
|
|
|
|
// param
|
|
#define GROUP_OBJECT_CONTENT (1 << 13)
|
|
|
|
// li
|
|
#define GROUP_LI (1 << 14)
|
|
|
|
// area
|
|
#define GROUP_MAP_CONTENT (1 << 15)
|
|
|
|
// optgroup, option
|
|
#define GROUP_SELECT_CONTENT (1 << 16)
|
|
|
|
// option
|
|
#define GROUP_OPTIONS (1 << 17)
|
|
|
|
// dd, dt
|
|
#define GROUP_DL_CONTENT (1 << 18)
|
|
|
|
// p
|
|
#define GROUP_P (1 << 19)
|
|
|
|
// text, white-space, newline, comment
|
|
#define GROUP_LEAF (1 << 20)
|
|
|
|
// XXX This is because the editor does sublists illegally.
|
|
// ol, ul
|
|
#define GROUP_OL_UL (1 << 21)
|
|
|
|
// h1, h2, h3, h4, h5, h6
|
|
#define GROUP_HEADING (1 << 22)
|
|
|
|
// figcaption
|
|
#define GROUP_FIGCAPTION (1 << 23)
|
|
|
|
// picture members (img, source)
|
|
#define GROUP_PICTURE_CONTENT (1 << 24)
|
|
|
|
#define GROUP_INLINE_ELEMENT \
|
|
(GROUP_FONTSTYLE | GROUP_PHRASE | GROUP_SPECIAL | GROUP_FORMCONTROL | \
|
|
GROUP_LEAF)
|
|
|
|
#define GROUP_FLOW_ELEMENT (GROUP_INLINE_ELEMENT | GROUP_BLOCK)
|
|
|
|
struct ElementInfo final {
|
|
#ifdef DEBUG
|
|
nsHTMLTag mTag;
|
|
#endif
|
|
// See `GROUP_NONE`'s comment.
|
|
uint32_t mGroup;
|
|
// See `GROUP_NONE`'s comment.
|
|
uint32_t mCanContainGroups;
|
|
bool mIsContainer;
|
|
bool mCanContainSelf;
|
|
};
|
|
|
|
#ifdef DEBUG
|
|
# define ELEM(_tag, _isContainer, _canContainSelf, _group, _canContainGroups) \
|
|
{eHTMLTag_##_tag, _group, _canContainGroups, _isContainer, _canContainSelf}
|
|
#else
|
|
# define ELEM(_tag, _isContainer, _canContainSelf, _group, _canContainGroups) \
|
|
{_group, _canContainGroups, _isContainer, _canContainSelf}
|
|
#endif
|
|
|
|
static const ElementInfo kElements[eHTMLTag_userdefined] = {
|
|
ELEM(a, true, false, GROUP_SPECIAL, GROUP_INLINE_ELEMENT),
|
|
ELEM(abbr, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(acronym, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(address, true, true, GROUP_BLOCK, GROUP_INLINE_ELEMENT | GROUP_P),
|
|
// While applet is no longer a valid tag, removing it here breaks the editor
|
|
// (compiles, but causes many tests to fail in odd ways). This list is
|
|
// tracked against the main HTML Tag list, so any changes will require more
|
|
// than just removing entries.
|
|
ELEM(applet, true, true, GROUP_SPECIAL | GROUP_BLOCK,
|
|
GROUP_FLOW_ELEMENT | GROUP_OBJECT_CONTENT),
|
|
ELEM(area, false, false, GROUP_MAP_CONTENT, GROUP_NONE),
|
|
ELEM(article, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(aside, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(audio, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(b, true, true, GROUP_FONTSTYLE, GROUP_INLINE_ELEMENT),
|
|
ELEM(base, false, false, GROUP_HEAD_CONTENT, GROUP_NONE),
|
|
ELEM(basefont, false, false, GROUP_SPECIAL, GROUP_NONE),
|
|
ELEM(bdi, true, true, GROUP_SPECIAL, GROUP_INLINE_ELEMENT),
|
|
ELEM(bdo, true, true, GROUP_SPECIAL, GROUP_INLINE_ELEMENT),
|
|
ELEM(bgsound, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(big, true, true, GROUP_FONTSTYLE, GROUP_INLINE_ELEMENT),
|
|
ELEM(blockquote, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(body, true, true, GROUP_TOPLEVEL, GROUP_FLOW_ELEMENT),
|
|
ELEM(br, false, false, GROUP_SPECIAL, GROUP_NONE),
|
|
ELEM(button, true, true, GROUP_FORMCONTROL | GROUP_BLOCK,
|
|
GROUP_FLOW_ELEMENT),
|
|
ELEM(canvas, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(caption, true, true, GROUP_NONE, GROUP_INLINE_ELEMENT),
|
|
ELEM(center, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(cite, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(code, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(col, false, false, GROUP_TABLE_CONTENT | GROUP_COLGROUP_CONTENT,
|
|
GROUP_NONE),
|
|
ELEM(colgroup, true, false, GROUP_NONE, GROUP_COLGROUP_CONTENT),
|
|
ELEM(data, true, false, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(datalist, true, false, GROUP_PHRASE,
|
|
GROUP_OPTIONS | GROUP_INLINE_ELEMENT),
|
|
ELEM(dd, true, false, GROUP_DL_CONTENT, GROUP_FLOW_ELEMENT),
|
|
ELEM(del, true, true, GROUP_PHRASE | GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(details, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(dfn, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(dialog, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(dir, true, false, GROUP_BLOCK, GROUP_LI),
|
|
ELEM(div, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(dl, true, false, GROUP_BLOCK, GROUP_DL_CONTENT),
|
|
ELEM(dt, true, true, GROUP_DL_CONTENT, GROUP_INLINE_ELEMENT),
|
|
ELEM(em, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(embed, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(fieldset, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(figcaption, true, false, GROUP_FIGCAPTION, GROUP_FLOW_ELEMENT),
|
|
ELEM(figure, true, true, GROUP_BLOCK,
|
|
GROUP_FLOW_ELEMENT | GROUP_FIGCAPTION),
|
|
ELEM(font, true, true, GROUP_SPECIAL, GROUP_INLINE_ELEMENT),
|
|
ELEM(footer, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(form, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(frame, false, false, GROUP_FRAME, GROUP_NONE),
|
|
ELEM(frameset, true, true, GROUP_FRAME, GROUP_FRAME),
|
|
ELEM(h1, true, false, GROUP_BLOCK | GROUP_HEADING, GROUP_INLINE_ELEMENT),
|
|
ELEM(h2, true, false, GROUP_BLOCK | GROUP_HEADING, GROUP_INLINE_ELEMENT),
|
|
ELEM(h3, true, false, GROUP_BLOCK | GROUP_HEADING, GROUP_INLINE_ELEMENT),
|
|
ELEM(h4, true, false, GROUP_BLOCK | GROUP_HEADING, GROUP_INLINE_ELEMENT),
|
|
ELEM(h5, true, false, GROUP_BLOCK | GROUP_HEADING, GROUP_INLINE_ELEMENT),
|
|
ELEM(h6, true, false, GROUP_BLOCK | GROUP_HEADING, GROUP_INLINE_ELEMENT),
|
|
ELEM(head, true, false, GROUP_TOPLEVEL, GROUP_HEAD_CONTENT),
|
|
ELEM(header, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(hgroup, true, false, GROUP_BLOCK, GROUP_HEADING),
|
|
ELEM(hr, false, false, GROUP_BLOCK, GROUP_NONE),
|
|
ELEM(html, true, false, GROUP_TOPLEVEL, GROUP_TOPLEVEL),
|
|
ELEM(i, true, true, GROUP_FONTSTYLE, GROUP_INLINE_ELEMENT),
|
|
ELEM(iframe, true, true, GROUP_SPECIAL | GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(image, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(img, false, false, GROUP_SPECIAL | GROUP_PICTURE_CONTENT, GROUP_NONE),
|
|
ELEM(input, false, false, GROUP_FORMCONTROL, GROUP_NONE),
|
|
ELEM(ins, true, true, GROUP_PHRASE | GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(kbd, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(keygen, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(label, true, false, GROUP_FORMCONTROL, GROUP_INLINE_ELEMENT),
|
|
ELEM(legend, true, true, GROUP_NONE, GROUP_INLINE_ELEMENT),
|
|
ELEM(li, true, false, GROUP_LI, GROUP_FLOW_ELEMENT),
|
|
ELEM(link, false, false, GROUP_HEAD_CONTENT, GROUP_NONE),
|
|
ELEM(listing, true, true, GROUP_BLOCK, GROUP_INLINE_ELEMENT),
|
|
ELEM(main, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(map, true, true, GROUP_SPECIAL, GROUP_BLOCK | GROUP_MAP_CONTENT),
|
|
ELEM(mark, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(marquee, true, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(menu, true, true, GROUP_BLOCK, GROUP_LI | GROUP_FLOW_ELEMENT),
|
|
ELEM(meta, false, false, GROUP_HEAD_CONTENT, GROUP_NONE),
|
|
ELEM(meter, true, false, GROUP_SPECIAL, GROUP_FLOW_ELEMENT),
|
|
ELEM(multicol, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(nav, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(nobr, true, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(noembed, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(noframes, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(noscript, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(object, true, true, GROUP_SPECIAL | GROUP_BLOCK,
|
|
GROUP_FLOW_ELEMENT | GROUP_OBJECT_CONTENT),
|
|
// XXX Can contain self and ul because editor does sublists illegally.
|
|
ELEM(ol, true, true, GROUP_BLOCK | GROUP_OL_UL, GROUP_LI | GROUP_OL_UL),
|
|
ELEM(optgroup, true, false, GROUP_SELECT_CONTENT, GROUP_OPTIONS),
|
|
ELEM(option, true, false, GROUP_SELECT_CONTENT | GROUP_OPTIONS, GROUP_LEAF),
|
|
ELEM(output, true, true, GROUP_SPECIAL, GROUP_INLINE_ELEMENT),
|
|
ELEM(p, true, false, GROUP_BLOCK | GROUP_P, GROUP_INLINE_ELEMENT),
|
|
ELEM(param, false, false, GROUP_OBJECT_CONTENT, GROUP_NONE),
|
|
ELEM(picture, true, false, GROUP_SPECIAL, GROUP_PICTURE_CONTENT),
|
|
ELEM(plaintext, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(pre, true, true, GROUP_BLOCK, GROUP_INLINE_ELEMENT),
|
|
ELEM(progress, true, false, GROUP_SPECIAL, GROUP_FLOW_ELEMENT),
|
|
ELEM(q, true, true, GROUP_SPECIAL, GROUP_INLINE_ELEMENT),
|
|
ELEM(rb, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(rp, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(rt, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(rtc, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(ruby, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(s, true, true, GROUP_FONTSTYLE, GROUP_INLINE_ELEMENT),
|
|
ELEM(samp, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(script, true, false, GROUP_HEAD_CONTENT | GROUP_SPECIAL, GROUP_LEAF),
|
|
ELEM(search, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(section, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(select, true, false, GROUP_FORMCONTROL, GROUP_SELECT_CONTENT),
|
|
ELEM(small, true, true, GROUP_FONTSTYLE, GROUP_INLINE_ELEMENT),
|
|
ELEM(slot, true, false, GROUP_NONE, GROUP_FLOW_ELEMENT),
|
|
ELEM(source, false, false, GROUP_PICTURE_CONTENT, GROUP_NONE),
|
|
ELEM(span, true, true, GROUP_SPECIAL, GROUP_INLINE_ELEMENT),
|
|
ELEM(strike, true, true, GROUP_FONTSTYLE, GROUP_INLINE_ELEMENT),
|
|
ELEM(strong, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(style, true, false, GROUP_HEAD_CONTENT, GROUP_LEAF),
|
|
ELEM(sub, true, true, GROUP_SPECIAL, GROUP_INLINE_ELEMENT),
|
|
ELEM(summary, true, true, GROUP_BLOCK, GROUP_FLOW_ELEMENT),
|
|
ELEM(sup, true, true, GROUP_SPECIAL, GROUP_INLINE_ELEMENT),
|
|
ELEM(table, true, false, GROUP_BLOCK, GROUP_TABLE_CONTENT),
|
|
ELEM(tbody, true, false, GROUP_TABLE_CONTENT, GROUP_TBODY_CONTENT),
|
|
ELEM(td, true, false, GROUP_TR_CONTENT, GROUP_FLOW_ELEMENT),
|
|
ELEM(textarea, true, false, GROUP_FORMCONTROL, GROUP_LEAF),
|
|
ELEM(tfoot, true, false, GROUP_NONE, GROUP_TBODY_CONTENT),
|
|
ELEM(th, true, false, GROUP_TR_CONTENT, GROUP_FLOW_ELEMENT),
|
|
ELEM(thead, true, false, GROUP_NONE, GROUP_TBODY_CONTENT),
|
|
ELEM(template, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(time, true, false, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(title, true, false, GROUP_HEAD_CONTENT, GROUP_LEAF),
|
|
ELEM(tr, true, false, GROUP_TBODY_CONTENT, GROUP_TR_CONTENT),
|
|
ELEM(track, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(tt, true, true, GROUP_FONTSTYLE, GROUP_INLINE_ELEMENT),
|
|
ELEM(u, true, true, GROUP_FONTSTYLE, GROUP_INLINE_ELEMENT),
|
|
// XXX Can contain self and ol because editor does sublists illegally.
|
|
ELEM(ul, true, true, GROUP_BLOCK | GROUP_OL_UL, GROUP_LI | GROUP_OL_UL),
|
|
ELEM(var, true, true, GROUP_PHRASE, GROUP_INLINE_ELEMENT),
|
|
ELEM(video, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(wbr, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(xmp, true, false, GROUP_BLOCK, GROUP_NONE),
|
|
|
|
// These aren't elements.
|
|
ELEM(text, false, false, GROUP_LEAF, GROUP_NONE),
|
|
ELEM(whitespace, false, false, GROUP_LEAF, GROUP_NONE),
|
|
ELEM(newline, false, false, GROUP_LEAF, GROUP_NONE),
|
|
ELEM(comment, false, false, GROUP_LEAF, GROUP_NONE),
|
|
ELEM(entity, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(doctypeDecl, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(markupDecl, false, false, GROUP_NONE, GROUP_NONE),
|
|
ELEM(instruction, false, false, GROUP_NONE, GROUP_NONE),
|
|
|
|
ELEM(userdefined, true, false, GROUP_NONE, GROUP_FLOW_ELEMENT)};
|
|
|
|
bool HTMLEditUtils::CanNodeContain(nsHTMLTag aParentTagId,
|
|
nsHTMLTag aChildTagId) {
|
|
NS_ASSERTION(
|
|
aParentTagId > eHTMLTag_unknown && aParentTagId <= eHTMLTag_userdefined,
|
|
"aParentTagId out of range!");
|
|
NS_ASSERTION(
|
|
aChildTagId > eHTMLTag_unknown && aChildTagId <= eHTMLTag_userdefined,
|
|
"aChildTagId out of range!");
|
|
|
|
#ifdef DEBUG
|
|
static bool checked = false;
|
|
if (!checked) {
|
|
checked = true;
|
|
int32_t i;
|
|
for (i = 1; i <= eHTMLTag_userdefined; ++i) {
|
|
NS_ASSERTION(kElements[i - 1].mTag == i,
|
|
"You need to update kElements (missing tags).");
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// Special-case button.
|
|
if (aParentTagId == eHTMLTag_button) {
|
|
static const nsHTMLTag kButtonExcludeKids[] = {
|
|
eHTMLTag_a, eHTMLTag_fieldset, eHTMLTag_form, eHTMLTag_iframe,
|
|
eHTMLTag_input, eHTMLTag_select, eHTMLTag_textarea};
|
|
|
|
uint32_t j;
|
|
for (j = 0; j < std::size(kButtonExcludeKids); ++j) {
|
|
if (kButtonExcludeKids[j] == aChildTagId) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Deprecated elements.
|
|
if (aChildTagId == eHTMLTag_bgsound) {
|
|
return false;
|
|
}
|
|
|
|
// Bug #67007, dont strip userdefined tags.
|
|
if (aChildTagId == eHTMLTag_userdefined) {
|
|
return true;
|
|
}
|
|
|
|
const ElementInfo& parent = kElements[aParentTagId - 1];
|
|
if (aParentTagId == aChildTagId) {
|
|
return parent.mCanContainSelf;
|
|
}
|
|
|
|
const ElementInfo& child = kElements[aChildTagId - 1];
|
|
return !!(parent.mCanContainGroups & child.mGroup);
|
|
}
|
|
|
|
bool HTMLEditUtils::ContentIsInert(const nsIContent& aContent) {
|
|
for (nsIContent* content :
|
|
aContent.InclusiveFlatTreeAncestorsOfType<nsIContent>()) {
|
|
if (nsIFrame* frame = content->GetPrimaryFrame()) {
|
|
return frame->StyleUI()->IsInert();
|
|
}
|
|
// If it doesn't have primary frame, we need to check its ancestors.
|
|
// This may occur if it's an invisible text node or element nodes whose
|
|
// display is an invisible value.
|
|
if (!content->IsElement()) {
|
|
continue;
|
|
}
|
|
if (content->AsElement()->State().HasState(dom::ElementState::INERT)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool HTMLEditUtils::IsContainerNode(nsHTMLTag aTagId) {
|
|
NS_ASSERTION(aTagId > eHTMLTag_unknown && aTagId <= eHTMLTag_userdefined,
|
|
"aTagId out of range!");
|
|
|
|
return kElements[aTagId - 1].mIsContainer;
|
|
}
|
|
|
|
bool HTMLEditUtils::IsNonListSingleLineContainer(const nsINode& aNode) {
|
|
return aNode.IsAnyOfHTMLElements(
|
|
nsGkAtoms::address, nsGkAtoms::div, nsGkAtoms::h1, nsGkAtoms::h2,
|
|
nsGkAtoms::h3, nsGkAtoms::h4, nsGkAtoms::h5, nsGkAtoms::h6,
|
|
nsGkAtoms::listing, nsGkAtoms::p, nsGkAtoms::pre, nsGkAtoms::xmp);
|
|
}
|
|
|
|
bool HTMLEditUtils::IsSingleLineContainer(const nsINode& aNode) {
|
|
return IsNonListSingleLineContainer(aNode) ||
|
|
aNode.IsAnyOfHTMLElements(nsGkAtoms::li, nsGkAtoms::dt, nsGkAtoms::dd);
|
|
}
|
|
|
|
// static
|
|
template <typename PT, typename CT>
|
|
nsIContent* HTMLEditUtils::GetPreviousContent(
|
|
const EditorDOMPointBase<PT, CT>& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck,
|
|
const Element* aAncestorLimiter /* = nullptr */) {
|
|
MOZ_ASSERT(aPoint.IsSetAndValid());
|
|
NS_WARNING_ASSERTION(
|
|
!aPoint.IsInDataNode() || aPoint.IsInTextNode(),
|
|
"GetPreviousContent() doesn't assume that the start point is a "
|
|
"data node except text node");
|
|
|
|
// If we are at the beginning of the node, or it is a text node, then just
|
|
// look before it.
|
|
if (aPoint.IsStartOfContainer() || aPoint.IsInTextNode()) {
|
|
if (aOptions.contains(WalkTreeOption::StopAtBlockBoundary) &&
|
|
aPoint.IsInContentNode() &&
|
|
HTMLEditUtils::IsBlockElement(
|
|
*aPoint.template ContainerAs<nsIContent>(), aBlockInlineCheck)) {
|
|
// If we aren't allowed to cross blocks, don't look before this block.
|
|
return nullptr;
|
|
}
|
|
return HTMLEditUtils::GetPreviousContent(
|
|
*aPoint.GetContainer(), aOptions, aBlockInlineCheck, aAncestorLimiter);
|
|
}
|
|
|
|
// else look before the child at 'aOffset'
|
|
if (aPoint.GetChild()) {
|
|
return HTMLEditUtils::GetPreviousContent(
|
|
*aPoint.GetChild(), aOptions, aBlockInlineCheck, aAncestorLimiter);
|
|
}
|
|
|
|
// unless there isn't one, in which case we are at the end of the node
|
|
// and want the deep-right child.
|
|
nsIContent* lastLeafContent = HTMLEditUtils::GetLastLeafContent(
|
|
*aPoint.GetContainer(),
|
|
{aOptions.contains(WalkTreeOption::StopAtBlockBoundary)
|
|
? LeafNodeType::LeafNodeOrChildBlock
|
|
: LeafNodeType::OnlyLeafNode},
|
|
aBlockInlineCheck);
|
|
if (!lastLeafContent) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (!HTMLEditUtils::IsContentIgnored(*lastLeafContent, aOptions)) {
|
|
return lastLeafContent;
|
|
}
|
|
|
|
// restart the search from the non-editable node we just found
|
|
return HTMLEditUtils::GetPreviousContent(*lastLeafContent, aOptions,
|
|
aBlockInlineCheck, aAncestorLimiter);
|
|
}
|
|
|
|
// static
|
|
template <typename PT, typename CT>
|
|
nsIContent* HTMLEditUtils::GetNextContent(
|
|
const EditorDOMPointBase<PT, CT>& aPoint, const WalkTreeOptions& aOptions,
|
|
BlockInlineCheck aBlockInlineCheck,
|
|
const Element* aAncestorLimiter /* = nullptr */) {
|
|
MOZ_ASSERT(aPoint.IsSetAndValid());
|
|
NS_WARNING_ASSERTION(
|
|
!aPoint.IsInDataNode() || aPoint.IsInTextNode(),
|
|
"GetNextContent() doesn't assume that the start point is a "
|
|
"data node except text node");
|
|
|
|
auto point = aPoint.template To<EditorRawDOMPoint>();
|
|
|
|
// if the container is a text node, use its location instead
|
|
if (point.IsInTextNode()) {
|
|
point.SetAfter(point.GetContainer());
|
|
if (NS_WARN_IF(!point.IsSet())) {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
if (point.GetChild()) {
|
|
if (aOptions.contains(WalkTreeOption::StopAtBlockBoundary) &&
|
|
HTMLEditUtils::IsBlockElement(*point.GetChild(), aBlockInlineCheck)) {
|
|
return point.GetChild();
|
|
}
|
|
|
|
nsIContent* firstLeafContent = HTMLEditUtils::GetFirstLeafContent(
|
|
*point.GetChild(),
|
|
{aOptions.contains(WalkTreeOption::StopAtBlockBoundary)
|
|
? LeafNodeType::LeafNodeOrChildBlock
|
|
: LeafNodeType::OnlyLeafNode},
|
|
aBlockInlineCheck);
|
|
if (!firstLeafContent) {
|
|
return point.GetChild();
|
|
}
|
|
|
|
// XXX Why do we need to do this check? The leaf node must be a descendant
|
|
// of `point.GetChild()`.
|
|
if (aAncestorLimiter &&
|
|
(firstLeafContent == aAncestorLimiter ||
|
|
!firstLeafContent->IsInclusiveDescendantOf(aAncestorLimiter))) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (!HTMLEditUtils::IsContentIgnored(*firstLeafContent, aOptions)) {
|
|
return firstLeafContent;
|
|
}
|
|
|
|
// restart the search from the non-editable node we just found
|
|
return HTMLEditUtils::GetNextContent(*firstLeafContent, aOptions,
|
|
aBlockInlineCheck, aAncestorLimiter);
|
|
}
|
|
|
|
// unless there isn't one, in which case we are at the end of the node
|
|
// and want the next one.
|
|
if (aOptions.contains(WalkTreeOption::StopAtBlockBoundary) &&
|
|
point.IsInContentNode() &&
|
|
HTMLEditUtils::IsBlockElement(*point.template ContainerAs<nsIContent>(),
|
|
aBlockInlineCheck)) {
|
|
// don't cross out of parent block
|
|
return nullptr;
|
|
}
|
|
|
|
return HTMLEditUtils::GetNextContent(*point.GetContainer(), aOptions,
|
|
aBlockInlineCheck, aAncestorLimiter);
|
|
}
|
|
|
|
// static
|
|
nsIContent* HTMLEditUtils::GetAdjacentLeafContent(
|
|
const nsINode& aNode, WalkTreeDirection aWalkTreeDirection,
|
|
const WalkTreeOptions& aOptions, BlockInlineCheck aBlockInlineCheck,
|
|
const Element* aAncestorLimiter /* = nullptr */) {
|
|
// called only by GetPriorNode so we don't need to check params.
|
|
MOZ_ASSERT(&aNode != aAncestorLimiter);
|
|
MOZ_ASSERT_IF(aAncestorLimiter,
|
|
aAncestorLimiter->IsInclusiveDescendantOf(aAncestorLimiter));
|
|
|
|
const nsINode* node = &aNode;
|
|
for (;;) {
|
|
// if aNode has a sibling in the right direction, return
|
|
// that sibling's closest child (or itself if it has no children)
|
|
nsIContent* sibling = aWalkTreeDirection == WalkTreeDirection::Forward
|
|
? node->GetNextSibling()
|
|
: node->GetPreviousSibling();
|
|
if (sibling) {
|
|
// XXX If `sibling` belongs to siblings of inclusive ancestors of aNode,
|
|
// perhaps, we need to use
|
|
// IgnoreInsideBlockBoundary(aBlockInlineCheck) here.
|
|
if (aOptions.contains(WalkTreeOption::StopAtBlockBoundary) &&
|
|
HTMLEditUtils::IsBlockElement(*sibling, aBlockInlineCheck)) {
|
|
// don't look inside previous sibling, since it is a block
|
|
return sibling;
|
|
}
|
|
const LeafNodeTypes leafNodeTypes = {
|
|
aOptions.contains(WalkTreeOption::StopAtBlockBoundary)
|
|
? LeafNodeType::LeafNodeOrChildBlock
|
|
: LeafNodeType::OnlyLeafNode};
|
|
nsIContent* leafContent =
|
|
aWalkTreeDirection == WalkTreeDirection::Forward
|
|
? HTMLEditUtils::GetFirstLeafContent(*sibling, leafNodeTypes,
|
|
aBlockInlineCheck)
|
|
: HTMLEditUtils::GetLastLeafContent(*sibling, leafNodeTypes,
|
|
aBlockInlineCheck);
|
|
return leafContent ? leafContent : sibling;
|
|
}
|
|
|
|
nsIContent* parent = node->GetParent();
|
|
if (!parent) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (parent == aAncestorLimiter ||
|
|
(aOptions.contains(WalkTreeOption::StopAtBlockBoundary) &&
|
|
HTMLEditUtils::IsBlockElement(*parent, aBlockInlineCheck))) {
|
|
return nullptr;
|
|
}
|
|
|
|
node = parent;
|
|
}
|
|
|
|
MOZ_ASSERT_UNREACHABLE("What part of for(;;) do you not understand?");
|
|
return nullptr;
|
|
}
|
|
|
|
// static
|
|
nsIContent* HTMLEditUtils::GetAdjacentContent(
|
|
const nsINode& aNode, WalkTreeDirection aWalkTreeDirection,
|
|
const WalkTreeOptions& aOptions, BlockInlineCheck aBlockInlineCheck,
|
|
const Element* aAncestorLimiter /* = nullptr */) {
|
|
if (&aNode == aAncestorLimiter) {
|
|
// Don't allow traversal above the root node! This helps
|
|
// prevent us from accidentally editing browser content
|
|
// when the editor is in a text widget.
|
|
return nullptr;
|
|
}
|
|
|
|
nsIContent* leafContent = HTMLEditUtils::GetAdjacentLeafContent(
|
|
aNode, aWalkTreeDirection, aOptions, aBlockInlineCheck, aAncestorLimiter);
|
|
if (!leafContent) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (!HTMLEditUtils::IsContentIgnored(*leafContent, aOptions)) {
|
|
return leafContent;
|
|
}
|
|
|
|
return HTMLEditUtils::GetAdjacentContent(*leafContent, aWalkTreeDirection,
|
|
aOptions, aBlockInlineCheck,
|
|
aAncestorLimiter);
|
|
}
|
|
|
|
// static
|
|
template <typename EditorDOMPointType>
|
|
EditorDOMPointType HTMLEditUtils::GetPreviousEditablePoint(
|
|
nsIContent& aContent, const Element* aAncestorLimiter,
|
|
InvisibleWhiteSpaces aInvisibleWhiteSpaces,
|
|
TableBoundary aHowToTreatTableBoundary) {
|
|
MOZ_ASSERT(HTMLEditUtils::IsSimplyEditableNode(aContent));
|
|
NS_ASSERTION(!HTMLEditUtils::IsAnyTableElement(&aContent) ||
|
|
HTMLEditUtils::IsTableCellOrCaption(aContent),
|
|
"HTMLEditUtils::GetPreviousEditablePoint() may return a point "
|
|
"between table structure elements");
|
|
|
|
if (&aContent == aAncestorLimiter) {
|
|
return EditorDOMPointType();
|
|
}
|
|
|
|
// First, look for previous content.
|
|
nsIContent* previousContent = aContent.GetPreviousSibling();
|
|
if (!previousContent) {
|
|
if (!aContent.GetParentElement()) {
|
|
return EditorDOMPointType();
|
|
}
|
|
nsIContent* inclusiveAncestor = &aContent;
|
|
for (Element* parentElement : aContent.AncestorsOfType<Element>()) {
|
|
if (parentElement == aAncestorLimiter ||
|
|
!HTMLEditUtils::IsSimplyEditableNode(*parentElement) ||
|
|
!HTMLEditUtils::CanCrossContentBoundary(*parentElement,
|
|
aHowToTreatTableBoundary)) {
|
|
// If cannot cross the parent element boundary, return the point of
|
|
// last inclusive ancestor point.
|
|
return EditorDOMPointType(inclusiveAncestor);
|
|
}
|
|
|
|
// Start of the parent element is a next editable point if it's an
|
|
// element which is not a table structure element.
|
|
if (!HTMLEditUtils::IsAnyTableElement(parentElement) ||
|
|
HTMLEditUtils::IsTableCellOrCaption(*parentElement)) {
|
|
inclusiveAncestor = parentElement;
|
|
}
|
|
|
|
previousContent = parentElement->GetPreviousSibling();
|
|
if (!previousContent) {
|
|
continue; // Keep looking for previous sibling of an ancestor.
|
|
}
|
|
|
|
// XXX Should we ignore data node like CDATA, Comment, etc?
|
|
|
|
// If previous content is not editable, let's return the point after it.
|
|
if (!HTMLEditUtils::IsSimplyEditableNode(*previousContent)) {
|
|
return EditorDOMPointType::After(*previousContent);
|
|
}
|
|
|
|
// If cannot cross previous content boundary, return start of last
|
|
// inclusive ancestor.
|
|
if (!HTMLEditUtils::CanCrossContentBoundary(*previousContent,
|
|
aHowToTreatTableBoundary)) {
|
|
return inclusiveAncestor == &aContent
|
|
? EditorDOMPointType(inclusiveAncestor)
|
|
: EditorDOMPointType(inclusiveAncestor, 0);
|
|
}
|
|
break;
|
|
}
|
|
if (!previousContent) {
|
|
return EditorDOMPointType(inclusiveAncestor);
|
|
}
|
|
} else if (!HTMLEditUtils::IsSimplyEditableNode(*previousContent)) {
|
|
return EditorDOMPointType::After(*previousContent);
|
|
} else if (!HTMLEditUtils::CanCrossContentBoundary(
|
|
*previousContent, aHowToTreatTableBoundary)) {
|
|
return EditorDOMPointType(&aContent);
|
|
}
|
|
|
|
// Next, look for end of the previous content.
|
|
nsIContent* leafContent = previousContent;
|
|
if (previousContent->GetChildCount() &&
|
|
HTMLEditUtils::IsContainerNode(*previousContent)) {
|
|
for (nsIContent* maybeLeafContent = previousContent->GetLastChild();
|
|
maybeLeafContent;
|
|
maybeLeafContent = maybeLeafContent->GetLastChild()) {
|
|
// If it's not an editable content or cannot cross the boundary,
|
|
// return the point after the content. Note that in this case,
|
|
// the content must not be any table elements except `<table>`
|
|
// because we've climbed down the tree.
|
|
if (!HTMLEditUtils::IsSimplyEditableNode(*maybeLeafContent) ||
|
|
!HTMLEditUtils::CanCrossContentBoundary(*maybeLeafContent,
|
|
aHowToTreatTableBoundary)) {
|
|
return EditorDOMPointType::After(*maybeLeafContent);
|
|
}
|
|
leafContent = maybeLeafContent;
|
|
if (!HTMLEditUtils::IsContainerNode(*leafContent)) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (leafContent->IsText()) {
|
|
Text* textNode = leafContent->AsText();
|
|
if (aInvisibleWhiteSpaces == InvisibleWhiteSpaces::Preserve) {
|
|
return EditorDOMPointType::AtEndOf(*textNode);
|
|
}
|
|
// There may be invisible trailing white-spaces which should be
|
|
// ignored. Let's scan its start.
|
|
return WSRunScanner::GetAfterLastVisiblePoint<EditorDOMPointType>(
|
|
WSRunScanner::Scan::EditableNodes, *textNode);
|
|
}
|
|
|
|
// If it's a container element, return end of it. Otherwise, return
|
|
// the point after the non-container element.
|
|
return HTMLEditUtils::IsContainerNode(*leafContent)
|
|
? EditorDOMPointType::AtEndOf(*leafContent)
|
|
: EditorDOMPointType::After(*leafContent);
|
|
}
|
|
|
|
// static
|
|
template <typename EditorDOMPointType>
|
|
EditorDOMPointType HTMLEditUtils::GetNextEditablePoint(
|
|
nsIContent& aContent, const Element* aAncestorLimiter,
|
|
InvisibleWhiteSpaces aInvisibleWhiteSpaces,
|
|
TableBoundary aHowToTreatTableBoundary) {
|
|
MOZ_ASSERT(HTMLEditUtils::IsSimplyEditableNode(aContent));
|
|
NS_ASSERTION(!HTMLEditUtils::IsAnyTableElement(&aContent) ||
|
|
HTMLEditUtils::IsTableCellOrCaption(aContent),
|
|
"HTMLEditUtils::GetPreviousEditablePoint() may return a point "
|
|
"between table structure elements");
|
|
|
|
if (&aContent == aAncestorLimiter) {
|
|
return EditorDOMPointType();
|
|
}
|
|
|
|
// First, look for next content.
|
|
nsIContent* nextContent = aContent.GetNextSibling();
|
|
if (!nextContent) {
|
|
if (!aContent.GetParentElement()) {
|
|
return EditorDOMPointType();
|
|
}
|
|
nsIContent* inclusiveAncestor = &aContent;
|
|
for (Element* parentElement : aContent.AncestorsOfType<Element>()) {
|
|
if (parentElement == aAncestorLimiter ||
|
|
!HTMLEditUtils::IsSimplyEditableNode(*parentElement) ||
|
|
!HTMLEditUtils::CanCrossContentBoundary(*parentElement,
|
|
aHowToTreatTableBoundary)) {
|
|
// If cannot cross the parent element boundary, return the point of
|
|
// last inclusive ancestor point.
|
|
return EditorDOMPointType(inclusiveAncestor);
|
|
}
|
|
|
|
// End of the parent element is a next editable point if it's an
|
|
// element which is not a table structure element.
|
|
if (!HTMLEditUtils::IsAnyTableElement(parentElement) ||
|
|
HTMLEditUtils::IsTableCellOrCaption(*parentElement)) {
|
|
inclusiveAncestor = parentElement;
|
|
}
|
|
|
|
nextContent = parentElement->GetNextSibling();
|
|
if (!nextContent) {
|
|
continue; // Keep looking for next sibling of an ancestor.
|
|
}
|
|
|
|
// XXX Should we ignore data node like CDATA, Comment, etc?
|
|
|
|
// If next content is not editable, let's return the point after
|
|
// the last inclusive ancestor.
|
|
if (!HTMLEditUtils::IsSimplyEditableNode(*nextContent)) {
|
|
return EditorDOMPointType::After(*parentElement);
|
|
}
|
|
|
|
// If cannot cross next content boundary, return after the last
|
|
// inclusive ancestor.
|
|
if (!HTMLEditUtils::CanCrossContentBoundary(*nextContent,
|
|
aHowToTreatTableBoundary)) {
|
|
return EditorDOMPointType::After(*inclusiveAncestor);
|
|
}
|
|
break;
|
|
}
|
|
if (!nextContent) {
|
|
return EditorDOMPointType::After(*inclusiveAncestor);
|
|
}
|
|
} else if (!HTMLEditUtils::IsSimplyEditableNode(*nextContent)) {
|
|
return EditorDOMPointType::After(aContent);
|
|
} else if (!HTMLEditUtils::CanCrossContentBoundary(
|
|
*nextContent, aHowToTreatTableBoundary)) {
|
|
return EditorDOMPointType::After(aContent);
|
|
}
|
|
|
|
// Next, look for start of the next content.
|
|
nsIContent* leafContent = nextContent;
|
|
if (nextContent->GetChildCount() &&
|
|
HTMLEditUtils::IsContainerNode(*nextContent)) {
|
|
for (nsIContent* maybeLeafContent = nextContent->GetFirstChild();
|
|
maybeLeafContent;
|
|
maybeLeafContent = maybeLeafContent->GetFirstChild()) {
|
|
// If it's not an editable content or cannot cross the boundary,
|
|
// return the point at the content (i.e., start of its parent). Note
|
|
// that in this case, the content must not be any table elements except
|
|
// `<table>` because we've climbed down the tree.
|
|
if (!HTMLEditUtils::IsSimplyEditableNode(*maybeLeafContent) ||
|
|
!HTMLEditUtils::CanCrossContentBoundary(*maybeLeafContent,
|
|
aHowToTreatTableBoundary)) {
|
|
return EditorDOMPointType(maybeLeafContent);
|
|
}
|
|
leafContent = maybeLeafContent;
|
|
if (!HTMLEditUtils::IsContainerNode(*leafContent)) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (leafContent->IsText()) {
|
|
Text* textNode = leafContent->AsText();
|
|
if (aInvisibleWhiteSpaces == InvisibleWhiteSpaces::Preserve) {
|
|
return EditorDOMPointType(textNode, 0);
|
|
}
|
|
// There may be invisible leading white-spaces which should be
|
|
// ignored. Let's scan its start.
|
|
return WSRunScanner::GetFirstVisiblePoint<EditorDOMPointType>(
|
|
WSRunScanner::Scan::EditableNodes, *textNode);
|
|
}
|
|
|
|
// If it's a container element, return start of it. Otherwise, return
|
|
// the point at the non-container element (i.e., start of its parent).
|
|
return HTMLEditUtils::IsContainerNode(*leafContent)
|
|
? EditorDOMPointType(leafContent, 0)
|
|
: EditorDOMPointType(leafContent);
|
|
}
|
|
|
|
// static
|
|
Element* HTMLEditUtils::GetAncestorElement(
|
|
const nsIContent& aContent, const AncestorTypes& aAncestorTypes,
|
|
BlockInlineCheck aBlockInlineCheck,
|
|
const Element* aAncestorLimiter /* = nullptr */) {
|
|
MOZ_ASSERT(
|
|
aAncestorTypes.contains(AncestorType::ClosestBlockElement) ||
|
|
aAncestorTypes.contains(AncestorType::MostDistantInlineElementInBlock) ||
|
|
aAncestorTypes.contains(AncestorType::ButtonElement) ||
|
|
aAncestorTypes.contains(AncestorType::AllowRootOrAncestorLimiterElement));
|
|
|
|
if (&aContent == aAncestorLimiter) {
|
|
return nullptr;
|
|
}
|
|
|
|
const Element* theBodyElement = aContent.OwnerDoc()->GetBody();
|
|
const Element* theDocumentElement = aContent.OwnerDoc()->GetDocumentElement();
|
|
Element* lastAncestorElement = nullptr;
|
|
const bool editableElementOnly =
|
|
aAncestorTypes.contains(AncestorType::EditableElement);
|
|
const bool lookingForClosestBlockElement =
|
|
aAncestorTypes.contains(AncestorType::ClosestBlockElement);
|
|
const bool lookingForMostDistantInlineElementInBlock =
|
|
aAncestorTypes.contains(AncestorType::MostDistantInlineElementInBlock);
|
|
const bool ignoreHRElement =
|
|
aAncestorTypes.contains(AncestorType::IgnoreHRElement);
|
|
const bool lookingForButtonElement =
|
|
aAncestorTypes.contains(AncestorType::ButtonElement);
|
|
const bool lookingForAnyElement =
|
|
aAncestorTypes.contains(AncestorType::AllowRootOrAncestorLimiterElement);
|
|
auto IsSearchingElementType = [&](const nsIContent& aContent) -> bool {
|
|
if (lookingForAnyElement) {
|
|
return aContent.IsElement();
|
|
}
|
|
if (!aContent.IsElement() ||
|
|
(ignoreHRElement && aContent.IsHTMLElement(nsGkAtoms::hr))) {
|
|
return false;
|
|
}
|
|
if (editableElementOnly &&
|
|
!EditorUtils::IsEditableContent(aContent, EditorType::HTML)) {
|
|
return false;
|
|
}
|
|
return (lookingForClosestBlockElement &&
|
|
HTMLEditUtils::IsBlockElement(aContent, aBlockInlineCheck)) ||
|
|
(lookingForMostDistantInlineElementInBlock &&
|
|
HTMLEditUtils::IsInlineContent(aContent, aBlockInlineCheck)) ||
|
|
(lookingForButtonElement &&
|
|
aContent.IsHTMLElement(nsGkAtoms::button));
|
|
};
|
|
for (Element* element : aContent.AncestorsOfType<Element>()) {
|
|
if (editableElementOnly &&
|
|
!EditorUtils::IsEditableContent(*element, EditorType::HTML)) {
|
|
return lastAncestorElement && IsSearchingElementType(*lastAncestorElement)
|
|
? lastAncestorElement // editing host (can be inline element)
|
|
: nullptr;
|
|
}
|
|
if (ignoreHRElement && element->IsHTMLElement(nsGkAtoms::hr)) {
|
|
if (element == aAncestorLimiter) {
|
|
if (lookingForAnyElement) {
|
|
lastAncestorElement = element;
|
|
}
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
if (lookingForButtonElement && element->IsHTMLElement(nsGkAtoms::button)) {
|
|
return element; // closest button element
|
|
}
|
|
if (HTMLEditUtils::IsBlockElement(*element, aBlockInlineCheck)) {
|
|
if (lookingForClosestBlockElement) {
|
|
return element; // closest block element
|
|
}
|
|
MOZ_ASSERT_IF(lastAncestorElement,
|
|
HTMLEditUtils::IsInlineContent(*lastAncestorElement,
|
|
aBlockInlineCheck));
|
|
return lastAncestorElement; // the last inline element which we found
|
|
}
|
|
if (element == aAncestorLimiter || element == theBodyElement ||
|
|
element == theDocumentElement) {
|
|
if (lookingForAnyElement) {
|
|
lastAncestorElement = element;
|
|
}
|
|
break;
|
|
}
|
|
lastAncestorElement = element;
|
|
}
|
|
return lastAncestorElement && IsSearchingElementType(*lastAncestorElement)
|
|
? lastAncestorElement
|
|
: nullptr;
|
|
}
|
|
|
|
// static
|
|
Element* HTMLEditUtils::GetInclusiveAncestorElement(
|
|
const nsIContent& aContent, const AncestorTypes& aAncestorTypes,
|
|
BlockInlineCheck aBlockInlineCheck,
|
|
const Element* aAncestorLimiter /* = nullptr */) {
|
|
MOZ_ASSERT(
|
|
aAncestorTypes.contains(AncestorType::ClosestBlockElement) ||
|
|
aAncestorTypes.contains(AncestorType::MostDistantInlineElementInBlock) ||
|
|
aAncestorTypes.contains(AncestorType::ButtonElement) ||
|
|
aAncestorTypes.contains(AncestorType::AllowRootOrAncestorLimiterElement));
|
|
|
|
const Element* theBodyElement = aContent.OwnerDoc()->GetBody();
|
|
const Element* theDocumentElement = aContent.OwnerDoc()->GetDocumentElement();
|
|
const bool editableElementOnly =
|
|
aAncestorTypes.contains(AncestorType::EditableElement);
|
|
const bool lookingForClosestBlockElement =
|
|
aAncestorTypes.contains(AncestorType::ClosestBlockElement);
|
|
const bool lookingForMostDistantInlineElementInBlock =
|
|
aAncestorTypes.contains(AncestorType::MostDistantInlineElementInBlock);
|
|
const bool lookingForButtonElement =
|
|
aAncestorTypes.contains(AncestorType::ButtonElement);
|
|
const bool ignoreHRElement =
|
|
aAncestorTypes.contains(AncestorType::IgnoreHRElement);
|
|
const bool lookingForAnyElement =
|
|
aAncestorTypes.contains(AncestorType::AllowRootOrAncestorLimiterElement);
|
|
auto IsSearchingElementType = [&](const nsIContent& aContent) -> bool {
|
|
if (lookingForAnyElement) {
|
|
return aContent.IsElement();
|
|
}
|
|
if (!aContent.IsElement() ||
|
|
(ignoreHRElement && aContent.IsHTMLElement(nsGkAtoms::hr))) {
|
|
return false;
|
|
}
|
|
if (editableElementOnly &&
|
|
!EditorUtils::IsEditableContent(aContent, EditorType::HTML)) {
|
|
return false;
|
|
}
|
|
return (lookingForClosestBlockElement &&
|
|
HTMLEditUtils::IsBlockElement(aContent, aBlockInlineCheck)) ||
|
|
(lookingForMostDistantInlineElementInBlock &&
|
|
HTMLEditUtils::IsInlineContent(aContent, aBlockInlineCheck)) ||
|
|
(lookingForButtonElement &&
|
|
aContent.IsHTMLElement(nsGkAtoms::button));
|
|
};
|
|
|
|
// If aContent is the body element or the document element, we shouldn't climb
|
|
// up to its parent.
|
|
if (editableElementOnly &&
|
|
(&aContent == theBodyElement || &aContent == theDocumentElement)) {
|
|
return IsSearchingElementType(aContent)
|
|
? const_cast<Element*>(aContent.AsElement())
|
|
: nullptr;
|
|
}
|
|
|
|
if (lookingForButtonElement && aContent.IsHTMLElement(nsGkAtoms::button)) {
|
|
return const_cast<Element*>(aContent.AsElement());
|
|
}
|
|
|
|
// If aContent is a block element, we don't need to climb up the tree.
|
|
// Consider the result right now.
|
|
if ((lookingForClosestBlockElement ||
|
|
lookingForMostDistantInlineElementInBlock) &&
|
|
HTMLEditUtils::IsBlockElement(aContent, aBlockInlineCheck) &&
|
|
!(ignoreHRElement && aContent.IsHTMLElement(nsGkAtoms::hr))) {
|
|
return IsSearchingElementType(aContent)
|
|
? const_cast<Element*>(aContent.AsElement())
|
|
: nullptr;
|
|
}
|
|
|
|
// If aContent is the last element to search range because of the parent
|
|
// element type, consider the result before calling GetAncestorElement()
|
|
// because it won't return aContent.
|
|
if (!aContent.GetParent() ||
|
|
(editableElementOnly && !EditorUtils::IsEditableContent(
|
|
*aContent.GetParent(), EditorType::HTML)) ||
|
|
(!lookingForClosestBlockElement &&
|
|
HTMLEditUtils::IsBlockElement(*aContent.GetParent(),
|
|
aBlockInlineCheck) &&
|
|
!(ignoreHRElement &&
|
|
aContent.GetParent()->IsHTMLElement(nsGkAtoms::hr)))) {
|
|
return IsSearchingElementType(aContent)
|
|
? const_cast<Element*>(aContent.AsElement())
|
|
: nullptr;
|
|
}
|
|
|
|
if (&aContent == aAncestorLimiter) {
|
|
return aAncestorTypes.contains(
|
|
AncestorType::AllowRootOrAncestorLimiterElement)
|
|
? Element::FromNode(const_cast<nsIContent&>(aContent))
|
|
: nullptr;
|
|
}
|
|
|
|
return HTMLEditUtils::GetAncestorElement(aContent, aAncestorTypes,
|
|
aBlockInlineCheck, aAncestorLimiter);
|
|
}
|
|
|
|
// static
|
|
Element* HTMLEditUtils::GetClosestAncestorAnyListElement(
|
|
const nsIContent& aContent) {
|
|
for (Element* element : aContent.AncestorsOfType<Element>()) {
|
|
if (HTMLEditUtils::IsAnyListElement(element)) {
|
|
return element;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
// static
|
|
Element* HTMLEditUtils::GetClosestInclusiveAncestorAnyListElement(
|
|
const nsIContent& aContent) {
|
|
for (Element* element : aContent.InclusiveAncestorsOfType<Element>()) {
|
|
if (HTMLEditUtils::IsAnyListElement(element)) {
|
|
return element;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
EditAction HTMLEditUtils::GetEditActionForInsert(const nsAtom& aTagName) {
|
|
// This method may be in a hot path. So, return only necessary
|
|
// EditAction::eInsert*Element.
|
|
if (&aTagName == nsGkAtoms::ul) {
|
|
// For InputEvent.inputType, "insertUnorderedList".
|
|
return EditAction::eInsertUnorderedListElement;
|
|
}
|
|
if (&aTagName == nsGkAtoms::ol) {
|
|
// For InputEvent.inputType, "insertOrderedList".
|
|
return EditAction::eInsertOrderedListElement;
|
|
}
|
|
if (&aTagName == nsGkAtoms::hr) {
|
|
// For InputEvent.inputType, "insertHorizontalRule".
|
|
return EditAction::eInsertHorizontalRuleElement;
|
|
}
|
|
return EditAction::eInsertNode;
|
|
}
|
|
|
|
EditAction HTMLEditUtils::GetEditActionForRemoveList(const nsAtom& aTagName) {
|
|
// This method may be in a hot path. So, return only necessary
|
|
// EditAction::eRemove*Element.
|
|
if (&aTagName == nsGkAtoms::ul) {
|
|
// For InputEvent.inputType, "insertUnorderedList".
|
|
return EditAction::eRemoveUnorderedListElement;
|
|
}
|
|
if (&aTagName == nsGkAtoms::ol) {
|
|
// For InputEvent.inputType, "insertOrderedList".
|
|
return EditAction::eRemoveOrderedListElement;
|
|
}
|
|
return EditAction::eRemoveListElement;
|
|
}
|
|
|
|
EditAction HTMLEditUtils::GetEditActionForInsert(const Element& aElement) {
|
|
return GetEditActionForInsert(*aElement.NodeInfo()->NameAtom());
|
|
}
|
|
|
|
EditAction HTMLEditUtils::GetEditActionForFormatText(const nsAtom& aProperty,
|
|
const nsAtom* aAttribute,
|
|
bool aToSetStyle) {
|
|
// This method may be in a hot path. So, return only necessary
|
|
// EditAction::eSet*Property or EditAction::eRemove*Property.
|
|
if (&aProperty == nsGkAtoms::b) {
|
|
return aToSetStyle ? EditAction::eSetFontWeightProperty
|
|
: EditAction::eRemoveFontWeightProperty;
|
|
}
|
|
if (&aProperty == nsGkAtoms::i) {
|
|
return aToSetStyle ? EditAction::eSetTextStyleProperty
|
|
: EditAction::eRemoveTextStyleProperty;
|
|
}
|
|
if (&aProperty == nsGkAtoms::u) {
|
|
return aToSetStyle ? EditAction::eSetTextDecorationPropertyUnderline
|
|
: EditAction::eRemoveTextDecorationPropertyUnderline;
|
|
}
|
|
if (&aProperty == nsGkAtoms::strike) {
|
|
return aToSetStyle ? EditAction::eSetTextDecorationPropertyLineThrough
|
|
: EditAction::eRemoveTextDecorationPropertyLineThrough;
|
|
}
|
|
if (&aProperty == nsGkAtoms::sup) {
|
|
return aToSetStyle ? EditAction::eSetVerticalAlignPropertySuper
|
|
: EditAction::eRemoveVerticalAlignPropertySuper;
|
|
}
|
|
if (&aProperty == nsGkAtoms::sub) {
|
|
return aToSetStyle ? EditAction::eSetVerticalAlignPropertySub
|
|
: EditAction::eRemoveVerticalAlignPropertySub;
|
|
}
|
|
if (&aProperty == nsGkAtoms::font) {
|
|
if (aAttribute == nsGkAtoms::face) {
|
|
return aToSetStyle ? EditAction::eSetFontFamilyProperty
|
|
: EditAction::eRemoveFontFamilyProperty;
|
|
}
|
|
if (aAttribute == nsGkAtoms::color) {
|
|
return aToSetStyle ? EditAction::eSetColorProperty
|
|
: EditAction::eRemoveColorProperty;
|
|
}
|
|
if (aAttribute == nsGkAtoms::bgcolor) {
|
|
return aToSetStyle ? EditAction::eSetBackgroundColorPropertyInline
|
|
: EditAction::eRemoveBackgroundColorPropertyInline;
|
|
}
|
|
}
|
|
return aToSetStyle ? EditAction::eSetInlineStyleProperty
|
|
: EditAction::eRemoveInlineStyleProperty;
|
|
}
|
|
|
|
EditAction HTMLEditUtils::GetEditActionForAlignment(
|
|
const nsAString& aAlignType) {
|
|
// This method may be in a hot path. So, return only necessary
|
|
// EditAction::eAlign*.
|
|
if (aAlignType.EqualsLiteral("left")) {
|
|
return EditAction::eAlignLeft;
|
|
}
|
|
if (aAlignType.EqualsLiteral("right")) {
|
|
return EditAction::eAlignRight;
|
|
}
|
|
if (aAlignType.EqualsLiteral("center")) {
|
|
return EditAction::eAlignCenter;
|
|
}
|
|
if (aAlignType.EqualsLiteral("justify")) {
|
|
return EditAction::eJustify;
|
|
}
|
|
return EditAction::eSetAlignment;
|
|
}
|
|
|
|
// static
|
|
template <typename EditorDOMPointType>
|
|
nsIContent* HTMLEditUtils::GetContentToPreserveInlineStyles(
|
|
const EditorDOMPointType& aPoint, const Element& aEditingHost) {
|
|
MOZ_ASSERT(aPoint.IsSetAndValid());
|
|
if (MOZ_UNLIKELY(!aPoint.IsInContentNode())) {
|
|
return nullptr;
|
|
}
|
|
// If it points middle of a text node, use it. Otherwise, scan next visible
|
|
// thing and use the style of following text node if there is.
|
|
if (aPoint.IsInTextNode() && !aPoint.IsEndOfContainer()) {
|
|
return aPoint.template ContainerAs<nsIContent>();
|
|
}
|
|
for (auto point = aPoint.template To<EditorRawDOMPoint>(); point.IsSet();) {
|
|
const WSScanResult nextVisibleThing =
|
|
WSRunScanner::ScanInclusiveNextVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::EditableNodes, point,
|
|
BlockInlineCheck::UseComputedDisplayOutsideStyle);
|
|
if (nextVisibleThing.InVisibleOrCollapsibleCharacters()) {
|
|
return nextVisibleThing.TextPtr();
|
|
}
|
|
if (nextVisibleThing.IsContentEditableRoot()) {
|
|
break;
|
|
}
|
|
// Ignore empty inline container elements because it's not visible for
|
|
// users so that using the style will appear suddenly from point of
|
|
// view of users.
|
|
if (nextVisibleThing.ReachedSpecialContent() &&
|
|
nextVisibleThing.IsContentEditable() &&
|
|
nextVisibleThing.ContentIsElement() &&
|
|
!nextVisibleThing.ElementPtr()->HasChildNodes() &&
|
|
HTMLEditUtils::IsContainerNode(*nextVisibleThing.ElementPtr())) {
|
|
point.SetAfter(nextVisibleThing.ElementPtr());
|
|
continue;
|
|
}
|
|
// Otherwise, we should use style of the container of the start point.
|
|
break;
|
|
}
|
|
return aPoint.template ContainerAs<nsIContent>();
|
|
}
|
|
|
|
template <typename EditorDOMPointType, typename EditorDOMPointTypeInput>
|
|
EditorDOMPointType HTMLEditUtils::GetBetterInsertionPointFor(
|
|
const nsIContent& aContentToInsert,
|
|
const EditorDOMPointTypeInput& aPointToInsert) {
|
|
if (NS_WARN_IF(!aPointToInsert.IsSet())) {
|
|
return EditorDOMPointType();
|
|
}
|
|
|
|
auto pointToInsert =
|
|
aPointToInsert.template GetNonAnonymousSubtreePoint<EditorDOMPointType>();
|
|
if (NS_WARN_IF(!pointToInsert.IsSet()) ||
|
|
NS_WARN_IF(!HTMLEditUtils::IsSimplyEditableNode(
|
|
*pointToInsert.GetContainer()))) {
|
|
// Cannot insert aContentToInsert into this DOM tree.
|
|
return EditorDOMPointType();
|
|
}
|
|
|
|
// If the node to insert is not a block level element, we can insert it
|
|
// at any point.
|
|
if (!HTMLEditUtils::IsBlockElement(
|
|
aContentToInsert, BlockInlineCheck::UseComputedDisplayStyle)) {
|
|
return pointToInsert;
|
|
}
|
|
|
|
const WSRunScanner wsScannerForPointToInsert(
|
|
WSRunScanner::Scan::EditableNodes, pointToInsert,
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
|
|
// If the insertion position is after the last visible item in a line,
|
|
// i.e., the insertion position is just before a visible line break <br>,
|
|
// we want to skip to the position just after the line break (see bug 68767).
|
|
const WSScanResult forwardScanFromPointToInsertResult =
|
|
wsScannerForPointToInsert.ScanInclusiveNextVisibleNodeOrBlockBoundaryFrom(
|
|
pointToInsert);
|
|
// So, if the next visible node isn't a <br> element, we can insert the block
|
|
// level element to the point.
|
|
if (!forwardScanFromPointToInsertResult.ReachedBRElement()) {
|
|
return pointToInsert;
|
|
}
|
|
|
|
// However, we must not skip next <br> element when the caret appears to be
|
|
// positioned at the beginning of a block, in that case skipping the <br>
|
|
// would not insert the <br> at the caret position, but after the current
|
|
// empty line.
|
|
const WSScanResult backwardScanFromPointToInsertResult =
|
|
wsScannerForPointToInsert.ScanPreviousVisibleNodeOrBlockBoundaryFrom(
|
|
pointToInsert);
|
|
// So, if there is no previous visible node,
|
|
// or, if both nodes of the insertion point is <br> elements,
|
|
// or, if the previous visible node is different block,
|
|
// we need to skip the following <br>. So, otherwise, we can insert the
|
|
// block at the insertion point.
|
|
if (NS_WARN_IF(backwardScanFromPointToInsertResult.Failed()) ||
|
|
backwardScanFromPointToInsertResult.ReachedInlineEditingHostBoundary() ||
|
|
backwardScanFromPointToInsertResult.ReachedBRElement() ||
|
|
backwardScanFromPointToInsertResult.ReachedCurrentBlockBoundary()) {
|
|
return pointToInsert;
|
|
}
|
|
|
|
return forwardScanFromPointToInsertResult
|
|
.template PointAfterReachedContent<EditorDOMPointType>();
|
|
}
|
|
|
|
// static
|
|
template <typename EditorDOMPointType, typename EditorDOMPointTypeInput>
|
|
EditorDOMPointType HTMLEditUtils::GetBetterCaretPositionToInsertText(
|
|
const EditorDOMPointTypeInput& aPoint) {
|
|
MOZ_ASSERT(aPoint.IsSetAndValid());
|
|
MOZ_ASSERT(HTMLEditUtils::IsSimplyEditableNode(*aPoint.GetContainer()));
|
|
|
|
if (aPoint.IsInTextNode()) {
|
|
return aPoint.template To<EditorDOMPointType>();
|
|
}
|
|
if (!aPoint.IsEndOfContainer() && aPoint.GetChild() &&
|
|
aPoint.GetChild()->IsText()) {
|
|
return EditorDOMPointType(aPoint.GetChild(), 0u);
|
|
}
|
|
if (aPoint.IsEndOfContainer()) {
|
|
const WSScanResult previousThing =
|
|
WSRunScanner::ScanPreviousVisibleNodeOrBlockBoundary(
|
|
WSRunScanner::Scan::EditableNodes, aPoint,
|
|
BlockInlineCheck::UseComputedDisplayStyle);
|
|
if (previousThing.InVisibleOrCollapsibleCharacters()) {
|
|
return EditorDOMPointType::AtEndOf(*previousThing.TextPtr());
|
|
}
|
|
}
|
|
if (HTMLEditUtils::CanNodeContain(*aPoint.GetContainer(),
|
|
*nsGkAtoms::textTagName)) {
|
|
return aPoint.template To<EditorDOMPointType>();
|
|
}
|
|
if (MOZ_UNLIKELY(aPoint.GetContainer()->IsEditingHost() ||
|
|
!aPoint.template GetContainerParentAs<nsIContent>() ||
|
|
!HTMLEditUtils::CanNodeContain(
|
|
*aPoint.template ContainerParentAs<nsIContent>(),
|
|
*nsGkAtoms::textTagName))) {
|
|
return EditorDOMPointType();
|
|
}
|
|
return aPoint.ParentPoint().template To<EditorDOMPointType>();
|
|
}
|
|
|
|
// static
|
|
template <typename EditorDOMPointType, typename EditorDOMPointTypeInput>
|
|
Result<EditorDOMPointType, nsresult>
|
|
HTMLEditUtils::ComputePointToPutCaretInElementIfOutside(
|
|
const Element& aElement, const EditorDOMPointTypeInput& aCurrentPoint) {
|
|
MOZ_ASSERT(aCurrentPoint.IsSet());
|
|
|
|
// FYI: This was moved from
|
|
// https://searchfox.org/mozilla-central/rev/d3c2f51d89c3ca008ff0cb5a057e77ccd973443e/editor/libeditor/HTMLEditSubActionHandler.cpp#9193
|
|
|
|
// Use range boundaries and RangeUtils::CompareNodeToRange() to compare
|
|
// selection start to new block.
|
|
bool nodeBefore, nodeAfter;
|
|
nsresult rv = RangeUtils::CompareNodeToRangeBoundaries(
|
|
const_cast<Element*>(&aElement), aCurrentPoint.ToRawRangeBoundary(),
|
|
aCurrentPoint.ToRawRangeBoundary(), &nodeBefore, &nodeAfter);
|
|
if (NS_FAILED(rv)) {
|
|
NS_WARNING("RangeUtils::CompareNodeToRange() failed");
|
|
return Err(rv);
|
|
}
|
|
|
|
if (nodeBefore && nodeAfter) {
|
|
return EditorDOMPointType(); // aCurrentPoint is in aElement
|
|
}
|
|
|
|
if (nodeBefore) {
|
|
// selection is after block. put at end of block.
|
|
const nsIContent* lastEditableContent = HTMLEditUtils::GetLastChild(
|
|
aElement, {WalkTreeOption::IgnoreNonEditableNode});
|
|
if (!lastEditableContent) {
|
|
lastEditableContent = &aElement;
|
|
}
|
|
if (lastEditableContent->IsText() ||
|
|
HTMLEditUtils::IsContainerNode(*lastEditableContent)) {
|
|
return EditorDOMPointType::AtEndOf(*lastEditableContent);
|
|
}
|
|
MOZ_ASSERT(lastEditableContent->GetParentNode());
|
|
return EditorDOMPointType::After(*lastEditableContent);
|
|
}
|
|
|
|
// selection is before block. put at start of block.
|
|
const nsIContent* firstEditableContent = HTMLEditUtils::GetFirstChild(
|
|
aElement, {WalkTreeOption::IgnoreNonEditableNode});
|
|
if (!firstEditableContent) {
|
|
firstEditableContent = &aElement;
|
|
}
|
|
if (firstEditableContent->IsText() ||
|
|
HTMLEditUtils::IsContainerNode(*firstEditableContent)) {
|
|
MOZ_ASSERT(firstEditableContent->GetParentNode());
|
|
// XXX Shouldn't this be EditorDOMPointType(firstEditableContent, 0u)?
|
|
return EditorDOMPointType(firstEditableContent);
|
|
}
|
|
// XXX And shouldn't this be EditorDOMPointType(firstEditableContent)?
|
|
return EditorDOMPointType(firstEditableContent, 0u);
|
|
}
|
|
|
|
// static
|
|
bool HTMLEditUtils::IsInlineStyleSetByElement(
|
|
const nsIContent& aContent, const EditorInlineStyle& aStyle,
|
|
const nsAString* aValue, nsAString* aOutValue /* = nullptr */) {
|
|
for (Element* element : aContent.InclusiveAncestorsOfType<Element>()) {
|
|
if (aStyle.mHTMLProperty != element->NodeInfo()->NameAtom()) {
|
|
continue;
|
|
}
|
|
if (!aStyle.mAttribute) {
|
|
return true;
|
|
}
|
|
nsAutoString value;
|
|
element->GetAttr(aStyle.mAttribute, value);
|
|
if (aOutValue) {
|
|
*aOutValue = value;
|
|
}
|
|
if (!value.IsEmpty()) {
|
|
if (!aValue) {
|
|
return true;
|
|
}
|
|
if (aValue->Equals(value, nsCaseInsensitiveStringComparator)) {
|
|
return true;
|
|
}
|
|
// We found the prop with the attribute, but the value doesn't match.
|
|
return false;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// static
|
|
size_t HTMLEditUtils::CollectChildren(
|
|
const nsINode& aNode,
|
|
nsTArray<OwningNonNull<nsIContent>>& aOutArrayOfContents,
|
|
size_t aIndexToInsertChildren, const CollectChildrenOptions& aOptions) {
|
|
// FYI: This was moved from
|
|
// https://searchfox.org/mozilla-central/rev/4bce7d85ba4796dd03c5dcc7cfe8eee0e4c07b3b/editor/libeditor/HTMLEditSubActionHandler.cpp#6261
|
|
|
|
size_t numberOfFoundChildren = 0;
|
|
for (nsIContent* content =
|
|
GetFirstChild(aNode, {WalkTreeOption::IgnoreNonEditableNode});
|
|
content; content = content->GetNextSibling()) {
|
|
if ((aOptions.contains(CollectChildrenOption::CollectListChildren) &&
|
|
(HTMLEditUtils::IsAnyListElement(content) ||
|
|
HTMLEditUtils::IsListItem(content))) ||
|
|
(aOptions.contains(CollectChildrenOption::CollectTableChildren) &&
|
|
HTMLEditUtils::IsAnyTableElement(content))) {
|
|
numberOfFoundChildren += HTMLEditUtils::CollectChildren(
|
|
*content, aOutArrayOfContents,
|
|
aIndexToInsertChildren + numberOfFoundChildren, aOptions);
|
|
continue;
|
|
}
|
|
|
|
if (aOptions.contains(CollectChildrenOption::IgnoreNonEditableChildren) &&
|
|
!EditorUtils::IsEditableContent(*content, EditorType::HTML)) {
|
|
continue;
|
|
}
|
|
if (aOptions.contains(CollectChildrenOption::IgnoreInvisibleTextNodes) &&
|
|
content->IsText() &&
|
|
!HTMLEditUtils::IsVisibleTextNode(*content->AsText())) {
|
|
continue;
|
|
}
|
|
aOutArrayOfContents.InsertElementAt(
|
|
aIndexToInsertChildren + numberOfFoundChildren++, *content);
|
|
}
|
|
return numberOfFoundChildren;
|
|
}
|
|
|
|
// static
|
|
size_t HTMLEditUtils::CollectEmptyInlineContainerDescendants(
|
|
const nsINode& aNode,
|
|
nsTArray<OwningNonNull<nsIContent>>& aOutArrayOfContents,
|
|
const EmptyCheckOptions& aOptions, BlockInlineCheck aBlockInlineCheck) {
|
|
size_t numberOfFoundElements = 0;
|
|
for (Element* element = aNode.GetFirstElementChild(); element;) {
|
|
if (HTMLEditUtils::IsEmptyInlineContainer(*element, aOptions,
|
|
aBlockInlineCheck)) {
|
|
aOutArrayOfContents.AppendElement(*element);
|
|
numberOfFoundElements++;
|
|
nsIContent* nextContent = element->GetNextNonChildNode(&aNode);
|
|
element = nullptr;
|
|
for (; nextContent; nextContent = nextContent->GetNextNode(&aNode)) {
|
|
if (nextContent->IsElement()) {
|
|
element = nextContent->AsElement();
|
|
break;
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
|
|
nsIContent* nextContent = element->GetNextNode(&aNode);
|
|
element = nullptr;
|
|
for (; nextContent; nextContent = nextContent->GetNextNode(&aNode)) {
|
|
if (nextContent->IsElement()) {
|
|
element = nextContent->AsElement();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return numberOfFoundElements;
|
|
}
|
|
|
|
// static
|
|
bool HTMLEditUtils::ElementHasAttributeExcept(const Element& aElement,
|
|
const nsAtom& aAttribute1,
|
|
const nsAtom& aAttribute2,
|
|
const nsAtom& aAttribute3) {
|
|
// FYI: This was moved from
|
|
// https://searchfox.org/mozilla-central/rev/0b1543e85d13c30a13c57e959ce9815a3f0fa1d3/editor/libeditor/HTMLStyleEditor.cpp#1626
|
|
for (auto i : IntegerRange<uint32_t>(aElement.GetAttrCount())) {
|
|
const nsAttrName* name = aElement.GetAttrNameAt(i);
|
|
if (!name->NamespaceEquals(kNameSpaceID_None)) {
|
|
return true;
|
|
}
|
|
|
|
if (name->LocalName() == &aAttribute1 ||
|
|
name->LocalName() == &aAttribute2 ||
|
|
name->LocalName() == &aAttribute3) {
|
|
continue; // Ignore the given attribute
|
|
}
|
|
|
|
// Ignore empty style, class and id attributes because those attributes are
|
|
// not meaningful with empty value.
|
|
if (name->LocalName() == nsGkAtoms::style ||
|
|
name->LocalName() == nsGkAtoms::_class ||
|
|
name->LocalName() == nsGkAtoms::id) {
|
|
if (aElement.HasNonEmptyAttr(name->LocalName())) {
|
|
return true;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// Ignore special _moz attributes
|
|
nsAutoString attrString;
|
|
name->LocalName()->ToString(attrString);
|
|
if (!StringBeginsWith(attrString, u"_moz"_ns)) {
|
|
return true;
|
|
}
|
|
}
|
|
// if we made it through all of them without finding a real attribute
|
|
// other than aAttribute, then return true
|
|
return false;
|
|
}
|
|
|
|
bool HTMLEditUtils::GetNormalizedHTMLColorValue(const nsAString& aColorValue,
|
|
nsAString& aNormalizedValue) {
|
|
nsAttrValue value;
|
|
if (!value.ParseColor(aColorValue)) {
|
|
aNormalizedValue = aColorValue;
|
|
return false;
|
|
}
|
|
nscolor color = NS_RGB(0, 0, 0);
|
|
MOZ_ALWAYS_TRUE(value.GetColorValue(color));
|
|
aNormalizedValue = NS_ConvertASCIItoUTF16(nsPrintfCString(
|
|
"#%02x%02x%02x", NS_GET_R(color), NS_GET_G(color), NS_GET_B(color)));
|
|
return true;
|
|
}
|
|
|
|
bool HTMLEditUtils::IsSameHTMLColorValue(
|
|
const nsAString& aColorA, const nsAString& aColorB,
|
|
TransparentKeyword aTransparentKeyword) {
|
|
if (aTransparentKeyword == TransparentKeyword::Allowed) {
|
|
const bool isATransparent = aColorA.LowerCaseEqualsLiteral("transparent");
|
|
const bool isBTransparent = aColorB.LowerCaseEqualsLiteral("transparent");
|
|
if (isATransparent || isBTransparent) {
|
|
return isATransparent && isBTransparent;
|
|
}
|
|
}
|
|
nsAttrValue valueA, valueB;
|
|
if (!valueA.ParseColor(aColorA) || !valueB.ParseColor(aColorB)) {
|
|
return false;
|
|
}
|
|
nscolor colorA = NS_RGB(0, 0, 0), colorB = NS_RGB(0, 0, 0);
|
|
MOZ_ALWAYS_TRUE(valueA.GetColorValue(colorA));
|
|
MOZ_ALWAYS_TRUE(valueB.GetColorValue(colorB));
|
|
return colorA == colorB;
|
|
}
|
|
|
|
bool HTMLEditUtils::MaybeCSSSpecificColorValue(const nsAString& aColorValue) {
|
|
if (aColorValue.IsEmpty() || aColorValue.First() == '#') {
|
|
return false; // Quick return for the most cases.
|
|
}
|
|
|
|
nsAutoString colorValue(aColorValue);
|
|
colorValue.CompressWhitespace(true, true);
|
|
if (colorValue.LowerCaseEqualsASCII("transparent")) {
|
|
return true;
|
|
}
|
|
nscolor color = NS_RGB(0, 0, 0);
|
|
if (colorValue.IsEmpty() || colorValue.First() == '#') {
|
|
return false;
|
|
}
|
|
const NS_ConvertUTF16toUTF8 colorU8(colorValue);
|
|
if (Servo_ColorNameToRgb(&colorU8, &color)) {
|
|
return false;
|
|
}
|
|
if (colorValue.LowerCaseEqualsASCII("initial") ||
|
|
colorValue.LowerCaseEqualsASCII("inherit") ||
|
|
colorValue.LowerCaseEqualsASCII("unset") ||
|
|
colorValue.LowerCaseEqualsASCII("revert") ||
|
|
colorValue.LowerCaseEqualsASCII("currentcolor")) {
|
|
return true;
|
|
}
|
|
return ServoCSSParser::IsValidCSSColor(colorU8);
|
|
}
|
|
|
|
static bool ComputeColor(const nsAString& aColorValue, nscolor* aColor,
|
|
bool* aIsCurrentColor) {
|
|
return ServoCSSParser::ComputeColor(nullptr, NS_RGB(0, 0, 0),
|
|
NS_ConvertUTF16toUTF8(aColorValue),
|
|
aColor, aIsCurrentColor);
|
|
}
|
|
|
|
static bool ComputeColor(const nsACString& aColorValue, nscolor* aColor,
|
|
bool* aIsCurrentColor) {
|
|
return ServoCSSParser::ComputeColor(nullptr, NS_RGB(0, 0, 0), aColorValue,
|
|
aColor, aIsCurrentColor);
|
|
}
|
|
|
|
bool HTMLEditUtils::CanConvertToHTMLColorValue(const nsAString& aColorValue) {
|
|
bool isCurrentColor = false;
|
|
nscolor color = NS_RGB(0, 0, 0);
|
|
return ComputeColor(aColorValue, &color, &isCurrentColor) &&
|
|
!isCurrentColor && NS_GET_A(color) == 0xFF;
|
|
}
|
|
|
|
bool HTMLEditUtils::ConvertToNormalizedHTMLColorValue(
|
|
const nsAString& aColorValue, nsAString& aNormalizedValue) {
|
|
bool isCurrentColor = false;
|
|
nscolor color = NS_RGB(0, 0, 0);
|
|
if (!ComputeColor(aColorValue, &color, &isCurrentColor) || isCurrentColor ||
|
|
NS_GET_A(color) != 0xFF) {
|
|
aNormalizedValue = aColorValue;
|
|
return false;
|
|
}
|
|
aNormalizedValue.Truncate();
|
|
aNormalizedValue.AppendPrintf("#%02x%02x%02x", NS_GET_R(color),
|
|
NS_GET_G(color), NS_GET_B(color));
|
|
return true;
|
|
}
|
|
|
|
bool HTMLEditUtils::GetNormalizedCSSColorValue(const nsAString& aColorValue,
|
|
ZeroAlphaColor aZeroAlphaColor,
|
|
nsAString& aNormalizedValue) {
|
|
bool isCurrentColor = false;
|
|
nscolor color = NS_RGB(0, 0, 0);
|
|
if (!ComputeColor(aColorValue, &color, &isCurrentColor)) {
|
|
aNormalizedValue = aColorValue;
|
|
return false;
|
|
}
|
|
|
|
// If it's currentcolor, let's return it as-is since we cannot resolve it
|
|
// without ancestors.
|
|
if (isCurrentColor) {
|
|
aNormalizedValue = aColorValue;
|
|
return true;
|
|
}
|
|
|
|
if (aZeroAlphaColor == ZeroAlphaColor::TransparentKeyword &&
|
|
NS_GET_A(color) == 0) {
|
|
aNormalizedValue.AssignLiteral("transparent");
|
|
return true;
|
|
}
|
|
|
|
// Get serialized color value (i.e., "rgb()" or "rgba()").
|
|
aNormalizedValue.Truncate();
|
|
nsStyleUtil::GetSerializedColorValue(color, aNormalizedValue);
|
|
return true;
|
|
}
|
|
|
|
template <typename CharType>
|
|
bool HTMLEditUtils::IsSameCSSColorValue(const nsTSubstring<CharType>& aColorA,
|
|
const nsTSubstring<CharType>& aColorB) {
|
|
bool isACurrentColor = false;
|
|
nscolor colorA = NS_RGB(0, 0, 0);
|
|
if (!ComputeColor(aColorA, &colorA, &isACurrentColor)) {
|
|
return false;
|
|
}
|
|
bool isBCurrentColor = false;
|
|
nscolor colorB = NS_RGB(0, 0, 0);
|
|
if (!ComputeColor(aColorB, &colorB, &isBCurrentColor)) {
|
|
return false;
|
|
}
|
|
if (isACurrentColor || isBCurrentColor) {
|
|
return isACurrentColor && isBCurrentColor;
|
|
}
|
|
return colorA == colorB;
|
|
}
|
|
|
|
bool HTMLEditUtils::IsTransparentCSSColor(const nsAString& aColor) {
|
|
nsAutoString normalizedCSSColorValue;
|
|
return GetNormalizedCSSColorValue(aColor, ZeroAlphaColor::TransparentKeyword,
|
|
normalizedCSSColorValue) &&
|
|
normalizedCSSColorValue.EqualsASCII("transparent");
|
|
}
|
|
|
|
/******************************************************************************
|
|
* SelectedTableCellScanner
|
|
******************************************************************************/
|
|
|
|
SelectedTableCellScanner::SelectedTableCellScanner(
|
|
const AutoClonedRangeArray& aRanges) {
|
|
if (aRanges.Ranges().IsEmpty()) {
|
|
return;
|
|
}
|
|
Element* firstSelectedCellElement =
|
|
HTMLEditUtils::GetTableCellElementIfOnlyOneSelected(
|
|
aRanges.FirstRangeRef());
|
|
if (!firstSelectedCellElement) {
|
|
return; // We're not in table cell selection mode.
|
|
}
|
|
mSelectedCellElements.SetCapacity(aRanges.Ranges().Length());
|
|
mSelectedCellElements.AppendElement(*firstSelectedCellElement);
|
|
for (uint32_t i = 1; i < aRanges.Ranges().Length(); i++) {
|
|
nsRange* range = aRanges.Ranges()[i];
|
|
if (NS_WARN_IF(!range) || NS_WARN_IF(!range->IsPositioned())) {
|
|
continue; // Shouldn't occur in normal conditions.
|
|
}
|
|
// Just ignore selection ranges which do not select only one table
|
|
// cell element. This is possible case if web apps sets multiple
|
|
// selections and first range selects a table cell element.
|
|
if (Element* selectedCellElement =
|
|
HTMLEditUtils::GetTableCellElementIfOnlyOneSelected(*range)) {
|
|
mSelectedCellElements.AppendElement(*selectedCellElement);
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace mozilla
|