diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 17:32:43 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 17:32:43 +0000 |
commit | 6bf0a5cb5034a7e684dcc3500e841785237ce2dd (patch) | |
tree | a68f146d7fa01f0134297619fbe7e33db084e0aa /editor/libeditor/HTMLEditor.cpp | |
parent | Initial commit. (diff) | |
download | thunderbird-6bf0a5cb5034a7e684dcc3500e841785237ce2dd.tar.xz thunderbird-6bf0a5cb5034a7e684dcc3500e841785237ce2dd.zip |
Adding upstream version 1:115.7.0.upstream/1%115.7.0upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'editor/libeditor/HTMLEditor.cpp')
-rw-r--r-- | editor/libeditor/HTMLEditor.cpp | 7199 |
1 files changed, 7199 insertions, 0 deletions
diff --git a/editor/libeditor/HTMLEditor.cpp b/editor/libeditor/HTMLEditor.cpp new file mode 100644 index 0000000000..474ce8b841 --- /dev/null +++ b/editor/libeditor/HTMLEditor.cpp @@ -0,0 +1,7199 @@ +/* -*- 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 "HTMLEditor.h" +#include "HTMLEditHelpers.h" +#include "HTMLEditorInlines.h" + +#include "AutoRangeArray.h" +#include "CSSEditUtils.h" +#include "EditAction.h" +#include "EditorBase.h" +#include "EditorDOMPoint.h" +#include "EditorUtils.h" +#include "ErrorList.h" +#include "HTMLEditorEventListener.h" +#include "HTMLEditUtils.h" +#include "InsertNodeTransaction.h" +#include "JoinNodesTransaction.h" +#include "JoinSplitNodeDirection.h" +#include "MoveNodeTransaction.h" +#include "PendingStyles.h" +#include "ReplaceTextTransaction.h" +#include "SplitNodeTransaction.h" +#include "WSRunObject.h" + +#include "mozilla/ComposerCommandsUpdater.h" +#include "mozilla/ContentIterator.h" +#include "mozilla/DebugOnly.h" +#include "mozilla/EditorForwards.h" +#include "mozilla/Encoding.h" // for Encoding +#include "mozilla/IMEStateManager.h" +#include "mozilla/IntegerRange.h" // for IntegerRange +#include "mozilla/InternalMutationEvent.h" +#include "mozilla/mozInlineSpellChecker.h" +#include "mozilla/Preferences.h" +#include "mozilla/PresShell.h" +#include "mozilla/StaticPrefs_editor.h" +#include "mozilla/StyleSheet.h" +#include "mozilla/StyleSheetInlines.h" +#include "mozilla/Telemetry.h" +#include "mozilla/TextEvents.h" +#include "mozilla/TextServicesDocument.h" +#include "mozilla/ToString.h" +#include "mozilla/css/Loader.h" +#include "mozilla/dom/AncestorIterator.h" +#include "mozilla/dom/Attr.h" +#include "mozilla/dom/DocumentFragment.h" +#include "mozilla/dom/DocumentInlines.h" +#include "mozilla/dom/Element.h" +#include "mozilla/dom/Event.h" +#include "mozilla/dom/EventTarget.h" +#include "mozilla/dom/HTMLAnchorElement.h" +#include "mozilla/dom/HTMLBodyElement.h" +#include "mozilla/dom/HTMLBRElement.h" +#include "mozilla/dom/NameSpaceConstants.h" +#include "mozilla/dom/Selection.h" + +#include "nsContentList.h" +#include "nsContentUtils.h" +#include "nsCRT.h" +#include "nsDebug.h" +#include "nsDOMAttributeMap.h" +#include "nsElementTable.h" +#include "nsFocusManager.h" +#include "nsGenericHTMLElement.h" +#include "nsGkAtoms.h" +#include "nsHTMLDocument.h" +#include "nsIContent.h" +#include "nsIContentInlines.h" +#include "nsIEditActionListener.h" +#include "nsIFrame.h" +#include "nsIPrincipal.h" +#include "nsISelectionController.h" +#include "nsIURI.h" +#include "nsIWidget.h" +#include "nsNetUtil.h" +#include "nsPresContext.h" +#include "nsPrintfCString.h" +#include "nsPIDOMWindow.h" +#include "nsStyledElement.h" +#include "nsTextFragment.h" +#include "nsUnicharUtils.h" + +namespace mozilla { + +using namespace dom; +using namespace widget; + +using EmptyCheckOption = HTMLEditUtils::EmptyCheckOption; +using LeafNodeType = HTMLEditUtils::LeafNodeType; +using LeafNodeTypes = HTMLEditUtils::LeafNodeTypes; +using WalkTreeOption = HTMLEditUtils::WalkTreeOption; + +// Some utilities to handle overloading of "A" tag for link and named anchor. +static bool IsLinkTag(const nsAtom& aTagName) { + return &aTagName == nsGkAtoms::href; +} + +static bool IsNamedAnchorTag(const nsAtom& aTagName) { + return &aTagName == nsGkAtoms::anchor; +} + +// Helper struct for DoJoinNodes() and DoSplitNode(). +struct MOZ_STACK_CLASS SavedRange final { + RefPtr<Selection> mSelection; + nsCOMPtr<nsINode> mStartContainer; + nsCOMPtr<nsINode> mEndContainer; + uint32_t mStartOffset = 0; + uint32_t mEndOffset = 0; +}; + +/****************************************************************************** + * HTMLEditor::AutoSelectionRestorer + *****************************************************************************/ + +HTMLEditor::AutoSelectionRestorer::AutoSelectionRestorer( + HTMLEditor& aHTMLEditor) + : mHTMLEditor(nullptr) { + if (aHTMLEditor.ArePreservingSelection()) { + // We already have initialized mParentData::mSavedSelection, so this must + // be nested call. + return; + } + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + mHTMLEditor = &aHTMLEditor; + mHTMLEditor->PreserveSelectionAcrossActions(); +} + +HTMLEditor::AutoSelectionRestorer::~AutoSelectionRestorer() { + if (!mHTMLEditor || !mHTMLEditor->ArePreservingSelection()) { + return; + } + DebugOnly<nsresult> rvIgnored = mHTMLEditor->RestorePreservedSelection(); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "EditorBase::RestorePreservedSelection() failed, but ignored"); +} + +void HTMLEditor::AutoSelectionRestorer::Abort() { + if (mHTMLEditor) { + mHTMLEditor->StopPreservingSelection(); + } +} + +/****************************************************************************** + * HTMLEditor + *****************************************************************************/ + +template Result<CreateContentResult, nsresult> +HTMLEditor::InsertNodeIntoProperAncestorWithTransaction( + nsIContent& aContentToInsert, const EditorDOMPoint& aPointToInsert, + SplitAtEdges aSplitAtEdges); +template Result<CreateElementResult, nsresult> +HTMLEditor::InsertNodeIntoProperAncestorWithTransaction( + Element& aContentToInsert, const EditorDOMPoint& aPointToInsert, + SplitAtEdges aSplitAtEdges); +template Result<CreateTextResult, nsresult> +HTMLEditor::InsertNodeIntoProperAncestorWithTransaction( + Text& aContentToInsert, const EditorDOMPoint& aPointToInsert, + SplitAtEdges aSplitAtEdges); + +HTMLEditor::InitializeInsertingElement HTMLEditor::DoNothingForNewElement = + [](HTMLEditor&, Element&, const EditorDOMPoint&) { return NS_OK; }; + +HTMLEditor::InitializeInsertingElement HTMLEditor::InsertNewBRElement = + [](HTMLEditor& aHTMLEditor, Element& aNewElement, const EditorDOMPoint&) + MOZ_CAN_RUN_SCRIPT_BOUNDARY { + MOZ_ASSERT(!aNewElement.IsInComposedDoc()); + Result<CreateElementResult, nsresult> createBRElementResult = + aHTMLEditor.InsertBRElement(WithTransaction::No, + EditorDOMPoint(&aNewElement, 0u)); + if (MOZ_UNLIKELY(createBRElementResult.isErr())) { + NS_WARNING_ASSERTION( + createBRElementResult.isOk(), + "HTMLEditor::InsertBRElement(WithTransaction::No) failed"); + return createBRElementResult.unwrapErr(); + } + createBRElementResult.unwrap().IgnoreCaretPointSuggestion(); + return NS_OK; + }; + +// static +Result<CreateElementResult, nsresult> +HTMLEditor::AppendNewElementToInsertingElement( + HTMLEditor& aHTMLEditor, const nsStaticAtom& aTagName, Element& aNewElement, + const InitializeInsertingElement& aInitializer) { + MOZ_ASSERT(!aNewElement.IsInComposedDoc()); + Result<CreateElementResult, nsresult> createNewElementResult = + aHTMLEditor.CreateAndInsertElement( + WithTransaction::No, const_cast<nsStaticAtom&>(aTagName), + EditorDOMPoint(&aNewElement, 0u), aInitializer); + NS_WARNING_ASSERTION( + createNewElementResult.isOk(), + "HTMLEditor::CreateAndInsertElement(WithTransaction::No) failed"); + return createNewElementResult; +} + +// static +Result<CreateElementResult, nsresult> +HTMLEditor::AppendNewElementWithBRToInsertingElement( + HTMLEditor& aHTMLEditor, const nsStaticAtom& aTagName, + Element& aNewElement) { + MOZ_ASSERT(!aNewElement.IsInComposedDoc()); + Result<CreateElementResult, nsresult> createNewElementWithBRResult = + HTMLEditor::AppendNewElementToInsertingElement( + aHTMLEditor, aTagName, aNewElement, HTMLEditor::InsertNewBRElement); + NS_WARNING_ASSERTION( + createNewElementWithBRResult.isOk(), + "HTMLEditor::AppendNewElementToInsertingElement() failed"); + return createNewElementWithBRResult; +} + +HTMLEditor::AttributeFilter HTMLEditor::CopyAllAttributes = + [](HTMLEditor&, const Element&, const Element&, const Attr&, nsString&) { + return true; + }; +HTMLEditor::AttributeFilter HTMLEditor::CopyAllAttributesExceptId = + [](HTMLEditor&, const Element&, const Element&, const Attr& aAttr, + nsString&) { + return aAttr.NodeInfo()->NamespaceID() != kNameSpaceID_None || + aAttr.NodeInfo()->NameAtom() != nsGkAtoms::id; + }; +HTMLEditor::AttributeFilter HTMLEditor::CopyAllAttributesExceptDir = + [](HTMLEditor&, const Element&, const Element&, const Attr& aAttr, + nsString&) { + return aAttr.NodeInfo()->NamespaceID() != kNameSpaceID_None || + aAttr.NodeInfo()->NameAtom() != nsGkAtoms::dir; + }; +HTMLEditor::AttributeFilter HTMLEditor::CopyAllAttributesExceptIdAndDir = + [](HTMLEditor&, const Element&, const Element&, const Attr& aAttr, + nsString&) { + return !(aAttr.NodeInfo()->NamespaceID() == kNameSpaceID_None && + (aAttr.NodeInfo()->NameAtom() == nsGkAtoms::id || + aAttr.NodeInfo()->NameAtom() == nsGkAtoms::dir)); + }; + +static bool ShouldUseTraditionalJoinSplitDirection(const Document& aDocument) { + if (nsIPrincipal* principal = aDocument.GetPrincipalForPrefBasedHacks()) { + if (principal->IsURIInPrefList("editor.join_split_direction." + "force_use_traditional_direction")) { + return true; + } + if (principal->IsURIInPrefList("editor.join_split_direction." + "force_use_compatible_direction")) { + return false; + } + } + return !StaticPrefs:: + editor_join_split_direction_compatible_with_the_other_browsers(); +} + +HTMLEditor::HTMLEditor(const Document& aDocument) + : EditorBase(EditorBase::EditorType::HTML), + mCRInParagraphCreatesParagraph(false), + mUseGeckoTraditionalJoinSplitBehavior( + ShouldUseTraditionalJoinSplitDirection(aDocument)), + mIsObjectResizingEnabled( + StaticPrefs::editor_resizing_enabled_by_default()), + mIsResizing(false), + mPreserveRatio(false), + mResizedObjectIsAnImage(false), + mIsAbsolutelyPositioningEnabled( + StaticPrefs::editor_positioning_enabled_by_default()), + mResizedObjectIsAbsolutelyPositioned(false), + mGrabberClicked(false), + mIsMoving(false), + mSnapToGridEnabled(false), + mIsInlineTableEditingEnabled( + StaticPrefs::editor_inline_table_editing_enabled_by_default()), + mIsCSSPrefChecked(StaticPrefs::editor_use_css()), + mOriginalX(0), + mOriginalY(0), + mResizedObjectX(0), + mResizedObjectY(0), + mResizedObjectWidth(0), + mResizedObjectHeight(0), + mResizedObjectMarginLeft(0), + mResizedObjectMarginTop(0), + mResizedObjectBorderLeft(0), + mResizedObjectBorderTop(0), + mXIncrementFactor(0), + mYIncrementFactor(0), + mWidthIncrementFactor(0), + mHeightIncrementFactor(0), + mInfoXIncrement(20), + mInfoYIncrement(20), + mPositionedObjectX(0), + mPositionedObjectY(0), + mPositionedObjectWidth(0), + mPositionedObjectHeight(0), + mPositionedObjectMarginLeft(0), + mPositionedObjectMarginTop(0), + mPositionedObjectBorderLeft(0), + mPositionedObjectBorderTop(0), + mGridSize(0), + mDefaultParagraphSeparator(ParagraphSeparator::div) {} + +HTMLEditor::~HTMLEditor() { + // Collect the data of `beforeinput` event only when it's enabled because + // web apps should switch their behavior with feature detection with + // checking `onbeforeinput` or `getTargetRanges`. + if (StaticPrefs::dom_input_events_beforeinput_enabled()) { + Telemetry::Accumulate( + Telemetry::HTMLEDITORS_WITH_BEFOREINPUT_LISTENERS, + MayHaveBeforeInputEventListenersForTelemetry() ? 1 : 0); + Telemetry::Accumulate( + Telemetry::HTMLEDITORS_OVERRIDDEN_BY_BEFOREINPUT_LISTENERS, + mHasBeforeInputBeenCanceled ? 1 : 0); + Telemetry::Accumulate( + Telemetry:: + HTMLEDITORS_WITH_MUTATION_LISTENERS_WITHOUT_BEFOREINPUT_LISTENERS, + !MayHaveBeforeInputEventListenersForTelemetry() && + MayHaveMutationEventListeners() + ? 1 + : 0); + Telemetry::Accumulate( + Telemetry:: + HTMLEDITORS_WITH_MUTATION_OBSERVERS_WITHOUT_BEFOREINPUT_LISTENERS, + !MayHaveBeforeInputEventListenersForTelemetry() && + MutationObserverHasObservedNodeForTelemetry() + ? 1 + : 0); + } + + mPendingStylesToApplyToNewContent = nullptr; + + if (mDisabledLinkHandling) { + if (Document* doc = GetDocument()) { + doc->SetLinkHandlingEnabled(mOldLinkHandlingEnabled); + } + } + + RemoveEventListeners(); + + HideAnonymousEditingUIs(); +} + +NS_IMPL_CYCLE_COLLECTION_CLASS(HTMLEditor) + +NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(HTMLEditor, EditorBase) + NS_IMPL_CYCLE_COLLECTION_UNLINK(mPendingStylesToApplyToNewContent) + NS_IMPL_CYCLE_COLLECTION_UNLINK(mComposerCommandsUpdater) + NS_IMPL_CYCLE_COLLECTION_UNLINK(mChangedRangeForTopLevelEditSubAction) + NS_IMPL_CYCLE_COLLECTION_UNLINK(mPaddingBRElementForEmptyEditor) + tmp->HideAnonymousEditingUIs(); +NS_IMPL_CYCLE_COLLECTION_UNLINK_END + +NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(HTMLEditor, EditorBase) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mPendingStylesToApplyToNewContent) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mComposerCommandsUpdater) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mChangedRangeForTopLevelEditSubAction) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mPaddingBRElementForEmptyEditor) + + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTopLeftHandle) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTopHandle) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTopRightHandle) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mLeftHandle) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRightHandle) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mBottomLeftHandle) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mBottomHandle) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mBottomRightHandle) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mActivatedHandle) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mResizingShadow) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mResizingInfo) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mResizedObject) + + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mAbsolutelyPositionedObject) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mGrabber) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mPositioningShadow) + + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mInlineEditedCell) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mAddColumnBeforeButton) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRemoveColumnButton) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mAddColumnAfterButton) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mAddRowBeforeButton) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRemoveRowButton) + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mAddRowAfterButton) +NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END + +NS_IMPL_ADDREF_INHERITED(HTMLEditor, EditorBase) +NS_IMPL_RELEASE_INHERITED(HTMLEditor, EditorBase) + +NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(HTMLEditor) + NS_INTERFACE_MAP_ENTRY(nsIHTMLEditor) + NS_INTERFACE_MAP_ENTRY(nsIHTMLObjectResizer) + NS_INTERFACE_MAP_ENTRY(nsIHTMLAbsPosEditor) + NS_INTERFACE_MAP_ENTRY(nsIHTMLInlineTableEditor) + NS_INTERFACE_MAP_ENTRY(nsITableEditor) + NS_INTERFACE_MAP_ENTRY(nsIMutationObserver) + NS_INTERFACE_MAP_ENTRY(nsIEditorMailSupport) +NS_INTERFACE_MAP_END_INHERITING(EditorBase) + +nsresult HTMLEditor::Init(Document& aDocument, + ComposerCommandsUpdater& aComposerCommandsUpdater, + uint32_t aFlags) { + MOZ_ASSERT(!mInitSucceeded, + "HTMLEditor::Init() called again without calling PreDestroy()?"); + + MOZ_DIAGNOSTIC_ASSERT(!mComposerCommandsUpdater || + mComposerCommandsUpdater == &aComposerCommandsUpdater); + mComposerCommandsUpdater = &aComposerCommandsUpdater; + + RefPtr<PresShell> presShell = aDocument.GetPresShell(); + if (NS_WARN_IF(!presShell)) { + return NS_ERROR_FAILURE; + } + nsresult rv = InitInternal(aDocument, nullptr, *presShell, aFlags); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::InitInternal() failed"); + return rv; + } + + // Init mutation observer + aDocument.AddMutationObserverUnlessExists(this); + + if (!mRootElement) { + UpdateRootElement(); + } + + // disable Composer-only features + if (IsMailEditor()) { + DebugOnly<nsresult> rvIgnored = SetAbsolutePositioningEnabled(false); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "HTMLEditor::SetAbsolutePositioningEnabled(false) failed, but ignored"); + rvIgnored = SetSnapToGridEnabled(false); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "HTMLEditor::SetSnapToGridEnabled(false) failed, but ignored"); + } + + // disable links + Document* document = GetDocument(); + if (NS_WARN_IF(!document)) { + return NS_ERROR_FAILURE; + } + if (!IsInPlaintextMode() && !IsInteractionAllowed()) { + mDisabledLinkHandling = true; + mOldLinkHandlingEnabled = document->LinkHandlingEnabled(); + document->SetLinkHandlingEnabled(false); + } + + // init the type-in state + mPendingStylesToApplyToNewContent = new PendingStyles(); + + if (!IsInteractionAllowed()) { + nsCOMPtr<nsIURI> uaURI; + rv = NS_NewURI(getter_AddRefs(uaURI), + "resource://gre/res/EditorOverride.css"); + NS_ENSURE_SUCCESS(rv, rv); + + rv = document->LoadAdditionalStyleSheet(Document::eAgentSheet, uaURI); + NS_ENSURE_SUCCESS(rv, rv); + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eInitializing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_FAILURE; + } + + rv = InitEditorContentAndSelection(); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::InitEditorContentAndSelection() failed"); + // XXX Sholdn't we expose `NS_ERROR_EDITOR_DESTROYED` even though this + // is a public method? + return EditorBase::ToGenericNSResult(rv); + } + + // Throw away the old transaction manager if this is not the first time that + // we're initializing the editor. + ClearUndoRedo(); + EnableUndoRedo(); // FYI: Creating mTransactionManager in this call + + if (mTransactionManager) { + mTransactionManager->Attach(*this); + } + + MOZ_ASSERT(!mInitSucceeded, "HTMLEditor::Init() shouldn't be nested"); + mInitSucceeded = true; + return NS_OK; +} + +nsresult HTMLEditor::PostCreate() { + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + + nsresult rv = PostCreateInternal(); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::PostCreatInternal() failed"); + return rv; +} + +void HTMLEditor::PreDestroy() { + if (mDidPreDestroy) { + return; + } + + mInitSucceeded = false; + + // FYI: Cannot create AutoEditActionDataSetter here. However, it does not + // necessary for the methods called by the following code. + + RefPtr<Document> document = GetDocument(); + if (document) { + document->RemoveMutationObserver(this); + + if (!IsInteractionAllowed()) { + nsCOMPtr<nsIURI> uaURI; + nsresult rv = NS_NewURI(getter_AddRefs(uaURI), + "resource://gre/res/EditorOverride.css"); + if (NS_SUCCEEDED(rv)) { + document->RemoveAdditionalStyleSheet(Document::eAgentSheet, uaURI); + } + } + } + + // Clean up after our anonymous content -- we don't want these nodes to + // stay around (which they would, since the frames have an owning reference). + PresShell* presShell = GetPresShell(); + if (presShell && presShell->IsDestroying()) { + // Just destroying PresShell now. + // We have to keep UI elements of anonymous content until PresShell + // is destroyed. + RefPtr<HTMLEditor> self = this; + nsContentUtils::AddScriptRunner( + NS_NewRunnableFunction("HTMLEditor::PreDestroy", + [self]() { self->HideAnonymousEditingUIs(); })); + } else { + // PresShell is alive or already gone. + HideAnonymousEditingUIs(); + } + + mPaddingBRElementForEmptyEditor = nullptr; + + PreDestroyInternal(); +} + +NS_IMETHODIMP HTMLEditor::GetDocumentCharacterSet(nsACString& aCharacterSet) { + nsresult rv = GetDocumentCharsetInternal(aCharacterSet); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::GetDocumentCharsetInternal() failed"); + return rv; +} + +NS_IMETHODIMP HTMLEditor::SetDocumentCharacterSet( + const nsACString& aCharacterSet) { + AutoEditActionDataSetter editActionData(*this, EditAction::eSetCharacterSet); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent() failed"); + return EditorBase::ToGenericNSResult(rv); + } + + RefPtr<Document> document = GetDocument(); + if (NS_WARN_IF(!document)) { + return EditorBase::ToGenericNSResult(NS_ERROR_NOT_INITIALIZED); + } + // This method is scriptable, so add-ons could pass in something other + // than a canonical name. + const Encoding* encoding = Encoding::ForLabelNoReplacement(aCharacterSet); + if (!encoding) { + NS_WARNING("Encoding::ForLabelNoReplacement() failed"); + return EditorBase::ToGenericNSResult(NS_ERROR_INVALID_ARG); + } + document->SetDocumentCharacterSet(WrapNotNull(encoding)); + + // Update META charset element. + if (UpdateMetaCharsetWithTransaction(*document, aCharacterSet)) { + return NS_OK; + } + + // Set attributes to the created element + if (aCharacterSet.IsEmpty()) { + return NS_OK; + } + + RefPtr<nsContentList> headElementList = + document->GetElementsByTagName(u"head"_ns); + if (NS_WARN_IF(!headElementList)) { + return NS_OK; + } + + nsCOMPtr<nsIContent> primaryHeadElement = headElementList->Item(0); + if (NS_WARN_IF(!primaryHeadElement)) { + return NS_OK; + } + + // Create a new meta charset tag + Result<CreateElementResult, nsresult> createNewMetaElementResult = + CreateAndInsertElement( + WithTransaction::Yes, *nsGkAtoms::meta, + EditorDOMPoint(primaryHeadElement, 0), + [&aCharacterSet](HTMLEditor&, Element& aMetaElement, + const EditorDOMPoint&) { + MOZ_ASSERT(!aMetaElement.IsInComposedDoc()); + DebugOnly<nsresult> rvIgnored = + aMetaElement.SetAttr(kNameSpaceID_None, nsGkAtoms::httpEquiv, + u"Content-Type"_ns, false); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "Element::SetAttr(nsGkAtoms::httpEquiv, \"Content-Type\", " + "false) failed, but ignored"); + rvIgnored = + aMetaElement.SetAttr(kNameSpaceID_None, nsGkAtoms::content, + u"text/html;charset="_ns + + NS_ConvertASCIItoUTF16(aCharacterSet), + false); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + nsPrintfCString( + "Element::SetAttr(nsGkAtoms::content, " + "\"text/html;charset=%s\", false) failed, but ignored", + nsPromiseFlatCString(aCharacterSet).get()) + .get()); + return NS_OK; + }); + NS_WARNING_ASSERTION(createNewMetaElementResult.isOk(), + "HTMLEditor::CreateAndInsertElement(WithTransaction::" + "Yes, nsGkAtoms::meta) failed, but ignored"); + // Probably, we don't need to update selection in this case since we should + // not put selection into <head> element. + createNewMetaElementResult.inspect().IgnoreCaretPointSuggestion(); + return NS_OK; +} + +bool HTMLEditor::UpdateMetaCharsetWithTransaction( + Document& aDocument, const nsACString& aCharacterSet) { + // get a list of META tags + RefPtr<nsContentList> metaElementList = + aDocument.GetElementsByTagName(u"meta"_ns); + if (NS_WARN_IF(!metaElementList)) { + return false; + } + + for (uint32_t i = 0; i < metaElementList->Length(true); ++i) { + RefPtr<Element> metaElement = metaElementList->Item(i)->AsElement(); + MOZ_ASSERT(metaElement); + + nsAutoString currentValue; + metaElement->GetAttr(kNameSpaceID_None, nsGkAtoms::httpEquiv, currentValue); + + if (!FindInReadable(u"content-type"_ns, currentValue, + nsCaseInsensitiveStringComparator)) { + continue; + } + + metaElement->GetAttr(kNameSpaceID_None, nsGkAtoms::content, currentValue); + + constexpr auto charsetEquals = u"charset="_ns; + nsAString::const_iterator originalStart, start, end; + originalStart = currentValue.BeginReading(start); + currentValue.EndReading(end); + if (!FindInReadable(charsetEquals, start, end, + nsCaseInsensitiveStringComparator)) { + continue; + } + + // set attribute to <original prefix> charset=text/html + nsresult rv = SetAttributeWithTransaction( + *metaElement, *nsGkAtoms::content, + Substring(originalStart, start) + charsetEquals + + NS_ConvertASCIItoUTF16(aCharacterSet)); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::SetAttributeWithTransaction(nsGkAtoms::content) failed"); + return NS_SUCCEEDED(rv); + } + return false; +} + +NS_IMETHODIMP HTMLEditor::NotifySelectionChanged(Document* aDocument, + Selection* aSelection, + int16_t aReason, + int32_t aAmount) { + if (NS_WARN_IF(!aDocument) || NS_WARN_IF(!aSelection)) { + return NS_ERROR_INVALID_ARG; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + + if (mPendingStylesToApplyToNewContent) { + RefPtr<PendingStyles> pendingStyles = mPendingStylesToApplyToNewContent; + pendingStyles->OnSelectionChange(*this, aReason); + + // We used a class which derived from nsISelectionListener to call + // HTMLEditor::RefreshEditingUI(). The lifetime of the class was + // exactly same as mPendingStylesToApplyToNewContent. So, call it only when + // mPendingStylesToApplyToNewContent is not nullptr. + if ((aReason & (nsISelectionListener::MOUSEDOWN_REASON | + nsISelectionListener::KEYPRESS_REASON | + nsISelectionListener::SELECTALL_REASON)) && + aSelection) { + // the selection changed and we need to check if we have to + // hide and/or redisplay resizing handles + // FYI: This is an XPCOM method. So, the caller, Selection, guarantees + // the lifetime of this instance. So, don't need to grab this with + // local variable. + DebugOnly<nsresult> rv = RefreshEditingUI(); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::RefreshEditingUI() failed, but ignored"); + } + } + + if (mComposerCommandsUpdater) { + RefPtr<ComposerCommandsUpdater> updater = mComposerCommandsUpdater; + updater->OnSelectionChange(); + } + + nsresult rv = EditorBase::NotifySelectionChanged(aDocument, aSelection, + aReason, aAmount); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::NotifySelectionChanged() failed"); + return rv; +} + +void HTMLEditor::UpdateRootElement() { + // Use the HTML documents body element as the editor root if we didn't + // get a root element during initialization. + + mRootElement = GetBodyElement(); + if (!mRootElement) { + RefPtr<Document> doc = GetDocument(); + if (doc) { + // If there is no HTML body element, + // we should use the document root element instead. + mRootElement = doc->GetDocumentElement(); + } + // else leave it null, for lack of anything better. + } +} + +nsresult HTMLEditor::FocusedElementOrDocumentBecomesEditable( + Document& aDocument, Element* aElement) { + // If we should've already handled focus event, selection limiter should not + // be set. Therefore, if it's set, we should do nothing here. + if (GetSelectionAncestorLimiter()) { + return NS_OK; + } + // If we should be in the design mode, we want to handle focus event fired + // on the document node. Therefore, we should emulate it here. + if (IsInDesignMode() && (!aElement || aElement->IsInDesignMode())) { + MOZ_ASSERT(&aDocument == GetDocument()); + nsresult rv = OnFocus(aDocument); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "HTMLEditor::OnFocus() failed"); + return rv; + } + + if (NS_WARN_IF(!aElement)) { + return NS_ERROR_INVALID_ARG; + } + + // Otherwise, we should've already handled focus event on the element, + // therefore, we need to emulate it here. + MOZ_ASSERT(nsFocusManager::GetFocusManager()->GetFocusedElement() == + aElement); + nsresult rv = OnFocus(*aElement); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "HTMLEditor::OnFocus() failed"); + + // Note that we don't need to call + // IMEStateManager::MaybeOnEditableStateDisabled here because + // EditorBase::OnFocus must have already been called IMEStateManager::OnFocus + // if succeeded. And perhaps, it's okay that IME is not enabled when + // HTMLEditor fails to start handling since nobody can handle composition + // events anyway... + + return rv; +} + +nsresult HTMLEditor::OnFocus(const nsINode& aOriginalEventTargetNode) { + // Before doing anything, we should check whether the original target is still + // valid focus event target because it may have already lost focus. + if (!CanKeepHandlingFocusEvent(aOriginalEventTargetNode)) { + return NS_OK; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_FAILURE; + } + + return EditorBase::OnFocus(aOriginalEventTargetNode); +} + +nsresult HTMLEditor::FocusedElementOrDocumentBecomesNotEditable( + HTMLEditor* aHTMLEditor, Document& aDocument, Element* aElement) { + nsresult rv = [&]() MOZ_CAN_RUN_SCRIPT { + // If HTMLEditor has not been created yet, we just need to adjust + // IMEStateManager. So, don't return error. + if (!aHTMLEditor) { + return NS_OK; + } + + nsIContent* const limiter = aHTMLEditor->GetSelectionAncestorLimiter(); + // The HTMLEditor has not received `focus` event so that it does not need to + // emulate `blur`. + if (!limiter) { + return NS_OK; + } + + // If we should be in the design mode, we should treat it as blur from + // the document node. + if (aHTMLEditor->IsInDesignMode() && + (!aElement || aElement->IsInDesignMode())) { + MOZ_ASSERT(aHTMLEditor->GetDocument() == &aDocument); + nsresult rv = aHTMLEditor->OnBlur(&aDocument); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "HTMLEditor::OnBlur() failed"); + return rv; + } + // If the HTMLEditor has already received `focus` event for different + // element than aElement, we'll receive `blur` event later so that we need + // to do nothing here. + if (aElement != limiter) { + return NS_OK; + } + + // Otherwise, even though the limiter keeps having focus but becomes not + // editable. From HTMLEditor point of view, this is equivalent to the + // elements gets blurred. Therefore, we should treat it as losing + // focus. + nsresult rv = aHTMLEditor->OnBlur(aElement); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "HTMLEditor::OnBlur() failed"); + return rv; + }(); + + // If the element becomes not editable without focus change, IMEStateManager + // does not have a chance to disable IME. Therefore, (even if we fail to + // handle the emulated blur above,) we should notify IMEStateManager of the + // editing state change. + RefPtr<Element> focusedElement = aElement ? aElement + : aHTMLEditor + ? aHTMLEditor->GetFocusedElement() + : nullptr; + RefPtr<nsPresContext> presContext = + focusedElement ? focusedElement->GetPresContext( + Element::PresContextFor::eForComposedDoc) + : aDocument.GetPresContext(); + if (presContext) { + IMEStateManager::MaybeOnEditableStateDisabled(*presContext, focusedElement); + } + + return rv; +} + +nsresult HTMLEditor::OnBlur(const EventTarget* aEventTarget) { + // check if something else is focused. If another element is focused, then + // we should not change the selection. + nsFocusManager* focusManager = nsFocusManager::GetFocusManager(); + if (MOZ_UNLIKELY(!focusManager)) { + return NS_OK; + } + + // If another element already has focus, we should not maintain the selection + // because we may not have the rights doing it. + if (focusManager->GetFocusedElement()) { + return NS_OK; + } + + // If it's in the designMode, and blur occurs, the target must be the + // document node. If a blur event is fired and the target is an element, it + // must be delayed blur event at initializing the `HTMLEditor`. + if (IsInDesignMode() && Element::FromEventTargetOrNull(aEventTarget)) { + return NS_OK; + } + nsresult rv = FinalizeSelection(); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::FinalizeSelection() failed"); + return rv; +} + +Element* HTMLEditor::FindSelectionRoot(const nsINode& aNode) const { + MOZ_ASSERT(aNode.IsDocument() || aNode.IsContent(), + "aNode must be content or document node"); + + if (NS_WARN_IF(!aNode.IsInComposedDoc())) { + return nullptr; + } + + if (aNode.IsInDesignMode()) { + return GetDocument()->GetRootElement(); + } + + nsIContent* content = const_cast<nsIContent*>(aNode.AsContent()); + if (!content->HasFlag(NODE_IS_EDITABLE)) { + // If the content is in read-write state but is not editable itself, + // return it as the selection root. + if (content->IsElement() && + content->AsElement()->State().HasState(ElementState::READWRITE)) { + return content->AsElement(); + } + return nullptr; + } + + // For non-readonly editors we want to find the root of the editable subtree + // containing aContent. + return content->GetEditingHost(); +} + +bool HTMLEditor::IsInDesignMode() const { + // TODO: If active editing host is in a shadow tree, it means that we should + // behave exactly same as contenteditable mode because shadow tree + // content is not editable even if composed document is in design mode, + // but contenteditable elements in shoadow trees are focusable and + // their content is editable. Changing this affects to drop event + // handler and blur event handler, so please add new tests for them + // when you change here. + Document* document = GetDocument(); + return document && document->IsInDesignMode(); +} + +bool HTMLEditor::EntireDocumentIsEditable() const { + Document* document = GetDocument(); + return document && document->GetDocumentElement() && + (document->GetDocumentElement()->IsEditable() || + (document->GetBody() && document->GetBody()->IsEditable())); +} + +void HTMLEditor::CreateEventListeners() { + // Don't create the handler twice + if (!mEventListener) { + mEventListener = new HTMLEditorEventListener(); + } +} + +nsresult HTMLEditor::InstallEventListeners() { + if (NS_WARN_IF(!IsInitialized()) || NS_WARN_IF(!mEventListener)) { + return NS_ERROR_NOT_INITIALIZED; + } + + // NOTE: HTMLEditor doesn't need to initialize mEventTarget here because + // the target must be document node and it must be referenced as weak pointer. + + HTMLEditorEventListener* listener = + reinterpret_cast<HTMLEditorEventListener*>(mEventListener.get()); + nsresult rv = listener->Connect(this); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditorEventListener::Connect() failed"); + return rv; +} + +void HTMLEditor::RemoveEventListeners() { + if (!IsInitialized()) { + return; + } + + EditorBase::RemoveEventListeners(); +} + +void HTMLEditor::Detach( + const ComposerCommandsUpdater& aComposerCommandsUpdater) { + MOZ_DIAGNOSTIC_ASSERT_IF( + mComposerCommandsUpdater, + &aComposerCommandsUpdater == mComposerCommandsUpdater); + if (mComposerCommandsUpdater == &aComposerCommandsUpdater) { + mComposerCommandsUpdater = nullptr; + if (mTransactionManager) { + mTransactionManager->Detach(*this); + } + } +} + +NS_IMETHODIMP HTMLEditor::BeginningOfDocument() { + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + + nsresult rv = MaybeCollapseSelectionAtFirstEditableNode(false); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::MaybeCollapseSelectionAtFirstEditableNode(false) failed"); + return rv; +} + +NS_IMETHODIMP HTMLEditor::EndOfDocument() { + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + nsresult rv = CollapseSelectionToEndOfLastLeafNodeOfDocument(); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::CollapseSelectionToEndOfLastLeafNodeOfDocument() failed"); + // This is low level API for embedders and chrome script so that we can return + // raw error code here. + return rv; +} + +nsresult HTMLEditor::CollapseSelectionToEndOfLastLeafNodeOfDocument() const { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // We should do nothing with the result of GetRoot() if only a part of the + // document is editable. + if (!EntireDocumentIsEditable()) { + return NS_OK; + } + + RefPtr<Element> bodyOrDocumentElement = GetRoot(); + if (NS_WARN_IF(!bodyOrDocumentElement)) { + return NS_ERROR_NULL_POINTER; + } + + auto pointToPutCaret = [&]() -> EditorRawDOMPoint { + nsCOMPtr<nsIContent> lastLeafContent = HTMLEditUtils::GetLastLeafContent( + *bodyOrDocumentElement, {LeafNodeType::OnlyLeafNode}); + if (!lastLeafContent) { + return EditorRawDOMPoint::AtEndOf(*bodyOrDocumentElement); + } + // TODO: We should put caret into text node if it's visible. + return lastLeafContent->IsText() || + HTMLEditUtils::IsContainerNode(*lastLeafContent) + ? EditorRawDOMPoint::AtEndOf(*lastLeafContent) + : EditorRawDOMPoint(lastLeafContent); + }(); + nsresult rv = CollapseSelectionTo(pointToPutCaret); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; +} + +void HTMLEditor::InitializeSelectionAncestorLimit( + nsIContent& aAncestorLimit) const { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // Hack for initializing selection. + // HTMLEditor::MaybeCollapseSelectionAtFirstEditableNode() will try to + // collapse selection at first editable text node or inline element which + // cannot have text nodes as its children. However, selection has already + // set into the new editing host by user, we should not change it. For + // solving this issue, we should do nothing if selection range is in active + // editing host except it's not collapsed at start of the editing host since + // aSelection.SetAncestorLimiter(aAncestorLimit) will collapse selection + // at start of the new limiter if focus node of aSelection is outside of the + // editing host. However, we need to check here if selection is already + // collapsed at start of the editing host because it's possible JS to do it. + // In such case, we should not modify selection with calling + // MaybeCollapseSelectionAtFirstEditableNode(). + + // Basically, we should try to collapse selection at first editable node + // in HTMLEditor. + bool tryToCollapseSelectionAtFirstEditableNode = true; + if (SelectionRef().RangeCount() == 1 && SelectionRef().IsCollapsed()) { + Element* editingHost = ComputeEditingHost(); + const nsRange* range = SelectionRef().GetRangeAt(0); + if (range->GetStartContainer() == editingHost && !range->StartOffset()) { + // JS or user operation has already collapsed selection at start of + // the editing host. So, we don't need to try to change selection + // in this case. + tryToCollapseSelectionAtFirstEditableNode = false; + } + } + + EditorBase::InitializeSelectionAncestorLimit(aAncestorLimit); + + // XXX Do we need to check if we still need to change selection? E.g., + // we could have already lost focus while we're changing the ancestor + // limiter because it may causes "selectionchange" event. + if (tryToCollapseSelectionAtFirstEditableNode) { + DebugOnly<nsresult> rvIgnored = + MaybeCollapseSelectionAtFirstEditableNode(true); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "HTMLEditor::MaybeCollapseSelectionAtFirstEditableNode(true) failed, " + "but ignored"); + } + + // If the target is a text control element, we won't handle user input + // for the `TextEditor` in it. However, we need to be open for `execCommand`. + // Therefore, we shouldn't set ancestor limit in this case. + // Note that we should do this once setting ancestor limiter for backward + // compatiblity of select events, etc. (Selection should be collapsed into + // the text control element.) + if (aAncestorLimit.HasIndependentSelection()) { + SelectionRef().SetAncestorLimiter(nullptr); + } +} + +nsresult HTMLEditor::MaybeCollapseSelectionAtFirstEditableNode( + bool aIgnoreIfSelectionInEditingHost) const { + MOZ_ASSERT(IsEditActionDataAvailable()); + + RefPtr<Element> editingHost = ComputeEditingHost(LimitInBodyElement::No); + if (NS_WARN_IF(!editingHost)) { + return NS_OK; + } + + // If selection range is already in the editing host and the range is not + // start of the editing host, we shouldn't reset selection. E.g., window + // is activated when the editor had focus before inactivated. + if (aIgnoreIfSelectionInEditingHost && SelectionRef().RangeCount() == 1) { + const nsRange* range = SelectionRef().GetRangeAt(0); + if (!range->Collapsed() || + range->GetStartContainer() != editingHost.get() || + range->StartOffset()) { + return NS_OK; + } + } + + for (nsIContent* leafContent = HTMLEditUtils::GetFirstLeafContent( + *editingHost, + {LeafNodeType::LeafNodeOrNonEditableNode, + LeafNodeType::LeafNodeOrChildBlock}, + editingHost); + leafContent;) { + // If we meet a non-editable node first, we should move caret to start + // of the container block or editing host. + if (!EditorUtils::IsEditableContent(*leafContent, EditorType::HTML)) { + MOZ_ASSERT(leafContent->GetParent()); + MOZ_ASSERT(EditorUtils::IsEditableContent(*leafContent->GetParent(), + EditorType::HTML)); + if (const Element* editableBlockElementOrInlineEditingHost = + HTMLEditUtils::GetAncestorElement( + *leafContent, + HTMLEditUtils:: + ClosestEditableBlockElementOrInlineEditingHost)) { + nsresult rv = CollapseSelectionTo( + EditorDOMPoint(editableBlockElementOrInlineEditingHost, 0)); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; + } + NS_WARNING("Found leaf content did not have editable parent, why?"); + return NS_ERROR_FAILURE; + } + + // When we meet an empty inline element, we should look for a next sibling. + // For example, if current editor is: + // <div contenteditable><span></span><b><br></b></div> + // then, we should put caret at the <br> element. So, let's check if found + // node is an empty inline container element. + if (leafContent->IsElement() && + HTMLEditUtils::IsInlineElement(*leafContent) && + !HTMLEditUtils::IsNeverElementContentsEditableByUser(*leafContent) && + HTMLEditUtils::CanNodeContain(*leafContent, *nsGkAtoms::textTagName)) { + // Chromium collaps selection to start of the editing host when this is + // the last leaf content. So, we don't need special handling here. + leafContent = HTMLEditUtils::GetNextLeafContentOrNextBlockElement( + *leafContent, *editingHost, + {LeafNodeType::LeafNodeOrNonEditableNode, + LeafNodeType::LeafNodeOrChildBlock}, + editingHost); + continue; + } + + if (Text* text = leafContent->GetAsText()) { + // If there is editable and visible text node, move caret at first of + // the visible character. + WSScanResult scanResultInTextNode = + WSRunScanner::ScanNextVisibleNodeOrBlockBoundary( + editingHost, EditorRawDOMPoint(text, 0)); + if ((scanResultInTextNode.InVisibleOrCollapsibleCharacters() || + scanResultInTextNode.ReachedPreformattedLineBreak()) && + scanResultInTextNode.TextPtr() == text) { + nsresult rv = CollapseSelectionTo( + scanResultInTextNode.Point<EditorRawDOMPoint>()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; + } + // If it's an invisible text node, keep scanning next leaf. + leafContent = HTMLEditUtils::GetNextLeafContentOrNextBlockElement( + *leafContent, *editingHost, + {LeafNodeType::LeafNodeOrNonEditableNode, + LeafNodeType::LeafNodeOrChildBlock}, + editingHost); + continue; + } + + // If there is editable <br> or something void element like <img>, <input>, + // <hr> etc, move caret before it. + if (!HTMLEditUtils::CanNodeContain(*leafContent, *nsGkAtoms::textTagName) || + HTMLEditUtils::IsNeverElementContentsEditableByUser(*leafContent)) { + MOZ_ASSERT(leafContent->GetParent()); + if (EditorUtils::IsEditableContent(*leafContent, EditorType::HTML)) { + nsresult rv = CollapseSelectionTo(EditorDOMPoint(leafContent)); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; + } + MOZ_ASSERT_UNREACHABLE( + "How do we reach editable leaf in non-editable element?"); + // But if it's not editable, let's put caret at start of editing host + // for now. + nsresult rv = CollapseSelectionTo(EditorDOMPoint(editingHost, 0)); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; + } + + // If we meet non-empty block element, we need to scan its child too. + if (HTMLEditUtils::IsBlockElement(*leafContent) && + !HTMLEditUtils::IsEmptyNode( + *leafContent, {EmptyCheckOption::TreatSingleBRElementAsVisible}) && + !HTMLEditUtils::IsNeverElementContentsEditableByUser(*leafContent)) { + leafContent = HTMLEditUtils::GetFirstLeafContent( + *leafContent, + {LeafNodeType::LeafNodeOrNonEditableNode, + LeafNodeType::LeafNodeOrChildBlock}, + editingHost); + continue; + } + + // Otherwise, we must meet an empty block element or a data node like + // comment node. Let's ignore it. + leafContent = HTMLEditUtils::GetNextLeafContentOrNextBlockElement( + *leafContent, *editingHost, + {LeafNodeType::LeafNodeOrNonEditableNode, + LeafNodeType::LeafNodeOrChildBlock}, + editingHost); + } + + // If there is no visible/editable node except another block element in + // current editing host, we should move caret to very first of the editing + // host. + // XXX This may not make sense, but Chromium behaves so. Therefore, the + // reason why we do this is just compatibility with Chromium. + nsresult rv = CollapseSelectionTo(EditorDOMPoint(editingHost, 0)); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; +} + +bool HTMLEditor::ArePreservingSelection() const { + return IsEditActionDataAvailable() && SavedSelectionRef().RangeCount(); +} + +void HTMLEditor::PreserveSelectionAcrossActions() { + MOZ_ASSERT(IsEditActionDataAvailable()); + + SavedSelectionRef().SaveSelection(SelectionRef()); + RangeUpdaterRef().RegisterSelectionState(SavedSelectionRef()); +} + +nsresult HTMLEditor::RestorePreservedSelection() { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (!SavedSelectionRef().RangeCount()) { + // XXX Returing error when it does not store is odd because no selection + // ranges is not illegal case in general. + return NS_ERROR_FAILURE; + } + DebugOnly<nsresult> rvIgnored = + SavedSelectionRef().RestoreSelection(SelectionRef()); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "SelectionState::RestoreSelection() failed, but ignored"); + StopPreservingSelection(); + return NS_OK; +} + +void HTMLEditor::StopPreservingSelection() { + MOZ_ASSERT(IsEditActionDataAvailable()); + + RangeUpdaterRef().DropSelectionState(SavedSelectionRef()); + SavedSelectionRef().RemoveAllRanges(); +} + +void HTMLEditor::PreHandleMouseDown(const MouseEvent& aMouseDownEvent) { + if (mPendingStylesToApplyToNewContent) { + // mPendingStylesToApplyToNewContent will be notified of selection change + // even if aMouseDownEvent is not an acceptable event for this editor. + // Therefore, we need to notify it of this event too. + mPendingStylesToApplyToNewContent->PreHandleMouseEvent(aMouseDownEvent); + } +} + +void HTMLEditor::PreHandleMouseUp(const MouseEvent& aMouseUpEvent) { + if (mPendingStylesToApplyToNewContent) { + // mPendingStylesToApplyToNewContent will be notified of selection change + // even if aMouseUpEvent is not an acceptable event for this editor. + // Therefore, we need to notify it of this event too. + mPendingStylesToApplyToNewContent->PreHandleMouseEvent(aMouseUpEvent); + } +} + +void HTMLEditor::PreHandleSelectionChangeCommand(Command aCommand) { + if (mPendingStylesToApplyToNewContent) { + mPendingStylesToApplyToNewContent->PreHandleSelectionChangeCommand( + aCommand); + } +} + +void HTMLEditor::PostHandleSelectionChangeCommand(Command aCommand) { + if (!mPendingStylesToApplyToNewContent) { + return; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (!editActionData.CanHandle()) { + return; + } + mPendingStylesToApplyToNewContent->PostHandleSelectionChangeCommand(*this, + aCommand); +} + +nsresult HTMLEditor::HandleKeyPressEvent(WidgetKeyboardEvent* aKeyboardEvent) { + // NOTE: When you change this method, you should also change: + // * editor/libeditor/tests/test_htmleditor_keyevent_handling.html + if (NS_WARN_IF(!aKeyboardEvent)) { + return NS_ERROR_UNEXPECTED; + } + + if (IsReadonly()) { + HandleKeyPressEventInReadOnlyMode(*aKeyboardEvent); + return NS_OK; + } + + MOZ_ASSERT(aKeyboardEvent->mMessage == eKeyPress, + "HandleKeyPressEvent gets non-keypress event"); + + switch (aKeyboardEvent->mKeyCode) { + case NS_VK_META: + case NS_VK_WIN: + case NS_VK_SHIFT: + case NS_VK_CONTROL: + case NS_VK_ALT: + // FYI: This shouldn't occur since modifier key shouldn't cause eKeyPress + // event. + aKeyboardEvent->PreventDefault(); + return NS_OK; + + case NS_VK_BACK: + case NS_VK_DELETE: { + nsresult rv = EditorBase::HandleKeyPressEvent(aKeyboardEvent); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::HandleKeyPressEvent() failed"); + return rv; + } + case NS_VK_TAB: { + // Basically, "Tab" key be used only for focus navigation. + // FYI: In web apps, this is always true. + if (IsTabbable()) { + return NS_OK; + } + + // If we're in the plaintext mode, and not tabbable editor, let's + // insert a horizontal tabulation. + if (IsInPlaintextMode()) { + if (aKeyboardEvent->IsShift() || aKeyboardEvent->IsControl() || + aKeyboardEvent->IsAlt() || aKeyboardEvent->IsMeta() || + aKeyboardEvent->IsOS()) { + return NS_OK; + } + + // else we insert the tab straight through + aKeyboardEvent->PreventDefault(); + nsresult rv = OnInputText(u"\t"_ns); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::OnInputText(\\t) failed"); + return rv; + } + + // Otherwise, e.g., we're an embedding editor in chrome, we can handle + // "Tab" key as an input. + if (aKeyboardEvent->IsControl() || aKeyboardEvent->IsAlt() || + aKeyboardEvent->IsMeta() || aKeyboardEvent->IsOS()) { + return NS_OK; + } + + RefPtr<Selection> selection = GetSelection(); + if (NS_WARN_IF(!selection) || NS_WARN_IF(!selection->RangeCount())) { + return NS_ERROR_FAILURE; + } + + nsINode* startContainer = selection->GetRangeAt(0)->GetStartContainer(); + MOZ_ASSERT(startContainer); + if (!startContainer->IsContent()) { + break; + } + + const Element* editableBlockElement = + HTMLEditUtils::GetInclusiveAncestorElement( + *startContainer->AsContent(), + HTMLEditUtils::ClosestEditableBlockElement); + if (!editableBlockElement) { + break; + } + + // If selection is in a table element, we need special handling. + if (HTMLEditUtils::IsAnyTableElement(editableBlockElement)) { + Result<EditActionResult, nsresult> result = + HandleTabKeyPressInTable(aKeyboardEvent); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::HandleTabKeyPressInTable() failed"); + return EditorBase::ToGenericNSResult(result.unwrapErr()); + } + if (!result.inspect().Handled()) { + return NS_OK; + } + nsresult rv = ScrollSelectionFocusIntoView(); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::ScrollSelectionFocusIntoView() failed"); + return EditorBase::ToGenericNSResult(rv); + } + + // If selection is in an list item element, treat it as indent or outdent. + if (HTMLEditUtils::IsListItem(editableBlockElement)) { + aKeyboardEvent->PreventDefault(); + if (!aKeyboardEvent->IsShift()) { + nsresult rv = IndentAsAction(); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::IndentAsAction() failed"); + return EditorBase::ToGenericNSResult(rv); + } + nsresult rv = OutdentAsAction(); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::OutdentAsAction() failed"); + return EditorBase::ToGenericNSResult(rv); + } + + // If only "Tab" key is pressed in normal context, just treat it as + // horizontal tab character input. + if (aKeyboardEvent->IsShift()) { + return NS_OK; + } + aKeyboardEvent->PreventDefault(); + nsresult rv = OnInputText(u"\t"_ns); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::OnInputText(\\t) failed"); + return EditorBase::ToGenericNSResult(rv); + } + case NS_VK_RETURN: + if (!aKeyboardEvent->IsInputtingLineBreak()) { + return NS_OK; + } + aKeyboardEvent->PreventDefault(); // consumed + if (aKeyboardEvent->IsShift()) { + // Only inserts a <br> element. + nsresult rv = InsertLineBreakAsAction(); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::InsertLineBreakAsAction() failed"); + return EditorBase::ToGenericNSResult(rv); + } + // uses rules to figure out what to insert + nsresult rv = InsertParagraphSeparatorAsAction(); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::InsertParagraphSeparatorAsAction() failed"); + return EditorBase::ToGenericNSResult(rv); + } + + if (!aKeyboardEvent->IsInputtingText()) { + // we don't PreventDefault() here or keybindings like control-x won't work + return NS_OK; + } + aKeyboardEvent->PreventDefault(); + nsAutoString str(aKeyboardEvent->mCharCode); + nsresult rv = OnInputText(str); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "EditorBase::OnInputText() failed"); + return rv; +} + +NS_IMETHODIMP HTMLEditor::NodeIsBlock(nsINode* aNode, bool* aIsBlock) { + *aIsBlock = aNode && aNode->IsContent() && + HTMLEditUtils::IsBlockElement(*aNode->AsContent()); + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::UpdateBaseURL() { + RefPtr<Document> document = GetDocument(); + if (NS_WARN_IF(!document)) { + return NS_ERROR_FAILURE; + } + + // Look for an HTML <base> tag + RefPtr<nsContentList> baseElementList = + document->GetElementsByTagName(u"base"_ns); + + // If no base tag, then set baseURL to the document's URL. This is very + // important, else relative URLs for links and images are wrong + if (!baseElementList || !baseElementList->Item(0)) { + document->SetBaseURI(document->GetDocumentURI()); + } + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::InsertLineBreak() { + // XPCOM method's InsertLineBreak() should insert paragraph separator in + // HTMLEditor. + AutoEditActionDataSetter editActionData( + *this, EditAction::eInsertParagraphSeparator); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + const RefPtr<Element> editingHost = ComputeEditingHost(); + if (!editingHost) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + Result<EditActionResult, nsresult> result = + InsertParagraphSeparatorAsSubAction(*editingHost); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::InsertParagraphSeparatorAsSubAction() failed"); + return EditorBase::ToGenericNSResult(result.unwrapErr()); + } + return NS_OK; +} + +nsresult HTMLEditor::InsertLineBreakAsAction(nsIPrincipal* aPrincipal) { + AutoEditActionDataSetter editActionData(*this, EditAction::eInsertLineBreak, + aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + if (IsSelectionRangeContainerNotContent()) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + rv = InsertLineBreakAsSubAction(); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::InsertLineBreakAsSubAction() failed"); + // Don't return NS_SUCCESS_DOM_NO_OPERATION for compatibility of `execCommand` + // result of Chrome. + return NS_FAILED(rv) ? rv : NS_OK; +} + +nsresult HTMLEditor::InsertParagraphSeparatorAsAction( + nsIPrincipal* aPrincipal) { + AutoEditActionDataSetter editActionData( + *this, EditAction::eInsertParagraphSeparator, aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + const RefPtr<Element> editingHost = ComputeEditingHost(); + if (!editingHost) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + Result<EditActionResult, nsresult> result = + InsertParagraphSeparatorAsSubAction(*editingHost); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::InsertParagraphSeparatorAsSubAction() failed"); + return EditorBase::ToGenericNSResult(result.unwrapErr()); + } + return NS_OK; +} + +Result<EditActionResult, nsresult> HTMLEditor::HandleTabKeyPressInTable( + WidgetKeyboardEvent* aKeyboardEvent) { + MOZ_ASSERT(aKeyboardEvent); + + AutoEditActionDataSetter dummyEditActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!dummyEditActionData.CanHandle())) { + // Do nothing if we didn't find a table cell. + return EditActionResult::IgnoredResult(); + } + + // Find enclosing table cell from selection (cell may be selected element) + const RefPtr<Element> cellElement = + GetInclusiveAncestorByTagNameAtSelection(*nsGkAtoms::td); + if (!cellElement) { + NS_WARNING( + "HTMLEditor::GetInclusiveAncestorByTagNameAtSelection(*nsGkAtoms::td) " + "returned nullptr"); + // Do nothing if we didn't find a table cell. + return EditActionResult::IgnoredResult(); + } + + // find enclosing table + RefPtr<Element> table = + HTMLEditUtils::GetClosestAncestorTableElement(*cellElement); + if (!table) { + NS_WARNING("HTMLEditor::GetClosestAncestorTableElement() failed"); + return EditActionResult::IgnoredResult(); + } + + // advance to next cell + // first create an iterator over the table + PostContentIterator postOrderIter; + nsresult rv = postOrderIter.Init(table); + if (NS_FAILED(rv)) { + NS_WARNING("PostContentIterator::Init() failed"); + return Err(rv); + } + // position postOrderIter at block + rv = postOrderIter.PositionAt(cellElement); + if (NS_FAILED(rv)) { + NS_WARNING("PostContentIterator::PositionAt() failed"); + return Err(rv); + } + + do { + if (aKeyboardEvent->IsShift()) { + postOrderIter.Prev(); + } else { + postOrderIter.Next(); + } + + nsCOMPtr<nsINode> node = postOrderIter.GetCurrentNode(); + if (node && HTMLEditUtils::IsTableCell(node) && + HTMLEditUtils::GetClosestAncestorTableElement(*node->AsElement()) == + table) { + aKeyboardEvent->PreventDefault(); + CollapseSelectionToDeepestNonTableFirstChild(node); + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + return EditActionResult::HandledResult(); + } + } while (!postOrderIter.IsDone()); + + if (aKeyboardEvent->IsShift()) { + return EditActionResult::IgnoredResult(); + } + + // If we haven't handled it yet, then we must have run off the end of the + // table. Insert a new row. + // XXX We should investigate whether this behavior is supported by other + // browsers later. + AutoEditActionDataSetter editActionData(*this, + EditAction::eInsertTableRowElement); + rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return Err(rv); + } + rv = InsertTableRowsWithTransaction(*cellElement, 1, + InsertPosition::eAfterSelectedCell); + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::InsertTableRowsWithTransaction(*cellElement, 1, " + "InsertPosition::eAfterSelectedCell) failed"); + return Err(rv); + } + aKeyboardEvent->PreventDefault(); + // Put selection in right place. Use table code to get selection and index + // to new row... + RefPtr<Element> tblElement, cell; + int32_t row; + rv = GetCellContext(getter_AddRefs(tblElement), getter_AddRefs(cell), nullptr, + nullptr, &row, nullptr); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::GetCellContext() failed"); + return Err(rv); + } + if (!tblElement) { + NS_WARNING("HTMLEditor::GetCellContext() didn't return table element"); + return Err(NS_ERROR_FAILURE); + } + // ...so that we can ask for first cell in that row... + cell = GetTableCellElementAt(*tblElement, row, 0); + // ...and then set selection there. (Note that normally you should use + // CollapseSelectionToDeepestNonTableFirstChild(), but we know cell is an + // empty new cell, so this works fine) + if (cell) { + nsresult rv = CollapseSelectionToStartOf(*cell); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionToStartOf() failed"); + return Err(NS_ERROR_EDITOR_DESTROYED); + } + } + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + return EditActionResult::HandledResult(); +} + +void HTMLEditor::CollapseSelectionToDeepestNonTableFirstChild(nsINode* aNode) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + MOZ_ASSERT(aNode); + + nsCOMPtr<nsINode> node = aNode; + + for (nsIContent* child = node->GetFirstChild(); child; + child = child->GetFirstChild()) { + // Stop if we find a table, don't want to go into nested tables + if (HTMLEditUtils::IsTable(child) || + !HTMLEditUtils::IsContainerNode(*child)) { + break; + } + node = child; + } + + DebugOnly<nsresult> rvIgnored = CollapseSelectionToStartOf(*node); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "EditorBase::CollapseSelectionToStartOf() failed, but ignored"); +} + +nsresult HTMLEditor::ReplaceHeadContentsWithSourceWithTransaction( + const nsAString& aSourceToInsert) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // don't do any post processing, rules get confused + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eReplaceHeadWithHTMLSource, nsIEditor::eNone, + ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return ignoredError.StealNSResult(); + } + NS_WARNING_ASSERTION( + !ignoredError.Failed(), + "HTMLEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored"); + + CommitComposition(); + + // Do not use AutoEditSubActionNotifier -- rules code won't let us insert in + // <head>. Use the head node as a parent and delete/insert directly. + // XXX We're using AutoEditSubActionNotifier above... + RefPtr<Document> document = GetDocument(); + if (NS_WARN_IF(!document)) { + return NS_ERROR_NOT_INITIALIZED; + } + + RefPtr<nsContentList> headElementList = + document->GetElementsByTagName(u"head"_ns); + if (NS_WARN_IF(!headElementList)) { + return NS_ERROR_FAILURE; + } + + RefPtr<Element> primaryHeadElement = headElementList->Item(0)->AsElement(); + if (NS_WARN_IF(!primaryHeadElement)) { + return NS_ERROR_FAILURE; + } + + // First, make sure there are no return chars in the source. Bad things + // happen if you insert returns (instead of dom newlines, \n) into an editor + // document. + nsAutoString inputString(aSourceToInsert); + + // Windows linebreaks: Map CRLF to LF: + inputString.ReplaceSubstring(u"\r\n"_ns, u"\n"_ns); + + // Mac linebreaks: Map any remaining CR to LF: + inputString.ReplaceSubstring(u"\r"_ns, u"\n"_ns); + + AutoPlaceholderBatch treatAsOneTransaction( + *this, ScrollSelectionIntoView::Yes, __FUNCTION__); + + // Get the first range in the selection, for context: + RefPtr<const nsRange> range = SelectionRef().GetRangeAt(0); + if (NS_WARN_IF(!range)) { + return NS_ERROR_FAILURE; + } + + ErrorResult error; + RefPtr<DocumentFragment> documentFragment = + range->CreateContextualFragment(inputString, error); + + // XXXX BUG 50965: This is not returning the text between <title>...</title> + // Special code is needed in JS to handle title anyway, so it doesn't matter! + + if (error.Failed()) { + NS_WARNING("nsRange::CreateContextualFragment() failed"); + return error.StealNSResult(); + } + if (NS_WARN_IF(!documentFragment)) { + NS_WARNING( + "nsRange::CreateContextualFragment() didn't create DocumentFragment"); + return NS_ERROR_FAILURE; + } + + // First delete all children in head + while (nsCOMPtr<nsIContent> child = primaryHeadElement->GetFirstChild()) { + nsresult rv = DeleteNodeWithTransaction(*child); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return rv; + } + } + + // Now insert the new nodes + int32_t offsetOfNewNode = 0; + + // Loop over the contents of the fragment and move into the document + while (nsCOMPtr<nsIContent> child = documentFragment->GetFirstChild()) { + Result<CreateContentResult, nsresult> insertChildContentResult = + InsertNodeWithTransaction( + *child, EditorDOMPoint(primaryHeadElement, offsetOfNewNode++)); + if (MOZ_UNLIKELY(insertChildContentResult.isErr())) { + NS_WARNING("EditorBase::InsertNodeWithTransaction() failed"); + return insertChildContentResult.unwrapErr(); + } + // We probably don't need to adjust selection here, although we've done it + // unless AutoTransactionsConserveSelection is created in a caller. + insertChildContentResult.inspect().IgnoreCaretPointSuggestion(); + } + + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::RebuildDocumentFromSource( + const nsAString& aSourceString) { + CommitComposition(); + + AutoEditActionDataSetter editActionData(*this, EditAction::eSetHTML); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + RefPtr<Element> rootElement = GetRoot(); + if (NS_WARN_IF(!rootElement)) { + return NS_ERROR_NULL_POINTER; + } + + // Find where the <body> tag starts. + nsReadingIterator<char16_t> beginbody; + nsReadingIterator<char16_t> endbody; + aSourceString.BeginReading(beginbody); + aSourceString.EndReading(endbody); + bool foundbody = + CaseInsensitiveFindInReadable(u"<body"_ns, beginbody, endbody); + + nsReadingIterator<char16_t> beginhead; + nsReadingIterator<char16_t> endhead; + aSourceString.BeginReading(beginhead); + aSourceString.EndReading(endhead); + bool foundhead = + CaseInsensitiveFindInReadable(u"<head"_ns, beginhead, endhead); + // a valid head appears before the body + if (foundbody && beginhead.get() > beginbody.get()) { + foundhead = false; + } + + nsReadingIterator<char16_t> beginclosehead; + nsReadingIterator<char16_t> endclosehead; + aSourceString.BeginReading(beginclosehead); + aSourceString.EndReading(endclosehead); + + // Find the index after "<head>" + bool foundclosehead = CaseInsensitiveFindInReadable( + u"</head>"_ns, beginclosehead, endclosehead); + // a valid close head appears after a found head + if (foundhead && beginhead.get() > beginclosehead.get()) { + foundclosehead = false; + } + // a valid close head appears before a found body + if (foundbody && beginclosehead.get() > beginbody.get()) { + foundclosehead = false; + } + + // Time to change the document + AutoPlaceholderBatch treatAsOneTransaction( + *this, ScrollSelectionIntoView::Yes, __FUNCTION__); + + nsReadingIterator<char16_t> endtotal; + aSourceString.EndReading(endtotal); + + if (foundhead) { + if (foundclosehead) { + nsresult rv = ReplaceHeadContentsWithSourceWithTransaction( + Substring(beginhead, beginclosehead)); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::ReplaceHeadContentsWithSourceWithTransaction() " + "failed"); + return rv; + } + } else if (foundbody) { + nsresult rv = ReplaceHeadContentsWithSourceWithTransaction( + Substring(beginhead, beginbody)); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::ReplaceHeadContentsWithSourceWithTransaction() " + "failed"); + return rv; + } + } else { + // XXX Without recourse to some parser/content sink/docshell hackery we + // don't really know where the head ends and the body begins so we assume + // that there is no body + nsresult rv = ReplaceHeadContentsWithSourceWithTransaction( + Substring(beginhead, endtotal)); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::ReplaceHeadContentsWithSourceWithTransaction() " + "failed"); + return rv; + } + } + } else { + nsReadingIterator<char16_t> begintotal; + aSourceString.BeginReading(begintotal); + constexpr auto head = u"<head>"_ns; + if (foundclosehead) { + nsresult rv = ReplaceHeadContentsWithSourceWithTransaction( + head + Substring(begintotal, beginclosehead)); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::ReplaceHeadContentsWithSourceWithTransaction() " + "failed"); + return rv; + } + } else if (foundbody) { + nsresult rv = ReplaceHeadContentsWithSourceWithTransaction( + head + Substring(begintotal, beginbody)); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::ReplaceHeadContentsWithSourceWithTransaction() " + "failed"); + return rv; + } + } else { + // XXX Without recourse to some parser/content sink/docshell hackery we + // don't really know where the head ends and the body begins so we assume + // that there is no head + nsresult rv = ReplaceHeadContentsWithSourceWithTransaction(head); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::ReplaceHeadContentsWithSourceWithTransaction() " + "failed"); + return rv; + } + } + } + + rv = SelectAll(); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::SelectAll() failed"); + return rv; + } + + if (!foundbody) { + constexpr auto body = u"<body>"_ns; + // XXX Without recourse to some parser/content sink/docshell hackery we + // don't really know where the head ends and the body begins + if (foundclosehead) { + // assume body starts after the head ends + nsresult rv = LoadHTML(body + Substring(endclosehead, endtotal)); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::LoadHTML() failed"); + return rv; + } + } else if (foundhead) { + // assume there is no body + nsresult rv = LoadHTML(body); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::LoadHTML() failed"); + return rv; + } + } else { + // assume there is no head, the entire source is body + nsresult rv = LoadHTML(body + aSourceString); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::LoadHTML() failed"); + return rv; + } + } + + RefPtr<Element> divElement = CreateElementWithDefaults(*nsGkAtoms::div); + if (!divElement) { + NS_WARNING( + "HTMLEditor::CreateElementWithDefaults(nsGkAtoms::div) failed"); + return NS_ERROR_FAILURE; + } + CloneAttributesWithTransaction(*rootElement, *divElement); + + nsresult rv = MaybeCollapseSelectionAtFirstEditableNode(false); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::MaybeCollapseSelectionAtFirstEditableNode(false) failed"); + return rv; + } + + rv = LoadHTML(Substring(beginbody, endtotal)); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::LoadHTML() failed"); + return rv; + } + + // Now we must copy attributes user might have edited on the <body> tag + // because InsertHTML (actually, CreateContextualFragment()) will never + // return a body node in the DOM fragment + + // We already know where "<body" begins + nsReadingIterator<char16_t> beginclosebody = beginbody; + nsReadingIterator<char16_t> endclosebody; + aSourceString.EndReading(endclosebody); + if (!FindInReadable(u">"_ns, beginclosebody, endclosebody)) { + NS_WARNING("'>' was not found"); + return NS_ERROR_FAILURE; + } + + // Truncate at the end of the body tag. Kludge of the year: fool the parser + // by replacing "body" with "div" so we get a node + nsAutoString bodyTag; + bodyTag.AssignLiteral("<div "); + bodyTag.Append(Substring(endbody, endclosebody)); + + RefPtr<const nsRange> range = SelectionRef().GetRangeAt(0); + if (NS_WARN_IF(!range)) { + return NS_ERROR_FAILURE; + } + + ErrorResult error; + RefPtr<DocumentFragment> documentFragment = + range->CreateContextualFragment(bodyTag, error); + if (error.Failed()) { + NS_WARNING("nsRange::CreateContextualFragment() failed"); + return error.StealNSResult(); + } + if (!documentFragment) { + NS_WARNING( + "nsRange::CreateContextualFragment() didn't create DocumentFagement"); + return NS_ERROR_FAILURE; + } + + nsCOMPtr<nsIContent> firstChild = documentFragment->GetFirstChild(); + if (!firstChild || !firstChild->IsElement()) { + NS_WARNING("First child of DocumentFragment was not an Element node"); + return NS_ERROR_FAILURE; + } + + // Copy all attributes from the div child to current body element + CloneAttributesWithTransaction(*rootElement, + MOZ_KnownLive(*firstChild->AsElement())); + + // place selection at first editable content + rv = MaybeCollapseSelectionAtFirstEditableNode(false); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::MaybeCollapseSelectionAtFirstEditableNode(false) failed"); + return rv; +} + +NS_IMETHODIMP HTMLEditor::InsertElementAtSelection(Element* aElement, + bool aDeleteSelection) { + nsresult rv = InsertElementAtSelectionAsAction(aElement, aDeleteSelection); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::InsertElementAtSelectionAsAction() failed"); + return rv; +} + +nsresult HTMLEditor::InsertElementAtSelectionAsAction( + Element* aElement, bool aDeleteSelection, nsIPrincipal* aPrincipal) { + if (NS_WARN_IF(!aElement)) { + return NS_ERROR_INVALID_ARG; + } + + if (IsReadonly()) { + return NS_OK; + } + + AutoEditActionDataSetter editActionData( + *this, HTMLEditUtils::GetEditActionForInsert(*aElement), aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + DebugOnly<nsresult> rvIgnored = CommitComposition(); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "EditorBase::CommitComposition() failed, but ignored"); + + { + Result<EditActionResult, nsresult> result = CanHandleHTMLEditSubAction(); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::CanHandleHTMLEditSubAction() failed"); + return EditorBase::ToGenericNSResult(result.unwrapErr()); + } + if (result.inspect().Canceled()) { + return NS_OK; + } + } + + UndefineCaretBidiLevel(); + + AutoPlaceholderBatch treatAsOneTransaction( + *this, ScrollSelectionIntoView::Yes, __FUNCTION__); + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eInsertElement, nsIEditor::eNext, ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return ignoredError.StealNSResult(); + } + NS_WARNING_ASSERTION( + !ignoredError.Failed(), + "HTMLEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored"); + + rv = EnsureNoPaddingBRElementForEmptyEditor(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::EnsureNoPaddingBRElementForEmptyEditor() " + "failed, but ignored"); + + if (NS_SUCCEEDED(rv) && SelectionRef().IsCollapsed()) { + nsresult rv = EnsureCaretNotAfterInvisibleBRElement(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::EnsureCaretNotAfterInvisibleBRElement() " + "failed, but ignored"); + if (NS_SUCCEEDED(rv)) { + nsresult rv = PrepareInlineStylesForCaret(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::PrepareInlineStylesForCaret() failed, but ignored"); + } + } + + if (aDeleteSelection) { + if (!HTMLEditUtils::IsBlockElement(*aElement)) { + // E.g., inserting an image. In this case we don't need to delete any + // inline wrappers before we do the insertion. Otherwise we let + // DeleteSelectionAndPrepareToCreateNode do the deletion for us, which + // calls DeleteSelection with aStripWrappers = eStrip. + nsresult rv = DeleteSelectionAsSubAction(eNone, eNoStrip); + if (NS_FAILED(rv)) { + NS_WARNING( + "EditorBase::DeleteSelectionAsSubAction(eNone, eNoStrip) failed"); + return EditorBase::ToGenericNSResult(rv); + } + } + + nsresult rv = DeleteSelectionAndPrepareToCreateNode(); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::DeleteSelectionAndPrepareToCreateNode() failed"); + return rv; + } + } + // If deleting, selection will be collapsed. + // so if not, we collapse it + else { + // Named Anchor is a special case, + // We collapse to insert element BEFORE the selection + // For all other tags, we insert AFTER the selection + if (HTMLEditUtils::IsNamedAnchor(aElement)) { + IgnoredErrorResult ignoredError; + SelectionRef().CollapseToStart(ignoredError); + if (NS_WARN_IF(Destroyed())) { + return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION(!ignoredError.Failed(), + "Selection::CollapseToStart() failed, but ignored"); + } else { + IgnoredErrorResult ignoredError; + SelectionRef().CollapseToEnd(ignoredError); + if (NS_WARN_IF(Destroyed())) { + return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION(!ignoredError.Failed(), + "Selection::CollapseToEnd() failed, but ignored"); + } + } + + if (!SelectionRef().GetAnchorNode()) { + return NS_OK; + } + + Element* editingHost = ComputeEditingHost(LimitInBodyElement::No); + if (NS_WARN_IF(!editingHost)) { + return EditorBase::ToGenericNSResult(NS_ERROR_FAILURE); + } + + EditorRawDOMPoint atAnchor(SelectionRef().AnchorRef()); + // Adjust position based on the node we are going to insert. + EditorDOMPoint pointToInsert = + HTMLEditUtils::GetBetterInsertionPointFor<EditorDOMPoint>( + *aElement, atAnchor, *editingHost); + if (!pointToInsert.IsSet()) { + NS_WARNING("HTMLEditUtils::GetBetterInsertionPointFor() failed"); + return NS_ERROR_FAILURE; + } + + { + Result<CreateElementResult, nsresult> insertElementResult = + InsertNodeIntoProperAncestorWithTransaction<Element>( + *aElement, pointToInsert, + SplitAtEdges::eAllowToCreateEmptyContainer); + if (MOZ_UNLIKELY(insertElementResult.isErr())) { + NS_WARNING( + "HTMLEditor::InsertNodeIntoProperAncestorWithTransaction(" + "SplitAtEdges::eAllowToCreateEmptyContainer) failed"); + return EditorBase::ToGenericNSResult(insertElementResult.unwrapErr()); + } + insertElementResult.inspect().IgnoreCaretPointSuggestion(); + } + // Set caret after element, but check for special case + // of inserting table-related elements: set in first cell instead + if (!SetCaretInTableCell(aElement)) { + if (NS_WARN_IF(Destroyed())) { + return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_DESTROYED); + } + nsresult rv = CollapseSelectionTo(EditorRawDOMPoint::After(*aElement)); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::CollapseSelectionTo() failed"); + return EditorBase::ToGenericNSResult(rv); + } + } + + // check for inserting a whole table at the end of a block. If so insert + // a br after it. + if (!HTMLEditUtils::IsTable(aElement) || + !HTMLEditUtils::IsLastChild(*aElement, + {WalkTreeOption::IgnoreNonEditableNode})) { + return NS_OK; + } + + const auto afterElement = EditorDOMPoint::After(*aElement); + // Collapse selection to the new `<br>` element node after creating it. + Result<CreateElementResult, nsresult> insertBRElementResult = + InsertBRElement(WithTransaction::Yes, afterElement, ePrevious); + if (MOZ_UNLIKELY(insertBRElementResult.isErr())) { + NS_WARNING( + "HTMLEditor::InsertBRElement(WithTransaction::Yes, ePrevious) failed"); + return EditorBase::ToGenericNSResult(insertBRElementResult.unwrapErr()); + } + rv = insertBRElementResult.inspect().SuggestCaretPointTo(*this, {}); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + MOZ_ASSERT(insertBRElementResult.inspect().GetNewNode()); + return EditorBase::ToGenericNSResult(rv); +} + +template <typename NodeType> +Result<CreateNodeResultBase<NodeType>, nsresult> +HTMLEditor::InsertNodeIntoProperAncestorWithTransaction( + NodeType& aContentToInsert, const EditorDOMPoint& aPointToInsert, + SplitAtEdges aSplitAtEdges) { + MOZ_ASSERT(aPointToInsert.IsSetAndValidInComposedDoc()); + if (NS_WARN_IF(!aPointToInsert.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + MOZ_ASSERT(aPointToInsert.IsSetAndValid()); + + if (aContentToInsert.NodeType() == nsINode::DOCUMENT_TYPE_NODE || + aContentToInsert.NodeType() == nsINode::PROCESSING_INSTRUCTION_NODE) { + return CreateNodeResultBase<NodeType>::NotHandled(); + } + + // Search up the parent chain to find a suitable container. + EditorDOMPoint pointToInsert(aPointToInsert); + MOZ_ASSERT(pointToInsert.IsSet()); + while (!HTMLEditUtils::CanNodeContain(*pointToInsert.GetContainer(), + aContentToInsert)) { + // If the current parent is a root (body or table element) + // then go no further - we can't insert. + if (MOZ_UNLIKELY( + pointToInsert.IsContainerHTMLElement(nsGkAtoms::body) || + HTMLEditUtils::IsAnyTableElement(pointToInsert.GetContainer()))) { + NS_WARNING( + "There was no proper container element to insert the content node in " + "the document"); + return Err(NS_ERROR_FAILURE); + } + + // Get the next point. + pointToInsert = pointToInsert.ParentPoint(); + + if (MOZ_UNLIKELY( + !pointToInsert.IsInContentNode() || + !EditorUtils::IsEditableContent( + *pointToInsert.ContainerAs<nsIContent>(), EditorType::HTML))) { + NS_WARNING( + "There was no proper container element to insert the content node in " + "the editing host"); + return Err(NS_ERROR_FAILURE); + } + } + + if (pointToInsert != aPointToInsert) { + // We need to split some levels above the original selection parent. + MOZ_ASSERT(pointToInsert.GetChild()); + Result<SplitNodeResult, nsresult> splitNodeResult = + SplitNodeDeepWithTransaction(MOZ_KnownLive(*pointToInsert.GetChild()), + aPointToInsert, aSplitAtEdges); + if (MOZ_UNLIKELY(splitNodeResult.isErr())) { + NS_WARNING("HTMLEditor::SplitNodeDeepWithTransaction() failed"); + return splitNodeResult.propagateErr(); + } + pointToInsert = + splitNodeResult.inspect().template AtSplitPoint<EditorDOMPoint>(); + MOZ_ASSERT(pointToInsert.IsSetAndValidInComposedDoc()); + // Caret should be set by the caller of this method so that we don't + // need to handle it here. + splitNodeResult.inspect().IgnoreCaretPointSuggestion(); + } + + // Now we can insert the new node. + Result<CreateNodeResultBase<NodeType>, nsresult> insertContentNodeResult = + InsertNodeWithTransaction<NodeType>(aContentToInsert, pointToInsert); + if (MOZ_LIKELY(insertContentNodeResult.isOk()) && + MOZ_UNLIKELY(NS_WARN_IF(!aContentToInsert.GetParentNode()) || + NS_WARN_IF(aContentToInsert.GetParentNode() != + pointToInsert.GetContainer()))) { + NS_WARNING( + "EditorBase::InsertNodeWithTransaction() succeeded, but the inserted " + "node was moved or removed by the web app"); + insertContentNodeResult.inspect().IgnoreCaretPointSuggestion(); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + NS_WARNING_ASSERTION(insertContentNodeResult.isOk(), + "EditorBase::InsertNodeWithTransaction() failed"); + return insertContentNodeResult; +} + +NS_IMETHODIMP HTMLEditor::SelectElement(Element* aElement) { + if (NS_WARN_IF(!aElement)) { + return NS_ERROR_INVALID_ARG; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + + nsresult rv = SelectContentInternal(MOZ_KnownLive(*aElement)); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::SelectContentInternal() failed"); + return rv; +} + +nsresult HTMLEditor::SelectContentInternal(nsIContent& aContentToSelect) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // Must be sure that element is contained in the editing host + const RefPtr<Element> editingHost = ComputeEditingHost(); + if (NS_WARN_IF(!editingHost) || + NS_WARN_IF(!aContentToSelect.IsInclusiveDescendantOf(editingHost))) { + return NS_ERROR_FAILURE; + } + + EditorRawDOMPoint newSelectionStart(&aContentToSelect); + if (NS_WARN_IF(!newSelectionStart.IsSet())) { + return NS_ERROR_FAILURE; + } + EditorRawDOMPoint newSelectionEnd(EditorRawDOMPoint::After(aContentToSelect)); + MOZ_ASSERT(newSelectionEnd.IsSet()); + ErrorResult error; + SelectionRef().SetStartAndEndInLimiter(newSelectionStart, newSelectionEnd, + error); + NS_WARNING_ASSERTION(!error.Failed(), + "Selection::SetStartAndEndInLimiter() failed"); + return error.StealNSResult(); +} + +nsresult HTMLEditor::AppendContentToSelectionAsRange(nsIContent& aContent) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + EditorRawDOMPoint atContent(&aContent); + if (NS_WARN_IF(!atContent.IsSet())) { + return NS_ERROR_FAILURE; + } + + RefPtr<nsRange> range = nsRange::Create( + atContent.ToRawRangeBoundary(), + atContent.NextPoint().ToRawRangeBoundary(), IgnoreErrors()); + if (NS_WARN_IF(!range)) { + NS_WARNING("nsRange::Create() failed"); + return NS_ERROR_FAILURE; + } + + ErrorResult error; + SelectionRef().AddRangeAndSelectFramesAndNotifyListeners(*range, error); + if (NS_WARN_IF(Destroyed())) { + if (error.Failed()) { + error.SuppressException(); + } + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION(!error.Failed(), "Failed to add range to Selection"); + return error.StealNSResult(); +} + +nsresult HTMLEditor::ClearSelection() { + MOZ_ASSERT(IsEditActionDataAvailable()); + + ErrorResult error; + SelectionRef().RemoveAllRanges(error); + if (NS_WARN_IF(Destroyed())) { + if (error.Failed()) { + error.SuppressException(); + } + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION(!error.Failed(), "Selection::RemoveAllRanges() failed"); + return error.StealNSResult(); +} + +nsresult HTMLEditor::SetParagraphFormatAsAction( + const nsAString& aParagraphFormat, nsIPrincipal* aPrincipal) { + AutoEditActionDataSetter editActionData( + *this, EditAction::eInsertBlockElement, aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + // TODO: Computing the editing host here makes the `execCommand` in + // docshell/base/crashtests/file_432114-2.xhtml cannot run + // `DOMNodeRemoved` event listener with deleting the bogus <br> element. + // So that it should be rewritten with different mutation event listener + // since we'd like to stop using it. + + nsAutoString lowerCaseTagName(aParagraphFormat); + ToLowerCase(lowerCaseTagName); + RefPtr<nsAtom> tagName = NS_Atomize(lowerCaseTagName); + MOZ_ASSERT(tagName); + if (tagName == nsGkAtoms::dd || tagName == nsGkAtoms::dt) { + // MOZ_KnownLive(tagName->AsStatic()) because nsStaticAtom instances live + // while the process is running. + Result<EditActionResult, nsresult> result = + MakeOrChangeListAndListItemAsSubAction( + MOZ_KnownLive(*tagName->AsStatic()), u""_ns, + SelectAllOfCurrentList::No); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING( + "HTMLEditor::MakeOrChangeListAndListItemAsSubAction(" + "SelectAllOfCurrentList::No) failed"); + return EditorBase::ToGenericNSResult(result.unwrapErr()); + } + return NS_OK; + } + + rv = FormatBlockContainerAsSubAction(*tagName); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::FormatBlockContainerAsSubAction() failed"); + return EditorBase::ToGenericNSResult(rv); +} + +NS_IMETHODIMP HTMLEditor::GetParagraphState(bool* aMixed, + nsAString& aFirstParagraphState) { + if (NS_WARN_IF(!aMixed)) { + return NS_ERROR_INVALID_ARG; + } + if (!mInitSucceeded) { + return NS_ERROR_NOT_INITIALIZED; + } + + ErrorResult error; + ParagraphStateAtSelection paragraphState(*this, error); + if (error.Failed()) { + NS_WARNING("ParagraphStateAtSelection failed"); + return error.StealNSResult(); + } + + *aMixed = paragraphState.IsMixed(); + if (NS_WARN_IF(!paragraphState.GetFirstParagraphStateAtSelection())) { + // XXX Odd result, but keep this behavior for now... + aFirstParagraphState.AssignASCII("x"); + } else { + paragraphState.GetFirstParagraphStateAtSelection()->ToString( + aFirstParagraphState); + } + return NS_OK; +} + +nsresult HTMLEditor::GetBackgroundColorState(bool* aMixed, + nsAString& aOutColor) { + if (NS_WARN_IF(!aMixed)) { + return NS_ERROR_INVALID_ARG; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + + if (IsCSSEnabled()) { + // if we are in CSS mode, we have to check if the containing block defines + // a background color + nsresult rv = GetCSSBackgroundColorState(aMixed, aOutColor, true); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::GetCSSBackgroundColorState() failed"); + return EditorBase::ToGenericNSResult(rv); + } + // in HTML mode, we look only at page's background + nsresult rv = GetHTMLBackgroundColorState(aMixed, aOutColor); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::GetCSSBackgroundColorState() failed"); + return EditorBase::ToGenericNSResult(rv); +} + +NS_IMETHODIMP HTMLEditor::GetHighlightColorState(bool* aMixed, + nsAString& aOutColor) { + if (NS_WARN_IF(!aMixed)) { + return NS_ERROR_INVALID_ARG; + } + + *aMixed = false; + aOutColor.AssignLiteral("transparent"); + if (!IsCSSEnabled()) { + return NS_OK; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + + // in CSS mode, text background can be added by the Text Highlight button + // we need to query the background of the selection without looking for + // the block container of the ranges in the selection + nsresult rv = GetCSSBackgroundColorState(aMixed, aOutColor, false); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::GetCSSBackgroundColorState() failed"); + return rv; +} + +nsresult HTMLEditor::GetCSSBackgroundColorState(bool* aMixed, + nsAString& aOutColor, + bool aBlockLevel) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (NS_WARN_IF(!aMixed)) { + return NS_ERROR_INVALID_ARG; + } + + *aMixed = false; + // the default background color is transparent + aOutColor.AssignLiteral("transparent"); + + RefPtr<const nsRange> firstRange = SelectionRef().GetRangeAt(0); + if (NS_WARN_IF(!firstRange)) { + return NS_ERROR_FAILURE; + } + + nsCOMPtr<nsINode> startContainer = firstRange->GetStartContainer(); + if (NS_WARN_IF(!startContainer) || NS_WARN_IF(!startContainer->IsContent())) { + return NS_ERROR_FAILURE; + } + + // is the selection collapsed? + nsIContent* contentToExamine; + if (SelectionRef().IsCollapsed() || startContainer->IsText()) { + if (NS_WARN_IF(!startContainer->IsContent())) { + return NS_ERROR_FAILURE; + } + // we want to look at the startContainer and ancestors + contentToExamine = startContainer->AsContent(); + } else { + // otherwise we want to look at the first editable node after + // {startContainer,offset} and its ancestors for divs with alignment on them + contentToExamine = firstRange->GetChildAtStartOffset(); + // GetNextNode(startContainer, offset, true, address_of(contentToExamine)); + } + + if (NS_WARN_IF(!contentToExamine)) { + return NS_ERROR_FAILURE; + } + + if (aBlockLevel) { + // we are querying the block background (and not the text background), let's + // climb to the block container. Note that background color of ancestor + // of editing host may be what the caller wants to know. Therefore, we + // should ignore the editing host boundaries. + Element* const closestBlockElement = + HTMLEditUtils::GetInclusiveAncestorElement( + *contentToExamine, HTMLEditUtils::ClosestBlockElement); + if (NS_WARN_IF(!closestBlockElement)) { + return NS_OK; + } + + for (RefPtr<Element> blockElement = closestBlockElement; blockElement;) { + RefPtr<Element> nextBlockElement = HTMLEditUtils::GetAncestorElement( + *blockElement, HTMLEditUtils::ClosestBlockElement); + DebugOnly<nsresult> rvIgnored = CSSEditUtils::GetComputedProperty( + *blockElement, *nsGkAtoms::backgroundColor, aOutColor); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + if (MayHaveMutationEventListeners() && + NS_WARN_IF(nextBlockElement != + HTMLEditUtils::GetAncestorElement( + *blockElement, HTMLEditUtils::ClosestBlockElement))) { + return NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE; + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "CSSEditUtils::GetComputedProperty(nsGkAtoms::" + "backgroundColor) failed, but ignored"); + // look at parent if the queried color is transparent and if the node to + // examine is not the root of the document + if (!aOutColor.EqualsLiteral("transparent") && + !aOutColor.EqualsLiteral("rgba(0, 0, 0, 0)")) { + break; + } + blockElement = std::move(nextBlockElement); + } + + if (aOutColor.EqualsLiteral("transparent") || + aOutColor.EqualsLiteral("rgba(0, 0, 0, 0)")) { + // we have hit the root of the document and the color is still transparent + // ! Grumble... Let's look at the default background color because that's + // the color we are looking for + CSSEditUtils::GetDefaultBackgroundColor(aOutColor); + } + } else { + // no, we are querying the text background for the Text Highlight button + if (contentToExamine->IsText()) { + // if the node of interest is a text node, let's climb a level + contentToExamine = contentToExamine->GetParent(); + } + // Return default value due to no parent node + if (!contentToExamine) { + return NS_OK; + } + + for (RefPtr<Element> element = + contentToExamine->GetAsElementOrParentElement(); + element; element = element->GetParentElement()) { + // is the node to examine a block ? + if (HTMLEditUtils::IsBlockElement(*element)) { + // yes it is a block; in that case, the text background color is + // transparent + aOutColor.AssignLiteral("transparent"); + break; + } + + // no, it's not; let's retrieve the computed style of background-color + // for the node to examine + nsCOMPtr<nsINode> parentNode = element->GetParentNode(); + DebugOnly<nsresult> rvIgnored = CSSEditUtils::GetComputedProperty( + *element, *nsGkAtoms::backgroundColor, aOutColor); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + if (NS_WARN_IF(parentNode != element->GetParentNode())) { + return NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE; + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "CSSEditUtils::GetComputedProperty(nsGkAtoms::" + "backgroundColor) failed, but ignored"); + if (!aOutColor.EqualsLiteral("transparent")) { + break; + } + } + } + return NS_OK; +} + +nsresult HTMLEditor::GetHTMLBackgroundColorState(bool* aMixed, + nsAString& aOutColor) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // TODO: We don't handle "mixed" correctly! + if (NS_WARN_IF(!aMixed)) { + return NS_ERROR_INVALID_ARG; + } + + *aMixed = false; + aOutColor.Truncate(); + + Result<RefPtr<Element>, nsresult> cellOrRowOrTableElementOrError = + GetSelectedOrParentTableElement(); + if (cellOrRowOrTableElementOrError.isErr()) { + NS_WARNING("HTMLEditor::GetSelectedOrParentTableElement() returned error"); + return cellOrRowOrTableElementOrError.unwrapErr(); + } + + for (RefPtr<Element> element = cellOrRowOrTableElementOrError.unwrap(); + element; element = element->GetParentElement()) { + // We are in a cell or selected table + element->GetAttr(kNameSpaceID_None, nsGkAtoms::bgcolor, aOutColor); + + // Done if we have a color explicitly set + if (!aOutColor.IsEmpty()) { + return NS_OK; + } + + // Once we hit the body, we're done + if (element->IsHTMLElement(nsGkAtoms::body)) { + return NS_OK; + } + + // No color is set, but we need to report visible color inherited + // from nested cells/tables, so search up parent chain so that + // let's keep checking the ancestors. + } + + // If no table or cell found, get page body + Element* rootElement = GetRoot(); + if (NS_WARN_IF(!rootElement)) { + return NS_ERROR_FAILURE; + } + + rootElement->GetAttr(kNameSpaceID_None, nsGkAtoms::bgcolor, aOutColor); + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::GetListState(bool* aMixed, bool* aOL, bool* aUL, + bool* aDL) { + if (NS_WARN_IF(!aMixed) || NS_WARN_IF(!aOL) || NS_WARN_IF(!aUL) || + NS_WARN_IF(!aDL)) { + return NS_ERROR_INVALID_ARG; + } + if (!mInitSucceeded) { + return NS_ERROR_NOT_INITIALIZED; + } + + ErrorResult error; + ListElementSelectionState state(*this, error); + if (error.Failed()) { + NS_WARNING("ListElementSelectionState failed"); + return error.StealNSResult(); + } + + *aMixed = state.IsNotOneTypeListElementSelected(); + *aOL = state.IsOLElementSelected(); + *aUL = state.IsULElementSelected(); + *aDL = state.IsDLElementSelected(); + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::GetListItemState(bool* aMixed, bool* aLI, bool* aDT, + bool* aDD) { + if (NS_WARN_IF(!aMixed) || NS_WARN_IF(!aLI) || NS_WARN_IF(!aDT) || + NS_WARN_IF(!aDD)) { + return NS_ERROR_INVALID_ARG; + } + if (!mInitSucceeded) { + return NS_ERROR_NOT_INITIALIZED; + } + + ErrorResult error; + ListItemElementSelectionState state(*this, error); + if (error.Failed()) { + NS_WARNING("ListItemElementSelectionState failed"); + return error.StealNSResult(); + } + + // XXX Why do we ignore `<li>` element selected state? + *aMixed = state.IsNotOneTypeDefinitionListItemElementSelected(); + *aLI = state.IsLIElementSelected(); + *aDT = state.IsDTElementSelected(); + *aDD = state.IsDDElementSelected(); + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::GetAlignment(bool* aMixed, + nsIHTMLEditor::EAlignment* aAlign) { + if (NS_WARN_IF(!aMixed) || NS_WARN_IF(!aAlign)) { + return NS_ERROR_INVALID_ARG; + } + if (!mInitSucceeded) { + return NS_ERROR_NOT_INITIALIZED; + } + + ErrorResult error; + AlignStateAtSelection state(*this, error); + if (error.Failed()) { + NS_WARNING("AlignStateAtSelection failed"); + return error.StealNSResult(); + } + + *aMixed = false; + *aAlign = state.AlignmentAtSelectionStart(); + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::MakeOrChangeList(const nsAString& aListType, + bool aEntireList, + const nsAString& aBulletType) { + RefPtr<nsAtom> listTagName = NS_Atomize(aListType); + if (NS_WARN_IF(!listTagName) || NS_WARN_IF(!listTagName->IsStatic())) { + return NS_ERROR_INVALID_ARG; + } + // MOZ_KnownLive(listTagName->AsStatic()) because nsStaticAtom instances live + // while the process is running. + nsresult rv = MakeOrChangeListAsAction( + MOZ_KnownLive(*listTagName->AsStatic()), aBulletType, + aEntireList ? SelectAllOfCurrentList::Yes : SelectAllOfCurrentList::No); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::MakeOrChangeListAsAction() failed"); + return rv; +} + +nsresult HTMLEditor::MakeOrChangeListAsAction( + const nsStaticAtom& aListElementTagName, const nsAString& aBulletType, + SelectAllOfCurrentList aSelectAllOfCurrentList, nsIPrincipal* aPrincipal) { + if (NS_WARN_IF(!mInitSucceeded)) { + return NS_ERROR_NOT_INITIALIZED; + } + + AutoEditActionDataSetter editActionData( + *this, HTMLEditUtils::GetEditActionForInsert(aListElementTagName), + aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + Result<EditActionResult, nsresult> result = + MakeOrChangeListAndListItemAsSubAction(aListElementTagName, aBulletType, + aSelectAllOfCurrentList); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::MakeOrChangeListAndListItemAsSubAction() failed"); + return EditorBase::ToGenericNSResult(result.unwrapErr()); + } + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::RemoveList(const nsAString& aListType) { + nsresult rv = RemoveListAsAction(aListType); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::RemoveListAsAction() failed"); + return rv; +} + +nsresult HTMLEditor::RemoveListAsAction(const nsAString& aListType, + nsIPrincipal* aPrincipal) { + if (NS_WARN_IF(!mInitSucceeded)) { + return NS_ERROR_NOT_INITIALIZED; + } + + // Note that we ignore aListType when we actually remove parent list elements. + // However, we need to set InputEvent.inputType to "insertOrderedList" or + // "insertedUnorderedList" when this is called for + // execCommand("insertorderedlist") or execCommand("insertunorderedlist"). + // Otherwise, comm-central UI may call this methods with "dl" or "". + // So, it's okay to use mismatched EditAction here if this is called in + // comm-central. + + RefPtr<nsAtom> listAtom = NS_Atomize(aListType); + if (NS_WARN_IF(!listAtom)) { + return NS_ERROR_INVALID_ARG; + } + AutoEditActionDataSetter editActionData( + *this, HTMLEditUtils::GetEditActionForRemoveList(*listAtom), aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + const RefPtr<Element> editingHost = ComputeEditingHost(); + if (!editingHost) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + rv = RemoveListAtSelectionAsSubAction(*editingHost); + NS_WARNING_ASSERTION(NS_FAILED(rv), + "HTMLEditor::RemoveListAtSelectionAsSubAction() failed"); + return rv; +} + +nsresult HTMLEditor::FormatBlockContainerAsSubAction(nsAtom& aTagName) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (NS_WARN_IF(!mInitSucceeded)) { + return NS_ERROR_NOT_INITIALIZED; + } + + MOZ_ASSERT(&aTagName != nsGkAtoms::dd && &aTagName != nsGkAtoms::dt); + + AutoPlaceholderBatch treatAsOneTransaction( + *this, ScrollSelectionIntoView::Yes, __FUNCTION__); + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eCreateOrRemoveBlock, nsIEditor::eNext, + ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return ignoredError.StealNSResult(); + } + NS_WARNING_ASSERTION( + !ignoredError.Failed(), + "HTMLEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored"); + + { + Result<EditActionResult, nsresult> result = CanHandleHTMLEditSubAction(); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::CanHandleHTMLEditSubAction() failed"); + return result.unwrapErr(); + } + if (result.inspect().Canceled()) { + return NS_OK; + } + } + + if (IsSelectionRangeContainerNotContent()) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + nsresult rv = EnsureNoPaddingBRElementForEmptyEditor(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::EnsureNoPaddingBRElementForEmptyEditor() " + "failed, but ignored"); + + if (NS_SUCCEEDED(rv) && SelectionRef().IsCollapsed()) { + nsresult rv = EnsureCaretNotAfterInvisibleBRElement(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::EnsureCaretNotAfterInvisibleBRElement() " + "failed, but ignored"); + if (NS_SUCCEEDED(rv)) { + nsresult rv = PrepareInlineStylesForCaret(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::PrepareInlineStylesForCaret() failed, but ignored"); + } + } + + // FormatBlockContainerWithTransaction() creates AutoSelectionRestorer. + // Therefore, even if it returns NS_OK, editor might have been destroyed + // at restoring Selection. + const RefPtr<Element> editingHost = ComputeEditingHost(); + if (MOZ_UNLIKELY(!editingHost)) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + AutoRangeArray selectionRanges(SelectionRef()); + Result<RefPtr<Element>, nsresult> suggestBlockElementToPutCaretOrError = + FormatBlockContainerWithTransaction(selectionRanges, aTagName, + *editingHost); + if (suggestBlockElementToPutCaretOrError.isErr()) { + NS_WARNING("HTMLEditor::FormatBlockContainerWithTransaction() failed"); + return suggestBlockElementToPutCaretOrError.unwrapErr(); + } + + if (selectionRanges.HasSavedRanges()) { + selectionRanges.RestoreFromSavedRanges(); + } + + rv = selectionRanges.ApplyTo(SelectionRef()); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + if (NS_FAILED(rv)) { + NS_WARNING("AutoRangeArray::ApplyTo(SelectionRef()) failed, but ignored"); + return rv; + } + + rv = MaybeInsertPaddingBRElementForEmptyLastLineAtSelection(); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::MaybeInsertPaddingBRElementForEmptyLastLineAtSelection() " + "failed"); + + if (!suggestBlockElementToPutCaretOrError.inspect() || + !SelectionRef().IsCollapsed()) { + return rv; + } + const auto firstSelectionStartPoint = + GetFirstSelectionStartPoint<EditorRawDOMPoint>(); + if (MOZ_UNLIKELY(!firstSelectionStartPoint.IsSet())) { + return rv; + } + Result<EditorRawDOMPoint, nsresult> pointInBlockElementOrError = + HTMLEditUtils::ComputePointToPutCaretInElementIfOutside< + EditorRawDOMPoint>(*suggestBlockElementToPutCaretOrError.inspect(), + firstSelectionStartPoint); + if (MOZ_UNLIKELY(pointInBlockElementOrError.isErr())) { + NS_WARNING( + "HTMLEditUtils::ComputePointToPutCaretInElementIfOutside() failed, but " + "ignored"); + return rv; + } + // Note that if the point is unset, it means that firstSelectionStartPoint is + // in the block element. + if (pointInBlockElementOrError.inspect().IsSet()) { + nsresult rvOfCollapseSelection = + CollapseSelectionTo(pointInBlockElementOrError.inspect()); + if (MOZ_UNLIKELY(rvOfCollapseSelection == NS_ERROR_EDITOR_DESTROYED)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvOfCollapseSelection), + "EditorBase::CollapseSelectionTo() failed, but ignored"); + } + return rv; +} + +nsresult HTMLEditor::IndentAsAction(nsIPrincipal* aPrincipal) { + if (NS_WARN_IF(!mInitSucceeded)) { + return NS_ERROR_NOT_INITIALIZED; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eIndent, + aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + const RefPtr<Element> editingHost = ComputeEditingHost(); + if (!editingHost) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + Result<EditActionResult, nsresult> result = IndentAsSubAction(*editingHost); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::IndentAsSubAction() failed"); + return EditorBase::ToGenericNSResult(result.unwrapErr()); + } + return NS_OK; +} + +nsresult HTMLEditor::OutdentAsAction(nsIPrincipal* aPrincipal) { + if (NS_WARN_IF(!mInitSucceeded)) { + return NS_ERROR_NOT_INITIALIZED; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eOutdent, + aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + const RefPtr<Element> editingHost = ComputeEditingHost(); + if (!editingHost) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + Result<EditActionResult, nsresult> result = OutdentAsSubAction(*editingHost); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::OutdentAsSubAction() failed"); + return EditorBase::ToGenericNSResult(result.unwrapErr()); + } + return NS_OK; +} + +// TODO: IMPLEMENT ALIGNMENT! + +nsresult HTMLEditor::AlignAsAction(const nsAString& aAlignType, + nsIPrincipal* aPrincipal) { + AutoEditActionDataSetter editActionData( + *this, HTMLEditUtils::GetEditActionForAlignment(aAlignType), aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + const RefPtr<Element> editingHost = ComputeEditingHost(); + if (!editingHost) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + Result<EditActionResult, nsresult> result = + AlignAsSubAction(aAlignType, *editingHost); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::AlignAsSubAction() failed"); + return EditorBase::ToGenericNSResult(result.unwrapErr()); + } + return NS_OK; +} + +Element* HTMLEditor::GetInclusiveAncestorByTagName(const nsStaticAtom& aTagName, + nsIContent& aContent) const { + MOZ_ASSERT(&aTagName != nsGkAtoms::_empty); + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return nullptr; + } + + return GetInclusiveAncestorByTagNameInternal(aTagName, aContent); +} + +Element* HTMLEditor::GetInclusiveAncestorByTagNameAtSelection( + const nsStaticAtom& aTagName) const { + MOZ_ASSERT(IsEditActionDataAvailable()); + MOZ_ASSERT(&aTagName != nsGkAtoms::_empty); + + // If no node supplied, get it from anchor node of current selection + const EditorRawDOMPoint atAnchor(SelectionRef().AnchorRef()); + if (NS_WARN_IF(!atAnchor.IsInContentNode())) { + return nullptr; + } + + // Try to get the actual selected node + nsIContent* content = nullptr; + if (atAnchor.GetContainer()->HasChildNodes() && + atAnchor.ContainerAs<nsIContent>()) { + content = atAnchor.GetChild(); + } + // Anchor node is probably a text node - just use that + if (!content) { + content = atAnchor.ContainerAs<nsIContent>(); + if (NS_WARN_IF(!content)) { + return nullptr; + } + } + return GetInclusiveAncestorByTagNameInternal(aTagName, *content); +} + +Element* HTMLEditor::GetInclusiveAncestorByTagNameInternal( + const nsStaticAtom& aTagName, const nsIContent& aContent) const { + MOZ_ASSERT(&aTagName != nsGkAtoms::_empty); + + Element* currentElement = aContent.GetAsElementOrParentElement(); + if (NS_WARN_IF(!currentElement)) { + MOZ_ASSERT(!aContent.GetParentNode()); + return nullptr; + } + + bool lookForLink = IsLinkTag(aTagName); + bool lookForNamedAnchor = IsNamedAnchorTag(aTagName); + for (Element* element : currentElement->InclusiveAncestorsOfType<Element>()) { + // Stop searching if parent is a body element. Note: Originally used + // IsRoot() to/ stop at table cells, but that's too messy when you are + // trying to find the parent table. + if (element->IsHTMLElement(nsGkAtoms::body)) { + return nullptr; + } + if (lookForLink) { + // Test if we have a link (an anchor with href set) + if (HTMLEditUtils::IsLink(element)) { + return element; + } + } else if (lookForNamedAnchor) { + // Test if we have a named anchor (an anchor with name set) + if (HTMLEditUtils::IsNamedAnchor(element)) { + return element; + } + } else if (&aTagName == nsGkAtoms::list_) { + // Match "ol", "ul", or "dl" for lists + if (HTMLEditUtils::IsAnyListElement(element)) { + return element; + } + } else if (&aTagName == nsGkAtoms::td) { + // Table cells are another special case: match either "td" or "th" + if (HTMLEditUtils::IsTableCell(element)) { + return element; + } + } else if (&aTagName == element->NodeInfo()->NameAtom()) { + return element; + } + } + return nullptr; +} + +NS_IMETHODIMP HTMLEditor::GetElementOrParentByTagName(const nsAString& aTagName, + nsINode* aNode, + Element** aReturn) { + if (NS_WARN_IF(aTagName.IsEmpty()) || NS_WARN_IF(!aReturn)) { + return NS_ERROR_INVALID_ARG; + } + + nsStaticAtom* tagName = EditorUtils::GetTagNameAtom(aTagName); + if (NS_WARN_IF(!tagName)) { + // We don't need to support custom elements since this is an internal API. + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + if (NS_WARN_IF(tagName == nsGkAtoms::_empty)) { + return NS_ERROR_INVALID_ARG; + } + + if (!aNode) { + AutoEditActionDataSetter dummyEditAction(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!dummyEditAction.CanHandle())) { + return NS_ERROR_NOT_AVAILABLE; + } + RefPtr<Element> parentElement = + GetInclusiveAncestorByTagNameAtSelection(*tagName); + if (!parentElement) { + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + parentElement.forget(aReturn); + return NS_OK; + } + + if (!aNode->IsContent() || !aNode->GetAsElementOrParentElement()) { + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + + RefPtr<Element> parentElement = + GetInclusiveAncestorByTagName(*tagName, *aNode->AsContent()); + if (!parentElement) { + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + parentElement.forget(aReturn); + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::GetSelectedElement(const nsAString& aTagName, + nsISupports** aReturn) { + if (NS_WARN_IF(!aReturn)) { + return NS_ERROR_INVALID_ARG; + } + *aReturn = nullptr; + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + + ErrorResult error; + nsStaticAtom* tagName = EditorUtils::GetTagNameAtom(aTagName); + if (!aTagName.IsEmpty() && !tagName) { + // We don't need to support custom elements becaus of internal API. + return NS_OK; + } + RefPtr<nsINode> selectedNode = GetSelectedElement(tagName, error); + NS_WARNING_ASSERTION(!error.Failed(), + "HTMLEditor::GetSelectedElement() failed"); + selectedNode.forget(aReturn); + return error.StealNSResult(); +} + +already_AddRefed<Element> HTMLEditor::GetSelectedElement(const nsAtom* aTagName, + ErrorResult& aRv) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + MOZ_ASSERT(!aRv.Failed()); + + // If there is no Selection or two or more selection ranges, that means that + // not only one element is selected so that return nullptr. + if (SelectionRef().RangeCount() != 1) { + return nullptr; + } + + bool isLinkTag = aTagName && IsLinkTag(*aTagName); + bool isNamedAnchorTag = aTagName && IsNamedAnchorTag(*aTagName); + + RefPtr<nsRange> firstRange = SelectionRef().GetRangeAt(0); + MOZ_ASSERT(firstRange); + + const RangeBoundary& startRef = firstRange->StartRef(); + if (NS_WARN_IF(!startRef.IsSet())) { + aRv.Throw(NS_ERROR_FAILURE); + return nullptr; + } + const RangeBoundary& endRef = firstRange->EndRef(); + if (NS_WARN_IF(!endRef.IsSet())) { + aRv.Throw(NS_ERROR_FAILURE); + return nullptr; + } + + // Optimization for a single selected element + if (startRef.Container() == endRef.Container()) { + nsIContent* startContent = startRef.GetChildAtOffset(); + nsIContent* endContent = endRef.GetChildAtOffset(); + if (startContent && endContent && + startContent->GetNextSibling() == endContent) { + if (!aTagName) { + if (!startContent->IsElement()) { + // This means only a text node or something is selected. We should + // return nullptr in this case since no other elements are selected. + return nullptr; + } + return do_AddRef(startContent->AsElement()); + } + // Test for appropriate node type requested + if (aTagName == startContent->NodeInfo()->NameAtom() || + (isLinkTag && HTMLEditUtils::IsLink(startContent)) || + (isNamedAnchorTag && HTMLEditUtils::IsNamedAnchor(startContent))) { + MOZ_ASSERT(startContent->IsElement()); + return do_AddRef(startContent->AsElement()); + } + } + } + + if (isLinkTag && startRef.Container()->IsContent() && + endRef.Container()->IsContent()) { + // Link node must be the same for both ends of selection. + Element* parentLinkOfStart = GetInclusiveAncestorByTagNameInternal( + *nsGkAtoms::href, *startRef.Container()->AsContent()); + if (parentLinkOfStart) { + if (SelectionRef().IsCollapsed()) { + // We have just a caret in the link. + return do_AddRef(parentLinkOfStart); + } + // Link node must be the same for both ends of selection. + Element* parentLinkOfEnd = GetInclusiveAncestorByTagNameInternal( + *nsGkAtoms::href, *endRef.Container()->AsContent()); + if (parentLinkOfStart == parentLinkOfEnd) { + return do_AddRef(parentLinkOfStart); + } + } + } + + if (SelectionRef().IsCollapsed()) { + return nullptr; + } + + PostContentIterator postOrderIter; + postOrderIter.Init(firstRange); + + RefPtr<Element> lastElementInRange; + for (nsINode* lastNodeInRange = nullptr; !postOrderIter.IsDone(); + postOrderIter.Next()) { + if (lastElementInRange) { + // When any node follows an element node, not only one element is + // selected so that return nullptr. + return nullptr; + } + + // This loop ignors any non-element nodes before first element node. + // Its purpose must be that this method treats this case as selecting + // the <b> element: + // - <p>abc <b>d[ef</b>}</p> + // because children of an element node is listed up before the element. + // However, this case must not be expected by the initial developer: + // - <p>a[bc <b>def</b>}</p> + // When we meet non-parent and non-next-sibling node of previous node, + // it means that the range across element boundary (open tag in HTML + // source). So, in this case, we should not say only the following + // element is selected. + nsINode* currentNode = postOrderIter.GetCurrentNode(); + MOZ_ASSERT(currentNode); + if (lastNodeInRange && lastNodeInRange->GetParentNode() != currentNode && + lastNodeInRange->GetNextSibling() != currentNode) { + return nullptr; + } + + lastNodeInRange = currentNode; + + lastElementInRange = Element::FromNodeOrNull(lastNodeInRange); + if (!lastElementInRange) { + continue; + } + + // And also, if it's followed by a <br> element, we shouldn't treat the + // the element is selected like this case: + // - <p><b>[def</b>}<br></p> + // Note that we don't need special handling for <a href> because double + // clicking it selects the element and we use the first path to handle it. + // Additionally, we have this case too: + // - <p><b>[def</b><b>}<br></b></p> + // In these cases, the <br> element is not listed up by PostContentIterator. + // So, we should return nullptr if next sibling is a `<br>` element or + // next sibling starts with `<br>` element. + if (nsIContent* nextSibling = lastElementInRange->GetNextSibling()) { + if (nextSibling->IsHTMLElement(nsGkAtoms::br)) { + return nullptr; + } + nsIContent* firstEditableLeaf = HTMLEditUtils::GetFirstLeafContent( + *nextSibling, {LeafNodeType::OnlyLeafNode}); + if (firstEditableLeaf && + firstEditableLeaf->IsHTMLElement(nsGkAtoms::br)) { + return nullptr; + } + } + + if (!aTagName) { + continue; + } + + if (isLinkTag && HTMLEditUtils::IsLink(lastElementInRange)) { + continue; + } + + if (isNamedAnchorTag && HTMLEditUtils::IsNamedAnchor(lastElementInRange)) { + continue; + } + + if (aTagName == lastElementInRange->NodeInfo()->NameAtom()) { + continue; + } + + // First element in the range does not match what the caller is looking + // for. + return nullptr; + } + return lastElementInRange.forget(); +} + +Result<CreateElementResult, nsresult> HTMLEditor::CreateAndInsertElement( + WithTransaction aWithTransaction, nsAtom& aTagName, + const EditorDOMPoint& aPointToInsert, + const InitializeInsertingElement& aInitializer) { + MOZ_ASSERT(IsEditActionDataAvailable()); + MOZ_ASSERT(aPointToInsert.IsSetAndValid()); + + // XXX We need offset at new node for RangeUpdaterRef(). Therefore, we need + // to compute the offset now but this is expensive. So, if it's possible, + // we need to redesign RangeUpdaterRef() as avoiding using indices. + Unused << aPointToInsert.Offset(); + + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eCreateNode, nsIEditor::eNext, ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION( + !ignoredError.Failed(), + "HTMLEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored"); + + // TODO: This method should have a callback function which is called + // immediately after creating an element but before it's inserted into + // the DOM tree. Then, caller can init the new element's attributes + // and children **without** transactions (it'll reduce the number of + // legacy mutation events). Finally, we can get rid of + // CreatElementTransaction since we can use InsertNodeTransaction + // instead. + + auto createNewElementResult = + [&]() MOZ_CAN_RUN_SCRIPT -> Result<CreateElementResult, nsresult> { + RefPtr<Element> newElement = CreateHTMLContent(&aTagName); + if (MOZ_UNLIKELY(!newElement)) { + NS_WARNING("EditorBase::CreateHTMLContent() failed"); + return Err(NS_ERROR_FAILURE); + } + nsresult rv = MarkElementDirty(*newElement); + if (MOZ_UNLIKELY(rv == NS_ERROR_EDITOR_DESTROYED)) { + NS_WARNING("EditorBase::MarkElementDirty() caused destroying the editor"); + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::MarkElementDirty() failed, but ignored"); + rv = aInitializer(*this, *newElement, aPointToInsert); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "aInitializer failed"); + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + RefPtr<InsertNodeTransaction> transaction = + InsertNodeTransaction::Create(*this, *newElement, aPointToInsert); + rv = aWithTransaction == WithTransaction::Yes + ? DoTransactionInternal(transaction) + : transaction->DoTransaction(); + // FYI: Transaction::DoTransaction never returns NS_ERROR_EDITOR_*. + if (MOZ_UNLIKELY(Destroyed())) { + NS_WARNING( + "InsertNodeTransaction::DoTransaction() caused destroying the " + "editor"); + return Err(NS_ERROR_EDITOR_DESTROYED); + } + if (NS_FAILED(rv)) { + NS_WARNING("InsertNodeTransaction::DoTransaction() failed"); + return Err(rv); + } + // Override the success code if new element was moved by the web apps. + if (newElement && + newElement->GetParentNode() != aPointToInsert.GetContainer()) { + NS_WARNING("The new element was not inserted into the expected node"); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + return CreateElementResult( + std::move(newElement), + transaction->SuggestPointToPutCaret<EditorDOMPoint>()); + }(); + + if (MOZ_UNLIKELY(createNewElementResult.isErr())) { + NS_WARNING("EditorBase::DoTransactionInternal() failed"); + // XXX Why do we do this even when DoTransaction() returned error? + DebugOnly<nsresult> rvIgnored = + RangeUpdaterRef().SelAdjCreateNode(aPointToInsert); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "RangeUpdater::SelAdjCreateNode() failed"); + return createNewElementResult; + } + + // If we succeeded to create and insert new element, we need to adjust + // ranges in RangeUpdaterRef(). It currently requires offset of the new + // node. So, let's call it with original offset. Note that if + // aPointToInsert stores child node, it may not be at the offset since new + // element must be inserted before the old child. Although, mutation + // observer can do anything, but currently, we don't check it. + DebugOnly<nsresult> rvIgnored = + RangeUpdaterRef().SelAdjCreateNode(EditorRawDOMPoint( + aPointToInsert.GetContainer(), aPointToInsert.Offset())); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "RangeUpdater::SelAdjCreateNode() failed, but ignored"); + if (MOZ_LIKELY(createNewElementResult.inspect().GetNewNode())) { + TopLevelEditSubActionDataRef().DidCreateElement( + *this, *createNewElementResult.inspect().GetNewNode()); + } + + return createNewElementResult; +} + +nsresult HTMLEditor::CopyAttributes(WithTransaction aWithTransaction, + Element& aDestElement, Element& aSrcElement, + const AttributeFilter& aFilterFunc) { + RefPtr<nsDOMAttributeMap> srcAttributes = aSrcElement.Attributes(); + if (!srcAttributes->Length()) { + return NS_OK; + } + AutoTArray<OwningNonNull<Attr>, 16> srcAttrs; + srcAttrs.SetCapacity(srcAttributes->Length()); + for (uint32_t i = 0; i < srcAttributes->Length(); i++) { + RefPtr<Attr> attr = srcAttributes->Item(i); + if (!attr) { + break; + } + srcAttrs.AppendElement(std::move(attr)); + } + if (aWithTransaction == WithTransaction::No) { + for (const OwningNonNull<Attr>& attr : srcAttrs) { + nsString value; + attr->GetValue(value); + if (!aFilterFunc(*this, aSrcElement, aDestElement, attr, value)) { + continue; + } + DebugOnly<nsresult> rvIgnored = + aDestElement.SetAttr(attr->NodeInfo()->NamespaceID(), + attr->NodeInfo()->NameAtom(), value, false); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "Element::SetAttr() failed, but ignored"); + } + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + return NS_OK; + } + MOZ_ASSERT_UNREACHABLE("Not implemented yet, but you try to use this"); + return NS_ERROR_NOT_IMPLEMENTED; +} + +already_AddRefed<Element> HTMLEditor::CreateElementWithDefaults( + const nsAtom& aTagName) { + // NOTE: Despite of public method, this can be called for internal use. + + // Although this creates an element, but won't change the DOM tree nor + // transaction. So, EditAtion::eNotEditing is proper value here. If + // this is called for internal when there is already AutoEditActionDataSetter + // instance, this would be initialized with its EditAction value. + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return nullptr; + } + + const nsAtom* realTagName = IsLinkTag(aTagName) || IsNamedAnchorTag(aTagName) + ? nsGkAtoms::a + : &aTagName; + + // We don't use editor's CreateElement because we don't want to go through + // the transaction system + + // New call to use instead to get proper HTML element, bug 39919 + RefPtr<Element> newElement = CreateHTMLContent(realTagName); + if (!newElement) { + return nullptr; + } + + // Mark the new element dirty, so it will be formatted + // XXX Don't we need to check the error result of setting _moz_dirty attr? + IgnoredErrorResult ignoredError; + newElement->SetAttribute(u"_moz_dirty"_ns, u""_ns, ignoredError); + NS_WARNING_ASSERTION(!ignoredError.Failed(), + "Element::SetAttribute(_moz_dirty) failed, but ignored"); + ignoredError.SuppressException(); + + // Set default values for new elements + if (realTagName == nsGkAtoms::table) { + newElement->SetAttr(nsGkAtoms::cellpadding, u"2"_ns, ignoredError); + if (ignoredError.Failed()) { + NS_WARNING("Element::SetAttr(nsGkAtoms::cellpadding, 2) failed"); + return nullptr; + } + ignoredError.SuppressException(); + + newElement->SetAttr(nsGkAtoms::cellspacing, u"2"_ns, ignoredError); + if (ignoredError.Failed()) { + NS_WARNING("Element::SetAttr(nsGkAtoms::cellspacing, 2) failed"); + return nullptr; + } + ignoredError.SuppressException(); + + newElement->SetAttr(nsGkAtoms::border, u"1"_ns, ignoredError); + if (ignoredError.Failed()) { + NS_WARNING("Element::SetAttr(nsGkAtoms::border, 1) failed"); + return nullptr; + } + } else if (realTagName == nsGkAtoms::td) { + nsresult rv = SetAttributeOrEquivalent(newElement, nsGkAtoms::valign, + u"top"_ns, true); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::SetAttributeOrEquivalent(nsGkAtoms::valign, top) " + "failed"); + return nullptr; + } + } + // ADD OTHER TAGS HERE + + return newElement.forget(); +} + +NS_IMETHODIMP HTMLEditor::CreateElementWithDefaults(const nsAString& aTagName, + Element** aReturn) { + if (NS_WARN_IF(aTagName.IsEmpty()) || NS_WARN_IF(!aReturn)) { + return NS_ERROR_INVALID_ARG; + } + + *aReturn = nullptr; + + nsStaticAtom* tagName = EditorUtils::GetTagNameAtom(aTagName); + if (NS_WARN_IF(!tagName)) { + return NS_ERROR_INVALID_ARG; + } + RefPtr<Element> newElement = + CreateElementWithDefaults(MOZ_KnownLive(*tagName)); + if (!newElement) { + NS_WARNING("HTMLEditor::CreateElementWithDefaults() failed"); + return NS_ERROR_FAILURE; + } + newElement.forget(aReturn); + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::InsertLinkAroundSelection(Element* aAnchorElement) { + nsresult rv = InsertLinkAroundSelectionAsAction(aAnchorElement); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::InsertLinkAroundSelectionAsAction() failed"); + return rv; +} + +nsresult HTMLEditor::InsertLinkAroundSelectionAsAction( + Element* aAnchorElement, nsIPrincipal* aPrincipal) { + if (NS_WARN_IF(!aAnchorElement)) { + return NS_ERROR_INVALID_ARG; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eInsertLinkElement, + aPrincipal); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + + if (SelectionRef().IsCollapsed()) { + NS_WARNING("Selection was collapsed"); + return NS_OK; + } + + // Be sure we were given an anchor element + RefPtr<HTMLAnchorElement> anchor = + HTMLAnchorElement::FromNodeOrNull(aAnchorElement); + if (!anchor) { + return NS_OK; + } + + nsAutoString rawHref; + anchor->GetAttr(kNameSpaceID_None, nsGkAtoms::href, rawHref); + editActionData.SetData(rawHref); + + nsresult rv = editActionData.MaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "MaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + nsAutoString href; + anchor->GetHref(href); + if (href.IsEmpty()) { + return NS_OK; + } + + AutoPlaceholderBatch treatAsOneTransaction( + *this, ScrollSelectionIntoView::Yes, __FUNCTION__); + + // Set all attributes found on the supplied anchor element + RefPtr<nsDOMAttributeMap> attributeMap = anchor->Attributes(); + if (NS_WARN_IF(!attributeMap)) { + return NS_ERROR_FAILURE; + } + + // TODO: We should stop using this loop for adding attributes to newly created + // `<a href="...">` elements. Then, we can avoid to increate the ref- + // counter of attribute names since we can use nsStaticAtom if we don't + // need to support unknown attributes. + AutoTArray<EditorInlineStyleAndValue, 32> stylesToSet; + stylesToSet.SetCapacity(attributeMap->Length()); + nsString value; + for (uint32_t i : IntegerRange(attributeMap->Length())) { + RefPtr<Attr> attribute = attributeMap->Item(i); + if (!attribute) { + continue; + } + + RefPtr<nsAtom> attributeName = attribute->NodeInfo()->NameAtom(); + + MOZ_ASSERT(value.IsEmpty()); + attribute->GetValue(value); + + stylesToSet.AppendElement(EditorInlineStyleAndValue( + *nsGkAtoms::a, std::move(attributeName), std::move(value))); + } + rv = SetInlinePropertiesAsSubAction(stylesToSet); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::SetInlinePropertiesAsSubAction() failed"); + return rv; +} + +nsresult HTMLEditor::SetHTMLBackgroundColorWithTransaction( + const nsAString& aColor) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // Find a selected or enclosing table element to set background on + bool isCellSelected = false; + Result<RefPtr<Element>, nsresult> cellOrRowOrTableElementOrError = + GetSelectedOrParentTableElement(&isCellSelected); + if (cellOrRowOrTableElementOrError.isErr()) { + NS_WARNING("HTMLEditor::GetSelectedOrParentTableElement() failed"); + return cellOrRowOrTableElementOrError.unwrapErr(); + } + + bool setColor = !aColor.IsEmpty(); + RefPtr<Element> rootElementOfBackgroundColor = + cellOrRowOrTableElementOrError.unwrap(); + if (rootElementOfBackgroundColor) { + // Needs to set or remove background color of each selected cell elements. + // Therefore, just the cell contains selection range, we don't need to + // do this. Note that users can select each cell, but with Selection API, + // web apps can select <tr> and <td> at same time. With <table>, looks + // odd, though. + if (isCellSelected || rootElementOfBackgroundColor->IsAnyOfHTMLElements( + nsGkAtoms::table, nsGkAtoms::tr)) { + SelectedTableCellScanner scanner(SelectionRef()); + if (scanner.IsInTableCellSelectionMode()) { + if (setColor) { + for (const OwningNonNull<Element>& cellElement : + scanner.ElementsRef()) { + // `MOZ_KnownLive(cellElement)` is safe because of `scanner` + // is stack only class and keeps grabbing it until it's destroyed. + nsresult rv = SetAttributeWithTransaction( + MOZ_KnownLive(cellElement), *nsGkAtoms::bgcolor, aColor); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + if (NS_FAILED(rv)) { + NS_WARNING( + "EditorBase::::SetAttributeWithTransaction(nsGkAtoms::" + "bgcolor) failed"); + return rv; + } + } + return NS_OK; + } + for (const OwningNonNull<Element>& cellElement : + scanner.ElementsRef()) { + // `MOZ_KnownLive(cellElement)` is safe because of `scanner` + // is stack only class and keeps grabbing it until it's destroyed. + nsresult rv = RemoveAttributeWithTransaction( + MOZ_KnownLive(cellElement), *nsGkAtoms::bgcolor); + if (NS_FAILED(rv)) { + NS_WARNING( + "EditorBase::RemoveAttributeWithTransaction(nsGkAtoms::bgcolor)" + " failed"); + return rv; + } + } + return NS_OK; + } + } + // If we failed to find a cell, fall through to use originally-found element + } else { + // No table element -- set the background color on the body tag + rootElementOfBackgroundColor = GetRoot(); + if (NS_WARN_IF(!rootElementOfBackgroundColor)) { + return NS_ERROR_FAILURE; + } + } + // Use the editor method that goes through the transaction system + if (setColor) { + nsresult rv = SetAttributeWithTransaction(*rootElementOfBackgroundColor, + *nsGkAtoms::bgcolor, aColor); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::SetAttributeWithTransaction(nsGkAtoms::bgcolor) failed"); + return rv; + } + nsresult rv = RemoveAttributeWithTransaction(*rootElementOfBackgroundColor, + *nsGkAtoms::bgcolor); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::RemoveAttributeWithTransaction(nsGkAtoms::bgcolor) failed"); + return rv; +} + +nsresult HTMLEditor::RemoveEmptyInclusiveAncestorInlineElements( + nsIContent& aContent) { + MOZ_ASSERT(IsEditActionDataAvailable()); + MOZ_ASSERT(!aContent.Length()); + + Element* editingHost = aContent.GetEditingHost(); + if (NS_WARN_IF(!editingHost)) { + return NS_ERROR_FAILURE; + } + + if (&aContent == editingHost || HTMLEditUtils::IsBlockElement(aContent) || + !EditorUtils::IsEditableContent(aContent, EditorType::HTML) || + !aContent.GetParent()) { + return NS_OK; + } + + // Don't strip wrappers if this is the only wrapper in the block. Then we'll + // add a <br> later, so it won't be an empty wrapper in the end. + // XXX This is different from Blink. We should delete empty inline element + // even if it's only child of the block element. + { + const Element* editableBlockElement = HTMLEditUtils::GetAncestorElement( + aContent, HTMLEditUtils::ClosestEditableBlockElement); + if (!editableBlockElement || + HTMLEditUtils::IsEmptyNode( + *editableBlockElement, + {EmptyCheckOption::TreatSingleBRElementAsVisible})) { + return NS_OK; + } + } + + OwningNonNull<nsIContent> content = aContent; + for (nsIContent* parentContent : aContent.AncestorsOfType<nsIContent>()) { + if (HTMLEditUtils::IsBlockElement(*parentContent) || + parentContent->Length() != 1 || + !EditorUtils::IsEditableContent(*parentContent, EditorType::HTML) || + parentContent == editingHost) { + break; + } + content = *parentContent; + } + + nsresult rv = DeleteNodeWithTransaction(content); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::DeleteNodeWithTransaction() failed"); + return rv; +} + +nsresult HTMLEditor::DeleteAllChildrenWithTransaction(Element& aElement) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // Prevent rules testing until we're done + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eDeleteNode, nsIEditor::eNext, ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return ignoredError.StealNSResult(); + } + NS_WARNING_ASSERTION( + !ignoredError.Failed(), + "OnStartToHandleTopLevelEditSubAction() failed, but ignored"); + + while (nsCOMPtr<nsIContent> child = aElement.GetLastChild()) { + nsresult rv = DeleteNodeWithTransaction(*child); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return rv; + } + } + return NS_OK; +} + +NS_IMETHODIMP HTMLEditor::DeleteNode(nsINode* aNode) { + if (NS_WARN_IF(!aNode) || NS_WARN_IF(!aNode->IsContent())) { + return NS_ERROR_INVALID_ARG; + } + + AutoEditActionDataSetter editActionData(*this, EditAction::eRemoveNode); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + rv = DeleteNodeWithTransaction(MOZ_KnownLive(*aNode->AsContent())); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::DeleteNodeWithTransaction() failed"); + return rv; +} + +Result<CaretPoint, nsresult> HTMLEditor::DeleteTextWithTransaction( + Text& aTextNode, uint32_t aOffset, uint32_t aLength) { + if (NS_WARN_IF(!HTMLEditUtils::IsSimplyEditableNode(aTextNode))) { + return Err(NS_ERROR_FAILURE); + } + + Result<CaretPoint, nsresult> caretPointOrError = + EditorBase::DeleteTextWithTransaction(aTextNode, aOffset, aLength); + NS_WARNING_ASSERTION(caretPointOrError.isOk(), + "EditorBase::DeleteTextWithTransaction() failed"); + return caretPointOrError; +} + +Result<InsertTextResult, nsresult> HTMLEditor::ReplaceTextWithTransaction( + Text& aTextNode, uint32_t aOffset, uint32_t aLength, + const nsAString& aStringToInsert) { + MOZ_ASSERT(IsEditActionDataAvailable()); + MOZ_ASSERT(aLength > 0 || !aStringToInsert.IsEmpty()); + + if (aStringToInsert.IsEmpty()) { + Result<CaretPoint, nsresult> caretPointOrError = + DeleteTextWithTransaction(aTextNode, aOffset, aLength); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("HTMLEditor::DeleteTextWithTransaction() failed"); + return caretPointOrError.propagateErr(); + } + return InsertTextResult(EditorDOMPointInText(&aTextNode, aOffset), + caretPointOrError.unwrap()); + } + + if (!aLength) { + RefPtr<Document> document = GetDocument(); + if (NS_WARN_IF(!document)) { + return Err(NS_ERROR_NOT_INITIALIZED); + } + Result<InsertTextResult, nsresult> insertTextResult = + InsertTextWithTransaction(*document, aStringToInsert, + EditorDOMPoint(&aTextNode, aOffset)); + NS_WARNING_ASSERTION(insertTextResult.isOk(), + "HTMLEditor::InsertTextWithTransaction() failed"); + return insertTextResult; + } + + if (NS_WARN_IF(!HTMLEditUtils::IsSimplyEditableNode(aTextNode))) { + return Err(NS_ERROR_FAILURE); + } + + // This should emulates inserting text for better undo/redo behavior. + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eInsertText, nsIEditor::eNext, ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION( + !ignoredError.Failed(), + "HTMLEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored"); + + // FYI: Create the insertion point before changing the DOM tree because + // the point may become invalid offset after that. + EditorDOMPointInText pointToInsert(&aTextNode, aOffset); + + RefPtr<ReplaceTextTransaction> transaction = ReplaceTextTransaction::Create( + *this, aStringToInsert, aTextNode, aOffset, aLength); + MOZ_ASSERT(transaction); + + if (aLength && !mActionListeners.IsEmpty()) { + for (auto& listener : mActionListeners.Clone()) { + DebugOnly<nsresult> rvIgnored = + listener->WillDeleteText(&aTextNode, aOffset, aLength); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "nsIEditActionListener::WillDeleteText() failed, but ignored"); + } + } + + nsresult rv = DoTransactionInternal(transaction); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::DoTransactionInternal() failed"); + + // Don't check whether we've been destroyed here because we need to notify + // listeners and observers below even if we've already destroyed. + + EditorDOMPointInText endOfInsertedText(&aTextNode, + aOffset + aStringToInsert.Length()); + + if (pointToInsert.IsSet()) { + auto [begin, end] = ComputeInsertedRange(pointToInsert, aStringToInsert); + if (begin.IsSet() && end.IsSet()) { + TopLevelEditSubActionDataRef().DidDeleteText( + *this, begin.To<EditorRawDOMPoint>()); + TopLevelEditSubActionDataRef().DidInsertText( + *this, begin.To<EditorRawDOMPoint>(), end.To<EditorRawDOMPoint>()); + } + + // XXX Should we update endOfInsertedText here? + } + + if (!mActionListeners.IsEmpty()) { + for (auto& listener : mActionListeners.Clone()) { + DebugOnly<nsresult> rvIgnored = + listener->DidInsertText(&aTextNode, aOffset, aStringToInsert, rv); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "nsIEditActionListener::DidInsertText() failed, but ignored"); + } + } + + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + + return InsertTextResult( + std::move(endOfInsertedText), + transaction->SuggestPointToPutCaret<EditorDOMPoint>()); +} + +Result<InsertTextResult, nsresult> HTMLEditor::InsertTextWithTransaction( + Document& aDocument, const nsAString& aStringToInsert, + const EditorDOMPoint& aPointToInsert) { + if (NS_WARN_IF(!aPointToInsert.IsSet())) { + return Err(NS_ERROR_INVALID_ARG); + } + + // Do nothing if the node is read-only + if (MOZ_UNLIKELY(NS_WARN_IF(!HTMLEditUtils::IsSimplyEditableNode( + *aPointToInsert.GetContainer())))) { + return Err(NS_ERROR_FAILURE); + } + + return EditorBase::InsertTextWithTransaction(aDocument, aStringToInsert, + aPointToInsert); +} + +Result<EditorDOMPoint, nsresult> HTMLEditor::PrepareToInsertBRElement( + const EditorDOMPoint& aPointToInsert) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (NS_WARN_IF(!aPointToInsert.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + + if (!aPointToInsert.IsInTextNode()) { + return aPointToInsert; + } + + if (aPointToInsert.IsStartOfContainer()) { + // Insert before the text node. + EditorDOMPoint pointInContainer(aPointToInsert.GetContainer()); + if (!pointInContainer.IsSet()) { + NS_WARNING("Failed to climb up the DOM tree from text node"); + return Err(NS_ERROR_FAILURE); + } + return pointInContainer; + } + + if (aPointToInsert.IsEndOfContainer()) { + // Insert after the text node. + EditorDOMPoint pointInContainer(aPointToInsert.GetContainer()); + if (NS_WARN_IF(!pointInContainer.IsSet())) { + NS_WARNING("Failed to climb up the DOM tree from text node"); + return Err(NS_ERROR_FAILURE); + } + MOZ_ALWAYS_TRUE(pointInContainer.AdvanceOffset()); + return pointInContainer; + } + + MOZ_DIAGNOSTIC_ASSERT(aPointToInsert.IsSetAndValid()); + + // Unfortunately, we need to split the text node at the offset. + Result<SplitNodeResult, nsresult> splitTextNodeResult = + SplitNodeWithTransaction(aPointToInsert); + if (MOZ_UNLIKELY(splitTextNodeResult.isErr())) { + NS_WARNING("HTMLEditor::SplitNodeWithTransaction() failed"); + return splitTextNodeResult.propagateErr(); + } + nsresult rv = splitTextNodeResult.inspect().SuggestCaretPointTo( + *this, {SuggestCaret::OnlyIfTransactionsAllowedToDoIt}); + if (NS_FAILED(rv)) { + NS_WARNING("SplitNodeResult::SuggestCaretPointTo() failed"); + return Err(rv); + } + + // Insert new <br> before the right node. + auto atNextContent = + splitTextNodeResult.inspect().AtNextContent<EditorDOMPoint>(); + if (MOZ_UNLIKELY(!atNextContent.IsSet())) { + NS_WARNING("The next node seems not in the DOM tree"); + return Err(NS_ERROR_FAILURE); + } + return atNextContent; +} + +Result<CreateElementResult, nsresult> HTMLEditor::InsertBRElement( + WithTransaction aWithTransaction, const EditorDOMPoint& aPointToInsert, + EDirection aSelect /* = eNone */) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + Result<EditorDOMPoint, nsresult> maybePointToInsert = + PrepareToInsertBRElement(aPointToInsert); + if (maybePointToInsert.isErr()) { + NS_WARNING( + nsPrintfCString("HTMLEditor::PrepareToInsertBRElement(%s) failed", + ToString(aWithTransaction).c_str()) + .get()); + return maybePointToInsert.propagateErr(); + } + MOZ_ASSERT(maybePointToInsert.inspect().IsSetAndValid()); + + Result<CreateElementResult, nsresult> createNewBRElementResult = + CreateAndInsertElement(aWithTransaction, *nsGkAtoms::br, + maybePointToInsert.inspect()); + if (MOZ_UNLIKELY(createNewBRElementResult.isErr())) { + NS_WARNING(nsPrintfCString("HTMLEditor::CreateAndInsertElement(%s) failed", + ToString(aWithTransaction).c_str()) + .get()); + return createNewBRElementResult.propagateErr(); + } + CreateElementResult unwrappedCreateNewBRElementResult = + createNewBRElementResult.unwrap(); + RefPtr<Element> newBRElement = + unwrappedCreateNewBRElementResult.UnwrapNewNode(); + MOZ_ASSERT(newBRElement); + + unwrappedCreateNewBRElementResult.IgnoreCaretPointSuggestion(); + switch (aSelect) { + case eNext: { + const auto pointToPutCaret = EditorDOMPoint::After( + *newBRElement, Selection::InterlinePosition::StartOfNextLine); + return CreateElementResult(std::move(newBRElement), pointToPutCaret); + } + case ePrevious: { + const auto pointToPutCaret = EditorDOMPoint( + newBRElement, Selection::InterlinePosition::StartOfNextLine); + return CreateElementResult(std::move(newBRElement), pointToPutCaret); + } + default: + NS_WARNING( + "aSelect has invalid value, the caller need to set selection " + "by itself"); + [[fallthrough]]; + case eNone: + return CreateElementResult( + std::move(newBRElement), + unwrappedCreateNewBRElementResult.UnwrapCaretPoint()); + } +} + +Result<CreateElementResult, nsresult> +HTMLEditor::InsertContainerWithTransaction( + nsIContent& aContentToBeWrapped, const nsAtom& aWrapperTagName, + const InitializeInsertingElement& aInitializer) { + EditorDOMPoint pointToInsertNewContainer(&aContentToBeWrapped); + if (NS_WARN_IF(!pointToInsertNewContainer.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + // aContentToBeWrapped will be moved to the new container before inserting the + // new container. So, when we insert the container, the insertion point is + // before the next sibling of aContentToBeWrapped. + // XXX If pointerToInsertNewContainer stores offset here, the offset and + // referring child node become mismatched. Although, currently this + // is not a problem since InsertNodeTransaction refers only child node. + MOZ_ALWAYS_TRUE(pointToInsertNewContainer.AdvanceOffset()); + + // Create new container. + RefPtr<Element> newContainer = CreateHTMLContent(&aWrapperTagName); + if (NS_WARN_IF(!newContainer)) { + return Err(NS_ERROR_FAILURE); + } + + if (&aInitializer != &HTMLEditor::DoNothingForNewElement) { + nsresult rv = aInitializer(*this, *newContainer, + EditorDOMPoint(&aContentToBeWrapped)); + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + if (NS_FAILED(rv)) { + NS_WARNING("aInitializer() failed"); + return Err(rv); + } + } + + // Notify our internal selection state listener + AutoInsertContainerSelNotify selNotify(RangeUpdaterRef()); + + // Put aNode in the new container, first. + // XXX Perhaps, we should not remove the container if it's not editable. + nsresult rv = DeleteNodeWithTransaction(aContentToBeWrapped); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return Err(rv); + } + + { + // TODO: Remove AutoTransactionsConserveSelection here. It's not necessary + // in normal cases. However, it may be required for nested edit + // actions which may be caused by legacy mutation event listeners or + // chrome script. + AutoTransactionsConserveSelection conserveSelection(*this); + Result<CreateContentResult, nsresult> insertContentNodeResult = + InsertNodeWithTransaction(aContentToBeWrapped, + EditorDOMPoint(newContainer, 0u)); + if (MOZ_UNLIKELY(insertContentNodeResult.isErr())) { + NS_WARNING("EditorBase::InsertNodeWithTransaction() failed"); + return insertContentNodeResult.propagateErr(); + } + insertContentNodeResult.inspect().IgnoreCaretPointSuggestion(); + } + + // Put the new container where aNode was. + Result<CreateElementResult, nsresult> insertNewContainerElementResult = + InsertNodeWithTransaction<Element>(*newContainer, + pointToInsertNewContainer); + NS_WARNING_ASSERTION(insertNewContainerElementResult.isOk(), + "EditorBase::InsertNodeWithTransaction() failed"); + return insertNewContainerElementResult; +} + +Result<CreateElementResult, nsresult> +HTMLEditor::ReplaceContainerWithTransactionInternal( + Element& aOldContainer, const nsAtom& aTagName, const nsAtom& aAttribute, + const nsAString& aAttributeValue, bool aCloneAllAttributes) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (NS_WARN_IF(!HTMLEditUtils::IsRemovableNode(aOldContainer)) || + NS_WARN_IF(!HTMLEditUtils::IsSimplyEditableNode(aOldContainer))) { + return Err(NS_ERROR_FAILURE); + } + + const RefPtr<Element> newContainer = CreateHTMLContent(&aTagName); + if (NS_WARN_IF(!newContainer)) { + return Err(NS_ERROR_FAILURE); + } + + // Set or clone attribute if needed. + if (aCloneAllAttributes) { + MOZ_ASSERT(&aAttribute == nsGkAtoms::_empty); + CloneAttributesWithTransaction(*newContainer, aOldContainer); + } else if (&aAttribute != nsGkAtoms::_empty) { + nsresult rv = newContainer->SetAttr(kNameSpaceID_None, + const_cast<nsAtom*>(&aAttribute), + aAttributeValue, true); + if (NS_FAILED(rv)) { + NS_WARNING("Element::SetAttr() failed"); + return Err(NS_ERROR_FAILURE); + } + } + + const OwningNonNull<nsINode> parentNode = *aOldContainer.GetParentNode(); + const nsCOMPtr<nsINode> referenceNode = aOldContainer.GetNextSibling(); + AutoReplaceContainerSelNotify selStateNotify(RangeUpdaterRef(), aOldContainer, + *newContainer); + { + AutoTArray<OwningNonNull<nsIContent>, 32> arrayOfChildren; + HTMLEditUtils::CollectChildren( + aOldContainer, arrayOfChildren, 0u, + // Move non-editable children too because its container, aElement, is + // editable so that all children must be removable node. + {}); + // TODO: Remove AutoTransactionsConserveSelection here. It's not necessary + // in normal cases. However, it may be required for nested edit + // actions which may be caused by legacy mutation event listeners or + // chrome script. + AutoTransactionsConserveSelection conserveSelection(*this); + // Move all children from the old container to the new container. + // For making all MoveNodeTransactions have a referenc node in the current + // parent, move nodes from last one to preceding ones. + for (const OwningNonNull<nsIContent>& child : Reversed(arrayOfChildren)) { + Result<MoveNodeResult, nsresult> moveChildResult = + MoveNodeWithTransaction(MOZ_KnownLive(child), // due to bug 1622253. + EditorDOMPoint(newContainer, 0u)); + if (MOZ_UNLIKELY(moveChildResult.isErr())) { + NS_WARNING("HTMLEditor::MoveNodeWithTransaction() failed"); + return moveChildResult.propagateErr(); + } + // We'll suggest new caret point which is suggested by new container + // element insertion result. Therefore, we need to do nothing here. + moveChildResult.inspect().IgnoreCaretPointSuggestion(); + } + } + + // Delete aOldContainer from the DOM tree to make it not referred by + // InsertNodeTransaction. + nsresult rv = DeleteNodeWithTransaction(aOldContainer); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return Err(rv); + } + + if (referenceNode && (!referenceNode->GetParentNode() || + parentNode != referenceNode->GetParentNode())) { + NS_WARNING( + "The reference node for insertion has been moved to different parent, " + "so we got lost the insertion point"); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + // Finally, insert the new node to where probably aOldContainer was. + Result<CreateElementResult, nsresult> insertNewContainerElementResult = + InsertNodeWithTransaction<Element>( + *newContainer, referenceNode ? EditorDOMPoint(referenceNode) + : EditorDOMPoint::AtEndOf(*parentNode)); + NS_WARNING_ASSERTION(insertNewContainerElementResult.isOk(), + "EditorBase::InsertNodeWithTransaction() failed"); + MOZ_ASSERT_IF( + insertNewContainerElementResult.isOk(), + insertNewContainerElementResult.inspect().GetNewNode() == newContainer); + return insertNewContainerElementResult; +} + +Result<EditorDOMPoint, nsresult> HTMLEditor::RemoveContainerWithTransaction( + Element& aElement) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (NS_WARN_IF(!HTMLEditUtils::IsRemovableNode(aElement)) || + NS_WARN_IF(!HTMLEditUtils::IsSimplyEditableNode(aElement))) { + return Err(NS_ERROR_FAILURE); + } + + // Notify our internal selection state listener. + AutoRemoveContainerSelNotify selNotify(RangeUpdaterRef(), + EditorRawDOMPoint(&aElement)); + + AutoTArray<OwningNonNull<nsIContent>, 32> arrayOfChildren; + HTMLEditUtils::CollectChildren( + aElement, arrayOfChildren, 0u, + // Move non-editable children too because its container, aElement, is + // editable so that all children must be removable node. + {}); + const OwningNonNull<nsINode> parentNode = *aElement.GetParentNode(); + nsCOMPtr<nsIContent> previousChild = aElement.GetPreviousSibling(); + // For making all MoveNodeTransactions have a referenc node in the current + // parent, move nodes from last one to preceding ones. + for (const OwningNonNull<nsIContent>& child : Reversed(arrayOfChildren)) { + if (MOZ_UNLIKELY(!HTMLEditUtils::IsRemovableNode(child))) { + continue; + } + Result<MoveNodeResult, nsresult> moveChildResult = MoveNodeWithTransaction( + MOZ_KnownLive(child), // due to bug 1622253. + previousChild ? EditorDOMPoint::After(previousChild) + : EditorDOMPoint(parentNode, 0u)); + if (MOZ_UNLIKELY(moveChildResult.isErr())) { + NS_WARNING("HTMLEditor::MoveNodeWithTransaction() failed"); + return moveChildResult.propagateErr(); + } + // If the reference node was moved to different container, try to recover + // the original position. + if (previousChild && + MOZ_UNLIKELY(previousChild->GetParentNode() != parentNode)) { + if (MOZ_UNLIKELY(child->GetParentNode() != parentNode)) { + NS_WARNING( + "Neither the reference (previous) sibling nor the moved child was " + "in the expected parent node"); + moveChildResult.inspect().IgnoreCaretPointSuggestion(); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + previousChild = child->GetPreviousSibling(); + } + // We'll need to put caret at next sibling of aElement if nobody moves + // content nodes under the parent node except us. + moveChildResult.inspect().IgnoreCaretPointSuggestion(); + } + + if (aElement.GetParentNode() && aElement.GetParentNode() != parentNode) { + NS_WARNING( + "The removing element has already been moved to another element"); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + NS_WARNING_ASSERTION(!aElement.GetFirstChild(), + "The removing container still has some children, but " + "they are removed by removing the container"); + + auto GetNextSiblingOf = + [](const nsTArray<OwningNonNull<nsIContent>>& aArrayOfMovedContent, + const nsINode& aExpectedParentNode) -> nsIContent* { + for (const OwningNonNull<nsIContent>& movedChild : + Reversed(aArrayOfMovedContent)) { + if (movedChild != &aExpectedParentNode) { + continue; // Ignore moved node which was moved to different place + } + return movedChild->GetNextSibling(); + } + // XXX If all nodes were moved by web apps, we cannot suggest "collect" + // position without computing the index of aElement. However, I + // don't think that it's necessary for the web apps in the wild. + return nullptr; + }; + + nsCOMPtr<nsIContent> nextSibling = + aElement.GetParentNode() ? aElement.GetNextSibling() + : GetNextSiblingOf(arrayOfChildren, *parentNode); + + nsresult rv = DeleteNodeWithTransaction(aElement); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::DeleteNodeTransaction() failed"); + return Err(rv); + } + + if (nextSibling && nextSibling->GetParentNode() != parentNode) { + nextSibling = GetNextSiblingOf(arrayOfChildren, *parentNode); + } + return nextSibling ? EditorDOMPoint(nextSibling) + : EditorDOMPoint::AtEndOf(*parentNode); +} + +MOZ_CAN_RUN_SCRIPT_BOUNDARY void HTMLEditor::ContentAppended( + nsIContent* aFirstNewContent) { + DoContentInserted(aFirstNewContent, ContentNodeIs::Appended); +} + +MOZ_CAN_RUN_SCRIPT_BOUNDARY void HTMLEditor::ContentInserted( + nsIContent* aChild) { + DoContentInserted(aChild, ContentNodeIs::Inserted); +} + +bool HTMLEditor::IsInObservedSubtree(nsIContent* aChild) { + if (!aChild) { + return false; + } + + // FIXME(emilio, bug 1596856): This should probably work if the root is in the + // same shadow tree as the child, probably? I don't know what the + // contenteditable-in-shadow-dom situation is. + if (Element* root = GetRoot()) { + // To be super safe here, check both ChromeOnlyAccess and NAC / Shadow DOM. + // That catches (also unbound) native anonymous content and ShadowDOM. + if (root->ChromeOnlyAccess() != aChild->ChromeOnlyAccess() || + root->IsInNativeAnonymousSubtree() != + aChild->IsInNativeAnonymousSubtree() || + root->IsInShadowTree() != aChild->IsInShadowTree()) { + return false; + } + } + + return !aChild->ChromeOnlyAccess() && !aChild->IsInShadowTree() && + !aChild->IsInNativeAnonymousSubtree(); +} + +void HTMLEditor::DoContentInserted(nsIContent* aChild, + ContentNodeIs aContentNodeIs) { + MOZ_ASSERT(aChild); + nsINode* container = aChild->GetParentNode(); + MOZ_ASSERT(container); + + if (!IsInObservedSubtree(aChild)) { + return; + } + + // XXX Why do we need this? This method is a helper of mutation observer. + // So, the callers of mutation observer should guarantee that this won't + // be deleted at least during the call. + RefPtr<HTMLEditor> kungFuDeathGrip(this); + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return; + } + + if (ShouldReplaceRootElement()) { + UpdateRootElement(); + if (mPendingRootElementUpdatedRunner) { + return; + } + mPendingRootElementUpdatedRunner = NewRunnableMethod( + "HTMLEditor::NotifyRootChanged", this, &HTMLEditor::NotifyRootChanged); + nsContentUtils::AddScriptRunner( + do_AddRef(mPendingRootElementUpdatedRunner)); + return; + } + + // We don't need to handle our own modifications + if (!GetTopLevelEditSubAction() && container->IsEditable()) { + if (EditorUtils::IsPaddingBRElementForEmptyEditor(*aChild)) { + // Ignore insertion of the padding <br> element. + return; + } + nsresult rv = OnDocumentModified(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::OnDocumentModified() failed, but ignored"); + + // Update spellcheck for only the newly-inserted node (bug 743819) + if (mInlineSpellChecker) { + nsIContent* endContent = aChild; + if (aContentNodeIs == ContentNodeIs::Appended) { + nsIContent* child = nullptr; + for (child = aChild; child; child = child->GetNextSibling()) { + if (child->InclusiveDescendantMayNeedSpellchecking(this)) { + break; + } + } + if (!child) { + // No child needed spellchecking, return. + return; + } + + // Maybe more than 1 child was appended. + endContent = container->GetLastChild(); + } else if (!aChild->InclusiveDescendantMayNeedSpellchecking(this)) { + return; + } + + RefPtr<nsRange> range = nsRange::Create(aChild); + range->SelectNodesInContainer(container, aChild, endContent); + DebugOnly<nsresult> rvIgnored = + mInlineSpellChecker->SpellCheckRange(range); + NS_WARNING_ASSERTION( + rvIgnored == NS_ERROR_NOT_INITIALIZED || NS_SUCCEEDED(rvIgnored), + "mozInlineSpellChecker::SpellCheckRange() failed, but ignored"); + } + } +} + +MOZ_CAN_RUN_SCRIPT_BOUNDARY void HTMLEditor::ContentRemoved( + nsIContent* aChild, nsIContent* aPreviousSibling) { + if (!IsInObservedSubtree(aChild)) { + return; + } + + // XXX Why do we need to do this? This method is a mutation observer's + // method. Therefore, the caller should guarantee that this won't be + // deleted during the call. + RefPtr<HTMLEditor> kungFuDeathGrip(this); + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return; + } + + if (SameCOMIdentity(aChild, mRootElement)) { + mRootElement = nullptr; + if (mPendingRootElementUpdatedRunner) { + return; + } + mPendingRootElementUpdatedRunner = NewRunnableMethod( + "HTMLEditor::NotifyRootChanged", this, &HTMLEditor::NotifyRootChanged); + nsContentUtils::AddScriptRunner( + do_AddRef(mPendingRootElementUpdatedRunner)); + return; + } + + // We don't need to handle our own modifications + if (!GetTopLevelEditSubAction() && aChild->GetParentNode()->IsEditable()) { + if (aChild && EditorUtils::IsPaddingBRElementForEmptyEditor(*aChild)) { + // Ignore removal of the padding <br> element for empty editor. + return; + } + + nsresult rv = OnDocumentModified(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::OnDocumentModified() failed, but ignored"); + } +} + +MOZ_CAN_RUN_SCRIPT_BOUNDARY void HTMLEditor::CharacterDataChanged( + nsIContent* aContent, const CharacterDataChangeInfo& aInfo) { + if (!mInlineSpellChecker || !aContent->IsEditable() || + !IsInObservedSubtree(aContent) || + GetTopLevelEditSubAction() != EditSubAction::eNone) { + return; + } + + nsIContent* parent = aContent->GetParent(); + if (!parent || !parent->InclusiveDescendantMayNeedSpellchecking(this)) { + return; + } + + RefPtr<nsRange> range = nsRange::Create(aContent); + range->SelectNodesInContainer(parent, aContent, aContent); + DebugOnly<nsresult> rvIgnored = mInlineSpellChecker->SpellCheckRange(range); +} + +nsresult HTMLEditor::SelectEntireDocument() { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (!mInitSucceeded) { + return NS_ERROR_NOT_INITIALIZED; + } + + // XXX It's odd to select all of the document body if an contenteditable + // element has focus. + RefPtr<Element> bodyOrDocumentElement = GetRoot(); + if (NS_WARN_IF(!bodyOrDocumentElement)) { + return NS_ERROR_NOT_INITIALIZED; + } + + // If we're empty, don't select all children because that would select the + // padding <br> element for empty editor. + if (IsEmpty()) { + nsresult rv = CollapseSelectionToStartOf(*bodyOrDocumentElement); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionToStartOf() failed"); + return rv; + } + + // Otherwise, select all children. + ErrorResult error; + SelectionRef().SelectAllChildren(*bodyOrDocumentElement, error); + if (NS_WARN_IF(Destroyed())) { + error.SuppressException(); + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION(!error.Failed(), + "Selection::SelectAllChildren() failed"); + return error.StealNSResult(); +} + +nsresult HTMLEditor::SelectAllInternal() { + MOZ_ASSERT(IsEditActionDataAvailable()); + + CommitComposition(); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + + auto GetBodyElementIfElementIsParentOfHTMLBody = + [](const Element& aElement) -> Element* { + if (!aElement.OwnerDoc()->IsHTMLDocument()) { + return const_cast<Element*>(&aElement); + } + HTMLBodyElement* bodyElement = aElement.OwnerDoc()->GetBodyElement(); + return bodyElement && nsContentUtils::ContentIsFlattenedTreeDescendantOf( + bodyElement, &aElement) + ? bodyElement + : const_cast<Element*>(&aElement); + }; + + nsCOMPtr<nsIContent> selectionRootContent = + [&]() MOZ_CAN_RUN_SCRIPT -> nsIContent* { + RefPtr<Element> elementToBeSelected = [&]() -> Element* { + // If there is at least one selection range, we should compute the + // selection root from the anchor node. + if (SelectionRef().RangeCount()) { + if (nsIContent* content = + nsIContent::FromNodeOrNull(SelectionRef().GetAnchorNode())) { + if (content->IsElement()) { + return content->AsElement(); + } + if (Element* parentElement = + content->GetParentElementCrossingShadowRoot()) { + return parentElement; + } + } + } + // If no element contains a selection range, we should select all children + // of the focused element at least. + if (Element* focusedElement = GetFocusedElement()) { + return focusedElement; + } + // of the body or document element. + Element* bodyOrDocumentElement = GetRoot(); + NS_WARNING_ASSERTION(bodyOrDocumentElement, + "There was no element in the document"); + return bodyOrDocumentElement; + }(); + + // If the element to be selected is <input type="text"> or <textarea>, + // GetSelectionRootContent() returns its anonymous <div> element, but we + // want to select all of the document or selection limiter. Therefore, + // we should use its parent to compute the selection root. + if (elementToBeSelected->HasIndependentSelection()) { + Element* parentElement = elementToBeSelected->GetParentElement(); + if (MOZ_LIKELY(parentElement)) { + elementToBeSelected = parentElement; + } + } + + // Then, compute the selection root content to select all including + // elementToBeSelected. + RefPtr<PresShell> presShell = GetPresShell(); + nsIContent* computedSelectionRootContent = + elementToBeSelected->GetSelectionRootContent(presShell); + if (NS_WARN_IF(!computedSelectionRootContent)) { + return nullptr; + } + if (MOZ_UNLIKELY(!computedSelectionRootContent->IsElement())) { + return computedSelectionRootContent; + } + return GetBodyElementIfElementIsParentOfHTMLBody( + *computedSelectionRootContent->AsElement()); + }(); + if (NS_WARN_IF(!selectionRootContent)) { + return NS_ERROR_FAILURE; + } + + Maybe<Selection::AutoUserInitiated> userSelection; + // XXX Do we need to mark it as "user initiated" for + // `Document.execCommand("selectAll")`? + if (!selectionRootContent->IsEditable()) { + userSelection.emplace(SelectionRef()); + } + ErrorResult error; + SelectionRef().SelectAllChildren(*selectionRootContent, error); + NS_WARNING_ASSERTION(!error.Failed(), + "Selection::SelectAllChildren() failed"); + return error.StealNSResult(); +} + +bool HTMLEditor::SetCaretInTableCell(Element* aElement) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (!aElement || !aElement->IsHTMLElement() || + !HTMLEditUtils::IsAnyTableElement(aElement)) { + return false; + } + const RefPtr<Element> editingHost = ComputeEditingHost(); + if (!editingHost || !aElement->IsInclusiveDescendantOf(editingHost)) { + return false; + } + + nsCOMPtr<nsIContent> deepestFirstChild = aElement; + while (deepestFirstChild->HasChildren()) { + deepestFirstChild = deepestFirstChild->GetFirstChild(); + } + + // Set selection at beginning of the found node + nsresult rv = CollapseSelectionToStartOf(*deepestFirstChild); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionToStartOf() failed"); + return NS_SUCCEEDED(rv); +} + +/** + * This method scans the selection for adjacent text nodes + * and collapses them into a single text node. + * "adjacent" means literally adjacent siblings of the same parent. + * Uses HTMLEditor::JoinNodesWithTransaction() so action is undoable. + * Should be called within the context of a batch transaction. + */ +nsresult HTMLEditor::CollapseAdjacentTextNodes(nsRange& aInRange) { + AutoTransactionsConserveSelection dontChangeMySelection(*this); + + // we can't actually do anything during iteration, so store the text nodes in + // an array first. + DOMSubtreeIterator subtreeIter; + if (NS_FAILED(subtreeIter.Init(aInRange))) { + NS_WARNING("DOMSubtreeIterator::Init() failed"); + return NS_ERROR_FAILURE; + } + AutoTArray<OwningNonNull<Text>, 8> textNodes; + subtreeIter.AppendNodesToArray( + +[](nsINode& aNode, void*) -> bool { + return EditorUtils::IsEditableContent(*aNode.AsText(), + EditorType::HTML); + }, + textNodes); + + // now that I have a list of text nodes, collapse adjacent text nodes + while (textNodes.Length() > 1u) { + OwningNonNull<Text>& leftTextNode = textNodes[0u]; + OwningNonNull<Text>& rightTextNode = textNodes[1u]; + + // If the text nodes are not direct siblings, we shouldn't join them, and + // we don't need to handle the left one anymore. + if (rightTextNode->GetPreviousSibling() != leftTextNode) { + textNodes.RemoveElementAt(0u); + continue; + } + + Result<JoinNodesResult, nsresult> joinNodesResult = + JoinNodesWithTransaction(MOZ_KnownLive(*leftTextNode), + MOZ_KnownLive(*rightTextNode)); + if (MOZ_UNLIKELY(joinNodesResult.isErr())) { + NS_WARNING("HTMLEditor::JoinNodesWithTransaction() failed"); + return joinNodesResult.unwrapErr(); + } + if (MOZ_LIKELY(joinNodesResult.inspect().RemovedContent() == + leftTextNode)) { + textNodes.RemoveElementAt(0u); + } else if (MOZ_LIKELY(joinNodesResult.inspect().RemovedContent() == + rightTextNode)) { + textNodes.RemoveElementAt(1u); + } else { + MOZ_ASSERT_UNREACHABLE( + "HTMLEditor::JoinNodesWithTransaction() removed unexpected node"); + return NS_ERROR_UNEXPECTED; + } + } + + return NS_OK; +} + +nsresult HTMLEditor::SetSelectionAtDocumentStart() { + MOZ_ASSERT(IsEditActionDataAvailable()); + + RefPtr<Element> rootElement = GetRoot(); + if (NS_WARN_IF(!rootElement)) { + return NS_ERROR_FAILURE; + } + + nsresult rv = CollapseSelectionToStartOf(*rootElement); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionToStartOf() failed"); + return rv; +} + +/** + * Remove aNode, reparenting any children into the parent of aNode. In + * addition, insert any br's needed to preserve identity of removed block. + */ +Result<EditorDOMPoint, nsresult> +HTMLEditor::RemoveBlockContainerWithTransaction(Element& aElement) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // Two possibilities: the container could be empty of editable content. If + // that is the case, we need to compare what is before and after aNode to + // determine if we need a br. + // + // Or it could be not empty, in which case we have to compare previous + // sibling and first child to determine if we need a leading br, and compare + // following sibling and last child to determine if we need a trailing br. + + EditorDOMPoint pointToPutCaret; + if (nsCOMPtr<nsIContent> child = HTMLEditUtils::GetFirstChild( + aElement, {WalkTreeOption::IgnoreNonEditableNode})) { + // The case of aNode not being empty. We need a br at start unless: + // 1) previous sibling of aNode is a block, OR + // 2) previous sibling of aNode is a br, OR + // 3) first child of aNode is a block OR + // 4) either is null + + if (nsIContent* previousSibling = HTMLEditUtils::GetPreviousSibling( + aElement, {WalkTreeOption::IgnoreNonEditableNode})) { + if (!HTMLEditUtils::IsBlockElement(*previousSibling) && + !previousSibling->IsHTMLElement(nsGkAtoms::br) && + !HTMLEditUtils::IsBlockElement(*child)) { + // Insert br node + Result<CreateElementResult, nsresult> insertBRElementResult = + InsertBRElement(WithTransaction::Yes, + EditorDOMPoint(&aElement, 0u)); + if (MOZ_UNLIKELY(insertBRElementResult.isErr())) { + NS_WARNING( + "HTMLEditor::InsertBRElement(WithTransaction::Yes) failed"); + return insertBRElementResult.propagateErr(); + } + CreateElementResult unwrappedInsertBRElementResult = + insertBRElementResult.unwrap(); + unwrappedInsertBRElementResult.MoveCaretPointTo( + pointToPutCaret, {SuggestCaret::OnlyIfHasSuggestion}); + MOZ_ASSERT(unwrappedInsertBRElementResult.GetNewNode()); + } + } + + // We need a br at end unless: + // 1) following sibling of aNode is a block, OR + // 2) last child of aNode is a block, OR + // 3) last child of aNode is a br OR + // 4) either is null + + if (nsIContent* nextSibling = HTMLEditUtils::GetNextSibling( + aElement, {WalkTreeOption::IgnoreNonEditableNode})) { + if (nextSibling && !HTMLEditUtils::IsBlockElement(*nextSibling)) { + if (nsIContent* lastChild = HTMLEditUtils::GetLastChild( + aElement, {WalkTreeOption::IgnoreNonEditableNode})) { + if (!HTMLEditUtils::IsBlockElement(*lastChild) && + !lastChild->IsHTMLElement(nsGkAtoms::br)) { + Result<CreateElementResult, nsresult> insertBRElementResult = + InsertBRElement(WithTransaction::Yes, + EditorDOMPoint::AtEndOf(aElement)); + if (MOZ_UNLIKELY(insertBRElementResult.isErr())) { + NS_WARNING( + "HTMLEditor::InsertBRElement(WithTransaction::Yes) failed"); + return insertBRElementResult.propagateErr(); + } + CreateElementResult unwrappedInsertBRElementResult = + insertBRElementResult.unwrap(); + unwrappedInsertBRElementResult.MoveCaretPointTo( + pointToPutCaret, {SuggestCaret::OnlyIfHasSuggestion}); + MOZ_ASSERT(unwrappedInsertBRElementResult.GetNewNode()); + } + } + } + } + } else if (nsIContent* previousSibling = HTMLEditUtils::GetPreviousSibling( + aElement, {WalkTreeOption::IgnoreNonEditableNode})) { + // The case of aNode being empty. We need a br at start unless: + // 1) previous sibling of aNode is a block, OR + // 2) previous sibling of aNode is a br, OR + // 3) following sibling of aNode is a block, OR + // 4) following sibling of aNode is a br OR + // 5) either is null + if (!HTMLEditUtils::IsBlockElement(*previousSibling) && + !previousSibling->IsHTMLElement(nsGkAtoms::br)) { + if (nsIContent* nextSibling = HTMLEditUtils::GetNextSibling( + aElement, {WalkTreeOption::IgnoreNonEditableNode})) { + if (!HTMLEditUtils::IsBlockElement(*nextSibling) && + !nextSibling->IsHTMLElement(nsGkAtoms::br)) { + Result<CreateElementResult, nsresult> insertBRElementResult = + InsertBRElement(WithTransaction::Yes, + EditorDOMPoint(&aElement, 0u)); + if (MOZ_UNLIKELY(insertBRElementResult.isErr())) { + NS_WARNING( + "HTMLEditor::InsertBRElement(WithTransaction::Yes) failed"); + return insertBRElementResult.propagateErr(); + } + CreateElementResult unwrappedInsertBRElementResult = + insertBRElementResult.unwrap(); + unwrappedInsertBRElementResult.MoveCaretPointTo( + pointToPutCaret, {SuggestCaret::OnlyIfHasSuggestion}); + MOZ_ASSERT(unwrappedInsertBRElementResult.GetNewNode()); + } + } + } + } + + // Now remove container + Result<EditorDOMPoint, nsresult> unwrapBlockElementResult = + RemoveContainerWithTransaction(aElement); + if (MOZ_UNLIKELY(unwrapBlockElementResult.isErr())) { + NS_WARNING("HTMLEditor::RemoveContainerWithTransaction() failed"); + return unwrapBlockElementResult; + } + if (AllowsTransactionsToChangeSelection() && + unwrapBlockElementResult.inspect().IsSet()) { + pointToPutCaret = unwrapBlockElementResult.unwrap(); + } + return pointToPutCaret; // May be unset +} + +Result<SplitNodeResult, nsresult> HTMLEditor::SplitNodeWithTransaction( + const EditorDOMPoint& aStartOfRightNode) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (NS_WARN_IF(!aStartOfRightNode.IsInContentNode())) { + return Err(NS_ERROR_INVALID_ARG); + } + MOZ_ASSERT(aStartOfRightNode.IsSetAndValid()); + + if (NS_WARN_IF(!HTMLEditUtils::IsSplittableNode( + *aStartOfRightNode.ContainerAs<nsIContent>()))) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eSplitNode, nsIEditor::eNext, ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION( + !ignoredError.Failed(), + "OnStartToHandleTopLevelEditSubAction() failed, but ignored"); + + mMaybeHasJoinSplitTransactions = true; + RefPtr<SplitNodeTransaction> transaction = + SplitNodeTransaction::Create(*this, aStartOfRightNode); + nsresult rv = DoTransactionInternal(transaction); + if (NS_WARN_IF(Destroyed())) { + NS_WARNING( + "EditorBase::DoTransactionInternal() caused destroying the editor"); + return Err(NS_ERROR_EDITOR_DESTROYED); + } + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DoTransactionInternal() failed"); + return Err(rv); + } + + nsIContent* newContent = transaction->GetNewContent(); + nsIContent* splitContent = transaction->GetSplitContent(); + if (NS_WARN_IF(!newContent) || NS_WARN_IF(!splitContent)) { + return Err(NS_ERROR_FAILURE); + } + TopLevelEditSubActionDataRef().DidSplitContent( + *this, *splitContent, *newContent, transaction->GetSplitNodeDirection()); + if (NS_WARN_IF(!newContent->IsInComposedDoc()) || + NS_WARN_IF(!splitContent->IsInComposedDoc())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + return SplitNodeResult(*newContent, *splitContent, + transaction->GetSplitNodeDirection()); +} + +Result<SplitNodeResult, nsresult> HTMLEditor::SplitNodeDeepWithTransaction( + nsIContent& aMostAncestorToSplit, + const EditorDOMPoint& aDeepestStartOfRightNode, + SplitAtEdges aSplitAtEdges) { + MOZ_ASSERT(aDeepestStartOfRightNode.IsSetAndValidInComposedDoc()); + MOZ_ASSERT( + aDeepestStartOfRightNode.GetContainer() == &aMostAncestorToSplit || + EditorUtils::IsDescendantOf(*aDeepestStartOfRightNode.GetContainer(), + aMostAncestorToSplit)); + + if (NS_WARN_IF(!aDeepestStartOfRightNode.IsInComposedDoc())) { + return Err(NS_ERROR_INVALID_ARG); + } + + nsCOMPtr<nsIContent> newLeftNodeOfMostAncestor; + EditorDOMPoint atStartOfRightNode(aDeepestStartOfRightNode); + // lastResult is as explained by its name, the last result which may not be + // split a node actually. + SplitNodeResult lastResult = + SplitNodeResult::NotHandled(atStartOfRightNode, GetSplitNodeDirection()); + MOZ_ASSERT(lastResult.AtSplitPoint<EditorRawDOMPoint>() + .IsSetAndValidInComposedDoc()); + + while (true) { + // Need to insert rules code call here to do things like not split a list + // if you are after the last <li> or before the first, etc. For now we + // just have some smarts about unnecessarily splitting text nodes, which + // should be universal enough to put straight in this EditorBase routine. + auto* splittingContent = atStartOfRightNode.GetContainerAs<nsIContent>(); + if (NS_WARN_IF(!splittingContent)) { + lastResult.IgnoreCaretPointSuggestion(); + return Err(NS_ERROR_FAILURE); + } + // If we meet an orphan node before meeting aMostAncestorToSplit, we need + // to stop splitting. This is a bug of the caller. + if (NS_WARN_IF(splittingContent != &aMostAncestorToSplit && + !atStartOfRightNode.GetContainerParentAs<nsIContent>())) { + lastResult.IgnoreCaretPointSuggestion(); + return Err(NS_ERROR_FAILURE); + } + // If the container is not splitable node such as comment node, atomic + // element, etc, we should keep it as-is, and try to split its parents. + if (!HTMLEditUtils::IsSplittableNode(*splittingContent)) { + if (splittingContent == &aMostAncestorToSplit) { + return lastResult; + } + atStartOfRightNode.Set(splittingContent); + continue; + } + + // If the split point is middle of the node or the node is not a text node + // and we're allowed to create empty element node, split it. + if ((aSplitAtEdges == SplitAtEdges::eAllowToCreateEmptyContainer && + !atStartOfRightNode.IsInTextNode()) || + (!atStartOfRightNode.IsStartOfContainer() && + !atStartOfRightNode.IsEndOfContainer())) { + Result<SplitNodeResult, nsresult> splitNodeResult = + SplitNodeWithTransaction(atStartOfRightNode); + if (MOZ_UNLIKELY(splitNodeResult.isErr())) { + lastResult.IgnoreCaretPointSuggestion(); + return splitNodeResult; + } + lastResult = SplitNodeResult::MergeWithDeeperSplitNodeResult( + splitNodeResult.unwrap(), lastResult); + if (NS_WARN_IF(!lastResult.AtSplitPoint<EditorRawDOMPoint>() + .IsInComposedDoc())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + MOZ_ASSERT(lastResult.HasCaretPointSuggestion()); + MOZ_ASSERT(lastResult.GetOriginalContent() == splittingContent); + if (splittingContent == &aMostAncestorToSplit) { + // Actually, we split aMostAncestorToSplit. + return lastResult; + } + + // Then, try to split its parent before current node. + atStartOfRightNode = lastResult.AtNextContent<EditorDOMPoint>(); + } + // If the split point is end of the node and it is a text node or we're not + // allowed to create empty container node, try to split its parent after it. + else if (!atStartOfRightNode.IsStartOfContainer()) { + lastResult = SplitNodeResult::HandledButDidNotSplitDueToEndOfContainer( + *splittingContent, GetSplitNodeDirection(), &lastResult); + MOZ_ASSERT(lastResult.AtSplitPoint<EditorRawDOMPoint>() + .IsSetAndValidInComposedDoc()); + if (splittingContent == &aMostAncestorToSplit) { + return lastResult; + } + + // Try to split its parent after current node. + atStartOfRightNode.SetAfter(splittingContent); + } + // If the split point is start of the node and it is a text node or we're + // not allowed to create empty container node, try to split its parent. + else { + if (splittingContent == &aMostAncestorToSplit) { + return SplitNodeResult::HandledButDidNotSplitDueToStartOfContainer( + *splittingContent, GetSplitNodeDirection(), &lastResult); + } + + // Try to split its parent before current node. + // XXX This is logically wrong. If we've already split something but + // this is the last splitable content node in the limiter, this + // method will return "not handled". + lastResult = SplitNodeResult::NotHandled( + atStartOfRightNode, GetSplitNodeDirection(), &lastResult); + MOZ_ASSERT(lastResult.AtSplitPoint<EditorRawDOMPoint>() + .IsSetAndValidInComposedDoc()); + atStartOfRightNode.Set(splittingContent); + MOZ_ASSERT(atStartOfRightNode.IsSetAndValidInComposedDoc()); + } + } + + // Not reached because while (true) loop never breaks. +} + +Result<SplitNodeResult, nsresult> HTMLEditor::DoSplitNode( + const EditorDOMPoint& aStartOfRightNode, nsIContent& aNewNode, + SplitNodeDirection aDirection) { + // Ensure computing the offset if it's intialized with a child content node. + Unused << aStartOfRightNode.Offset(); + + // XXX Perhaps, aStartOfRightNode may be invalid if this is a redo + // operation after modifying DOM node with JS. + if (NS_WARN_IF(!aStartOfRightNode.IsInContentNode())) { + return Err(NS_ERROR_INVALID_ARG); + } + MOZ_ASSERT(aStartOfRightNode.IsSetAndValid()); + + // Remember all selection points. + AutoTArray<SavedRange, 10> savedRanges; + for (SelectionType selectionType : kPresentSelectionTypes) { + SavedRange savingRange; + savingRange.mSelection = GetSelection(selectionType); + if (NS_WARN_IF(!savingRange.mSelection && + selectionType == SelectionType::eNormal)) { + return Err(NS_ERROR_FAILURE); + } + if (!savingRange.mSelection) { + // For non-normal selections, skip over the non-existing ones. + continue; + } + + for (uint32_t j : IntegerRange(savingRange.mSelection->RangeCount())) { + const nsRange* r = savingRange.mSelection->GetRangeAt(j); + MOZ_ASSERT(r); + MOZ_ASSERT(r->IsPositioned()); + // XXX Looks like that SavedRange should have mStart and mEnd which + // are RangeBoundary. Then, we can avoid to compute offset here. + savingRange.mStartContainer = r->GetStartContainer(); + savingRange.mStartOffset = r->StartOffset(); + savingRange.mEndContainer = r->GetEndContainer(); + savingRange.mEndOffset = r->EndOffset(); + + savedRanges.AppendElement(savingRange); + } + } + + nsCOMPtr<nsINode> parent = aStartOfRightNode.GetContainerParent(); + if (NS_WARN_IF(!parent)) { + return Err(NS_ERROR_FAILURE); + } + + // Fix the child before mutation observer may touch the DOM tree. + nsIContent* firstChildOfRightNode = aStartOfRightNode.GetChild(); + IgnoredErrorResult error; + parent->InsertBefore(aNewNode, + aDirection == SplitNodeDirection::LeftNodeIsNewOne + ? aStartOfRightNode.GetContainer() + : aStartOfRightNode.GetContainer()->GetNextSibling(), + error); + if (MOZ_UNLIKELY(error.Failed())) { + NS_WARNING("nsINode::InsertBefore() failed"); + return Err(error.StealNSResult()); + } + + MOZ_DIAGNOSTIC_ASSERT_IF(aStartOfRightNode.IsInTextNode(), aNewNode.IsText()); + MOZ_DIAGNOSTIC_ASSERT_IF(!aStartOfRightNode.IsInTextNode(), + !aNewNode.IsText()); + + // If we are splitting a text node, we need to move its some data to the + // new text node. + if (aStartOfRightNode.IsInTextNode()) { + if (!(aDirection == SplitNodeDirection::LeftNodeIsNewOne && + aStartOfRightNode.IsStartOfContainer()) && + !(aDirection == SplitNodeDirection::RightNodeIsNewOne && + aStartOfRightNode.IsEndOfContainer())) { + Text* originalTextNode = aStartOfRightNode.ContainerAs<Text>(); + Text* newTextNode = aNewNode.AsText(); + nsAutoString movingText; + const uint32_t cutStartOffset = + aDirection == SplitNodeDirection::LeftNodeIsNewOne + ? 0u + : aStartOfRightNode.Offset(); + const uint32_t cutLength = + aDirection == SplitNodeDirection::LeftNodeIsNewOne + ? aStartOfRightNode.Offset() + : originalTextNode->Length() - aStartOfRightNode.Offset(); + IgnoredErrorResult error; + originalTextNode->SubstringData(cutStartOffset, cutLength, movingText, + error); + NS_WARNING_ASSERTION(!error.Failed(), + "Text::SubstringData() failed, but ignored"); + error.SuppressException(); + + // XXX This call may destroy us. + DoDeleteText(MOZ_KnownLive(*originalTextNode), cutStartOffset, cutLength, + error); + NS_WARNING_ASSERTION(!error.Failed(), + "EditorBase::DoDeleteText() failed, but ignored"); + error.SuppressException(); + + // XXX This call may destroy us. + DoSetText(MOZ_KnownLive(*newTextNode), movingText, error); + NS_WARNING_ASSERTION(!error.Failed(), + "EditorBase::DoSetText() failed, but ignored"); + } + } + // If the node has been moved to different parent, we should do nothing + // since web apps should handle eventhing in such case. + else if (firstChildOfRightNode && + aStartOfRightNode.GetContainer() != + firstChildOfRightNode->GetParentNode()) { + NS_WARNING( + "The web app interupped us and touched the DOM tree, we stopped " + "splitting anything"); + } else if (aDirection == SplitNodeDirection::LeftNodeIsNewOne) { + // If Splitting at end of container which is not a text node, we need to + // move all children if the left node is new one. Otherwise, nothing to do. + if (!firstChildOfRightNode) { + // XXX Why do we ignore an error while moving nodes from the right + // node to the left node? + IgnoredErrorResult error; + MoveAllChildren(*aStartOfRightNode.GetContainer(), + EditorRawDOMPoint(&aNewNode, 0u), error); + NS_WARNING_ASSERTION(!error.Failed(), + "HTMLEditor::MoveAllChildren() failed, but ignored"); + } + // If the left node is new one and splitting middle of it, we need to + // previous siblings of the given point to the new left node. + else if (firstChildOfRightNode->GetPreviousSibling()) { + // XXX Why do we ignore an error while moving nodes from the right node + // to the left node? + IgnoredErrorResult error; + MovePreviousSiblings(*firstChildOfRightNode, + EditorRawDOMPoint(&aNewNode, 0u), error); + NS_WARNING_ASSERTION( + !error.Failed(), + "HTMLEditor::MovePreviousSiblings() failed, but ignored"); + } + } else { + MOZ_ASSERT(aDirection == SplitNodeDirection::RightNodeIsNewOne); + // If the right node is new one and there is no children or splitting at + // end of the node, we need to do nothing. + if (!firstChildOfRightNode) { + // Do nothing. + } + // If the right node is new one and splitting at start of the container, + // we need to move all children to the new right node. + else if (!firstChildOfRightNode->GetPreviousSibling()) { + // XXX Why do we ignore an error while moving nodes from the right + // node to the left node? + IgnoredErrorResult error; + MoveAllChildren(*aStartOfRightNode.GetContainer(), + EditorRawDOMPoint(&aNewNode, 0u), error); + NS_WARNING_ASSERTION(!error.Failed(), + "HTMLEditor::MoveAllChildren() failed, but ignored"); + } + // If the right node is new one and splitting at middle of the node, we need + // to move inclusive next siblings of the split point to the new right node. + else { + // XXX Why do we ignore an error while moving nodes from the right node + // to the left node? + IgnoredErrorResult error; + MoveInclusiveNextSiblings(*firstChildOfRightNode, + EditorRawDOMPoint(&aNewNode, 0u), error); + NS_WARNING_ASSERTION( + !error.Failed(), + "HTMLEditor::MoveInclusiveNextSiblings() failed, but ignored"); + } + } + + // Handle selection + // TODO: Stop doing this, this shouldn't be necessary to update selection. + if (RefPtr<PresShell> presShell = GetPresShell()) { + presShell->FlushPendingNotifications(FlushType::Frames); + } + NS_WARNING_ASSERTION(!Destroyed(), + "The editor is destroyed during splitting a node"); + + const bool allowedTransactionsToChangeSelection = + AllowsTransactionsToChangeSelection(); + + RefPtr<Selection> previousSelection; + for (SavedRange& savedRange : savedRanges) { + // If we have not seen the selection yet, clear all of its ranges. + if (savedRange.mSelection != previousSelection) { + MOZ_KnownLive(savedRange.mSelection)->RemoveAllRanges(error); + if (MOZ_UNLIKELY(error.Failed())) { + NS_WARNING("Selection::RemoveAllRanges() failed"); + return Err(error.StealNSResult()); + } + previousSelection = savedRange.mSelection; + } + + // XXX Looks like that we don't need to modify normal selection here + // because selection will be modified by the caller if + // AllowsTransactionsToChangeSelection() will return true. + if (allowedTransactionsToChangeSelection && + savedRange.mSelection->Type() == SelectionType::eNormal) { + // If the editor should adjust the selection, don't bother restoring + // the ranges for the normal selection here. + continue; + } + + auto AdjustDOMPoint = [&](nsCOMPtr<nsINode>& aContainer, + uint32_t& aOffset) { + if (aContainer != aStartOfRightNode.GetContainer()) { + return; + } + + if (aDirection == SplitNodeDirection::LeftNodeIsNewOne) { + // If the container is the right node and offset is before the split + // point, the content was moved into aNewNode. So, just changing the + // container will point proper position. + if (aOffset < aStartOfRightNode.Offset()) { + aContainer = &aNewNode; + return; + } + + // If the container is the right node and offset equals or is larger + // than the split point, we need to decrease the offset since some + // content before the split point was moved to aNewNode. + if (aOffset >= aStartOfRightNode.Offset()) { + aOffset -= aStartOfRightNode.Offset(); + return; + } + + NS_WARNING("The stored offset was smaller than the right node offset"); + aOffset = 0u; + return; + } + + // If the container is the left node and offset is after the split + // point, the content was moved from the right node to aNewNode. + // So, we need to change the container to aNewNode and decrease the + // offset. + if (aOffset >= aStartOfRightNode.Offset()) { + aContainer = &aNewNode; + aOffset -= aStartOfRightNode.Offset(); + } + }; + AdjustDOMPoint(savedRange.mStartContainer, savedRange.mStartOffset); + AdjustDOMPoint(savedRange.mEndContainer, savedRange.mEndOffset); + + RefPtr<nsRange> newRange = + nsRange::Create(savedRange.mStartContainer, savedRange.mStartOffset, + savedRange.mEndContainer, savedRange.mEndOffset, error); + if (MOZ_UNLIKELY(error.Failed())) { + NS_WARNING("nsRange::Create() failed"); + return Err(error.StealNSResult()); + } + // The `MOZ_KnownLive` annotation is only necessary because of a bug + // (https://bugzilla.mozilla.org/show_bug.cgi?id=1622253) in the + // static analyzer. + MOZ_KnownLive(savedRange.mSelection) + ->AddRangeAndSelectFramesAndNotifyListeners(*newRange, error); + if (MOZ_UNLIKELY(error.Failed())) { + NS_WARNING( + "Selection::AddRangeAndSelectFramesAndNotifyListeners() failed"); + return Err(error.StealNSResult()); + } + } + + // We don't need to set selection here because the caller should do that + // in any case. + + // If splitting the node causes running mutation event listener and we've + // got unexpected result, we should return error because callers will + // continue to do their work without complicated DOM tree result. + // NOTE: Perhaps, we shouldn't do this immediately after each DOM tree change + // because stopping handling it causes some data loss. E.g., user + // may loose the text which is moved to the new text node. + // XXX We cannot check all descendants in the right node and the new left + // node for performance reason. I think that if caller needs to access + // some of the descendants, they should check by themselves. + if (NS_WARN_IF(parent != aStartOfRightNode.GetContainer()->GetParentNode()) || + NS_WARN_IF(parent != aNewNode.GetParentNode()) || + (aDirection == SplitNodeDirection::LeftNodeIsNewOne && + NS_WARN_IF(aNewNode.GetNextSibling() != + aStartOfRightNode.GetContainer())) || + (aDirection == SplitNodeDirection::RightNodeIsNewOne && + NS_WARN_IF(aNewNode.GetPreviousSibling() != + aStartOfRightNode.GetContainer()))) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + DebugOnly<nsresult> rvIgnored = RangeUpdaterRef().SelAdjSplitNode( + *aStartOfRightNode.ContainerAs<nsIContent>(), aStartOfRightNode.Offset(), + aNewNode, aDirection); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "RangeUpdater::SelAdjSplitNode() failed, but ignored"); + + return SplitNodeResult(aNewNode, *aStartOfRightNode.ContainerAs<nsIContent>(), + aDirection); +} + +Result<JoinNodesResult, nsresult> HTMLEditor::JoinNodesWithTransaction( + nsIContent& aLeftContent, nsIContent& aRightContent) { + MOZ_ASSERT(IsEditActionDataAvailable()); + MOZ_ASSERT(&aLeftContent != &aRightContent); + MOZ_ASSERT(aLeftContent.GetParentNode()); + MOZ_ASSERT(aRightContent.GetParentNode()); + MOZ_ASSERT(aLeftContent.GetParentNode() == aRightContent.GetParentNode()); + + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eJoinNodes, nsIEditor::ePrevious, ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return Err(ignoredError.StealNSResult()); + } + NS_WARNING_ASSERTION( + !ignoredError.Failed(), + "HTMLEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored"); + + if (NS_WARN_IF(!aRightContent.GetParentNode())) { + return Err(NS_ERROR_FAILURE); + } + + RefPtr<JoinNodesTransaction> transaction = + JoinNodesTransaction::MaybeCreate(*this, aLeftContent, aRightContent); + if (MOZ_UNLIKELY(!transaction)) { + NS_WARNING("JoinNodesTransaction::MaybeCreate() failed"); + return Err(NS_ERROR_FAILURE); + } + + mMaybeHasJoinSplitTransactions = true; + const nsresult rv = DoTransactionInternal(transaction); + // FYI: Now, DidJoinNodesTransaction() must have been run if succeeded. + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + + // This shouldn't occur unless the cycle collector runs by chrome script + // forcibly. + if (NS_WARN_IF(!transaction->GetRemovedContent()) || + NS_WARN_IF(!transaction->GetExistingContent())) { + return Err(NS_ERROR_UNEXPECTED); + } + + // If joined node is moved to different place, offset may not have any + // meaning. In this case, the web app modified the DOM tree takes on the + // responsibility for the remaning things. + if (NS_WARN_IF(transaction->GetExistingContent()->GetParent() != + transaction->GetParentNode())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DoTransactionInternal() failed"); + return Err(rv); + } + + return JoinNodesResult(transaction->CreateJoinedPoint<EditorDOMPoint>(), + *transaction->GetRemovedContent(), + transaction->GetJoinNodesDirection()); +} + +void HTMLEditor::DidJoinNodesTransaction( + const JoinNodesTransaction& aTransaction, nsresult aDoJoinNodesResult) { + // This shouldn't occur unless the cycle collector runs by chrome script + // forcibly. + if (MOZ_UNLIKELY(NS_WARN_IF(!aTransaction.GetRemovedContent()) || + NS_WARN_IF(!aTransaction.GetExistingContent()))) { + return; + } + + // If joined node is moved to different place, offset may not have any + // meaning. In this case, the web app modified the DOM tree takes on the + // responsibility for the remaning things. + if (MOZ_UNLIKELY(aTransaction.GetExistingContent()->GetParentNode() != + aTransaction.GetParentNode())) { + return; + } + + // Be aware, the joined point should be created for each call because + // they may refer the child node, but some of them may change the DOM tree + // after that, thus we need to avoid invalid point (Although it shouldn't + // occur). + TopLevelEditSubActionDataRef().DidJoinContents( + *this, aTransaction.CreateJoinedPoint<EditorRawDOMPoint>()); + + if (NS_SUCCEEDED(aDoJoinNodesResult)) { + if (RefPtr<TextServicesDocument> textServicesDocument = + mTextServicesDocument) { + textServicesDocument->DidJoinContents( + aTransaction.CreateJoinedPoint<EditorRawDOMPoint>(), + *aTransaction.GetRemovedContent(), + aTransaction.GetJoinNodesDirection()); + } + } + + if (!mActionListeners.IsEmpty()) { + for (auto& listener : mActionListeners.Clone()) { + DebugOnly<nsresult> rvIgnored = listener->DidJoinContents( + aTransaction.CreateJoinedPoint<EditorRawDOMPoint>(), + aTransaction.GetRemovedContent(), + aTransaction.GetJoinNodesDirection() == + JoinNodesDirection::LeftNodeIntoRightNode); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "nsIEditActionListener::DidJoinContents() failed, but ignored"); + } + } +} + +nsresult HTMLEditor::DoJoinNodes(nsIContent& aContentToKeep, + nsIContent& aContentToRemove, + JoinNodesDirection aDirection) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + const uint32_t keepingContentLength = aContentToKeep.Length(); + const uint32_t removingContentLength = aContentToRemove.Length(); + const EditorDOMPoint oldPointAtRightContent( + aDirection == JoinNodesDirection::LeftNodeIntoRightNode + ? &aContentToKeep + : &aContentToRemove); + if (MOZ_LIKELY(oldPointAtRightContent.IsSet())) { + Unused << oldPointAtRightContent.Offset(); // Fix the offset + } + + // Remember all selection points. + // XXX Do we need to restore all types of selections by ourselves? Normal + // selection should be modified later as result of handling edit action. + // IME selections shouldn't be there when nodes are joined. Spellcheck + // selections should be recreated with newer text. URL selections + // shouldn't be there because of used only by the URL bar. + AutoTArray<SavedRange, 10> savedRanges; + { + EditorRawDOMPoint atRemovingNode(&aContentToRemove); + EditorRawDOMPoint atNodeToKeep(&aContentToKeep); + for (SelectionType selectionType : kPresentSelectionTypes) { + SavedRange savingRange; + savingRange.mSelection = GetSelection(selectionType); + if (selectionType == SelectionType::eNormal) { + if (NS_WARN_IF(!savingRange.mSelection)) { + return NS_ERROR_FAILURE; + } + } else if (!savingRange.mSelection) { + // For non-normal selections, skip over the non-existing ones. + continue; + } + + const uint32_t rangeCount = savingRange.mSelection->RangeCount(); + for (const uint32_t j : IntegerRange(rangeCount)) { + MOZ_ASSERT(savingRange.mSelection->RangeCount() == rangeCount); + const RefPtr<nsRange> r = savingRange.mSelection->GetRangeAt(j); + MOZ_ASSERT(r); + MOZ_ASSERT(r->IsPositioned()); + savingRange.mStartContainer = r->GetStartContainer(); + savingRange.mStartOffset = r->StartOffset(); + savingRange.mEndContainer = r->GetEndContainer(); + savingRange.mEndOffset = r->EndOffset(); + + // If selection endpoint is between the nodes, remember it as being + // in the one that is going away instead. This simplifies later + // selection adjustment logic at end of this method. + if (savingRange.mStartContainer) { + MOZ_ASSERT(savingRange.mEndContainer); + auto AdjustDOMPoint = [&](nsCOMPtr<nsINode>& aContainer, + uint32_t& aOffset) { + if (aDirection == JoinNodesDirection::LeftNodeIntoRightNode) { + // If range boundary points aContentToKeep and aContentToRemove + // is its left node, remember it as being at end of the removing + // node. Then, only chaning the container to aContentToKeep will + // point start of the current first content of aContentToKeep. + if (aContainer == atRemovingNode.GetContainer() && + atRemovingNode.Offset() < aOffset && + aOffset <= atNodeToKeep.Offset()) { + aContainer = &aContentToRemove; + aOffset = removingContentLength; + } + return; + } + // If range boundary points aContentToRemove and aContentToKeep is + // its left node, remember it as being at end of aContentToKeep. + // Then, it will point start of the first content of moved content + // from aContentToRemove. + if (aContainer == atRemovingNode.GetContainer() && + atNodeToKeep.Offset() < aOffset && + aOffset <= atRemovingNode.Offset()) { + aContainer = &aContentToKeep; + aOffset = keepingContentLength; + } + }; + AdjustDOMPoint(savingRange.mStartContainer, savingRange.mStartOffset); + AdjustDOMPoint(savingRange.mEndContainer, savingRange.mEndOffset); + } + + savedRanges.AppendElement(savingRange); + } + } + } + + // OK, ready to do join now. + nsresult rv = [&]() MOZ_CAN_RUN_SCRIPT { + // If it's a text node, just shuffle around some text. + if (aContentToKeep.IsText() && aContentToRemove.IsText()) { + nsAutoString rightText; + nsAutoString leftText; + const nsIContent& rightTextNode = + aDirection == JoinNodesDirection::LeftNodeIntoRightNode + ? aContentToKeep + : aContentToRemove; + const nsIContent& leftTextNode = + aDirection == JoinNodesDirection::LeftNodeIntoRightNode + ? aContentToRemove + : aContentToKeep; + rightTextNode.AsText()->GetData(rightText); + leftTextNode.AsText()->GetData(leftText); + leftText += rightText; + IgnoredErrorResult ignoredError; + DoSetText(MOZ_KnownLive(*aContentToKeep.AsText()), leftText, + ignoredError); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION(!ignoredError.Failed(), + "EditorBase::DoSetText() failed, but ignored"); + return NS_OK; + } + // Otherwise it's an interior node, so shuffle around the children. + AutoTArray<OwningNonNull<nsIContent>, 64> arrayOfChildContents; + HTMLEditUtils::CollectAllChildren(aContentToRemove, arrayOfChildContents); + + if (aDirection == JoinNodesDirection::LeftNodeIntoRightNode) { + for (const OwningNonNull<nsIContent>& child : + Reversed(arrayOfChildContents)) { + // Note that it's safe to pass the reference node to insert the child + // without making it grabbed by nsINode::mNextSibling before touching + // the DOM tree. + IgnoredErrorResult error; + aContentToKeep.InsertBefore(child, aContentToKeep.GetFirstChild(), + error); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + if (error.Failed()) { + NS_WARNING("nsINode::InsertBefore() failed"); + return error.StealNSResult(); + } + } + } else { + for (const OwningNonNull<nsIContent>& child : arrayOfChildContents) { + IgnoredErrorResult error; + aContentToKeep.AppendChild(child, error); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + if (error.Failed()) { + NS_WARNING("nsINode::AppendChild() failed"); + return error.StealNSResult(); + } + } + } + return NS_OK; + }(); + + // Delete the extra node. + if (NS_SUCCEEDED(rv)) { + aContentToRemove.Remove(); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + } + + if (MOZ_LIKELY(oldPointAtRightContent.IsSet())) { + DebugOnly<nsresult> rvIgnored = RangeUpdaterRef().SelAdjJoinNodes( + EditorRawDOMPoint( + &aContentToKeep, + aDirection == JoinNodesDirection::LeftNodeIntoRightNode + ? std::min(removingContentLength, aContentToKeep.Length()) + : std::min(keepingContentLength, aContentToKeep.Length())), + aContentToRemove, oldPointAtRightContent, aDirection); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "RangeUpdater::SelAdjJoinNodes() failed, but ignored"); + } + if (MOZ_UNLIKELY(NS_FAILED(rv))) { + return rv; + } + + const bool allowedTransactionsToChangeSelection = + AllowsTransactionsToChangeSelection(); + + // And adjust the selection if needed. + RefPtr<Selection> previousSelection; + for (SavedRange& savedRange : savedRanges) { + // If we have not seen the selection yet, clear all of its ranges. + if (savedRange.mSelection != previousSelection) { + IgnoredErrorResult error; + MOZ_KnownLive(savedRange.mSelection)->RemoveAllRanges(error); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + if (error.Failed()) { + NS_WARNING("Selection::RemoveAllRanges() failed"); + return error.StealNSResult(); + } + previousSelection = savedRange.mSelection; + } + + if (allowedTransactionsToChangeSelection && + savedRange.mSelection->Type() == SelectionType::eNormal) { + // If the editor should adjust the selection, don't bother restoring + // the ranges for the normal selection here. + continue; + } + + auto AdjustDOMPoint = [&](nsCOMPtr<nsINode>& aContainer, + uint32_t& aOffset) { + if (aDirection == JoinNodesDirection::LeftNodeIntoRightNode) { + // Now, all content of aContentToRemove are moved to start of + // aContentToKeep. Therefore, if a range boundary was in + // aContentToRemove, we just need to change the container to + // aContentToKeep. + if (aContainer == &aContentToRemove) { + aContainer = &aContentToKeep; + return; + } + // And also if the range boundary was in aContentToKeep, we need to + // adjust the offset because the content in aContentToRemove was + // instarted before ex-start content of aContentToKeep. + if (aContainer == &aContentToKeep) { + aOffset += removingContentLength; + } + return; + } + // Now, all content of aContentToRemove are moved to end of + // aContentToKeep. Therefore, if a range boundary was in + // aContentToRemove, we need to change the container to aContentToKeep and + // adjust the offset to after the original content of aContentToKeep. + if (aContainer == &aContentToRemove) { + aContainer = &aContentToKeep; + aOffset += keepingContentLength; + } + }; + AdjustDOMPoint(savedRange.mStartContainer, savedRange.mStartOffset); + AdjustDOMPoint(savedRange.mEndContainer, savedRange.mEndOffset); + + const RefPtr<nsRange> newRange = nsRange::Create( + savedRange.mStartContainer, savedRange.mStartOffset, + savedRange.mEndContainer, savedRange.mEndOffset, IgnoreErrors()); + if (!newRange) { + NS_WARNING("nsRange::Create() failed"); + return NS_ERROR_FAILURE; + } + + IgnoredErrorResult error; + // The `MOZ_KnownLive` annotation is only necessary because of a bug + // (https://bugzilla.mozilla.org/show_bug.cgi?id=1622253) in the + // static analyzer. + MOZ_KnownLive(savedRange.mSelection) + ->AddRangeAndSelectFramesAndNotifyListeners(*newRange, error); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + if (NS_WARN_IF(error.Failed())) { + return error.StealNSResult(); + } + } + + if (allowedTransactionsToChangeSelection) { + // Editor wants us to set selection at join point. + DebugOnly<nsresult> rvIgnored = CollapseSelectionTo( + aDirection == JoinNodesDirection::LeftNodeIntoRightNode + ? EditorRawDOMPoint(&aContentToKeep, removingContentLength) + : EditorRawDOMPoint(&aContentToKeep, 0u)); + if (MOZ_UNLIKELY(rv == NS_ERROR_EDITOR_DESTROYED)) { + NS_WARNING( + "EditorBase::CollapseSelectionTo() caused destroying the editor"); + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBases::CollapseSelectionTos() failed, but ignored"); + } + + return NS_OK; +} + +Result<MoveNodeResult, nsresult> HTMLEditor::MoveNodeWithTransaction( + nsIContent& aContentToMove, const EditorDOMPoint& aPointToInsert) { + MOZ_ASSERT(aPointToInsert.IsSetAndValid()); + + EditorDOMPoint oldPoint(&aContentToMove); + if (NS_WARN_IF(!oldPoint.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + + // Don't do anything if it's already in right place. + if (aPointToInsert == oldPoint) { + return MoveNodeResult::IgnoredResult(aPointToInsert.NextPoint()); + } + + RefPtr<MoveNodeTransaction> moveNodeTransaction = + MoveNodeTransaction::MaybeCreate(*this, aContentToMove, aPointToInsert); + if (MOZ_UNLIKELY(!moveNodeTransaction)) { + NS_WARNING("MoveNodeTransaction::MaybeCreate() failed"); + return Err(NS_ERROR_FAILURE); + } + + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eMoveNode, nsIEditor::eNext, ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return Err(ignoredError.StealNSResult()); + } + NS_WARNING_ASSERTION( + !ignoredError.Failed(), + "TextEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored"); + + TopLevelEditSubActionDataRef().WillDeleteContent(*this, aContentToMove); + + nsresult rv = DoTransactionInternal(moveNodeTransaction); + if (NS_SUCCEEDED(rv)) { + if (mTextServicesDocument) { + const OwningNonNull<TextServicesDocument> textServicesDocument = + *mTextServicesDocument; + textServicesDocument->DidDeleteContent(aContentToMove); + } + } + + if (!mActionListeners.IsEmpty()) { + for (auto& listener : mActionListeners.Clone()) { + DebugOnly<nsresult> rvIgnored = + listener->DidDeleteNode(&aContentToMove, rv); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "nsIEditActionListener::DidDeleteNode() failed, but ignored"); + } + } + + if (MOZ_UNLIKELY(Destroyed())) { + NS_WARNING( + "MoveNodeTransaction::DoTransaction() caused destroying the editor"); + return Err(NS_ERROR_EDITOR_DESTROYED); + } + + if (NS_FAILED(rv)) { + NS_WARNING("MoveNodeTransaction::DoTransaction() failed"); + return Err(rv); + } + + TopLevelEditSubActionDataRef().DidInsertContent(*this, aContentToMove); + + return MoveNodeResult::HandledResult( + moveNodeTransaction->SuggestNextInsertionPoint<EditorDOMPoint>(), + moveNodeTransaction->SuggestPointToPutCaret<EditorDOMPoint>()); +} + +Result<RefPtr<Element>, nsresult> HTMLEditor::DeleteSelectionAndCreateElement( + nsAtom& aTag, const InitializeInsertingElement& aInitializer) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + nsresult rv = DeleteSelectionAndPrepareToCreateNode(); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::DeleteSelectionAndPrepareToCreateNode() failed"); + return Err(rv); + } + + EditorDOMPoint pointToInsert(SelectionRef().AnchorRef()); + if (!pointToInsert.IsSet()) { + return Err(NS_ERROR_FAILURE); + } + Result<CreateElementResult, nsresult> createNewElementResult = + CreateAndInsertElement(WithTransaction::Yes, aTag, pointToInsert, + aInitializer); + if (MOZ_UNLIKELY(createNewElementResult.isErr())) { + NS_WARNING( + "HTMLEditor::CreateAndInsertElement(WithTransaction::Yes) failed"); + return createNewElementResult.propagateErr(); + } + MOZ_ASSERT(createNewElementResult.inspect().GetNewNode()); + + // We want the selection to be just after the new node + createNewElementResult.inspect().IgnoreCaretPointSuggestion(); + rv = CollapseSelectionTo( + EditorRawDOMPoint::After(*createNewElementResult.inspect().GetNewNode())); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return Err(rv); + } + return createNewElementResult.unwrap().UnwrapNewNode(); +} + +nsresult HTMLEditor::DeleteSelectionAndPrepareToCreateNode() { + MOZ_ASSERT(IsEditActionDataAvailable()); + + if (NS_WARN_IF(!SelectionRef().GetAnchorFocusRange())) { + return NS_OK; + } + + if (!SelectionRef().GetAnchorFocusRange()->Collapsed()) { + nsresult rv = + DeleteSelectionAsSubAction(nsIEditor::eNone, nsIEditor::eStrip); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteSelectionAsSubAction() failed"); + return rv; + } + MOZ_ASSERT(SelectionRef().GetAnchorFocusRange() && + SelectionRef().GetAnchorFocusRange()->Collapsed(), + "Selection not collapsed after delete"); + } + + // If the selection is a chardata node, split it if necessary and compute + // where to put the new node + EditorDOMPoint atAnchor(SelectionRef().AnchorRef()); + if (NS_WARN_IF(!atAnchor.IsSet()) || !atAnchor.IsInDataNode()) { + return NS_OK; + } + + if (NS_WARN_IF(!atAnchor.GetContainerParent())) { + return NS_ERROR_FAILURE; + } + + if (atAnchor.IsStartOfContainer()) { + const EditorRawDOMPoint atAnchorContainer(atAnchor.GetContainer()); + if (NS_WARN_IF(!atAnchorContainer.IsSetAndValid())) { + return NS_ERROR_FAILURE; + } + nsresult rv = CollapseSelectionTo(atAnchorContainer); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; + } + + if (atAnchor.IsEndOfContainer()) { + EditorRawDOMPoint afterAnchorContainer(atAnchor.GetContainer()); + if (NS_WARN_IF(!afterAnchorContainer.AdvanceOffset())) { + return NS_ERROR_FAILURE; + } + nsresult rv = CollapseSelectionTo(afterAnchorContainer); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; + } + + Result<SplitNodeResult, nsresult> splitAtAnchorResult = + SplitNodeWithTransaction(atAnchor); + if (MOZ_UNLIKELY(splitAtAnchorResult.isErr())) { + NS_WARNING("HTMLEditor::SplitNodeWithTransaction() failed"); + return splitAtAnchorResult.unwrapErr(); + } + + splitAtAnchorResult.inspect().IgnoreCaretPointSuggestion(); + const auto atRightContent = + splitAtAnchorResult.inspect().AtNextContent<EditorRawDOMPoint>(); + if (NS_WARN_IF(!atRightContent.IsSet())) { + return NS_ERROR_FAILURE; + } + MOZ_ASSERT(atRightContent.IsSetAndValid()); + nsresult rv = CollapseSelectionTo(atRightContent); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; +} + +bool HTMLEditor::IsEmpty() const { + if (mPaddingBRElementForEmptyEditor) { + return true; + } + + // XXX Oddly, we check body or document element's state instead of + // active editing host. Must be a bug. + Element* bodyOrDocumentElement = GetRoot(); + if (!bodyOrDocumentElement) { + return true; + } + + for (nsIContent* childContent = bodyOrDocumentElement->GetFirstChild(); + childContent; childContent = childContent->GetNextSibling()) { + if (!childContent->IsText() || childContent->Length()) { + return false; + } + } + return true; +} + +// add to aElement the CSS inline styles corresponding to the HTML attribute +// aAttribute with its value aValue +nsresult HTMLEditor::SetAttributeOrEquivalent(Element* aElement, + nsAtom* aAttribute, + const nsAString& aValue, + bool aSuppressTransaction) { + MOZ_ASSERT(aElement); + MOZ_ASSERT(aAttribute); + + nsAutoScriptBlocker scriptBlocker; + nsStyledElement* styledElement = nsStyledElement::FromNodeOrNull(aElement); + if (!IsCSSEnabled()) { + // we are not in an HTML+CSS editor; let's set the attribute the HTML way + if (EditorElementStyle::IsHTMLStyle(aAttribute)) { + const EditorElementStyle elementStyle = + EditorElementStyle::Create(*aAttribute); + if (styledElement && elementStyle.IsCSSRemovable(*styledElement)) { + // MOZ_KnownLive(*styledElement): It's aElement and its lifetime must + // be guaranteed by the caller because of MOZ_CAN_RUN_SCRIPT method. + nsresult rv = CSSEditUtils::RemoveCSSEquivalentToStyle( + aSuppressTransaction ? WithTransaction::No : WithTransaction::Yes, + *this, MOZ_KnownLive(*styledElement), elementStyle, nullptr); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "CSSEditUtils::RemoveCSSEquivalentToStyle() failed, but ignored"); + } + } + if (aSuppressTransaction) { + nsresult rv = + aElement->SetAttr(kNameSpaceID_None, aAttribute, aValue, true); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "Element::SetAttr() failed"); + return rv; + } + nsresult rv = SetAttributeWithTransaction(*aElement, *aAttribute, aValue); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::SetAttributeWithTransaction() failed"); + return rv; + } + + if (EditorElementStyle::IsHTMLStyle(aAttribute)) { + const EditorElementStyle elementStyle = + EditorElementStyle::Create(*aAttribute); + if (styledElement && elementStyle.IsCSSSettable(*styledElement)) { + // MOZ_KnownLive(*styledElement): It's aElement and its lifetime must + // be guaranteed by the caller because of MOZ_CAN_RUN_SCRIPT method. + Result<size_t, nsresult> count = CSSEditUtils::SetCSSEquivalentToStyle( + aSuppressTransaction ? WithTransaction::No : WithTransaction::Yes, + *this, MOZ_KnownLive(*styledElement), elementStyle, &aValue); + if (MOZ_UNLIKELY(count.isErr())) { + if (NS_WARN_IF(count.inspectErr() == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING( + "CSSEditUtils::SetCSSEquivalentToStyle() failed, but ignored"); + } + if (count.inspect()) { + // we found an equivalence ; let's remove the HTML attribute itself if + // it is set + nsAutoString existingValue; + if (!aElement->GetAttr(kNameSpaceID_None, aAttribute, existingValue)) { + return NS_OK; + } + + if (aSuppressTransaction) { + nsresult rv = + aElement->UnsetAttr(kNameSpaceID_None, aAttribute, true); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "Element::UnsetAttr() failed"); + return rv; + } + nsresult rv = RemoveAttributeWithTransaction(*aElement, *aAttribute); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::RemoveAttributeWithTransaction() failed"); + return rv; + } + } + } + + // count is an integer that represents the number of CSS declarations + // applied to the element. If it is zero, we found no equivalence in this + // implementation for the attribute + if (aAttribute == nsGkAtoms::style) { + // if it is the style attribute, just add the new value to the existing + // style attribute's value + nsString existingValue; // Use nsString to avoid copying the string + // buffer at setting the attribute below. + aElement->GetAttr(kNameSpaceID_None, nsGkAtoms::style, existingValue); + if (!existingValue.IsEmpty()) { + existingValue.Append(HTMLEditUtils::kSpace); + } + existingValue.Append(aValue); + if (aSuppressTransaction) { + nsresult rv = aElement->SetAttr(kNameSpaceID_None, nsGkAtoms::style, + existingValue, true); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "Element::SetAttr(nsGkAtoms::style) failed"); + return rv; + } + nsresult rv = SetAttributeWithTransaction(*aElement, *nsGkAtoms::style, + existingValue); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::SetAttributeWithTransaction(nsGkAtoms::style) failed"); + return rv; + } + + // we have no CSS equivalence for this attribute and it is not the style + // attribute; let's set it the good'n'old HTML way + if (aSuppressTransaction) { + nsresult rv = + aElement->SetAttr(kNameSpaceID_None, aAttribute, aValue, true); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "Element::SetAttr() failed"); + return rv; + } + nsresult rv = SetAttributeWithTransaction(*aElement, *aAttribute, aValue); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::SetAttributeWithTransaction() failed"); + return rv; +} + +nsresult HTMLEditor::RemoveAttributeOrEquivalent(Element* aElement, + nsAtom* aAttribute, + bool aSuppressTransaction) { + MOZ_ASSERT(aElement); + MOZ_ASSERT(aAttribute); + + if (IsCSSEnabled() && EditorElementStyle::IsHTMLStyle(aAttribute)) { + const EditorElementStyle elementStyle = + EditorElementStyle::Create(*aAttribute); + if (elementStyle.IsCSSRemovable(*aElement)) { + // XXX It might be keep handling attribute even if aElement is not + // an nsStyledElement instance. + nsStyledElement* styledElement = + nsStyledElement::FromNodeOrNull(aElement); + if (NS_WARN_IF(!styledElement)) { + return NS_ERROR_INVALID_ARG; + } + // MOZ_KnownLive(*styledElement): It's aElement and its lifetime must + // be guaranteed by the caller because of MOZ_CAN_RUN_SCRIPT method. + nsresult rv = CSSEditUtils::RemoveCSSEquivalentToStyle( + aSuppressTransaction ? WithTransaction::No : WithTransaction::Yes, + *this, MOZ_KnownLive(*styledElement), elementStyle, nullptr); + if (NS_FAILED(rv)) { + NS_WARNING("CSSEditUtils::RemoveCSSEquivalentToStyle() failed"); + return rv; + } + } + } + + if (!aElement->HasAttr(kNameSpaceID_None, aAttribute)) { + return NS_OK; + } + + if (aSuppressTransaction) { + nsresult rv = aElement->UnsetAttr(kNameSpaceID_None, aAttribute, + /* aNotify = */ true); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "Element::UnsetAttr() failed"); + return rv; + } + nsresult rv = RemoveAttributeWithTransaction(*aElement, *aAttribute); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::RemoveAttributeWithTransaction() failed"); + return rv; +} + +NS_IMETHODIMP HTMLEditor::SetIsCSSEnabled(bool aIsCSSPrefChecked) { + AutoEditActionDataSetter editActionData(*this, + EditAction::eEnableOrDisableCSS); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_INITIALIZED; + } + + mIsCSSPrefChecked = aIsCSSPrefChecked; + return NS_OK; +} + +// Set the block background color +nsresult HTMLEditor::SetBlockBackgroundColorWithCSSAsSubAction( + const nsAString& aColor) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // background-color change and committing composition should be undone + // together + AutoPlaceholderBatch treatAsOneTransaction( + *this, ScrollSelectionIntoView::Yes, __FUNCTION__); + + CommitComposition(); + + // XXX Shouldn't we do this before calling `CommitComposition()`? + if (IsInPlaintextMode()) { + return NS_OK; + } + + { + Result<EditActionResult, nsresult> result = CanHandleHTMLEditSubAction(); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("HTMLEditor::CanHandleHTMLEditSubAction() failed"); + return result.unwrapErr(); + } + if (result.inspect().Canceled()) { + return NS_OK; + } + } + + IgnoredErrorResult ignoredError; + AutoEditSubActionNotifier startToHandleEditSubAction( + *this, EditSubAction::eInsertElement, nsIEditor::eNext, ignoredError); + if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) { + return ignoredError.StealNSResult(); + } + NS_WARNING_ASSERTION(!ignoredError.Failed(), + "HTMLEditor::OnStartToHandleTopLevelEditSubAction() " + "failed, but ignored"); + + // TODO: We don't need AutoTransactionsConserveSelection here in the normal + // cases, but removing this may cause the behavior with the legacy + // mutation event listeners. We should try to delete this in a bug. + AutoTransactionsConserveSelection dontChangeMySelection(*this); + + AutoRangeArray selectionRanges(SelectionRef()); + MOZ_ALWAYS_TRUE(selectionRanges.SaveAndTrackRanges(*this)); + for (const OwningNonNull<nsRange>& domRange : selectionRanges.Ranges()) { + EditorDOMRange range(domRange); + if (NS_WARN_IF(!range.IsPositioned())) { + continue; + } + + if (range.InSameContainer()) { + // If the range is in a text node, set background color of its parent + // block. + if (range.StartRef().IsInTextNode()) { + const RefPtr<nsStyledElement> editableBlockStyledElement = + nsStyledElement::FromNodeOrNull(HTMLEditUtils::GetAncestorElement( + *range.StartRef().ContainerAs<Text>(), + HTMLEditUtils::ClosestEditableBlockElement)); + if (!editableBlockStyledElement || + !EditorElementStyle::BGColor().IsCSSSettable( + *editableBlockStyledElement)) { + continue; + } + Result<size_t, nsresult> result = CSSEditUtils::SetCSSEquivalentToStyle( + WithTransaction::Yes, *this, *editableBlockStyledElement, + EditorElementStyle::BGColor(), &aColor); + if (MOZ_UNLIKELY(result.isErr())) { + if (NS_WARN_IF(result.inspectErr() == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING( + "CSSEditUtils::SetCSSEquivalentToStyle(EditorElementStyle::" + "BGColor()) failed, but ignored"); + } + continue; + } + + // If `Selection` is collapsed in a `<body>` element, set background + // color of the `<body>` element. + if (range.Collapsed() && + range.StartRef().IsContainerHTMLElement(nsGkAtoms::body)) { + const RefPtr<nsStyledElement> styledElement = + range.StartRef().GetContainerAs<nsStyledElement>(); + if (!styledElement || + !EditorElementStyle::BGColor().IsCSSSettable(*styledElement)) { + continue; + } + Result<size_t, nsresult> result = CSSEditUtils::SetCSSEquivalentToStyle( + WithTransaction::Yes, *this, *styledElement, + EditorElementStyle::BGColor(), &aColor); + if (MOZ_UNLIKELY(result.isErr())) { + if (NS_WARN_IF(result.inspectErr() == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING( + "CSSEditUtils::SetCSSEquivalentToStyle(EditorElementStyle::" + "BGColor()) failed, but ignored"); + } + continue; + } + + // If one node is selected, set background color of it if it's a + // block, or of its parent block otherwise. + if ((range.StartRef().IsStartOfContainer() && + range.EndRef().IsStartOfContainer()) || + range.StartRef().Offset() + 1 == range.EndRef().Offset()) { + if (NS_WARN_IF(range.StartRef().IsInDataNode())) { + continue; + } + const RefPtr<nsStyledElement> editableBlockStyledElement = + nsStyledElement::FromNodeOrNull( + HTMLEditUtils::GetInclusiveAncestorElement( + *range.StartRef().GetChild(), + HTMLEditUtils::ClosestEditableBlockElement)); + if (!editableBlockStyledElement || + !EditorElementStyle::BGColor().IsCSSSettable( + *editableBlockStyledElement)) { + continue; + } + Result<size_t, nsresult> result = CSSEditUtils::SetCSSEquivalentToStyle( + WithTransaction::Yes, *this, *editableBlockStyledElement, + EditorElementStyle::BGColor(), &aColor); + if (MOZ_UNLIKELY(result.isErr())) { + if (NS_WARN_IF(result.inspectErr() == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING( + "CSSEditUtils::SetCSSEquivalentToStyle(EditorElementStyle::" + "BGColor()) failed, but ignored"); + } + continue; + } + } // if (range.InSameContainer()) + + // Collect editable nodes which are entirely contained in the range. + AutoTArray<OwningNonNull<nsIContent>, 64> arrayOfContents; + { + ContentSubtreeIterator subtreeIter; + // If there is no node which is entirely in the range, + // `ContentSubtreeIterator::Init()` fails, but this is possible case, + // don't warn it. + nsresult rv = subtreeIter.Init(range.StartRef().ToRawRangeBoundary(), + range.EndRef().ToRawRangeBoundary()); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "ContentSubtreeIterator::Init() failed, but ignored"); + if (NS_SUCCEEDED(rv)) { + for (; !subtreeIter.IsDone(); subtreeIter.Next()) { + nsINode* node = subtreeIter.GetCurrentNode(); + if (NS_WARN_IF(!node)) { + return NS_ERROR_FAILURE; + } + if (node->IsContent() && EditorUtils::IsEditableContent( + *node->AsContent(), EditorType::HTML)) { + arrayOfContents.AppendElement(*node->AsContent()); + } + } + } + } + + // This caches block parent if we set its background color. + RefPtr<Element> handledBlockParent; + + // If start node is a text node, set background color of its parent + // block. + if (range.StartRef().IsInTextNode() && + EditorUtils::IsEditableContent(*range.StartRef().ContainerAs<Text>(), + EditorType::HTML)) { + Element* const editableBlockElement = HTMLEditUtils::GetAncestorElement( + *range.StartRef().ContainerAs<Text>(), + HTMLEditUtils::ClosestEditableBlockElement); + if (editableBlockElement && handledBlockParent != editableBlockElement) { + handledBlockParent = editableBlockElement; + nsStyledElement* const blockStyledElement = + nsStyledElement::FromNode(handledBlockParent); + if (blockStyledElement && + EditorElementStyle::BGColor().IsCSSSettable(*blockStyledElement)) { + // MOZ_KnownLive(*blockStyledElement): It's handledBlockParent + // whose type is RefPtr. + Result<size_t, nsresult> result = + CSSEditUtils::SetCSSEquivalentToStyle( + WithTransaction::Yes, *this, + MOZ_KnownLive(*blockStyledElement), + EditorElementStyle::BGColor(), &aColor); + if (MOZ_UNLIKELY(result.isErr())) { + if (NS_WARN_IF(result.inspectErr() == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING( + "CSSEditUtils::SetCSSEquivalentToStyle(EditorElementStyle::" + "BGColor()) failed, but ignored"); + } + } + } + } + + // Then, set background color of each block or block parent of all nodes + // in the range entirely. + for (OwningNonNull<nsIContent>& content : arrayOfContents) { + Element* const editableBlockElement = + HTMLEditUtils::GetInclusiveAncestorElement( + content, HTMLEditUtils::ClosestEditableBlockElement); + if (editableBlockElement && handledBlockParent != editableBlockElement) { + handledBlockParent = editableBlockElement; + nsStyledElement* const blockStyledElement = + nsStyledElement::FromNode(handledBlockParent); + if (blockStyledElement && + EditorElementStyle::BGColor().IsCSSSettable(*blockStyledElement)) { + // MOZ_KnownLive(*blockStyledElement): It's handledBlockParent whose + // type is RefPtr. + Result<size_t, nsresult> result = + CSSEditUtils::SetCSSEquivalentToStyle( + WithTransaction::Yes, *this, + MOZ_KnownLive(*blockStyledElement), + EditorElementStyle::BGColor(), &aColor); + if (MOZ_UNLIKELY(result.isErr())) { + if (NS_WARN_IF(result.inspectErr() == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING( + "CSSEditUtils::SetCSSEquivalentToStyle(EditorElementStyle::" + "BGColor()) failed, but ignored"); + } + } + } + } + + // Finally, if end node is a text node, set background color of its + // parent block. + if (range.EndRef().IsInTextNode() && + EditorUtils::IsEditableContent(*range.EndRef().ContainerAs<Text>(), + EditorType::HTML)) { + Element* const editableBlockElement = HTMLEditUtils::GetAncestorElement( + *range.EndRef().ContainerAs<Text>(), + HTMLEditUtils::ClosestEditableBlockElement); + if (editableBlockElement && handledBlockParent != editableBlockElement) { + const RefPtr<nsStyledElement> blockStyledElement = + nsStyledElement::FromNode(editableBlockElement); + if (blockStyledElement && + EditorElementStyle::BGColor().IsCSSSettable(*blockStyledElement)) { + Result<size_t, nsresult> result = + CSSEditUtils::SetCSSEquivalentToStyle( + WithTransaction::Yes, *this, *blockStyledElement, + EditorElementStyle::BGColor(), &aColor); + if (MOZ_UNLIKELY(result.isErr())) { + if (NS_WARN_IF(result.inspectErr() == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING( + "CSSEditUtils::SetCSSEquivalentToStyle(EditorElementStyle::" + "BGColor()) failed, but ignored"); + } + } + } + } + } // for-loop of selectionRanges + + MOZ_ASSERT(selectionRanges.HasSavedRanges()); + selectionRanges.RestoreFromSavedRanges(); + nsresult rv = selectionRanges.ApplyTo(SelectionRef()); + if (NS_WARN_IF(Destroyed())) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "AutoRangeArray::ApplyTo() failed"); + return rv; +} + +NS_IMETHODIMP HTMLEditor::SetBackgroundColor(const nsAString& aColor) { + nsresult rv = SetBackgroundColorAsAction(aColor); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::SetBackgroundColorAsAction() failed"); + return rv; +} + +nsresult HTMLEditor::SetBackgroundColorAsAction(const nsAString& aColor, + nsIPrincipal* aPrincipal) { + AutoEditActionDataSetter editActionData( + *this, EditAction::eSetBackgroundColor, aPrincipal); + nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent(); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED, + "CanHandleAndMaybeDispatchBeforeInputEvent(), failed"); + return EditorBase::ToGenericNSResult(rv); + } + + if (IsCSSEnabled()) { + // if we are in CSS mode, we have to apply the background color to the + // containing block (or the body if we have no block-level element in + // the document) + nsresult rv = SetBlockBackgroundColorWithCSSAsSubAction(aColor); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::SetBlockBackgroundColorWithCSSAsSubAction() failed"); + return EditorBase::ToGenericNSResult(rv); + } + + // but in HTML mode, we can only set the document's background color + rv = SetHTMLBackgroundColorWithTransaction(aColor); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "HTMLEditor::SetHTMLBackgroundColorWithTransaction() failed"); + return EditorBase::ToGenericNSResult(rv); +} + +Result<EditorDOMPoint, nsresult> +HTMLEditor::CopyLastEditableChildStylesWithTransaction( + Element& aPreviousBlock, Element& aNewBlock, const Element& aEditingHost) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // First, clear out aNewBlock. Contract is that we want only the styles + // from aPreviousBlock. + AutoTArray<OwningNonNull<nsIContent>, 32> newBlockChildren; + HTMLEditUtils::CollectAllChildren(aNewBlock, newBlockChildren); + for (const OwningNonNull<nsIContent>& child : newBlockChildren) { + // MOZ_KNownLive(child) because of bug 1622253 + nsresult rv = DeleteNodeWithTransaction(MOZ_KnownLive(child)); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return Err(rv); + } + } + if (MOZ_UNLIKELY(aNewBlock.GetFirstChild())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + // XXX aNewBlock may be moved or removed. Even in such case, we should + // keep cloning the styles? + + // Look for the deepest last editable leaf node in aPreviousBlock. + // Then, if found one is a <br> element, look for non-<br> element. + nsIContent* deepestEditableContent = nullptr; + for (nsCOMPtr<nsIContent> child = &aPreviousBlock; child; + child = HTMLEditUtils::GetLastChild( + *child, {WalkTreeOption::IgnoreNonEditableNode})) { + deepestEditableContent = child; + } + while (deepestEditableContent && + deepestEditableContent->IsHTMLElement(nsGkAtoms::br)) { + deepestEditableContent = HTMLEditUtils::GetPreviousContent( + *deepestEditableContent, {WalkTreeOption::IgnoreNonEditableNode}, + &aEditingHost); + } + if (!deepestEditableContent) { + return EditorDOMPoint(&aNewBlock, 0u); + } + + Element* deepestVisibleEditableElement = + deepestEditableContent->GetAsElementOrParentElement(); + if (!deepestVisibleEditableElement) { + return EditorDOMPoint(&aNewBlock, 0u); + } + + // Clone inline elements to keep current style in the new block. + // XXX Looks like that this is really slow if lastEditableDescendant is + // far from aPreviousBlock. Probably, we should clone inline containers + // from ancestor to descendants without transactions, then, insert it + // after that with transaction. + RefPtr<Element> lastClonedElement, firstClonedElement; + for (RefPtr<Element> elementInPreviousBlock = deepestVisibleEditableElement; + elementInPreviousBlock && elementInPreviousBlock != &aPreviousBlock; + elementInPreviousBlock = elementInPreviousBlock->GetParentElement()) { + if (!HTMLEditUtils::IsInlineStyle(elementInPreviousBlock) && + !elementInPreviousBlock->IsHTMLElement(nsGkAtoms::span)) { + continue; + } + OwningNonNull<nsAtom> tagName = + *elementInPreviousBlock->NodeInfo()->NameAtom(); + // At first time, just create the most descendant inline container + // element. + if (!firstClonedElement) { + Result<CreateElementResult, nsresult> createNewElementResult = + CreateAndInsertElement( + WithTransaction::Yes, tagName, EditorDOMPoint(&aNewBlock, 0u), + // MOZ_CAN_RUN_SCRIPT_BOUNDARY due to bug 1758868 + [&elementInPreviousBlock]( + HTMLEditor& aHTMLEditor, Element& aNewElement, + const EditorDOMPoint&) MOZ_CAN_RUN_SCRIPT_BOUNDARY { + // Clone all attributes. Note that despite the method name, + // CloneAttributesWithTransaction does not create + // transactions in this case because aNewElement has not + // been connected yet. + // XXX Looks like that this clones id attribute too. + aHTMLEditor.CloneAttributesWithTransaction( + aNewElement, *elementInPreviousBlock); + return NS_OK; + }); + if (MOZ_UNLIKELY(createNewElementResult.isErr())) { + NS_WARNING( + "HTMLEditor::CreateAndInsertElement(WithTransaction::Yes) failed"); + return createNewElementResult.propagateErr(); + } + CreateElementResult unwrappedCreateNewElementResult = + createNewElementResult.unwrap(); + // We'll return with a point suggesting new caret position and the + // following path does not require an update of selection here. + // Therefore, we don't need to update selection here. + unwrappedCreateNewElementResult.IgnoreCaretPointSuggestion(); + firstClonedElement = lastClonedElement = + unwrappedCreateNewElementResult.UnwrapNewNode(); + continue; + } + // Otherwise, inserts new parent inline container to the previous inserted + // inline container. + Result<CreateElementResult, nsresult> wrapClonedElementResult = + InsertContainerWithTransaction(*lastClonedElement, tagName); + if (MOZ_UNLIKELY(wrapClonedElementResult.isErr())) { + NS_WARNING("HTMLEditor::InsertContainerWithTransaction() failed"); + return wrapClonedElementResult.propagateErr(); + } + CreateElementResult unwrappedWrapClonedElementResult = + wrapClonedElementResult.unwrap(); + // We'll return with a point suggesting new caret so that we don't need to + // update selection here. + unwrappedWrapClonedElementResult.IgnoreCaretPointSuggestion(); + MOZ_ASSERT(unwrappedWrapClonedElementResult.GetNewNode()); + lastClonedElement = unwrappedWrapClonedElementResult.UnwrapNewNode(); + CloneAttributesWithTransaction(*lastClonedElement, *elementInPreviousBlock); + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + } + + if (!firstClonedElement) { + // XXX Even if no inline elements are cloned, shouldn't we create new + // <br> element for aNewBlock? + return EditorDOMPoint(&aNewBlock, 0u); + } + + Result<CreateElementResult, nsresult> insertBRElementResult = InsertBRElement( + WithTransaction::Yes, EditorDOMPoint(firstClonedElement, 0u)); + if (MOZ_UNLIKELY(insertBRElementResult.isErr())) { + NS_WARNING("HTMLEditor::InsertBRElement(WithTransaction::Yes) failed"); + return insertBRElementResult.propagateErr(); + } + insertBRElementResult.inspect().IgnoreCaretPointSuggestion(); + MOZ_ASSERT(insertBRElementResult.inspect().GetNewNode()); + return EditorDOMPoint(insertBRElementResult.inspect().GetNewNode()); +} + +nsresult HTMLEditor::GetElementOrigin(Element& aElement, int32_t& aX, + int32_t& aY) { + aX = 0; + aY = 0; + + if (NS_WARN_IF(!IsInitialized())) { + return NS_ERROR_NOT_INITIALIZED; + } + PresShell* presShell = GetPresShell(); + if (NS_WARN_IF(!presShell)) { + return NS_ERROR_NOT_INITIALIZED; + } + + nsIFrame* frame = aElement.GetPrimaryFrame(); + if (NS_WARN_IF(!frame)) { + return NS_OK; + } + + nsIFrame* absoluteContainerBlockFrame = + presShell->GetAbsoluteContainingBlock(frame); + if (NS_WARN_IF(!absoluteContainerBlockFrame)) { + return NS_OK; + } + nsPoint off = frame->GetOffsetTo(absoluteContainerBlockFrame); + aX = nsPresContext::AppUnitsToIntCSSPixels(off.x); + aY = nsPresContext::AppUnitsToIntCSSPixels(off.y); + + return NS_OK; +} + +Element* HTMLEditor::GetSelectionContainerElement() const { + MOZ_ASSERT(IsEditActionDataAvailable()); + + nsINode* focusNode = nullptr; + if (SelectionRef().IsCollapsed()) { + focusNode = SelectionRef().GetFocusNode(); + if (NS_WARN_IF(!focusNode)) { + return nullptr; + } + } else { + const uint32_t rangeCount = SelectionRef().RangeCount(); + MOZ_ASSERT(rangeCount, "If 0, Selection::IsCollapsed() should return true"); + + if (rangeCount == 1) { + const nsRange* range = SelectionRef().GetRangeAt(0); + + const RangeBoundary& startRef = range->StartRef(); + const RangeBoundary& endRef = range->EndRef(); + + // This method called GetSelectedElement() to retrieve proper container + // when only one node is selected. However, it simply returns start + // node of Selection with additional cost. So, we do not need to call + // it anymore. + if (startRef.Container()->IsElement() && + startRef.Container() == endRef.Container() && + startRef.GetChildAtOffset() && + startRef.GetChildAtOffset()->GetNextSibling() == + endRef.GetChildAtOffset()) { + focusNode = startRef.GetChildAtOffset(); + MOZ_ASSERT(focusNode, "Start container must not be nullptr"); + } else { + focusNode = range->GetClosestCommonInclusiveAncestor(); + if (!focusNode) { + NS_WARNING( + "AbstractRange::GetClosestCommonInclusiveAncestor() returned " + "nullptr"); + return nullptr; + } + } + } else { + for (const uint32_t i : IntegerRange(rangeCount)) { + MOZ_ASSERT(SelectionRef().RangeCount() == rangeCount); + const nsRange* range = SelectionRef().GetRangeAt(i); + MOZ_ASSERT(range); + nsINode* startContainer = range->GetStartContainer(); + if (!focusNode) { + focusNode = startContainer; + } else if (focusNode != startContainer) { + // XXX Looks odd to use parent of startContainer because previous + // range may not be in the parent node of current + // startContainer. + focusNode = startContainer->GetParentNode(); + // XXX Looks odd to break the for-loop here because we refer only + // first range and another range which starts from different + // container, and the latter range is preferred. Why? + break; + } + } + if (!focusNode) { + NS_WARNING("Focused node of selection was not found"); + return nullptr; + } + } + } + + if (focusNode->IsText()) { + focusNode = focusNode->GetParentNode(); + if (NS_WARN_IF(!focusNode)) { + return nullptr; + } + } + + if (NS_WARN_IF(!focusNode->IsElement())) { + return nullptr; + } + return focusNode->AsElement(); +} + +NS_IMETHODIMP HTMLEditor::IsAnonymousElement(Element* aElement, bool* aReturn) { + if (NS_WARN_IF(!aElement)) { + return NS_ERROR_INVALID_ARG; + } + *aReturn = aElement->IsRootOfNativeAnonymousSubtree(); + return NS_OK; +} + +nsresult HTMLEditor::SetReturnInParagraphCreatesNewParagraph( + bool aCreatesNewParagraph) { + mCRInParagraphCreatesParagraph = aCreatesNewParagraph; + return NS_OK; +} + +bool HTMLEditor::GetReturnInParagraphCreatesNewParagraph() const { + return mCRInParagraphCreatesParagraph; +} + +nsresult HTMLEditor::GetReturnInParagraphCreatesNewParagraph( + bool* aCreatesNewParagraph) { + *aCreatesNewParagraph = mCRInParagraphCreatesParagraph; + return NS_OK; +} + +Element* HTMLEditor::GetFocusedElement() const { + nsFocusManager* focusManager = nsFocusManager::GetFocusManager(); + if (NS_WARN_IF(!focusManager)) { + return nullptr; + } + + Element* const focusedElement = focusManager->GetFocusedElement(); + + Document* document = GetDocument(); + if (NS_WARN_IF(!document)) { + return nullptr; + } + const bool inDesignMode = IsInDesignMode(); + if (!focusedElement) { + // in designMode, nobody gets focus in most cases. + if (inDesignMode && OurWindowHasFocus()) { + return document->GetRootElement(); + } + return nullptr; + } + + if (inDesignMode) { + return OurWindowHasFocus() && + focusedElement->IsInclusiveDescendantOf(document) + ? focusedElement + : nullptr; + } + + // We're HTML editor for contenteditable + + // If the focused content isn't editable, or it has independent selection, + // we don't have focus. + if (!focusedElement->HasFlag(NODE_IS_EDITABLE) || + focusedElement->HasIndependentSelection()) { + return nullptr; + } + // If our window is focused, we're focused. + return OurWindowHasFocus() ? focusedElement : nullptr; +} + +bool HTMLEditor::IsActiveInDOMWindow() const { + nsFocusManager* focusManager = nsFocusManager::GetFocusManager(); + if (NS_WARN_IF(!focusManager)) { + return false; + } + + Document* document = GetDocument(); + if (NS_WARN_IF(!document)) { + return false; + } + const bool inDesignMode = IsInDesignMode(); + + // If we're in designMode, we're always active in the DOM window. + if (inDesignMode) { + return true; + } + + nsPIDOMWindowOuter* ourWindow = document->GetWindow(); + nsCOMPtr<nsPIDOMWindowOuter> win; + nsIContent* content = nsFocusManager::GetFocusedDescendant( + ourWindow, nsFocusManager::eOnlyCurrentWindow, getter_AddRefs(win)); + if (!content) { + return false; + } + + // We're HTML editor for contenteditable + + // If the active content isn't editable, or it has independent selection, + // we're not active). + if (!content->HasFlag(NODE_IS_EDITABLE) || + content->HasIndependentSelection()) { + return false; + } + return true; +} + +Element* HTMLEditor::ComputeEditingHostInternal( + const nsIContent* aContent, LimitInBodyElement aLimitInBodyElement) const { + Document* document = GetDocument(); + if (NS_WARN_IF(!document)) { + return nullptr; + } + + auto MaybeLimitInBodyElement = + [&](const Element* aCandidiateEditingHost) -> Element* { + if (!aCandidiateEditingHost) { + return nullptr; + } + if (aLimitInBodyElement != LimitInBodyElement::Yes) { + return const_cast<Element*>(aCandidiateEditingHost); + } + // By default, we should limit editing host to the <body> element for + // avoiding deleting or creating unexpected elements outside the <body>. + // However, this is incompatible with Chrome so that we should stop + // doing this with adding safety checks more. + if (document->GetBodyElement() && + nsContentUtils::ContentIsFlattenedTreeDescendantOf( + aCandidiateEditingHost, document->GetBodyElement())) { + return const_cast<Element*>(aCandidiateEditingHost); + } + // XXX If aContent is an editing host and has no parent node, we reach here, + // but returing the <body> which is not connected to aContent is odd. + return document->GetBodyElement(); + }; + + if (IsInDesignMode()) { + // TODO: In this case, we need to compute editing host from aContent or the + // focus node of selection, and it may be in an editing host in a + // shadow DOM tree etc. We need to do more complicated things. + // See also InDesignMode(). + return document->GetBodyElement(); + } + + // We're HTML editor for contenteditable + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return nullptr; + } + + const nsIContent* const content = + aContent ? aContent + : nsIContent::FromNodeOrNull(SelectionRef().GetFocusNode()); + if (NS_WARN_IF(!content)) { + return nullptr; + } + + // If the active content isn't editable, we're not active. + if (!content->HasFlag(NODE_IS_EDITABLE)) { + return nullptr; + } + + // Although the content shouldn't be in a native anonymous subtree, but + // perhaps due to a bug of Selection or Range API, it may occur. HTMLEditor + // shouldn't touch native anonymous subtree so that return nullptr in such + // case. + if (MOZ_UNLIKELY(content->IsInNativeAnonymousSubtree())) { + return nullptr; + } + + // Note that `Selection` can be in <input> or <textarea>. In the case, we + // need to look for an ancestor which does not have editable parent. + return MaybeLimitInBodyElement( + const_cast<nsIContent*>(content)->GetEditingHost()); +} + +void HTMLEditor::NotifyEditingHostMaybeChanged() { + // Note that even if the document is in design mode, a contenteditable element + // in a shadow tree is focusable. Therefore, we may need to update editing + // host even when the document is in design mode. + if (MOZ_UNLIKELY(NS_WARN_IF(!GetDocument()))) { + return; + } + + // We're HTML editor for contenteditable + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return; + } + + // Get selection ancestor limit which may be old editing host. + nsIContent* ancestorLimiter = SelectionRef().GetAncestorLimiter(); + if (!ancestorLimiter) { + // If we've not initialized selection ancestor limit, we should wait focus + // event to set proper limiter. + return; + } + + // Compute current editing host. + nsIContent* editingHost = ComputeEditingHost(); + if (NS_WARN_IF(!editingHost)) { + return; + } + + // Update selection ancestor limit if current editing host includes the + // previous editing host. + // Additionally, the editing host may be an element in shadow DOM and the + // shadow host is in designMode. In this case, we need to set the editing + // host as the new selection limiter. + if (ancestorLimiter->IsInclusiveDescendantOf(editingHost) || + (ancestorLimiter->IsInDesignMode() != editingHost->IsInDesignMode())) { + // Note that don't call HTMLEditor::InitializeSelectionAncestorLimit() + // here because it may collapse selection to the first editable node. + EditorBase::InitializeSelectionAncestorLimit(*editingHost); + } +} + +EventTarget* HTMLEditor::GetDOMEventTarget() const { + // Don't use getDocument here, because we have no way of knowing + // whether Init() was ever called. So we need to get the document + // ourselves, if it exists. + MOZ_ASSERT(IsInitialized(), "The HTMLEditor has not been initialized yet"); + return GetDocument(); +} + +bool HTMLEditor::ShouldReplaceRootElement() const { + if (!mRootElement) { + // If we don't know what is our root element, we should find our root. + return true; + } + + // If we temporary set document root element to mRootElement, but there is + // body element now, we should replace the root element by the body element. + return mRootElement != GetBodyElement(); +} + +void HTMLEditor::NotifyRootChanged() { + MOZ_ASSERT(mPendingRootElementUpdatedRunner, + "HTMLEditor::NotifyRootChanged() should be called via a runner"); + mPendingRootElementUpdatedRunner = nullptr; + + nsCOMPtr<nsIMutationObserver> kungFuDeathGrip(this); + + AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return; + } + + RemoveEventListeners(); + nsresult rv = InstallEventListeners(); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::InstallEventListeners() failed, but ignored"); + return; + } + + UpdateRootElement(); + if (!mRootElement) { + return; + } + + rv = MaybeCollapseSelectionAtFirstEditableNode(false); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::MaybeCollapseSelectionAtFirstEditableNode(false) " + "failed, " + "but ignored"); + return; + } + + // When this editor has focus, we need to reset the selection limiter to + // new root. Otherwise, that is going to be done when this gets focus. + nsCOMPtr<nsINode> node = GetFocusedNode(); + if (node) { + DebugOnly<nsresult> rvIgnored = InitializeSelection(*node); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvIgnored), + "EditorBase::InitializeSelection() failed, but ignored"); + } + + SyncRealTimeSpell(); +} + +Element* HTMLEditor::GetBodyElement() const { + MOZ_ASSERT(IsInitialized(), "The HTMLEditor hasn't been initialized yet"); + Document* document = GetDocument(); + if (NS_WARN_IF(!document)) { + return nullptr; + } + return document->GetBody(); +} + +nsINode* HTMLEditor::GetFocusedNode() const { + Element* focusedElement = GetFocusedElement(); + if (!focusedElement) { + return nullptr; + } + + // focusedElement might be non-null even focusManager->GetFocusedElement() + // is null. That's the designMode case, and in that case our + // FocusedContent() returns the root element, but we want to return + // the document. + + nsFocusManager* focusManager = nsFocusManager::GetFocusManager(); + NS_ASSERTION(focusManager, "Focus manager is null"); + if ((focusedElement = focusManager->GetFocusedElement())) { + return focusedElement; + } + + return GetDocument(); +} + +bool HTMLEditor::OurWindowHasFocus() const { + nsFocusManager* focusManager = nsFocusManager::GetFocusManager(); + if (NS_WARN_IF(!focusManager)) { + return false; + } + nsPIDOMWindowOuter* focusedWindow = focusManager->GetFocusedWindow(); + if (!focusedWindow) { + return false; + } + Document* document = GetDocument(); + if (NS_WARN_IF(!document)) { + return false; + } + nsPIDOMWindowOuter* ourWindow = document->GetWindow(); + return ourWindow == focusedWindow; +} + +bool HTMLEditor::IsAcceptableInputEvent(WidgetGUIEvent* aGUIEvent) const { + if (!EditorBase::IsAcceptableInputEvent(aGUIEvent)) { + return false; + } + + // While there is composition, all composition events in its top level + // window are always fired on the composing editor. Therefore, if this + // editor has composition, the composition events should be handled in this + // editor. + if (mComposition && aGUIEvent->AsCompositionEvent()) { + return true; + } + + nsCOMPtr<nsINode> eventTargetNode = + nsINode::FromEventTargetOrNull(aGUIEvent->GetOriginalDOMEventTarget()); + if (NS_WARN_IF(!eventTargetNode)) { + return false; + } + + if (eventTargetNode->IsContent()) { + eventTargetNode = + eventTargetNode->AsContent()->FindFirstNonChromeOnlyAccessContent(); + if (NS_WARN_IF(!eventTargetNode)) { + return false; + } + } + + RefPtr<Document> document = GetDocument(); + if (NS_WARN_IF(!document)) { + return false; + } + + if (IsInDesignMode()) { + // If this editor is in designMode and the event target is the document, + // the event is for this editor. + if (eventTargetNode->IsDocument()) { + return eventTargetNode == document; + } + // Otherwise, check whether the event target is in this document or not. + if (NS_WARN_IF(!eventTargetNode->IsContent())) { + return false; + } + if (document == eventTargetNode->GetUncomposedDoc()) { + return true; + } + // If the event target is in a shadow tree, the content is not editable + // by default, but if the focused content is an editing host, we need to + // handle it as contenteditable mode. + if (!eventTargetNode->IsInShadowTree()) { + return false; + } + } + + // This HTML editor is for contenteditable. We need to check the validity + // of the target. + if (NS_WARN_IF(!eventTargetNode->IsContent())) { + return false; + } + + // If the event is a mouse event, we need to check if the target content is + // the focused editing host or its descendant. + if (aGUIEvent->AsMouseEventBase()) { + nsIContent* editingHost = ComputeEditingHost(); + // If there is no active editing host, we cannot handle the mouse event + // correctly. + if (!editingHost) { + return false; + } + // If clicked on non-editable root element but the body element is the + // active editing host, we should assume that the click event is + // targetted. + if (eventTargetNode == document->GetRootElement() && + !eventTargetNode->HasFlag(NODE_IS_EDITABLE) && + editingHost == document->GetBodyElement()) { + eventTargetNode = editingHost; + } + // If the target element is neither the active editing host nor a + // descendant of it, we may not be able to handle the event. + if (!eventTargetNode->IsInclusiveDescendantOf(editingHost)) { + return false; + } + // If the clicked element has an independent selection, we shouldn't + // handle this click event. + if (eventTargetNode->AsContent()->HasIndependentSelection()) { + return false; + } + // If the target content is editable, we should handle this event. + return eventTargetNode->HasFlag(NODE_IS_EDITABLE); + } + + // If the target of the other events which target focused element isn't + // editable or has an independent selection, this editor shouldn't handle + // the event. + if (!eventTargetNode->HasFlag(NODE_IS_EDITABLE) || + eventTargetNode->AsContent()->HasIndependentSelection()) { + return false; + } + + // Finally, check whether we're actually focused or not. When we're not + // focused, we should ignore the dispatched event by script (or something) + // because content editable element needs selection in itself for editing. + // However, when we're not focused, it's not guaranteed. + return IsActiveInDOMWindow(); +} + +nsresult HTMLEditor::GetPreferredIMEState(IMEState* aState) { + // HTML editor don't prefer the CSS ime-mode because IE didn't do so too. + aState->mOpen = IMEState::DONT_CHANGE_OPEN_STATE; + if (IsReadonly()) { + aState->mEnabled = IMEEnabled::Disabled; + } else { + aState->mEnabled = IMEEnabled::Enabled; + } + return NS_OK; +} + +already_AddRefed<Element> HTMLEditor::GetInputEventTargetElement() const { + RefPtr<Element> target = ComputeEditingHost(LimitInBodyElement::No); + if (target) { + return target.forget(); + } + + // When there is no active editing host due to focus node is a + // non-editable node, we should look for its editable parent to + // dispatch `beforeinput` event. + nsIContent* focusContent = + nsIContent::FromNodeOrNull(SelectionRef().GetFocusNode()); + if (!focusContent || focusContent->IsEditable()) { + return nullptr; + } + for (Element* element : focusContent->AncestorsOfType<Element>()) { + if (element->IsEditable()) { + target = element->GetEditingHost(); + return target.forget(); + } + } + return nullptr; +} + +nsresult HTMLEditor::OnModifyDocument() { + MOZ_ASSERT(mPendingDocumentModifiedRunner, + "HTMLEditor::OnModifyDocument() should be called via a runner"); + mPendingDocumentModifiedRunner = nullptr; + + if (IsEditActionDataAvailable()) { + return OnModifyDocumentInternal(); + } + + AutoEditActionDataSetter editActionData( + *this, EditAction::eCreatePaddingBRElementForEmptyEditor); + if (NS_WARN_IF(!editActionData.CanHandle())) { + return NS_ERROR_NOT_AVAILABLE; + } + + nsresult rv = OnModifyDocumentInternal(); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "HTMLEditor::OnModifyDocumentInternal() failed"); + return rv; +} + +nsresult HTMLEditor::OnModifyDocumentInternal() { + MOZ_ASSERT(IsEditActionDataAvailable()); + MOZ_ASSERT(!mPendingDocumentModifiedRunner); + + // EnsureNoPaddingBRElementForEmptyEditor() below may cause a flush, which + // could destroy the editor + nsAutoScriptBlockerSuppressNodeRemoved scriptBlocker; + + // Delete our padding <br> element for empty editor, if we have one, since + // the document might not be empty any more. + nsresult rv = EnsureNoPaddingBRElementForEmptyEditor(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return rv; + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::EnsureNoPaddingBRElementForEmptyEditor() " + "failed, but ignored"); + + // Try to recreate the padding <br> element for empty editor if needed. + rv = MaybeCreatePaddingBRElementForEmptyEditor(); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::MaybeCreatePaddingBRElementForEmptyEditor() failed"); + + return rv; +} + +} // namespace mozilla |