diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-21 11:44:51 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-21 11:44:51 +0000 |
commit | 9e3c08db40b8916968b9f30096c7be3f00ce9647 (patch) | |
tree | a68f146d7fa01f0134297619fbe7e33db084e0aa /editor/libeditor/HTMLEditorDeleteHandler.cpp | |
parent | Initial commit. (diff) | |
download | thunderbird-9e3c08db40b8916968b9f30096c7be3f00ce9647.tar.xz thunderbird-9e3c08db40b8916968b9f30096c7be3f00ce9647.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/HTMLEditorDeleteHandler.cpp')
-rw-r--r-- | editor/libeditor/HTMLEditorDeleteHandler.cpp | 6854 |
1 files changed, 6854 insertions, 0 deletions
diff --git a/editor/libeditor/HTMLEditorDeleteHandler.cpp b/editor/libeditor/HTMLEditorDeleteHandler.cpp new file mode 100644 index 0000000000..928d9b3b4c --- /dev/null +++ b/editor/libeditor/HTMLEditorDeleteHandler.cpp @@ -0,0 +1,6854 @@ +/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=2 sw=2 et tw=80: */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +#include "HTMLEditor.h" +#include "HTMLEditorNestedClasses.h" + +#include <algorithm> +#include <utility> + +#include "AutoRangeArray.h" +#include "CSSEditUtils.h" +#include "EditAction.h" +#include "EditorDOMPoint.h" +#include "EditorUtils.h" +#include "HTMLEditHelpers.h" +#include "HTMLEditUtils.h" +#include "WSRunObject.h" + +#include "ErrorList.h" +#include "js/ErrorReport.h" +#include "mozilla/Assertions.h" +#include "mozilla/CheckedInt.h" +#include "mozilla/ComputedStyle.h" // for ComputedStyle +#include "mozilla/ContentIterator.h" +#include "mozilla/EditorDOMPoint.h" +#include "mozilla/EditorForwards.h" +#include "mozilla/InternalMutationEvent.h" +#include "mozilla/Maybe.h" +#include "mozilla/OwningNonNull.h" +#include "mozilla/SelectionState.h" +#include "mozilla/StaticPrefs_editor.h" // for StaticPrefs::editor_* +#include "mozilla/Unused.h" +#include "mozilla/dom/AncestorIterator.h" +#include "mozilla/dom/Element.h" +#include "mozilla/dom/HTMLBRElement.h" +#include "mozilla/dom/Selection.h" +#include "mozilla/mozalloc.h" +#include "nsAString.h" +#include "nsAtom.h" +#include "nsComputedDOMStyle.h" // for nsComputedDOMStyle +#include "nsContentUtils.h" +#include "nsDebug.h" +#include "nsError.h" +#include "nsFrameSelection.h" +#include "nsGkAtoms.h" +#include "nsIContent.h" +#include "nsINode.h" +#include "nsRange.h" +#include "nsString.h" +#include "nsStringFwd.h" +#include "nsStyleConsts.h" // for StyleWhiteSpace +#include "nsTArray.h" + +// NOTE: This file was split from: +// https://searchfox.org/mozilla-central/rev/c409dd9235c133ab41eba635f906aa16e050c197/editor/libeditor/HTMLEditSubActionHandler.cpp + +namespace mozilla { + +using namespace dom; +using EmptyCheckOption = HTMLEditUtils::EmptyCheckOption; +using InvisibleWhiteSpaces = HTMLEditUtils::InvisibleWhiteSpaces; +using LeafNodeType = HTMLEditUtils::LeafNodeType; +using ScanLineBreak = HTMLEditUtils::ScanLineBreak; +using TableBoundary = HTMLEditUtils::TableBoundary; +using WalkTreeOption = HTMLEditUtils::WalkTreeOption; + +template Result<CaretPoint, nsresult> +HTMLEditor::DeleteTextAndTextNodesWithTransaction( + const EditorDOMPoint& aStartPoint, const EditorDOMPoint& aEndPoint, + TreatEmptyTextNodes aTreatEmptyTextNodes); +template Result<CaretPoint, nsresult> +HTMLEditor::DeleteTextAndTextNodesWithTransaction( + const EditorDOMPointInText& aStartPoint, + const EditorDOMPointInText& aEndPoint, + TreatEmptyTextNodes aTreatEmptyTextNodes); + +/***************************************************************************** + * AutoSetTemporaryAncestorLimiter + ****************************************************************************/ + +class MOZ_RAII AutoSetTemporaryAncestorLimiter final { + public: + AutoSetTemporaryAncestorLimiter(const HTMLEditor& aHTMLEditor, + Selection& aSelection, + nsINode& aStartPointNode, + AutoRangeArray* aRanges = nullptr) { + MOZ_ASSERT(aSelection.GetType() == SelectionType::eNormal); + + if (aSelection.GetAncestorLimiter()) { + return; + } + + Element* selectionRootElement = + aHTMLEditor.FindSelectionRoot(aStartPointNode); + if (!selectionRootElement) { + return; + } + aHTMLEditor.InitializeSelectionAncestorLimit(*selectionRootElement); + mSelection = &aSelection; + // Setting ancestor limiter may change ranges which were outer of + // the new limiter. Therefore, we need to reinitialize aRanges. + if (aRanges) { + aRanges->Initialize(aSelection); + } + } + + ~AutoSetTemporaryAncestorLimiter() { + if (mSelection) { + mSelection->SetAncestorLimiter(nullptr); + } + } + + private: + RefPtr<Selection> mSelection; +}; + +/***************************************************************************** + * AutoDeleteRangesHandler + ****************************************************************************/ + +class MOZ_STACK_CLASS HTMLEditor::AutoDeleteRangesHandler final { + public: + explicit AutoDeleteRangesHandler( + const AutoDeleteRangesHandler* aParent = nullptr) + : mParent(aParent), + mOriginalDirectionAndAmount(nsIEditor::eNone), + mOriginalStripWrappers(nsIEditor::eNoStrip) {} + + /** + * ComputeRangesToDelete() computes actual deletion ranges. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT nsresult ComputeRangesToDelete( + const HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, const Element& aEditingHost); + + /** + * Deletes content in or around aRangesToDelete. + * NOTE: This method creates SelectionBatcher. Therefore, each caller + * needs to check if the editor is still available even if this returns + * NS_OK. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> Run( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, AutoRangeArray& aRangesToDelete, + const Element& aEditingHost); + + private: + bool IsHandlingRecursively() const { return mParent != nullptr; } + + bool CanFallbackToDeleteRangesWithTransaction( + const AutoRangeArray& aRangesToDelete) const { + return !IsHandlingRecursively() && !aRangesToDelete.Ranges().IsEmpty() && + (!aRangesToDelete.IsCollapsed() || + EditorBase::HowToHandleCollapsedRangeFor( + mOriginalDirectionAndAmount) != + EditorBase::HowToHandleCollapsedRange::Ignore); + } + + /** + * HandleDeleteAroundCollapsedRanges() handles deletion with collapsed + * ranges. Callers must guarantee that this is called only when + * aRangesToDelete.IsCollapsed() returns true. + * + * @param aDirectionAndAmount Direction of the deletion. + * @param aStripWrappers Must be eStrip or eNoStrip. + * @param aRangesToDelete Ranges to delete. This `IsCollapsed()` must + * return true. + * @param aWSRunScannerAtCaret Scanner instance which scanned from + * caret point. + * @param aScanFromCaretPointResult Scan result of aWSRunScannerAtCaret + * toward aDirectionAndAmount. + * @param aEditingHost The editing host. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + HandleDeleteAroundCollapsedRanges( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, AutoRangeArray& aRangesToDelete, + const WSRunScanner& aWSRunScannerAtCaret, + const WSScanResult& aScanFromCaretPointResult, + const Element& aEditingHost); + nsresult ComputeRangesToDeleteAroundCollapsedRanges( + const HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, const WSRunScanner& aWSRunScannerAtCaret, + const WSScanResult& aScanFromCaretPointResult, + const Element& aEditingHost) const; + + /** + * HandleDeleteNonCollapsedRanges() handles deletion with non-collapsed + * ranges. Callers must guarantee that this is called only when + * aRangesToDelete.IsCollapsed() returns false. + * + * @param aDirectionAndAmount Direction of the deletion. + * @param aStripWrappers Must be eStrip or eNoStrip. + * @param aRangesToDelete The ranges to delete. + * @param aSelectionWasCollapsed If the caller extended `Selection` + * from collapsed, set this to `Yes`. + * Otherwise, i.e., `Selection` is not + * collapsed from the beginning, set + * this to `No`. + * @param aEditingHost The editing host. + */ + enum class SelectionWasCollapsed { Yes, No }; + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + HandleDeleteNonCollapsedRanges(HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + AutoRangeArray& aRangesToDelete, + SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost); + nsresult ComputeRangesToDeleteNonCollapsedRanges( + const HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, + SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost) const; + + /** + * Handle deletion of collapsed ranges in a text node. + * + * @param aDirectionAndAmount Must be eNext or ePrevious. + * @param aCaretPosition The position where caret is. This container + * must be a text node. + * @param aEditingHost The editing host. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<CaretPoint, nsresult> + HandleDeleteTextAroundCollapsedRanges( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, const Element& aEditingHost); + nsresult ComputeRangesToDeleteTextAroundCollapsedRanges( + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, const Element& aEditingHost) const; + + /** + * Handles deletion of collapsed selection at white-spaces in a text node. + * + * @param aDirectionAndAmount Direction of the deletion. + * @param aPointToDelete The point to delete. I.e., typically, caret + * position. + * @param aEditingHost The editing host. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<CaretPoint, nsresult> + HandleDeleteCollapsedSelectionAtWhiteSpaces( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + const EditorDOMPoint& aPointToDelete, const Element& aEditingHost); + + /** + * Handle deletion of collapsed selection in a text node. + * + * @param aDirectionAndAmount Direction of the deletion. + * @param aRangesToDelete Computed selection ranges to delete. + * @param aPointAtDeletingChar The visible char position which you want to + * delete. + * @param aEditingHost The editing host. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<CaretPoint, nsresult> + HandleDeleteCollapsedSelectionAtVisibleChar( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, + const EditorDOMPoint& aPointAtDeletingChar, const Element& aEditingHost); + + /** + * Handle deletion of atomic elements like <br>, <hr>, <img>, <input>, etc and + * data nodes except text node (e.g., comment node). Note that don't call this + * directly with `<hr>` element. + * + * @param aAtomicContent The atomic content to be deleted. + * @param aCaretPoint The caret point (i.e., selection start or + * end). + * @param aWSRunScannerAtCaret WSRunScanner instance which was initialized + * with the caret point. + * @param aEditingHost The editing host. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<CaretPoint, nsresult> + HandleDeleteAtomicContent(HTMLEditor& aHTMLEditor, nsIContent& aAtomicContent, + const EditorDOMPoint& aCaretPoint, + const WSRunScanner& aWSRunScannerAtCaret, + const Element& aEditingHost); + nsresult ComputeRangesToDeleteAtomicContent( + Element* aEditingHost, const nsIContent& aAtomicContent, + AutoRangeArray& aRangesToDelete) const; + + /** + * GetAtomicContnetToDelete() returns better content that is deletion of + * atomic element. If aScanFromCaretPointResult is special, since this + * point may not be editable, we look for better point to remove atomic + * content. + * + * @param aDirectionAndAmount Direction of the deletion. + * @param aWSRunScannerAtCaret WSRunScanner instance which was + * initialized with the caret point. + * @param aScanFromCaretPointResult Scan result of aWSRunScannerAtCaret + * toward aDirectionAndAmount. + */ + static nsIContent* GetAtomicContentToDelete( + nsIEditor::EDirection aDirectionAndAmount, + const WSRunScanner& aWSRunScannerAtCaret, + const WSScanResult& aScanFromCaretPointResult) MOZ_NONNULL_RETURN; + + /** + * HandleDeleteAtOtherBlockBoundary() handles deletion at other block boundary + * (i.e., immediately before or after a block). If this does not join blocks, + * `Run()` may be called recursively with creating another instance. + * + * @param aDirectionAndAmount Direction of the deletion. + * @param aStripWrappers Must be eStrip or eNoStrip. + * @param aOtherBlockElement The block element which follows the caret or + * is followed by caret. + * @param aCaretPoint The caret point (i.e., selection start or + * end). + * @param aWSRunScannerAtCaret WSRunScanner instance which was initialized + * with the caret point. + * @param aRangesToDelete Ranges to delete of the caller. This should + * be collapsed and the point should match with + * aCaretPoint. + * @param aEditingHost The editing host. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + HandleDeleteAtOtherBlockBoundary( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, Element& aOtherBlockElement, + const EditorDOMPoint& aCaretPoint, WSRunScanner& aWSRunScannerAtCaret, + AutoRangeArray& aRangesToDelete, const Element& aEditingHost); + + /** + * ExtendOrShrinkRangeToDelete() extends aRangeToDelete if there are + * an invisible <br> element and/or some parent empty elements. + * + * @param aFrameSelection If the caller wants range in selection limiter, + * set this to non-nullptr which knows the limiter. + * @param aRangeToDelete The range to be extended for deletion. This + * must not be collapsed, must be positioned. + */ + template <typename EditorDOMRangeType> + Result<EditorRawDOMRange, nsresult> ExtendOrShrinkRangeToDelete( + const HTMLEditor& aHTMLEditor, const nsFrameSelection* aFrameSelection, + const EditorDOMRangeType& aRangeToDelete) const; + + /** + * A helper method for ExtendOrShrinkRangeToDelete(). This returns shrunken + * range if aRangeToDelete selects all over list elements which have some list + * item elements to avoid to delete all list items from the list element. + */ + MOZ_NEVER_INLINE_DEBUG static EditorRawDOMRange + GetRangeToAvoidDeletingAllListItemsIfSelectingAllOverListElements( + const EditorRawDOMRange& aRangeToDelete); + + /** + * DeleteUnnecessaryNodes() removes unnecessary nodes around aRange. + * Note that aRange is tracked with AutoTrackDOMRange. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT nsresult + DeleteUnnecessaryNodes(HTMLEditor& aHTMLEditor, EditorDOMRange& aRange); + + /** + * DeleteUnnecessaryNodesAndCollapseSelection() calls DeleteUnnecessaryNodes() + * and then, collapse selection at tracked aSelectionStartPoint or + * aSelectionEndPoint (depending on aDirectionAndAmount). + * + * @param aDirectionAndAmount Direction of the deletion. + * If nsIEditor::ePrevious, selection + * will be collapsed to aSelectionEndPoint. + * Otherwise, selection will be collapsed + * to aSelectionStartPoint. + * @param aSelectionStartPoint First selection range start after + * computing the deleting range. + * @param aSelectionEndPoint First selection range end after + * computing the deleting range. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT nsresult + DeleteUnnecessaryNodesAndCollapseSelection( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + const EditorDOMPoint& aSelectionStartPoint, + const EditorDOMPoint& aSelectionEndPoint); + + /** + * If aContent is a text node that contains only collapsed white-space or + * empty and editable. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT nsresult + DeleteNodeIfInvisibleAndEditableTextNode(HTMLEditor& aHTMLEditor, + nsIContent& aContent); + + /** + * DeleteParentBlocksIfEmpty() removes parent block elements if they + * don't have visible contents. Note that due performance issue of + * WhiteSpaceVisibilityKeeper, this call may be expensive. And also note that + * this removes a empty block with a transaction. So, please make sure that + * you've already created `AutoPlaceholderBatch`. + * + * @param aPoint The point whether this method climbing up the DOM + * tree to remove empty parent blocks. + * @return NS_OK if one or more empty block parents are deleted. + * NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND if the point is + * not in empty block. + * Or NS_ERROR_* if something unexpected occurs. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT nsresult + DeleteParentBlocksWithTransactionIfEmpty(HTMLEditor& aHTMLEditor, + const EditorDOMPoint& aPoint); + + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<CaretPoint, nsresult> + FallbackToDeleteRangesWithTransaction(HTMLEditor& aHTMLEditor, + AutoRangeArray& aRangesToDelete) const { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(CanFallbackToDeleteRangesWithTransaction(aRangesToDelete)); + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteRangesWithTransaction(mOriginalDirectionAndAmount, + mOriginalStripWrappers, + aRangesToDelete); + NS_WARNING_ASSERTION(caretPointOrError.isOk(), + "HTMLEditor::DeleteRangesWithTransaction() failed"); + return caretPointOrError; + } + + /** + * ComputeRangesToDeleteRangesWithTransaction() computes target ranges + * which will be called by `EditorBase::DeleteRangesWithTransaction()`. + * TODO: We should not use it for consistency with each deletion handler + * in this and nested classes. + */ + nsresult ComputeRangesToDeleteRangesWithTransaction( + const HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete) const; + + nsresult FallbackToComputeRangesToDeleteRangesWithTransaction( + const HTMLEditor& aHTMLEditor, AutoRangeArray& aRangesToDelete) const { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(CanFallbackToDeleteRangesWithTransaction(aRangesToDelete)); + nsresult rv = ComputeRangesToDeleteRangesWithTransaction( + aHTMLEditor, mOriginalDirectionAndAmount, aRangesToDelete); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::" + "ComputeRangesToDeleteRangesWithTransaction() failed"); + return rv; + } + + class MOZ_STACK_CLASS AutoBlockElementsJoiner final { + public: + AutoBlockElementsJoiner() = delete; + explicit AutoBlockElementsJoiner( + AutoDeleteRangesHandler& aDeleteRangesHandler) + : mDeleteRangesHandler(&aDeleteRangesHandler), + mDeleteRangesHandlerConst(aDeleteRangesHandler) {} + explicit AutoBlockElementsJoiner( + const AutoDeleteRangesHandler& aDeleteRangesHandler) + : mDeleteRangesHandler(nullptr), + mDeleteRangesHandlerConst(aDeleteRangesHandler) {} + + /** + * PrepareToDeleteAtCurrentBlockBoundary() considers left content and right + * content which are joined for handling deletion at current block boundary + * (i.e., at start or end of the current block). + * + * @param aHTMLEditor The HTML editor. + * @param aDirectionAndAmount Direction of the deletion. + * @param aCurrentBlockElement The current block element. + * @param aCaretPoint The caret point (i.e., selection start + * or end). + * @return true if can continue to handle the + * deletion. + */ + bool PrepareToDeleteAtCurrentBlockBoundary( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + Element& aCurrentBlockElement, const EditorDOMPoint& aCaretPoint); + + /** + * PrepareToDeleteAtOtherBlockBoundary() considers left content and right + * content which are joined for handling deletion at other block boundary + * (i.e., immediately before or after a block). + * + * @param aHTMLEditor The HTML editor. + * @param aDirectionAndAmount Direction of the deletion. + * @param aOtherBlockElement The block element which follows the + * caret or is followed by caret. + * @param aCaretPoint The caret point (i.e., selection start + * or end). + * @param aWSRunScannerAtCaret WSRunScanner instance which was + * initialized with the caret point. + * @return true if can continue to handle the + * deletion. + */ + bool PrepareToDeleteAtOtherBlockBoundary( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, Element& aOtherBlockElement, + const EditorDOMPoint& aCaretPoint, + const WSRunScanner& aWSRunScannerAtCaret); + + /** + * PrepareToDeleteNonCollapsedRanges() considers left block element and + * right block element which are inclusive ancestor block element of + * start and end container of first range of aRangesToDelete. + * + * @param aHTMLEditor The HTML editor. + * @param aRangesToDelete Ranges to delete. Must not be + * collapsed. + * @return true if can continue to handle the + * deletion. + */ + bool PrepareToDeleteNonCollapsedRanges( + const HTMLEditor& aHTMLEditor, const AutoRangeArray& aRangesToDelete); + + /** + * Run() executes the joining. + * + * @param aHTMLEditor The HTML editor. + * @param aDirectionAndAmount Direction of the deletion. + * @param aStripWrappers Must be eStrip or eNoStrip. + * @param aCaretPoint The caret point (i.e., selection start + * or end). + * @param aRangesToDelete Ranges to delete of the caller. + * This should be collapsed and match + * with aCaretPoint. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> Run( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + const EditorDOMPoint& aCaretPoint, AutoRangeArray& aRangesToDelete, + const Element& aEditingHost) { + switch (mMode) { + case Mode::JoinCurrentBlock: { + Result<EditActionResult, nsresult> result = + HandleDeleteAtCurrentBlockBoundary( + aHTMLEditor, aDirectionAndAmount, aCaretPoint, aEditingHost); + NS_WARNING_ASSERTION(result.isOk(), + "AutoBlockElementsJoiner::" + "HandleDeleteAtCurrentBlockBoundary() failed"); + return result; + } + case Mode::JoinOtherBlock: { + Result<EditActionResult, nsresult> result = + HandleDeleteAtOtherBlockBoundary(aHTMLEditor, aDirectionAndAmount, + aStripWrappers, aCaretPoint, + aRangesToDelete, aEditingHost); + NS_WARNING_ASSERTION(result.isOk(), + "AutoBlockElementsJoiner::" + "HandleDeleteAtOtherBlockBoundary() failed"); + return result; + } + case Mode::DeleteBRElement: { + Result<EditActionResult, nsresult> result = + DeleteBRElement(aHTMLEditor, aDirectionAndAmount, aEditingHost); + NS_WARNING_ASSERTION( + result.isOk(), + "AutoBlockElementsJoiner::DeleteBRElement() failed"); + return result; + } + case Mode::JoinBlocksInSameParent: + case Mode::DeleteContentInRanges: + case Mode::DeleteNonCollapsedRanges: + MOZ_ASSERT_UNREACHABLE( + "This mode should be handled in the other Run()"); + return Err(NS_ERROR_UNEXPECTED); + case Mode::NotInitialized: + return EditActionResult::IgnoredResult(); + } + return Err(NS_ERROR_NOT_INITIALIZED); + } + + nsresult ComputeRangesToDelete(const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + const EditorDOMPoint& aCaretPoint, + AutoRangeArray& aRangesToDelete, + const Element& aEditingHost) const { + switch (mMode) { + case Mode::JoinCurrentBlock: { + nsresult rv = ComputeRangesToDeleteAtCurrentBlockBoundary( + aHTMLEditor, aCaretPoint, aRangesToDelete, aEditingHost); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::" + "ComputeRangesToDeleteAtCurrentBlockBoundary() failed"); + return rv; + } + case Mode::JoinOtherBlock: { + nsresult rv = ComputeRangesToDeleteAtOtherBlockBoundary( + aHTMLEditor, aDirectionAndAmount, aCaretPoint, aRangesToDelete, + aEditingHost); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::" + "ComputeRangesToDeleteAtOtherBlockBoundary() failed"); + return rv; + } + case Mode::DeleteBRElement: { + nsresult rv = ComputeRangesToDeleteBRElement(aRangesToDelete); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::" + "ComputeRangesToDeleteBRElement() failed"); + return rv; + } + case Mode::JoinBlocksInSameParent: + case Mode::DeleteContentInRanges: + case Mode::DeleteNonCollapsedRanges: + MOZ_ASSERT_UNREACHABLE( + "This mode should be handled in the other " + "ComputeRangesToDelete()"); + return NS_ERROR_UNEXPECTED; + case Mode::NotInitialized: + return NS_OK; + } + return NS_ERROR_NOT_IMPLEMENTED; + } + + /** + * Run() executes the joining. + * + * @param aHTMLEditor The HTML editor. + * @param aDirectionAndAmount Direction of the deletion. + * @param aStripWrappers Whether delete or keep new empty + * ancestor elements. + * @param aRangesToDelete Ranges to delete. Must not be + * collapsed. + * @param aSelectionWasCollapsed Whether selection was or was not + * collapsed when starting to handle + * deletion. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> Run( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + AutoRangeArray& aRangesToDelete, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost) { + switch (mMode) { + case Mode::JoinCurrentBlock: + case Mode::JoinOtherBlock: + case Mode::DeleteBRElement: + MOZ_ASSERT_UNREACHABLE( + "This mode should be handled in the other Run()"); + return Err(NS_ERROR_UNEXPECTED); + case Mode::JoinBlocksInSameParent: { + Result<EditActionResult, nsresult> result = + JoinBlockElementsInSameParent( + aHTMLEditor, aDirectionAndAmount, aStripWrappers, + aRangesToDelete, aSelectionWasCollapsed, aEditingHost); + NS_WARNING_ASSERTION(result.isOk(), + "AutoBlockElementsJoiner::" + "JoinBlockElementsInSameParent() failed"); + return result; + } + case Mode::DeleteContentInRanges: { + Result<EditActionResult, nsresult> result = + DeleteContentInRanges(aHTMLEditor, aDirectionAndAmount, + aStripWrappers, aRangesToDelete); + NS_WARNING_ASSERTION( + result.isOk(), + "AutoBlockElementsJoiner::DeleteContentInRanges() failed"); + return result; + } + case Mode::DeleteNonCollapsedRanges: { + Result<EditActionResult, nsresult> result = + HandleDeleteNonCollapsedRanges( + aHTMLEditor, aDirectionAndAmount, aStripWrappers, + aRangesToDelete, aSelectionWasCollapsed, aEditingHost); + NS_WARNING_ASSERTION(result.isOk(), + "AutoBlockElementsJoiner::" + "HandleDeleteNonCollapsedRange() failed"); + return result; + } + case Mode::NotInitialized: + MOZ_ASSERT_UNREACHABLE( + "Call Run() after calling a preparation method"); + return EditActionResult::IgnoredResult(); + } + return Err(NS_ERROR_NOT_INITIALIZED); + } + + nsresult ComputeRangesToDelete( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost) const { + switch (mMode) { + case Mode::JoinCurrentBlock: + case Mode::JoinOtherBlock: + case Mode::DeleteBRElement: + MOZ_ASSERT_UNREACHABLE( + "This mode should be handled in the other " + "ComputeRangesToDelete()"); + return NS_ERROR_UNEXPECTED; + case Mode::JoinBlocksInSameParent: { + nsresult rv = ComputeRangesToJoinBlockElementsInSameParent( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::" + "ComputeRangesToJoinBlockElementsInSameParent() failed"); + return rv; + } + case Mode::DeleteContentInRanges: { + nsresult rv = ComputeRangesToDeleteContentInRanges( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::" + "ComputeRangesToDeleteContentInRanges() failed"); + return rv; + } + case Mode::DeleteNonCollapsedRanges: { + nsresult rv = ComputeRangesToDeleteNonCollapsedRanges( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete, + aSelectionWasCollapsed, aEditingHost); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::" + "ComputeRangesToDeleteNonCollapsedRanges() failed"); + return rv; + } + case Mode::NotInitialized: + MOZ_ASSERT_UNREACHABLE( + "Call ComputeRangesToDelete() after calling a preparation " + "method"); + return NS_ERROR_NOT_INITIALIZED; + } + return NS_ERROR_NOT_INITIALIZED; + } + + nsIContent* GetLeafContentInOtherBlockElement() const { + MOZ_ASSERT(mMode == Mode::JoinOtherBlock); + return mLeafContentInOtherBlock; + } + + private: + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + HandleDeleteAtCurrentBlockBoundary( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + const EditorDOMPoint& aCaretPoint, const Element& aEditingHost); + nsresult ComputeRangesToDeleteAtCurrentBlockBoundary( + const HTMLEditor& aHTMLEditor, const EditorDOMPoint& aCaretPoint, + AutoRangeArray& aRangesToDelete, const Element& aEditingHost) const; + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + HandleDeleteAtOtherBlockBoundary(HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + const EditorDOMPoint& aCaretPoint, + AutoRangeArray& aRangesToDelete, + const Element& aEditingHost); + // FYI: This method may modify selection, but it won't cause running + // script because of `AutoHideSelectionChanges` which blocks + // selection change listeners and the selection change event + // dispatcher. + MOZ_CAN_RUN_SCRIPT_BOUNDARY nsresult + ComputeRangesToDeleteAtOtherBlockBoundary( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + const EditorDOMPoint& aCaretPoint, AutoRangeArray& aRangesToDelete, + const Element& aEditingHost) const; + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + JoinBlockElementsInSameParent( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + AutoRangeArray& aRangesToDelete, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost); + nsresult ComputeRangesToJoinBlockElementsInSameParent( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete) const; + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + DeleteBRElement(HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + const Element& aEditingHost); + nsresult ComputeRangesToDeleteBRElement( + AutoRangeArray& aRangesToDelete) const; + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + DeleteContentInRanges(HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + AutoRangeArray& aRangesToDelete); + nsresult ComputeRangesToDeleteContentInRanges( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete) const; + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + HandleDeleteNonCollapsedRanges( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + AutoRangeArray& aRangesToDelete, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost); + nsresult ComputeRangesToDeleteNonCollapsedRanges( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost) const; + + /** + * JoinNodesDeepWithTransaction() joins aLeftNode and aRightNode "deeply". + * First, they are joined simply, then, new right node is assumed as the + * child at length of the left node before joined and new left node is + * assumed as its previous sibling. Then, they will be joined again. + * And then, these steps are repeated. + * + * @param aLeftContent The node which will be removed form the tree. + * @param aRightContent The node which will be inserted the contents of + * aRightContent. + * @return The point of the first child of the last right + * node. The result is always set if this succeeded. + */ + MOZ_CAN_RUN_SCRIPT Result<EditorDOMPoint, nsresult> + JoinNodesDeepWithTransaction(HTMLEditor& aHTMLEditor, + nsIContent& aLeftContent, + nsIContent& aRightContent); + + /** + * DeleteNodesEntirelyInRangeButKeepTableStructure() removes nodes which are + * entirely in aRange. Howevers, if some nodes are part of a table, + * removes all children of them instead. I.e., this does not make damage to + * table structure at the range, but may remove table entirely if it's + * in the range. + * + * @return true if inclusive ancestor block elements at + * start and end of the range should be joined. + */ + MOZ_CAN_RUN_SCRIPT Result<bool, nsresult> + DeleteNodesEntirelyInRangeButKeepTableStructure( + HTMLEditor& aHTMLEditor, nsRange& aRange, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed); + bool NeedsToJoinNodesAfterDeleteNodesEntirelyInRangeButKeepTableStructure( + const HTMLEditor& aHTMLEditor, + const nsTArray<OwningNonNull<nsIContent>>& aArrayOfContents, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed) + const; + Result<bool, nsresult> + ComputeRangesToDeleteNodesEntirelyInRangeButKeepTableStructure( + const HTMLEditor& aHTMLEditor, nsRange& aRange, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed) + const; + + /** + * DeleteContentButKeepTableStructure() removes aContent if it's an element + * which is part of a table structure. If it's a part of table structure, + * removes its all children recursively. I.e., this may delete all of a + * table, but won't break table structure partially. + * + * @param aContent The content which or whose all children should + * be removed. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT nsresult + DeleteContentButKeepTableStructure(HTMLEditor& aHTMLEditor, + nsIContent& aContent); + + /** + * DeleteTextAtStartAndEndOfRange() removes text if start and/or end of + * aRange is in a text node. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT nsresult + DeleteTextAtStartAndEndOfRange(HTMLEditor& aHTMLEditor, nsRange& aRange); + + class MOZ_STACK_CLASS AutoInclusiveAncestorBlockElementsJoiner final { + public: + AutoInclusiveAncestorBlockElementsJoiner() = delete; + AutoInclusiveAncestorBlockElementsJoiner( + nsIContent& aInclusiveDescendantOfLeftBlockElement, + nsIContent& aInclusiveDescendantOfRightBlockElement) + : mInclusiveDescendantOfLeftBlockElement( + aInclusiveDescendantOfLeftBlockElement), + mInclusiveDescendantOfRightBlockElement( + aInclusiveDescendantOfRightBlockElement), + mCanJoinBlocks(false), + mFallbackToDeleteLeafContent(false) {} + + bool IsSet() const { return mLeftBlockElement && mRightBlockElement; } + bool IsSameBlockElement() const { + return mLeftBlockElement && mLeftBlockElement == mRightBlockElement; + } + + const EditorDOMPoint& PointRefToPutCaret() const { + return mPointToPutCaret; + } + + /** + * Prepare for joining inclusive ancestor block elements. When this + * returns false, the deletion should be canceled. + */ + Result<bool, nsresult> Prepare(const HTMLEditor& aHTMLEditor, + const Element& aEditingHost); + + /** + * When this returns true, this can join the blocks with `Run()`. + */ + bool CanJoinBlocks() const { return mCanJoinBlocks; } + + /** + * When this returns true, `Run()` must return "ignored" so that + * caller can skip calling `Run()`. This is available only when + * `CanJoinBlocks()` returns `true`. + * TODO: This should be merged into `CanJoinBlocks()` in the future. + */ + bool ShouldDeleteLeafContentInstead() const { + MOZ_ASSERT(CanJoinBlocks()); + return mFallbackToDeleteLeafContent; + } + + /** + * ComputeRangesToDelete() extends aRangesToDelete includes the element + * boundaries between joining blocks. If they won't be joined, this + * collapses the range to aCaretPoint. + */ + nsresult ComputeRangesToDelete(const HTMLEditor& aHTMLEditor, + const EditorDOMPoint& aCaretPoint, + AutoRangeArray& aRangesToDelete) const; + + /** + * Join inclusive ancestor block elements which are found by preceding + * Preare() call. + * The right element is always joined to the left element. + * If the elements are the same type and not nested within each other, + * JoinEditableNodesWithTransaction() is called (example, joining two + * list items together into one). + * If the elements are not the same type, or one is a descendant of the + * other, we instead destroy the right block placing its children into + * left block. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> Run( + HTMLEditor& aHTMLEditor, const Element& aEditingHost); + + private: + /** + * This method returns true when + * `MergeFirstLineOfRightBlockElementIntoDescendantLeftBlockElement()`, + * `MergeFirstLineOfRightBlockElementIntoAncestorLeftBlockElement()` and + * `MergeFirstLineOfRightBlockElementIntoLeftBlockElement()` handle it + * with the `if` block of their main blocks. + */ + bool CanMergeLeftAndRightBlockElements() const { + if (!IsSet()) { + return false; + } + // `MergeFirstLineOfRightBlockElementIntoDescendantLeftBlockElement()` + if (mPointContainingTheOtherBlockElement.GetContainer() == + mRightBlockElement) { + return mNewListElementTagNameOfRightListElement.isSome(); + } + // `MergeFirstLineOfRightBlockElementIntoAncestorLeftBlockElement()` + if (mPointContainingTheOtherBlockElement.GetContainer() == + mLeftBlockElement) { + return mNewListElementTagNameOfRightListElement.isSome() && + !mRightBlockElement->GetChildCount(); + } + MOZ_ASSERT(!mPointContainingTheOtherBlockElement.IsSet()); + // `MergeFirstLineOfRightBlockElementIntoLeftBlockElement()` + return mNewListElementTagNameOfRightListElement.isSome() || + mLeftBlockElement->NodeInfo()->NameAtom() == + mRightBlockElement->NodeInfo()->NameAtom(); + } + + OwningNonNull<nsIContent> mInclusiveDescendantOfLeftBlockElement; + OwningNonNull<nsIContent> mInclusiveDescendantOfRightBlockElement; + RefPtr<Element> mLeftBlockElement; + RefPtr<Element> mRightBlockElement; + Maybe<nsAtom*> mNewListElementTagNameOfRightListElement; + EditorDOMPoint mPointContainingTheOtherBlockElement; + EditorDOMPoint mPointToPutCaret; + RefPtr<dom::HTMLBRElement> mPrecedingInvisibleBRElement; + bool mCanJoinBlocks; + bool mFallbackToDeleteLeafContent; + }; // HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + // AutoInclusiveAncestorBlockElementsJoiner + + enum class Mode { + NotInitialized, + JoinCurrentBlock, + JoinOtherBlock, + JoinBlocksInSameParent, + DeleteBRElement, + DeleteContentInRanges, + DeleteNonCollapsedRanges, + }; + AutoDeleteRangesHandler* mDeleteRangesHandler; + const AutoDeleteRangesHandler& mDeleteRangesHandlerConst; + nsCOMPtr<nsIContent> mLeftContent; + nsCOMPtr<nsIContent> mRightContent; + nsCOMPtr<nsIContent> mLeafContentInOtherBlock; + // mSkippedInvisibleContents stores all content nodes which are skipped at + // scanning mLeftContent and mRightContent. The content nodes should be + // removed at deletion. + AutoTArray<OwningNonNull<nsIContent>, 8> mSkippedInvisibleContents; + RefPtr<dom::HTMLBRElement> mBRElement; + Mode mMode = Mode::NotInitialized; + }; // HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner + + class MOZ_STACK_CLASS AutoEmptyBlockAncestorDeleter final { + public: + /** + * ScanEmptyBlockInclusiveAncestor() scans an inclusive ancestor element + * which is empty and a block element. Then, stores the result and + * returns the found empty block element. + * + * @param aHTMLEditor The HTMLEditor. + * @param aStartContent Start content to look for empty ancestors. + */ + [[nodiscard]] Element* ScanEmptyBlockInclusiveAncestor( + const HTMLEditor& aHTMLEditor, nsIContent& aStartContent); + + /** + * ComputeTargetRanges() computes "target ranges" for deleting + * `mEmptyInclusiveAncestorBlockElement`. + */ + nsresult ComputeTargetRanges(const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + const Element& aEditingHost, + AutoRangeArray& aRangesToDelete) const; + + /** + * Deletes found empty block element by `ScanEmptyBlockInclusiveAncestor()`. + * If found one is a list item element, calls + * `MaybeInsertBRElementBeforeEmptyListItemElement()` before deleting + * the list item element. + * If found empty ancestor is not a list item element, + * `GetNewCaretPosition()` will be called to determine new caret position. + * Finally, removes the empty block ancestor. + * + * @param aHTMLEditor The HTMLEditor. + * @param aDirectionAndAmount If found empty ancestor block is a list item + * element, this is ignored. Otherwise: + * - If eNext, eNextWord or eToEndOfLine, + * collapse Selection to after found empty + * ancestor. + * - If ePrevious, ePreviousWord or + * eToBeginningOfLine, collapse Selection to + * end of previous editable node. + * - Otherwise, eNone is allowed but does + * nothing. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> Run( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount); + + private: + /** + * MaybeReplaceSubListWithNewListItem() replaces + * mEmptyInclusiveAncestorBlockElement with new list item element + * (containing <br>) if: + * - mEmptyInclusiveAncestorBlockElement is a list element + * - The parent of mEmptyInclusiveAncestorBlockElement is a list element + * - The parent becomes empty after deletion + * If this does not perform the replacement, returns "ignored". + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<EditActionResult, nsresult> + MaybeReplaceSubListWithNewListItem(HTMLEditor& aHTMLEditor); + + /** + * MaybeInsertBRElementBeforeEmptyListItemElement() inserts a `<br>` element + * if `mEmptyInclusiveAncestorBlockElement` is a list item element which + * is first editable element in its parent, and its grand parent is not a + * list element, inserts a `<br>` element before the empty list item. + */ + [[nodiscard]] MOZ_CAN_RUN_SCRIPT Result<RefPtr<Element>, nsresult> + MaybeInsertBRElementBeforeEmptyListItemElement(HTMLEditor& aHTMLEditor); + + /** + * GetNewCaretPosition() returns new caret position after deleting + * `mEmptyInclusiveAncestorBlockElement`. + */ + [[nodiscard]] Result<CaretPoint, nsresult> GetNewCaretPosition( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount) const; + + RefPtr<Element> mEmptyInclusiveAncestorBlockElement; + }; // HTMLEditor::AutoDeleteRangesHandler::AutoEmptyBlockAncestorDeleter + + const AutoDeleteRangesHandler* const mParent; + nsIEditor::EDirection mOriginalDirectionAndAmount; + nsIEditor::EStripWrappers mOriginalStripWrappers; +}; // HTMLEditor::AutoDeleteRangesHandler + +nsresult HTMLEditor::ComputeTargetRanges( + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete) const { + MOZ_ASSERT(IsEditActionDataAvailable()); + + Element* editingHost = ComputeEditingHost(); + if (!editingHost) { + aRangesToDelete.RemoveAllRanges(); + return NS_ERROR_EDITOR_NO_EDITABLE_RANGE; + } + + // First check for table selection mode. If so, hand off to table editor. + SelectedTableCellScanner scanner(aRangesToDelete); + if (scanner.IsInTableCellSelectionMode()) { + // If it's in table cell selection mode, we'll delete all childen in + // the all selected table cell elements, + if (scanner.ElementsRef().Length() == aRangesToDelete.Ranges().Length()) { + return NS_OK; + } + // but will ignore all ranges which does not select a table cell. + size_t removedRanges = 0; + for (size_t i = 1; i < scanner.ElementsRef().Length(); i++) { + if (HTMLEditUtils::GetTableCellElementIfOnlyOneSelected( + aRangesToDelete.Ranges()[i - removedRanges]) != + scanner.ElementsRef()[i]) { + // XXX Need to manage anchor-focus range too! + aRangesToDelete.Ranges().RemoveElementAt(i - removedRanges); + removedRanges++; + } + } + return NS_OK; + } + + aRangesToDelete.EnsureOnlyEditableRanges(*editingHost); + if (aRangesToDelete.Ranges().IsEmpty()) { + NS_WARNING( + "There is no range which we can delete entire of or around the caret"); + return NS_ERROR_EDITOR_NO_EDITABLE_RANGE; + } + AutoDeleteRangesHandler deleteHandler; + // Should we delete target ranges which cannot delete actually? + nsresult rv = deleteHandler.ComputeRangesToDelete( + *this, aDirectionAndAmount, aRangesToDelete, *editingHost); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::ComputeRangesToDelete() failed"); + return rv; +} + +Result<EditActionResult, nsresult> HTMLEditor::HandleDeleteSelection( + nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers) { + MOZ_ASSERT(IsEditActionDataAvailable()); + MOZ_ASSERT(aStripWrappers == nsIEditor::eStrip || + aStripWrappers == nsIEditor::eNoStrip); + + if (MOZ_UNLIKELY(!SelectionRef().RangeCount())) { + return Err(NS_ERROR_EDITOR_NO_EDITABLE_RANGE); + } + + RefPtr<Element> editingHost = ComputeEditingHost(); + if (MOZ_UNLIKELY(!editingHost)) { + return Err(NS_ERROR_EDITOR_NO_EDITABLE_RANGE); + } + + // Remember that we did a selection deletion. Used by + // CreateStyleForInsertText() + TopLevelEditSubActionDataRef().mDidDeleteSelection = true; + + if (MOZ_UNLIKELY(IsEmpty())) { + return EditActionResult::CanceledResult(); + } + + // First check for table selection mode. If so, hand off to table editor. + if (HTMLEditUtils::IsInTableCellSelectionMode(SelectionRef())) { + nsresult rv = DeleteTableCellContentsWithTransaction(); + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::DeleteTableCellContentsWithTransaction() failed"); + return Err(rv); + } + return EditActionResult::HandledResult(); + } + + AutoRangeArray rangesToDelete(SelectionRef()); + rangesToDelete.EnsureOnlyEditableRanges(*editingHost); + if (MOZ_UNLIKELY(rangesToDelete.Ranges().IsEmpty())) { + NS_WARNING( + "There is no range which we can delete entire the ranges or around the " + "caret"); + return Err(NS_ERROR_EDITOR_NO_EDITABLE_RANGE); + } + AutoDeleteRangesHandler deleteHandler; + Result<EditActionResult, nsresult> result = deleteHandler.Run( + *this, aDirectionAndAmount, aStripWrappers, rangesToDelete, *editingHost); + if (MOZ_UNLIKELY(result.isErr()) || result.inspect().Canceled()) { + NS_WARNING_ASSERTION(result.isOk(), + "AutoDeleteRangesHandler::Run() failed"); + return result; + } + + // XXX At here, selection may have no range because of mutation event + // listeners can do anything so that we should just return NS_OK instead + // of returning error. + const auto atNewStartOfSelection = + GetFirstSelectionStartPoint<EditorDOMPoint>(); + if (NS_WARN_IF(!atNewStartOfSelection.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + if (atNewStartOfSelection.IsInContentNode()) { + nsresult rv = DeleteMostAncestorMailCiteElementIfEmpty( + MOZ_KnownLive(*atNewStartOfSelection.ContainerAs<nsIContent>())); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::DeleteMostAncestorMailCiteElementIfEmpty() failed"); + return Err(rv); + } + } + return EditActionResult::HandledResult(); +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::ComputeRangesToDelete( + const HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.Ranges().IsEmpty()); + + mOriginalDirectionAndAmount = aDirectionAndAmount; + mOriginalStripWrappers = nsIEditor::eNoStrip; + + if (aHTMLEditor.mPaddingBRElementForEmptyEditor) { + nsresult rv = aRangesToDelete.Collapse( + EditorRawDOMPoint(aHTMLEditor.mPaddingBRElementForEmptyEditor)); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "AutoRangeArray::Collapse() failed"); + return rv; + } + + SelectionWasCollapsed selectionWasCollapsed = aRangesToDelete.IsCollapsed() + ? SelectionWasCollapsed::Yes + : SelectionWasCollapsed::No; + if (selectionWasCollapsed == SelectionWasCollapsed::Yes) { + const auto startPoint = + aRangesToDelete.GetFirstRangeStartPoint<EditorDOMPoint>(); + if (NS_WARN_IF(!startPoint.IsSet())) { + return NS_ERROR_FAILURE; + } + RefPtr<Element> editingHost = aHTMLEditor.ComputeEditingHost(); + if (NS_WARN_IF(!editingHost)) { + return NS_ERROR_FAILURE; + } + if (startPoint.IsInContentNode()) { + AutoEmptyBlockAncestorDeleter deleter; + if (deleter.ScanEmptyBlockInclusiveAncestor( + aHTMLEditor, *startPoint.ContainerAs<nsIContent>())) { + nsresult rv = deleter.ComputeTargetRanges( + aHTMLEditor, aDirectionAndAmount, *editingHost, aRangesToDelete); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoEmptyBlockAncestorDeleter::ComputeTargetRanges() failed"); + return rv; + } + } + + // We shouldn't update caret bidi level right now, but we need to check + // whether the deletion will be canceled or not. + AutoCaretBidiLevelManager bidiLevelManager(aHTMLEditor, aDirectionAndAmount, + startPoint); + if (bidiLevelManager.Failed()) { + NS_WARNING( + "EditorBase::AutoCaretBidiLevelManager failed to initialize itself"); + return NS_ERROR_FAILURE; + } + if (bidiLevelManager.Canceled()) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + // AutoRangeArray::ExtendAnchorFocusRangeFor() will use `nsFrameSelection` + // to extend the range for deletion. But if focus event doesn't receive + // yet, ancestor isn't set. So we must set root element of editor to + // ancestor temporarily. + AutoSetTemporaryAncestorLimiter autoSetter( + aHTMLEditor, aHTMLEditor.SelectionRef(), *startPoint.GetContainer(), + &aRangesToDelete); + + Result<nsIEditor::EDirection, nsresult> extendResult = + aRangesToDelete.ExtendAnchorFocusRangeFor(aHTMLEditor, + aDirectionAndAmount); + if (extendResult.isErr()) { + NS_WARNING("AutoRangeArray::ExtendAnchorFocusRangeFor() failed"); + return extendResult.unwrapErr(); + } + + // For compatibility with other browsers, we should set target ranges + // to start from and/or end after an atomic content rather than start + // from preceding text node end nor end at following text node start. + Result<bool, nsresult> shrunkenResult = + aRangesToDelete.ShrinkRangesIfStartFromOrEndAfterAtomicContent( + aHTMLEditor, aDirectionAndAmount, + AutoRangeArray::IfSelectingOnlyOneAtomicContent::Collapse, + editingHost); + if (shrunkenResult.isErr()) { + NS_WARNING( + "AutoRangeArray::ShrinkRangesIfStartFromOrEndAfterAtomicContent() " + "failed"); + return shrunkenResult.unwrapErr(); + } + + if (!shrunkenResult.inspect() || !aRangesToDelete.IsCollapsed()) { + aDirectionAndAmount = extendResult.unwrap(); + } + + if (aDirectionAndAmount == nsIEditor::eNone) { + MOZ_ASSERT(aRangesToDelete.Ranges().Length() == 1); + if (!CanFallbackToDeleteRangesWithTransaction(aRangesToDelete)) { + // XXX In this case, do we need to modify the range again? + return NS_SUCCESS_DOM_NO_OPERATION; + } + nsresult rv = FallbackToComputeRangesToDeleteRangesWithTransaction( + aHTMLEditor, aRangesToDelete); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::" + "FallbackToComputeRangesToDeleteRangesWithTransaction() failed"); + return rv; + } + + if (aRangesToDelete.IsCollapsed()) { + const auto caretPoint = + aRangesToDelete.GetFirstRangeStartPoint<EditorDOMPoint>(); + if (MOZ_UNLIKELY(NS_WARN_IF(!caretPoint.IsInContentNode()))) { + return NS_ERROR_FAILURE; + } + if (!EditorUtils::IsEditableContent(*caretPoint.ContainerAs<nsIContent>(), + EditorType::HTML)) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + WSRunScanner wsRunScannerAtCaret(editingHost, caretPoint); + WSScanResult scanFromCaretPointResult = + aDirectionAndAmount == nsIEditor::eNext + ? wsRunScannerAtCaret.ScanNextVisibleNodeOrBlockBoundaryFrom( + caretPoint) + : wsRunScannerAtCaret.ScanPreviousVisibleNodeOrBlockBoundaryFrom( + caretPoint); + if (scanFromCaretPointResult.Failed()) { + NS_WARNING( + "WSRunScanner::Scan(Next|Previous)VisibleNodeOrBlockBoundaryFrom() " + "failed"); + return NS_ERROR_FAILURE; + } + if (!scanFromCaretPointResult.GetContent()) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + if (scanFromCaretPointResult.ReachedBRElement()) { + if (scanFromCaretPointResult.BRElementPtr() == + wsRunScannerAtCaret.GetEditingHost()) { + return NS_OK; + } + if (!EditorUtils::IsEditableContent( + *scanFromCaretPointResult.BRElementPtr(), EditorType::HTML)) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + if (HTMLEditUtils::IsInvisibleBRElement( + *scanFromCaretPointResult.BRElementPtr())) { + EditorDOMPoint newCaretPosition = + aDirectionAndAmount == nsIEditor::eNext + ? EditorDOMPoint::After( + *scanFromCaretPointResult.BRElementPtr()) + : EditorDOMPoint(scanFromCaretPointResult.BRElementPtr()); + if (NS_WARN_IF(!newCaretPosition.IsSet())) { + return NS_ERROR_FAILURE; + } + AutoHideSelectionChanges blockSelectionListeners( + aHTMLEditor.SelectionRef()); + nsresult rv = aHTMLEditor.CollapseSelectionTo(newCaretPosition); + if (MOZ_UNLIKELY(NS_FAILED(rv))) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return NS_ERROR_FAILURE; + } + if (NS_WARN_IF(!aHTMLEditor.SelectionRef().RangeCount())) { + return NS_ERROR_UNEXPECTED; + } + aRangesToDelete.Initialize(aHTMLEditor.SelectionRef()); + AutoDeleteRangesHandler anotherHandler(this); + rv = anotherHandler.ComputeRangesToDelete( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete, aEditingHost); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "Recursive AutoDeleteRangesHandler::ComputeRangesToDelete() " + "failed"); + + rv = aHTMLEditor.CollapseSelectionTo(caretPoint); + 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), + "EditorBase::CollapseSelectionTo() failed to " + "restore original selection, but ignored"); + + MOZ_ASSERT(aRangesToDelete.Ranges().Length() == 1); + // If the range is collapsed, there is no content which should + // be removed together. In this case, only the invisible `<br>` + // element should be selected. + if (aRangesToDelete.IsCollapsed()) { + nsresult rv = aRangesToDelete.SelectNode( + *scanFromCaretPointResult.BRElementPtr()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoRangeArray::SelectNode() failed"); + return rv; + } + + // Otherwise, extend the range to contain the invisible `<br>` + // element. + if (EditorRawDOMPoint(scanFromCaretPointResult.BRElementPtr()) + .IsBefore( + aRangesToDelete + .GetFirstRangeStartPoint<EditorRawDOMPoint>())) { + nsresult rv = aRangesToDelete.FirstRangeRef()->SetStartAndEnd( + EditorRawDOMPoint(scanFromCaretPointResult.BRElementPtr()) + .ToRawRangeBoundary(), + aRangesToDelete.FirstRangeRef()->EndRef()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "nsRange::SetStartAndEnd() failed"); + return rv; + } + if (aRangesToDelete.GetFirstRangeEndPoint<EditorRawDOMPoint>() + .IsBefore(EditorRawDOMPoint::After( + *scanFromCaretPointResult.BRElementPtr()))) { + nsresult rv = aRangesToDelete.FirstRangeRef()->SetStartAndEnd( + aRangesToDelete.FirstRangeRef()->StartRef(), + EditorRawDOMPoint::After( + *scanFromCaretPointResult.BRElementPtr()) + .ToRawRangeBoundary()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "nsRange::SetStartAndEnd() failed"); + return rv; + } + NS_WARNING("Was the invisible `<br>` element selected?"); + return NS_OK; + } + } + + nsresult rv = ComputeRangesToDeleteAroundCollapsedRanges( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete, + wsRunScannerAtCaret, scanFromCaretPointResult, aEditingHost); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::ComputeRangesToDeleteAroundCollapsedRanges(" + ") failed"); + return rv; + } + } + + nsresult rv = ComputeRangesToDeleteNonCollapsedRanges( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete, selectionWasCollapsed, + aEditingHost); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::" + "ComputeRangesToDeleteNonCollapsedRanges() failed"); + return rv; +} + +Result<EditActionResult, nsresult> HTMLEditor::AutoDeleteRangesHandler::Run( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, AutoRangeArray& aRangesToDelete, + const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(aStripWrappers == nsIEditor::eStrip || + aStripWrappers == nsIEditor::eNoStrip); + MOZ_ASSERT(!aRangesToDelete.Ranges().IsEmpty()); + + mOriginalDirectionAndAmount = aDirectionAndAmount; + mOriginalStripWrappers = aStripWrappers; + + if (MOZ_UNLIKELY(aHTMLEditor.IsEmpty())) { + return EditActionResult::CanceledResult(); + } + + // selectionWasCollapsed is used later to determine whether we should join + // blocks in HandleDeleteNonCollapsedRanges(). We don't really care about + // collapsed because it will be modified by + // AutoRangeArray::ExtendAnchorFocusRangeFor() later. + // AutoBlockElementsJoiner::AutoInclusiveAncestorBlockElementsJoiner should + // happen if the original selection is collapsed and the cursor is at the end + // of a block element, in which case + // AutoRangeArray::ExtendAnchorFocusRangeFor() would always make the selection + // not collapsed. + SelectionWasCollapsed selectionWasCollapsed = aRangesToDelete.IsCollapsed() + ? SelectionWasCollapsed::Yes + : SelectionWasCollapsed::No; + + if (selectionWasCollapsed == SelectionWasCollapsed::Yes) { + const auto startPoint = + aRangesToDelete.GetFirstRangeStartPoint<EditorDOMPoint>(); + if (NS_WARN_IF(!startPoint.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + + // If we are inside an empty block, delete it. + if (startPoint.IsInContentNode()) { +#ifdef DEBUG + nsMutationGuard debugMutation; +#endif // #ifdef DEBUG + AutoEmptyBlockAncestorDeleter deleter; + if (deleter.ScanEmptyBlockInclusiveAncestor( + aHTMLEditor, *startPoint.ContainerAs<nsIContent>())) { + Result<EditActionResult, nsresult> result = + deleter.Run(aHTMLEditor, aDirectionAndAmount); + if (MOZ_UNLIKELY(result.isErr()) || result.inspect().Handled()) { + NS_WARNING_ASSERTION(result.isOk(), + "AutoEmptyBlockAncestorDeleter::Run() failed"); + return result; + } + } + MOZ_ASSERT(!debugMutation.Mutated(0), + "AutoEmptyBlockAncestorDeleter shouldn't modify the DOM tree " + "if it returns not handled nor error"); + } + + // Test for distance between caret and text that will be deleted. + // Note that this call modifies `nsFrameSelection` without modifying + // `Selection`. However, it does not have problem for now because + // it'll be referred by `AutoRangeArray::ExtendAnchorFocusRangeFor()` + // before modifying `Selection`. + // XXX This looks odd. `ExtendAnchorFocusRangeFor()` will extend + // anchor-focus range, but here refers the first range. + AutoCaretBidiLevelManager bidiLevelManager(aHTMLEditor, aDirectionAndAmount, + startPoint); + if (MOZ_UNLIKELY(bidiLevelManager.Failed())) { + NS_WARNING( + "EditorBase::AutoCaretBidiLevelManager failed to initialize itself"); + return Err(NS_ERROR_FAILURE); + } + bidiLevelManager.MaybeUpdateCaretBidiLevel(aHTMLEditor); + if (bidiLevelManager.Canceled()) { + return EditActionResult::CanceledResult(); + } + + // AutoRangeArray::ExtendAnchorFocusRangeFor() will use `nsFrameSelection` + // to extend the range for deletion. But if focus event doesn't receive + // yet, ancestor isn't set. So we must set root element of editor to + // ancestor temporarily. + AutoSetTemporaryAncestorLimiter autoSetter( + aHTMLEditor, aHTMLEditor.SelectionRef(), *startPoint.GetContainer(), + &aRangesToDelete); + + // Calling `ExtendAnchorFocusRangeFor()` and + // `ShrinkRangesIfStartFromOrEndAfterAtomicContent()` may move caret to + // the container of deleting atomic content. However, it may be different + // from the original caret's container. The original caret container may + // be important to put caret after deletion so that let's cache the + // original position. + Maybe<EditorDOMPoint> caretPoint; + if (aRangesToDelete.IsCollapsed() && !aRangesToDelete.Ranges().IsEmpty()) { + caretPoint = + Some(aRangesToDelete.GetFirstRangeStartPoint<EditorDOMPoint>()); + if (NS_WARN_IF(!caretPoint.ref().IsInContentNode())) { + return Err(NS_ERROR_FAILURE); + } + } + + Result<nsIEditor::EDirection, nsresult> extendResult = + aRangesToDelete.ExtendAnchorFocusRangeFor(aHTMLEditor, + aDirectionAndAmount); + if (MOZ_UNLIKELY(extendResult.isErr())) { + NS_WARNING("AutoRangeArray::ExtendAnchorFocusRangeFor() failed"); + return extendResult.propagateErr(); + } + if (caretPoint.isSome() && + MOZ_UNLIKELY(!caretPoint.ref().IsSetAndValid())) { + NS_WARNING("The caret position became invalid"); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + // If there is only one range and it selects an atomic content, we should + // delete it with collapsed range path for making consistent behavior + // between both cases, the content is selected case and caret is at it or + // after it case. + Result<bool, nsresult> shrunkenResult = + aRangesToDelete.ShrinkRangesIfStartFromOrEndAfterAtomicContent( + aHTMLEditor, aDirectionAndAmount, + AutoRangeArray::IfSelectingOnlyOneAtomicContent::Collapse, + &aEditingHost); + if (MOZ_UNLIKELY(shrunkenResult.isErr())) { + NS_WARNING( + "AutoRangeArray::ShrinkRangesIfStartFromOrEndAfterAtomicContent() " + "failed"); + return shrunkenResult.propagateErr(); + } + + if (!shrunkenResult.inspect() || !aRangesToDelete.IsCollapsed()) { + aDirectionAndAmount = extendResult.unwrap(); + } + + if (aDirectionAndAmount == nsIEditor::eNone) { + MOZ_ASSERT(aRangesToDelete.Ranges().Length() == 1); + if (!CanFallbackToDeleteRangesWithTransaction(aRangesToDelete)) { + return EditActionResult::IgnoredResult(); + } + Result<CaretPoint, nsresult> caretPointOrError = + FallbackToDeleteRangesWithTransaction(aHTMLEditor, aRangesToDelete); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "AutoDeleteRangesHandler::FallbackToDeleteRangesWithTransaction() " + "failed"); + } + nsresult rv = caretPointOrError.inspect().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + // Don't return "ignored" to avoid to fall it back to delete ranges + // recursively. + return EditActionResult::HandledResult(); + } + + if (aRangesToDelete.IsCollapsed()) { + // Use the original caret position for handling the deletion around + // collapsed range because the container may be different from the + // new collapsed position's container. + if (!EditorUtils::IsEditableContent( + *caretPoint.ref().ContainerAs<nsIContent>(), EditorType::HTML)) { + return EditActionResult::CanceledResult(); + } + WSRunScanner wsRunScannerAtCaret(&aEditingHost, caretPoint.ref()); + WSScanResult scanFromCaretPointResult = + aDirectionAndAmount == nsIEditor::eNext + ? wsRunScannerAtCaret.ScanNextVisibleNodeOrBlockBoundaryFrom( + caretPoint.ref()) + : wsRunScannerAtCaret.ScanPreviousVisibleNodeOrBlockBoundaryFrom( + caretPoint.ref()); + if (MOZ_UNLIKELY(scanFromCaretPointResult.Failed())) { + NS_WARNING( + "WSRunScanner::Scan(Next|Previous)VisibleNodeOrBlockBoundaryFrom() " + "failed"); + return Err(NS_ERROR_FAILURE); + } + if (!scanFromCaretPointResult.GetContent()) { + return EditActionResult::CanceledResult(); + } + // Short circuit for invisible breaks. delete them and recurse. + if (scanFromCaretPointResult.ReachedBRElement()) { + if (scanFromCaretPointResult.BRElementPtr() == &aEditingHost) { + return EditActionResult::HandledResult(); + } + if (!EditorUtils::IsEditableContent( + *scanFromCaretPointResult.BRElementPtr(), EditorType::HTML)) { + return EditActionResult::CanceledResult(); + } + if (HTMLEditUtils::IsInvisibleBRElement( + *scanFromCaretPointResult.BRElementPtr())) { + // TODO: We should extend the range to delete again before/after + // the caret point and use `HandleDeleteNonCollapsedRanges()` + // instead after we would create delete range computation + // method at switching to the new white-space normalizer. + Result<CaretPoint, nsresult> caretPointOrError = + WhiteSpaceVisibilityKeeper:: + DeleteContentNodeAndJoinTextNodesAroundIt( + aHTMLEditor, + MOZ_KnownLive(*scanFromCaretPointResult.BRElementPtr()), + caretPoint.ref(), aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "WhiteSpaceVisibilityKeeper::" + "DeleteContentNodeAndJoinTextNodesAroundIt() failed"); + return caretPointOrError.propagateErr(); + } + if (caretPointOrError.inspect().HasCaretPointSuggestion()) { + caretPoint = Some(caretPointOrError.unwrap().UnwrapCaretPoint()); + } + if (NS_WARN_IF(!caretPoint->IsSetAndValid())) { + return Err(NS_ERROR_FAILURE); + } + AutoRangeArray rangesToDelete(caretPoint.ref()); + if (aHTMLEditor.MayHaveMutationEventListeners( + NS_EVENT_BITS_MUTATION_SUBTREEMODIFIED | + NS_EVENT_BITS_MUTATION_NODEREMOVED | + NS_EVENT_BITS_MUTATION_NODEREMOVEDFROMDOCUMENT)) { + // Let's check whether there is new invisible `<br>` element + // for avoiding infinite recursive calls. + WSRunScanner wsRunScannerAtCaret(&aEditingHost, caretPoint.ref()); + WSScanResult scanFromCaretPointResult = + aDirectionAndAmount == nsIEditor::eNext + ? wsRunScannerAtCaret + .ScanNextVisibleNodeOrBlockBoundaryFrom( + caretPoint.ref()) + : wsRunScannerAtCaret + .ScanPreviousVisibleNodeOrBlockBoundaryFrom( + caretPoint.ref()); + if (MOZ_UNLIKELY(scanFromCaretPointResult.Failed())) { + NS_WARNING( + "WSRunScanner::Scan(Next|Previous)" + "VisibleNodeOrBlockBoundaryFrom() failed"); + return Err(NS_ERROR_FAILURE); + } + if (MOZ_UNLIKELY( + scanFromCaretPointResult.ReachedInvisibleBRElement())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + } + AutoDeleteRangesHandler anotherHandler(this); + Result<EditActionResult, nsresult> result = + anotherHandler.Run(aHTMLEditor, aDirectionAndAmount, + aStripWrappers, rangesToDelete, aEditingHost); + NS_WARNING_ASSERTION( + result.isOk(), "Recursive AutoDeleteRangesHandler::Run() failed"); + return result; + } + } + + Result<EditActionResult, nsresult> result = + HandleDeleteAroundCollapsedRanges( + aHTMLEditor, aDirectionAndAmount, aStripWrappers, aRangesToDelete, + wsRunScannerAtCaret, scanFromCaretPointResult, aEditingHost); + NS_WARNING_ASSERTION(result.isOk(), + "AutoDeleteRangesHandler::" + "HandleDeleteAroundCollapsedRanges() failed"); + return result; + } + } + + Result<EditActionResult, nsresult> result = HandleDeleteNonCollapsedRanges( + aHTMLEditor, aDirectionAndAmount, aStripWrappers, aRangesToDelete, + selectionWasCollapsed, aEditingHost); + NS_WARNING_ASSERTION( + result.isOk(), + "AutoDeleteRangesHandler::HandleDeleteNonCollapsedRanges() failed"); + return result; +} + +nsresult +HTMLEditor::AutoDeleteRangesHandler::ComputeRangesToDeleteAroundCollapsedRanges( + const HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, const WSRunScanner& aWSRunScannerAtCaret, + const WSScanResult& aScanFromCaretPointResult, + const Element& aEditingHost) const { + if (aScanFromCaretPointResult.InCollapsibleWhiteSpaces() || + aScanFromCaretPointResult.InNonCollapsibleCharacters() || + aScanFromCaretPointResult.ReachedPreformattedLineBreak()) { + nsresult rv = aRangesToDelete.Collapse( + aScanFromCaretPointResult.Point<EditorRawDOMPoint>()); + if (MOZ_UNLIKELY(NS_FAILED(rv))) { + NS_WARNING("AutoRangeArray::Collapse() failed"); + return NS_ERROR_FAILURE; + } + rv = ComputeRangesToDeleteTextAroundCollapsedRanges( + aDirectionAndAmount, aRangesToDelete, aEditingHost); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::" + "ComputeRangesToDeleteTextAroundCollapsedRanges() failed"); + return rv; + } + + if (aScanFromCaretPointResult.ReachedSpecialContent() || + aScanFromCaretPointResult.ReachedBRElement() || + aScanFromCaretPointResult.ReachedHRElement() || + aScanFromCaretPointResult.ReachedNonEditableOtherBlockElement()) { + if (aScanFromCaretPointResult.GetContent() == + aWSRunScannerAtCaret.GetEditingHost()) { + return NS_OK; + } + nsIContent* atomicContent = GetAtomicContentToDelete( + aDirectionAndAmount, aWSRunScannerAtCaret, aScanFromCaretPointResult); + if (!HTMLEditUtils::IsRemovableNode(*atomicContent)) { + NS_WARNING( + "AutoDeleteRangesHandler::GetAtomicContentToDelete() cannot find " + "removable atomic content"); + return NS_ERROR_FAILURE; + } + nsresult rv = ComputeRangesToDeleteAtomicContent( + aWSRunScannerAtCaret.GetEditingHost(), *atomicContent, aRangesToDelete); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::ComputeRangesToDeleteAtomicContent() failed"); + return rv; + } + + if (aScanFromCaretPointResult.ReachedOtherBlockElement()) { + if (NS_WARN_IF(!aScanFromCaretPointResult.GetContent()->IsElement())) { + return NS_ERROR_FAILURE; + } + AutoBlockElementsJoiner joiner(*this); + if (!joiner.PrepareToDeleteAtOtherBlockBoundary( + aHTMLEditor, aDirectionAndAmount, + *aScanFromCaretPointResult.ElementPtr(), + aWSRunScannerAtCaret.ScanStartRef(), aWSRunScannerAtCaret)) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + nsresult rv = joiner.ComputeRangesToDelete( + aHTMLEditor, aDirectionAndAmount, aWSRunScannerAtCaret.ScanStartRef(), + aRangesToDelete, aEditingHost); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::ComputeRangesToDelete() " + "failed (other block boundary)"); + return rv; + } + + if (aScanFromCaretPointResult.ReachedCurrentBlockBoundary()) { + if (NS_WARN_IF(!aScanFromCaretPointResult.GetContent()->IsElement())) { + return NS_ERROR_FAILURE; + } + AutoBlockElementsJoiner joiner(*this); + if (!joiner.PrepareToDeleteAtCurrentBlockBoundary( + aHTMLEditor, aDirectionAndAmount, + *aScanFromCaretPointResult.ElementPtr(), + aWSRunScannerAtCaret.ScanStartRef())) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + nsresult rv = joiner.ComputeRangesToDelete( + aHTMLEditor, aDirectionAndAmount, aWSRunScannerAtCaret.ScanStartRef(), + aRangesToDelete, aEditingHost); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::ComputeRangesToDelete() " + "failed (current block boundary)"); + return rv; + } + + return NS_OK; +} + +Result<EditActionResult, nsresult> +HTMLEditor::AutoDeleteRangesHandler::HandleDeleteAroundCollapsedRanges( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, AutoRangeArray& aRangesToDelete, + const WSRunScanner& aWSRunScannerAtCaret, + const WSScanResult& aScanFromCaretPointResult, + const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsTopLevelEditSubActionDataAvailable()); + MOZ_ASSERT(aRangesToDelete.IsCollapsed()); + MOZ_ASSERT(aDirectionAndAmount != nsIEditor::eNone); + MOZ_ASSERT(aWSRunScannerAtCaret.ScanStartRef().IsInContentNode()); + MOZ_ASSERT(EditorUtils::IsEditableContent( + *aWSRunScannerAtCaret.ScanStartRef().ContainerAs<nsIContent>(), + EditorType::HTML)); + + if (StaticPrefs::editor_white_space_normalization_blink_compatible()) { + if (aScanFromCaretPointResult.InCollapsibleWhiteSpaces() || + aScanFromCaretPointResult.InNonCollapsibleCharacters() || + aScanFromCaretPointResult.ReachedPreformattedLineBreak()) { + nsresult rv = aRangesToDelete.Collapse( + aScanFromCaretPointResult.Point<EditorRawDOMPoint>()); + if (NS_FAILED(rv)) { + NS_WARNING("AutoRangeArray::Collapse() failed"); + return Err(NS_ERROR_FAILURE); + } + Result<CaretPoint, nsresult> caretPointOrError = + HandleDeleteTextAroundCollapsedRanges( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete, aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "AutoDeleteRangesHandler::HandleDeleteTextAroundCollapsedRanges() " + "failed"); + return caretPointOrError.propagateErr(); + } + rv = caretPointOrError.unwrap().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPoint() failed, but ignored"); + return EditActionResult::HandledResult(); + } + } + + if (aScanFromCaretPointResult.InCollapsibleWhiteSpaces() || + aScanFromCaretPointResult.ReachedPreformattedLineBreak()) { + Result<CaretPoint, nsresult> caretPointOrError = + HandleDeleteCollapsedSelectionAtWhiteSpaces( + aHTMLEditor, aDirectionAndAmount, + aWSRunScannerAtCaret.ScanStartRef(), aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "AutoDeleteRangesHandler::" + "HandleDeleteCollapsedSelectionAtWhiteSpaces() failed"); + return caretPointOrError.propagateErr(); + } + nsresult rv = caretPointOrError.unwrap().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + return EditActionResult::HandledResult(); + } + + if (aScanFromCaretPointResult.InNonCollapsibleCharacters()) { + if (NS_WARN_IF(!aScanFromCaretPointResult.GetContent()->IsText())) { + return Err(NS_ERROR_FAILURE); + } + Result<CaretPoint, nsresult> caretPointOrError = + HandleDeleteCollapsedSelectionAtVisibleChar( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete, + aScanFromCaretPointResult.Point<EditorDOMPoint>(), aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "AutoDeleteRangesHandler::" + "HandleDeleteCollapsedSelectionAtVisibleChar() failed"); + return caretPointOrError.propagateErr(); + } + nsresult rv = caretPointOrError.unwrap().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + return EditActionResult::HandledResult(); + } + + if (aScanFromCaretPointResult.ReachedSpecialContent() || + aScanFromCaretPointResult.ReachedBRElement() || + aScanFromCaretPointResult.ReachedHRElement() || + aScanFromCaretPointResult.ReachedNonEditableOtherBlockElement()) { + if (aScanFromCaretPointResult.GetContent() == &aEditingHost) { + return EditActionResult::HandledResult(); + } + nsCOMPtr<nsIContent> atomicContent = GetAtomicContentToDelete( + aDirectionAndAmount, aWSRunScannerAtCaret, aScanFromCaretPointResult); + if (MOZ_UNLIKELY(!HTMLEditUtils::IsRemovableNode(*atomicContent))) { + NS_WARNING( + "AutoDeleteRangesHandler::GetAtomicContentToDelete() cannot find " + "removable atomic content"); + return Err(NS_ERROR_FAILURE); + } + Result<CaretPoint, nsresult> caretPointOrError = HandleDeleteAtomicContent( + aHTMLEditor, *atomicContent, aWSRunScannerAtCaret.ScanStartRef(), + aWSRunScannerAtCaret, aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("AutoDeleteRangesHandler::HandleDeleteAtomicContent() failed"); + return caretPointOrError.propagateErr(); + } + nsresult rv = caretPointOrError.unwrap().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + return EditActionResult::HandledResult(); + } + + if (aScanFromCaretPointResult.ReachedOtherBlockElement()) { + if (NS_WARN_IF(!aScanFromCaretPointResult.GetContent()->IsElement())) { + return Err(NS_ERROR_FAILURE); + } + AutoBlockElementsJoiner joiner(*this); + if (!joiner.PrepareToDeleteAtOtherBlockBoundary( + aHTMLEditor, aDirectionAndAmount, + *aScanFromCaretPointResult.ElementPtr(), + aWSRunScannerAtCaret.ScanStartRef(), aWSRunScannerAtCaret)) { + return EditActionResult::CanceledResult(); + } + Result<EditActionResult, nsresult> result = joiner.Run( + aHTMLEditor, aDirectionAndAmount, aStripWrappers, + aWSRunScannerAtCaret.ScanStartRef(), aRangesToDelete, aEditingHost); + NS_WARNING_ASSERTION( + result.isOk(), + "AutoBlockElementsJoiner::Run() failed (other block boundary)"); + return result; + } + + if (aScanFromCaretPointResult.ReachedCurrentBlockBoundary()) { + if (NS_WARN_IF(!aScanFromCaretPointResult.GetContent()->IsElement())) { + return Err(NS_ERROR_FAILURE); + } + AutoBlockElementsJoiner joiner(*this); + if (!joiner.PrepareToDeleteAtCurrentBlockBoundary( + aHTMLEditor, aDirectionAndAmount, + *aScanFromCaretPointResult.ElementPtr(), + aWSRunScannerAtCaret.ScanStartRef())) { + return EditActionResult::CanceledResult(); + } + Result<EditActionResult, nsresult> result = joiner.Run( + aHTMLEditor, aDirectionAndAmount, aStripWrappers, + aWSRunScannerAtCaret.ScanStartRef(), aRangesToDelete, aEditingHost); + NS_WARNING_ASSERTION( + result.isOk(), + "AutoBlockElementsJoiner::Run() failed (current block boundary)"); + return result; + } + + MOZ_ASSERT_UNREACHABLE("New type of reached content hasn't been handled yet"); + return EditActionResult::IgnoredResult(); +} + +nsresult HTMLEditor::AutoDeleteRangesHandler:: + ComputeRangesToDeleteTextAroundCollapsedRanges( + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, const Element& aEditingHost) const { + MOZ_ASSERT(aDirectionAndAmount == nsIEditor::eNext || + aDirectionAndAmount == nsIEditor::ePrevious); + + const auto caretPosition = + aRangesToDelete.GetFirstRangeStartPoint<EditorDOMPoint>(); + MOZ_ASSERT(caretPosition.IsSetAndValid()); + if (MOZ_UNLIKELY(NS_WARN_IF(!caretPosition.IsInContentNode()))) { + return NS_ERROR_FAILURE; + } + + EditorDOMRangeInTexts rangeToDelete; + if (aDirectionAndAmount == nsIEditor::eNext) { + Result<EditorDOMRangeInTexts, nsresult> result = + WSRunScanner::GetRangeInTextNodesToForwardDeleteFrom(caretPosition, + aEditingHost); + if (result.isErr()) { + NS_WARNING( + "WSRunScanner::GetRangeInTextNodesToForwardDeleteFrom() failed"); + return result.unwrapErr(); + } + rangeToDelete = result.unwrap(); + if (!rangeToDelete.IsPositioned()) { + return NS_OK; // no range to delete, but consume it. + } + } else { + Result<EditorDOMRangeInTexts, nsresult> result = + WSRunScanner::GetRangeInTextNodesToBackspaceFrom(caretPosition, + aEditingHost); + if (result.isErr()) { + NS_WARNING("WSRunScanner::GetRangeInTextNodesToBackspaceFrom() failed"); + return result.unwrapErr(); + } + rangeToDelete = result.unwrap(); + if (!rangeToDelete.IsPositioned()) { + return NS_OK; // no range to delete, but consume it. + } + } + + nsresult rv = aRangesToDelete.SetStartAndEnd(rangeToDelete.StartRef(), + rangeToDelete.EndRef()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoArrayRanges::SetStartAndEnd() failed"); + return rv; +} + +Result<CaretPoint, nsresult> +HTMLEditor::AutoDeleteRangesHandler::HandleDeleteTextAroundCollapsedRanges( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(aDirectionAndAmount == nsIEditor::eNext || + aDirectionAndAmount == nsIEditor::ePrevious); + + nsresult rv = ComputeRangesToDeleteTextAroundCollapsedRanges( + aDirectionAndAmount, aRangesToDelete, aEditingHost); + if (NS_FAILED(rv)) { + return Err(NS_ERROR_FAILURE); + } + if (MOZ_UNLIKELY(aRangesToDelete.IsCollapsed())) { + return CaretPoint(EditorDOMPoint()); // no range to delete + } + + // FYI: rangeToDelete does not contain newly empty inline ancestors which + // are removed by DeleteTextAndNormalizeSurroundingWhiteSpaces(). + // So, if `getTargetRanges()` needs to include parent empty elements, + // we need to extend the range with + // HTMLEditUtils::GetMostDistantAncestorEditableEmptyInlineElement(). + EditorRawDOMRange rangeToDelete(aRangesToDelete.FirstRangeRef()); + if (MOZ_UNLIKELY(!rangeToDelete.IsInTextNodes())) { + NS_WARNING("The extended range to delete character was not in text nodes"); + return Err(NS_ERROR_FAILURE); + } + + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteTextAndNormalizeSurroundingWhiteSpaces( + rangeToDelete.StartRef().AsInText(), + rangeToDelete.EndRef().AsInText(), + TreatEmptyTextNodes::RemoveAllEmptyInlineAncestors, + aDirectionAndAmount == nsIEditor::eNext ? DeleteDirection::Forward + : DeleteDirection::Backward); + aHTMLEditor.TopLevelEditSubActionDataRef().mDidNormalizeWhitespaces = true; + NS_WARNING_ASSERTION( + caretPointOrError.isOk(), + "HTMLEditor::DeleteTextAndNormalizeSurroundingWhiteSpaces() failed"); + return caretPointOrError; +} + +Result<CaretPoint, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + HandleDeleteCollapsedSelectionAtWhiteSpaces( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + const EditorDOMPoint& aPointToDelete, const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!StaticPrefs::editor_white_space_normalization_blink_compatible()); + + EditorDOMPoint pointToPutCaret; + if (aDirectionAndAmount == nsIEditor::eNext) { + Result<CaretPoint, nsresult> caretPointOrError = + WhiteSpaceVisibilityKeeper::DeleteInclusiveNextWhiteSpace( + aHTMLEditor, aPointToDelete, aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "WhiteSpaceVisibilityKeeper::DeleteInclusiveNextWhiteSpace() failed"); + return caretPointOrError; + } + caretPointOrError.unwrap().MoveCaretPointTo( + pointToPutCaret, aHTMLEditor, + {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt}); + } else { + Result<CaretPoint, nsresult> caretPointOrError = + WhiteSpaceVisibilityKeeper::DeletePreviousWhiteSpace( + aHTMLEditor, aPointToDelete, aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "WhiteSpaceVisibilityKeeper::DeletePreviousWhiteSpace() failed"); + return caretPointOrError; + } + caretPointOrError.unwrap().MoveCaretPointTo( + pointToPutCaret, aHTMLEditor, + {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt}); + } + const auto newCaretPosition = + aHTMLEditor.GetFirstSelectionStartPoint<EditorDOMPoint>(); + if (MOZ_UNLIKELY(!newCaretPosition.IsSet())) { + NS_WARNING("There was no selection range"); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.InsertBRElementIfHardLineIsEmptyAndEndsWithBlockBoundary( + newCaretPosition); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "HTMLEditor::InsertBRElementIfHardLineIsEmptyAndEndsWithBlockBoundary()" + " failed"); + return caretPointOrError; + } + caretPointOrError.unwrap().MoveCaretPointTo( + pointToPutCaret, {SuggestCaret::OnlyIfHasSuggestion}); + return CaretPoint(std::move(pointToPutCaret)); +} + +Result<CaretPoint, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + HandleDeleteCollapsedSelectionAtVisibleChar( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, + const EditorDOMPoint& aPointAtDeletingChar, + const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsTopLevelEditSubActionDataAvailable()); + MOZ_ASSERT(!StaticPrefs::editor_white_space_normalization_blink_compatible()); + MOZ_ASSERT(aPointAtDeletingChar.IsSet()); + MOZ_ASSERT(aPointAtDeletingChar.IsInTextNode()); + + OwningNonNull<Text> visibleTextNode = + *aPointAtDeletingChar.ContainerAs<Text>(); + EditorDOMPoint startToDelete, endToDelete; + // FIXME: This does not care grapheme cluster of complicate character + // sequence like Emoji. + // TODO: Investigate what happens if a grapheme cluster which should be + // delete once is split to multiple text nodes. + // TODO: We should stop using this path, instead, we should extend the range + // before calling this method. + if (aDirectionAndAmount == nsIEditor::ePrevious) { + if (MOZ_UNLIKELY(aPointAtDeletingChar.IsStartOfContainer())) { + return Err(NS_ERROR_UNEXPECTED); + } + startToDelete = aPointAtDeletingChar.PreviousPoint(); + endToDelete = aPointAtDeletingChar; + // Bug 1068979: delete both codepoints if surrogate pair + if (!startToDelete.IsStartOfContainer()) { + const nsTextFragment* text = &visibleTextNode->TextFragment(); + if (text->IsLowSurrogateFollowingHighSurrogateAt( + startToDelete.Offset())) { + startToDelete.RewindOffset(); + } + } + } else { + if (NS_WARN_IF(aRangesToDelete.Ranges().IsEmpty()) || + NS_WARN_IF(aRangesToDelete.FirstRangeRef()->GetStartContainer() != + aPointAtDeletingChar.GetContainer()) || + NS_WARN_IF(aRangesToDelete.FirstRangeRef()->GetEndContainer() != + aPointAtDeletingChar.GetContainer())) { + return Err(NS_ERROR_FAILURE); + } + startToDelete = aRangesToDelete.FirstRangeRef()->StartRef(); + endToDelete = aRangesToDelete.FirstRangeRef()->EndRef(); + } + + { + Result<CaretPoint, nsresult> caretPointOrError = + WhiteSpaceVisibilityKeeper::PrepareToDeleteRangeAndTrackPoints( + aHTMLEditor, &startToDelete, &endToDelete, aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "WhiteSpaceVisibilityKeeper::PrepareToDeleteRangeAndTrackPoints() " + "failed"); + return caretPointOrError.propagateErr(); + } + // Ignore caret position because we'll set caret position below + caretPointOrError.unwrap().IgnoreCaretPointSuggestion(); + } + + if (aHTMLEditor.MayHaveMutationEventListeners( + NS_EVENT_BITS_MUTATION_NODEREMOVED | + NS_EVENT_BITS_MUTATION_NODEREMOVEDFROMDOCUMENT | + NS_EVENT_BITS_MUTATION_ATTRMODIFIED | + NS_EVENT_BITS_MUTATION_CHARACTERDATAMODIFIED) && + (NS_WARN_IF(!startToDelete.IsSetAndValid()) || + NS_WARN_IF(!startToDelete.IsInTextNode()) || + NS_WARN_IF(!endToDelete.IsSetAndValid()) || + NS_WARN_IF(!endToDelete.IsInTextNode()) || + NS_WARN_IF(startToDelete.ContainerAs<Text>() != visibleTextNode) || + NS_WARN_IF(endToDelete.ContainerAs<Text>() != visibleTextNode) || + NS_WARN_IF(startToDelete.Offset() >= endToDelete.Offset()))) { + NS_WARNING("Mutation event listener changed the DOM tree"); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + EditorDOMPoint pointToPutCaret = startToDelete; + { + AutoTrackDOMPoint trackPointToPutCaret(aHTMLEditor.RangeUpdaterRef(), + &pointToPutCaret); + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteTextWithTransaction( + visibleTextNode, startToDelete.Offset(), + endToDelete.Offset() - startToDelete.Offset()); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("HTMLEditor::DeleteTextWithTransaction() failed"); + return caretPointOrError.propagateErr(); + } + trackPointToPutCaret.FlushAndStopTracking(); + caretPointOrError.unwrap().MoveCaretPointTo( + pointToPutCaret, aHTMLEditor, + {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt}); + } + + // XXX When Backspace key is pressed, Chromium removes following empty + // text nodes when removing the last character of the non-empty text + // node. However, Edge never removes empty text nodes even if + // selection is in the following empty text node(s). For now, we + // should keep our traditional behavior same as Edge for backward + // compatibility. + // XXX When Delete key is pressed, Edge removes all preceding empty + // text nodes when removing the first character of the non-empty + // text node. Chromium removes only selected empty text node and + // following empty text nodes and the first character of the + // non-empty text node. For now, we should keep our traditional + // behavior same as Chromium for backward compatibility. + { + AutoTrackDOMPoint trackPointToPutCaret(aHTMLEditor.RangeUpdaterRef(), + &pointToPutCaret); + nsresult rv = + DeleteNodeIfInvisibleAndEditableTextNode(aHTMLEditor, visibleTextNode); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::DeleteNodeIfInvisibleAndEditableTextNode() " + "failed, but ignored"); + } + + if (NS_WARN_IF(!pointToPutCaret.IsSet())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + // XXX `Selection` may be modified by mutation event listeners so + // that we should use EditorDOMPoint::AtEndOf(visibleTextNode) + // instead. (Perhaps, we don't and/or shouldn't need to do this + // if the text node is preformatted.) + AutoTrackDOMPoint trackPointToPutCaret(aHTMLEditor.RangeUpdaterRef(), + &pointToPutCaret); + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.InsertBRElementIfHardLineIsEmptyAndEndsWithBlockBoundary( + pointToPutCaret); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "HTMLEditor::InsertBRElementIfHardLineIsEmptyAndEndsWithBlockBoundary()" + " failed"); + return caretPointOrError.propagateErr(); + } + trackPointToPutCaret.FlushAndStopTracking(); + caretPointOrError.unwrap().MoveCaretPointTo( + pointToPutCaret, {SuggestCaret::OnlyIfHasSuggestion}); + // Remember that we did a ranged delete for the benefit of + // AfterEditInner(). + aHTMLEditor.TopLevelEditSubActionDataRef().mDidDeleteNonCollapsedRange = true; + return CaretPoint(std::move(pointToPutCaret)); +} + +// static +nsIContent* HTMLEditor::AutoDeleteRangesHandler::GetAtomicContentToDelete( + nsIEditor::EDirection aDirectionAndAmount, + const WSRunScanner& aWSRunScannerAtCaret, + const WSScanResult& aScanFromCaretPointResult) { + MOZ_ASSERT(aScanFromCaretPointResult.GetContent()); + + if (!aScanFromCaretPointResult.ReachedSpecialContent()) { + return aScanFromCaretPointResult.GetContent(); + } + + if (!aScanFromCaretPointResult.GetContent()->IsText() || + HTMLEditUtils::IsRemovableNode(*aScanFromCaretPointResult.GetContent())) { + return aScanFromCaretPointResult.GetContent(); + } + + // aScanFromCaretPointResult is non-removable text node. + // Since we try removing atomic content, we look for removable node from + // scanned point that is non-removable text. + nsIContent* removableRoot = aScanFromCaretPointResult.GetContent(); + while (removableRoot && !HTMLEditUtils::IsRemovableNode(*removableRoot)) { + removableRoot = removableRoot->GetParent(); + } + + if (removableRoot) { + return removableRoot; + } + + // Not found better content. This content may not be removable. + return aScanFromCaretPointResult.GetContent(); +} + +nsresult +HTMLEditor::AutoDeleteRangesHandler::ComputeRangesToDeleteAtomicContent( + Element* aEditingHost, const nsIContent& aAtomicContent, + AutoRangeArray& aRangesToDelete) const { + EditorDOMRange rangeToDelete = + WSRunScanner::GetRangesForDeletingAtomicContent(aEditingHost, + aAtomicContent); + if (!rangeToDelete.IsPositioned()) { + NS_WARNING("WSRunScanner::GetRangeForDeleteAContentNode() failed"); + return NS_ERROR_FAILURE; + } + nsresult rv = aRangesToDelete.SetStartAndEnd(rangeToDelete.StartRef(), + rangeToDelete.EndRef()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoRangeArray::SetStartAndEnd() failed"); + return rv; +} + +Result<CaretPoint, nsresult> +HTMLEditor::AutoDeleteRangesHandler::HandleDeleteAtomicContent( + HTMLEditor& aHTMLEditor, nsIContent& aAtomicContent, + const EditorDOMPoint& aCaretPoint, const WSRunScanner& aWSRunScannerAtCaret, + const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!HTMLEditUtils::IsInvisibleBRElement(aAtomicContent)); + MOZ_ASSERT(&aAtomicContent != aWSRunScannerAtCaret.GetEditingHost()); + + EditorDOMPoint pointToPutCaret = aCaretPoint; + { + AutoTrackDOMPoint trackPointToPutCaret(aHTMLEditor.RangeUpdaterRef(), + &pointToPutCaret); + Result<CaretPoint, nsresult> caretPointOrError = + WhiteSpaceVisibilityKeeper::DeleteContentNodeAndJoinTextNodesAroundIt( + aHTMLEditor, aAtomicContent, aCaretPoint, aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "WhiteSpaceVisibilityKeeper::" + "DeleteContentNodeAndJoinTextNodesAroundIt() failed"); + return caretPointOrError; + } + trackPointToPutCaret.FlushAndStopTracking(); + caretPointOrError.unwrap().MoveCaretPointTo( + pointToPutCaret, aHTMLEditor, + {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt}); + if (NS_WARN_IF(!pointToPutCaret.IsSet())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + } + + { + AutoTrackDOMPoint trackPointToPutCaret(aHTMLEditor.RangeUpdaterRef(), + &pointToPutCaret); + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.InsertBRElementIfHardLineIsEmptyAndEndsWithBlockBoundary( + pointToPutCaret); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "HTMLEditor::" + "InsertBRElementIfHardLineIsEmptyAndEndsWithBlockBoundary()" + " failed"); + return caretPointOrError; + } + trackPointToPutCaret.FlushAndStopTracking(); + caretPointOrError.unwrap().MoveCaretPointTo( + pointToPutCaret, aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion}); + if (NS_WARN_IF(!pointToPutCaret.IsSet())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + } + return CaretPoint(std::move(pointToPutCaret)); +} + +bool HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + PrepareToDeleteAtOtherBlockBoundary( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, Element& aOtherBlockElement, + const EditorDOMPoint& aCaretPoint, + const WSRunScanner& aWSRunScannerAtCaret) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(aCaretPoint.IsSetAndValid()); + + mMode = Mode::JoinOtherBlock; + + // Make sure it's not a table element. If so, cancel the operation + // (translation: users cannot backspace or delete across table cells) + if (HTMLEditUtils::IsAnyTableElement(&aOtherBlockElement)) { + return false; + } + + // First find the adjacent node in the block + if (aDirectionAndAmount == nsIEditor::ePrevious) { + mLeafContentInOtherBlock = HTMLEditUtils::GetLastLeafContent( + aOtherBlockElement, {LeafNodeType::OnlyEditableLeafNode}, + &aOtherBlockElement); + mLeftContent = mLeafContentInOtherBlock; + mRightContent = aCaretPoint.GetContainerAs<nsIContent>(); + } else { + mLeafContentInOtherBlock = HTMLEditUtils::GetFirstLeafContent( + aOtherBlockElement, {LeafNodeType::OnlyEditableLeafNode}, + &aOtherBlockElement); + mLeftContent = aCaretPoint.GetContainerAs<nsIContent>(); + mRightContent = mLeafContentInOtherBlock; + } + + // Next to a block. See if we are between the block and a `<br>`. + // If so, we really want to delete the `<br>`. Else join content at + // selection to the block. + WSScanResult scanFromCaretResult = + aDirectionAndAmount == nsIEditor::eNext + ? aWSRunScannerAtCaret.ScanPreviousVisibleNodeOrBlockBoundaryFrom( + aCaretPoint) + : aWSRunScannerAtCaret.ScanNextVisibleNodeOrBlockBoundaryFrom( + aCaretPoint); + // If we found a `<br>` element, we need to delete it instead of joining the + // contents. + if (scanFromCaretResult.ReachedBRElement()) { + mBRElement = scanFromCaretResult.BRElementPtr(); + mMode = Mode::DeleteBRElement; + return true; + } + + return mLeftContent && mRightContent; +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + ComputeRangesToDeleteBRElement(AutoRangeArray& aRangesToDelete) const { + MOZ_ASSERT(mBRElement); + // XXX Why don't we scan invisible leading white-spaces which follows the + // `<br>` element? + nsresult rv = aRangesToDelete.SelectNode(*mBRElement); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "AutoRangeArray::SelectNode() failed"); + return rv; +} + +Result<EditActionResult, nsresult> +HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner::DeleteBRElement( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(mBRElement); + + // If we're deleting selection (not replacing with new content), we should + // put caret to end of preceding text node if there is. Then, users can type + // text in it like the other browsers. + EditorDOMPoint pointToPutCaret = [&]() { + if (!MayEditActionDeleteAroundCollapsedSelection( + aHTMLEditor.GetEditAction())) { + return EditorDOMPoint(); + } + WSRunScanner scanner(&aEditingHost, EditorRawDOMPoint(mBRElement)); + WSScanResult maybePreviousText = + scanner.ScanPreviousVisibleNodeOrBlockBoundaryFrom( + EditorRawDOMPoint(mBRElement)); + if (maybePreviousText.IsContentEditable() && + maybePreviousText.InVisibleOrCollapsibleCharacters() && + !HTMLEditor::GetLinkElement(maybePreviousText.TextPtr())) { + return maybePreviousText.Point<EditorDOMPoint>(); + } + WSScanResult maybeNextText = scanner.ScanNextVisibleNodeOrBlockBoundaryFrom( + EditorRawDOMPoint::After(*mBRElement)); + if (maybeNextText.IsContentEditable() && + maybeNextText.InVisibleOrCollapsibleCharacters()) { + return maybeNextText.Point<EditorDOMPoint>(); + } + return EditorDOMPoint(); + }(); + + // If we found a `<br>` element, we should delete it instead of joining the + // contents. + nsresult rv = + aHTMLEditor.DeleteNodeWithTransaction(MOZ_KnownLive(*mBRElement)); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return Err(rv); + } + + if (mLeftContent && mRightContent && + HTMLEditUtils::GetInclusiveAncestorAnyTableElement(*mLeftContent) != + HTMLEditUtils::GetInclusiveAncestorAnyTableElement(*mRightContent)) { + return EditActionResult::HandledResult(); + } + + // Put selection at edge of block and we are done. + if (NS_WARN_IF(!mLeafContentInOtherBlock)) { + // XXX This must be odd case. The other block can be empty. + return Err(NS_ERROR_FAILURE); + } + + if (pointToPutCaret.IsSet()) { + nsresult rv = aHTMLEditor.CollapseSelectionTo(pointToPutCaret); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + if (NS_SUCCEEDED(rv)) { + // If we prefer to use style in the previous line, we should forget + // previous styles since the caret position has all styles which we want + // to use with new content. + if (nsIEditor::DirectionIsBackspace(aDirectionAndAmount)) { + aHTMLEditor.TopLevelEditSubActionDataRef() + .mCachedPendingStyles->Clear(); + } + // And we don't want to keep extending a link at ex-end of the previous + // paragraph. + if (HTMLEditor::GetLinkElement(pointToPutCaret.GetContainer())) { + aHTMLEditor.mPendingStylesToApplyToNewContent + ->ClearLinkAndItsSpecifiedStyle(); + } + } else { + NS_WARNING("EditorBase::CollapseSelectionTo() failed, but ignored"); + } + return EditActionResult::HandledResult(); + } + + EditorRawDOMPoint newCaretPosition = + HTMLEditUtils::GetGoodCaretPointFor<EditorRawDOMPoint>( + *mLeafContentInOtherBlock, aDirectionAndAmount); + if (MOZ_UNLIKELY(!newCaretPosition.IsSet())) { + NS_WARNING("HTMLEditUtils::GetGoodCaretPointFor() failed"); + return Err(NS_ERROR_FAILURE); + } + rv = aHTMLEditor.CollapseSelectionTo(newCaretPosition); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed, but ignored"); + return EditActionResult::HandledResult(); +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + ComputeRangesToDeleteAtOtherBlockBoundary( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + const EditorDOMPoint& aCaretPoint, AutoRangeArray& aRangesToDelete, + const Element& aEditingHost) const { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(aCaretPoint.IsSetAndValid()); + MOZ_ASSERT(mLeftContent); + MOZ_ASSERT(mRightContent); + + if (HTMLEditUtils::GetInclusiveAncestorAnyTableElement(*mLeftContent) != + HTMLEditUtils::GetInclusiveAncestorAnyTableElement(*mRightContent)) { + if (!mDeleteRangesHandlerConst.CanFallbackToDeleteRangesWithTransaction( + aRangesToDelete)) { + nsresult rv = aRangesToDelete.Collapse(aCaretPoint); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoRangeArray::Collapse() failed"); + return rv; + } + nsresult rv = mDeleteRangesHandlerConst + .FallbackToComputeRangesToDeleteRangesWithTransaction( + aHTMLEditor, aRangesToDelete); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::" + "FallbackToComputeRangesToDeleteRangesWithTransaction() failed"); + return rv; + } + + AutoInclusiveAncestorBlockElementsJoiner joiner(*mLeftContent, + *mRightContent); + Result<bool, nsresult> canJoinThem = + joiner.Prepare(aHTMLEditor, aEditingHost); + if (canJoinThem.isErr()) { + NS_WARNING("AutoInclusiveAncestorBlockElementsJoiner::Prepare() failed"); + return canJoinThem.unwrapErr(); + } + if (canJoinThem.inspect() && joiner.CanJoinBlocks() && + !joiner.ShouldDeleteLeafContentInstead()) { + nsresult rv = + joiner.ComputeRangesToDelete(aHTMLEditor, aCaretPoint, aRangesToDelete); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoInclusiveAncestorBlockElementsJoiner::ComputeRangesToDelete() " + "failed"); + return rv; + } + + // If AutoInclusiveAncestorBlockElementsJoiner didn't handle it and it's not + // canceled, user may want to modify the start leaf node or the last leaf + // node of the block. + if (mLeafContentInOtherBlock == aCaretPoint.GetContainer()) { + return NS_OK; + } + + AutoHideSelectionChanges hideSelectionChanges(aHTMLEditor.SelectionRef()); + + // If it's ignored, it didn't modify the DOM tree. In this case, user must + // want to delete nearest leaf node in the other block element. + // TODO: We need to consider this before calling ComputeRangesToDelete() for + // computing the deleting range. + EditorRawDOMPoint newCaretPoint = + aDirectionAndAmount == nsIEditor::ePrevious + ? EditorRawDOMPoint::AtEndOf(*mLeafContentInOtherBlock) + : EditorRawDOMPoint(mLeafContentInOtherBlock, 0); + // If new caret position is same as current caret position, we can do + // nothing anymore. + if (aRangesToDelete.IsCollapsed() && + aRangesToDelete.FocusRef() == newCaretPoint.ToRawRangeBoundary()) { + return NS_OK; + } + // TODO: Stop modifying the `Selection` for computing the targer ranges. + nsresult rv = aHTMLEditor.CollapseSelectionTo(newCaretPoint); + 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), + "EditorBase::CollapseSelectionTo() failed"); + if (NS_SUCCEEDED(rv)) { + aRangesToDelete.Initialize(aHTMLEditor.SelectionRef()); + AutoDeleteRangesHandler anotherHandler(mDeleteRangesHandlerConst); + rv = anotherHandler.ComputeRangesToDelete(aHTMLEditor, aDirectionAndAmount, + aRangesToDelete, aEditingHost); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "Recursive AutoDeleteRangesHandler::ComputeRangesToDelete() failed"); + } + // Restore selection. + nsresult rvCollapsingSelectionTo = + aHTMLEditor.CollapseSelectionTo(aCaretPoint); + if (MOZ_UNLIKELY(rvCollapsingSelectionTo == NS_ERROR_EDITOR_DESTROYED)) { + NS_WARNING( + "EditorBase::CollapseSelectionTo() caused destroying the editor"); + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rvCollapsingSelectionTo), + "EditorBase::CollapseSelectionTo() failed to restore caret position"); + return NS_SUCCEEDED(rv) && NS_SUCCEEDED(rvCollapsingSelectionTo) + ? NS_OK + : NS_ERROR_FAILURE; +} + +Result<EditActionResult, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoBlockElementsJoiner::HandleDeleteAtOtherBlockBoundary( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + const EditorDOMPoint& aCaretPoint, AutoRangeArray& aRangesToDelete, + const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(aCaretPoint.IsSetAndValid()); + MOZ_ASSERT(mDeleteRangesHandler); + MOZ_ASSERT(mLeftContent); + MOZ_ASSERT(mRightContent); + + if (HTMLEditUtils::GetInclusiveAncestorAnyTableElement(*mLeftContent) != + HTMLEditUtils::GetInclusiveAncestorAnyTableElement(*mRightContent)) { + // If we have not deleted `<br>` element and are not called recursively, + // we should call `DeleteRangesWithTransaction()` here. + if (!mDeleteRangesHandler->CanFallbackToDeleteRangesWithTransaction( + aRangesToDelete)) { + return EditActionResult::IgnoredResult(); + } + Result<CaretPoint, nsresult> caretPointOrError = + mDeleteRangesHandler->FallbackToDeleteRangesWithTransaction( + aHTMLEditor, aRangesToDelete); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING( + "AutoDeleteRangesHandler::FallbackToDeleteRangesWithTransaction() " + "failed"); + return caretPointOrError.propagateErr(); + } + nsresult rv = caretPointOrError.inspect().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + // Don't return "ignored" to avoid to fall it back to delete ranges + // recursively. + return EditActionResult::HandledResult(); + } + + // Else we are joining content to block + AutoInclusiveAncestorBlockElementsJoiner joiner(*mLeftContent, + *mRightContent); + Result<bool, nsresult> canJoinThem = + joiner.Prepare(aHTMLEditor, aEditingHost); + if (MOZ_UNLIKELY(canJoinThem.isErr())) { + NS_WARNING("AutoInclusiveAncestorBlockElementsJoiner::Prepare() failed"); + return canJoinThem.propagateErr(); + } + + if (!canJoinThem.inspect()) { + nsresult rv = aHTMLEditor.CollapseSelectionTo(aCaretPoint); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed, but ignored"); + return EditActionResult::CanceledResult(); + } + + auto result = EditActionResult::IgnoredResult(); + EditorDOMPoint pointToPutCaret(aCaretPoint); + if (joiner.CanJoinBlocks()) { + { + AutoTrackDOMPoint tracker(aHTMLEditor.RangeUpdaterRef(), + &pointToPutCaret); + Result<EditActionResult, nsresult> joinResult = + joiner.Run(aHTMLEditor, aEditingHost); + if (MOZ_UNLIKELY(joinResult.isErr())) { + NS_WARNING("AutoInclusiveAncestorBlockElementsJoiner::Run() failed"); + return joinResult; + } + result |= joinResult.unwrap(); +#ifdef DEBUG + if (joiner.ShouldDeleteLeafContentInstead()) { + NS_ASSERTION( + result.Ignored(), + "Assumed `AutoInclusiveAncestorBlockElementsJoiner::Run()` " + "returning ignored, but returned not ignored"); + } else { + NS_ASSERTION( + !result.Ignored(), + "Assumed `AutoInclusiveAncestorBlockElementsJoiner::Run()` " + "returning handled, but returned ignored"); + } +#endif // #ifdef DEBUG + // If we're deleting selection (not replacing with new content) and + // AutoInclusiveAncestorBlockElementsJoiner computed new caret position, + // we should use it. Otherwise, we should keep the our traditional + // behavior. + if (result.Handled() && joiner.PointRefToPutCaret().IsSet()) { + nsresult rv = + aHTMLEditor.CollapseSelectionTo(joiner.PointRefToPutCaret()); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed, but ignored"); + return result; + } + // If we prefer to use style in the previous line, we should forget + // previous styles since the caret position has all styles which we want + // to use with new content. + if (nsIEditor::DirectionIsBackspace(aDirectionAndAmount)) { + aHTMLEditor.TopLevelEditSubActionDataRef() + .mCachedPendingStyles->Clear(); + } + // And we don't want to keep extending a link at ex-end of the previous + // paragraph. + if (HTMLEditor::GetLinkElement( + joiner.PointRefToPutCaret().GetContainer())) { + aHTMLEditor.mPendingStylesToApplyToNewContent + ->ClearLinkAndItsSpecifiedStyle(); + } + return result; + } + } + + // If AutoInclusiveAncestorBlockElementsJoiner didn't handle it and it's not + // canceled, user may want to modify the start leaf node or the last leaf + // node of the block. + if (result.Ignored() && + mLeafContentInOtherBlock != aCaretPoint.GetContainer()) { + // If it's ignored, it didn't modify the DOM tree. In this case, user + // must want to delete nearest leaf node in the other block element. + // TODO: We need to consider this before calling Run() for computing the + // deleting range. + EditorRawDOMPoint newCaretPoint = + aDirectionAndAmount == nsIEditor::ePrevious + ? EditorRawDOMPoint::AtEndOf(*mLeafContentInOtherBlock) + : EditorRawDOMPoint(mLeafContentInOtherBlock, 0); + // If new caret position is same as current caret position, we can do + // nothing anymore. + if (aRangesToDelete.IsCollapsed() && + aRangesToDelete.FocusRef() == newCaretPoint.ToRawRangeBoundary()) { + return EditActionResult::CanceledResult(); + } + nsresult rv = aHTMLEditor.CollapseSelectionTo(newCaretPoint); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return Err(rv); + } + AutoRangeArray rangesToDelete(aHTMLEditor.SelectionRef()); + AutoDeleteRangesHandler anotherHandler(mDeleteRangesHandler); + Result<EditActionResult, nsresult> fallbackResult = + anotherHandler.Run(aHTMLEditor, aDirectionAndAmount, aStripWrappers, + rangesToDelete, aEditingHost); + if (MOZ_UNLIKELY(fallbackResult.isErr())) { + NS_WARNING("Recursive AutoDeleteRangesHandler::Run() failed"); + return fallbackResult; + } + result |= fallbackResult.unwrap(); + return result; + } + } else { + result.MarkAsHandled(); + } + + // Otherwise, we must have deleted the selection as user expected. + nsresult rv = aHTMLEditor.CollapseSelectionTo(pointToPutCaret); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed, but ignored"); + return result; +} + +bool HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + PrepareToDeleteAtCurrentBlockBoundary( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + Element& aCurrentBlockElement, const EditorDOMPoint& aCaretPoint) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + + // At edge of our block. Look beside it and see if we can join to an + // adjacent block + mMode = Mode::JoinCurrentBlock; + + // Don't break the basic structure of the HTML document. + if (aCurrentBlockElement.IsAnyOfHTMLElements(nsGkAtoms::html, nsGkAtoms::head, + nsGkAtoms::body)) { + return false; + } + + // Make sure it's not a table element. If so, cancel the operation + // (translation: users cannot backspace or delete across table cells) + if (HTMLEditUtils::IsAnyTableElement(&aCurrentBlockElement)) { + return false; + } + + Element* editingHost = aHTMLEditor.ComputeEditingHost(); + if (NS_WARN_IF(!editingHost)) { + return false; + } + + auto ScanJoinTarget = [&]() -> nsIContent* { + nsIContent* targetContent = + aDirectionAndAmount == nsIEditor::ePrevious + ? HTMLEditUtils::GetPreviousContent( + aCurrentBlockElement, {WalkTreeOption::IgnoreNonEditableNode}, + editingHost) + : HTMLEditUtils::GetNextContent( + aCurrentBlockElement, {WalkTreeOption::IgnoreNonEditableNode}, + editingHost); + // If found content is an invisible text node, let's scan visible things. + auto IsIgnorableDataNode = [](nsIContent* aContent) { + return aContent && HTMLEditUtils::IsRemovableNode(*aContent) && + ((aContent->IsText() && + aContent->AsText()->TextIsOnlyWhitespace() && + !HTMLEditUtils::IsVisibleTextNode(*aContent->AsText())) || + (aContent->IsCharacterData() && !aContent->IsText())); + }; + if (!IsIgnorableDataNode(targetContent)) { + return targetContent; + } + MOZ_ASSERT(mSkippedInvisibleContents.IsEmpty()); + for (nsIContent* adjacentContent = + aDirectionAndAmount == nsIEditor::ePrevious + ? HTMLEditUtils::GetPreviousContent( + *targetContent, {WalkTreeOption::StopAtBlockBoundary}, + editingHost) + : HTMLEditUtils::GetNextContent( + *targetContent, {WalkTreeOption::StopAtBlockBoundary}, + editingHost); + adjacentContent; + adjacentContent = + aDirectionAndAmount == nsIEditor::ePrevious + ? HTMLEditUtils::GetPreviousContent( + *adjacentContent, {WalkTreeOption::StopAtBlockBoundary}, + editingHost) + : HTMLEditUtils::GetNextContent( + *adjacentContent, {WalkTreeOption::StopAtBlockBoundary}, + editingHost)) { + // If non-editable element is found, we should not skip it to avoid + // joining too far nodes. + if (!HTMLEditUtils::IsSimplyEditableNode(*adjacentContent)) { + break; + } + // If block element is found, we should join last leaf content in it. + if (HTMLEditUtils::IsBlockElement(*adjacentContent)) { + nsIContent* leafContent = + aDirectionAndAmount == nsIEditor::ePrevious + ? HTMLEditUtils::GetLastLeafContent( + *adjacentContent, {LeafNodeType::OnlyEditableLeafNode}) + : HTMLEditUtils::GetFirstLeafContent( + *adjacentContent, {LeafNodeType::OnlyEditableLeafNode}); + mSkippedInvisibleContents.AppendElement(*targetContent); + return leafContent ? leafContent : adjacentContent; + } + // Only when the found node is an invisible text node or a non-text data + // node, we should keep scanning. + if (IsIgnorableDataNode(adjacentContent)) { + mSkippedInvisibleContents.AppendElement(*targetContent); + targetContent = adjacentContent; + continue; + } + // Otherwise, we find a visible things. We should join with last found + // invisible text node. + break; + } + return targetContent; + }; + + if (aDirectionAndAmount == nsIEditor::ePrevious) { + mLeftContent = ScanJoinTarget(); + mRightContent = aCaretPoint.GetContainerAs<nsIContent>(); + } else { + mRightContent = ScanJoinTarget(); + mLeftContent = aCaretPoint.GetContainerAs<nsIContent>(); + } + + // Nothing to join + if (!mLeftContent || !mRightContent) { + return false; + } + + // Don't cross table boundaries. + return HTMLEditUtils::GetInclusiveAncestorAnyTableElement(*mLeftContent) == + HTMLEditUtils::GetInclusiveAncestorAnyTableElement(*mRightContent); +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + ComputeRangesToDeleteAtCurrentBlockBoundary( + const HTMLEditor& aHTMLEditor, const EditorDOMPoint& aCaretPoint, + AutoRangeArray& aRangesToDelete, const Element& aEditingHost) const { + MOZ_ASSERT(mLeftContent); + MOZ_ASSERT(mRightContent); + + AutoInclusiveAncestorBlockElementsJoiner joiner(*mLeftContent, + *mRightContent); + Result<bool, nsresult> canJoinThem = + joiner.Prepare(aHTMLEditor, aEditingHost); + if (canJoinThem.isErr()) { + NS_WARNING("AutoInclusiveAncestorBlockElementsJoiner::Prepare() failed"); + return canJoinThem.unwrapErr(); + } + if (canJoinThem.inspect()) { + nsresult rv = + joiner.ComputeRangesToDelete(aHTMLEditor, aCaretPoint, aRangesToDelete); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoInclusiveAncestorBlockElementsJoiner::" + "ComputeRangesToDelete() failed"); + return rv; + } + + // In this case, nothing will be deleted so that the affected range should + // be collapsed. + nsresult rv = aRangesToDelete.Collapse(aCaretPoint); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "AutoRangeArray::Collapse() failed"); + return rv; +} + +Result<EditActionResult, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoBlockElementsJoiner::HandleDeleteAtCurrentBlockBoundary( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + const EditorDOMPoint& aCaretPoint, const Element& aEditingHost) { + MOZ_ASSERT(mLeftContent); + MOZ_ASSERT(mRightContent); + + AutoInclusiveAncestorBlockElementsJoiner joiner(*mLeftContent, + *mRightContent); + Result<bool, nsresult> canJoinThem = + joiner.Prepare(aHTMLEditor, aEditingHost); + if (MOZ_UNLIKELY(canJoinThem.isErr())) { + NS_WARNING("AutoInclusiveAncestorBlockElementsJoiner::Prepare() failed"); + return Err(canJoinThem.unwrapErr()); + } + + if (!canJoinThem.inspect()) { + nsresult rv = aHTMLEditor.CollapseSelectionTo(aCaretPoint); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed, but ignored"); + return EditActionResult::CanceledResult(); + } + + EditActionResult result = EditActionResult::IgnoredResult(); + EditorDOMPoint pointToPutCaret(aCaretPoint); + if (joiner.CanJoinBlocks()) { + AutoTrackDOMPoint tracker(aHTMLEditor.RangeUpdaterRef(), &pointToPutCaret); + Result<EditActionResult, nsresult> joinResult = + joiner.Run(aHTMLEditor, aEditingHost); + if (MOZ_UNLIKELY(joinResult.isErr())) { + NS_WARNING("AutoInclusiveAncestorBlockElementsJoiner::Run() failed"); + return joinResult; + } + result |= joinResult.unwrap(); +#ifdef DEBUG + if (joiner.ShouldDeleteLeafContentInstead()) { + NS_ASSERTION(result.Ignored(), + "Assumed `AutoInclusiveAncestorBlockElementsJoiner::Run()` " + "returning ignored, but returned not ignored"); + } else { + NS_ASSERTION(!result.Ignored(), + "Assumed `AutoInclusiveAncestorBlockElementsJoiner::Run()` " + "returning handled, but returned ignored"); + } +#endif // #ifdef DEBUG + + // Cleaning up invisible nodes which are skipped at scanning mLeftContent or + // mRightContent. + for (const OwningNonNull<nsIContent>& content : mSkippedInvisibleContents) { + nsresult rv = + aHTMLEditor.DeleteNodeWithTransaction(MOZ_KnownLive(content)); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return Err(rv); + } + } + mSkippedInvisibleContents.Clear(); + + // If we're deleting selection (not replacing with new content) and + // AutoInclusiveAncestorBlockElementsJoiner computed new caret position, we + // should use it. Otherwise, we should keep the our traditional behavior. + if (result.Handled() && joiner.PointRefToPutCaret().IsSet()) { + nsresult rv = + aHTMLEditor.CollapseSelectionTo(joiner.PointRefToPutCaret()); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed, but ignored"); + return result; + } + // If we prefer to use style in the previous line, we should forget + // previous styles since the caret position has all styles which we want + // to use with new content. + if (nsIEditor::DirectionIsBackspace(aDirectionAndAmount)) { + aHTMLEditor.TopLevelEditSubActionDataRef() + .mCachedPendingStyles->Clear(); + } + // And we don't want to keep extending a link at ex-end of the previous + // paragraph. + if (HTMLEditor::GetLinkElement( + joiner.PointRefToPutCaret().GetContainer())) { + aHTMLEditor.mPendingStylesToApplyToNewContent + ->ClearLinkAndItsSpecifiedStyle(); + } + return result; + } + } + // This should claim that trying to join the block means that + // this handles the action because the caller shouldn't do anything + // anymore in this case. + result.MarkAsHandled(); + + nsresult rv = aHTMLEditor.CollapseSelectionTo(pointToPutCaret); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed, but ignored"); + return result; +} + +nsresult +HTMLEditor::AutoDeleteRangesHandler::ComputeRangesToDeleteNonCollapsedRanges( + const HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost) const { + MOZ_ASSERT(!aRangesToDelete.IsCollapsed()); + + if (NS_WARN_IF(!aRangesToDelete.FirstRangeRef()->StartRef().IsSet()) || + NS_WARN_IF(!aRangesToDelete.FirstRangeRef()->EndRef().IsSet())) { + return NS_ERROR_FAILURE; + } + + if (aRangesToDelete.Ranges().Length() == 1) { + nsFrameSelection* frameSelection = + aHTMLEditor.SelectionRef().GetFrameSelection(); + if (NS_WARN_IF(!frameSelection)) { + return NS_ERROR_FAILURE; + } + Result<EditorRawDOMRange, nsresult> result = ExtendOrShrinkRangeToDelete( + aHTMLEditor, frameSelection, + EditorRawDOMRange(aRangesToDelete.FirstRangeRef())); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING( + "AutoDeleteRangesHandler::ExtendOrShrinkRangeToDelete() failed"); + return NS_ERROR_FAILURE; + } + EditorRawDOMRange newRange(result.unwrap()); + if (MOZ_UNLIKELY(NS_FAILED(aRangesToDelete.FirstRangeRef()->SetStartAndEnd( + newRange.StartRef().ToRawRangeBoundary(), + newRange.EndRef().ToRawRangeBoundary())))) { + NS_WARNING("nsRange::SetStartAndEnd() failed"); + return NS_ERROR_FAILURE; + } + if (MOZ_UNLIKELY( + NS_WARN_IF(!aRangesToDelete.FirstRangeRef()->IsPositioned()))) { + return NS_ERROR_FAILURE; + } + if (NS_WARN_IF(aRangesToDelete.FirstRangeRef()->Collapsed())) { + return NS_OK; // Hmm, there is nothing to delete...? + } + } + + if (!aHTMLEditor.IsInPlaintextMode()) { + EditorDOMRange firstRange(aRangesToDelete.FirstRangeRef()); + EditorDOMRange extendedRange = + WSRunScanner::GetRangeContainingInvisibleWhiteSpacesAtRangeBoundaries( + aHTMLEditor.ComputeEditingHost(), + EditorDOMRange(aRangesToDelete.FirstRangeRef())); + if (firstRange != extendedRange) { + nsresult rv = aRangesToDelete.FirstRangeRef()->SetStartAndEnd( + extendedRange.StartRef().ToRawRangeBoundary(), + extendedRange.EndRef().ToRawRangeBoundary()); + if (NS_FAILED(rv)) { + NS_WARNING("nsRange::SetStartAndEnd() failed"); + return NS_ERROR_FAILURE; + } + } + } + + if (aRangesToDelete.FirstRangeRef()->GetStartContainer() == + aRangesToDelete.FirstRangeRef()->GetEndContainer()) { + if (!aRangesToDelete.FirstRangeRef()->Collapsed()) { + nsresult rv = ComputeRangesToDeleteRangesWithTransaction( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::ComputeRangesToDeleteRangesWithTransaction(" + ") failed"); + return rv; + } + // `DeleteUnnecessaryNodesAndCollapseSelection()` may delete parent + // elements, but it does not affect computing target ranges. Therefore, + // we don't need to touch aRangesToDelete in this case. + return NS_OK; + } + + Element* startCiteNode = aHTMLEditor.GetMostDistantAncestorMailCiteElement( + *aRangesToDelete.FirstRangeRef()->GetStartContainer()); + Element* endCiteNode = aHTMLEditor.GetMostDistantAncestorMailCiteElement( + *aRangesToDelete.FirstRangeRef()->GetEndContainer()); + + if (startCiteNode && !endCiteNode) { + aDirectionAndAmount = nsIEditor::eNext; + } else if (!startCiteNode && endCiteNode) { + aDirectionAndAmount = nsIEditor::ePrevious; + } + + AutoBlockElementsJoiner joiner(*this); + if (!joiner.PrepareToDeleteNonCollapsedRanges(aHTMLEditor, aRangesToDelete)) { + return NS_ERROR_FAILURE; + } + nsresult rv = joiner.ComputeRangesToDelete( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete, aSelectionWasCollapsed, + aEditingHost); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::ComputeRangesToDelete() failed"); + return rv; +} + +Result<EditActionResult, nsresult> +HTMLEditor::AutoDeleteRangesHandler::HandleDeleteNonCollapsedRanges( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, AutoRangeArray& aRangesToDelete, + SelectionWasCollapsed aSelectionWasCollapsed, const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsTopLevelEditSubActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.IsCollapsed()); + + if (NS_WARN_IF(!aRangesToDelete.FirstRangeRef()->StartRef().IsSet()) || + NS_WARN_IF(!aRangesToDelete.FirstRangeRef()->EndRef().IsSet())) { + return Err(NS_ERROR_FAILURE); + } + + MOZ_ASSERT_IF(aRangesToDelete.Ranges().Length() == 1, + aRangesToDelete.IsFirstRangeEditable(aEditingHost)); + + // Else we have a non-collapsed selection. First adjust the selection. + // XXX Why do we extend selection only when there is only one range? + if (aRangesToDelete.Ranges().Length() == 1) { + nsFrameSelection* frameSelection = + aHTMLEditor.SelectionRef().GetFrameSelection(); + if (NS_WARN_IF(!frameSelection)) { + return Err(NS_ERROR_FAILURE); + } + Result<EditorRawDOMRange, nsresult> result = ExtendOrShrinkRangeToDelete( + aHTMLEditor, frameSelection, + EditorRawDOMRange(aRangesToDelete.FirstRangeRef())); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING( + "AutoDeleteRangesHandler::ExtendOrShrinkRangeToDelete() failed"); + return Err(NS_ERROR_FAILURE); + } + EditorRawDOMRange newRange(result.unwrap()); + if (NS_FAILED(aRangesToDelete.FirstRangeRef()->SetStartAndEnd( + newRange.StartRef().ToRawRangeBoundary(), + newRange.EndRef().ToRawRangeBoundary()))) { + NS_WARNING("nsRange::SetStartAndEnd() failed"); + return Err(NS_ERROR_FAILURE); + } + if (NS_WARN_IF(!aRangesToDelete.FirstRangeRef()->IsPositioned())) { + return Err(NS_ERROR_FAILURE); + } + if (NS_WARN_IF(aRangesToDelete.FirstRangeRef()->Collapsed())) { + // Hmm, there is nothing to delete...? + // In this case, the callers want collapsed selection. Therefore, we need + // to change the `Selection` here. + nsresult rv = aHTMLEditor.CollapseSelectionTo( + aRangesToDelete.GetFirstRangeStartPoint<EditorRawDOMPoint>()); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return Err(rv); + } + return EditActionResult::HandledResult(); + } + MOZ_ASSERT(aRangesToDelete.IsFirstRangeEditable(aEditingHost)); + } + + // Remember that we did a ranged delete for the benefit of AfterEditInner(). + aHTMLEditor.TopLevelEditSubActionDataRef().mDidDeleteNonCollapsedRange = true; + + // Figure out if the endpoints are in nodes that can be merged. Adjust + // surrounding white-space in preparation to delete selection. + if (!aHTMLEditor.IsInPlaintextMode()) { + { + AutoTrackDOMRange firstRangeTracker(aHTMLEditor.RangeUpdaterRef(), + &aRangesToDelete.FirstRangeRef()); + Result<CaretPoint, nsresult> caretPointOrError = + WhiteSpaceVisibilityKeeper::PrepareToDeleteRange( + aHTMLEditor, EditorDOMRange(aRangesToDelete.FirstRangeRef()), + aEditingHost); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("WhiteSpaceVisibilityKeeper::PrepareToDeleteRange() failed"); + return caretPointOrError.propagateErr(); + } + // Ignore caret point suggestion because there was + // AutoTransactionsConserveSelection. + caretPointOrError.unwrap().IgnoreCaretPointSuggestion(); + } + if (NS_WARN_IF(!aRangesToDelete.FirstRangeRef()->IsPositioned()) || + (aHTMLEditor.MayHaveMutationEventListeners() && + NS_WARN_IF(!aRangesToDelete.IsFirstRangeEditable(aEditingHost)))) { + NS_WARNING( + "WhiteSpaceVisibilityKeeper::PrepareToDeleteRange() made the first " + "range invalid"); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + } + + // XXX This is odd. We do we simply use `DeleteRangesWithTransaction()` + // only when **first** range is in same container? + if (aRangesToDelete.FirstRangeRef()->GetStartContainer() == + aRangesToDelete.FirstRangeRef()->GetEndContainer()) { + // Because of previous DOM tree changes, the range may be collapsed. + // If we've already removed all contents in the range, we shouldn't + // delete anything around the caret. + if (!aRangesToDelete.FirstRangeRef()->Collapsed()) { + { + AutoTrackDOMRange firstRangeTracker(aHTMLEditor.RangeUpdaterRef(), + &aRangesToDelete.FirstRangeRef()); + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteRangesWithTransaction( + aDirectionAndAmount, aStripWrappers, aRangesToDelete); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("EditorBase::DeleteRangesWithTransaction() failed"); + return caretPointOrError.propagateErr(); + } + nsresult rv = caretPointOrError.inspect().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + } + if (NS_WARN_IF(!aRangesToDelete.FirstRangeRef()->IsPositioned()) || + (aHTMLEditor.MayHaveMutationEventListeners( + NS_EVENT_BITS_MUTATION_NODEREMOVED | + NS_EVENT_BITS_MUTATION_NODEREMOVEDFROMDOCUMENT | + NS_EVENT_BITS_MUTATION_SUBTREEMODIFIED) && + NS_WARN_IF(!aRangesToDelete.IsFirstRangeEditable(aEditingHost)))) { + NS_WARNING( + "EditorBase::DeleteRangesWithTransaction() made the first range " + "invalid"); + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + } + // However, even if the range is removed, we may need to clean up the + // containers which become empty. + nsresult rv = DeleteUnnecessaryNodesAndCollapseSelection( + aHTMLEditor, aDirectionAndAmount, + EditorDOMPoint(aRangesToDelete.FirstRangeRef()->StartRef()), + EditorDOMPoint(aRangesToDelete.FirstRangeRef()->EndRef())); + if (NS_FAILED(rv)) { + NS_WARNING( + "AutoDeleteRangesHandler::DeleteUnnecessaryNodesAndCollapseSelection(" + ") failed"); + return Err(rv); + } + return EditActionResult::HandledResult(); + } + + if (NS_WARN_IF( + !aRangesToDelete.FirstRangeRef()->GetStartContainer()->IsContent()) || + NS_WARN_IF( + !aRangesToDelete.FirstRangeRef()->GetEndContainer()->IsContent())) { + return Err(NS_ERROR_FAILURE); + } + + // Figure out mailcite ancestors + RefPtr<Element> startCiteNode = + aHTMLEditor.GetMostDistantAncestorMailCiteElement( + *aRangesToDelete.FirstRangeRef()->GetStartContainer()); + RefPtr<Element> endCiteNode = + aHTMLEditor.GetMostDistantAncestorMailCiteElement( + *aRangesToDelete.FirstRangeRef()->GetEndContainer()); + + // If we only have a mailcite at one of the two endpoints, set the + // directionality of the deletion so that the selection will end up + // outside the mailcite. + if (startCiteNode && !endCiteNode) { + aDirectionAndAmount = nsIEditor::eNext; + } else if (!startCiteNode && endCiteNode) { + aDirectionAndAmount = nsIEditor::ePrevious; + } + + AutoBlockElementsJoiner joiner(*this); + if (!joiner.PrepareToDeleteNonCollapsedRanges(aHTMLEditor, aRangesToDelete)) { + return Err(NS_ERROR_FAILURE); + } + Result<EditActionResult, nsresult> result = + joiner.Run(aHTMLEditor, aDirectionAndAmount, aStripWrappers, + aRangesToDelete, aSelectionWasCollapsed, aEditingHost); + NS_WARNING_ASSERTION(result.isOk(), "AutoBlockElementsJoiner::Run() failed"); + return result; +} + +bool HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + PrepareToDeleteNonCollapsedRanges(const HTMLEditor& aHTMLEditor, + const AutoRangeArray& aRangesToDelete) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.IsCollapsed()); + + mLeftContent = HTMLEditUtils::GetInclusiveAncestorElement( + *aRangesToDelete.FirstRangeRef()->GetStartContainer()->AsContent(), + HTMLEditUtils::ClosestEditableBlockElement); + mRightContent = HTMLEditUtils::GetInclusiveAncestorElement( + *aRangesToDelete.FirstRangeRef()->GetEndContainer()->AsContent(), + HTMLEditUtils::ClosestEditableBlockElement); + // Note that mLeftContent and/or mRightContent can be nullptr if editing host + // is an inline element. If both editable ancestor block is exactly same + // one or one reaches an inline editing host, we can just delete the content + // in ranges. + if (mLeftContent == mRightContent || !mLeftContent || !mRightContent) { + MOZ_ASSERT_IF(!mLeftContent || !mRightContent, + aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->AsContent() + ->GetEditingHost() == aRangesToDelete.FirstRangeRef() + ->GetEndContainer() + ->AsContent() + ->GetEditingHost()); + mMode = Mode::DeleteContentInRanges; + return true; + } + + // If left block and right block are adjuscent siblings and they are same + // type of elements, we can merge them after deleting the selected contents. + // MOOSE: this could conceivably screw up a table.. fix me. + if (mLeftContent->GetParentNode() == mRightContent->GetParentNode() && + HTMLEditUtils::CanContentsBeJoined(*mLeftContent, *mRightContent) && + // XXX What's special about these three types of block? + (mLeftContent->IsHTMLElement(nsGkAtoms::p) || + HTMLEditUtils::IsListItem(mLeftContent) || + HTMLEditUtils::IsHeader(*mLeftContent))) { + mMode = Mode::JoinBlocksInSameParent; + return true; + } + + mMode = Mode::DeleteNonCollapsedRanges; + return true; +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + ComputeRangesToDeleteContentInRanges( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete) const { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.IsCollapsed()); + MOZ_ASSERT(mMode == Mode::DeleteContentInRanges); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->AsContent() + ->GetEditingHost()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->AsContent() + ->GetEditingHost() == aRangesToDelete.FirstRangeRef() + ->GetEndContainer() + ->AsContent() + ->GetEditingHost()); + MOZ_ASSERT(!mLeftContent == !mRightContent); + MOZ_ASSERT_IF(mLeftContent, mLeftContent->IsElement()); + MOZ_ASSERT_IF(mLeftContent, aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->IsInclusiveDescendantOf(mLeftContent)); + MOZ_ASSERT_IF(mRightContent, mRightContent->IsElement()); + MOZ_ASSERT_IF(mRightContent, aRangesToDelete.FirstRangeRef() + ->GetEndContainer() + ->IsInclusiveDescendantOf(mRightContent)); + MOZ_ASSERT_IF(!mLeftContent, + HTMLEditUtils::IsInlineElement(*aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->AsContent() + ->GetEditingHost())); + + nsresult rv = + mDeleteRangesHandlerConst.ComputeRangesToDeleteRangesWithTransaction( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::" + "ComputeRangesToDeleteRangesWithTransaction() failed"); + return rv; +} + +Result<EditActionResult, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoBlockElementsJoiner::DeleteContentInRanges( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + AutoRangeArray& aRangesToDelete) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.IsCollapsed()); + MOZ_ASSERT(mMode == Mode::DeleteContentInRanges); + MOZ_ASSERT(mDeleteRangesHandler); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->AsContent() + ->GetEditingHost()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->AsContent() + ->GetEditingHost() == aRangesToDelete.FirstRangeRef() + ->GetEndContainer() + ->AsContent() + ->GetEditingHost()); + MOZ_ASSERT_IF(mLeftContent, mLeftContent->IsElement()); + MOZ_ASSERT_IF(mLeftContent, aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->IsInclusiveDescendantOf(mLeftContent)); + MOZ_ASSERT_IF(mRightContent, mRightContent->IsElement()); + MOZ_ASSERT_IF(mRightContent, aRangesToDelete.FirstRangeRef() + ->GetEndContainer() + ->IsInclusiveDescendantOf(mRightContent)); + MOZ_ASSERT_IF(!mLeftContent, + HTMLEditUtils::IsInlineElement(*aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->AsContent() + ->GetEditingHost())); + + // XXX This is also odd. We do we simply use + // `DeleteRangesWithTransaction()` only when **first** range is in + // same block? + { + AutoTrackDOMRange firstRangeTracker(aHTMLEditor.RangeUpdaterRef(), + &aRangesToDelete.FirstRangeRef()); + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteRangesWithTransaction( + aDirectionAndAmount, aStripWrappers, aRangesToDelete); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + if (NS_WARN_IF(caretPointOrError.inspectErr() == + NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING( + "EditorBase::DeleteRangesWithTransaction() failed, but ignored"); + } else { + nsresult rv = caretPointOrError.inspect().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + } + } + nsresult rv = + mDeleteRangesHandler->DeleteUnnecessaryNodesAndCollapseSelection( + aHTMLEditor, aDirectionAndAmount, + EditorDOMPoint(aRangesToDelete.FirstRangeRef()->StartRef()), + EditorDOMPoint(aRangesToDelete.FirstRangeRef()->EndRef())); + if (NS_FAILED(rv)) { + NS_WARNING( + "AutoDeleteRangesHandler::DeleteUnnecessaryNodesAndCollapseSelection() " + "failed"); + return Err(rv); + } + return EditActionResult::HandledResult(); +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + ComputeRangesToJoinBlockElementsInSameParent( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete) const { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.IsCollapsed()); + MOZ_ASSERT(mMode == Mode::JoinBlocksInSameParent); + MOZ_ASSERT(mLeftContent); + MOZ_ASSERT(mLeftContent->IsElement()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->IsInclusiveDescendantOf(mLeftContent)); + MOZ_ASSERT(mRightContent); + MOZ_ASSERT(mRightContent->IsElement()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetEndContainer() + ->IsInclusiveDescendantOf(mRightContent)); + MOZ_ASSERT(mLeftContent->GetParentNode() == mRightContent->GetParentNode()); + + nsresult rv = + mDeleteRangesHandlerConst.ComputeRangesToDeleteRangesWithTransaction( + aHTMLEditor, aDirectionAndAmount, aRangesToDelete); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::" + "ComputeRangesToDeleteRangesWithTransaction() failed"); + return rv; +} + +Result<EditActionResult, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoBlockElementsJoiner::JoinBlockElementsInSameParent( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + AutoRangeArray& aRangesToDelete, + SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.IsCollapsed()); + MOZ_ASSERT(mMode == Mode::JoinBlocksInSameParent); + MOZ_ASSERT(mLeftContent); + MOZ_ASSERT(mLeftContent->IsElement()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->IsInclusiveDescendantOf(mLeftContent)); + MOZ_ASSERT(mRightContent); + MOZ_ASSERT(mRightContent->IsElement()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetEndContainer() + ->IsInclusiveDescendantOf(mRightContent)); + MOZ_ASSERT(mLeftContent->GetParentNode() == mRightContent->GetParentNode()); + + const bool backspaceInRightBlock = + aSelectionWasCollapsed == SelectionWasCollapsed::Yes && + nsIEditor::DirectionIsBackspace(aDirectionAndAmount); + + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteRangesWithTransaction(aDirectionAndAmount, + aStripWrappers, aRangesToDelete); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("EditorBase::DeleteRangesWithTransaction() failed"); + return caretPointOrError.propagateErr(); + } + + nsresult rv = caretPointOrError.inspect().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION(rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + + if (NS_WARN_IF(!mLeftContent->GetParentNode()) || + NS_WARN_IF(!mRightContent->GetParentNode()) || + NS_WARN_IF(mLeftContent->GetParentNode() != + mRightContent->GetParentNode())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + auto startOfRightContent = + HTMLEditUtils::GetDeepestEditableStartPointOf<EditorDOMPoint>( + *mRightContent); + AutoTrackDOMPoint trackStartOfRightContent(aHTMLEditor.RangeUpdaterRef(), + &startOfRightContent); + Result<EditorDOMPoint, nsresult> atFirstChildOfTheLastRightNodeOrError = + JoinNodesDeepWithTransaction(aHTMLEditor, MOZ_KnownLive(*mLeftContent), + MOZ_KnownLive(*mRightContent)); + if (MOZ_UNLIKELY(atFirstChildOfTheLastRightNodeOrError.isErr())) { + NS_WARNING("HTMLEditor::JoinNodesDeepWithTransaction() failed"); + return atFirstChildOfTheLastRightNodeOrError.propagateErr(); + } + MOZ_ASSERT(atFirstChildOfTheLastRightNodeOrError.inspect().IsSet()); + trackStartOfRightContent.FlushAndStopTracking(); + if (NS_WARN_IF(!startOfRightContent.IsSet()) || + NS_WARN_IF(!startOfRightContent.GetContainer()->IsInComposedDoc())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + // If we're deleting selection (not replacing with new content) and the joined + // point follows a text node, we should put caret to end of the preceding text + // node because the other browsers insert following inputs into there. + if (MayEditActionDeleteAroundCollapsedSelection( + aHTMLEditor.GetEditAction())) { + WSRunScanner scanner(&aEditingHost, startOfRightContent); + WSScanResult maybePreviousText = + scanner.ScanPreviousVisibleNodeOrBlockBoundaryFrom(startOfRightContent); + if (maybePreviousText.IsContentEditable() && + maybePreviousText.InVisibleOrCollapsibleCharacters()) { + nsresult rv = aHTMLEditor.CollapseSelectionTo( + maybePreviousText.Point<EditorRawDOMPoint>()); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return Err(rv); + } + // If we prefer to use style in the previous line, we should forget + // previous styles since the caret position has all styles which we want + // to use with new content. + if (backspaceInRightBlock) { + aHTMLEditor.TopLevelEditSubActionDataRef() + .mCachedPendingStyles->Clear(); + } + // And we don't want to keep extending a link at ex-end of the previous + // paragraph. + if (HTMLEditor::GetLinkElement(maybePreviousText.TextPtr())) { + aHTMLEditor.mPendingStylesToApplyToNewContent + ->ClearLinkAndItsSpecifiedStyle(); + } + return EditActionResult::HandledResult(); + } + } + + // Otherwise, we should put caret at start of the right content. + rv = aHTMLEditor.CollapseSelectionTo( + atFirstChildOfTheLastRightNodeOrError.inspect()); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return Err(rv); + } + return EditActionResult::HandledResult(); +} + +Result<bool, nsresult> +HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + ComputeRangesToDeleteNodesEntirelyInRangeButKeepTableStructure( + const HTMLEditor& aHTMLEditor, nsRange& aRange, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed) + const { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + + AutoTArray<OwningNonNull<nsIContent>, 10> arrayOfTopChildren; + DOMSubtreeIterator iter; + nsresult rv = iter.Init(aRange); + if (NS_FAILED(rv)) { + NS_WARNING("DOMSubtreeIterator::Init() failed"); + return Err(rv); + } + iter.AppendAllNodesToArray(arrayOfTopChildren); + return NeedsToJoinNodesAfterDeleteNodesEntirelyInRangeButKeepTableStructure( + aHTMLEditor, arrayOfTopChildren, aSelectionWasCollapsed); +} + +Result<bool, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoBlockElementsJoiner::DeleteNodesEntirelyInRangeButKeepTableStructure( + HTMLEditor& aHTMLEditor, nsRange& aRange, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + + // Build a list of direct child nodes in the range + AutoTArray<OwningNonNull<nsIContent>, 10> arrayOfTopChildren; + DOMSubtreeIterator iter; + nsresult rv = iter.Init(aRange); + if (NS_FAILED(rv)) { + NS_WARNING("DOMSubtreeIterator::Init() failed"); + return Err(rv); + } + iter.AppendAllNodesToArray(arrayOfTopChildren); + + // Now that we have the list, delete non-table elements + bool needsToJoinLater = + NeedsToJoinNodesAfterDeleteNodesEntirelyInRangeButKeepTableStructure( + aHTMLEditor, arrayOfTopChildren, aSelectionWasCollapsed); + for (auto& content : arrayOfTopChildren) { + // XXX After here, the child contents in the array may have been moved + // to somewhere or removed. We should handle it. + // + // MOZ_KnownLive because 'arrayOfTopChildren' is guaranteed to + // keep it alive. + // + // Even with https://bugzilla.mozilla.org/show_bug.cgi?id=1620312 fixed + // this might need to stay, because 'arrayOfTopChildren' is not const, + // so it's not obvious how to prove via static analysis that it won't + // change and release us. + nsresult rv = + DeleteContentButKeepTableStructure(aHTMLEditor, MOZ_KnownLive(content)); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoBlockElementsJoiner::DeleteContentButKeepTableStructure() failed, " + "but ignored"); + } + return needsToJoinLater; +} + +bool HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + NeedsToJoinNodesAfterDeleteNodesEntirelyInRangeButKeepTableStructure( + const HTMLEditor& aHTMLEditor, + const nsTArray<OwningNonNull<nsIContent>>& aArrayOfContents, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed) + const { + // If original selection was collapsed, we need always to join the nodes. + // XXX Why? + if (aSelectionWasCollapsed == + AutoDeleteRangesHandler::SelectionWasCollapsed::No) { + return true; + } + // If something visible is deleted, no need to join. Visible means + // all nodes except non-visible textnodes and breaks. + if (aArrayOfContents.IsEmpty()) { + return true; + } + for (const OwningNonNull<nsIContent>& content : aArrayOfContents) { + if (content->IsText()) { + if (HTMLEditUtils::IsInVisibleTextFrames(aHTMLEditor.GetPresContext(), + *content->AsText())) { + return false; + } + continue; + } + // XXX If it's an element node, we should check whether it has visible + // frames or not. + if (!content->IsElement() || + HTMLEditUtils::IsEmptyNode( + *content->AsElement(), + {EmptyCheckOption::TreatSingleBRElementAsVisible})) { + continue; + } + if (!HTMLEditUtils::IsInvisibleBRElement(*content)) { + return false; + } + } + return true; +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + DeleteTextAtStartAndEndOfRange(HTMLEditor& aHTMLEditor, nsRange& aRange) { + EditorDOMPoint rangeStart(aRange.StartRef()); + EditorDOMPoint rangeEnd(aRange.EndRef()); + if (rangeStart.IsInTextNode() && !rangeStart.IsEndOfContainer()) { + // Delete to last character + OwningNonNull<Text> textNode = *rangeStart.ContainerAs<Text>(); + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteTextWithTransaction( + textNode, rangeStart.Offset(), + rangeStart.GetContainer()->Length() - rangeStart.Offset()); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("HTMLEditor::DeleteTextWithTransaction() failed"); + return caretPointOrError.unwrapErr(); + } + nsresult rv = caretPointOrError.inspect().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return rv; + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + } + if (rangeEnd.IsInTextNode() && !rangeEnd.IsStartOfContainer()) { + // Delete to first character + OwningNonNull<Text> textNode = *rangeEnd.ContainerAs<Text>(); + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteTextWithTransaction(textNode, 0, rangeEnd.Offset()); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("HTMLEditor::DeleteTextWithTransaction() failed"); + return caretPointOrError.unwrapErr(); + } + nsresult rv = caretPointOrError.inspect().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return rv; + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + } + return NS_OK; +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + ComputeRangesToDeleteNonCollapsedRanges( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost) const { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.IsCollapsed()); + MOZ_ASSERT(mLeftContent); + MOZ_ASSERT(mLeftContent->IsElement()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->IsInclusiveDescendantOf(mLeftContent)); + MOZ_ASSERT(mRightContent); + MOZ_ASSERT(mRightContent->IsElement()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetEndContainer() + ->IsInclusiveDescendantOf(mRightContent)); + + for (OwningNonNull<nsRange>& range : aRangesToDelete.Ranges()) { + Result<bool, nsresult> result = + ComputeRangesToDeleteNodesEntirelyInRangeButKeepTableStructure( + aHTMLEditor, range, aSelectionWasCollapsed); + if (result.isErr()) { + NS_WARNING( + "AutoBlockElementsJoiner::" + "ComputeRangesToDeleteNodesEntirelyInRangeButKeepTableStructure() " + "failed"); + return result.unwrapErr(); + } + if (!result.unwrap()) { + return NS_OK; + } + } + + AutoInclusiveAncestorBlockElementsJoiner joiner(*mLeftContent, + *mRightContent); + Result<bool, nsresult> canJoinThem = + joiner.Prepare(aHTMLEditor, aEditingHost); + if (canJoinThem.isErr()) { + NS_WARNING("AutoInclusiveAncestorBlockElementsJoiner::Prepare() failed"); + return canJoinThem.unwrapErr(); + } + + if (!canJoinThem.unwrap()) { + return NS_SUCCESS_DOM_NO_OPERATION; + } + + if (!joiner.CanJoinBlocks()) { + return NS_OK; + } + + nsresult rv = joiner.ComputeRangesToDelete(aHTMLEditor, EditorDOMPoint(), + aRangesToDelete); + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoInclusiveAncestorBlockElementsJoiner::ComputeRangesToDelete() " + "failed"); + return rv; +} + +Result<EditActionResult, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoBlockElementsJoiner::HandleDeleteNonCollapsedRanges( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + nsIEditor::EStripWrappers aStripWrappers, + AutoRangeArray& aRangesToDelete, + AutoDeleteRangesHandler::SelectionWasCollapsed aSelectionWasCollapsed, + const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.IsCollapsed()); + MOZ_ASSERT(mDeleteRangesHandler); + MOZ_ASSERT(mLeftContent); + MOZ_ASSERT(mLeftContent->IsElement()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetStartContainer() + ->IsInclusiveDescendantOf(mLeftContent)); + MOZ_ASSERT(mRightContent); + MOZ_ASSERT(mRightContent->IsElement()); + MOZ_ASSERT(aRangesToDelete.FirstRangeRef() + ->GetEndContainer() + ->IsInclusiveDescendantOf(mRightContent)); + + const bool backspaceInRightBlock = + aSelectionWasCollapsed == SelectionWasCollapsed::Yes && + nsIEditor::DirectionIsBackspace(aDirectionAndAmount); + + // Otherwise, delete every nodes in all ranges, then, clean up something. + EditActionResult result = EditActionResult::IgnoredResult(); + EditorDOMPoint pointToPutCaret; + while (true) { + AutoTrackDOMRange firstRangeTracker(aHTMLEditor.RangeUpdaterRef(), + &aRangesToDelete.FirstRangeRef()); + + bool joinInclusiveAncestorBlockElements = true; + for (auto& range : aRangesToDelete.Ranges()) { + Result<bool, nsresult> deleteResult = + DeleteNodesEntirelyInRangeButKeepTableStructure( + aHTMLEditor, MOZ_KnownLive(range), aSelectionWasCollapsed); + if (MOZ_UNLIKELY(deleteResult.isErr())) { + NS_WARNING( + "AutoBlockElementsJoiner::" + "DeleteNodesEntirelyInRangeButKeepTableStructure() failed"); + return deleteResult.propagateErr(); + } + // XXX Completely odd. Why don't we join blocks around each range? + joinInclusiveAncestorBlockElements &= deleteResult.unwrap(); + } + + // Check endpoints for possible text deletion. We can assume that if + // text node is found, we can delete to end or to begining as + // appropriate, since the case where both sel endpoints in same text + // node was already handled (we wouldn't be here) + nsresult rv = DeleteTextAtStartAndEndOfRange( + aHTMLEditor, MOZ_KnownLive(aRangesToDelete.FirstRangeRef())); + if (NS_FAILED(rv)) { + NS_WARNING( + "AutoBlockElementsJoiner::DeleteTextAtStartAndEndOfRange() failed"); + return Err(rv); + } + + if (!joinInclusiveAncestorBlockElements) { + break; + } + + AutoInclusiveAncestorBlockElementsJoiner joiner(*mLeftContent, + *mRightContent); + Result<bool, nsresult> canJoinThem = + joiner.Prepare(aHTMLEditor, aEditingHost); + if (canJoinThem.isErr()) { + NS_WARNING("AutoInclusiveAncestorBlockElementsJoiner::Prepare() failed"); + return canJoinThem.propagateErr(); + } + + // If we're joining blocks: if deleting forward the selection should + // be collapsed to the end of the selection, if deleting backward the + // selection should be collapsed to the beginning of the selection. + // But if we're not joining then the selection should collapse to the + // beginning of the selection if we'redeleting forward, because the + // end of the selection will still be in the next block. And same + // thing for deleting backwards (selection should collapse to the end, + // because the beginning will still be in the first block). See Bug + // 507936. + if (aDirectionAndAmount == nsIEditor::eNext) { + aDirectionAndAmount = nsIEditor::ePrevious; + } else { + aDirectionAndAmount = nsIEditor::eNext; + } + + if (!canJoinThem.inspect()) { + result.MarkAsCanceled(); + break; + } + + if (!joiner.CanJoinBlocks()) { + break; + } + + Result<EditActionResult, nsresult> joinResult = + joiner.Run(aHTMLEditor, aEditingHost); + if (MOZ_UNLIKELY(joinResult.isErr())) { + NS_WARNING("AutoInclusiveAncestorBlockElementsJoiner::Run() failed"); + return joinResult; + } + result |= joinResult.unwrap(); +#ifdef DEBUG + if (joiner.ShouldDeleteLeafContentInstead()) { + NS_ASSERTION(result.Ignored(), + "Assumed `AutoInclusiveAncestorBlockElementsJoiner::Run()` " + "returning ignored, but returned not ignored"); + } else { + NS_ASSERTION(!result.Ignored(), + "Assumed `AutoInclusiveAncestorBlockElementsJoiner::Run()` " + "returning handled, but returned ignored"); + } +#endif // #ifdef DEBUG + pointToPutCaret = joiner.PointRefToPutCaret(); + break; + } + + // If we're deleting selection (not replacing with new content) and + // AutoInclusiveAncestorBlockElementsJoiner computed new caret position, we + // should use it. Otherwise, we should keep the traditional behavior. + if (result.Handled() && pointToPutCaret.IsSet()) { + EditorDOMRange range(aRangesToDelete.FirstRangeRef()); + nsresult rv = + mDeleteRangesHandler->DeleteUnnecessaryNodes(aHTMLEditor, range); + if (NS_FAILED(rv)) { + NS_WARNING("AutoDeleteRangesHandler::DeleteUnnecessaryNodes() failed"); + return Err(rv); + } + rv = aHTMLEditor.CollapseSelectionTo(pointToPutCaret); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return Err(rv); + } + // If we prefer to use style in the previous line, we should forget + // previous styles since the caret position has all styles which we want + // to use with new content. + if (backspaceInRightBlock) { + aHTMLEditor.TopLevelEditSubActionDataRef().mCachedPendingStyles->Clear(); + } + // And we don't want to keep extending a link at ex-end of the previous + // paragraph. + if (HTMLEditor::GetLinkElement(pointToPutCaret.GetContainer())) { + aHTMLEditor.mPendingStylesToApplyToNewContent + ->ClearLinkAndItsSpecifiedStyle(); + } + return result; + } + + nsresult rv = + mDeleteRangesHandler->DeleteUnnecessaryNodesAndCollapseSelection( + aHTMLEditor, aDirectionAndAmount, + EditorDOMPoint(aRangesToDelete.FirstRangeRef()->StartRef()), + EditorDOMPoint(aRangesToDelete.FirstRangeRef()->EndRef())); + if (NS_FAILED(rv)) { + NS_WARNING( + "AutoDeleteRangesHandler::DeleteUnnecessaryNodesAndCollapseSelection() " + "failed"); + return Err(rv); + } + + result.MarkAsHandled(); + return result; +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::DeleteUnnecessaryNodes( + HTMLEditor& aHTMLEditor, EditorDOMRange& aRange) { + MOZ_ASSERT(aHTMLEditor.IsTopLevelEditSubActionDataAvailable()); + MOZ_ASSERT(EditorUtils::IsEditableContent( + *aRange.StartRef().ContainerAs<nsIContent>(), EditorType::HTML)); + MOZ_ASSERT(EditorUtils::IsEditableContent( + *aRange.EndRef().ContainerAs<nsIContent>(), EditorType::HTML)); + + // If we're handling DnD, this is called to delete dragging item from the + // tree. In this case, we should remove parent blocks if it becomes empty. + if (aHTMLEditor.GetEditAction() == EditAction::eDrop || + aHTMLEditor.GetEditAction() == EditAction::eDeleteByDrag) { + MOZ_ASSERT(aRange.Collapsed() || + (aRange.StartRef().GetContainer()->GetNextSibling() == + aRange.EndRef().GetContainer() && + aRange.StartRef().IsEndOfContainer() && + aRange.EndRef().IsStartOfContainer())); + AutoTrackDOMRange trackRange(aHTMLEditor.RangeUpdaterRef(), &aRange); + + nsresult rv = DeleteParentBlocksWithTransactionIfEmpty(aHTMLEditor, + aRange.StartRef()); + if (NS_FAILED(rv)) { + NS_WARNING( + "HTMLEditor::DeleteParentBlocksWithTransactionIfEmpty() failed"); + return rv; + } + aHTMLEditor.TopLevelEditSubActionDataRef().mDidDeleteEmptyParentBlocks = + rv == NS_OK; + // If we removed parent blocks, Selection should be collapsed at where + // the most ancestor empty block has been. + if (aHTMLEditor.TopLevelEditSubActionDataRef() + .mDidDeleteEmptyParentBlocks) { + return NS_OK; + } + } + + if (NS_WARN_IF(!aRange.IsInContentNodes()) || + NS_WARN_IF(!EditorUtils::IsEditableContent( + *aRange.StartRef().ContainerAs<nsIContent>(), EditorType::HTML)) || + NS_WARN_IF(!EditorUtils::IsEditableContent( + *aRange.EndRef().ContainerAs<nsIContent>(), EditorType::HTML))) { + return NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE; + } + + // We might have left only collapsed white-space in the start/end nodes + AutoTrackDOMRange trackRange(aHTMLEditor.RangeUpdaterRef(), &aRange); + + OwningNonNull<nsIContent> startContainer = + *aRange.StartRef().ContainerAs<nsIContent>(); + OwningNonNull<nsIContent> endContainer = + *aRange.EndRef().ContainerAs<nsIContent>(); + nsresult rv = + DeleteNodeIfInvisibleAndEditableTextNode(aHTMLEditor, startContainer); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::DeleteNodeIfInvisibleAndEditableTextNode() " + "failed to remove start node, but ignored"); + // If we've not handled the selection end container, and it's still + // editable, let's handle it. + if (aRange.InSameContainer() || + !EditorUtils::IsEditableContent( + *aRange.EndRef().ContainerAs<nsIContent>(), EditorType::HTML)) { + return NS_OK; + } + rv = DeleteNodeIfInvisibleAndEditableTextNode(aHTMLEditor, endContainer); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::DeleteNodeIfInvisibleAndEditableTextNode() " + "failed to remove end node, but ignored"); + return NS_OK; +} + +nsresult +HTMLEditor::AutoDeleteRangesHandler::DeleteUnnecessaryNodesAndCollapseSelection( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + const EditorDOMPoint& aSelectionStartPoint, + const EditorDOMPoint& aSelectionEndPoint) { + EditorDOMRange range(aSelectionStartPoint, aSelectionEndPoint); + nsresult rv = DeleteUnnecessaryNodes(aHTMLEditor, range); + if (NS_FAILED(rv)) { + NS_WARNING("AutoDeleteRangesHandler::DeleteUnnecessaryNodes() failed"); + return rv; + } + + if (aHTMLEditor.GetEditAction() == EditAction::eDrop || + aHTMLEditor.GetEditAction() == EditAction::eDeleteByDrag) { + // If we removed parent blocks, Selection should be collapsed at where + // the most ancestor empty block has been. + // XXX I think that if the range is not in active editing host, we should + // not try to collapse selection here. + if (aHTMLEditor.TopLevelEditSubActionDataRef() + .mDidDeleteEmptyParentBlocks) { + nsresult rv = aHTMLEditor.CollapseSelectionTo(range.StartRef()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; + } + } + + rv = aHTMLEditor.CollapseSelectionTo( + aDirectionAndAmount == nsIEditor::ePrevious ? range.EndRef() + : range.StartRef()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return rv; +} + +nsresult +HTMLEditor::AutoDeleteRangesHandler::DeleteNodeIfInvisibleAndEditableTextNode( + HTMLEditor& aHTMLEditor, nsIContent& aContent) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + + Text* text = aContent.GetAsText(); + if (!text) { + return NS_OK; + } + + if (!HTMLEditUtils::IsRemovableFromParentNode(*text) || + HTMLEditUtils::IsVisibleTextNode(*text)) { + return NS_OK; + } + + nsresult rv = aHTMLEditor.DeleteNodeWithTransaction(aContent); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::DeleteNodeWithTransaction() failed"); + return rv; +} + +nsresult +HTMLEditor::AutoDeleteRangesHandler::DeleteParentBlocksWithTransactionIfEmpty( + HTMLEditor& aHTMLEditor, const EditorDOMPoint& aPoint) { + MOZ_ASSERT(aPoint.IsSet()); + MOZ_ASSERT(aHTMLEditor.mPlaceholderBatch); + + // First, check there is visible contents before the point in current block. + RefPtr<Element> editingHost = aHTMLEditor.ComputeEditingHost(); + WSRunScanner wsScannerForPoint(editingHost, aPoint); + if (!wsScannerForPoint.StartsFromCurrentBlockBoundary()) { + // If there is visible node before the point, we shouldn't remove the + // parent block. + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + if (NS_WARN_IF(!wsScannerForPoint.GetStartReasonContent()) || + NS_WARN_IF(!wsScannerForPoint.GetStartReasonContent()->GetParentNode())) { + return NS_ERROR_FAILURE; + } + if (editingHost == wsScannerForPoint.GetStartReasonContent()) { + // If we reach editing host, there is no parent blocks which can be removed. + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + if (HTMLEditUtils::IsTableCellOrCaption( + *wsScannerForPoint.GetStartReasonContent())) { + // If we reach a <td>, <th> or <caption>, we shouldn't remove it even + // becomes empty because removing such element changes the structure of + // the <table>. + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + + // Next, check there is visible contents after the point in current block. + WSScanResult forwardScanFromPointResult = + wsScannerForPoint.ScanNextVisibleNodeOrBlockBoundaryFrom(aPoint); + if (forwardScanFromPointResult.Failed()) { + NS_WARNING("WSRunScanner::ScanNextVisibleNodeOrBlockBoundaryFrom() failed"); + return NS_ERROR_FAILURE; + } + if (forwardScanFromPointResult.ReachedBRElement()) { + // XXX In my understanding, this is odd. The end reason may not be + // same as the reached <br> element because the equality is + // guaranteed only when ReachedCurrentBlockBoundary() returns true. + // However, looks like that this code assumes that + // GetEndReasonContent() returns the (or a) <br> element. + NS_ASSERTION(wsScannerForPoint.GetEndReasonContent() == + forwardScanFromPointResult.BRElementPtr(), + "End reason is not the reached <br> element"); + // If the <br> element is visible, we shouldn't remove the parent block. + if (HTMLEditUtils::IsVisibleBRElement( + *wsScannerForPoint.GetEndReasonContent())) { + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + if (wsScannerForPoint.GetEndReasonContent()->GetNextSibling()) { + WSScanResult scanResult = + WSRunScanner::ScanNextVisibleNodeOrBlockBoundary( + editingHost, EditorRawDOMPoint::After( + *wsScannerForPoint.GetEndReasonContent())); + if (scanResult.Failed()) { + NS_WARNING("WSRunScanner::ScanNextVisibleNodeOrBlockBoundary() failed"); + return NS_ERROR_FAILURE; + } + if (!scanResult.ReachedCurrentBlockBoundary()) { + // If we couldn't reach the block's end after the invisible <br>, + // that means that there is visible content. + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + } + } else if (!forwardScanFromPointResult.ReachedCurrentBlockBoundary()) { + // If we couldn't reach the block's end, the block has visible content. + return NS_SUCCESS_EDITOR_ELEMENT_NOT_FOUND; + } + + // Delete the parent block. + EditorDOMPoint nextPoint( + wsScannerForPoint.GetStartReasonContent()->GetParentNode(), 0); + nsresult rv = aHTMLEditor.DeleteNodeWithTransaction( + MOZ_KnownLive(*wsScannerForPoint.GetStartReasonContent())); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return rv; + } + // If we reach editing host, return NS_OK. + if (nextPoint.GetContainer() == editingHost) { + return NS_OK; + } + + // Otherwise, we need to check whether we're still in empty block or not. + + // If we have mutation event listeners, the next point is now outside of + // editing host or editing hos has been changed. + if (aHTMLEditor.MayHaveMutationEventListeners( + NS_EVENT_BITS_MUTATION_NODEREMOVED | + NS_EVENT_BITS_MUTATION_NODEREMOVEDFROMDOCUMENT | + NS_EVENT_BITS_MUTATION_SUBTREEMODIFIED)) { + Element* newEditingHost = aHTMLEditor.ComputeEditingHost(); + if (NS_WARN_IF(!newEditingHost) || + NS_WARN_IF(newEditingHost != editingHost)) { + return NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE; + } + if (NS_WARN_IF(!EditorUtils::IsDescendantOf(*nextPoint.GetContainer(), + *newEditingHost))) { + return NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE; + } + } + + rv = DeleteParentBlocksWithTransactionIfEmpty(aHTMLEditor, nextPoint); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoDeleteRangesHandler::" + "DeleteParentBlocksWithTransactionIfEmpty() failed"); + return rv; +} + +nsresult +HTMLEditor::AutoDeleteRangesHandler::ComputeRangesToDeleteRangesWithTransaction( + const HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount, + AutoRangeArray& aRangesToDelete) const { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangesToDelete.Ranges().IsEmpty()); + + EditorBase::HowToHandleCollapsedRange howToHandleCollapsedRange = + EditorBase::HowToHandleCollapsedRangeFor(aDirectionAndAmount); + if (NS_WARN_IF(aRangesToDelete.IsCollapsed() && + howToHandleCollapsedRange == + EditorBase::HowToHandleCollapsedRange::Ignore)) { + return NS_ERROR_FAILURE; + } + + auto extendRangeToSelectCharacterForward = + [](nsRange& aRange, const EditorRawDOMPointInText& aCaretPoint) -> void { + const nsTextFragment& textFragment = + aCaretPoint.ContainerAs<Text>()->TextFragment(); + if (!textFragment.GetLength()) { + return; + } + if (textFragment.IsHighSurrogateFollowedByLowSurrogateAt( + aCaretPoint.Offset())) { + DebugOnly<nsresult> rvIgnored = aRange.SetStartAndEnd( + aCaretPoint.ContainerAs<Text>(), aCaretPoint.Offset(), + aCaretPoint.ContainerAs<Text>(), aCaretPoint.Offset() + 2); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "nsRange::SetStartAndEnd() failed"); + return; + } + DebugOnly<nsresult> rvIgnored = aRange.SetStartAndEnd( + aCaretPoint.ContainerAs<Text>(), aCaretPoint.Offset(), + aCaretPoint.ContainerAs<Text>(), aCaretPoint.Offset() + 1); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "nsRange::SetStartAndEnd() failed"); + }; + auto extendRangeToSelectCharacterBackward = + [](nsRange& aRange, const EditorRawDOMPointInText& aCaretPoint) -> void { + if (aCaretPoint.IsStartOfContainer()) { + return; + } + const nsTextFragment& textFragment = + aCaretPoint.ContainerAs<Text>()->TextFragment(); + if (!textFragment.GetLength()) { + return; + } + if (textFragment.IsLowSurrogateFollowingHighSurrogateAt( + aCaretPoint.Offset() - 1)) { + DebugOnly<nsresult> rvIgnored = aRange.SetStartAndEnd( + aCaretPoint.ContainerAs<Text>(), aCaretPoint.Offset() - 2, + aCaretPoint.ContainerAs<Text>(), aCaretPoint.Offset()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "nsRange::SetStartAndEnd() failed"); + return; + } + DebugOnly<nsresult> rvIgnored = aRange.SetStartAndEnd( + aCaretPoint.ContainerAs<Text>(), aCaretPoint.Offset() - 1, + aCaretPoint.ContainerAs<Text>(), aCaretPoint.Offset()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "nsRange::SetStartAndEnd() failed"); + }; + + RefPtr<Element> editingHost = aHTMLEditor.ComputeEditingHost(); + for (OwningNonNull<nsRange>& range : aRangesToDelete.Ranges()) { + // If it's not collapsed, `DeleteRangeTransaction::Create()` will be called + // with it and `DeleteRangeTransaction` won't modify the range. + if (!range->Collapsed()) { + continue; + } + + if (howToHandleCollapsedRange == + EditorBase::HowToHandleCollapsedRange::Ignore) { + continue; + } + + // In the other cases, `EditorBase::CreateTransactionForCollapsedRange()` + // will handle the collapsed range. + EditorRawDOMPoint caretPoint(range->StartRef()); + if (howToHandleCollapsedRange == + EditorBase::HowToHandleCollapsedRange::ExtendBackward && + caretPoint.IsStartOfContainer()) { + nsIContent* previousEditableContent = HTMLEditUtils::GetPreviousContent( + *caretPoint.GetContainer(), {WalkTreeOption::IgnoreNonEditableNode}, + editingHost); + if (!previousEditableContent) { + continue; + } + if (!previousEditableContent->IsText()) { + IgnoredErrorResult ignoredError; + range->SelectNode(*previousEditableContent, ignoredError); + NS_WARNING_ASSERTION(!ignoredError.Failed(), + "nsRange::SelectNode() failed"); + continue; + } + + extendRangeToSelectCharacterBackward( + range, + EditorRawDOMPointInText::AtEndOf(*previousEditableContent->AsText())); + continue; + } + + if (howToHandleCollapsedRange == + EditorBase::HowToHandleCollapsedRange::ExtendForward && + caretPoint.IsEndOfContainer()) { + nsIContent* nextEditableContent = HTMLEditUtils::GetNextContent( + *caretPoint.GetContainer(), {WalkTreeOption::IgnoreNonEditableNode}, + editingHost); + if (!nextEditableContent) { + continue; + } + + if (!nextEditableContent->IsText()) { + IgnoredErrorResult ignoredError; + range->SelectNode(*nextEditableContent, ignoredError); + NS_WARNING_ASSERTION(!ignoredError.Failed(), + "nsRange::SelectNode() failed"); + continue; + } + + extendRangeToSelectCharacterForward( + range, EditorRawDOMPointInText(nextEditableContent->AsText(), 0)); + continue; + } + + if (caretPoint.IsInTextNode()) { + if (howToHandleCollapsedRange == + EditorBase::HowToHandleCollapsedRange::ExtendBackward) { + extendRangeToSelectCharacterBackward( + range, EditorRawDOMPointInText(caretPoint.ContainerAs<Text>(), + caretPoint.Offset())); + continue; + } + extendRangeToSelectCharacterForward( + range, EditorRawDOMPointInText(caretPoint.ContainerAs<Text>(), + caretPoint.Offset())); + continue; + } + + nsIContent* editableContent = + howToHandleCollapsedRange == + EditorBase::HowToHandleCollapsedRange::ExtendBackward + ? HTMLEditUtils::GetPreviousContent( + caretPoint, {WalkTreeOption::IgnoreNonEditableNode}, + editingHost) + : HTMLEditUtils::GetNextContent( + caretPoint, {WalkTreeOption::IgnoreNonEditableNode}, + editingHost); + if (!editableContent) { + continue; + } + while (editableContent && editableContent->IsCharacterData() && + !editableContent->Length()) { + editableContent = + howToHandleCollapsedRange == + EditorBase::HowToHandleCollapsedRange::ExtendBackward + ? HTMLEditUtils::GetPreviousContent( + *editableContent, {WalkTreeOption::IgnoreNonEditableNode}, + editingHost) + : HTMLEditUtils::GetNextContent( + *editableContent, {WalkTreeOption::IgnoreNonEditableNode}, + editingHost); + } + if (!editableContent) { + continue; + } + + if (!editableContent->IsText()) { + IgnoredErrorResult ignoredError; + range->SelectNode(*editableContent, ignoredError); + NS_WARNING_ASSERTION(!ignoredError.Failed(), + "nsRange::SelectNode() failed"); + continue; + } + + if (howToHandleCollapsedRange == + EditorBase::HowToHandleCollapsedRange::ExtendBackward) { + extendRangeToSelectCharacterBackward( + range, EditorRawDOMPointInText::AtEndOf(*editableContent->AsText())); + continue; + } + extendRangeToSelectCharacterForward( + range, EditorRawDOMPointInText(editableContent->AsText(), 0)); + } + + return NS_OK; +} + +template <typename EditorDOMPointType> +Result<CaretPoint, nsresult> HTMLEditor::DeleteTextAndTextNodesWithTransaction( + const EditorDOMPointType& aStartPoint, const EditorDOMPointType& aEndPoint, + TreatEmptyTextNodes aTreatEmptyTextNodes) { + if (NS_WARN_IF(!aStartPoint.IsSet()) || NS_WARN_IF(!aEndPoint.IsSet())) { + return Err(NS_ERROR_INVALID_ARG); + } + + // MOOSE: this routine needs to be modified to preserve the integrity of the + // wsFragment info. + + if (aStartPoint == aEndPoint) { + // Nothing to delete + return CaretPoint(EditorDOMPoint()); + } + + RefPtr<Element> editingHost = ComputeEditingHost(); + auto DeleteEmptyContentNodeWithTransaction = + [this, &aTreatEmptyTextNodes, &editingHost](nsIContent& aContent) + MOZ_CAN_RUN_SCRIPT_FOR_DEFINITION -> nsresult { + OwningNonNull<nsIContent> nodeToRemove = aContent; + if (aTreatEmptyTextNodes == + TreatEmptyTextNodes::RemoveAllEmptyInlineAncestors) { + Element* emptyParentElementToRemove = + HTMLEditUtils::GetMostDistantAncestorEditableEmptyInlineElement( + nodeToRemove, editingHost); + if (emptyParentElementToRemove) { + nodeToRemove = *emptyParentElementToRemove; + } + } + nsresult rv = DeleteNodeWithTransaction(nodeToRemove); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::DeleteNodeWithTransaction() failed"); + return rv; + }; + + if (aStartPoint.GetContainer() == aEndPoint.GetContainer() && + aStartPoint.IsInTextNode()) { + if (aTreatEmptyTextNodes != + TreatEmptyTextNodes::KeepIfContainerOfRangeBoundaries && + aStartPoint.IsStartOfContainer() && aEndPoint.IsEndOfContainer()) { + nsresult rv = DeleteEmptyContentNodeWithTransaction( + MOZ_KnownLive(*aStartPoint.template ContainerAs<Text>())); + if (NS_FAILED(rv)) { + NS_WARNING("deleteEmptyContentNodeWithTransaction() failed"); + return Err(rv); + } + return CaretPoint(EditorDOMPoint()); + } + RefPtr<Text> textNode = aStartPoint.template ContainerAs<Text>(); + Result<CaretPoint, nsresult> caretPointOrError = + DeleteTextWithTransaction(*textNode, aStartPoint.Offset(), + aEndPoint.Offset() - aStartPoint.Offset()); + NS_WARNING_ASSERTION(caretPointOrError.isOk(), + "HTMLEditor::DeleteTextWithTransaction() failed"); + return caretPointOrError; + } + + RefPtr<nsRange> range = + nsRange::Create(aStartPoint.ToRawRangeBoundary(), + aEndPoint.ToRawRangeBoundary(), IgnoreErrors()); + if (!range) { + NS_WARNING("nsRange::Create() failed"); + return Err(NS_ERROR_FAILURE); + } + + // Collect editable text nodes in the given range. + AutoTArray<OwningNonNull<Text>, 16> arrayOfTextNodes; + DOMIterator iter; + if (NS_FAILED(iter.Init(*range))) { + return CaretPoint(EditorDOMPoint()); // Nothing to delete in the range. + } + iter.AppendNodesToArray( + +[](nsINode& aNode, void*) { + MOZ_ASSERT(aNode.IsText()); + return HTMLEditUtils::IsSimplyEditableNode(aNode); + }, + arrayOfTextNodes); + EditorDOMPoint pointToPutCaret; + for (OwningNonNull<Text>& textNode : arrayOfTextNodes) { + if (textNode == aStartPoint.GetContainer()) { + if (aStartPoint.IsEndOfContainer()) { + continue; + } + if (aStartPoint.IsStartOfContainer() && + aTreatEmptyTextNodes != + TreatEmptyTextNodes::KeepIfContainerOfRangeBoundaries) { + AutoTrackDOMPoint trackPointToPutCaret(RangeUpdaterRef(), + &pointToPutCaret); + nsresult rv = DeleteEmptyContentNodeWithTransaction( + MOZ_KnownLive(*aStartPoint.template ContainerAs<Text>())); + if (NS_FAILED(rv)) { + NS_WARNING("DeleteEmptyContentNodeWithTransaction() failed"); + return Err(rv); + } + continue; + } + AutoTrackDOMPoint trackPointToPutCaret(RangeUpdaterRef(), + &pointToPutCaret); + Result<CaretPoint, nsresult> caretPointOrError = + DeleteTextWithTransaction(MOZ_KnownLive(textNode), + aStartPoint.Offset(), + textNode->Length() - aStartPoint.Offset()); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("HTMLEditor::DeleteTextWithTransaction() failed"); + return caretPointOrError; + } + trackPointToPutCaret.FlushAndStopTracking(); + caretPointOrError.unwrap().MoveCaretPointTo( + pointToPutCaret, {SuggestCaret::OnlyIfHasSuggestion}); + continue; + } + + if (textNode == aEndPoint.GetContainer()) { + if (aEndPoint.IsStartOfContainer()) { + break; + } + if (aEndPoint.IsEndOfContainer() && + aTreatEmptyTextNodes != + TreatEmptyTextNodes::KeepIfContainerOfRangeBoundaries) { + AutoTrackDOMPoint trackPointToPutCaret(RangeUpdaterRef(), + &pointToPutCaret); + nsresult rv = DeleteEmptyContentNodeWithTransaction( + MOZ_KnownLive(*aEndPoint.template ContainerAs<Text>())); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "DeleteEmptyContentNodeWithTransaction() failed"); + return Err(rv); + } + AutoTrackDOMPoint trackPointToPutCaret(RangeUpdaterRef(), + &pointToPutCaret); + Result<CaretPoint, nsresult> caretPointOrError = + DeleteTextWithTransaction(MOZ_KnownLive(textNode), 0, + aEndPoint.Offset()); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("HTMLEditor::DeleteTextWithTransaction() failed"); + return caretPointOrError; + } + trackPointToPutCaret.FlushAndStopTracking(); + caretPointOrError.unwrap().MoveCaretPointTo( + pointToPutCaret, {SuggestCaret::OnlyIfHasSuggestion}); + return CaretPoint(pointToPutCaret); + } + + nsresult rv = + DeleteEmptyContentNodeWithTransaction(MOZ_KnownLive(textNode)); + if (NS_FAILED(rv)) { + NS_WARNING("DeleteEmptyContentNodeWithTransaction() failed"); + return Err(rv); + } + } + + return CaretPoint(pointToPutCaret); +} + +Result<EditorDOMPoint, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoBlockElementsJoiner::JoinNodesDeepWithTransaction( + HTMLEditor& aHTMLEditor, nsIContent& aLeftContent, + nsIContent& aRightContent) { + // While the rightmost children and their descendants of the left node match + // the leftmost children and their descendants of the right node, join them + // up. + + nsCOMPtr<nsIContent> leftContentToJoin = &aLeftContent; + nsCOMPtr<nsIContent> rightContentToJoin = &aRightContent; + nsCOMPtr<nsINode> parentNode = aRightContent.GetParentNode(); + + EditorDOMPoint ret; + while (leftContentToJoin && rightContentToJoin && parentNode && + HTMLEditUtils::CanContentsBeJoined(*leftContentToJoin, + *rightContentToJoin)) { + // Do the join + Result<JoinNodesResult, nsresult> joinNodesResult = + aHTMLEditor.JoinNodesWithTransaction(*leftContentToJoin, + *rightContentToJoin); + if (MOZ_UNLIKELY(joinNodesResult.isErr())) { + NS_WARNING("HTMLEditor::JoinNodesWithTransaction() failed"); + return joinNodesResult.propagateErr(); + } + + ret = joinNodesResult.inspect().AtJoinedPoint<EditorDOMPoint>(); + if (NS_WARN_IF(!ret.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + + if (parentNode->IsText()) { + // We've joined all the way down to text nodes, we're done! + return ret; + } + + // Get new left and right nodes, and begin anew + rightContentToJoin = ret.GetCurrentChildAtOffset(); + if (rightContentToJoin) { + leftContentToJoin = rightContentToJoin->GetPreviousSibling(); + } else { + leftContentToJoin = nullptr; + } + + // Skip over non-editable nodes + while (leftContentToJoin && !EditorUtils::IsEditableContent( + *leftContentToJoin, EditorType::HTML)) { + leftContentToJoin = leftContentToJoin->GetPreviousSibling(); + } + if (!leftContentToJoin) { + return ret; + } + + while (rightContentToJoin && !EditorUtils::IsEditableContent( + *rightContentToJoin, EditorType::HTML)) { + rightContentToJoin = rightContentToJoin->GetNextSibling(); + } + if (!rightContentToJoin) { + return ret; + } + } + + if (!ret.IsSet()) { + NS_WARNING("HTMLEditor::JoinNodesDeepWithTransaction() joined no contents"); + return Err(NS_ERROR_FAILURE); + } + return ret; +} + +Result<bool, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoBlockElementsJoiner::AutoInclusiveAncestorBlockElementsJoiner::Prepare( + const HTMLEditor& aHTMLEditor, const Element& aEditingHost) { + mLeftBlockElement = HTMLEditUtils::GetInclusiveAncestorElement( + mInclusiveDescendantOfLeftBlockElement, + HTMLEditUtils::ClosestEditableBlockElementExceptHRElement); + mRightBlockElement = HTMLEditUtils::GetInclusiveAncestorElement( + mInclusiveDescendantOfRightBlockElement, + HTMLEditUtils::ClosestEditableBlockElementExceptHRElement); + + if (NS_WARN_IF(!IsSet())) { + mCanJoinBlocks = false; + return Err(NS_ERROR_UNEXPECTED); + } + + // Don't join the blocks if both of them are basic structure of the HTML + // document (Note that `<body>` can be joined with its children). + if (mLeftBlockElement->IsAnyOfHTMLElements(nsGkAtoms::html, nsGkAtoms::head, + nsGkAtoms::body) && + mRightBlockElement->IsAnyOfHTMLElements(nsGkAtoms::html, nsGkAtoms::head, + nsGkAtoms::body)) { + mCanJoinBlocks = false; + return false; + } + + if (HTMLEditUtils::IsAnyTableElement(mLeftBlockElement) || + HTMLEditUtils::IsAnyTableElement(mRightBlockElement)) { + // Do not try to merge table elements, cancel the deletion. + mCanJoinBlocks = false; + return false; + } + + // Bail if both blocks the same + if (IsSameBlockElement()) { + mCanJoinBlocks = true; // XXX Anyway, Run() will ingore this case. + mFallbackToDeleteLeafContent = true; + return true; + } + + // Joining a list item to its parent is a NOP. + if (HTMLEditUtils::IsAnyListElement(mLeftBlockElement) && + HTMLEditUtils::IsListItem(mRightBlockElement) && + mRightBlockElement->GetParentNode() == mLeftBlockElement) { + mCanJoinBlocks = false; + return true; + } + + // Special rule here: if we are trying to join list items, and they are in + // different lists, join the lists instead. + if (HTMLEditUtils::IsListItem(mLeftBlockElement) && + HTMLEditUtils::IsListItem(mRightBlockElement)) { + // XXX leftListElement and/or rightListElement may be not list elements. + Element* leftListElement = mLeftBlockElement->GetParentElement(); + Element* rightListElement = mRightBlockElement->GetParentElement(); + EditorDOMPoint atChildInBlock; + if (leftListElement && rightListElement && + leftListElement != rightListElement && + !EditorUtils::IsDescendantOf(*leftListElement, *mRightBlockElement, + &atChildInBlock) && + !EditorUtils::IsDescendantOf(*rightListElement, *mLeftBlockElement, + &atChildInBlock)) { + // There are some special complications if the lists are descendants of + // the other lists' items. Note that it is okay for them to be + // descendants of the other lists themselves, which is the usual case for + // sublists in our implementation. + MOZ_DIAGNOSTIC_ASSERT(!atChildInBlock.IsSet()); + mLeftBlockElement = leftListElement; + mRightBlockElement = rightListElement; + mNewListElementTagNameOfRightListElement = + Some(leftListElement->NodeInfo()->NameAtom()); + } + } + + if (!EditorUtils::IsDescendantOf(*mLeftBlockElement, *mRightBlockElement, + &mPointContainingTheOtherBlockElement)) { + Unused << EditorUtils::IsDescendantOf( + *mRightBlockElement, *mLeftBlockElement, + &mPointContainingTheOtherBlockElement); + } + + if (mPointContainingTheOtherBlockElement.GetContainer() == + mRightBlockElement) { + mPrecedingInvisibleBRElement = + WSRunScanner::GetPrecedingBRElementUnlessVisibleContentFound( + aHTMLEditor.ComputeEditingHost(), + EditorDOMPoint::AtEndOf(mLeftBlockElement)); + // `WhiteSpaceVisibilityKeeper:: + // MergeFirstLineOfRightBlockElementIntoDescendantLeftBlockElement()` + // returns ignored when: + // - No preceding invisible `<br>` element and + // - mNewListElementTagNameOfRightListElement is nothing and + // - There is no content to move from right block element. + if (!mPrecedingInvisibleBRElement) { + if (CanMergeLeftAndRightBlockElements()) { + // Always marked as handled in this case. + mFallbackToDeleteLeafContent = false; + } else { + // Marked as handled only when it actually moves a content node. + Result<bool, nsresult> firstLineHasContent = + AutoMoveOneLineHandler::CanMoveOrDeleteSomethingInLine( + mPointContainingTheOtherBlockElement + .NextPoint<EditorDOMPoint>(), + aEditingHost); + mFallbackToDeleteLeafContent = + firstLineHasContent.isOk() && !firstLineHasContent.inspect(); + } + } else { + // Marked as handled when deleting the invisible `<br>` element. + mFallbackToDeleteLeafContent = false; + } + } else if (mPointContainingTheOtherBlockElement.GetContainer() == + mLeftBlockElement) { + mPrecedingInvisibleBRElement = + WSRunScanner::GetPrecedingBRElementUnlessVisibleContentFound( + aHTMLEditor.ComputeEditingHost(), + mPointContainingTheOtherBlockElement); + // `WhiteSpaceVisibilityKeeper:: + // MergeFirstLineOfRightBlockElementIntoAncestorLeftBlockElement()` + // returns ignored when: + // - No preceding invisible `<br>` element and + // - mNewListElementTagNameOfRightListElement is some and + // - The right block element has no children + // or, + // - No preceding invisible `<br>` element and + // - mNewListElementTagNameOfRightListElement is nothing and + // - There is no content to move from right block element. + if (!mPrecedingInvisibleBRElement) { + if (CanMergeLeftAndRightBlockElements()) { + // Marked as handled only when it actualy moves a content node. + Result<bool, nsresult> rightBlockHasContent = + aHTMLEditor.CanMoveChildren(*mRightBlockElement, + *mLeftBlockElement); + mFallbackToDeleteLeafContent = + rightBlockHasContent.isOk() && !rightBlockHasContent.inspect(); + } else { + // Marked as handled only when it actually moves a content node. + Result<bool, nsresult> firstLineHasContent = + AutoMoveOneLineHandler::CanMoveOrDeleteSomethingInLine( + EditorDOMPoint(mRightBlockElement, 0u), aEditingHost); + mFallbackToDeleteLeafContent = + firstLineHasContent.isOk() && !firstLineHasContent.inspect(); + } + } else { + // Marked as handled when deleting the invisible `<br>` element. + mFallbackToDeleteLeafContent = false; + } + } else { + mPrecedingInvisibleBRElement = + WSRunScanner::GetPrecedingBRElementUnlessVisibleContentFound( + aHTMLEditor.ComputeEditingHost(), + EditorDOMPoint::AtEndOf(mLeftBlockElement)); + // `WhiteSpaceVisibilityKeeper:: + // MergeFirstLineOfRightBlockElementIntoLeftBlockElement()` always + // return "handled". + mFallbackToDeleteLeafContent = false; + } + + mCanJoinBlocks = true; + return true; +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + AutoInclusiveAncestorBlockElementsJoiner::ComputeRangesToDelete( + const HTMLEditor& aHTMLEditor, const EditorDOMPoint& aCaretPoint, + AutoRangeArray& aRangesToDelete) const { + MOZ_ASSERT(!aRangesToDelete.Ranges().IsEmpty()); + MOZ_ASSERT(mLeftBlockElement); + MOZ_ASSERT(mRightBlockElement); + + if (IsSameBlockElement()) { + if (!aCaretPoint.IsSet()) { + return NS_OK; // The ranges are not collapsed, keep them as-is. + } + nsresult rv = aRangesToDelete.Collapse(aCaretPoint); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "AutoRangeArray::Collapse() failed"); + return rv; + } + + EditorDOMPoint pointContainingTheOtherBlock; + if (!EditorUtils::IsDescendantOf(*mLeftBlockElement, *mRightBlockElement, + &pointContainingTheOtherBlock)) { + Unused << EditorUtils::IsDescendantOf( + *mRightBlockElement, *mLeftBlockElement, &pointContainingTheOtherBlock); + } + EditorDOMRange range = + WSRunScanner::GetRangeForDeletingBlockElementBoundaries( + aHTMLEditor, *mLeftBlockElement, *mRightBlockElement, + pointContainingTheOtherBlock); + if (!range.IsPositioned()) { + NS_WARNING( + "WSRunScanner::GetRangeForDeletingBlockElementBoundaries() failed"); + return NS_ERROR_FAILURE; + } + if (!aCaretPoint.IsSet()) { + // Don't shrink the original range. + bool noNeedToChangeStart = false; + const auto atStart = + aRangesToDelete.GetFirstRangeStartPoint<EditorDOMPoint>(); + if (atStart.IsBefore(range.StartRef())) { + // If the range starts from end of a container, and computed block + // boundaries range starts from an invisible `<br>` element, we + // may need to shrink the range. + Element* editingHost = aHTMLEditor.ComputeEditingHost(); + NS_WARNING_ASSERTION(editingHost, "There was no editing host"); + nsIContent* nextContent = + atStart.IsEndOfContainer() && range.StartRef().GetChild() && + HTMLEditUtils::IsInvisibleBRElement( + *range.StartRef().GetChild()) + ? HTMLEditUtils::GetNextContent( + *atStart.ContainerAs<nsIContent>(), + {WalkTreeOption::IgnoreDataNodeExceptText, + WalkTreeOption::StopAtBlockBoundary}, + editingHost) + : nullptr; + if (!nextContent || nextContent != range.StartRef().GetChild()) { + noNeedToChangeStart = true; + range.SetStart( + aRangesToDelete.GetFirstRangeStartPoint<EditorDOMPoint>()); + } + } + if (range.EndRef().IsBefore( + aRangesToDelete.GetFirstRangeEndPoint<EditorRawDOMPoint>())) { + if (noNeedToChangeStart) { + return NS_OK; // We don't need to modify the range. + } + range.SetEnd(aRangesToDelete.GetFirstRangeEndPoint<EditorDOMPoint>()); + } + } + // XXX Oddly, we join blocks only at the first range. + nsresult rv = aRangesToDelete.FirstRangeRef()->SetStartAndEnd( + range.StartRef().ToRawRangeBoundary(), + range.EndRef().ToRawRangeBoundary()); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoRangeArray::SetStartAndEnd() failed"); + return rv; +} + +Result<EditActionResult, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoBlockElementsJoiner::AutoInclusiveAncestorBlockElementsJoiner::Run( + HTMLEditor& aHTMLEditor, const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(mLeftBlockElement); + MOZ_ASSERT(mRightBlockElement); + + if (IsSameBlockElement()) { + return EditActionResult::IgnoredResult(); + } + + if (!mCanJoinBlocks) { + return EditActionResult::HandledResult(); + } + + EditorDOMPoint startOfRightContent; + + // If the left block element is in the right block element, move the hard + // line including the right block element to end of the left block. + // However, if we are merging list elements, we don't join them. + Result<EditActionResult, nsresult> result(NS_ERROR_NOT_INITIALIZED); + if (mPointContainingTheOtherBlockElement.GetContainer() == + mRightBlockElement) { + startOfRightContent = mPointContainingTheOtherBlockElement.NextPoint(); + if (Element* element = startOfRightContent.GetChildAs<Element>()) { + startOfRightContent = + HTMLEditUtils::GetDeepestEditableStartPointOf<EditorDOMPoint>( + *element); + } + AutoTrackDOMPoint trackStartOfRightBlock(aHTMLEditor.RangeUpdaterRef(), + &startOfRightContent); + result = WhiteSpaceVisibilityKeeper:: + MergeFirstLineOfRightBlockElementIntoDescendantLeftBlockElement( + aHTMLEditor, MOZ_KnownLive(*mLeftBlockElement), + MOZ_KnownLive(*mRightBlockElement), + mPointContainingTheOtherBlockElement, + mNewListElementTagNameOfRightListElement, + MOZ_KnownLive(mPrecedingInvisibleBRElement), aEditingHost); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING( + "WhiteSpaceVisibilityKeeper::" + "MergeFirstLineOfRightBlockElementIntoDescendantLeftBlockElement() " + "failed"); + return result; + } + if (NS_WARN_IF(!startOfRightContent.IsSet()) || + NS_WARN_IF(!startOfRightContent.GetContainer()->IsInComposedDoc())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + } + + // If the right block element is in the left block element: + // - move list item elements in the right block element to where the left + // list element is + // - or first hard line in the right block element to where: + // - the left block element is. + // - or the given left content in the left block is. + else if (mPointContainingTheOtherBlockElement.GetContainer() == + mLeftBlockElement) { + startOfRightContent = + HTMLEditUtils::GetDeepestEditableStartPointOf<EditorDOMPoint>( + *mRightBlockElement); + AutoTrackDOMPoint trackStartOfRightBlock(aHTMLEditor.RangeUpdaterRef(), + &startOfRightContent); + result = WhiteSpaceVisibilityKeeper:: + MergeFirstLineOfRightBlockElementIntoAncestorLeftBlockElement( + aHTMLEditor, MOZ_KnownLive(*mLeftBlockElement), + MOZ_KnownLive(*mRightBlockElement), + mPointContainingTheOtherBlockElement, + MOZ_KnownLive(*mInclusiveDescendantOfLeftBlockElement), + mNewListElementTagNameOfRightListElement, + MOZ_KnownLive(mPrecedingInvisibleBRElement), aEditingHost); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING( + "WhiteSpaceVisibilityKeeper::" + "MergeFirstLineOfRightBlockElementIntoAncestorLeftBlockElement() " + "failed"); + return result; + } + trackStartOfRightBlock.FlushAndStopTracking(); + if (NS_WARN_IF(!startOfRightContent.IsSet()) || + NS_WARN_IF(!startOfRightContent.GetContainer()->IsInComposedDoc())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + } + + // Normal case. Blocks are siblings, or at least close enough. An example + // of the latter is <p>paragraph</p><ul><li>one<li>two<li>three</ul>. The + // first li and the p are not true siblings, but we still want to join them + // if you backspace from li into p. + else { + MOZ_ASSERT(!mPointContainingTheOtherBlockElement.IsSet()); + + startOfRightContent = + HTMLEditUtils::GetDeepestEditableStartPointOf<EditorDOMPoint>( + *mRightBlockElement); + AutoTrackDOMPoint trackStartOfRightBlock(aHTMLEditor.RangeUpdaterRef(), + &startOfRightContent); + result = WhiteSpaceVisibilityKeeper:: + MergeFirstLineOfRightBlockElementIntoLeftBlockElement( + aHTMLEditor, MOZ_KnownLive(*mLeftBlockElement), + MOZ_KnownLive(*mRightBlockElement), + mNewListElementTagNameOfRightListElement, + MOZ_KnownLive(mPrecedingInvisibleBRElement), aEditingHost); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING( + "WhiteSpaceVisibilityKeeper::" + "MergeFirstLineOfRightBlockElementIntoLeftBlockElement() failed"); + return result; + } + trackStartOfRightBlock.FlushAndStopTracking(); + if (NS_WARN_IF(!startOfRightContent.IsSet()) || + NS_WARN_IF(!startOfRightContent.GetContainer()->IsInComposedDoc())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + } + + // If we're deleting selection (meaning not replacing selection with new + // content), we should put caret to end of preceding text node if there is. + // Then, users can type text into it like the other browsers. + if (MayEditActionDeleteAroundCollapsedSelection( + aHTMLEditor.GetEditAction())) { + WSRunScanner scanner(&aEditingHost, startOfRightContent); + WSScanResult maybePreviousText = + scanner.ScanPreviousVisibleNodeOrBlockBoundaryFrom(startOfRightContent); + if (maybePreviousText.IsContentEditable() && + maybePreviousText.InVisibleOrCollapsibleCharacters()) { + mPointToPutCaret = maybePreviousText.Point<EditorDOMPoint>(); + } + } + return result; +} + +// static +Result<bool, nsresult> +HTMLEditor::AutoMoveOneLineHandler::CanMoveOrDeleteSomethingInLine( + const EditorDOMPoint& aPointInHardLine, const Element& aEditingHost) { + if (NS_WARN_IF(!aPointInHardLine.IsSet()) || + NS_WARN_IF(aPointInHardLine.IsInNativeAnonymousSubtree())) { + return Err(NS_ERROR_INVALID_ARG); + } + + RefPtr<nsRange> oneLineRange = + AutoRangeArray::CreateRangeWrappingStartAndEndLinesContainingBoundaries( + aPointInHardLine, aPointInHardLine, + EditSubAction::eMergeBlockContents, aEditingHost); + if (!oneLineRange || oneLineRange->Collapsed() || + !oneLineRange->IsPositioned() || + !oneLineRange->GetStartContainer()->IsContent() || + !oneLineRange->GetEndContainer()->IsContent()) { + return false; + } + + // If there is only a padding `<br>` element in a empty block, it's selected + // by `UpdatePointsToSelectAllChildrenIfCollapsedInEmptyBlockElement()`. + // However, it won't be moved. Although it'll be deleted, + // AutoMoveOneLineHandler returns "ignored". Therefore, we should return + // `false` in this case. + if (nsIContent* childContent = oneLineRange->GetChildAtStartOffset()) { + if (childContent->IsHTMLElement(nsGkAtoms::br) && + childContent->GetParent()) { + if (const Element* blockElement = + HTMLEditUtils::GetInclusiveAncestorElement( + *childContent->GetParent(), + HTMLEditUtils::ClosestBlockElement)) { + if (HTMLEditUtils::IsEmptyNode(*blockElement)) { + return false; + } + } + } + } + + nsINode* commonAncestor = oneLineRange->GetClosestCommonInclusiveAncestor(); + // Currently, we move non-editable content nodes too. + EditorRawDOMPoint startPoint(oneLineRange->StartRef()); + if (!startPoint.IsEndOfContainer()) { + return true; + } + EditorRawDOMPoint endPoint(oneLineRange->EndRef()); + if (!endPoint.IsStartOfContainer()) { + return true; + } + if (startPoint.GetContainer() != commonAncestor) { + while (true) { + EditorRawDOMPoint pointInParent(startPoint.GetContainerAs<nsIContent>()); + if (NS_WARN_IF(!pointInParent.IsInContentNode())) { + return Err(NS_ERROR_FAILURE); + } + if (pointInParent.GetContainer() == commonAncestor) { + startPoint = pointInParent; + break; + } + if (!pointInParent.IsEndOfContainer()) { + return true; + } + } + } + if (endPoint.GetContainer() != commonAncestor) { + while (true) { + EditorRawDOMPoint pointInParent(endPoint.GetContainerAs<nsIContent>()); + if (NS_WARN_IF(!pointInParent.IsInContentNode())) { + return Err(NS_ERROR_FAILURE); + } + if (pointInParent.GetContainer() == commonAncestor) { + endPoint = pointInParent; + break; + } + if (!pointInParent.IsStartOfContainer()) { + return true; + } + } + } + // If start point and end point in the common ancestor are direct siblings, + // there is no content to move or delete. + // E.g., `<b>abc<br>[</b><i>]<br>def</i>`. + return startPoint.GetNextSiblingOfChild() != endPoint.GetChild(); +} + +nsresult HTMLEditor::AutoMoveOneLineHandler::Prepare( + HTMLEditor& aHTMLEditor, const EditorDOMPoint& aPointInHardLine, + const Element& aEditingHost) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(aPointInHardLine.IsInContentNode()); + MOZ_ASSERT(mPointToInsert.IsSetAndValid()); + + if (NS_WARN_IF(mPointToInsert.IsInNativeAnonymousSubtree())) { + return Err(NS_ERROR_INVALID_ARG); + } + + mSrcInclusiveAncestorBlock = + aPointInHardLine.IsInContentNode() + ? HTMLEditUtils::GetInclusiveAncestorElement( + *aPointInHardLine.ContainerAs<nsIContent>(), + HTMLEditUtils::ClosestBlockElement) + : nullptr; + mDestInclusiveAncestorBlock = + mPointToInsert.IsInContentNode() + ? HTMLEditUtils::GetInclusiveAncestorElement( + *mPointToInsert.ContainerAs<nsIContent>(), + HTMLEditUtils::ClosestBlockElement) + : nullptr; + mMovingToParentBlock = + mDestInclusiveAncestorBlock && mSrcInclusiveAncestorBlock && + mDestInclusiveAncestorBlock != mSrcInclusiveAncestorBlock && + mSrcInclusiveAncestorBlock->IsInclusiveDescendantOf( + mDestInclusiveAncestorBlock); + mTopmostSrcAncestorBlockInDestBlock = + mMovingToParentBlock + ? AutoMoveOneLineHandler:: + GetMostDistantInclusiveAncestorBlockInSpecificAncestorElement( + *mSrcInclusiveAncestorBlock, *mDestInclusiveAncestorBlock) + : nullptr; + MOZ_ASSERT_IF(mMovingToParentBlock, mTopmostSrcAncestorBlockInDestBlock); + + mPreserveWhiteSpaceStyle = + AutoMoveOneLineHandler::ConsiderWhetherPreserveWhiteSpaceStyle( + aPointInHardLine.GetContainerAs<nsIContent>(), + mDestInclusiveAncestorBlock); + + AutoRangeArray rangesToWrapTheLine(aPointInHardLine); + rangesToWrapTheLine.ExtendRangesToWrapLinesToHandleBlockLevelEditAction( + EditSubAction::eMergeBlockContents, aEditingHost); + MOZ_ASSERT(rangesToWrapTheLine.Ranges().Length() <= 1u); + mLineRange = EditorDOMRange(rangesToWrapTheLine.FirstRangeRef()); + return NS_OK; +} + +Result<CaretPoint, nsresult> +HTMLEditor::AutoMoveOneLineHandler::SplitToMakeTheLineIsolated( + HTMLEditor& aHTMLEditor, const nsIContent& aNewContainer, + const Element& aEditingHost, + nsTArray<OwningNonNull<nsIContent>>& aOutArrayOfContents) const { + AutoRangeArray rangesToWrapTheLine(mLineRange); + Result<EditorDOMPoint, nsresult> splitResult = + rangesToWrapTheLine + .SplitTextAtEndBoundariesAndInlineAncestorsAtBothBoundaries( + aHTMLEditor, aEditingHost, &aNewContainer); + if (MOZ_UNLIKELY(splitResult.isErr())) { + NS_WARNING( + "AutoRangeArray::" + "SplitTextAtEndBoundariesAndInlineAncestorsAtBothBoundaries() failed"); + return Err(splitResult.unwrapErr()); + } + EditorDOMPoint pointToPutCaret; + if (splitResult.inspect().IsSet()) { + pointToPutCaret = splitResult.unwrap(); + } + nsresult rv = rangesToWrapTheLine.CollectEditTargetNodes( + aHTMLEditor, aOutArrayOfContents, EditSubAction::eMergeBlockContents, + AutoRangeArray::CollectNonEditableNodes::Yes); + if (NS_FAILED(rv)) { + NS_WARNING( + "AutoRangeArray::CollectEditTargetNodes(EditSubAction::" + "eMergeBlockContents, CollectNonEditableNodes::Yes) failed"); + return Err(rv); + } + return CaretPoint(pointToPutCaret); +} + +// static +Element* HTMLEditor::AutoMoveOneLineHandler:: + GetMostDistantInclusiveAncestorBlockInSpecificAncestorElement( + Element& aBlockElement, const Element& aAncestorElement) { + MOZ_ASSERT(aBlockElement.IsInclusiveDescendantOf(&aAncestorElement)); + MOZ_ASSERT(HTMLEditUtils::IsBlockElement(aBlockElement)); + + if (&aBlockElement == &aAncestorElement) { + return nullptr; + } + + Element* lastBlockAncestor = &aBlockElement; + for (Element* element : aBlockElement.InclusiveAncestorsOfType<Element>()) { + if (element == &aAncestorElement) { + return lastBlockAncestor; + } + if (HTMLEditUtils::IsBlockElement(*lastBlockAncestor)) { + lastBlockAncestor = element; + } + } + return nullptr; +} + +// static +HTMLEditor::PreserveWhiteSpaceStyle +HTMLEditor::AutoMoveOneLineHandler::ConsiderWhetherPreserveWhiteSpaceStyle( + const nsIContent* aContentInLine, + const Element* aInclusiveAncestorBlockOfInsertionPoint) { + if (MOZ_UNLIKELY(!aInclusiveAncestorBlockOfInsertionPoint)) { + return PreserveWhiteSpaceStyle::No; + } + + // If we move content from or to <pre>, we don't need to preserve the + // white-space style for compatibility with both our traditional behavior + // and the other browsers. + + // TODO: If `white-space` is specified by non-UA stylesheet, we should + // preserve it even if the right block is <pre> for compatibility with the + // other browsers. + const auto IsInclusiveDescendantOfPre = [](const nsIContent& aContent) { + // If the content has different `white-space` style from <pre>, we + // shouldn't treat it as a descendant of <pre> because web apps or + // the user intent to treat the white-spaces in aContent not as `pre`. + if (EditorUtils::GetComputedWhiteSpaceStyle(aContent).valueOr( + StyleWhiteSpace::Normal) != StyleWhiteSpace::Pre) { + return false; + } + for (const Element* element : + aContent.InclusiveAncestorsOfType<Element>()) { + if (element->IsHTMLElement(nsGkAtoms::pre)) { + return true; + } + } + return false; + }; + if (IsInclusiveDescendantOfPre(*aInclusiveAncestorBlockOfInsertionPoint) || + MOZ_UNLIKELY(!aContentInLine) || + IsInclusiveDescendantOfPre(*aContentInLine)) { + return PreserveWhiteSpaceStyle::No; + } + return PreserveWhiteSpaceStyle::Yes; +} + +Result<MoveNodeResult, nsresult> HTMLEditor::AutoMoveOneLineHandler::Run( + HTMLEditor& aHTMLEditor, const Element& aEditingHost) { + EditorDOMPoint pointToInsert(NextInsertionPointRef()); + MOZ_ASSERT(pointToInsert.IsInContentNode()); + + EditorDOMPoint pointToPutCaret; + AutoTArray<OwningNonNull<nsIContent>, 64> arrayOfContents; + { + AutoTrackDOMPoint tackPointToInsert(aHTMLEditor.RangeUpdaterRef(), + &pointToInsert); + + Result<CaretPoint, nsresult> splitAtLineEdgesResult = + SplitToMakeTheLineIsolated( + aHTMLEditor, + MOZ_KnownLive(*pointToInsert.ContainerAs<nsIContent>()), + aEditingHost, arrayOfContents); + if (MOZ_UNLIKELY(splitAtLineEdgesResult.isErr())) { + NS_WARNING("AutoMoveOneLineHandler::SplitToMakeTheLineIsolated() failed"); + return splitAtLineEdgesResult.propagateErr(); + } + splitAtLineEdgesResult.unwrap().MoveCaretPointTo( + pointToPutCaret, {SuggestCaret::OnlyIfHasSuggestion}); + + Result<EditorDOMPoint, nsresult> splitAtBRElementsResult = + aHTMLEditor.MaybeSplitElementsAtEveryBRElement( + arrayOfContents, EditSubAction::eMergeBlockContents); + if (MOZ_UNLIKELY(splitAtBRElementsResult.isErr())) { + NS_WARNING( + "HTMLEditor::MaybeSplitElementsAtEveryBRElement(EditSubAction::" + "eMergeBlockContents) failed"); + return splitAtBRElementsResult.propagateErr(); + } + if (splitAtBRElementsResult.inspect().IsSet()) { + pointToPutCaret = splitAtBRElementsResult.unwrap(); + } + } + + if (!pointToInsert.IsSetAndValid()) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + + if (aHTMLEditor.AllowsTransactionsToChangeSelection() && + pointToPutCaret.IsSet()) { + nsresult rv = aHTMLEditor.CollapseSelectionTo(pointToPutCaret); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return Err(rv); + } + } + + if (arrayOfContents.IsEmpty()) { + return MoveNodeResult::IgnoredResult(std::move(pointToInsert)); + } + + // Track the range which contains the moved contents. + if (ForceMoveToEndOfContainer()) { + pointToInsert = NextInsertionPointRef(); + } + EditorDOMRange movedContentRange(pointToInsert); + MoveNodeResult moveContentsInLineResult = + MoveNodeResult::IgnoredResult(pointToInsert); + for (const OwningNonNull<nsIContent>& content : arrayOfContents) { + { + AutoEditorDOMRangeChildrenInvalidator lockOffsets(movedContentRange); + // If the content is a block element, move all children of it to the + // new container, and then, remove the (probably) empty block element. + if (HTMLEditUtils::IsBlockElement(content)) { + Result<MoveNodeResult, nsresult> moveChildrenResult = + aHTMLEditor.MoveChildrenWithTransaction( + MOZ_KnownLive(*content->AsElement()), pointToInsert, + mPreserveWhiteSpaceStyle, RemoveIfCommentNode::Yes); + if (MOZ_UNLIKELY(moveChildrenResult.isErr())) { + NS_WARNING("HTMLEditor::MoveChildrenWithTransaction() failed"); + moveContentsInLineResult.IgnoreCaretPointSuggestion(); + return moveChildrenResult; + } + moveContentsInLineResult |= moveChildrenResult.inspect(); + moveContentsInLineResult.MarkAsHandled(); + // MOZ_KnownLive due to bug 1620312 + nsresult rv = + aHTMLEditor.DeleteNodeWithTransaction(MOZ_KnownLive(content)); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + moveContentsInLineResult.IgnoreCaretPointSuggestion(); + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::DeleteNodeWithTransaction() failed, but ignored"); + } + // If the moving content is a comment node or an empty inline node, we + // don't want it to appear in the dist paragraph. + else if (content->IsComment() || + HTMLEditUtils::IsEmptyInlineContainer( + content, {EmptyCheckOption::TreatSingleBRElementAsVisible, + EmptyCheckOption::TreatListItemAsVisible, + EmptyCheckOption::TreatTableCellAsVisible})) { + nsCOMPtr<nsIContent> emptyContent = + HTMLEditUtils::GetMostDistantAncestorEditableEmptyInlineElement( + content, &aEditingHost, + pointToInsert.ContainerAs<nsIContent>()); + if (!emptyContent) { + emptyContent = content; + } + nsresult rv = aHTMLEditor.DeleteNodeWithTransaction(*emptyContent); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + moveContentsInLineResult.IgnoreCaretPointSuggestion(); + return Err(rv); + } + } else { + // MOZ_KnownLive due to bug 1620312 + Result<MoveNodeResult, nsresult> moveNodeOrChildrenResult = + aHTMLEditor.MoveNodeOrChildrenWithTransaction( + MOZ_KnownLive(content), pointToInsert, mPreserveWhiteSpaceStyle, + RemoveIfCommentNode::Yes); + if (MOZ_UNLIKELY(moveNodeOrChildrenResult.isErr())) { + NS_WARNING("HTMLEditor::MoveNodeOrChildrenWithTransaction() failed"); + moveContentsInLineResult.IgnoreCaretPointSuggestion(); + return moveNodeOrChildrenResult; + } + moveContentsInLineResult |= moveNodeOrChildrenResult.inspect(); + } + } + // For backward compatibility, we should move contents to end of the + // container if the instance is created without specific insertion point. + if (ForceMoveToEndOfContainer()) { + pointToInsert = NextInsertionPointRef(); + movedContentRange.SetEnd(pointToInsert); + } + // And also if pointToInsert has been made invalid with removing preceding + // children, we should move the content to the end of the container. + else if (aHTMLEditor.MayHaveMutationEventListeners() && + MOZ_UNLIKELY(!moveContentsInLineResult.NextInsertionPointRef() + .IsSetAndValid())) { + mPointToInsert.SetToEndOf(mPointToInsert.GetContainer()); + pointToInsert = NextInsertionPointRef(); + movedContentRange.SetEnd(pointToInsert); + } else { + MOZ_DIAGNOSTIC_ASSERT( + moveContentsInLineResult.NextInsertionPointRef().IsSet()); + mPointToInsert = moveContentsInLineResult.NextInsertionPointRef(); + pointToInsert = NextInsertionPointRef(); + if (!aHTMLEditor.MayHaveMutationEventListeners() || + movedContentRange.EndRef().IsBefore(pointToInsert)) { + movedContentRange.SetEnd(pointToInsert); + } + } + } + + // Nothing has been moved, we don't need to clean up unnecessary <br> element. + // And also if we're not moving content into a block, we can quit right now. + if (moveContentsInLineResult.Ignored() || + MOZ_UNLIKELY(!mDestInclusiveAncestorBlock)) { + return moveContentsInLineResult; + } + + // If we couldn't track the range to clean up, we should just stop cleaning up + // because returning error from here may change the behavior of web apps using + // mutation event listeners. + if (MOZ_UNLIKELY(!movedContentRange.IsPositioned() || + movedContentRange.Collapsed())) { + return moveContentsInLineResult; + } + + nsresult rv = DeleteUnnecessaryTrailingLineBreakInMovedLineEnd( + aHTMLEditor, movedContentRange, aEditingHost); + if (NS_FAILED(rv)) { + NS_WARNING( + "AutoMoveOneLineHandler::" + "DeleteUnnecessaryTrailingLineBreakInMovedLineEnd() failed"); + moveContentsInLineResult.IgnoreCaretPointSuggestion(); + return Err(rv); + } + return moveContentsInLineResult; +} + +nsresult HTMLEditor::AutoMoveOneLineHandler:: + DeleteUnnecessaryTrailingLineBreakInMovedLineEnd( + HTMLEditor& aHTMLEditor, const EditorDOMRange& aMovedContentRange, + const Element& aEditingHost) const { + MOZ_ASSERT(mDestInclusiveAncestorBlock); + MOZ_ASSERT(aMovedContentRange.IsPositioned()); + MOZ_ASSERT(!aMovedContentRange.Collapsed()); + + // If we didn't preserve white-space for backward compatibility and + // white-space becomes not preformatted, we need to clean it up the last text + // node if it ends with a preformatted line break. + if (mPreserveWhiteSpaceStyle == PreserveWhiteSpaceStyle::No) { + const RefPtr<Text> textNodeEndingWithUnnecessaryLineBreak = [&]() -> Text* { + Text* lastTextNode = Text::FromNodeOrNull( + mMovingToParentBlock + ? HTMLEditUtils::GetPreviousContent( + *mTopmostSrcAncestorBlockInDestBlock, + {WalkTreeOption::StopAtBlockBoundary}, + mDestInclusiveAncestorBlock) + : HTMLEditUtils::GetLastLeafContent( + *mDestInclusiveAncestorBlock, + {LeafNodeType::LeafNodeOrNonEditableNode})); + if (!lastTextNode || + !HTMLEditUtils::IsSimplyEditableNode(*lastTextNode)) { + return nullptr; + } + const nsTextFragment& textFragment = lastTextNode->TextFragment(); + const char16_t lastCh = + textFragment.GetLength() + ? textFragment.CharAt(textFragment.GetLength() - 1u) + : 0; + return lastCh == HTMLEditUtils::kNewLine && + !EditorUtils::IsNewLinePreformatted(*lastTextNode) + ? lastTextNode + : nullptr; + }(); + if (textNodeEndingWithUnnecessaryLineBreak) { + if (textNodeEndingWithUnnecessaryLineBreak->TextDataLength() == 1u) { + const RefPtr<Element> inlineElement = + HTMLEditUtils::GetMostDistantAncestorEditableEmptyInlineElement( + *textNodeEndingWithUnnecessaryLineBreak, &aEditingHost); + nsresult rv = aHTMLEditor.DeleteNodeWithTransaction( + inlineElement ? static_cast<nsIContent&>(*inlineElement) + : static_cast<nsIContent&>( + *textNodeEndingWithUnnecessaryLineBreak)); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return Err(rv); + } + } else { + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteTextWithTransaction( + *textNodeEndingWithUnnecessaryLineBreak, + textNodeEndingWithUnnecessaryLineBreak->TextDataLength() - 1u, + 1u); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("HTMLEditor::DeleteTextWithTransaction() failed"); + return caretPointOrError.propagateErr(); + } + nsresult rv = caretPointOrError.inspect().SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + NS_WARNING_ASSERTION( + rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR, + "CaretPoint::SuggestCaretPointTo() failed, but ignored"); + } + } + } + + nsCOMPtr<nsIContent> lastLineBreakContent = + mMovingToParentBlock + ? HTMLEditUtils::GetUnnecessaryLineBreakContent( + *mTopmostSrcAncestorBlockInDestBlock, + ScanLineBreak::BeforeBlock) + : HTMLEditUtils::GetUnnecessaryLineBreakContent( + *mDestInclusiveAncestorBlock, ScanLineBreak::AtEndOfBlock); + if (!lastLineBreakContent) { + return NS_OK; + } + EditorRawDOMPoint atUnnecessaryLineBreak(lastLineBreakContent); + if (NS_WARN_IF(!atUnnecessaryLineBreak.IsSet())) { + return NS_ERROR_FAILURE; + } + // If the found unnecessary line break is not what we moved above, we + // shouldn't remove it. E.g., the web app may have inserted it intentionally. + MOZ_ASSERT(aMovedContentRange.StartRef().IsSetAndValid()); + MOZ_ASSERT(aMovedContentRange.EndRef().IsSetAndValid()); + if (!aMovedContentRange.Contains(atUnnecessaryLineBreak)) { + return NS_OK; + } + + AutoTransactionsConserveSelection dontChangeMySelection(aHTMLEditor); + // If it's a text node and ending with a preformatted line break, we should + // delete it. + if (Text* textNode = Text::FromNode(lastLineBreakContent)) { + MOZ_ASSERT(EditorUtils::IsNewLinePreformatted(*textNode)); + if (textNode->TextDataLength() > 1) { + Result<CaretPoint, nsresult> caretPointOrError = + aHTMLEditor.DeleteTextWithTransaction( + MOZ_KnownLive(*textNode), textNode->TextDataLength() - 1u, 1u); + if (MOZ_UNLIKELY(caretPointOrError.isErr())) { + NS_WARNING("HTMLEditor::DeleteTextWithTransaction() failed"); + return caretPointOrError.unwrapErr(); + } + // IgnoreCaretPointSuggestion() because of dontChangeMySelection above. + caretPointOrError.unwrap().IgnoreCaretPointSuggestion(); + return NS_OK; + } + } else { + MOZ_ASSERT(lastLineBreakContent->IsHTMLElement(nsGkAtoms::br)); + } + // If last line break content is the only content of its inline parent, we + // should remove the parent too. + if (const RefPtr<Element> inlineElement = + HTMLEditUtils::GetMostDistantAncestorEditableEmptyInlineElement( + *lastLineBreakContent, &aEditingHost)) { + nsresult rv = aHTMLEditor.DeleteNodeWithTransaction(*inlineElement); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::DeleteNodeWithTransaction() failed"); + return rv; + } + // Or if the text node has only the preformatted line break or <br> element, + // we should remove it. + nsresult rv = aHTMLEditor.DeleteNodeWithTransaction(*lastLineBreakContent); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::DeleteNodeWithTransaction() failed"); + return rv; +} + +Result<bool, nsresult> HTMLEditor::CanMoveNodeOrChildren( + const nsIContent& aContent, const nsINode& aNewContainer) const { + if (HTMLEditUtils::CanNodeContain(aNewContainer, aContent)) { + return true; + } + if (aContent.IsElement()) { + return CanMoveChildren(*aContent.AsElement(), aNewContainer); + } + return true; +} + +Result<MoveNodeResult, nsresult> HTMLEditor::MoveNodeOrChildrenWithTransaction( + nsIContent& aContentToMove, const EditorDOMPoint& aPointToInsert, + PreserveWhiteSpaceStyle aPreserveWhiteSpaceStyle, + RemoveIfCommentNode aRemoveIfCommentNode) { + MOZ_ASSERT(IsEditActionDataAvailable()); + MOZ_ASSERT(aPointToInsert.IsInContentNode()); + + const auto destWhiteSpaceStyle = [&]() -> Maybe<StyleWhiteSpace> { + if (aPreserveWhiteSpaceStyle == PreserveWhiteSpaceStyle::No || + !aPointToInsert.IsInContentNode()) { + return Nothing(); + } + auto style = EditorUtils::GetComputedWhiteSpaceStyle( + *aPointToInsert.ContainerAs<nsIContent>()); + if (NS_WARN_IF(style.isSome() && + style.value() == StyleWhiteSpace::PreSpace)) { + return Nothing(); + } + return style; + }(); + const auto srcWhiteSpaceStyle = [&]() -> Maybe<StyleWhiteSpace> { + if (aPreserveWhiteSpaceStyle == PreserveWhiteSpaceStyle::No) { + return Nothing(); + } + auto style = EditorUtils::GetComputedWhiteSpaceStyle(aContentToMove); + if (NS_WARN_IF(style.isSome() && + style.value() == StyleWhiteSpace::PreSpace)) { + return Nothing(); + } + return style; + }(); + const auto GetWhiteSpaceStyleValue = [](StyleWhiteSpace aStyleWhiteSpace) { + switch (aStyleWhiteSpace) { + case StyleWhiteSpace::Normal: + return u"normal"_ns; + case StyleWhiteSpace::Pre: + return u"pre"_ns; + case StyleWhiteSpace::Nowrap: + return u"nowrap"_ns; + case StyleWhiteSpace::PreWrap: + return u"pre-wrap"_ns; + case StyleWhiteSpace::PreLine: + return u"pre-line"_ns; + case StyleWhiteSpace::BreakSpaces: + return u"break-spaces"_ns; + case StyleWhiteSpace::PreSpace: + MOZ_ASSERT_UNREACHABLE("Don't handle -moz-pre-space"); + return u""_ns; + default: + MOZ_ASSERT_UNREACHABLE("Handle the new white-space value"); + return u""_ns; + } + }; + + if (aRemoveIfCommentNode == RemoveIfCommentNode::Yes && + aContentToMove.IsComment()) { + EditorDOMPoint pointToInsert(aPointToInsert); + { + AutoTrackDOMPoint trackPointToInsert(RangeUpdaterRef(), &pointToInsert); + nsresult rv = DeleteNodeWithTransaction(aContentToMove); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return Err(rv); + } + } + if (NS_WARN_IF(!pointToInsert.IsSetAndValid())) { + return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + } + return MoveNodeResult::HandledResult(std::move(pointToInsert)); + } + + // Check if this node can go into the destination node + if (HTMLEditUtils::CanNodeContain(*aPointToInsert.GetContainer(), + aContentToMove)) { + EditorDOMPoint pointToInsert(aPointToInsert); + // Preserve white-space in the new position with using `style` attribute. + // This is additional path from point of view of our traditional behavior. + // Therefore, ignore errors especially if we got unexpected DOM tree. + if (destWhiteSpaceStyle.isSome() && srcWhiteSpaceStyle.isSome() && + destWhiteSpaceStyle.value() != srcWhiteSpaceStyle.value()) { + // Set `white-space` with `style` attribute if it's nsStyledElement. + if (nsStyledElement* styledElement = + nsStyledElement::FromNode(&aContentToMove)) { + DebugOnly<nsresult> rvIgnored = + CSSEditUtils::SetCSSPropertyWithTransaction( + *this, MOZ_KnownLive(*styledElement), *nsGkAtoms::white_space, + GetWhiteSpaceStyleValue(srcWhiteSpaceStyle.value())); + if (NS_WARN_IF(Destroyed())) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored), + "CSSEditUtils::SetCSSPropertyWithTransaction(" + "nsGkAtoms::white_space) failed, but ignored"); + } + // Otherwise, if the dest container can have <span> element and <span> + // element can have the moving content node, we should insert it. + else if (HTMLEditUtils::CanNodeContain(*aPointToInsert.GetContainer(), + *nsGkAtoms::span) && + HTMLEditUtils::CanNodeContain(*nsGkAtoms::span, + aContentToMove)) { + RefPtr<Element> newSpanElement = CreateHTMLContent(nsGkAtoms::span); + if (NS_WARN_IF(!newSpanElement)) { + return Err(NS_ERROR_FAILURE); + } + nsAutoString styleAttrValue(u"white-space: "_ns); + styleAttrValue.Append( + GetWhiteSpaceStyleValue(srcWhiteSpaceStyle.value())); + IgnoredErrorResult error; + newSpanElement->SetAttr(nsGkAtoms::style, styleAttrValue, error); + NS_WARNING_ASSERTION(!error.Failed(), + "Element::SetAttr(nsGkAtoms::span) failed"); + if (MOZ_LIKELY(!error.Failed())) { + Result<CreateElementResult, nsresult> insertSpanElementResult = + InsertNodeWithTransaction<Element>(*newSpanElement, + aPointToInsert); + if (MOZ_UNLIKELY(insertSpanElementResult.isErr())) { + if (NS_WARN_IF(insertSpanElementResult.inspectErr() == + NS_ERROR_EDITOR_DESTROYED)) { + return Err(NS_ERROR_EDITOR_DESTROYED); + } + NS_WARNING( + "HTMLEditor::InsertNodeWithTransaction() failed, but ignored"); + } else { + // We should move the node into the new <span> to preserve the + // style. + pointToInsert.Set(newSpanElement, 0u); + // We should put caret after aContentToMove after moving it so that + // we do not need the suggested caret point here. + insertSpanElementResult.inspect().IgnoreCaretPointSuggestion(); + } + } + } + } + // If it can, move it there. + Result<MoveNodeResult, nsresult> moveNodeResult = + MoveNodeWithTransaction(aContentToMove, pointToInsert); + NS_WARNING_ASSERTION(moveNodeResult.isOk(), + "HTMLEditor::MoveNodeWithTransaction() failed"); + // XXX This is odd to override the handled state here, but stopping this + // hits an NS_ASSERTION in WhiteSpaceVisibilityKeeper:: + // MergeFirstLineOfRightBlockElementIntoAncestorLeftBlockElement. + if (moveNodeResult.isOk()) { + MoveNodeResult unwrappedMoveNodeResult = moveNodeResult.unwrap(); + unwrappedMoveNodeResult.MarkAsHandled(); + return unwrappedMoveNodeResult; + } + return moveNodeResult; + } + + // If it can't, move its children (if any), and then delete it. + auto moveNodeResult = + [&]() MOZ_CAN_RUN_SCRIPT -> Result<MoveNodeResult, nsresult> { + if (!aContentToMove.IsElement()) { + return MoveNodeResult::HandledResult(aPointToInsert); + } + Result<MoveNodeResult, nsresult> moveChildrenResult = + MoveChildrenWithTransaction(MOZ_KnownLive(*aContentToMove.AsElement()), + aPointToInsert, aPreserveWhiteSpaceStyle, + aRemoveIfCommentNode); + NS_WARNING_ASSERTION(moveChildrenResult.isOk(), + "HTMLEditor::MoveChildrenWithTransaction() failed"); + return moveChildrenResult; + }(); + if (MOZ_UNLIKELY(moveNodeResult.isErr())) { + return moveNodeResult; // Already warned in the lambda. + } + + nsresult rv = DeleteNodeWithTransaction(aContentToMove); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + moveNodeResult.inspect().IgnoreCaretPointSuggestion(); + return Err(rv); + } + if (!MayHaveMutationEventListeners()) { + return moveNodeResult; + } + // Mutation event listener may make `offset` value invalid with + // removing some previous children while we call + // `DeleteNodeWithTransaction()` so that we should adjust it here. + if (moveNodeResult.inspect().NextInsertionPointRef().IsSetAndValid()) { + return moveNodeResult; + } + moveNodeResult.inspect().IgnoreCaretPointSuggestion(); + return MoveNodeResult::HandledResult( + EditorDOMPoint::AtEndOf(*aPointToInsert.GetContainer())); +} + +Result<bool, nsresult> HTMLEditor::CanMoveChildren( + const Element& aElement, const nsINode& aNewContainer) const { + if (NS_WARN_IF(&aElement == &aNewContainer)) { + return Err(NS_ERROR_FAILURE); + } + for (nsIContent* childContent = aElement.GetFirstChild(); childContent; + childContent = childContent->GetNextSibling()) { + Result<bool, nsresult> result = + CanMoveNodeOrChildren(*childContent, aNewContainer); + if (result.isErr() || result.inspect()) { + return result; + } + } + return false; +} + +Result<MoveNodeResult, nsresult> HTMLEditor::MoveChildrenWithTransaction( + Element& aElement, const EditorDOMPoint& aPointToInsert, + PreserveWhiteSpaceStyle aPreserveWhiteSpaceStyle, + RemoveIfCommentNode aRemoveIfCommentNode) { + MOZ_ASSERT(aPointToInsert.IsSet()); + + if (NS_WARN_IF(&aElement == aPointToInsert.GetContainer())) { + return Err(NS_ERROR_INVALID_ARG); + } + + MoveNodeResult moveChildrenResult = + MoveNodeResult::IgnoredResult(aPointToInsert); + while (aElement.GetFirstChild()) { + Result<MoveNodeResult, nsresult> moveNodeOrChildrenResult = + MoveNodeOrChildrenWithTransaction( + MOZ_KnownLive(*aElement.GetFirstChild()), + moveChildrenResult.NextInsertionPointRef(), + aPreserveWhiteSpaceStyle, aRemoveIfCommentNode); + if (MOZ_UNLIKELY(moveNodeOrChildrenResult.isErr())) { + NS_WARNING("HTMLEditor::MoveNodeOrChildrenWithTransaction() failed"); + moveChildrenResult.IgnoreCaretPointSuggestion(); + return moveNodeOrChildrenResult; + } + moveChildrenResult |= moveNodeOrChildrenResult.inspect(); + } + return moveChildrenResult; +} + +void HTMLEditor::MoveAllChildren(nsINode& aContainer, + const EditorRawDOMPoint& aPointToInsert, + ErrorResult& aError) { + MOZ_ASSERT(!aError.Failed()); + + if (!aContainer.HasChildren()) { + return; + } + nsIContent* firstChild = aContainer.GetFirstChild(); + if (NS_WARN_IF(!firstChild)) { + aError.Throw(NS_ERROR_FAILURE); + return; + } + nsIContent* lastChild = aContainer.GetLastChild(); + if (NS_WARN_IF(!lastChild)) { + aError.Throw(NS_ERROR_FAILURE); + return; + } + MoveChildrenBetween(*firstChild, *lastChild, aPointToInsert, aError); + NS_WARNING_ASSERTION(!aError.Failed(), + "HTMLEditor::MoveChildrenBetween() failed"); +} + +void HTMLEditor::MoveChildrenBetween(nsIContent& aFirstChild, + nsIContent& aLastChild, + const EditorRawDOMPoint& aPointToInsert, + ErrorResult& aError) { + nsCOMPtr<nsINode> oldContainer = aFirstChild.GetParentNode(); + if (NS_WARN_IF(oldContainer != aLastChild.GetParentNode()) || + NS_WARN_IF(!aPointToInsert.IsInContentNode()) || + NS_WARN_IF(!aPointToInsert.CanContainerHaveChildren())) { + aError.Throw(NS_ERROR_INVALID_ARG); + return; + } + + // First, store all children which should be moved to the new container. + AutoTArray<nsCOMPtr<nsIContent>, 10> children; + for (nsIContent* child = &aFirstChild; child; + child = child->GetNextSibling()) { + children.AppendElement(child); + if (child == &aLastChild) { + break; + } + } + + if (NS_WARN_IF(children.LastElement() != &aLastChild)) { + aError.Throw(NS_ERROR_INVALID_ARG); + return; + } + + nsCOMPtr<nsIContent> newContainer = aPointToInsert.ContainerAs<nsIContent>(); + nsCOMPtr<nsIContent> nextNode = aPointToInsert.GetChild(); + for (size_t i = children.Length(); i > 0; --i) { + nsCOMPtr<nsIContent>& child = children[i - 1]; + if (child->GetParentNode() != oldContainer) { + // If the child has been moved to different container, we shouldn't + // touch it. + continue; + } + if (NS_WARN_IF(!HTMLEditUtils::IsRemovableNode(*child))) { + aError.Throw(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + return; + } + oldContainer->RemoveChild(*child, aError); + if (NS_WARN_IF(Destroyed())) { + aError.Throw(NS_ERROR_EDITOR_DESTROYED); + return; + } + if (aError.Failed()) { + NS_WARNING("nsINode::RemoveChild() failed"); + return; + } + if (nextNode) { + // If we're not appending the children to the new container, we should + // check if referring next node of insertion point is still in the new + // container. + EditorRawDOMPoint pointToInsert(nextNode); + if (NS_WARN_IF(!pointToInsert.IsSet()) || + NS_WARN_IF(pointToInsert.GetContainer() != newContainer)) { + // The next node of insertion point has been moved by mutation observer. + // Let's stop moving the remaining nodes. + // XXX Or should we move remaining children after the last moved child? + aError.Throw(NS_ERROR_FAILURE); + return; + } + } + if (NS_WARN_IF( + !EditorUtils::IsEditableContent(*newContainer, EditorType::HTML))) { + aError.Throw(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE); + return; + } + newContainer->InsertBefore(*child, nextNode, aError); + if (NS_WARN_IF(Destroyed())) { + aError.Throw(NS_ERROR_EDITOR_DESTROYED); + return; + } + if (aError.Failed()) { + NS_WARNING("nsINode::InsertBefore() failed"); + return; + } + // If the child was inserted or appended properly, the following children + // should be inserted before it. Otherwise, keep using current position. + if (child->GetParentNode() == newContainer) { + nextNode = child; + } + } +} + +void HTMLEditor::MovePreviousSiblings(nsIContent& aChild, + const EditorRawDOMPoint& aPointToInsert, + ErrorResult& aError) { + MOZ_ASSERT(!aError.Failed()); + + if (NS_WARN_IF(!aChild.GetParentNode())) { + aError.Throw(NS_ERROR_INVALID_ARG); + return; + } + nsIContent* firstChild = aChild.GetParentNode()->GetFirstChild(); + if (NS_WARN_IF(!firstChild)) { + aError.Throw(NS_ERROR_FAILURE); + return; + } + nsIContent* lastChild = + &aChild == firstChild ? firstChild : aChild.GetPreviousSibling(); + if (NS_WARN_IF(!lastChild)) { + aError.Throw(NS_ERROR_FAILURE); + return; + } + MoveChildrenBetween(*firstChild, *lastChild, aPointToInsert, aError); + NS_WARNING_ASSERTION(!aError.Failed(), + "HTMLEditor::MoveChildrenBetween() failed"); +} + +void HTMLEditor::MoveInclusiveNextSiblings( + nsIContent& aChild, const EditorRawDOMPoint& aPointToInsert, + ErrorResult& aError) { + MOZ_ASSERT(!aError.Failed()); + + if (NS_WARN_IF(!aChild.GetParentNode())) { + aError.Throw(NS_ERROR_INVALID_ARG); + return; + } + nsIContent* lastChild = aChild.GetParentNode()->GetLastChild(); + if (NS_WARN_IF(!lastChild)) { + aError.Throw(NS_ERROR_FAILURE); + return; + } + MoveChildrenBetween(aChild, *lastChild, aPointToInsert, aError); + NS_WARNING_ASSERTION(!aError.Failed(), + "HTMLEditor::MoveChildrenBetween() failed"); +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoBlockElementsJoiner:: + DeleteContentButKeepTableStructure(HTMLEditor& aHTMLEditor, + nsIContent& aContent) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + + if (!HTMLEditUtils::IsAnyTableElementButNotTable(&aContent)) { + nsresult rv = aHTMLEditor.DeleteNodeWithTransaction(aContent); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::DeleteNodeWithTransaction() failed"); + return rv; + } + + // XXX For performance, this should just call + // DeleteContentButKeepTableStructure() while there are children in + // aContent. If we need to avoid infinite loop because mutation event + // listeners can add unexpected nodes into aContent, we should just loop + // only original count of the children. + AutoTArray<OwningNonNull<nsIContent>, 10> childList; + for (nsIContent* child = aContent.GetFirstChild(); child; + child = child->GetNextSibling()) { + childList.AppendElement(*child); + } + + for (const auto& child : childList) { + // MOZ_KnownLive because 'childList' is guaranteed to + // keep it alive. + nsresult rv = + DeleteContentButKeepTableStructure(aHTMLEditor, MOZ_KnownLive(child)); + if (NS_FAILED(rv)) { + NS_WARNING("HTMLEditor::DeleteContentButKeepTableStructure() failed"); + return rv; + } + } + return NS_OK; +} + +nsresult HTMLEditor::DeleteMostAncestorMailCiteElementIfEmpty( + nsIContent& aContent) { + MOZ_ASSERT(IsEditActionDataAvailable()); + + // The element must be `<blockquote type="cite">` or + // `<span _moz_quote="true">`. + RefPtr<Element> mailCiteElement = + GetMostDistantAncestorMailCiteElement(aContent); + if (!mailCiteElement) { + return NS_OK; + } + bool seenBR = false; + if (!HTMLEditUtils::IsEmptyNode(*mailCiteElement, + {EmptyCheckOption::TreatListItemAsVisible, + EmptyCheckOption::TreatTableCellAsVisible}, + &seenBR)) { + return NS_OK; + } + EditorDOMPoint atEmptyMailCiteElement(mailCiteElement); + { + AutoEditorDOMPointChildInvalidator lockOffset(atEmptyMailCiteElement); + nsresult rv = DeleteNodeWithTransaction(*mailCiteElement); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return rv; + } + } + + if (!atEmptyMailCiteElement.IsSet() || !seenBR) { + NS_WARNING_ASSERTION( + atEmptyMailCiteElement.IsSet(), + "Mutation event listener might changed the DOM tree during " + "EditorBase::DeleteNodeWithTransaction(), but ignored"); + return NS_OK; + } + + Result<CreateElementResult, nsresult> insertBRElementResult = + InsertBRElement(WithTransaction::Yes, atEmptyMailCiteElement); + if (MOZ_UNLIKELY(insertBRElementResult.isErr())) { + NS_WARNING("HTMLEditor::InsertBRElement(WithTransaction::Yes) failed"); + return insertBRElementResult.unwrapErr(); + } + MOZ_ASSERT(insertBRElementResult.inspect().GetNewNode()); + insertBRElementResult.inspect().IgnoreCaretPointSuggestion(); + nsresult rv = CollapseSelectionTo( + EditorRawDOMPoint(insertBRElementResult.inspect().GetNewNode())); + if (NS_WARN_IF(rv == NS_ERROR_EDITOR_DESTROYED)) { + return NS_ERROR_EDITOR_DESTROYED; + } + NS_WARNING_ASSERTION( + NS_SUCCEEDED(rv), + "EditorBase::::CollapseSelectionTo() failed, but ignored"); + return NS_OK; +} + +Element* HTMLEditor::AutoDeleteRangesHandler::AutoEmptyBlockAncestorDeleter:: + ScanEmptyBlockInclusiveAncestor(const HTMLEditor& aHTMLEditor, + nsIContent& aStartContent) { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!mEmptyInclusiveAncestorBlockElement); + + // If we are inside an empty block, delete it. + // Note: do NOT delete table elements this way. + // Note: do NOT delete non-editable block element. + Element* editableBlockElement = HTMLEditUtils::GetInclusiveAncestorElement( + aStartContent, HTMLEditUtils::ClosestEditableBlockElement); + if (!editableBlockElement) { + return nullptr; + } + // XXX Perhaps, this is slow loop. If empty blocks are nested, then, + // each block checks whether it's empty or not. However, descendant + // blocks are checked again and again by IsEmptyNode(). Perhaps, it + // should be able to take "known empty element" for avoiding same checks. + while (editableBlockElement && + HTMLEditUtils::IsRemovableFromParentNode(*editableBlockElement) && + !HTMLEditUtils::IsAnyTableElement(editableBlockElement) && + HTMLEditUtils::IsEmptyNode(*editableBlockElement)) { + // If the removable empty list item is a child of editing host list element, + // we should not delete it. + if (HTMLEditUtils::IsListItem(editableBlockElement)) { + Element* const parentElement = editableBlockElement->GetParentElement(); + if (parentElement && HTMLEditUtils::IsAnyListElement(parentElement) && + !HTMLEditUtils::IsRemovableFromParentNode(*parentElement) && + HTMLEditUtils::IsEmptyNode(*parentElement)) { + break; + } + } + mEmptyInclusiveAncestorBlockElement = editableBlockElement; + editableBlockElement = HTMLEditUtils::GetAncestorElement( + *mEmptyInclusiveAncestorBlockElement, + HTMLEditUtils::ClosestEditableBlockElement); + } + if (!mEmptyInclusiveAncestorBlockElement) { + return nullptr; + } + + // XXX Because of not checking whether found block element is editable + // in the above loop, empty ediable block element may be overwritten + // with empty non-editable clock element. Therefore, we fail to + // remove the found empty nodes. + if (NS_WARN_IF(!mEmptyInclusiveAncestorBlockElement->IsEditable()) || + NS_WARN_IF(!mEmptyInclusiveAncestorBlockElement->GetParentElement())) { + mEmptyInclusiveAncestorBlockElement = nullptr; + } + return mEmptyInclusiveAncestorBlockElement; +} + +nsresult HTMLEditor::AutoDeleteRangesHandler::AutoEmptyBlockAncestorDeleter:: + ComputeTargetRanges(const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount, + const Element& aEditingHost, + AutoRangeArray& aRangesToDelete) const { + MOZ_ASSERT(mEmptyInclusiveAncestorBlockElement); + + // We'll delete `mEmptyInclusiveAncestorBlockElement` node from the tree, but + // we should return the range from start/end of next/previous editable content + // to end/start of the element for compatiblity with the other browsers. + switch (aDirectionAndAmount) { + case nsIEditor::eNone: + break; + case nsIEditor::ePrevious: + case nsIEditor::ePreviousWord: + case nsIEditor::eToBeginningOfLine: { + EditorRawDOMPoint startPoint = + HTMLEditUtils::GetPreviousEditablePoint<EditorRawDOMPoint>( + *mEmptyInclusiveAncestorBlockElement, &aEditingHost, + // In this case, we don't join block elements so that we won't + // delete invisible trailing whitespaces in the previous element. + InvisibleWhiteSpaces::Preserve, + // In this case, we won't join table cells so that we should + // get a range which is in a table cell even if it's in a + // table. + TableBoundary::NoCrossAnyTableElement); + if (!startPoint.IsSet()) { + NS_WARNING( + "HTMLEditUtils::GetPreviousEditablePoint() didn't return a valid " + "point"); + return NS_ERROR_FAILURE; + } + nsresult rv = aRangesToDelete.SetStartAndEnd( + startPoint, + EditorRawDOMPoint::AtEndOf(mEmptyInclusiveAncestorBlockElement)); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoRangeArray::SetStartAndEnd() failed"); + return rv; + } + case nsIEditor::eNext: + case nsIEditor::eNextWord: + case nsIEditor::eToEndOfLine: { + EditorRawDOMPoint endPoint = + HTMLEditUtils::GetNextEditablePoint<EditorRawDOMPoint>( + *mEmptyInclusiveAncestorBlockElement, &aEditingHost, + // In this case, we don't join block elements so that we won't + // delete invisible trailing whitespaces in the next element. + InvisibleWhiteSpaces::Preserve, + // In this case, we won't join table cells so that we should + // get a range which is in a table cell even if it's in a + // table. + TableBoundary::NoCrossAnyTableElement); + if (!endPoint.IsSet()) { + NS_WARNING( + "HTMLEditUtils::GetNextEditablePoint() didn't return a valid " + "point"); + return NS_ERROR_FAILURE; + } + nsresult rv = aRangesToDelete.SetStartAndEnd( + EditorRawDOMPoint(mEmptyInclusiveAncestorBlockElement, 0), endPoint); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "AutoRangeArray::SetStartAndEnd() failed"); + return rv; + } + default: + MOZ_ASSERT_UNREACHABLE("Handle the nsIEditor::EDirection value"); + break; + } + // No direction, let's select the element to be deleted. + nsresult rv = + aRangesToDelete.SelectNode(*mEmptyInclusiveAncestorBlockElement); + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "AutoRangeArray::SelectNode() failed"); + return rv; +} + +Result<RefPtr<Element>, nsresult> +HTMLEditor::AutoDeleteRangesHandler::AutoEmptyBlockAncestorDeleter:: + MaybeInsertBRElementBeforeEmptyListItemElement(HTMLEditor& aHTMLEditor) { + MOZ_ASSERT(mEmptyInclusiveAncestorBlockElement); + MOZ_ASSERT(mEmptyInclusiveAncestorBlockElement->GetParentElement()); + MOZ_ASSERT(HTMLEditUtils::IsListItem(mEmptyInclusiveAncestorBlockElement)); + + // If the found empty block is a list item element and its grand parent + // (i.e., parent of list element) is NOT a list element, insert <br> + // element before the list element which has the empty list item. + // This odd list structure may occur if `Document.execCommand("indent")` + // is performed for list items. + // XXX Chrome does not remove empty list elements when last content in + // last list item is deleted. We should follow it since current + // behavior is annoying when you type new list item with selecting + // all list items. + if (!HTMLEditUtils::IsFirstChild(*mEmptyInclusiveAncestorBlockElement, + {WalkTreeOption::IgnoreNonEditableNode})) { + return RefPtr<Element>(); + } + + EditorDOMPoint atParentOfEmptyListItem( + mEmptyInclusiveAncestorBlockElement->GetParentElement()); + if (NS_WARN_IF(!atParentOfEmptyListItem.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + if (HTMLEditUtils::IsAnyListElement(atParentOfEmptyListItem.GetContainer())) { + return RefPtr<Element>(); + } + Result<CreateElementResult, nsresult> insertBRElementResult = + aHTMLEditor.InsertBRElement(WithTransaction::Yes, + atParentOfEmptyListItem); + if (MOZ_UNLIKELY(insertBRElementResult.isErr())) { + NS_WARNING("HTMLEditor::InsertBRElement(WithTransaction::Yes) failed"); + return insertBRElementResult.propagateErr(); + } + CreateElementResult unwrappedInsertBRElementResult = + insertBRElementResult.unwrap(); + nsresult rv = unwrappedInsertBRElementResult.SuggestCaretPointTo( + aHTMLEditor, {SuggestCaret::OnlyIfHasSuggestion, + SuggestCaret::OnlyIfTransactionsAllowedToDoIt, + SuggestCaret::AndIgnoreTrivialError}); + if (NS_FAILED(rv)) { + NS_WARNING("CreateElementResult::SuggestCaretPointTo() failed"); + return Err(rv); + } + MOZ_ASSERT(unwrappedInsertBRElementResult.GetNewNode()); + return unwrappedInsertBRElementResult.UnwrapNewNode(); +} + +Result<CaretPoint, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoEmptyBlockAncestorDeleter::GetNewCaretPosition( + const HTMLEditor& aHTMLEditor, + nsIEditor::EDirection aDirectionAndAmount) const { + MOZ_ASSERT(mEmptyInclusiveAncestorBlockElement); + MOZ_ASSERT(mEmptyInclusiveAncestorBlockElement->GetParentElement()); + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + + switch (aDirectionAndAmount) { + case nsIEditor::eNext: + case nsIEditor::eNextWord: + case nsIEditor::eToEndOfLine: { + // Collapse Selection to next node of after empty block element + // if there is. Otherwise, to just after the empty block. + auto afterEmptyBlock( + EditorDOMPoint::After(mEmptyInclusiveAncestorBlockElement)); + MOZ_ASSERT(afterEmptyBlock.IsSet()); + if (nsIContent* nextContentOfEmptyBlock = HTMLEditUtils::GetNextContent( + afterEmptyBlock, {}, aHTMLEditor.ComputeEditingHost())) { + EditorDOMPoint pt = HTMLEditUtils::GetGoodCaretPointFor<EditorDOMPoint>( + *nextContentOfEmptyBlock, aDirectionAndAmount); + if (!pt.IsSet()) { + NS_WARNING("HTMLEditUtils::GetGoodCaretPointFor() failed"); + return Err(NS_ERROR_FAILURE); + } + return CaretPoint(std::move(pt)); + } + if (NS_WARN_IF(!afterEmptyBlock.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + return CaretPoint(std::move(afterEmptyBlock)); + } + case nsIEditor::ePrevious: + case nsIEditor::ePreviousWord: + case nsIEditor::eToBeginningOfLine: { + // Collapse Selection to previous editable node of the empty block + // if there is. Otherwise, to after the empty block. + EditorRawDOMPoint atEmptyBlock(mEmptyInclusiveAncestorBlockElement); + if (nsIContent* previousContentOfEmptyBlock = + HTMLEditUtils::GetPreviousContent( + atEmptyBlock, {WalkTreeOption::IgnoreNonEditableNode}, + aHTMLEditor.ComputeEditingHost())) { + EditorDOMPoint pt = HTMLEditUtils::GetGoodCaretPointFor<EditorDOMPoint>( + *previousContentOfEmptyBlock, aDirectionAndAmount); + if (!pt.IsSet()) { + NS_WARNING("HTMLEditUtils::GetGoodCaretPointFor() failed"); + return Err(NS_ERROR_FAILURE); + } + return CaretPoint(std::move(pt)); + } + auto afterEmptyBlock = + EditorDOMPoint::After(*mEmptyInclusiveAncestorBlockElement); + if (NS_WARN_IF(!afterEmptyBlock.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + return CaretPoint(std::move(afterEmptyBlock)); + } + case nsIEditor::eNone: { + // Collapse selection at the removing block when we are replacing + // selected content. + EditorDOMPoint atEmptyBlock(mEmptyInclusiveAncestorBlockElement); + if (NS_WARN_IF(!atEmptyBlock.IsSet())) { + return Err(NS_ERROR_FAILURE); + } + return CaretPoint(std::move(atEmptyBlock)); + } + default: + MOZ_CRASH( + "AutoEmptyBlockAncestorDeleter doesn't support this action yet"); + return Err(NS_ERROR_FAILURE); + } +} + +Result<EditActionResult, nsresult> +HTMLEditor::AutoDeleteRangesHandler::AutoEmptyBlockAncestorDeleter::Run( + HTMLEditor& aHTMLEditor, nsIEditor::EDirection aDirectionAndAmount) { + MOZ_ASSERT(mEmptyInclusiveAncestorBlockElement); + MOZ_ASSERT(mEmptyInclusiveAncestorBlockElement->GetParentElement()); + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + + { + Result<EditActionResult, nsresult> result = + MaybeReplaceSubListWithNewListItem(aHTMLEditor); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING( + "AutoEmptyBlockAncestorDeleter::MaybeReplaceSubListWithNewListItem() " + "failed"); + return result; + } + if (result.inspect().Handled()) { + return result; + } + } + + if (HTMLEditUtils::IsListItem(mEmptyInclusiveAncestorBlockElement)) { + Result<RefPtr<Element>, nsresult> result = + MaybeInsertBRElementBeforeEmptyListItemElement(aHTMLEditor); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING( + "AutoEmptyBlockAncestorDeleter::" + "MaybeInsertBRElementBeforeEmptyListItemElement() failed"); + return result.propagateErr(); + } + // If a `<br>` element is inserted, caret should be moved to after it. + if (RefPtr<Element> brElement = result.unwrap()) { + nsresult rv = + aHTMLEditor.CollapseSelectionTo(EditorRawDOMPoint(brElement)); + if (NS_FAILED(rv)) { + NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), + "EditorBase::CollapseSelectionTo() failed"); + return Err(rv); + } + } + } else { + Result<CaretPoint, nsresult> result = + GetNewCaretPosition(aHTMLEditor, aDirectionAndAmount); + if (MOZ_UNLIKELY(result.isErr())) { + NS_WARNING("AutoEmptyBlockAncestorDeleter::GetNewCaretPosition() failed"); + return result.propagateErr(); + } + MOZ_ASSERT(result.inspect().HasCaretPointSuggestion()); + nsresult rv = result.inspect().SuggestCaretPointTo(aHTMLEditor, {}); + if (NS_FAILED(rv)) { + NS_WARNING("CaretPoint::SuggestCaretPointTo() failed"); + return Err(rv); + } + } + nsresult rv = aHTMLEditor.DeleteNodeWithTransaction( + MOZ_KnownLive(*mEmptyInclusiveAncestorBlockElement)); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return Err(rv); + } + return EditActionResult::HandledResult(); +} + +Result<EditActionResult, nsresult> HTMLEditor::AutoDeleteRangesHandler:: + AutoEmptyBlockAncestorDeleter::MaybeReplaceSubListWithNewListItem( + HTMLEditor& aHTMLEditor) { + // If we're deleting sublist element and it's the last list item of its parent + // list, we should replace it with a list element. + if (!HTMLEditUtils::IsAnyListElement(mEmptyInclusiveAncestorBlockElement)) { + return EditActionResult::IgnoredResult(); + } + RefPtr<Element> parentElement = + mEmptyInclusiveAncestorBlockElement->GetParentElement(); + if (!parentElement || !HTMLEditUtils::IsAnyListElement(parentElement) || + !HTMLEditUtils::IsEmptyNode(*parentElement)) { + return EditActionResult::IgnoredResult(); + } + + nsCOMPtr<nsINode> nextSibling = + mEmptyInclusiveAncestorBlockElement->GetNextSibling(); + nsresult rv = aHTMLEditor.DeleteNodeWithTransaction( + MOZ_KnownLive(*mEmptyInclusiveAncestorBlockElement)); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed"); + return Err(rv); + } + Result<CreateElementResult, nsresult> insertListItemResult = + aHTMLEditor.CreateAndInsertElement( + WithTransaction::Yes, + parentElement->IsHTMLElement(nsGkAtoms::dl) ? *nsGkAtoms::dd + : *nsGkAtoms::li, + !nextSibling || nextSibling->GetParentNode() != parentElement + ? EditorDOMPoint::AtEndOf(*parentElement) + : EditorDOMPoint(nextSibling), + [](HTMLEditor& aHTMLEditor, Element& aNewElement, + const EditorDOMPoint& aPointToInsert) -> nsresult { + RefPtr<Element> brElement = + aHTMLEditor.CreateHTMLContent(nsGkAtoms::br); + if (MOZ_UNLIKELY(!brElement)) { + NS_WARNING( + "EditorBase::CreateHTMLContent(nsGkAtoms::br) failed, but " + "ignored"); + return NS_OK; // Just gives up to insert <br> + } + IgnoredErrorResult error; + aNewElement.AppendChild(*brElement, error); + NS_WARNING_ASSERTION(!error.Failed(), + "nsINode::AppendChild() failed, but ignored"); + return NS_OK; + }); + if (MOZ_UNLIKELY(insertListItemResult.isErr())) { + NS_WARNING("HTMLEditor::CreateAndInsertElement() failed"); + return insertListItemResult.propagateErr(); + } + CreateElementResult unwrappedInsertListItemResult = + insertListItemResult.unwrap(); + unwrappedInsertListItemResult.IgnoreCaretPointSuggestion(); + rv = aHTMLEditor.CollapseSelectionTo( + EditorRawDOMPoint(unwrappedInsertListItemResult.GetNewNode(), 0u)); + if (NS_FAILED(rv)) { + NS_WARNING("EditorBase::CollapseSelectionTo() failed"); + return Err(rv); + } + return EditActionResult::HandledResult(); +} + +template <typename EditorDOMRangeType> +Result<EditorRawDOMRange, nsresult> +HTMLEditor::AutoDeleteRangesHandler::ExtendOrShrinkRangeToDelete( + const HTMLEditor& aHTMLEditor, const nsFrameSelection* aFrameSelection, + const EditorDOMRangeType& aRangeToDelete) const { + MOZ_ASSERT(aHTMLEditor.IsEditActionDataAvailable()); + MOZ_ASSERT(!aRangeToDelete.Collapsed()); + MOZ_ASSERT(aRangeToDelete.IsPositioned()); + + const nsIContent* commonAncestor = nsIContent::FromNodeOrNull( + nsContentUtils::GetClosestCommonInclusiveAncestor( + aRangeToDelete.StartRef().GetContainer(), + aRangeToDelete.EndRef().GetContainer())); + if (MOZ_UNLIKELY(NS_WARN_IF(!commonAncestor))) { + return Err(NS_ERROR_FAILURE); + } + + // Look for the common ancestor's block element. It's fine that we get + // non-editable block element which is ancestor of inline editing host + // because the following code checks editing host too. + const Element* const maybeNonEditableBlockElement = + HTMLEditUtils::GetInclusiveAncestorElement( + *commonAncestor, HTMLEditUtils::ClosestBlockElement); + if (NS_WARN_IF(!maybeNonEditableBlockElement)) { + return Err(NS_ERROR_FAILURE); + } + + // Set up for loops and cache our root element + RefPtr<Element> editingHost = aHTMLEditor.ComputeEditingHost(); + if (NS_WARN_IF(!editingHost)) { + return Err(NS_ERROR_FAILURE); + } + + // If only one list element is selected, and if the list element is empty, + // we should delete only the list element. Or if the list element is not + // empty, we should make the list has only one empty list item element. + if (const Element* maybeListElement = + HTMLEditUtils::GetElementIfOnlyOneSelected(aRangeToDelete)) { + if (HTMLEditUtils::IsAnyListElement(maybeListElement) && + !HTMLEditUtils::IsEmptyAnyListElement(*maybeListElement)) { + EditorRawDOMRange range = + HTMLEditUtils::GetRangeSelectingAllContentInAllListItems< + EditorRawDOMRange>(*maybeListElement); + if (range.IsPositioned()) { + if (EditorUtils::IsEditableContent( + *range.StartRef().ContainerAs<nsIContent>(), + EditorType::HTML) && + EditorUtils::IsEditableContent( + *range.EndRef().ContainerAs<nsIContent>(), EditorType::HTML)) { + return range; + } + } + // If the first and/or last list item is not editable, we need to do more + // complicated things probably, but we just delete the list element with + // invisible things around it for now since it must be rare case. + } + // Otherwise, if the list item is empty, we should delete it with invisible + // things around it. + } + + // Find previous visible things before start of selection + EditorRawDOMRange rangeToDelete(aRangeToDelete); + if (rangeToDelete.StartRef().GetContainer() != maybeNonEditableBlockElement && + rangeToDelete.StartRef().GetContainer() != editingHost) { + for (;;) { + WSScanResult backwardScanFromStartResult = + WSRunScanner::ScanPreviousVisibleNodeOrBlockBoundary( + editingHost, rangeToDelete.StartRef()); + if (!backwardScanFromStartResult.ReachedCurrentBlockBoundary()) { + break; + } + MOZ_ASSERT(backwardScanFromStartResult.GetContent() == + WSRunScanner(editingHost, rangeToDelete.StartRef()) + .GetStartReasonContent()); + // We want to keep looking up. But stop if we are crossing table + // element boundaries, or if we hit the root. + if (HTMLEditUtils::IsAnyTableElement( + backwardScanFromStartResult.GetContent()) || + backwardScanFromStartResult.GetContent() == + maybeNonEditableBlockElement || + backwardScanFromStartResult.GetContent() == editingHost) { + break; + } + // Don't cross list element boundary because we don't want to delete list + // element at start position unless it's empty. + if (HTMLEditUtils::IsAnyListElement( + backwardScanFromStartResult.GetContent()) && + !HTMLEditUtils::IsEmptyAnyListElement( + *backwardScanFromStartResult.ElementPtr())) { + break; + } + rangeToDelete.SetStart( + backwardScanFromStartResult.PointAtContent<EditorRawDOMPoint>()); + } + if (aFrameSelection && !aFrameSelection->IsValidSelectionPoint( + rangeToDelete.StartRef().GetContainer())) { + NS_WARNING("Computed start container was out of selection limiter"); + return Err(NS_ERROR_FAILURE); + } + } + + // Expand selection endpoint only if we don't pass an invisible `<br>`, or if + // we really needed to pass that `<br>` (i.e., its block is now totally + // selected). + + // Find next visible things after end of selection + EditorDOMPoint atFirstInvisibleBRElement; + if (rangeToDelete.EndRef().GetContainer() != maybeNonEditableBlockElement && + rangeToDelete.EndRef().GetContainer() != editingHost) { + for (;;) { + WSRunScanner wsScannerAtEnd(editingHost, rangeToDelete.EndRef()); + WSScanResult forwardScanFromEndResult = + wsScannerAtEnd.ScanNextVisibleNodeOrBlockBoundaryFrom( + rangeToDelete.EndRef()); + if (forwardScanFromEndResult.ReachedBRElement()) { + // XXX In my understanding, this is odd. The end reason may not be + // same as the reached <br> element because the equality is + // guaranteed only when ReachedCurrentBlockBoundary() returns true. + // However, looks like that this code assumes that + // GetEndReasonContent() returns the (or a) <br> element. + NS_ASSERTION(wsScannerAtEnd.GetEndReasonContent() == + forwardScanFromEndResult.BRElementPtr(), + "End reason is not the reached <br> element"); + if (HTMLEditUtils::IsVisibleBRElement( + *wsScannerAtEnd.GetEndReasonContent())) { + break; + } + if (!atFirstInvisibleBRElement.IsSet()) { + atFirstInvisibleBRElement = + rangeToDelete.EndRef().To<EditorDOMPoint>(); + } + rangeToDelete.SetEnd( + EditorRawDOMPoint::After(*wsScannerAtEnd.GetEndReasonContent())); + continue; + } + + if (forwardScanFromEndResult.ReachedCurrentBlockBoundary()) { + MOZ_ASSERT(forwardScanFromEndResult.GetContent() == + wsScannerAtEnd.GetEndReasonContent()); + // We want to keep looking up. But stop if we are crossing table + // element boundaries, or if we hit the root. + if (HTMLEditUtils::IsAnyTableElement( + forwardScanFromEndResult.GetContent()) || + forwardScanFromEndResult.GetContent() == + maybeNonEditableBlockElement || + forwardScanFromEndResult.GetContent() == editingHost) { + break; + } + rangeToDelete.SetEnd( + forwardScanFromEndResult.PointAfterContent<EditorRawDOMPoint>()); + continue; + } + + break; + } + + if (aFrameSelection && !aFrameSelection->IsValidSelectionPoint( + rangeToDelete.EndRef().GetContainer())) { + NS_WARNING("Computed end container was out of selection limiter"); + return Err(NS_ERROR_FAILURE); + } + } + + // If range boundaries are in list element, and the positions are very + // start/end of first/last list item, we may need to shrink the ranges for + // preventing to remove only all list item elements. + { + EditorRawDOMRange rangeToDeleteListOrLeaveOneEmptyListItem = + AutoDeleteRangesHandler:: + GetRangeToAvoidDeletingAllListItemsIfSelectingAllOverListElements( + rangeToDelete); + if (rangeToDeleteListOrLeaveOneEmptyListItem.IsPositioned()) { + rangeToDelete = std::move(rangeToDeleteListOrLeaveOneEmptyListItem); + } + } + + if (atFirstInvisibleBRElement.IsInContentNode()) { + // Find block node containing invisible `<br>` element. + if (const RefPtr<const Element> editableBlockContainingBRElement = + HTMLEditUtils::GetInclusiveAncestorElement( + *atFirstInvisibleBRElement.ContainerAs<nsIContent>(), + HTMLEditUtils::ClosestEditableBlockElement)) { + if (rangeToDelete.Contains( + EditorRawDOMPoint(editableBlockContainingBRElement))) { + return rangeToDelete; + } + // Otherwise, the new range should end at the invisible `<br>`. + if (aFrameSelection && !aFrameSelection->IsValidSelectionPoint( + atFirstInvisibleBRElement.GetContainer())) { + NS_WARNING( + "Computed end container (`<br>` element) was out of selection " + "limiter"); + return Err(NS_ERROR_FAILURE); + } + rangeToDelete.SetEnd(atFirstInvisibleBRElement); + } + } + + return rangeToDelete; +} + +// static +EditorRawDOMRange HTMLEditor::AutoDeleteRangesHandler:: + GetRangeToAvoidDeletingAllListItemsIfSelectingAllOverListElements( + const EditorRawDOMRange& aRangeToDelete) { + MOZ_ASSERT(aRangeToDelete.IsPositionedAndValid()); + + auto GetDeepestEditableStartPointOfList = [](Element& aListElement) { + Element* const firstListItemElement = + HTMLEditUtils::GetFirstListItemElement(aListElement); + if (MOZ_UNLIKELY(!firstListItemElement)) { + return EditorRawDOMPoint(); + } + if (MOZ_UNLIKELY(!EditorUtils::IsEditableContent(*firstListItemElement, + EditorType::HTML))) { + return EditorRawDOMPoint(firstListItemElement); + } + return HTMLEditUtils::GetDeepestEditableStartPointOf<EditorRawDOMPoint>( + *firstListItemElement); + }; + + auto GetDeepestEditableEndPointOfList = [](Element& aListElement) { + Element* const lastListItemElement = + HTMLEditUtils::GetLastListItemElement(aListElement); + if (MOZ_UNLIKELY(!lastListItemElement)) { + return EditorRawDOMPoint(); + } + if (MOZ_UNLIKELY(!EditorUtils::IsEditableContent(*lastListItemElement, + EditorType::HTML))) { + return EditorRawDOMPoint::After(*lastListItemElement); + } + return HTMLEditUtils::GetDeepestEditableEndPointOf<EditorRawDOMPoint>( + *lastListItemElement); + }; + + Element* const startListElement = + aRangeToDelete.StartRef().IsInContentNode() + ? HTMLEditUtils::GetClosestInclusiveAncestorAnyListElement( + *aRangeToDelete.StartRef().ContainerAs<nsIContent>()) + : nullptr; + Element* const endListElement = + aRangeToDelete.EndRef().IsInContentNode() + ? HTMLEditUtils::GetClosestInclusiveAncestorAnyListElement( + *aRangeToDelete.EndRef().ContainerAs<nsIContent>()) + : nullptr; + if (!startListElement && !endListElement) { + return EditorRawDOMRange(); + } + + // FIXME: If there are invalid children, we cannot handle first/last list item + // elements properly. In that case, we should treat list elements and list + // item elements as normal block elements. + if (startListElement && + NS_WARN_IF(!HTMLEditUtils::IsValidListElement( + *startListElement, HTMLEditUtils::TreatSubListElementAs::Valid))) { + return EditorRawDOMRange(); + } + if (endListElement && startListElement != endListElement && + NS_WARN_IF(!HTMLEditUtils::IsValidListElement( + *endListElement, HTMLEditUtils::TreatSubListElementAs::Valid))) { + return EditorRawDOMRange(); + } + + const bool startListElementIsEmpty = + startListElement && + HTMLEditUtils::IsEmptyAnyListElement(*startListElement); + const bool endListElementIsEmpty = + startListElement == endListElement + ? startListElementIsEmpty + : endListElement && + HTMLEditUtils::IsEmptyAnyListElement(*endListElement); + // If both list elements are empty, we should not shrink the range since + // we want to delete the list. + if (startListElementIsEmpty && endListElementIsEmpty) { + return EditorRawDOMRange(); + } + + // There may be invisible white-spaces and there are elements in the + // list items. Therefore, we need to compare the deepest positions + // and range boundaries. + EditorRawDOMPoint deepestStartPointOfStartList = + startListElement ? GetDeepestEditableStartPointOfList(*startListElement) + : EditorRawDOMPoint(); + EditorRawDOMPoint deepestEndPointOfEndList = + endListElement ? GetDeepestEditableEndPointOfList(*endListElement) + : EditorRawDOMPoint(); + if (MOZ_UNLIKELY(!deepestStartPointOfStartList.IsSet() && + !deepestEndPointOfEndList.IsSet())) { + // FIXME: This does not work well if there is non-list-item contents in the + // list elements. Perhaps, for fixing this invalid cases, we need to wrap + // the content into new list item like Chrome. + return EditorRawDOMRange(); + } + + // We don't want to shrink the range into empty sublist. + if (deepestStartPointOfStartList.IsSet()) { + for (nsIContent* const maybeList : + deepestStartPointOfStartList.GetContainer() + ->InclusiveAncestorsOfType<nsIContent>()) { + if (aRangeToDelete.StartRef().GetContainer() == maybeList) { + break; + } + if (HTMLEditUtils::IsAnyListElement(maybeList) && + HTMLEditUtils::IsEmptyAnyListElement(*maybeList->AsElement())) { + deepestStartPointOfStartList.Set(maybeList); + } + } + } + if (deepestEndPointOfEndList.IsSet()) { + for (nsIContent* const maybeList : + deepestEndPointOfEndList.GetContainer() + ->InclusiveAncestorsOfType<nsIContent>()) { + if (aRangeToDelete.EndRef().GetContainer() == maybeList) { + break; + } + if (HTMLEditUtils::IsAnyListElement(maybeList) && + HTMLEditUtils::IsEmptyAnyListElement(*maybeList->AsElement())) { + deepestEndPointOfEndList.SetAfter(maybeList); + } + } + } + + const EditorRawDOMPoint deepestEndPointOfStartList = + startListElement ? GetDeepestEditableEndPointOfList(*startListElement) + : EditorRawDOMPoint(); + MOZ_ASSERT_IF(deepestStartPointOfStartList.IsSet(), + deepestEndPointOfStartList.IsSet()); + MOZ_ASSERT_IF(!deepestStartPointOfStartList.IsSet(), + !deepestEndPointOfStartList.IsSet()); + + const bool rangeStartsFromBeginningOfStartList = + deepestStartPointOfStartList.IsSet() && + aRangeToDelete.StartRef().EqualsOrIsBefore(deepestStartPointOfStartList); + const bool rangeEndsByEndingOfStartListOrLater = + !deepestEndPointOfStartList.IsSet() || + deepestEndPointOfStartList.EqualsOrIsBefore(aRangeToDelete.EndRef()); + const bool rangeEndsByEndingOfEndList = + deepestEndPointOfEndList.IsSet() && + deepestEndPointOfEndList.EqualsOrIsBefore(aRangeToDelete.EndRef()); + + EditorRawDOMRange newRangeToDelete; + // If all over the list element at start boundary is selected, we should + // shrink the range to start from the first list item to avoid to delete + // all list items. + if (!startListElementIsEmpty && rangeStartsFromBeginningOfStartList && + rangeEndsByEndingOfStartListOrLater) { + newRangeToDelete.SetStart(EditorRawDOMPoint( + deepestStartPointOfStartList.ContainerAs<nsIContent>(), 0u)); + } + // If all over the list element at end boundary is selected, and... + if (!endListElementIsEmpty && rangeEndsByEndingOfEndList) { + // If the range starts before the range at end boundary of the range, + // we want to delete the list completely, thus, we should extend the + // range to contain the list element. + if (aRangeToDelete.StartRef().IsBefore( + EditorRawDOMPoint(endListElement, 0u))) { + newRangeToDelete.SetEnd(EditorRawDOMPoint::After(*endListElement)); + MOZ_ASSERT_IF(newRangeToDelete.StartRef().IsSet(), + newRangeToDelete.IsPositionedAndValid()); + } + // Otherwise, if the range starts in the end list element, we shouldn't + // delete the list. Therefore, we should shrink the range to end by end + // of the last list item element to avoid to delete all list items. + else { + newRangeToDelete.SetEnd(EditorRawDOMPoint::AtEndOf( + *deepestEndPointOfEndList.ContainerAs<nsIContent>())); + MOZ_ASSERT_IF(newRangeToDelete.StartRef().IsSet(), + newRangeToDelete.IsPositionedAndValid()); + } + } + + if (!newRangeToDelete.StartRef().IsSet() && + !newRangeToDelete.EndRef().IsSet()) { + return EditorRawDOMRange(); + } + + if (!newRangeToDelete.StartRef().IsSet()) { + newRangeToDelete.SetStart(aRangeToDelete.StartRef()); + MOZ_ASSERT(newRangeToDelete.IsPositionedAndValid()); + } + if (!newRangeToDelete.EndRef().IsSet()) { + newRangeToDelete.SetEnd(aRangeToDelete.EndRef()); + MOZ_ASSERT(newRangeToDelete.IsPositionedAndValid()); + } + + return newRangeToDelete; +} + +} // namespace mozilla |