summaryrefslogtreecommitdiffstats
path: root/editor/libeditor/EditorBase.cpp
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 19:33:14 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 19:33:14 +0000
commit36d22d82aa202bb199967e9512281e9a53db42c9 (patch)
tree105e8c98ddea1c1e4784a60a5a6410fa416be2de /editor/libeditor/EditorBase.cpp
parentInitial commit. (diff)
downloadfirefox-esr-upstream.tar.xz
firefox-esr-upstream.zip
Adding upstream version 115.7.0esr.upstream/115.7.0esrupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
-rw-r--r--editor/libeditor/EditorBase.cpp7065
1 files changed, 7065 insertions, 0 deletions
diff --git a/editor/libeditor/EditorBase.cpp b/editor/libeditor/EditorBase.cpp
new file mode 100644
index 0000000000..0f864a4bb2
--- /dev/null
+++ b/editor/libeditor/EditorBase.cpp
@@ -0,0 +1,7065 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#include "EditorBase.h"
+
+#include <stdio.h> // for nullptr, stdout
+#include <string.h> // for strcmp
+
+#include "AutoRangeArray.h" // for AutoRangeArray
+#include "ChangeAttributeTransaction.h"
+#include "CompositionTransaction.h"
+#include "DeleteContentTransactionBase.h"
+#include "DeleteMultipleRangesTransaction.h"
+#include "DeleteNodeTransaction.h"
+#include "DeleteRangeTransaction.h"
+#include "DeleteTextTransaction.h"
+#include "EditAction.h" // for EditSubAction
+#include "EditorDOMPoint.h" // for EditorDOMPoint
+#include "EditorUtils.h" // for various helper classes.
+#include "EditTransactionBase.h" // for EditTransactionBase
+#include "EditorEventListener.h" // for EditorEventListener
+#include "HTMLEditor.h" // for HTMLEditor
+#include "HTMLEditorInlines.h"
+#include "HTMLEditUtils.h" // for HTMLEditUtils
+#include "InsertNodeTransaction.h" // for InsertNodeTransaction
+#include "InsertTextTransaction.h" // for InsertTextTransaction
+#include "JoinNodesTransaction.h" // for JoinNodesTransaction
+#include "PlaceholderTransaction.h" // for PlaceholderTransaction
+#include "SplitNodeTransaction.h" // for SplitNodeTransaction
+#include "TextEditor.h" // for TextEditor
+
+#include "ErrorList.h"
+#include "gfxFontUtils.h" // for gfxFontUtils
+#include "mozilla/Assertions.h"
+#include "mozilla/intl/BidiEmbeddingLevel.h"
+#include "mozilla/BasePrincipal.h" // for BasePrincipal
+#include "mozilla/CheckedInt.h" // for CheckedInt
+#include "mozilla/ComposerCommandsUpdater.h" // for ComposerCommandsUpdater
+#include "mozilla/ContentEvents.h" // for InternalClipboardEvent
+#include "mozilla/DebugOnly.h" // for DebugOnly
+#include "mozilla/EditorSpellCheck.h" // for EditorSpellCheck
+#include "mozilla/Encoding.h" // for Encoding (used in Document::GetDocumentCharacterSet)
+#include "mozilla/EventDispatcher.h" // for EventChainPreVisitor, etc.
+#include "mozilla/FlushType.h" // for FlushType::Frames
+#include "mozilla/IMEContentObserver.h" // for IMEContentObserver
+#include "mozilla/IMEStateManager.h" // for IMEStateManager
+#include "mozilla/InputEventOptions.h" // for InputEventOptions
+#include "mozilla/IntegerRange.h" // for IntegerRange
+#include "mozilla/InternalMutationEvent.h" // for NS_EVENT_BITS_MUTATION_CHARACTERDATAMODIFIED
+#include "mozilla/mozalloc.h" // for operator new, etc.
+#include "mozilla/mozInlineSpellChecker.h" // for mozInlineSpellChecker
+#include "mozilla/mozSpellChecker.h" // for mozSpellChecker
+#include "mozilla/Preferences.h" // for Preferences
+#include "mozilla/PresShell.h" // for PresShell
+#include "mozilla/RangeBoundary.h" // for RawRangeBoundary, RangeBoundary
+#include "mozilla/Services.h" // for GetObserverService
+#include "mozilla/StaticPrefs_bidi.h" // for StaticPrefs::bidi_*
+#include "mozilla/StaticPrefs_dom.h" // for StaticPrefs::dom_*
+#include "mozilla/StaticPrefs_editor.h" // for StaticPrefs::editor_*
+#include "mozilla/StaticPrefs_layout.h" // for StaticPrefs::layout_*
+#include "mozilla/TextComposition.h" // for TextComposition
+#include "mozilla/TextControlElement.h" // for TextControlElement
+#include "mozilla/TextInputListener.h" // for TextInputListener
+#include "mozilla/TextServicesDocument.h" // for TextServicesDocument
+#include "mozilla/TextEvents.h"
+#include "mozilla/TransactionManager.h" // for TransactionManager
+#include "mozilla/dom/AbstractRange.h" // for AbstractRange
+#include "mozilla/dom/Attr.h" // for Attr
+#include "mozilla/dom/BrowsingContext.h" // for BrowsingContext
+#include "mozilla/dom/CharacterData.h" // for CharacterData
+#include "mozilla/dom/DataTransfer.h" // for DataTransfer
+#include "mozilla/dom/Document.h" // for Document
+#include "mozilla/dom/DocumentInlines.h" // for GetObservingPresShell
+#include "mozilla/dom/DragEvent.h" // for DragEvent
+#include "mozilla/dom/Element.h" // for Element, nsINode::AsElement
+#include "mozilla/dom/EventTarget.h" // for EventTarget
+#include "mozilla/dom/HTMLBodyElement.h"
+#include "mozilla/dom/HTMLBRElement.h"
+#include "mozilla/dom/Selection.h" // for Selection, etc.
+#include "mozilla/dom/StaticRange.h" // for StaticRange
+#include "mozilla/dom/Text.h"
+#include "mozilla/dom/Event.h"
+#include "nsAString.h" // for nsAString::Length, etc.
+#include "nsCCUncollectableMarker.h" // for nsCCUncollectableMarker
+#include "nsCaret.h" // for nsCaret
+#include "nsCaseTreatment.h"
+#include "nsCharTraits.h" // for NS_IS_HIGH_SURROGATE, etc.
+#include "nsContentUtils.h" // for nsContentUtils
+#include "nsCopySupport.h" // for nsCopySupport
+#include "nsDOMString.h" // for DOMStringIsNull
+#include "nsDebug.h" // for NS_WARNING, etc.
+#include "nsError.h" // for NS_OK, etc.
+#include "nsFocusManager.h" // for nsFocusManager
+#include "nsFrameSelection.h" // for nsFrameSelection
+#include "nsGenericHTMLElement.h" // for nsGenericHTMLElement
+#include "nsGkAtoms.h" // for nsGkAtoms, nsGkAtoms::dir
+#include "nsIClipboard.h" // for nsIClipboard
+#include "nsIContent.h" // for nsIContent
+#include "nsIContentInlines.h" // for nsINode::IsInDesignMode()
+#include "nsIDocumentEncoder.h" // for nsIDocumentEncoder
+#include "nsIDocumentStateListener.h" // for nsIDocumentStateListener
+#include "nsIDocShell.h" // for nsIDocShell
+#include "nsIEditActionListener.h" // for nsIEditActionListener
+#include "nsIFrame.h" // for nsIFrame
+#include "nsIInlineSpellChecker.h" // for nsIInlineSpellChecker, etc.
+#include "nsNameSpaceManager.h" // for kNameSpaceID_None, etc.
+#include "nsINode.h" // for nsINode, etc.
+#include "nsISelectionController.h" // for nsISelectionController, etc.
+#include "nsISelectionDisplay.h" // for nsISelectionDisplay, etc.
+#include "nsISupports.h" // for nsISupports
+#include "nsISupportsUtils.h" // for NS_ADDREF, NS_IF_ADDREF
+#include "nsITransferable.h" // for nsITransferable
+#include "nsIWeakReference.h" // for nsISupportsWeakReference
+#include "nsIWidget.h" // for nsIWidget, IMEState, etc.
+#include "nsPIDOMWindow.h" // for nsPIDOMWindow
+#include "nsPresContext.h" // for nsPresContext
+#include "nsRange.h" // for nsRange
+#include "nsReadableUtils.h" // for EmptyString, ToNewCString
+#include "nsString.h" // for nsAutoString, nsString, etc.
+#include "nsStringFwd.h" // for nsString
+#include "nsStyleConsts.h" // for StyleDirection::Rtl, etc.
+#include "nsStyleStruct.h" // for nsStyleDisplay, nsStyleText, etc.
+#include "nsStyleStructFwd.h" // for nsIFrame::StyleUIReset, etc.
+#include "nsTextNode.h" // for nsTextNode
+#include "nsThreadUtils.h" // for nsRunnable
+#include "prtime.h" // for PR_Now
+
+class nsIOutputStream;
+class nsITransferable;
+
+namespace mozilla {
+
+using namespace dom;
+using namespace widget;
+
+using EmptyCheckOption = HTMLEditUtils::EmptyCheckOption;
+using LeafNodeType = HTMLEditUtils::LeafNodeType;
+using LeafNodeTypes = HTMLEditUtils::LeafNodeTypes;
+using WalkTreeOption = HTMLEditUtils::WalkTreeOption;
+
+/*****************************************************************************
+ * mozilla::EditorBase
+ *****************************************************************************/
+template EditorDOMPoint EditorBase::GetFirstIMESelectionStartPoint() const;
+template EditorRawDOMPoint EditorBase::GetFirstIMESelectionStartPoint() const;
+template EditorDOMPoint EditorBase::GetLastIMESelectionEndPoint() const;
+template EditorRawDOMPoint EditorBase::GetLastIMESelectionEndPoint() const;
+
+template Result<CreateContentResult, nsresult>
+EditorBase::InsertNodeWithTransaction(nsIContent& aContentToInsert,
+ const EditorDOMPoint& aPointToInsert);
+template Result<CreateElementResult, nsresult>
+EditorBase::InsertNodeWithTransaction(Element& aContentToInsert,
+ const EditorDOMPoint& aPointToInsert);
+template Result<CreateTextResult, nsresult>
+EditorBase::InsertNodeWithTransaction(Text& aContentToInsert,
+ const EditorDOMPoint& aPointToInsert);
+
+template EditorDOMPoint EditorBase::GetFirstSelectionStartPoint() const;
+template EditorRawDOMPoint EditorBase::GetFirstSelectionStartPoint() const;
+template EditorDOMPoint EditorBase::GetFirstSelectionEndPoint() const;
+template EditorRawDOMPoint EditorBase::GetFirstSelectionEndPoint() const;
+
+template EditorDOMPoint EditorBase::FindBetterInsertionPoint(
+ const EditorDOMPoint& aPoint) const;
+template EditorRawDOMPoint EditorBase::FindBetterInsertionPoint(
+ const EditorRawDOMPoint& aPoint) const;
+
+template EditorBase::AutoCaretBidiLevelManager::AutoCaretBidiLevelManager(
+ const EditorBase& aEditorBase, nsIEditor::EDirection aDirectionAndAmount,
+ const EditorDOMPoint& aPointAtCaret);
+template EditorBase::AutoCaretBidiLevelManager::AutoCaretBidiLevelManager(
+ const EditorBase& aEditorBase, nsIEditor::EDirection aDirectionAndAmount,
+ const EditorRawDOMPoint& aPointAtCaret);
+
+EditorBase::EditorBase(EditorType aEditorType)
+ : mEditActionData(nullptr),
+ mPlaceholderName(nullptr),
+ mModCount(0),
+ mFlags(0),
+ mUpdateCount(0),
+ mPlaceholderBatch(0),
+ mWrapColumn(0),
+ mNewlineHandling(StaticPrefs::editor_singleLine_pasteNewlines()),
+ mCaretStyle(StaticPrefs::layout_selection_caret_style()),
+ mDocDirtyState(-1),
+ mSpellcheckCheckboxState(eTriUnset),
+ mInitSucceeded(false),
+ mAllowsTransactionsToChangeSelection(true),
+ mDidPreDestroy(false),
+ mDidPostCreate(false),
+ mDispatchInputEvent(true),
+ mIsInEditSubAction(false),
+ mHidingCaret(false),
+ mSpellCheckerDictionaryUpdated(true),
+ mIsHTMLEditorClass(aEditorType == EditorType::HTML) {
+#ifdef XP_WIN
+ if (!mCaretStyle && !IsTextEditor()) {
+ // Wordpad-like caret behavior.
+ mCaretStyle = 1;
+ }
+#endif // #ifdef XP_WIN
+ if (mNewlineHandling < nsIEditor::eNewlinesPasteIntact ||
+ mNewlineHandling > nsIEditor::eNewlinesStripSurroundingWhitespace) {
+ mNewlineHandling = nsIEditor::eNewlinesPasteToFirst;
+ }
+}
+
+EditorBase::~EditorBase() {
+ MOZ_ASSERT(!IsInitialized() || mDidPreDestroy,
+ "Why PreDestroy hasn't been called?");
+
+ if (mComposition) {
+ mComposition->OnEditorDestroyed();
+ mComposition = nullptr;
+ }
+ // If this editor is still hiding the caret, we need to restore it.
+ HideCaret(false);
+ mTransactionManager = nullptr;
+}
+
+NS_IMPL_CYCLE_COLLECTION_CLASS(EditorBase)
+
+NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(EditorBase)
+ // Remove event listeners first since EditorEventListener may need
+ // mDocument, mEventTarget, etc.
+ if (tmp->mEventListener) {
+ tmp->mEventListener->Disconnect();
+ tmp->mEventListener = nullptr;
+ }
+
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mRootElement)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mSelectionController)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mDocument)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mIMEContentObserver)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mInlineSpellChecker)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mTextServicesDocument)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mTextInputListener)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mTransactionManager)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mActionListeners)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mDocStateListeners)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mEventTarget)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mPlaceholderTransaction)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mCachedDocumentEncoder)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK_WEAK_REFERENCE
+NS_IMPL_CYCLE_COLLECTION_UNLINK_END
+
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(EditorBase)
+ Document* currentDoc =
+ tmp->mRootElement ? tmp->mRootElement->GetUncomposedDoc() : nullptr;
+ if (currentDoc && nsCCUncollectableMarker::InGeneration(
+ cb, currentDoc->GetMarkedCCGeneration())) {
+ return NS_SUCCESS_INTERRUPTED_TRAVERSE;
+ }
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRootElement)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mSelectionController)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mDocument)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mIMEContentObserver)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mInlineSpellChecker)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTextServicesDocument)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTextInputListener)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTransactionManager)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mActionListeners)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mDocStateListeners)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mEventTarget)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mEventListener)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mPlaceholderTransaction)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mCachedDocumentEncoder)
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
+
+NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(EditorBase)
+ NS_INTERFACE_MAP_ENTRY(nsISelectionListener)
+ NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference)
+ NS_INTERFACE_MAP_ENTRY(nsIEditor)
+ NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIEditor)
+NS_INTERFACE_MAP_END
+
+NS_IMPL_CYCLE_COLLECTING_ADDREF(EditorBase)
+NS_IMPL_CYCLE_COLLECTING_RELEASE(EditorBase)
+
+nsresult EditorBase::InitInternal(Document& aDocument, Element* aRootElement,
+ nsISelectionController& aSelectionController,
+ uint32_t aFlags) {
+ MOZ_ASSERT_IF(
+ !mEditActionData ||
+ !mEditActionData->HasEditorDestroyedDuringHandlingEditAction(),
+ GetTopLevelEditSubAction() == EditSubAction::eNone);
+
+ // First only set flags, but other stuff shouldn't be initialized now.
+ // Note that SetFlags() will be called by PostCreate().
+ mFlags = aFlags;
+
+ mDocument = &aDocument;
+ // nsISelectionController should be stored only when we're a `TextEditor`.
+ // Otherwise, in `HTMLEditor`, it's `PresShell`, and grabbing it causes
+ // a circular reference and memory leak.
+ // XXX Should we move `mSelectionController to `TextEditor`?
+ MOZ_ASSERT_IF(!IsTextEditor(), &aSelectionController == GetPresShell());
+ if (IsTextEditor()) {
+ MOZ_ASSERT(&aSelectionController != GetPresShell());
+ mSelectionController = &aSelectionController;
+ }
+
+ if (mEditActionData) {
+ // During edit action, selection is cached. But this selection is invalid
+ // now since selection controller is updated, so we have to update this
+ // cache.
+ Selection* selection = aSelectionController.GetSelection(
+ nsISelectionController::SELECTION_NORMAL);
+ NS_WARNING_ASSERTION(selection,
+ "SelectionController::GetSelection() failed");
+ if (selection) {
+ mEditActionData->UpdateSelectionCache(*selection);
+ }
+ }
+
+ // set up root element if we are passed one.
+ if (aRootElement) {
+ mRootElement = aRootElement;
+ }
+
+ // If this is an editor for <input> or <textarea>, the text node which
+ // has composition string is always recreated with same content. Therefore,
+ // we need to nodify mComposition of text node destruction and replacing
+ // composing string when this receives eCompositionChange event next time.
+ if (mComposition && mComposition->GetContainerTextNode() &&
+ !mComposition->GetContainerTextNode()->IsInComposedDoc()) {
+ mComposition->OnTextNodeRemoved();
+ }
+
+ // Show the caret.
+ DebugOnly<nsresult> rvIgnored = aSelectionController.SetCaretReadOnly(false);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsISelectionController::SetCaretReadOnly(false) failed, but ignored");
+ // Show all the selection reflected to user.
+ rvIgnored =
+ aSelectionController.SetSelectionFlags(nsISelectionDisplay::DISPLAY_ALL);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "nsISelectionController::SetSelectionFlags("
+ "nsISelectionDisplay::DISPLAY_ALL) failed, but ignored");
+
+ MOZ_ASSERT(IsInitialized());
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eInitializing);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ SelectionRef().AddSelectionListener(this);
+
+ // Make sure that the editor will be destroyed properly
+ mDidPreDestroy = false;
+ // Make sure that the editor will be created properly
+ mDidPostCreate = false;
+
+ return NS_OK;
+}
+
+nsresult EditorBase::EnsureEmptyTextFirstChild() {
+ MOZ_ASSERT(IsTextEditor());
+ RefPtr<Element> root = GetRoot();
+ nsIContent* firstChild = root->GetFirstChild();
+
+ if (!firstChild || !firstChild->IsText()) {
+ RefPtr<nsTextNode> newTextNode = CreateTextNode(u""_ns);
+ if (!newTextNode) {
+ NS_WARNING("EditorBase::CreateTextNode() failed");
+ return NS_ERROR_UNEXPECTED;
+ }
+ IgnoredErrorResult ignoredError;
+ root->InsertChildBefore(newTextNode, root->GetFirstChild(), true,
+ ignoredError);
+ MOZ_ASSERT(!ignoredError.Failed());
+ }
+
+ return NS_OK;
+}
+
+nsresult EditorBase::PostCreateInternal() {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ // Synchronize some stuff for the flags. SetFlags() will initialize
+ // something by the flag difference. This is first time of that, so, all
+ // initializations must be run. For such reason, we need to invert mFlags
+ // value first.
+ mFlags = ~mFlags;
+ nsresult rv = SetFlags(~mFlags);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::SetFlags() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ // These operations only need to happen on the first PostCreate call
+ if (!mDidPostCreate) {
+ mDidPostCreate = true;
+
+ // Set up listeners
+ CreateEventListeners();
+ nsresult rv = InstallEventListeners();
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::InstallEventListeners() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ // nuke the modification count, so the doc appears unmodified
+ // do this before we notify listeners
+ DebugOnly<nsresult> rvIgnored = ResetModificationCount();
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::ResetModificationCount() failed, but ignored");
+
+ // update the UI with our state
+ rvIgnored = NotifyDocumentListeners(eDocumentCreated);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "EditorBase::NotifyDocumentListeners(eDocumentCreated)"
+ " failed, but ignored");
+ rvIgnored = NotifyDocumentListeners(eDocumentStateChanged);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "EditorBase::NotifyDocumentListeners("
+ "eDocumentStateChanged) failed, but ignored");
+ }
+
+ // update nsTextStateManager and caret if we have focus
+ if (RefPtr<Element> focusedElement = GetFocusedElement()) {
+ DebugOnly<nsresult> rvIgnored = InitializeSelection(*focusedElement);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::InitializeSelection() failed, but ignored");
+
+ // If the text control gets reframed during focus, Focus() would not be
+ // called, so take a chance here to see if we need to spell check the text
+ // control.
+ nsresult rv = FlushPendingSpellCheck();
+ if (MOZ_UNLIKELY(rv == NS_ERROR_EDITOR_DESTROYED)) {
+ NS_WARNING(
+ "EditorBase::FlushPendingSpellCheck() caused destroying the editor");
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_DESTROYED);
+ }
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::FlushPendingSpellCheck() failed, but ignored");
+
+ IMEState newState;
+ rv = GetPreferredIMEState(&newState);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::GetPreferredIMEState() failed");
+ return NS_OK;
+ }
+ IMEStateManager::UpdateIMEState(newState, focusedElement, *this);
+ }
+
+ // FYI: This call might cause destroying this editor.
+ IMEStateManager::OnEditorInitialized(*this);
+
+ return NS_OK;
+}
+
+void EditorBase::SetTextInputListener(TextInputListener* aTextInputListener) {
+ MOZ_ASSERT(!mTextInputListener || !aTextInputListener ||
+ mTextInputListener == aTextInputListener);
+ mTextInputListener = aTextInputListener;
+}
+
+void EditorBase::SetIMEContentObserver(
+ IMEContentObserver* aIMEContentObserver) {
+ MOZ_ASSERT(!mIMEContentObserver || !aIMEContentObserver ||
+ mIMEContentObserver == aIMEContentObserver);
+ mIMEContentObserver = aIMEContentObserver;
+}
+
+void EditorBase::CreateEventListeners() {
+ // Don't create the handler twice
+ if (!mEventListener) {
+ mEventListener = new EditorEventListener();
+ }
+}
+
+nsresult EditorBase::InstallEventListeners() {
+ if (NS_WARN_IF(!IsInitialized()) || NS_WARN_IF(!mEventListener)) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ // Initialize the event target.
+ mEventTarget = GetExposedRoot();
+ if (NS_WARN_IF(!mEventTarget)) {
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+
+ nsresult rv = mEventListener->Connect(this);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorEventListener::Connect() failed");
+ if (mComposition) {
+ // If mComposition has already been destroyed, we should forget it.
+ // This may happen if it ended while we don't listen to composition
+ // events.
+ if (mComposition->Destroyed()) {
+ // XXX We may need to fix existing composition transaction here.
+ // However, this may be called when it's not safe.
+ // Perhaps, we should stop handling composition with events.
+ mComposition = nullptr;
+ }
+ // Otherwise, Restart to handle composition with new editor contents.
+ else {
+ mComposition->StartHandlingComposition(this);
+ }
+ }
+ return rv;
+}
+
+void EditorBase::RemoveEventListeners() {
+ if (!IsInitialized() || !mEventListener) {
+ return;
+ }
+ mEventListener->Disconnect();
+ if (mComposition) {
+ // Even if this is called, don't release mComposition because this is
+ // may be reused after reframing.
+ mComposition->EndHandlingComposition(this);
+ }
+ mEventTarget = nullptr;
+}
+
+bool EditorBase::IsListeningToEvents() const {
+ return IsInitialized() && mEventListener &&
+ !mEventListener->DetachedFromEditor();
+}
+
+bool EditorBase::GetDesiredSpellCheckState() {
+ // Check user override on this element
+ if (mSpellcheckCheckboxState != eTriUnset) {
+ return (mSpellcheckCheckboxState == eTriTrue);
+ }
+
+ // Check user preferences
+ int32_t spellcheckLevel = Preferences::GetInt("layout.spellcheckDefault", 1);
+
+ if (!spellcheckLevel) {
+ return false; // Spellchecking forced off globally
+ }
+
+ if (!CanEnableSpellCheck()) {
+ return false;
+ }
+
+ PresShell* presShell = GetPresShell();
+ if (presShell) {
+ nsPresContext* context = presShell->GetPresContext();
+ if (context && !context->IsDynamic()) {
+ return false;
+ }
+ }
+
+ // Check DOM state
+ nsCOMPtr<nsIContent> content = GetExposedRoot();
+ if (!content) {
+ return false;
+ }
+
+ auto element = nsGenericHTMLElement::FromNode(content);
+ if (!element) {
+ return false;
+ }
+
+ if (!IsInPlaintextMode()) {
+ // Some of the page content might be editable and some not, if spellcheck=
+ // is explicitly set anywhere, so if there's anything editable on the page,
+ // return true and let the spellchecker figure it out.
+ Document* doc = content->GetComposedDoc();
+ return doc && doc->IsEditingOn();
+ }
+
+ return element->Spellcheck();
+}
+
+void EditorBase::PreDestroyInternal() {
+ MOZ_ASSERT(!mDidPreDestroy);
+
+ mInitSucceeded = false;
+
+ Selection* selection = GetSelection();
+ if (selection) {
+ selection->RemoveSelectionListener(this);
+ }
+
+ IMEStateManager::OnEditorDestroying(*this);
+
+ // Let spellchecker clean up its observers etc. It is important not to
+ // actually free the spellchecker here, since the spellchecker could have
+ // caused flush notifications, which could have gotten here if a textbox
+ // is being removed. Setting the spellchecker to nullptr could free the
+ // object that is still in use! It will be freed when the editor is
+ // destroyed.
+ if (mInlineSpellChecker) {
+ DebugOnly<nsresult> rvIgnored =
+ mInlineSpellChecker->Cleanup(IsTextEditor());
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "mozInlineSpellChecker::Cleanup() failed, but ignored");
+ }
+
+ // tell our listeners that the doc is going away
+ DebugOnly<nsresult> rvIgnored =
+ NotifyDocumentListeners(eDocumentToBeDestroyed);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "EditorBase::NotifyDocumentListeners("
+ "eDocumentToBeDestroyed) failed, but ignored");
+
+ // Unregister event listeners
+ RemoveEventListeners();
+ // If this editor is still hiding the caret, we need to restore it.
+ HideCaret(false);
+ mActionListeners.Clear();
+ mDocStateListeners.Clear();
+ mInlineSpellChecker = nullptr;
+ mTextServicesDocument = nullptr;
+ mTextInputListener = nullptr;
+ mSpellcheckCheckboxState = eTriUnset;
+ mRootElement = nullptr;
+
+ // Transaction may grab this instance. Therefore, they should be released
+ // here for stopping the circular reference with this instance.
+ if (mTransactionManager) {
+ DebugOnly<bool> disabledUndoRedo = DisableUndoRedo();
+ NS_WARNING_ASSERTION(disabledUndoRedo,
+ "EditorBase::DisableUndoRedo() failed, but ignored");
+ mTransactionManager = nullptr;
+ }
+
+ if (mEditActionData) {
+ mEditActionData->OnEditorDestroy();
+ }
+
+ mDidPreDestroy = true;
+}
+
+NS_IMETHODIMP EditorBase::GetFlags(uint32_t* aFlags) {
+ // NOTE: If you need to override this method, you need to make Flags()
+ // virtual.
+ *aFlags = Flags();
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::SetFlags(uint32_t aFlags) {
+ if (mFlags == aFlags) {
+ return NS_OK;
+ }
+
+ // If we're a `TextEditor` instance, the plaintext mode should always be set.
+ // If we're an `HTMLEditor` instance, either is fine.
+ MOZ_ASSERT_IF(IsTextEditor(), !!(aFlags & nsIEditor::eEditorPlaintextMask));
+ // If we're an `HTMLEditor` instance, we cannot treat it as a single line
+ // editor. So, eEditorSingleLineMask is available only when we're a
+ // `TextEditor` instance.
+ MOZ_ASSERT_IF(IsHTMLEditor(), !(aFlags & nsIEditor::eEditorSingleLineMask));
+ // If we're an `HTMLEditor` instance, we cannot treat it as a password editor.
+ // So, eEditorPasswordMask is available only when we're a `TextEditor`
+ // instance.
+ MOZ_ASSERT_IF(IsHTMLEditor(), !(aFlags & nsIEditor::eEditorPasswordMask));
+ // eEditorAllowInteraction changes the behavior of `HTMLEditor`. So, it's
+ // not available with `TextEditor` instance.
+ MOZ_ASSERT_IF(IsTextEditor(), !(aFlags & nsIEditor::eEditorAllowInteraction));
+
+ const bool isCalledByPostCreate = (mFlags == ~aFlags);
+ // We don't support dynamic password flag change.
+ MOZ_ASSERT_IF(!isCalledByPostCreate,
+ !((mFlags ^ aFlags) & nsIEditor::eEditorPasswordMask));
+ bool spellcheckerWasEnabled = !isCalledByPostCreate && CanEnableSpellCheck();
+ mFlags = aFlags;
+
+ if (!IsInitialized()) {
+ // If we're initializing, we shouldn't do anything now.
+ // SetFlags() will be called by PostCreate(),
+ // we should synchronize some stuff for the flags at that time.
+ return NS_OK;
+ }
+
+ // The flag change may cause the spellchecker state change
+ if (CanEnableSpellCheck() != spellcheckerWasEnabled) {
+ SyncRealTimeSpell();
+ }
+
+ // If this is called from PostCreate(), it will update the IME state if it's
+ // necessary.
+ if (!mDidPostCreate) {
+ return NS_OK;
+ }
+
+ // Might be changing editable state, so, we need to reset current IME state
+ // if we're focused and the flag change causes IME state change.
+ if (RefPtr<Element> focusedElement = GetFocusedElement()) {
+ IMEState newState;
+ nsresult rv = GetPreferredIMEState(&newState);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::GetPreferredIMEState() failed, but ignored");
+ if (NS_SUCCEEDED(rv)) {
+ // NOTE: When the enabled state isn't going to be modified, this method
+ // is going to do nothing.
+ IMEStateManager::UpdateIMEState(newState, focusedElement, *this);
+ }
+ }
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetIsSelectionEditable(bool* aIsSelectionEditable) {
+ if (NS_WARN_IF(!aIsSelectionEditable)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aIsSelectionEditable = IsSelectionEditable();
+ return NS_OK;
+}
+
+bool EditorBase::IsSelectionEditable() {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return false;
+ }
+
+ if (IsTextEditor()) {
+ // XXX we just check that the anchor node is editable at the moment
+ // we should check that all nodes in the selection are editable
+ const nsINode* anchorNode = SelectionRef().GetAnchorNode();
+ return anchorNode && anchorNode->IsContent() && anchorNode->IsEditable();
+ }
+
+ const nsINode* anchorNode = SelectionRef().GetAnchorNode();
+ const nsINode* focusNode = SelectionRef().GetFocusNode();
+ if (!anchorNode || !focusNode) {
+ return false;
+ }
+
+ // if anchorNode or focusNode is in a native anonymous subtree, HTMLEditor
+ // shouldn't edit content in it.
+ // XXX This must be a bug of Selection API.
+ if (MOZ_UNLIKELY(anchorNode->IsInNativeAnonymousSubtree() ||
+ focusNode->IsInNativeAnonymousSubtree())) {
+ return false;
+ }
+
+ // Per the editing spec as of June 2012: we have to have a selection whose
+ // start and end nodes are editable, and which share an ancestor editing
+ // host. (Bug 766387.)
+ bool isSelectionEditable = SelectionRef().RangeCount() &&
+ anchorNode->IsEditable() &&
+ focusNode->IsEditable();
+ if (!isSelectionEditable) {
+ return false;
+ }
+
+ const nsINode* commonAncestor =
+ SelectionRef().GetAnchorFocusRange()->GetClosestCommonInclusiveAncestor();
+ while (commonAncestor && !commonAncestor->IsEditable()) {
+ commonAncestor = commonAncestor->GetParentNode();
+ }
+ // If there is no editable common ancestor, return false.
+ return !!commonAncestor;
+}
+
+NS_IMETHODIMP EditorBase::GetIsDocumentEditable(bool* aIsDocumentEditable) {
+ if (NS_WARN_IF(!aIsDocumentEditable)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ RefPtr<Document> document = GetDocument();
+ *aIsDocumentEditable = document && IsModifiable();
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetDocument(Document** aDocument) {
+ if (NS_WARN_IF(!aDocument)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aDocument = do_AddRef(mDocument).take();
+ return NS_WARN_IF(!*aDocument) ? NS_ERROR_NOT_INITIALIZED : NS_OK;
+}
+
+already_AddRefed<nsIWidget> EditorBase::GetWidget() const {
+ nsPresContext* presContext = GetPresContext();
+ if (NS_WARN_IF(!presContext)) {
+ return nullptr;
+ }
+ nsCOMPtr<nsIWidget> widget = presContext->GetRootWidget();
+ return NS_WARN_IF(!widget) ? nullptr : widget.forget();
+}
+
+NS_IMETHODIMP EditorBase::GetContentsMIMEType(nsAString& aContentsMIMEType) {
+ aContentsMIMEType = mContentMIMEType;
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::SetContentsMIMEType(
+ const nsAString& aContentsMIMEType) {
+ mContentMIMEType.Assign(aContentsMIMEType);
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetSelectionController(
+ nsISelectionController** aSelectionController) {
+ if (NS_WARN_IF(!aSelectionController)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aSelectionController = do_AddRef(GetSelectionController()).take();
+ return NS_WARN_IF(!*aSelectionController) ? NS_ERROR_FAILURE : NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::DeleteSelection(EDirection aAction,
+ EStripWrappers aStripWrappers) {
+ nsresult rv = DeleteSelectionAsAction(aAction, aStripWrappers);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DeleteSelectionAsAction() failed");
+ return rv;
+}
+
+NS_IMETHODIMP EditorBase::GetSelection(Selection** aSelection) {
+ nsresult rv = GetSelection(SelectionType::eNormal, aSelection);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::GetSelection(SelectionType::eNormal) failed");
+ return rv;
+}
+
+nsresult EditorBase::GetSelection(SelectionType aSelectionType,
+ Selection** aSelection) const {
+ if (NS_WARN_IF(!aSelection)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ if (IsEditActionDataAvailable()) {
+ *aSelection = do_AddRef(&SelectionRef()).take();
+ return NS_OK;
+ }
+ nsISelectionController* selectionController = GetSelectionController();
+ if (NS_WARN_IF(!selectionController)) {
+ *aSelection = nullptr;
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+ *aSelection = do_AddRef(selectionController->GetSelection(
+ ToRawSelectionType(aSelectionType)))
+ .take();
+ return NS_WARN_IF(!*aSelection) ? NS_ERROR_FAILURE : NS_OK;
+}
+
+nsresult EditorBase::DoTransactionInternal(nsITransaction* aTransaction) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(!ShouldAlreadyHaveHandledBeforeInputEventDispatching(),
+ "beforeinput event hasn't been dispatched yet");
+
+ if (mPlaceholderBatch && !mPlaceholderTransaction) {
+ MOZ_DIAGNOSTIC_ASSERT(mPlaceholderName);
+ mPlaceholderTransaction = PlaceholderTransaction::Create(
+ *this, *mPlaceholderName, std::move(mSelState));
+ MOZ_ASSERT(mSelState.isNothing());
+
+ // We will recurse, but will not hit this case in the nested call
+ RefPtr<PlaceholderTransaction> placeholderTransaction =
+ mPlaceholderTransaction;
+ DebugOnly<nsresult> rvIgnored =
+ DoTransactionInternal(placeholderTransaction);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::DoTransactionInternal() failed, but ignored");
+
+ if (mTransactionManager) {
+ if (nsCOMPtr<nsITransaction> topTransaction =
+ mTransactionManager->PeekUndoStack()) {
+ if (RefPtr<EditTransactionBase> topTransactionBase =
+ topTransaction->GetAsEditTransactionBase()) {
+ if (PlaceholderTransaction* topPlaceholderTransaction =
+ topTransactionBase->GetAsPlaceholderTransaction()) {
+ // there is a placeholder transaction on top of the undo stack. It
+ // is either the one we just created, or an earlier one that we are
+ // now merging into. From here on out remember this placeholder
+ // instead of the one we just created.
+ mPlaceholderTransaction = topPlaceholderTransaction;
+ }
+ }
+ }
+ }
+ }
+
+ if (aTransaction) {
+ // XXX: Why are we doing selection specific batching stuff here?
+ // XXX: Most entry points into the editor have auto variables that
+ // XXX: should trigger Begin/EndUpdateViewBatch() calls that will make
+ // XXX: these selection batch calls no-ops.
+ // XXX:
+ // XXX: I suspect that this was placed here to avoid multiple
+ // XXX: selection changed notifications from happening until after
+ // XXX: the transaction was done. I suppose that can still happen
+ // XXX: if an embedding application called DoTransaction() directly
+ // XXX: to pump its own transactions through the system, but in that
+ // XXX: case, wouldn't we want to use Begin/EndUpdateViewBatch() or
+ // XXX: its auto equivalent AutoUpdateViewBatch to ensure that
+ // XXX: selection listeners have access to accurate frame data?
+ // XXX:
+ // XXX: Note that if we did add Begin/EndUpdateViewBatch() calls
+ // XXX: we will need to make sure that they are disabled during
+ // XXX: the init of the editor for text widgets to avoid layout
+ // XXX: re-entry during initial reflow. - kin
+
+ // get the selection and start a batch change
+ SelectionBatcher selectionBatcher(SelectionRef(), __FUNCTION__);
+
+ if (mTransactionManager) {
+ RefPtr<TransactionManager> transactionManager(mTransactionManager);
+ nsresult rv = transactionManager->DoTransaction(aTransaction);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("TransactionManager::DoTransaction() failed");
+ return rv;
+ }
+ } else {
+ nsresult rv = aTransaction->DoTransaction();
+ if (NS_FAILED(rv)) {
+ NS_WARNING("nsITransaction::DoTransaction() failed");
+ return rv;
+ }
+ }
+
+ DoAfterDoTransaction(aTransaction);
+ }
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::EnableUndo(bool aEnable) {
+ // XXX Should we return NS_ERROR_FAILURE if EdnableUndoRedo() or
+ // DisableUndoRedo() returns false?
+ if (aEnable) {
+ DebugOnly<bool> enabledUndoRedo = EnableUndoRedo();
+ NS_WARNING_ASSERTION(enabledUndoRedo,
+ "EditorBase::EnableUndoRedo() failed, but ignored");
+ return NS_OK;
+ }
+ DebugOnly<bool> disabledUndoRedo = DisableUndoRedo();
+ NS_WARNING_ASSERTION(disabledUndoRedo,
+ "EditorBase::DisableUndoRedo() failed, but ignored");
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::ClearUndoRedoXPCOM() {
+ if (MOZ_UNLIKELY(!ClearUndoRedo())) {
+ return NS_ERROR_FAILURE; // We're handling a transaction
+ }
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::Undo() {
+ nsresult rv = UndoAsAction(1u);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "EditorBase::UndoAsAction() failed");
+ return rv;
+}
+
+NS_IMETHODIMP EditorBase::UndoAll() {
+ if (!mTransactionManager) {
+ return NS_OK;
+ }
+ size_t numberOfUndoItems = mTransactionManager->NumberOfUndoItems();
+ if (!numberOfUndoItems) {
+ return NS_OK; // no transactions
+ }
+ nsresult rv = UndoAsAction(numberOfUndoItems);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "EditorBase::UndoAsAction() failed");
+ return rv;
+}
+
+NS_IMETHODIMP EditorBase::GetUndoRedoEnabled(bool* aIsEnabled) {
+ MOZ_ASSERT(aIsEnabled);
+ *aIsEnabled = IsUndoRedoEnabled();
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetCanUndo(bool* aCanUndo) {
+ MOZ_ASSERT(aCanUndo);
+ *aCanUndo = CanUndo();
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::Redo() {
+ nsresult rv = RedoAsAction(1u);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "EditorBase::RedoAsAction() failed");
+ return rv;
+}
+
+NS_IMETHODIMP EditorBase::GetCanRedo(bool* aCanRedo) {
+ MOZ_ASSERT(aCanRedo);
+ *aCanRedo = CanRedo();
+ return NS_OK;
+}
+
+nsresult EditorBase::UndoAsAction(uint32_t aCount, nsIPrincipal* aPrincipal) {
+ if (aCount == 0 || IsReadonly()) {
+ return NS_OK;
+ }
+
+ // If we don't have transaction in the undo stack, we shouldn't notify
+ // anybody of trying to undo since it's not useful notification but we
+ // need to pay some runtime cost.
+ if (!CanUndo()) {
+ return NS_OK;
+ }
+
+ // If there is composition, we shouldn't allow to undo with committing
+ // composition since Chrome doesn't allow it and it doesn't make sense
+ // because committing composition causes one transaction and Undo(1)
+ // undoes the committing composition.
+ if (GetComposition()) {
+ return NS_OK;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eUndo, aPrincipal);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ AutoUpdateViewBatch preventSelectionChangeEvent(*this, __FUNCTION__);
+
+ NotifyEditorObservers(eNotifyEditorObserversOfBefore);
+ if (NS_WARN_IF(!CanUndo()) || NS_WARN_IF(Destroyed())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ rv = NS_OK;
+ {
+ IgnoredErrorResult ignoredError;
+ AutoEditSubActionNotifier startToHandleEditSubAction(
+ *this, EditSubAction::eUndo, nsIEditor::eNone, ignoredError);
+ if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ return EditorBase::ToGenericNSResult(ignoredError.StealNSResult());
+ }
+ NS_WARNING_ASSERTION(!ignoredError.Failed(),
+ "TextEditor::OnStartToHandleTopLevelEditSubAction() "
+ "failed, but ignored");
+
+ RefPtr<TransactionManager> transactionManager(mTransactionManager);
+ for (uint32_t i = 0; i < aCount; ++i) {
+ if (NS_FAILED(transactionManager->Undo())) {
+ NS_WARNING("TransactionManager::Undo() failed");
+ break;
+ }
+ DoAfterUndoTransaction();
+ }
+
+ if (IsHTMLEditor()) {
+ rv = AsHTMLEditor()->ReflectPaddingBRElementForEmptyEditor();
+ }
+ }
+
+ NotifyEditorObservers(eNotifyEditorObserversOfEnd);
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::RedoAsAction(uint32_t aCount, nsIPrincipal* aPrincipal) {
+ if (aCount == 0 || IsReadonly()) {
+ return NS_OK;
+ }
+
+ // If we don't have transaction in the redo stack, we shouldn't notify
+ // anybody of trying to redo since it's not useful notification but we
+ // need to pay some runtime cost.
+ if (!CanRedo()) {
+ return NS_OK;
+ }
+
+ // If there is composition, we shouldn't allow to redo with committing
+ // composition since Chrome doesn't allow it and it doesn't make sense
+ // because committing composition causes removing all transactions from
+ // the redo queue. So, it becomes impossible to redo anything.
+ if (GetComposition()) {
+ return NS_OK;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eRedo, aPrincipal);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ AutoUpdateViewBatch preventSelectionChangeEvent(*this, __FUNCTION__);
+
+ NotifyEditorObservers(eNotifyEditorObserversOfBefore);
+ if (NS_WARN_IF(!CanRedo()) || NS_WARN_IF(Destroyed())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ rv = NS_OK;
+ {
+ IgnoredErrorResult ignoredError;
+ AutoEditSubActionNotifier startToHandleEditSubAction(
+ *this, EditSubAction::eRedo, nsIEditor::eNone, ignoredError);
+ if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ return ignoredError.StealNSResult();
+ }
+ NS_WARNING_ASSERTION(!ignoredError.Failed(),
+ "TextEditor::OnStartToHandleTopLevelEditSubAction() "
+ "failed, but ignored");
+
+ RefPtr<TransactionManager> transactionManager(mTransactionManager);
+ for (uint32_t i = 0; i < aCount; ++i) {
+ if (NS_FAILED(transactionManager->Redo())) {
+ NS_WARNING("TransactionManager::Redo() failed");
+ break;
+ }
+ DoAfterRedoTransaction();
+ }
+
+ if (IsHTMLEditor()) {
+ rv = AsHTMLEditor()->ReflectPaddingBRElementForEmptyEditor();
+ }
+ }
+
+ NotifyEditorObservers(eNotifyEditorObserversOfEnd);
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+NS_IMETHODIMP EditorBase::BeginTransaction() {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eUnknown);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ BeginTransactionInternal(__FUNCTION__);
+ return NS_OK;
+}
+
+void EditorBase::BeginTransactionInternal(const char* aRequesterFuncName) {
+ BeginUpdateViewBatch(aRequesterFuncName);
+
+ if (NS_WARN_IF(!mTransactionManager)) {
+ return;
+ }
+
+ RefPtr<TransactionManager> transactionManager(mTransactionManager);
+ DebugOnly<nsresult> rvIgnored = transactionManager->BeginBatch(nullptr);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "TransactionManager::BeginBatch() failed, but ignored");
+}
+
+NS_IMETHODIMP EditorBase::EndTransaction() {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eUnknown);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ EndTransactionInternal(__FUNCTION__);
+ return NS_OK;
+}
+
+void EditorBase::EndTransactionInternal(const char* aRequesterFuncName) {
+ if (mTransactionManager) {
+ RefPtr<TransactionManager> transactionManager(mTransactionManager);
+ DebugOnly<nsresult> rvIgnored = transactionManager->EndBatch(false);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "TransactionManager::EndBatch() failed, but ignored");
+ }
+
+ EndUpdateViewBatch(aRequesterFuncName);
+}
+
+void EditorBase::BeginPlaceholderTransaction(nsStaticAtom& aTransactionName,
+ const char* aRequesterFuncName) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(mPlaceholderBatch >= 0, "negative placeholder batch count!");
+
+ if (!mPlaceholderBatch) {
+ NotifyEditorObservers(eNotifyEditorObserversOfBefore);
+ // time to turn on the batch
+ BeginUpdateViewBatch(aRequesterFuncName);
+ mPlaceholderTransaction = nullptr;
+ mPlaceholderName = &aTransactionName;
+ mSelState.emplace();
+ mSelState->SaveSelection(SelectionRef());
+ // Composition transaction can modify multiple nodes and it merges text
+ // node for ime into single text node.
+ // So if current selection is into IME text node, it might be failed
+ // to restore selection by UndoTransaction.
+ // So we need update selection by range updater.
+ if (mPlaceholderName == nsGkAtoms::IMETxnName) {
+ RangeUpdaterRef().RegisterSelectionState(*mSelState);
+ }
+ }
+ mPlaceholderBatch++;
+}
+
+void EditorBase::EndPlaceholderTransaction(
+ ScrollSelectionIntoView aScrollSelectionIntoView,
+ const char* aRequesterFuncName) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(mPlaceholderBatch > 0,
+ "zero or negative placeholder batch count when ending batch!");
+
+ if (!(--mPlaceholderBatch)) {
+ // By making the assumption that no reflow happens during the calls
+ // to EndUpdateViewBatch and ScrollSelectionFocusIntoView, we are able to
+ // allow the selection to cache a frame offset which is used by the
+ // caret drawing code. We only enable this cache here; at other times,
+ // we have no way to know whether reflow invalidates it
+ // See bugs 35296 and 199412.
+ SelectionRef().SetCanCacheFrameOffset(true);
+
+ // time to turn off the batch
+ EndUpdateViewBatch(aRequesterFuncName);
+ // make sure selection is in view
+
+ // After ScrollSelectionFocusIntoView(), the pending notifications might be
+ // flushed and PresShell/PresContext/Frames may be dead. See bug 418470.
+ // XXX Even if we're destroyed, we need to keep handling below because
+ // this method changes a lot of status. We should rewrite this safer.
+ if (aScrollSelectionIntoView == ScrollSelectionIntoView::Yes) {
+ DebugOnly<nsresult> rvIgnored = ScrollSelectionFocusIntoView();
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::ScrollSelectionFocusIntoView() failed, but Ignored");
+ }
+
+ // cached for frame offset are Not available now
+ SelectionRef().SetCanCacheFrameOffset(false);
+
+ if (mSelState) {
+ // we saved the selection state, but never got to hand it to placeholder
+ // (else we ould have nulled out this pointer), so destroy it to prevent
+ // leaks.
+ if (mPlaceholderName == nsGkAtoms::IMETxnName) {
+ RangeUpdaterRef().DropSelectionState(*mSelState);
+ }
+ mSelState.reset();
+ }
+ // We might have never made a placeholder if no action took place.
+ if (mPlaceholderTransaction) {
+ // FYI: Disconnect placeholder transaction before dispatching "input"
+ // event because an input event listener may start other things.
+ // TODO: We should forget EditActionDataSetter too.
+ RefPtr<PlaceholderTransaction> placeholderTransaction =
+ std::move(mPlaceholderTransaction);
+ DebugOnly<nsresult> rvIgnored =
+ placeholderTransaction->EndPlaceHolderBatch();
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "PlaceholderTransaction::EndPlaceHolderBatch() failed, but ignored");
+ // notify editor observers of action but if composing, it's done by
+ // compositionchange event handler.
+ if (!mComposition) {
+ NotifyEditorObservers(eNotifyEditorObserversOfEnd);
+ }
+ } else {
+ NotifyEditorObservers(eNotifyEditorObserversOfCancel);
+ }
+ }
+}
+
+NS_IMETHODIMP EditorBase::SetShouldTxnSetSelection(bool aShould) {
+ MakeThisAllowTransactionsToChangeSelection(aShould);
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetDocumentIsEmpty(bool* aDocumentIsEmpty) {
+ MOZ_ASSERT(aDocumentIsEmpty);
+ *aDocumentIsEmpty = IsEmpty();
+ return NS_OK;
+}
+
+// XXX: The rule system should tell us which node to select all on (ie, the
+// root, or the body)
+NS_IMETHODIMP EditorBase::SelectAll() {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ nsresult rv = SelectAllInternal();
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "SelectAllInternal() failed");
+ // This is low level API for XUL applcation. So, we should return raw
+ // error code here.
+ return rv;
+}
+
+nsresult EditorBase::SelectAllInternal() {
+ MOZ_ASSERT(IsInitialized());
+
+ DebugOnly<nsresult> rvIgnored = CommitComposition();
+ if (NS_WARN_IF(Destroyed())) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "EditorBase::CommitComposition() failed, but ignored");
+
+ // XXX Do we need to keep handling after committing composition causes moving
+ // focus to different element? Although TextEditor has independent
+ // selection, so, we may not see any odd behavior even in such case.
+
+ nsresult rv = SelectEntireDocument();
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::SelectEntireDocument() failed");
+ return rv;
+}
+
+MOZ_CAN_RUN_SCRIPT_BOUNDARY NS_IMETHODIMP EditorBase::BeginningOfDocument() {
+ MOZ_ASSERT(IsTextEditor());
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ // get the root element
+ RefPtr<Element> rootElement = GetRoot();
+ if (NS_WARN_IF(!rootElement)) {
+ return NS_ERROR_NULL_POINTER;
+ }
+
+ // find first editable thingy
+ nsCOMPtr<nsIContent> firstEditableLeaf;
+ // If we're `TextEditor`, the first editable leaf node is a text node or
+ // padding `<br>` element. In the first case, we need to collapse selection
+ // into it.
+ if (rootElement->GetFirstChild() && rootElement->GetFirstChild()->IsText()) {
+ firstEditableLeaf = rootElement->GetFirstChild();
+ }
+ if (!firstEditableLeaf) {
+ // just the root node, set selection to inside the root
+ nsresult rv = CollapseSelectionToStartOf(*rootElement);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::CollapseSelectionToStartOf() failed");
+ return rv;
+ }
+
+ if (firstEditableLeaf->IsText()) {
+ // If firstEditableLeaf is text, set selection to beginning of the text
+ // node.
+ nsresult rv = CollapseSelectionToStartOf(*firstEditableLeaf);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::CollapseSelectionToStartOf() failed");
+ return rv;
+ }
+
+ // Otherwise, it's a leaf node and we set the selection just in front of it.
+ nsCOMPtr<nsIContent> parent = firstEditableLeaf->GetParent();
+ if (NS_WARN_IF(!parent)) {
+ return NS_ERROR_NULL_POINTER;
+ }
+
+ MOZ_ASSERT(
+ parent->ComputeIndexOf(firstEditableLeaf).valueOr(UINT32_MAX) == 0,
+ "How come the first node isn't the left most child in its parent?");
+ nsresult rv = CollapseSelectionToStartOf(*parent);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::CollapseSelectionToStartOf() failed");
+ return rv;
+}
+
+NS_IMETHODIMP EditorBase::EndOfDocument() { return NS_ERROR_NOT_IMPLEMENTED; }
+
+NS_IMETHODIMP EditorBase::GetDocumentModified(bool* aOutDocModified) {
+ if (NS_WARN_IF(!aOutDocModified)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ int32_t modCount = 0;
+ DebugOnly<nsresult> rvIgnored = GetModificationCount(&modCount);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::GetModificationCount() failed, but ignored");
+
+ *aOutDocModified = (modCount != 0);
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetDocumentCharacterSet(nsACString& aCharacterSet) {
+ return NS_ERROR_NOT_AVAILABLE;
+}
+
+nsresult EditorBase::GetDocumentCharsetInternal(nsACString& aCharset) const {
+ Document* document = GetDocument();
+ if (NS_WARN_IF(!document)) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+ document->GetDocumentCharacterSet()->Name(aCharset);
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::SetDocumentCharacterSet(
+ const nsACString& aCharacterSet) {
+ return NS_ERROR_NOT_AVAILABLE;
+}
+
+NS_IMETHODIMP EditorBase::OutputToString(const nsAString& aFormatType,
+ uint32_t aDocumentEncoderFlags,
+ nsAString& aOutputString) {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ nsresult rv =
+ ComputeValueInternal(aFormatType, aDocumentEncoderFlags, aOutputString);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::ComputeValueInternal() failed");
+ // This is low level API for XUL application. So, we should return raw
+ // error code here.
+ return rv;
+}
+
+nsresult EditorBase::ComputeValueInternal(const nsAString& aFormatType,
+ uint32_t aDocumentEncoderFlags,
+ nsAString& aOutputString) const {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ // First, let's try to get the value simply only from text node if the
+ // caller wants plaintext value.
+ if (aFormatType.LowerCaseEqualsLiteral("text/plain") &&
+ !(aDocumentEncoderFlags & (nsIDocumentEncoder::OutputSelectionOnly |
+ nsIDocumentEncoder::OutputWrap))) {
+ // Shortcut for empty editor case.
+ if (IsEmpty()) {
+ aOutputString.Truncate();
+ return NS_OK;
+ }
+ // NOTE: If it's neither <input type="text"> nor <textarea>, e.g., an HTML
+ // editor which is in plaintext mode (e.g., plaintext email composer on
+ // Thunderbird), it should be handled by the expensive path.
+ if (IsTextEditor()) {
+ // If it's necessary to check selection range or the editor wraps hard,
+ // we need some complicated handling. In such case, we need to use the
+ // expensive path.
+ // XXX Anything else what we cannot return the text node data simply?
+ Result<EditActionResult, nsresult> result =
+ AsTextEditor()->ComputeValueFromTextNodeAndBRElement(aOutputString);
+ if (MOZ_UNLIKELY(result.isErr())) {
+ NS_WARNING("TextEditor::ComputeValueFromTextNodeAndBRElement() failed");
+ return result.unwrapErr();
+ }
+ if (!result.inspect().Ignored()) {
+ return NS_OK;
+ }
+ }
+ }
+
+ nsAutoCString charset;
+ nsresult rv = GetDocumentCharsetInternal(charset);
+ if (NS_FAILED(rv) || charset.IsEmpty()) {
+ charset.AssignLiteral("windows-1252"); // XXX Why don't we use "UTF-8"?
+ }
+
+ nsCOMPtr<nsIDocumentEncoder> encoder =
+ GetAndInitDocEncoder(aFormatType, aDocumentEncoderFlags, charset);
+ if (!encoder) {
+ NS_WARNING("EditorBase::GetAndInitDocEncoder() failed");
+ return NS_ERROR_FAILURE;
+ }
+
+ rv = encoder->EncodeToString(aOutputString);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "nsIDocumentEncoder::EncodeToString() failed");
+ return rv;
+}
+
+already_AddRefed<nsIDocumentEncoder> EditorBase::GetAndInitDocEncoder(
+ const nsAString& aFormatType, uint32_t aDocumentEncoderFlags,
+ const nsACString& aCharset) const {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ nsCOMPtr<nsIDocumentEncoder> docEncoder;
+ if (!mCachedDocumentEncoder ||
+ !mCachedDocumentEncoderType.Equals(aFormatType)) {
+ nsAutoCString formatType;
+ LossyAppendUTF16toASCII(aFormatType, formatType);
+ docEncoder = do_createDocumentEncoder(PromiseFlatCString(formatType).get());
+ if (NS_WARN_IF(!docEncoder)) {
+ return nullptr;
+ }
+ mCachedDocumentEncoder = docEncoder;
+ mCachedDocumentEncoderType = aFormatType;
+ } else {
+ docEncoder = mCachedDocumentEncoder;
+ }
+
+ RefPtr<Document> doc = GetDocument();
+ NS_ASSERTION(doc, "Need a document");
+
+ nsresult rv = docEncoder->NativeInit(
+ doc, aFormatType,
+ aDocumentEncoderFlags | nsIDocumentEncoder::RequiresReinitAfterOutput);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("nsIDocumentEncoder::NativeInit() failed");
+ return nullptr;
+ }
+
+ if (!aCharset.IsEmpty() && !aCharset.EqualsLiteral("null")) {
+ DebugOnly<nsresult> rvIgnored = docEncoder->SetCharset(aCharset);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsIDocumentEncoder::SetCharset() failed, but ignored");
+ }
+
+ const int32_t wrapWidth = std::max(WrapWidth(), 0);
+ DebugOnly<nsresult> rvIgnored = docEncoder->SetWrapColumn(wrapWidth);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsIDocumentEncoder::SetWrapColumn() failed, but ignored");
+
+ // Set the selection, if appropriate.
+ // We do this either if the OutputSelectionOnly flag is set,
+ // in which case we use our existing selection ...
+ if (aDocumentEncoderFlags & nsIDocumentEncoder::OutputSelectionOnly) {
+ if (NS_FAILED(docEncoder->SetSelection(&SelectionRef()))) {
+ NS_WARNING("nsIDocumentEncoder::SetSelection() failed");
+ return nullptr;
+ }
+ }
+ // ... or if the root element is not a body,
+ // in which case we set the selection to encompass the root.
+ else {
+ Element* rootElement = GetRoot();
+ if (NS_WARN_IF(!rootElement)) {
+ return nullptr;
+ }
+ if (!rootElement->IsHTMLElement(nsGkAtoms::body)) {
+ if (NS_FAILED(docEncoder->SetContainerNode(rootElement))) {
+ NS_WARNING("nsIDocumentEncoder::SetContainerNode() failed");
+ return nullptr;
+ }
+ }
+ }
+
+ return docEncoder.forget();
+}
+
+bool EditorBase::AreClipboardCommandsUnconditionallyEnabled() const {
+ Document* document = GetDocument();
+ return document && document->AreClipboardCommandsUnconditionallyEnabled();
+}
+
+bool EditorBase::CheckForClipboardCommandListener(
+ nsAtom* aCommand, EventMessage aEventMessage) const {
+ RefPtr<Document> document = GetDocument();
+ if (!document) {
+ return false;
+ }
+
+ // We exclude XUL and chrome docs here to maintain current behavior where
+ // in these cases the editor element alone is expected to handle clipboard
+ // command availability.
+ if (!document->AreClipboardCommandsUnconditionallyEnabled()) {
+ return false;
+ }
+
+ // So in web content documents, "unconditionally" enabled Cut/Copy are not
+ // really unconditional; they're enabled if there is a listener that wants
+ // to handle them. What they're not conditional on here is whether there is
+ // currently a selection in the editor.
+ RefPtr<PresShell> presShell = document->GetObservingPresShell();
+ if (!presShell) {
+ return false;
+ }
+ RefPtr<nsPresContext> presContext = presShell->GetPresContext();
+ if (!presContext) {
+ return false;
+ }
+
+ RefPtr<EventTarget> et = GetDOMEventTarget();
+ while (et) {
+ EventListenerManager* elm = et->GetExistingListenerManager();
+ if (elm && elm->HasListenersFor(aCommand)) {
+ return true;
+ }
+ InternalClipboardEvent event(true, aEventMessage);
+ EventChainPreVisitor visitor(presContext, &event, nullptr,
+ nsEventStatus_eIgnore, false, et);
+ et->GetEventTargetParent(visitor);
+ et = visitor.GetParentTarget();
+ }
+
+ return false;
+}
+
+Result<EditorBase::ClipboardEventResult, nsresult>
+EditorBase::DispatchClipboardEventAndUpdateClipboard(EventMessage aEventMessage,
+ int32_t aClipboardType) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ const bool isPasting =
+ aEventMessage == ePaste || aEventMessage == ePasteNoFormatting;
+ if (isPasting) {
+ CommitComposition();
+ if (NS_WARN_IF(Destroyed())) {
+ return Err(NS_ERROR_EDITOR_DESTROYED);
+ }
+ }
+
+ RefPtr<PresShell> presShell = GetPresShell();
+ if (NS_WARN_IF(!presShell)) {
+ return Err(NS_ERROR_NOT_AVAILABLE);
+ }
+
+ const RefPtr<Selection> sel = [&]() {
+ if (IsHTMLEditor() && aEventMessage == eCopy &&
+ SelectionRef().IsCollapsed()) {
+ // If we don't have a usable selection for copy and we're an HTML
+ // editor (which is global for the document) try to use the last
+ // focused selection instead.
+ return nsCopySupport::GetSelectionForCopy(GetDocument());
+ }
+ return do_AddRef(&SelectionRef());
+ }();
+
+ bool actionTaken = false;
+ const bool doDefault = nsCopySupport::FireClipboardEvent(
+ aEventMessage, aClipboardType, presShell, sel, &actionTaken);
+ NotifyOfDispatchingClipboardEvent();
+
+ if (NS_WARN_IF(Destroyed())) {
+ return Err(NS_ERROR_EDITOR_DESTROYED);
+ }
+
+ if (doDefault) {
+ MOZ_ASSERT(actionTaken);
+ return ClipboardEventResult::DoDefault;
+ }
+ // If we handle a "paste" and nsCopySupport::FireClipboardEvent sets
+ // actionTaken to "false" means that it's an error. Otherwise, the "paste"
+ // event is just canceled.
+ if (isPasting) {
+ return actionTaken ? ClipboardEventResult::DefaultPreventedOfPaste
+ : ClipboardEventResult::IgnoredOrError;
+ }
+ // If we handle a "copy", actionTaken is set to true only when
+ // nsCopySupport::FireClipboardEvent does not meet an error.
+ // If we handle a "cut", actionTaken is set to true only when
+ // nsCopySupport::FireClipboardEvent does not meet an error and
+ // - the selection is collapsed in editable elements when the event is not
+ // canceled.
+ // - the event is canceled but update the clipboard with the dataTransfer
+ // of the event.
+ return actionTaken ? ClipboardEventResult::CopyOrCutHandled
+ : ClipboardEventResult::IgnoredOrError;
+}
+
+NS_IMETHODIMP EditorBase::Cut() {
+ nsresult rv = CutAsAction();
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "EditorBase::CutAsAction() failed");
+ return rv;
+}
+
+nsresult EditorBase::CutAsAction(nsIPrincipal* aPrincipal) {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eCut, aPrincipal);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ {
+ RefPtr<nsFocusManager> focusManager = nsFocusManager::GetFocusManager();
+ if (NS_WARN_IF(!focusManager)) {
+ return NS_ERROR_UNEXPECTED;
+ }
+ const RefPtr<Element> focusedElement = focusManager->GetFocusedElement();
+
+ Result<ClipboardEventResult, nsresult> ret =
+ DispatchClipboardEventAndUpdateClipboard(
+ eCut, nsIClipboard::kGlobalClipboard);
+ if (MOZ_UNLIKELY(ret.isErr())) {
+ NS_WARNING(
+ "EditorBase::DispatchClipboardEventAndUpdateClipboard(eCut, "
+ "nsIClipboard::kGlobalClipboard) failed");
+ return EditorBase::ToGenericNSResult(ret.unwrapErr());
+ }
+ switch (ret.unwrap()) {
+ case ClipboardEventResult::DoDefault:
+ break;
+ case ClipboardEventResult::CopyOrCutHandled:
+ return NS_OK;
+ case ClipboardEventResult::IgnoredOrError:
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ case ClipboardEventResult::DefaultPreventedOfPaste:
+ MOZ_ASSERT_UNREACHABLE("Invalid result for eCut");
+ }
+
+ // If focus is changed by a "cut" event listener, we should stop handling
+ // the cut.
+ const RefPtr<Element> newFocusedElement = focusManager->GetFocusedElement();
+ if (MOZ_UNLIKELY(focusedElement != newFocusedElement)) {
+ if (focusManager->GetFocusedWindow() != GetWindow()) {
+ return NS_OK;
+ }
+ RefPtr<EditorBase> editorBase =
+ nsContentUtils::GetActiveEditor(GetPresContext());
+ if (!editorBase || (editorBase->IsHTMLEditor() &&
+ !editorBase->AsHTMLEditor()->IsActiveInDOMWindow())) {
+ return NS_OK;
+ }
+ if (editorBase != this) {
+ return NS_OK;
+ }
+ }
+ }
+
+ // Dispatch "beforeinput" event after dispatching "cut" event.
+ nsresult rv = editActionData.MaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "MaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+ // XXX This transaction name is referred by PlaceholderTransaction::Merge()
+ // so that we need to keep using it here.
+ AutoPlaceholderBatch treatAsOneTransaction(*this, *nsGkAtoms::DeleteTxnName,
+ ScrollSelectionIntoView::Yes,
+ __FUNCTION__);
+ rv = DeleteSelectionAsSubAction(
+ eNone, IsTextEditor() ? nsIEditor::eNoStrip : nsIEditor::eStrip);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::DeleteSelectionAsSubAction(eNone) failed, but ignored");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+NS_IMETHODIMP EditorBase::CanCut(bool* aCanCut) {
+ if (NS_WARN_IF(!aCanCut)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aCanCut = IsCutCommandEnabled();
+ return NS_OK;
+}
+
+bool EditorBase::IsCutCommandEnabled() const {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return false;
+ }
+
+ if (IsModifiable() && IsCopyToClipboardAllowedInternal()) {
+ return true;
+ }
+
+ // If there's an event listener for "cut", we always enable the command
+ // as we don't really know what the listener may want to do in response.
+ // We look up the event target chain for a possible listener on a parent
+ // in addition to checking the immediate target.
+ return CheckForClipboardCommandListener(nsGkAtoms::oncut, eCut);
+}
+
+NS_IMETHODIMP EditorBase::Copy() {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eCopy);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ Result<ClipboardEventResult, nsresult> ret =
+ DispatchClipboardEventAndUpdateClipboard(eCopy,
+ nsIClipboard::kGlobalClipboard);
+ if (MOZ_UNLIKELY(ret.isErr())) {
+ NS_WARNING(
+ "EditorBase::DispatchClipboardEventAndUpdateClipboard(eCopy, "
+ "nsIClipboard::kGlobalClipboard) failed");
+ return EditorBase::ToGenericNSResult(ret.unwrapErr());
+ }
+ switch (ret.unwrap()) {
+ case ClipboardEventResult::DoDefault:
+ case ClipboardEventResult::CopyOrCutHandled:
+ return NS_OK;
+ case ClipboardEventResult::IgnoredOrError:
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ case ClipboardEventResult::DefaultPreventedOfPaste:
+ MOZ_ASSERT_UNREACHABLE("Invalid result for eCopy");
+ }
+ return NS_ERROR_UNEXPECTED;
+}
+
+NS_IMETHODIMP EditorBase::CanCopy(bool* aCanCopy) {
+ if (NS_WARN_IF(!aCanCopy)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aCanCopy = IsCopyCommandEnabled();
+ return NS_OK;
+}
+
+bool EditorBase::IsCopyCommandEnabled() const {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return false;
+ }
+
+ if (IsCopyToClipboardAllowedInternal()) {
+ return true;
+ }
+
+ // Like "cut", always enable "copy" if there's a listener.
+ return CheckForClipboardCommandListener(nsGkAtoms::oncopy, eCopy);
+}
+
+NS_IMETHODIMP EditorBase::Paste(int32_t aClipboardType) {
+ const nsresult rv = PasteAsAction(aClipboardType, DispatchPasteEvent::Yes);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::PasteAsAction(DispatchPasteEvent::Yes) failed");
+ return rv;
+}
+
+nsresult EditorBase::PasteAsAction(int32_t aClipboardType,
+ DispatchPasteEvent aDispatchPasteEvent,
+ nsIPrincipal* aPrincipal /* = nullptr */) {
+ if (IsHTMLEditor() && IsReadonly()) {
+ return NS_OK;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::ePaste,
+ aPrincipal);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ if (aDispatchPasteEvent == DispatchPasteEvent::Yes) {
+ RefPtr<nsFocusManager> focusManager = nsFocusManager::GetFocusManager();
+ if (NS_WARN_IF(!focusManager)) {
+ return NS_ERROR_UNEXPECTED;
+ }
+ const RefPtr<Element> focusedElement = focusManager->GetFocusedElement();
+
+ Result<ClipboardEventResult, nsresult> ret =
+ DispatchClipboardEventAndUpdateClipboard(ePaste, aClipboardType);
+ if (MOZ_UNLIKELY(ret.isErr())) {
+ NS_WARNING(
+ "EditorBase::DispatchClipboardEventAndUpdateClipboard(ePaste) "
+ "failed");
+ return EditorBase::ToGenericNSResult(ret.unwrapErr());
+ }
+ switch (ret.inspect()) {
+ case ClipboardEventResult::DoDefault:
+ break;
+ case ClipboardEventResult::DefaultPreventedOfPaste:
+ case ClipboardEventResult::IgnoredOrError:
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ case ClipboardEventResult::CopyOrCutHandled:
+ MOZ_ASSERT_UNREACHABLE("Invalid result for ePaste");
+ }
+
+ // If focus is changed by a "paste" event listener, we should keep handling
+ // the "pasting" in new focused editor because Chrome works as so.
+ const RefPtr<Element> newFocusedElement = focusManager->GetFocusedElement();
+ if (MOZ_UNLIKELY(focusedElement != newFocusedElement)) {
+ // For the privacy reason, let's top handling it if new focused element is
+ // in different document.
+ if (focusManager->GetFocusedWindow() != GetWindow()) {
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ }
+ RefPtr<EditorBase> editorBase =
+ nsContentUtils::GetActiveEditor(GetPresContext());
+ if (!editorBase || (editorBase->IsHTMLEditor() &&
+ !editorBase->AsHTMLEditor()->IsActiveInDOMWindow())) {
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ }
+ if (editorBase != this) {
+ nsresult rv = editorBase->PasteAsAction(
+ aClipboardType, DispatchPasteEvent::No, aPrincipal);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::PasteAsAction(DispatchPasteEvent::No) failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+ }
+ } else {
+ // The caller must already have dispatched a "paste" event.
+ editActionData.NotifyOfDispatchingClipboardEvent();
+ }
+
+ nsresult rv = HandlePaste(editActionData, aClipboardType);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "EditorBase::HandlePaste() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::PasteAsQuotationAsAction(
+ int32_t aClipboardType, DispatchPasteEvent aDispatchPasteEvent,
+ nsIPrincipal* aPrincipal /* = nullptr */) {
+ MOZ_ASSERT(aClipboardType == nsIClipboard::kGlobalClipboard ||
+ aClipboardType == nsIClipboard::kSelectionClipboard);
+
+ if (IsHTMLEditor() && IsReadonly()) {
+ return NS_OK;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::ePasteAsQuotation,
+ aPrincipal);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ if (aDispatchPasteEvent == DispatchPasteEvent::Yes) {
+ RefPtr<nsFocusManager> focusManager = nsFocusManager::GetFocusManager();
+ if (NS_WARN_IF(!focusManager)) {
+ return NS_ERROR_UNEXPECTED;
+ }
+ const RefPtr<Element> focusedElement = focusManager->GetFocusedElement();
+
+ Result<ClipboardEventResult, nsresult> ret =
+ DispatchClipboardEventAndUpdateClipboard(ePaste, aClipboardType);
+ if (MOZ_UNLIKELY(ret.isErr())) {
+ NS_WARNING(
+ "EditorBase::DispatchClipboardEventAndUpdateClipboard(ePaste) "
+ "failed");
+ return EditorBase::ToGenericNSResult(ret.unwrapErr());
+ }
+ switch (ret.inspect()) {
+ case ClipboardEventResult::DoDefault:
+ break;
+ case ClipboardEventResult::DefaultPreventedOfPaste:
+ case ClipboardEventResult::IgnoredOrError:
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ case ClipboardEventResult::CopyOrCutHandled:
+ MOZ_ASSERT_UNREACHABLE("Invalid result for ePaste");
+ }
+
+ // If focus is changed by a "paste" event listener, we should keep handling
+ // the "pasting" in new focused editor because Chrome works as so.
+ const RefPtr<Element> newFocusedElement = focusManager->GetFocusedElement();
+ if (MOZ_UNLIKELY(focusedElement != newFocusedElement)) {
+ // For the privacy reason, let's top handling it if new focused element is
+ // in different document.
+ if (focusManager->GetFocusedWindow() != GetWindow()) {
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ }
+ RefPtr<EditorBase> editorBase =
+ nsContentUtils::GetActiveEditor(GetPresContext());
+ if (!editorBase || (editorBase->IsHTMLEditor() &&
+ !editorBase->AsHTMLEditor()->IsActiveInDOMWindow())) {
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ }
+ if (editorBase != this) {
+ nsresult rv = editorBase->PasteAsQuotationAsAction(
+ aClipboardType, DispatchPasteEvent::No, aPrincipal);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::PasteAsQuotationAsAction("
+ "DispatchPasteEvent::No) failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+ }
+ } else {
+ // The caller must already have dispatched a "paste" event.
+ editActionData.NotifyOfDispatchingClipboardEvent();
+ }
+
+ nsresult rv = HandlePasteAsQuotation(editActionData, aClipboardType);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::HandlePasteAsQuotation() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::PasteTransferableAsAction(
+ nsITransferable* aTransferable, DispatchPasteEvent aDispatchPasteEvent,
+ nsIPrincipal* aPrincipal /* = nullptr */) {
+ // FIXME: This may be called as a call of nsIEditor::PasteTransferable.
+ // In this case, we should keep handling the paste even in the readonly mode.
+ if (IsHTMLEditor() && IsReadonly()) {
+ return NS_OK;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::ePaste,
+ aPrincipal);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ if (aDispatchPasteEvent == DispatchPasteEvent::Yes) {
+ RefPtr<nsFocusManager> focusManager = nsFocusManager::GetFocusManager();
+ if (NS_WARN_IF(!focusManager)) {
+ return NS_ERROR_UNEXPECTED;
+ }
+ const RefPtr<Element> focusedElement = focusManager->GetFocusedElement();
+
+ // Use an invalid value for the clipboard type as data comes from
+ // aTransferable and we don't currently implement a way to put that in the
+ // data transfer in TextEditor yet.
+ Result<ClipboardEventResult, nsresult> ret =
+ DispatchClipboardEventAndUpdateClipboard(
+ ePaste, IsTextEditor() ? -1 : nsIClipboard::kGlobalClipboard);
+ if (MOZ_UNLIKELY(ret.isErr())) {
+ NS_WARNING(
+ "EditorBase::DispatchClipboardEventAndUpdateClipboard(ePaste) "
+ "failed");
+ return EditorBase::ToGenericNSResult(ret.unwrapErr());
+ }
+ switch (ret.inspect()) {
+ case ClipboardEventResult::DoDefault:
+ break;
+ case ClipboardEventResult::DefaultPreventedOfPaste:
+ case ClipboardEventResult::IgnoredOrError:
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ case ClipboardEventResult::CopyOrCutHandled:
+ MOZ_ASSERT_UNREACHABLE("Invalid result for ePaste");
+ }
+
+ // If focus is changed by a "paste" event listener, we should keep handling
+ // the "pasting" in new focused editor because Chrome works as so.
+ const RefPtr<Element> newFocusedElement = focusManager->GetFocusedElement();
+ if (MOZ_UNLIKELY(focusedElement != newFocusedElement)) {
+ // For the privacy reason, let's top handling it if new focused element is
+ // in different document.
+ if (focusManager->GetFocusedWindow() != GetWindow()) {
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ }
+ RefPtr<EditorBase> editorBase =
+ nsContentUtils::GetActiveEditor(GetPresContext());
+ if (!editorBase || (editorBase->IsHTMLEditor() &&
+ !editorBase->AsHTMLEditor()->IsActiveInDOMWindow())) {
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_ACTION_CANCELED);
+ }
+ if (editorBase != this) {
+ nsresult rv = editorBase->PasteTransferableAsAction(
+ aTransferable, DispatchPasteEvent::No, aPrincipal);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::PasteTransferableAsAction("
+ "DispatchPasteEvent::No) failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+ }
+ } else {
+ // The caller must already have dispatched a "paste" event.
+ editActionData.NotifyOfDispatchingClipboardEvent();
+ }
+
+ if (NS_WARN_IF(!aTransferable)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ nsresult rv = HandlePasteTransferable(editActionData, *aTransferable);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::HandlePasteTransferable() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::PrepareToInsertContent(
+ const EditorDOMPoint& aPointToInsert,
+ DeleteSelectedContent aDeleteSelectedContent) {
+ // TODO: Move this method to `EditorBase`.
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ MOZ_ASSERT(aPointToInsert.IsSet());
+
+ EditorDOMPoint pointToInsert(aPointToInsert);
+ if (aDeleteSelectedContent == DeleteSelectedContent::Yes) {
+ AutoTrackDOMPoint tracker(RangeUpdaterRef(), &pointToInsert);
+ nsresult rv = DeleteSelectionAsSubAction(
+ nsIEditor::eNone,
+ IsTextEditor() ? nsIEditor::eNoStrip : nsIEditor::eStrip);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::DeleteSelectionAsSubAction(eNone) failed");
+ return rv;
+ }
+ }
+
+ nsresult rv = CollapseSelectionTo(pointToInsert);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::CollapseSelectionTo() failed");
+ return rv;
+}
+
+nsresult EditorBase::InsertTextAt(
+ const nsAString& aStringToInsert, const EditorDOMPoint& aPointToInsert,
+ DeleteSelectedContent aDeleteSelectedContent) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(aPointToInsert.IsSet());
+
+ nsresult rv = PrepareToInsertContent(aPointToInsert, aDeleteSelectedContent);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::PrepareToInsertContent() failed");
+ return rv;
+ }
+
+ rv = InsertTextAsSubAction(aStringToInsert, SelectionHandling::Delete);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::InsertTextAsSubAction() failed");
+ return rv;
+}
+
+EditorBase::SafeToInsertData EditorBase::IsSafeToInsertData(
+ nsIPrincipal* aSourcePrincipal) const {
+ // Try to determine whether we should use a sanitizing fragment sink
+ RefPtr<Document> destdoc = GetDocument();
+ NS_ASSERTION(destdoc, "Where is our destination doc?");
+
+ nsIDocShell* docShell = nullptr;
+ if (RefPtr<BrowsingContext> bc = destdoc->GetBrowsingContext()) {
+ RefPtr<BrowsingContext> root = bc->Top();
+ MOZ_ASSERT(root, "root should not be null");
+
+ docShell = root->GetDocShell();
+ }
+
+ bool isSafe =
+ docShell && docShell->GetAppType() == nsIDocShell::APP_TYPE_EDITOR;
+
+ if (!isSafe && aSourcePrincipal) {
+ nsIPrincipal* destPrincipal = destdoc->NodePrincipal();
+ NS_ASSERTION(destPrincipal, "How come we don't have a principal?");
+ DebugOnly<nsresult> rvIgnored =
+ aSourcePrincipal->Subsumes(destPrincipal, &isSafe);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "nsIPrincipal::Subsumes() failed, but ignored");
+ }
+
+ return isSafe ? SafeToInsertData::Yes : SafeToInsertData::No;
+}
+
+NS_IMETHODIMP EditorBase::PasteTransferable(nsITransferable* aTransferable) {
+ nsresult rv =
+ PasteTransferableAsAction(aTransferable, DispatchPasteEvent::Yes);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::PasteTransferableAsAction(DispatchPasteEvent::Yes) failed");
+ return rv;
+}
+
+NS_IMETHODIMP EditorBase::CanPaste(int32_t aClipboardType, bool* aCanPaste) {
+ if (NS_WARN_IF(!aCanPaste)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aCanPaste = CanPaste(aClipboardType);
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::SetAttribute(Element* aElement,
+ const nsAString& aAttribute,
+ const nsAString& aValue) {
+ if (NS_WARN_IF(aAttribute.IsEmpty()) || NS_WARN_IF(!aElement)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eSetAttribute);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ RefPtr<nsAtom> attribute = NS_Atomize(aAttribute);
+ rv = SetAttributeWithTransaction(*aElement, *attribute, aValue);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::SetAttributeWithTransaction() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::SetAttributeWithTransaction(Element& aElement,
+ nsAtom& aAttribute,
+ const nsAString& aValue) {
+ RefPtr<ChangeAttributeTransaction> transaction =
+ ChangeAttributeTransaction::Create(aElement, aAttribute, aValue);
+ nsresult rv = DoTransactionInternal(transaction);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DoTransactionInternal() failed");
+ return rv;
+}
+
+NS_IMETHODIMP EditorBase::RemoveAttribute(Element* aElement,
+ const nsAString& aAttribute) {
+ if (NS_WARN_IF(aAttribute.IsEmpty()) || NS_WARN_IF(!aElement)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eRemoveAttribute);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ RefPtr<nsAtom> attribute = NS_Atomize(aAttribute);
+ rv = RemoveAttributeWithTransaction(*aElement, *attribute);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::RemoveAttributeWithTransaction() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::RemoveAttributeWithTransaction(Element& aElement,
+ nsAtom& aAttribute) {
+ if (!aElement.HasAttr(&aAttribute)) {
+ return NS_OK;
+ }
+ RefPtr<ChangeAttributeTransaction> transaction =
+ ChangeAttributeTransaction::CreateToRemove(aElement, aAttribute);
+ nsresult rv = DoTransactionInternal(transaction);
+ if (NS_WARN_IF(Destroyed())) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DoTransactionInternal() failed");
+ return rv;
+}
+
+nsresult EditorBase::MarkElementDirty(Element& aElement) const {
+ // Mark the node dirty, but not for webpages (bug 599983)
+ if (!OutputsMozDirty()) {
+ return NS_OK;
+ }
+ DebugOnly<nsresult> rvIgnored =
+ aElement.SetAttr(kNameSpaceID_None, nsGkAtoms::mozdirty, u""_ns, false);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "Element::SetAttr(nsGkAtoms::mozdirty) failed, but ignored");
+ return NS_WARN_IF(Destroyed()) ? NS_ERROR_EDITOR_DESTROYED : NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetInlineSpellChecker(
+ bool aAutoCreate, nsIInlineSpellChecker** aInlineSpellChecker) {
+ if (NS_WARN_IF(!aInlineSpellChecker)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (mDidPreDestroy) {
+ // Don't allow people to get or create the spell checker once the editor
+ // is going away.
+ *aInlineSpellChecker = nullptr;
+ return aAutoCreate ? NS_ERROR_NOT_AVAILABLE : NS_OK;
+ }
+
+ // We don't want to show the spell checking UI if there are no spell check
+ // dictionaries available.
+ if (!mozInlineSpellChecker::CanEnableInlineSpellChecking()) {
+ *aInlineSpellChecker = nullptr;
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!mInlineSpellChecker && aAutoCreate) {
+ mInlineSpellChecker = new mozInlineSpellChecker();
+ }
+
+ if (mInlineSpellChecker) {
+ nsresult rv = mInlineSpellChecker->Init(this);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("mozInlineSpellChecker::Init() failed");
+ mInlineSpellChecker = nullptr;
+ return rv;
+ }
+ }
+
+ *aInlineSpellChecker = do_AddRef(mInlineSpellChecker).take();
+ return NS_OK;
+}
+
+void EditorBase::SyncRealTimeSpell() {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return;
+ }
+
+ bool enable = GetDesiredSpellCheckState();
+
+ // Initializes mInlineSpellChecker
+ nsCOMPtr<nsIInlineSpellChecker> spellChecker;
+ DebugOnly<nsresult> rvIgnored =
+ GetInlineSpellChecker(enable, getter_AddRefs(spellChecker));
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::GetInlineSpellChecker() failed, but ignored");
+
+ if (mInlineSpellChecker) {
+ if (!mSpellCheckerDictionaryUpdated && enable) {
+ DebugOnly<nsresult> rvIgnored =
+ mInlineSpellChecker->UpdateCurrentDictionary();
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "mozInlineSpellChecker::UpdateCurrentDictionary() "
+ "failed, but ignored");
+ mSpellCheckerDictionaryUpdated = true;
+ }
+
+ // We might have a mInlineSpellChecker even if there are no dictionaries
+ // available since we don't destroy the mInlineSpellChecker when the last
+ // dictionariy is removed, but in that case spellChecker is null
+ DebugOnly<nsresult> rvIgnored =
+ mInlineSpellChecker->SetEnableRealTimeSpell(enable && spellChecker);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "mozInlineSpellChecker::SetEnableRealTimeSpell() failed, but ignored");
+ }
+}
+
+NS_IMETHODIMP EditorBase::SetSpellcheckUserOverride(bool enable) {
+ mSpellcheckCheckboxState = enable ? eTriTrue : eTriFalse;
+ SyncRealTimeSpell();
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::InsertNode(nsINode* aNodeToInsert,
+ nsINode* aContainer, uint32_t aOffset) {
+ nsCOMPtr<nsIContent> contentToInsert = do_QueryInterface(aNodeToInsert);
+ if (NS_WARN_IF(!contentToInsert) || NS_WARN_IF(!aContainer)) {
+ return NS_ERROR_NULL_POINTER;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eInsertNode);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ const uint32_t offset = std::min(aOffset, aContainer->Length());
+ Result<CreateContentResult, nsresult> insertContentResult =
+ InsertNodeWithTransaction(*contentToInsert,
+ EditorDOMPoint(aContainer, offset));
+ if (MOZ_UNLIKELY(insertContentResult.isErr())) {
+ NS_WARNING("EditorBase::InsertNodeWithTransaction() failed");
+ return EditorBase::ToGenericNSResult(insertContentResult.unwrapErr());
+ }
+ rv = insertContentResult.inspect().SuggestCaretPointTo(
+ *this, {SuggestCaret::OnlyIfHasSuggestion,
+ SuggestCaret::OnlyIfTransactionsAllowedToDoIt,
+ SuggestCaret::AndIgnoreTrivialError});
+ if (NS_FAILED(rv)) {
+ NS_WARNING("CreateContentResult::SuggestCaretPointTo() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+ NS_WARNING_ASSERTION(
+ rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR,
+ "CreateContentResult::SuggestCaretPointTo() failed, but ignored");
+ return NS_OK;
+}
+
+template <typename ContentNodeType>
+Result<CreateNodeResultBase<ContentNodeType>, nsresult>
+EditorBase::InsertNodeWithTransaction(ContentNodeType& aContentToInsert,
+ const EditorDOMPoint& aPointToInsert) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT_IF(IsTextEditor(), !aContentToInsert.IsText());
+
+ if (NS_WARN_IF(!aPointToInsert.IsSet())) {
+ return Err(NS_ERROR_INVALID_ARG);
+ }
+ MOZ_ASSERT(aPointToInsert.IsSetAndValid());
+
+ IgnoredErrorResult ignoredError;
+ AutoEditSubActionNotifier startToHandleEditSubAction(
+ *this, EditSubAction::eInsertNode, nsIEditor::eNext, ignoredError);
+ if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ return Err(ignoredError.StealNSResult());
+ }
+ NS_WARNING_ASSERTION(
+ !ignoredError.Failed(),
+ "TextEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored");
+
+ RefPtr<InsertNodeTransaction> transaction =
+ InsertNodeTransaction::Create(*this, aContentToInsert, aPointToInsert);
+ nsresult rv = DoTransactionInternal(transaction);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DoTransactionInternal() failed");
+
+ DebugOnly<nsresult> rvIgnored =
+ RangeUpdaterRef().SelAdjInsertNode(aPointToInsert);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "RangeUpdater::SelAdjInsertNode() failed, but ignored");
+
+ if (NS_WARN_IF(Destroyed())) {
+ return Err(NS_ERROR_EDITOR_DESTROYED);
+ }
+ if (NS_WARN_IF(aContentToInsert.GetParentNode() !=
+ aPointToInsert.GetContainer())) {
+ return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE);
+ }
+ if (NS_FAILED(rv)) {
+ return Err(rv);
+ }
+
+ if (IsHTMLEditor()) {
+ TopLevelEditSubActionDataRef().DidInsertContent(*this, aContentToInsert);
+ }
+
+ return CreateNodeResultBase<ContentNodeType>(
+ &aContentToInsert, transaction->SuggestPointToPutCaret<EditorDOMPoint>());
+}
+
+Result<CreateElementResult, nsresult>
+EditorBase::InsertPaddingBRElementForEmptyLastLineWithTransaction(
+ const EditorDOMPoint& aPointToInsert) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(IsHTMLEditor() || !aPointToInsert.IsInTextNode());
+
+ if (MOZ_UNLIKELY(!aPointToInsert.IsSet())) {
+ return Err(NS_ERROR_FAILURE);
+ }
+
+ EditorDOMPoint pointToInsert;
+ if (IsTextEditor()) {
+ pointToInsert = aPointToInsert;
+ } else {
+ Result<EditorDOMPoint, nsresult> maybePointToInsert =
+ MOZ_KnownLive(AsHTMLEditor())->PrepareToInsertBRElement(aPointToInsert);
+ if (maybePointToInsert.isErr()) {
+ return maybePointToInsert.propagateErr();
+ }
+ MOZ_ASSERT(maybePointToInsert.inspect().IsSetAndValid());
+ pointToInsert = maybePointToInsert.unwrap();
+ }
+
+ RefPtr<Element> newBRElement = CreateHTMLContent(nsGkAtoms::br);
+ if (NS_WARN_IF(!newBRElement)) {
+ return Err(NS_ERROR_FAILURE);
+ }
+ newBRElement->SetFlags(NS_PADDING_FOR_EMPTY_LAST_LINE);
+
+ Result<CreateElementResult, nsresult> insertBRElementResult =
+ InsertNodeWithTransaction<Element>(*newBRElement, pointToInsert);
+ NS_WARNING_ASSERTION(insertBRElementResult.isOk(),
+ "EditorBase::InsertNodeWithTransaction() failed");
+ return insertBRElementResult;
+}
+
+NS_IMETHODIMP EditorBase::DeleteNode(nsINode* aNode) {
+ if (NS_WARN_IF(!aNode) || NS_WARN_IF(!aNode->IsContent())) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eRemoveNode);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ rv = DeleteNodeWithTransaction(MOZ_KnownLive(*aNode->AsContent()));
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DeleteNodeWithTransaction() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::DeleteNodeWithTransaction(nsIContent& aContent) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT_IF(IsTextEditor(), !aContent.IsText());
+
+ // Do nothing if the node is read-only.
+ if (IsHTMLEditor() && NS_WARN_IF(!HTMLEditUtils::IsRemovableNode(aContent))) {
+ return NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE;
+ }
+
+ IgnoredErrorResult ignoredError;
+ AutoEditSubActionNotifier startToHandleEditSubAction(
+ *this, EditSubAction::eDeleteNode, nsIEditor::ePrevious, ignoredError);
+ if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ return ignoredError.StealNSResult();
+ }
+ NS_WARNING_ASSERTION(
+ !ignoredError.Failed(),
+ "TextEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored");
+
+ if (IsHTMLEditor()) {
+ TopLevelEditSubActionDataRef().WillDeleteContent(*this, aContent);
+ }
+
+ // FYI: DeleteNodeTransaction grabs aContent while it's alive. So, it's safe
+ // to refer aContent even after calling DoTransaction().
+ RefPtr<DeleteNodeTransaction> deleteNodeTransaction =
+ DeleteNodeTransaction::MaybeCreate(*this, aContent);
+ NS_WARNING_ASSERTION(deleteNodeTransaction,
+ "DeleteNodeTransaction::MaybeCreate() failed");
+ nsresult rv;
+ if (deleteNodeTransaction) {
+ rv = DoTransactionInternal(deleteNodeTransaction);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DoTransactionInternal() failed");
+
+ if (mTextServicesDocument && NS_SUCCEEDED(rv)) {
+ RefPtr<TextServicesDocument> textServicesDocument = mTextServicesDocument;
+ textServicesDocument->DidDeleteContent(aContent);
+ }
+ } else {
+ rv = NS_ERROR_FAILURE;
+ }
+
+ if (!mActionListeners.IsEmpty()) {
+ for (auto& listener : mActionListeners.Clone()) {
+ DebugOnly<nsresult> rvIgnored = listener->DidDeleteNode(&aContent, rv);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsIEditActionListener::DidDeleteNode() failed, but ignored");
+ }
+ }
+
+ return NS_WARN_IF(Destroyed()) ? NS_ERROR_EDITOR_DESTROYED : rv;
+}
+
+NS_IMETHODIMP EditorBase::NotifySelectionChanged(Document* aDocument,
+ Selection* aSelection,
+ int16_t aReason,
+ int32_t aAmount) {
+ if (NS_WARN_IF(!aDocument) || NS_WARN_IF(!aSelection)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (mTextInputListener) {
+ RefPtr<TextInputListener> textInputListener = mTextInputListener;
+ textInputListener->OnSelectionChange(*aSelection, aReason);
+ }
+
+ if (mIMEContentObserver) {
+ RefPtr<IMEContentObserver> observer = mIMEContentObserver;
+ observer->OnSelectionChange(*aSelection);
+ }
+
+ return NS_OK;
+}
+
+void EditorBase::NotifyEditorObservers(
+ NotificationForEditorObservers aNotification) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ switch (aNotification) {
+ case eNotifyEditorObserversOfEnd:
+ mIsInEditSubAction = false;
+
+ if (mEditActionData) {
+ mEditActionData->MarkAsHandled();
+ }
+
+ if (mTextInputListener) {
+ // TODO: TextInputListener::OnEditActionHandled() may return
+ // NS_ERROR_OUT_OF_MEMORY. If so and if
+ // TextControlState::SetValue() setting value with us, we should
+ // return the result to EditorBase::ReplaceTextAsAction(),
+ // EditorBase::DeleteSelectionAsAction() and
+ // TextEditor::InsertTextAsAction(). However, it requires a lot
+ // of changes in editor classes, but it's not so important since
+ // editor does not use fallible allocation. Therefore, normally,
+ // the process must be crashed anyway.
+ RefPtr<TextInputListener> listener = mTextInputListener;
+ nsresult rv =
+ listener->OnEditActionHandled(MOZ_KnownLive(*AsTextEditor()));
+ MOZ_RELEASE_ASSERT(rv != NS_ERROR_OUT_OF_MEMORY,
+ "Setting value failed due to out of memory");
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "TextInputListener::OnEditActionHandled() failed, but ignored");
+ }
+
+ if (mIMEContentObserver) {
+ RefPtr<IMEContentObserver> observer = mIMEContentObserver;
+ observer->OnEditActionHandled();
+ }
+
+ if (!mDispatchInputEvent || IsEditActionAborted() ||
+ IsEditActionCanceled()) {
+ break;
+ }
+
+ DispatchInputEvent();
+ break;
+ case eNotifyEditorObserversOfBefore:
+ if (NS_WARN_IF(mIsInEditSubAction)) {
+ return;
+ }
+
+ mIsInEditSubAction = true;
+
+ if (mIMEContentObserver) {
+ RefPtr<IMEContentObserver> observer = mIMEContentObserver;
+ observer->BeforeEditAction();
+ }
+ return;
+ case eNotifyEditorObserversOfCancel:
+ mIsInEditSubAction = false;
+
+ if (mEditActionData) {
+ mEditActionData->MarkAsHandled();
+ }
+
+ if (mIMEContentObserver) {
+ RefPtr<IMEContentObserver> observer = mIMEContentObserver;
+ observer->CancelEditAction();
+ }
+ break;
+ default:
+ MOZ_CRASH("Handle all notifications here");
+ break;
+ }
+
+ if (IsHTMLEditor() && !Destroyed()) {
+ // We may need to show resizing handles or update existing ones after
+ // all transactions are done. This way of doing is preferred to DOM
+ // mutation events listeners because all the changes the user can apply
+ // to a document may result in multiple events, some of them quite hard
+ // to listen too (in particular when an ancestor of the selection is
+ // changed but the selection itself is not changed).
+ DebugOnly<nsresult> rvIgnored =
+ MOZ_KnownLive(AsHTMLEditor())->RefreshEditingUI();
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "HTMLEditor::RefreshEditingUI() failed, but ignored");
+ }
+}
+
+void EditorBase::DispatchInputEvent() {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(!IsEditActionCanceled(),
+ "If preceding beforeinput event is canceled, we shouldn't "
+ "dispatch input event");
+ MOZ_ASSERT(
+ !ShouldAlreadyHaveHandledBeforeInputEventDispatching(),
+ "We've not handled beforeinput event but trying to dispatch input event");
+
+ // We don't need to dispatch multiple input events if there is a pending
+ // input event. However, it may have different event target. If we resolved
+ // this issue, we need to manage the pending events in an array. But it's
+ // overwork. We don't need to do it for the very rare case.
+ // TODO: However, we start to set InputEvent.inputType. So, each "input"
+ // event now notifies web app each change. So, perhaps, we should
+ // not omit input events.
+
+ RefPtr<Element> targetElement = GetInputEventTargetElement();
+ if (NS_WARN_IF(!targetElement)) {
+ return;
+ }
+ RefPtr<DataTransfer> dataTransfer = GetInputEventDataTransfer();
+ DebugOnly<nsresult> rvIgnored = nsContentUtils::DispatchInputEvent(
+ targetElement, eEditorInput, ToInputType(GetEditAction()), this,
+ dataTransfer ? InputEventOptions(dataTransfer,
+ InputEventOptions::NeverCancelable::No)
+ : InputEventOptions(GetInputEventData(),
+ InputEventOptions::NeverCancelable::No));
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsContentUtils::DispatchInputEvent() failed, but ignored");
+}
+
+NS_IMETHODIMP EditorBase::AddEditActionListener(
+ nsIEditActionListener* aListener) {
+ if (NS_WARN_IF(!aListener)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ // If given edit action listener is text services document for the inline
+ // spell checker, store it as reference of concrete class for performance
+ // reason.
+ if (mInlineSpellChecker) {
+ EditorSpellCheck* editorSpellCheck =
+ mInlineSpellChecker->GetEditorSpellCheck();
+ if (editorSpellCheck) {
+ mozSpellChecker* spellChecker = editorSpellCheck->GetSpellChecker();
+ if (spellChecker) {
+ TextServicesDocument* textServicesDocument =
+ spellChecker->GetTextServicesDocument();
+ if (static_cast<nsIEditActionListener*>(textServicesDocument) ==
+ aListener) {
+ mTextServicesDocument = textServicesDocument;
+ return NS_OK;
+ }
+ }
+ }
+ }
+
+ // Make sure the listener isn't already on the list
+ if (!mActionListeners.Contains(aListener)) {
+ mActionListeners.AppendElement(*aListener);
+ NS_WARNING_ASSERTION(
+ mActionListeners.Length() != 1,
+ "nsIEditActionListener installed, this editor becomes slower");
+ }
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::RemoveEditActionListener(
+ nsIEditActionListener* aListener) {
+ if (NS_WARN_IF(!aListener)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (static_cast<nsIEditActionListener*>(mTextServicesDocument) == aListener) {
+ mTextServicesDocument = nullptr;
+ return NS_OK;
+ }
+
+ NS_WARNING_ASSERTION(mActionListeners.Length() != 1,
+ "All nsIEditActionListeners have been removed, this "
+ "editor becomes faster");
+ mActionListeners.RemoveElement(aListener);
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::AddDocumentStateListener(
+ nsIDocumentStateListener* aListener) {
+ if (NS_WARN_IF(!aListener)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (!mDocStateListeners.Contains(aListener)) {
+ mDocStateListeners.AppendElement(*aListener);
+ }
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::RemoveDocumentStateListener(
+ nsIDocumentStateListener* aListener) {
+ if (NS_WARN_IF(!aListener)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ mDocStateListeners.RemoveElement(aListener);
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::ForceCompositionEnd() {
+ nsresult rv = CommitComposition();
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::CommitComposition() failed");
+ return rv;
+}
+
+nsresult EditorBase::CommitComposition() {
+ nsPresContext* presContext = GetPresContext();
+ if (NS_WARN_IF(!presContext)) {
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+
+ if (!mComposition) {
+ return NS_OK;
+ }
+ nsresult rv =
+ IMEStateManager::NotifyIME(REQUEST_TO_COMMIT_COMPOSITION, presContext);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "IMEStateManager::NotifyIME() failed");
+ return rv;
+}
+
+nsresult EditorBase::GetPreferredIMEState(IMEState* aState) {
+ if (NS_WARN_IF(!aState)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ aState->mEnabled = IMEEnabled::Enabled;
+ aState->mOpen = IMEState::DONT_CHANGE_OPEN_STATE;
+
+ if (IsReadonly()) {
+ aState->mEnabled = IMEEnabled::Disabled;
+ return NS_OK;
+ }
+
+ Element* rootElement = GetRoot();
+ if (NS_WARN_IF(!rootElement)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ nsIFrame* frameForRootElement = rootElement->GetPrimaryFrame();
+ if (NS_WARN_IF(!frameForRootElement)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ switch (frameForRootElement->StyleUIReset()->mIMEMode) {
+ case StyleImeMode::Auto:
+ if (IsPasswordEditor()) {
+ aState->mEnabled = IMEEnabled::Password;
+ }
+ break;
+ case StyleImeMode::Disabled:
+ // we should use password state for |ime-mode: disabled;|.
+ aState->mEnabled = IMEEnabled::Password;
+ break;
+ case StyleImeMode::Active:
+ aState->mOpen = IMEState::OPEN;
+ break;
+ case StyleImeMode::Inactive:
+ aState->mOpen = IMEState::CLOSED;
+ break;
+ case StyleImeMode::Normal:
+ break;
+ }
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetComposing(bool* aResult) {
+ if (NS_WARN_IF(!aResult)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aResult = IsIMEComposing();
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetRootElement(Element** aRootElement) {
+ if (NS_WARN_IF(!aRootElement)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aRootElement = do_AddRef(mRootElement).take();
+ return NS_WARN_IF(!*aRootElement) ? NS_ERROR_NOT_AVAILABLE : NS_OK;
+}
+
+void EditorBase::OnStartToHandleTopLevelEditSubAction(
+ EditSubAction aTopLevelEditSubAction,
+ nsIEditor::EDirection aDirectionOfTopLevelEditSubAction, ErrorResult& aRv) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(!aRv.Failed());
+ mEditActionData->SetTopLevelEditSubAction(aTopLevelEditSubAction,
+ aDirectionOfTopLevelEditSubAction);
+}
+
+nsresult EditorBase::OnEndHandlingTopLevelEditSubAction() {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ mEditActionData->SetTopLevelEditSubAction(EditSubAction::eNone, eNone);
+ return NS_OK;
+}
+
+void EditorBase::DoInsertText(Text& aText, uint32_t aOffset,
+ const nsAString& aStringToInsert,
+ ErrorResult& aRv) {
+ aText.InsertData(aOffset, aStringToInsert, aRv);
+ if (NS_WARN_IF(Destroyed())) {
+ aRv = NS_ERROR_EDITOR_DESTROYED;
+ return;
+ }
+ if (aRv.Failed()) {
+ NS_WARNING("Text::InsertData() failed");
+ return;
+ }
+ if (IsTextEditor() && !aStringToInsert.IsEmpty()) {
+ aRv = MOZ_KnownLive(AsTextEditor())
+ ->DidInsertText(aText.TextLength(), aOffset,
+ aStringToInsert.Length());
+ NS_WARNING_ASSERTION(!aRv.Failed(), "TextEditor::DidInsertText() failed");
+ }
+}
+
+void EditorBase::DoDeleteText(Text& aText, uint32_t aOffset, uint32_t aCount,
+ ErrorResult& aRv) {
+ if (IsTextEditor() && aCount > 0) {
+ AsTextEditor()->WillDeleteText(aText.TextLength(), aOffset, aCount);
+ }
+ aText.DeleteData(aOffset, aCount, aRv);
+ if (NS_WARN_IF(Destroyed())) {
+ aRv = NS_ERROR_EDITOR_DESTROYED;
+ return;
+ }
+ NS_WARNING_ASSERTION(!aRv.Failed(), "Text::DeleteData() failed");
+}
+
+void EditorBase::DoReplaceText(Text& aText, uint32_t aOffset, uint32_t aCount,
+ const nsAString& aStringToInsert,
+ ErrorResult& aRv) {
+ if (IsTextEditor() && aCount > 0) {
+ AsTextEditor()->WillDeleteText(aText.TextLength(), aOffset, aCount);
+ }
+ aText.ReplaceData(aOffset, aCount, aStringToInsert, aRv);
+ if (NS_WARN_IF(Destroyed())) {
+ aRv = NS_ERROR_EDITOR_DESTROYED;
+ return;
+ }
+ if (aRv.Failed()) {
+ NS_WARNING("Text::ReplaceData() failed");
+ return;
+ }
+ if (IsTextEditor() && !aStringToInsert.IsEmpty()) {
+ aRv = MOZ_KnownLive(AsTextEditor())
+ ->DidInsertText(aText.TextLength(), aOffset,
+ aStringToInsert.Length());
+ NS_WARNING_ASSERTION(!aRv.Failed(), "TextEditor::DidInsertText() failed");
+ }
+}
+
+void EditorBase::DoSetText(Text& aText, const nsAString& aStringToSet,
+ ErrorResult& aRv) {
+ if (IsTextEditor()) {
+ uint32_t length = aText.TextLength();
+ if (length > 0) {
+ AsTextEditor()->WillDeleteText(length, 0, length);
+ }
+ }
+ aText.SetData(aStringToSet, aRv);
+ if (NS_WARN_IF(Destroyed())) {
+ aRv = NS_ERROR_EDITOR_DESTROYED;
+ return;
+ }
+ if (aRv.Failed()) {
+ NS_WARNING("Text::SetData() failed");
+ return;
+ }
+ if (IsTextEditor() && !aStringToSet.IsEmpty()) {
+ aRv = MOZ_KnownLive(AsTextEditor())
+ ->DidInsertText(aText.Length(), 0, aStringToSet.Length());
+ NS_WARNING_ASSERTION(!aRv.Failed(), "TextEditor::DidInsertText() failed");
+ }
+}
+
+nsresult EditorBase::CloneAttributeWithTransaction(nsAtom& aAttribute,
+ Element& aDestElement,
+ Element& aSourceElement) {
+ nsAutoString attrValue;
+ if (aSourceElement.GetAttr(kNameSpaceID_None, &aAttribute, attrValue)) {
+ nsresult rv =
+ SetAttributeWithTransaction(aDestElement, aAttribute, attrValue);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::SetAttributeWithTransaction() failed");
+ return rv;
+ }
+ nsresult rv = RemoveAttributeWithTransaction(aDestElement, aAttribute);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::RemoveAttributeWithTransaction() failed");
+ return rv;
+}
+
+NS_IMETHODIMP EditorBase::CloneAttributes(Element* aDestElement,
+ Element* aSourceElement) {
+ if (NS_WARN_IF(!aDestElement) || NS_WARN_IF(!aSourceElement)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eSetAttribute);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ CloneAttributesWithTransaction(*aDestElement, *aSourceElement);
+
+ return NS_OK;
+}
+
+void EditorBase::CloneAttributesWithTransaction(Element& aDestElement,
+ Element& aSourceElement) {
+ AutoPlaceholderBatch treatAsOneTransaction(
+ *this, ScrollSelectionIntoView::Yes, __FUNCTION__);
+
+ // Use transaction system for undo only if destination is already in the
+ // document
+ Element* rootElement = GetRoot();
+ if (NS_WARN_IF(!rootElement)) {
+ return;
+ }
+
+ OwningNonNull<Element> destElement(aDestElement);
+ OwningNonNull<Element> sourceElement(aSourceElement);
+ bool isDestElementInBody = rootElement->Contains(destElement);
+
+ // Clear existing attributes
+ RefPtr<nsDOMAttributeMap> destAttributes = destElement->Attributes();
+ while (RefPtr<Attr> attr = destAttributes->Item(0)) {
+ if (isDestElementInBody) {
+ DebugOnly<nsresult> rvIgnored = RemoveAttributeWithTransaction(
+ destElement, MOZ_KnownLive(*attr->NodeInfo()->NameAtom()));
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::RemoveAttributeWithTransaction() failed, but ignored");
+ } else {
+ DebugOnly<nsresult> rvIgnored = destElement->UnsetAttr(
+ kNameSpaceID_None, attr->NodeInfo()->NameAtom(), true);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "Element::UnsetAttr() failed, but ignored");
+ }
+ }
+
+ // Set just the attributes that the source element has
+ RefPtr<nsDOMAttributeMap> sourceAttributes = sourceElement->Attributes();
+ uint32_t sourceCount = sourceAttributes->Length();
+ for (uint32_t i = 0; i < sourceCount; i++) {
+ RefPtr<Attr> attr = sourceAttributes->Item(i);
+ nsAutoString value;
+ attr->GetValue(value);
+ if (isDestElementInBody) {
+ DebugOnly<nsresult> rvIgnored = SetAttributeOrEquivalent(
+ destElement, MOZ_KnownLive(attr->NodeInfo()->NameAtom()), value,
+ false);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::SetAttributeOrEquivalent() failed, but ignored");
+ } else {
+ // The element is not inserted in the document yet, we don't want to put
+ // a transaction on the UndoStack
+ DebugOnly<nsresult> rvIgnored = SetAttributeOrEquivalent(
+ destElement, MOZ_KnownLive(attr->NodeInfo()->NameAtom()), value,
+ true);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::SetAttributeOrEquivalent() failed, but ignored");
+ }
+ }
+}
+
+nsresult EditorBase::ScrollSelectionFocusIntoView() const {
+ nsISelectionController* selectionController = GetSelectionController();
+ if (!selectionController) {
+ return NS_OK;
+ }
+
+ DebugOnly<nsresult> rvIgnored = selectionController->ScrollSelectionIntoView(
+ nsISelectionController::SELECTION_NORMAL,
+ nsISelectionController::SELECTION_FOCUS_REGION,
+ nsISelectionController::SCROLL_OVERFLOW_HIDDEN);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsISelectionController::ScrollSelectionIntoView() failed, but ignored");
+ return NS_WARN_IF(Destroyed()) ? NS_ERROR_EDITOR_DESTROYED : NS_OK;
+}
+
+template <typename EditorDOMPointType>
+EditorDOMPointType EditorBase::FindBetterInsertionPoint(
+ const EditorDOMPointType& aPoint) const {
+ if (MOZ_UNLIKELY(NS_WARN_IF(!aPoint.IsInContentNode()))) {
+ return aPoint;
+ }
+
+ MOZ_ASSERT(aPoint.IsSetAndValid());
+
+ if (aPoint.IsInTextNode()) {
+ // There is no "better" insertion point.
+ return aPoint;
+ }
+
+ if (!IsInPlaintextMode()) {
+ // We cannot find "better" insertion point in HTML editor.
+ // WARNING: When you add some code to find better node in HTML editor,
+ // you need to call this before calling InsertTextWithTransaction()
+ // in HTMLEditor.
+ return aPoint;
+ }
+
+ RefPtr<Element> rootElement = GetRoot();
+ if (aPoint.GetContainer() == rootElement) {
+ // In some cases, aNode is the anonymous DIV, and offset is 0. To avoid
+ // injecting unneeded text nodes, we first look to see if we have one
+ // available. In that case, we'll just adjust node and offset accordingly.
+ if (aPoint.IsStartOfContainer() && aPoint.GetContainer()->HasChildren() &&
+ aPoint.GetContainer()->GetFirstChild()->IsText()) {
+ return EditorDOMPointType(aPoint.GetContainer()->GetFirstChild(), 0u);
+ }
+
+ // In some other cases, aNode is the anonymous DIV, and offset points to
+ // the terminating padding <br> element for empty last line. In that case,
+ // we'll adjust aInOutNode and aInOutOffset to the preceding text node,
+ // if any.
+ if (!aPoint.IsStartOfContainer()) {
+ if (IsHTMLEditor()) {
+ // Fall back to a slow path that uses GetChildAt_Deprecated() for
+ // Thunderbird's plaintext editor.
+ nsIContent* child = aPoint.GetPreviousSiblingOfChild();
+ if (child && child->IsText()) {
+ return EditorDOMPointType::AtEndOf(*child);
+ }
+ } else {
+ // If we're in a real plaintext editor, use a fast path that avoids
+ // calling GetChildAt_Deprecated() which may perform a linear search.
+ nsIContent* child = aPoint.GetContainer()->GetLastChild();
+ while (child) {
+ if (child->IsText()) {
+ return EditorDOMPointType::AtEndOf(*child);
+ }
+ child = child->GetPreviousSibling();
+ }
+ }
+ }
+ }
+
+ // Sometimes, aNode is the padding <br> element itself. In that case, we'll
+ // adjust the insertion point to the previous text node, if one exists, or
+ // to the parent anonymous DIV.
+ if (EditorUtils::IsPaddingBRElementForEmptyLastLine(
+ *aPoint.template ContainerAs<nsIContent>()) &&
+ aPoint.IsStartOfContainer()) {
+ nsIContent* previousSibling = aPoint.GetContainer()->GetPreviousSibling();
+ if (previousSibling && previousSibling->IsText()) {
+ return EditorDOMPointType::AtEndOf(*previousSibling);
+ }
+
+ nsINode* parentOfContainer = aPoint.GetContainerParent();
+ if (parentOfContainer && parentOfContainer == rootElement) {
+ return EditorDOMPointType(parentOfContainer,
+ aPoint.template ContainerAs<nsIContent>(), 0u);
+ }
+ }
+
+ return aPoint;
+}
+
+Result<InsertTextResult, nsresult> EditorBase::InsertTextWithTransaction(
+ Document& aDocument, const nsAString& aStringToInsert,
+ const EditorDOMPoint& aPointToInsert) {
+ if (NS_WARN_IF(!aPointToInsert.IsSet())) {
+ return Err(NS_ERROR_INVALID_ARG);
+ }
+
+ MOZ_ASSERT(aPointToInsert.IsSetAndValid());
+
+ if (!ShouldHandleIMEComposition() && aStringToInsert.IsEmpty()) {
+ return InsertTextResult();
+ }
+
+ // In some cases, the node may be the anonymous div element or a padding
+ // <br> element for empty last line. Let's try to look for better insertion
+ // point in the nearest text node if there is.
+ EditorDOMPoint pointToInsert = FindBetterInsertionPoint(aPointToInsert);
+
+ // If a neighboring text node already exists, use that
+ if (!pointToInsert.IsInTextNode()) {
+ nsIContent* child = nullptr;
+ if (!pointToInsert.IsStartOfContainer() &&
+ (child = pointToInsert.GetPreviousSiblingOfChild()) &&
+ child->IsText()) {
+ pointToInsert.Set(child, child->Length());
+ } else if (!pointToInsert.IsEndOfContainer() &&
+ (child = pointToInsert.GetChild()) && child->IsText()) {
+ pointToInsert.Set(child, 0);
+ }
+ }
+
+ if (ShouldHandleIMEComposition()) {
+ if (!pointToInsert.IsInTextNode()) {
+ // create a text node
+ RefPtr<nsTextNode> newTextNode = CreateTextNode(u""_ns);
+ if (NS_WARN_IF(!newTextNode)) {
+ return Err(NS_ERROR_FAILURE);
+ }
+ // then we insert it into the dom tree
+ Result<CreateTextResult, nsresult> insertTextNodeResult =
+ InsertNodeWithTransaction<Text>(*newTextNode, pointToInsert);
+ if (MOZ_UNLIKELY(insertTextNodeResult.isErr())) {
+ NS_WARNING("EditorBase::InsertNodeWithTransaction() failed");
+ return insertTextNodeResult.propagateErr();
+ }
+ insertTextNodeResult.unwrap().IgnoreCaretPointSuggestion();
+ pointToInsert.Set(newTextNode, 0u);
+ }
+ Result<InsertTextResult, nsresult> insertTextResult =
+ InsertTextIntoTextNodeWithTransaction(aStringToInsert,
+ pointToInsert.AsInText());
+ NS_WARNING_ASSERTION(
+ insertTextResult.isOk(),
+ "EditorBase::InsertTextIntoTextNodeWithTransaction() failed");
+ return insertTextResult;
+ }
+
+ if (pointToInsert.IsInTextNode()) {
+ // we are inserting text into an existing text node.
+ Result<InsertTextResult, nsresult> insertTextResult =
+ InsertTextIntoTextNodeWithTransaction(aStringToInsert,
+ pointToInsert.AsInText());
+ NS_WARNING_ASSERTION(
+ insertTextResult.isOk(),
+ "EditorBase::InsertTextIntoTextNodeWithTransaction() failed");
+ return insertTextResult;
+ }
+
+ // we are inserting text into a non-text node. first we have to create a
+ // textnode (this also populates it with the text)
+ RefPtr<nsTextNode> newTextNode = CreateTextNode(aStringToInsert);
+ if (NS_WARN_IF(!newTextNode)) {
+ return Err(NS_ERROR_FAILURE);
+ }
+ // then we insert it into the dom tree
+ Result<CreateTextResult, nsresult> insertTextNodeResult =
+ InsertNodeWithTransaction<Text>(*newTextNode, pointToInsert);
+ if (MOZ_UNLIKELY(insertTextNodeResult.isErr())) {
+ NS_WARNING("EditorBase::InsertNodeWithTransaction() failed");
+ return Err(insertTextNodeResult.unwrapErr());
+ }
+ insertTextNodeResult.unwrap().IgnoreCaretPointSuggestion();
+ if (NS_WARN_IF(!newTextNode->IsInComposedDoc())) {
+ return Err(NS_ERROR_EDITOR_UNEXPECTED_DOM_TREE);
+ }
+ return InsertTextResult(EditorDOMPointInText::AtEndOf(*newTextNode),
+ EditorDOMPoint::AtEndOf(*newTextNode));
+}
+
+static bool TextFragmentBeginsWithStringAtOffset(
+ const nsTextFragment& aTextFragment, const uint32_t aOffset,
+ const nsAString& aString) {
+ const uint32_t stringLength = aString.Length();
+
+ if (aOffset + stringLength > aTextFragment.GetLength()) {
+ return false;
+ }
+
+ if (aTextFragment.Is2b()) {
+ return aString.Equals(aTextFragment.Get2b() + aOffset);
+ }
+
+ return aString.EqualsLatin1(aTextFragment.Get1b() + aOffset, stringLength);
+}
+
+static std::tuple<EditorDOMPointInText, EditorDOMPointInText>
+AdjustTextInsertionRange(const EditorDOMPointInText& aInsertedPoint,
+ const nsAString& aInsertedString) {
+ if (TextFragmentBeginsWithStringAtOffset(
+ aInsertedPoint.ContainerAs<Text>()->TextFragment(),
+ aInsertedPoint.Offset(), aInsertedString)) {
+ return {aInsertedPoint,
+ EditorDOMPointInText(
+ aInsertedPoint.ContainerAs<Text>(),
+ aInsertedPoint.Offset() + aInsertedString.Length())};
+ }
+
+ return {EditorDOMPointInText(aInsertedPoint.ContainerAs<Text>(), 0),
+ EditorDOMPointInText::AtEndOf(*aInsertedPoint.ContainerAs<Text>())};
+}
+
+std::tuple<EditorDOMPointInText, EditorDOMPointInText>
+EditorBase::ComputeInsertedRange(const EditorDOMPointInText& aInsertedPoint,
+ const nsAString& aInsertedString) const {
+ MOZ_ASSERT(aInsertedPoint.IsSet());
+
+ // The DOM was potentially modified during the transaction. This is possible
+ // through mutation event listeners. That is, the node could've been removed
+ // from the doc or otherwise modified.
+ if (!MayHaveMutationEventListeners(
+ NS_EVENT_BITS_MUTATION_CHARACTERDATAMODIFIED)) {
+ EditorDOMPointInText endOfInsertion(
+ aInsertedPoint.ContainerAs<Text>(),
+ aInsertedPoint.Offset() + aInsertedString.Length());
+ return {aInsertedPoint, endOfInsertion};
+ }
+ if (aInsertedPoint.ContainerAs<Text>()->IsInComposedDoc()) {
+ EditorDOMPointInText begin, end;
+ return AdjustTextInsertionRange(aInsertedPoint, aInsertedString);
+ }
+ return {EditorDOMPointInText(), EditorDOMPointInText()};
+}
+
+Result<InsertTextResult, nsresult>
+EditorBase::InsertTextIntoTextNodeWithTransaction(
+ const nsAString& aStringToInsert,
+ const EditorDOMPointInText& aPointToInsert) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(aPointToInsert.IsSetAndValid());
+
+ RefPtr<EditTransactionBase> transaction;
+ bool isIMETransaction = false;
+ if (ShouldHandleIMEComposition()) {
+ transaction =
+ CompositionTransaction::Create(*this, aStringToInsert, aPointToInsert);
+ isIMETransaction = true;
+ } else {
+ transaction =
+ InsertTextTransaction::Create(*this, aStringToInsert, aPointToInsert);
+ }
+
+ // XXX We may not need these view batches anymore. This is handled at a
+ // higher level now I believe.
+ BeginUpdateViewBatch(__FUNCTION__);
+ nsresult rv = DoTransactionInternal(transaction);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DoTransactionInternal() failed");
+ EndUpdateViewBatch(__FUNCTION__);
+
+ // Don't check whether we've been destroyed here because we need to notify
+ // listeners and observers below even if we've already destroyed.
+
+ auto pointToInsert = [&]() -> EditorDOMPointInText {
+ if (!isIMETransaction) {
+ return aPointToInsert;
+ }
+ if (NS_WARN_IF(!mComposition->GetContainerTextNode())) {
+ return aPointToInsert;
+ }
+ return EditorDOMPointInText(
+ mComposition->GetContainerTextNode(),
+ std::min(mComposition->XPOffsetInTextNode(),
+ mComposition->GetContainerTextNode()->TextDataLength()));
+ }();
+
+ EditorDOMPointInText endOfInsertedText(
+ pointToInsert.ContainerAs<Text>(),
+ pointToInsert.Offset() + aStringToInsert.Length());
+
+ if (IsHTMLEditor()) {
+ auto [begin, end] = ComputeInsertedRange(pointToInsert, aStringToInsert);
+ if (begin.IsSet() && end.IsSet()) {
+ TopLevelEditSubActionDataRef().DidInsertText(
+ *this, begin.To<EditorRawDOMPoint>(), end.To<EditorRawDOMPoint>());
+ }
+ if (isIMETransaction) {
+ // Let's mark the text node as "modified frequently" if it interact with
+ // IME since non-ASCII character may be inserted into it in most cases.
+ pointToInsert.ContainerAs<Text>()->MarkAsMaybeModifiedFrequently();
+ }
+ // XXX Should we update endOfInsertedText here?
+ }
+
+ // let listeners know what happened
+ if (!mActionListeners.IsEmpty()) {
+ for (auto& listener : mActionListeners.Clone()) {
+ // TODO: might need adaptation because of mutation event listeners called
+ // during `DoTransactionInternal`.
+ DebugOnly<nsresult> rvIgnored = listener->DidInsertText(
+ pointToInsert.ContainerAs<Text>(),
+ static_cast<int32_t>(pointToInsert.Offset()), aStringToInsert, rv);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsIEditActionListener::DidInsertText() failed, but ignored");
+ }
+ }
+
+ // Added some cruft here for bug 43366. Layout was crashing because we left
+ // an empty text node lying around in the document. So I delete empty text
+ // nodes caused by IME. I have to mark the IME transaction as "fixed", which
+ // means that furure IME txns won't merge with it. This is because we don't
+ // want future IME txns trying to put their text into a node that is no
+ // longer in the document. This does not break undo/redo, because all these
+ // txns are wrapped in a parent PlaceHolder txn, and placeholder txns are
+ // already savvy to having multiple ime txns inside them.
+
+ // Delete empty IME text node if there is one
+ if (IsHTMLEditor() && isIMETransaction && mComposition) {
+ RefPtr<Text> textNode = mComposition->GetContainerTextNode();
+ if (textNode && !textNode->Length()) {
+ rv = DeleteNodeWithTransaction(*textNode);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DeleteNodeTransaction() failed");
+ if (MOZ_LIKELY(!textNode->IsInComposedDoc())) {
+ mComposition->OnTextNodeRemoved();
+ }
+ static_cast<CompositionTransaction*>(transaction.get())->MarkFixed();
+ }
+ }
+
+ if (NS_WARN_IF(Destroyed())) {
+ return Err(NS_ERROR_EDITOR_DESTROYED);
+ }
+
+ InsertTextTransaction* const insertTextTransaction =
+ transaction->GetAsInsertTextTransaction();
+ return insertTextTransaction
+ ? InsertTextResult(std::move(endOfInsertedText),
+ insertTextTransaction
+ ->SuggestPointToPutCaret<EditorDOMPoint>())
+ : InsertTextResult(std::move(endOfInsertedText));
+}
+
+nsresult EditorBase::NotifyDocumentListeners(
+ TDocumentListenerNotification aNotificationType) {
+ switch (aNotificationType) {
+ case eDocumentCreated:
+ if (IsTextEditor()) {
+ return NS_OK;
+ }
+ if (RefPtr<ComposerCommandsUpdater> composerCommandsUpdate =
+ AsHTMLEditor()->mComposerCommandsUpdater) {
+ composerCommandsUpdate->OnHTMLEditorCreated();
+ }
+ return NS_OK;
+
+ case eDocumentToBeDestroyed: {
+ RefPtr<ComposerCommandsUpdater> composerCommandsUpdate =
+ IsHTMLEditor() ? AsHTMLEditor()->mComposerCommandsUpdater : nullptr;
+ if (!mDocStateListeners.Length() && !composerCommandsUpdate) {
+ return NS_OK;
+ }
+ // Needs to store all listeners before notifying ComposerCommandsUpdate
+ // since notifying it might change mDocStateListeners.
+ const AutoDocumentStateListenerArray listeners(
+ mDocStateListeners.Clone());
+ if (composerCommandsUpdate) {
+ composerCommandsUpdate->OnBeforeHTMLEditorDestroyed();
+ }
+ for (auto& listener : listeners) {
+ // MOZ_KnownLive because 'listeners' is guaranteed to
+ // keep it alive.
+ //
+ // This can go away once
+ // https://bugzilla.mozilla.org/show_bug.cgi?id=1620312 is fixed.
+ nsresult rv = MOZ_KnownLive(listener)->NotifyDocumentWillBeDestroyed();
+ if (NS_FAILED(rv)) {
+ NS_WARNING(
+ "nsIDocumentStateListener::NotifyDocumentWillBeDestroyed() "
+ "failed");
+ return rv;
+ }
+ }
+ return NS_OK;
+ }
+ case eDocumentStateChanged: {
+ bool docIsDirty;
+ nsresult rv = GetDocumentModified(&docIsDirty);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::GetDocumentModified() failed");
+ return rv;
+ }
+
+ if (static_cast<int8_t>(docIsDirty) == mDocDirtyState) {
+ return NS_OK;
+ }
+
+ mDocDirtyState = docIsDirty;
+
+ RefPtr<ComposerCommandsUpdater> composerCommandsUpdate =
+ IsHTMLEditor() ? AsHTMLEditor()->mComposerCommandsUpdater : nullptr;
+ if (!mDocStateListeners.Length() && !composerCommandsUpdate) {
+ return NS_OK;
+ }
+ // Needs to store all listeners before notifying ComposerCommandsUpdate
+ // since notifying it might change mDocStateListeners.
+ const AutoDocumentStateListenerArray listeners(
+ mDocStateListeners.Clone());
+ if (composerCommandsUpdate) {
+ composerCommandsUpdate->OnHTMLEditorDirtyStateChanged(mDocDirtyState);
+ }
+ for (auto& listener : listeners) {
+ // MOZ_KnownLive because 'listeners' is guaranteed to
+ // keep it alive.
+ //
+ // This can go away once
+ // https://bugzilla.mozilla.org/show_bug.cgi?id=1620312 is fixed.
+ nsresult rv =
+ MOZ_KnownLive(listener)->NotifyDocumentStateChanged(mDocDirtyState);
+ if (NS_FAILED(rv)) {
+ NS_WARNING(
+ "nsIDocumentStateListener::NotifyDocumentStateChanged() failed");
+ return rv;
+ }
+ }
+ return NS_OK;
+ }
+ default:
+ MOZ_ASSERT_UNREACHABLE("Unknown notification");
+ return NS_ERROR_FAILURE;
+ }
+}
+
+nsresult EditorBase::SetTextNodeWithoutTransaction(const nsAString& aString,
+ Text& aTextNode) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(IsTextEditor());
+ MOZ_ASSERT(!IsUndoRedoEnabled());
+
+ const uint32_t length = aTextNode.Length();
+
+ // Let listeners know what's up
+ if (!mActionListeners.IsEmpty() && length) {
+ for (auto& listener : mActionListeners.Clone()) {
+ DebugOnly<nsresult> rvIgnored =
+ listener->WillDeleteText(MOZ_KnownLive(&aTextNode), 0, length);
+ if (NS_WARN_IF(Destroyed())) {
+ NS_WARNING(
+ "nsIEditActionListener::WillDeleteText() failed, but ignored");
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+ }
+ }
+
+ // We don't support undo here, so we don't really need all of the transaction
+ // machinery, therefore we can run our transaction directly, breaking all of
+ // the rules!
+ IgnoredErrorResult error;
+ DoSetText(aTextNode, aString, error);
+ if (MOZ_UNLIKELY(error.Failed())) {
+ NS_WARNING("EditorBase::DoSetText() failed");
+ return error.StealNSResult();
+ }
+
+ CollapseSelectionTo(EditorRawDOMPoint(&aTextNode, aString.Length()), error);
+ if (MOZ_UNLIKELY(error.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ NS_WARNING("EditorBase::CollapseSelection() caused destroying the editor");
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+ NS_ASSERTION(!error.Failed(),
+ "EditorBase::CollapseSelectionTo() failed, but ignored");
+
+ RangeUpdaterRef().SelAdjReplaceText(aTextNode, 0, length, aString.Length());
+
+ // Let listeners know what happened
+ if (!mActionListeners.IsEmpty() && !aString.IsEmpty()) {
+ for (auto& listener : mActionListeners.Clone()) {
+ DebugOnly<nsresult> rvIgnored =
+ listener->DidInsertText(&aTextNode, 0, aString, NS_OK);
+ if (NS_WARN_IF(Destroyed())) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsIEditActionListener::DidInsertText() failed, but ignored");
+ }
+ }
+
+ return NS_OK;
+}
+
+Result<CaretPoint, nsresult> EditorBase::DeleteTextWithTransaction(
+ Text& aTextNode, uint32_t aOffset, uint32_t aLength) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ RefPtr<DeleteTextTransaction> transaction =
+ DeleteTextTransaction::MaybeCreate(*this, aTextNode, aOffset, aLength);
+ if (MOZ_UNLIKELY(!transaction)) {
+ NS_WARNING("DeleteTextTransaction::MaybeCreate() failed");
+ return Err(NS_ERROR_FAILURE);
+ }
+
+ IgnoredErrorResult ignoredError;
+ AutoEditSubActionNotifier startToHandleEditSubAction(
+ *this, EditSubAction::eDeleteText, nsIEditor::ePrevious, ignoredError);
+ if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ return Err(ignoredError.StealNSResult());
+ }
+ NS_WARNING_ASSERTION(
+ !ignoredError.Failed(),
+ "TextEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored");
+
+ // Let listeners know what's up
+ if (!mActionListeners.IsEmpty()) {
+ for (auto& listener : mActionListeners.Clone()) {
+ DebugOnly<nsresult> rvIgnored =
+ listener->WillDeleteText(&aTextNode, aOffset, aLength);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsIEditActionListener::WillDeleteText() failed, but ignored");
+ }
+ }
+
+ nsresult rv = DoTransactionInternal(transaction);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DoTransactionInternal() failed");
+
+ if (IsHTMLEditor()) {
+ TopLevelEditSubActionDataRef().DidDeleteText(
+ *this, EditorRawDOMPoint(&aTextNode, aOffset));
+ }
+
+ if (NS_WARN_IF(Destroyed())) {
+ return Err(NS_ERROR_EDITOR_DESTROYED);
+ }
+ if (NS_FAILED(rv)) {
+ return Err(rv);
+ }
+
+ return CaretPoint(transaction->SuggestPointToPutCaret());
+}
+
+bool EditorBase::IsRoot(const nsINode* inNode) const {
+ if (NS_WARN_IF(!inNode)) {
+ return false;
+ }
+ nsINode* rootNode = GetRoot();
+ return inNode == rootNode;
+}
+
+bool EditorBase::IsDescendantOfRoot(const nsINode* inNode) const {
+ if (NS_WARN_IF(!inNode)) {
+ return false;
+ }
+ nsIContent* root = GetRoot();
+ if (NS_WARN_IF(!root)) {
+ return false;
+ }
+
+ return inNode->IsInclusiveDescendantOf(root);
+}
+
+NS_IMETHODIMP EditorBase::IncrementModificationCount(int32_t inNumMods) {
+ uint32_t oldModCount = mModCount;
+
+ mModCount += inNumMods;
+
+ if ((!oldModCount && mModCount) || (oldModCount && !mModCount)) {
+ DebugOnly<nsresult> rvIgnored =
+ NotifyDocumentListeners(eDocumentStateChanged);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::NotifyDocumentListeners() failed, but ignored");
+ }
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetModificationCount(int32_t* aOutModCount) {
+ if (NS_WARN_IF(!aOutModCount)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aOutModCount = mModCount;
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::ResetModificationCount() {
+ bool doNotify = (mModCount != 0);
+
+ mModCount = 0;
+
+ if (!doNotify) {
+ return NS_OK;
+ }
+
+ DebugOnly<nsresult> rvIgnored =
+ NotifyDocumentListeners(eDocumentStateChanged);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::NotifyDocumentListeners() failed, but ignored");
+ return NS_OK;
+}
+
+template <typename EditorDOMPointType>
+EditorDOMPointType EditorBase::GetFirstSelectionStartPoint() const {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ if (MOZ_UNLIKELY(!SelectionRef().RangeCount())) {
+ return EditorDOMPointType();
+ }
+
+ const nsRange* range = SelectionRef().GetRangeAt(0);
+ if (MOZ_UNLIKELY(NS_WARN_IF(!range) || NS_WARN_IF(!range->IsPositioned()))) {
+ return EditorDOMPointType();
+ }
+
+ return EditorDOMPointType(range->StartRef());
+}
+
+template <typename EditorDOMPointType>
+EditorDOMPointType EditorBase::GetFirstSelectionEndPoint() const {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ if (MOZ_UNLIKELY(!SelectionRef().RangeCount())) {
+ return EditorDOMPointType();
+ }
+
+ const nsRange* range = SelectionRef().GetRangeAt(0);
+ if (MOZ_UNLIKELY(NS_WARN_IF(!range) || NS_WARN_IF(!range->IsPositioned()))) {
+ return EditorDOMPointType();
+ }
+
+ return EditorDOMPointType(range->EndRef());
+}
+
+// static
+nsresult EditorBase::GetEndChildNode(const Selection& aSelection,
+ nsIContent** aEndNode) {
+ MOZ_ASSERT(aEndNode);
+
+ *aEndNode = nullptr;
+
+ if (NS_WARN_IF(!aSelection.RangeCount())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ const nsRange* range = aSelection.GetRangeAt(0);
+ if (NS_WARN_IF(!range)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (NS_WARN_IF(!range->IsPositioned())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ NS_IF_ADDREF(*aEndNode = range->GetChildAtEndOffset());
+ return NS_OK;
+}
+
+nsresult EditorBase::EnsurePaddingBRElementInMultilineEditor() {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(IsInPlaintextMode());
+ MOZ_ASSERT(!IsSingleLineEditor());
+
+ Element* anonymousDivOrBodyElement = GetRoot();
+ if (NS_WARN_IF(!anonymousDivOrBodyElement)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Assuming EditorBase::MaybeCreatePaddingBRElementForEmptyEditor() has been
+ // called first.
+ // XXX This assumption is wrong. This method may be called alone. Actually,
+ // we see this warning in mochitest log. So, we should fix this bug
+ // later.
+ if (NS_WARN_IF(!anonymousDivOrBodyElement->GetLastChild())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ RefPtr<HTMLBRElement> brElement =
+ HTMLBRElement::FromNode(anonymousDivOrBodyElement->GetLastChild());
+ if (!brElement) {
+ // TODO: Remove AutoTransactionsConserveSelection here. It's not necessary
+ // in normal cases. However, it may be required for nested edit
+ // actions which may be caused by legacy mutation event listeners or
+ // chrome script.
+ AutoTransactionsConserveSelection dontChangeMySelection(*this);
+ EditorDOMPoint endOfAnonymousDiv(
+ EditorDOMPoint::AtEndOf(*anonymousDivOrBodyElement));
+ Result<CreateElementResult, nsresult> insertPaddingBRElementResult =
+ InsertPaddingBRElementForEmptyLastLineWithTransaction(
+ endOfAnonymousDiv);
+ if (MOZ_UNLIKELY(insertPaddingBRElementResult.isErr())) {
+ NS_WARNING(
+ "EditorBase::InsertPaddingBRElementForEmptyLastLineWithTransaction() "
+ "failed");
+ return insertPaddingBRElementResult.unwrapErr();
+ }
+ insertPaddingBRElementResult.inspect().IgnoreCaretPointSuggestion();
+ return NS_OK;
+ }
+
+ // Check to see if the trailing BR is a former padding <br> element for empty
+ // editor - this will have stuck around if we previously morphed a trailing
+ // node into a padding <br> element.
+ if (!brElement->IsPaddingForEmptyEditor()) {
+ return NS_OK;
+ }
+
+ // Morph it back to a padding <br> element for empty last line.
+ brElement->UnsetFlags(NS_PADDING_FOR_EMPTY_EDITOR);
+ brElement->SetFlags(NS_PADDING_FOR_EMPTY_LAST_LINE);
+
+ return NS_OK;
+}
+
+void EditorBase::BeginUpdateViewBatch(const char* aRequesterFuncName) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(mUpdateCount >= 0, "bad state");
+
+ if (!mUpdateCount) {
+ // Turn off selection updates and notifications.
+ SelectionRef().StartBatchChanges(aRequesterFuncName);
+ }
+
+ mUpdateCount++;
+}
+
+void EditorBase::EndUpdateViewBatch(const char* aRequesterFuncName) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(mUpdateCount > 0, "bad state");
+
+ if (NS_WARN_IF(mUpdateCount <= 0)) {
+ mUpdateCount = 0;
+ return;
+ }
+
+ if (--mUpdateCount) {
+ return;
+ }
+
+ // Turn selection updating and notifications back on.
+ SelectionRef().EndBatchChanges(aRequesterFuncName);
+}
+
+TextComposition* EditorBase::GetComposition() const { return mComposition; }
+
+template <typename EditorDOMPointType>
+EditorDOMPointType EditorBase::GetFirstIMESelectionStartPoint() const {
+ return mComposition
+ ? EditorDOMPointType(mComposition->FirstIMESelectionStartRef())
+ : EditorDOMPointType();
+}
+
+template <typename EditorDOMPointType>
+EditorDOMPointType EditorBase::GetLastIMESelectionEndPoint() const {
+ return mComposition
+ ? EditorDOMPointType(mComposition->LastIMESelectionEndRef())
+ : EditorDOMPointType();
+}
+
+bool EditorBase::IsIMEComposing() const {
+ return mComposition && mComposition->IsComposing();
+}
+
+bool EditorBase::ShouldHandleIMEComposition() const {
+ // When the editor is being reframed, the old value may be restored with
+ // InsertText(). In this time, the text should be inserted as not a part
+ // of the composition.
+ return mComposition && mDidPostCreate;
+}
+
+bool EditorBase::EnsureComposition(WidgetCompositionEvent& aCompositionEvent) {
+ if (mComposition) {
+ return true;
+ }
+ // The compositionstart event must cause creating new TextComposition
+ // instance at being dispatched by IMEStateManager.
+ mComposition = IMEStateManager::GetTextCompositionFor(&aCompositionEvent);
+ if (!mComposition) {
+ // However, TextComposition may be committed before the composition
+ // event comes here.
+ return false;
+ }
+ mComposition->StartHandlingComposition(this);
+ return true;
+}
+
+nsresult EditorBase::OnCompositionStart(
+ WidgetCompositionEvent& aCompositionStartEvent) {
+ if (mComposition) {
+ NS_WARNING("There was a composition at receiving compositionstart event");
+ return NS_OK;
+ }
+
+ // "beforeinput" event shouldn't be fired before "compositionstart".
+ AutoEditActionDataSetter editActionData(*this, EditAction::eStartComposition);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ EnsureComposition(aCompositionStartEvent);
+ NS_WARNING_ASSERTION(mComposition, "Failed to get TextComposition instance?");
+ return NS_OK;
+}
+
+nsresult EditorBase::OnCompositionChange(
+ WidgetCompositionEvent& aCompositionChangeEvent) {
+ MOZ_ASSERT(aCompositionChangeEvent.mMessage == eCompositionChange,
+ "The event should be eCompositionChange");
+
+ if (!mComposition) {
+ NS_WARNING(
+ "There is no composition, but receiving compositionchange event");
+ return NS_ERROR_FAILURE;
+ }
+
+ AutoEditActionDataSetter editActionData(*this,
+ EditAction::eUpdateComposition);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ // If:
+ // - new composition string is not empty,
+ // - there is no composition string in the DOM tree,
+ // - and there is non-collapsed Selection,
+ // the selected content will be removed by this composition.
+ if (aCompositionChangeEvent.mData.IsEmpty() &&
+ mComposition->String().IsEmpty() && !SelectionRef().IsCollapsed()) {
+ editActionData.UpdateEditAction(EditAction::eDeleteByComposition);
+ }
+
+ // If Input Events Level 2 is enabled, EditAction::eDeleteByComposition is
+ // mapped to EditorInputType::eDeleteByComposition and it requires null
+ // for InputEvent.data. Therefore, only otherwise, we should set data.
+ if (ToInputType(editActionData.GetEditAction()) !=
+ EditorInputType::eDeleteByComposition) {
+ MOZ_ASSERT(ToInputType(editActionData.GetEditAction()) ==
+ EditorInputType::eInsertCompositionText);
+ MOZ_ASSERT(!aCompositionChangeEvent.mData.IsVoid());
+ editActionData.SetData(aCompositionChangeEvent.mData);
+ }
+
+ // If we're an `HTMLEditor` and this is second or later composition change,
+ // we should set target range to the range of composition string.
+ // Otherwise, set target ranges to selection ranges (will be done by
+ // editActionData itself before dispatching `beforeinput` event).
+ if (IsHTMLEditor() && mComposition->GetContainerTextNode()) {
+ RefPtr<StaticRange> targetRange = StaticRange::Create(
+ mComposition->GetContainerTextNode(),
+ mComposition->XPOffsetInTextNode(),
+ mComposition->GetContainerTextNode(),
+ mComposition->XPEndOffsetInTextNode(), IgnoreErrors());
+ NS_WARNING_ASSERTION(targetRange && targetRange->IsPositioned(),
+ "StaticRange::Create() failed");
+ if (targetRange && targetRange->IsPositioned()) {
+ editActionData.AppendTargetRange(*targetRange);
+ }
+ }
+
+ // TODO: We need to use different EditAction value for beforeinput event
+ // if the event is followed by "compositionend" because corresponding
+ // "input" event will be fired from OnCompositionEnd() later with
+ // different EditAction value.
+ // TODO: If Input Events Level 2 is enabled, "beforeinput" event may be
+ // actually canceled if edit action is eDeleteByComposition. In such
+ // case, we might need to keep selected text, but insert composition
+ // string before or after the selection. However, the spec is still
+ // unstable. We should keep handling the composition since other
+ // parts including widget may not be ready for such complicated
+ // behavior.
+ nsresult rv = editActionData.MaybeDispatchBeforeInputEvent();
+ if (rv != NS_ERROR_EDITOR_ACTION_CANCELED && NS_FAILED(rv)) {
+ NS_WARNING("MaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ if (!EnsureComposition(aCompositionChangeEvent)) {
+ NS_WARNING("EditorBase::EnsureComposition() failed");
+ return NS_OK;
+ }
+
+ if (NS_WARN_IF(!GetPresShell())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ // NOTE: TextComposition should receive selection change notification before
+ // CompositionChangeEventHandlingMarker notifies TextComposition of the
+ // end of handling compositionchange event because TextComposition may
+ // need to ignore selection changes caused by composition. Therefore,
+ // CompositionChangeEventHandlingMarker must be destroyed after a call
+ // of NotifiyEditorObservers(eNotifyEditorObserversOfEnd) or
+ // NotifiyEditorObservers(eNotifyEditorObserversOfCancel) which notifies
+ // TextComposition of a selection change.
+ MOZ_ASSERT(
+ !mPlaceholderBatch,
+ "UpdateIMEComposition() must be called without place holder batch");
+ nsString data(aCompositionChangeEvent.mData);
+ if (IsHTMLEditor()) {
+ nsContentUtils::PlatformToDOMLineBreaks(data);
+ }
+
+ {
+ // This needs to be destroyed before dispatching "input" event from
+ // the following call of `NotifyEditorObservers`. Therefore, we need to
+ // put this in this block rather than outside of this.
+ const bool wasComposing = mComposition->IsComposing();
+ TextComposition::CompositionChangeEventHandlingMarker
+ compositionChangeEventHandlingMarker(mComposition,
+ &aCompositionChangeEvent);
+ AutoPlaceholderBatch treatAsOneTransaction(*this, *nsGkAtoms::IMETxnName,
+ ScrollSelectionIntoView::Yes,
+ __FUNCTION__);
+
+ // XXX Why don't we get caret after the DOM mutation?
+ RefPtr<nsCaret> caret = GetCaret();
+
+ MOZ_ASSERT(
+ mIsInEditSubAction,
+ "AutoPlaceholderBatch should've notified the observes of before-edit");
+ // If we're updating composition, we need to ignore normal selection
+ // which may be updated by the web content.
+ rv = InsertTextAsSubAction(data, wasComposing ? SelectionHandling::Ignore
+ : SelectionHandling::Delete);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::InsertTextAsSubAction() failed");
+
+ if (caret) {
+ caret->SetSelection(&SelectionRef());
+ }
+ }
+
+ // If still composing, we should fire input event via observer.
+ // Note that if the composition will be committed by the following
+ // compositionend event, we don't need to notify editor observes of this
+ // change.
+ // NOTE: We must notify after the auto batch will be gone.
+ if (!aCompositionChangeEvent.IsFollowedByCompositionEnd()) {
+ // If we're a TextEditor, we'll be initialized with a new anonymous subtree,
+ // which can be caused by reframing from a "input" event listener. At that
+ // time, we'll move composition from current text node to the new text node
+ // with using mComposition's data. Therefore, it's important that
+ // mComposition already has the latest information here.
+ MOZ_ASSERT_IF(mComposition, mComposition->String() == data);
+ NotifyEditorObservers(eNotifyEditorObserversOfEnd);
+ }
+
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+void EditorBase::OnCompositionEnd(
+ WidgetCompositionEvent& aCompositionEndEvent) {
+ if (!mComposition) {
+ NS_WARNING("There is no composition, but receiving compositionend event");
+ return;
+ }
+
+ EditAction editAction = aCompositionEndEvent.mData.IsEmpty()
+ ? EditAction::eCancelComposition
+ : EditAction::eCommitComposition;
+ AutoEditActionDataSetter editActionData(*this, editAction);
+ // If Input Events Level 2 is enabled, EditAction::eCancelComposition is
+ // mapped to EditorInputType::eDeleteCompositionText and it requires null
+ // for InputEvent.data. Therefore, only otherwise, we should set data.
+ if (ToInputType(editAction) != EditorInputType::eDeleteCompositionText) {
+ MOZ_ASSERT(
+ ToInputType(editAction) == EditorInputType::eInsertCompositionText ||
+ ToInputType(editAction) == EditorInputType::eInsertFromComposition);
+ MOZ_ASSERT(!aCompositionEndEvent.mData.IsVoid());
+ editActionData.SetData(aCompositionEndEvent.mData);
+ }
+
+ // commit the IME transaction..we can get at it via the transaction mgr.
+ // Note that this means IME won't work without an undo stack!
+ if (mTransactionManager) {
+ if (nsCOMPtr<nsITransaction> transaction =
+ mTransactionManager->PeekUndoStack()) {
+ if (RefPtr<EditTransactionBase> transactionBase =
+ transaction->GetAsEditTransactionBase()) {
+ if (PlaceholderTransaction* placeholderTransaction =
+ transactionBase->GetAsPlaceholderTransaction()) {
+ placeholderTransaction->Commit();
+ }
+ }
+ }
+ }
+
+ // Note that this just marks as that we've already handled "beforeinput" for
+ // preventing assertions in FireInputEvent(). Note that corresponding
+ // "beforeinput" event for the following "input" event should've already
+ // been dispatched from `OnCompositionChange()`.
+ DebugOnly<nsresult> rvIgnored =
+ editActionData.MaybeDispatchBeforeInputEvent();
+ MOZ_ASSERT(rvIgnored != NS_ERROR_EDITOR_ACTION_CANCELED,
+ "Why beforeinput event was canceled in this case?");
+ MOZ_ASSERT(NS_SUCCEEDED(rvIgnored),
+ "MaybeDispatchBeforeInputEvent() should just mark the instance as "
+ "handled it");
+
+ // Composition string may have hidden the caret. Therefore, we need to
+ // cancel it here.
+ HideCaret(false);
+
+ // FYI: mComposition still keeps storing container text node of committed
+ // string, its offset and length. However, they will be invalidated
+ // soon since its Destroy() will be called by IMEStateManager.
+ mComposition->EndHandlingComposition(this);
+ mComposition = nullptr;
+
+ // notify editor observers of action
+ // FYI: With current draft, "input" event should be fired from
+ // OnCompositionChange(), however, it requires a lot of our UI code
+ // change and does not make sense. See spec issue:
+ // https://github.com/w3c/uievents/issues/202
+ NotifyEditorObservers(eNotifyEditorObserversOfEnd);
+}
+
+void EditorBase::DoAfterDoTransaction(nsITransaction* aTransaction) {
+ bool isTransientTransaction;
+ MOZ_ALWAYS_SUCCEEDS(aTransaction->GetIsTransient(&isTransientTransaction));
+
+ if (!isTransientTransaction) {
+ // we need to deal here with the case where the user saved after some
+ // edits, then undid one or more times. Then, the undo count is -ve,
+ // but we can't let a do take it back to zero. So we flip it up to
+ // a +ve number.
+ int32_t modCount;
+ DebugOnly<nsresult> rvIgnored = GetModificationCount(&modCount);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::GetModificationCount() failed, but ignored");
+ if (modCount < 0) {
+ modCount = -modCount;
+ }
+
+ // don't count transient transactions
+ MOZ_ALWAYS_SUCCEEDS(IncrementModificationCount(1));
+ }
+}
+
+void EditorBase::DoAfterUndoTransaction() {
+ // all undoable transactions are non-transient
+ MOZ_ALWAYS_SUCCEEDS(IncrementModificationCount(-1));
+}
+
+void EditorBase::DoAfterRedoTransaction() {
+ // all redoable transactions are non-transient
+ MOZ_ALWAYS_SUCCEEDS(IncrementModificationCount(1));
+}
+
+already_AddRefed<DeleteMultipleRangesTransaction>
+EditorBase::CreateTransactionForDeleteSelection(
+ HowToHandleCollapsedRange aHowToHandleCollapsedRange,
+ const AutoRangeArray& aRangesToDelete) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(!aRangesToDelete.Ranges().IsEmpty());
+
+ // Check whether the selection is collapsed and we should do nothing:
+ if (NS_WARN_IF(aRangesToDelete.IsCollapsed() &&
+ aHowToHandleCollapsedRange ==
+ HowToHandleCollapsedRange::Ignore)) {
+ return nullptr;
+ }
+
+ // allocate the out-param transaction
+ RefPtr<DeleteMultipleRangesTransaction> transaction =
+ DeleteMultipleRangesTransaction::Create();
+ for (const OwningNonNull<nsRange>& range : aRangesToDelete.Ranges()) {
+ // Same with range as with selection; if it is collapsed and action
+ // is eNone, do nothing.
+ if (!range->Collapsed()) {
+ RefPtr<DeleteRangeTransaction> deleteRangeTransaction =
+ DeleteRangeTransaction::Create(*this, range);
+ // XXX Oh, not checking if deleteRangeTransaction can modify the range...
+ transaction->AppendChild(*deleteRangeTransaction);
+ continue;
+ }
+
+ if (aHowToHandleCollapsedRange == HowToHandleCollapsedRange::Ignore) {
+ continue;
+ }
+
+ // Let's extend the collapsed range to delete content around it.
+ RefPtr<DeleteContentTransactionBase> deleteNodeOrTextTransaction =
+ CreateTransactionForCollapsedRange(range, aHowToHandleCollapsedRange);
+ // XXX When there are two or more ranges and at least one of them is
+ // not editable, deleteNodeOrTextTransaction may be nullptr.
+ // In such case, should we stop removing other ranges too?
+ if (!deleteNodeOrTextTransaction) {
+ NS_WARNING("EditorBase::CreateTransactionForCollapsedRange() failed");
+ return nullptr;
+ }
+ transaction->AppendChild(*deleteNodeOrTextTransaction);
+ }
+
+ return transaction.forget();
+}
+
+// XXX: currently, this doesn't handle edge conditions because GetNext/GetPrior
+// are not implemented
+already_AddRefed<DeleteContentTransactionBase>
+EditorBase::CreateTransactionForCollapsedRange(
+ const nsRange& aCollapsedRange,
+ HowToHandleCollapsedRange aHowToHandleCollapsedRange) {
+ MOZ_ASSERT(aCollapsedRange.Collapsed());
+ MOZ_ASSERT(
+ aHowToHandleCollapsedRange == HowToHandleCollapsedRange::ExtendBackward ||
+ aHowToHandleCollapsedRange == HowToHandleCollapsedRange::ExtendForward);
+
+ EditorRawDOMPoint point(aCollapsedRange.StartRef());
+ if (NS_WARN_IF(!point.IsSet())) {
+ return nullptr;
+ }
+ if (IsTextEditor()) {
+ // There should be only one text node in the anonymous `<div>` (but may
+ // be followed by a padding `<br>`). We should adjust the point into
+ // the text node (or return nullptr if there is no text to delete) for
+ // avoiding finding the text node with complicated API.
+ if (!point.IsInTextNode()) {
+ const Element* anonymousDiv = GetRoot();
+ if (NS_WARN_IF(!anonymousDiv)) {
+ return nullptr;
+ }
+ if (!anonymousDiv->GetFirstChild() ||
+ !anonymousDiv->GetFirstChild()->IsText()) {
+ return nullptr; // The value is empty.
+ }
+ if (point.GetContainer() == anonymousDiv) {
+ if (point.IsStartOfContainer()) {
+ point.Set(anonymousDiv->GetFirstChild(), 0);
+ } else {
+ point.SetToEndOf(anonymousDiv->GetFirstChild());
+ }
+ } else {
+ // Must be referring a padding `<br>` element or after the text node.
+ point.SetToEndOf(anonymousDiv->GetFirstChild());
+ }
+ }
+ MOZ_ASSERT(!point.ContainerAs<Text>()->GetPreviousSibling());
+ MOZ_ASSERT(!point.ContainerAs<Text>()->GetNextSibling() ||
+ !point.ContainerAs<Text>()->GetNextSibling()->IsText());
+ if (aHowToHandleCollapsedRange ==
+ HowToHandleCollapsedRange::ExtendBackward &&
+ point.IsStartOfContainer()) {
+ return nullptr;
+ }
+ if (aHowToHandleCollapsedRange ==
+ HowToHandleCollapsedRange::ExtendForward &&
+ point.IsEndOfContainer()) {
+ return nullptr;
+ }
+ }
+
+ // XXX: if the container of point is empty, then we'll need to delete the node
+ // as well as the 1 child
+
+ // build a transaction for deleting the appropriate data
+ // XXX: this has to come from rule section
+ const Element* const anonymousDivOrEditingHost =
+ IsTextEditor() ? GetRoot() : AsHTMLEditor()->ComputeEditingHost();
+ if (aHowToHandleCollapsedRange == HowToHandleCollapsedRange::ExtendBackward &&
+ point.IsStartOfContainer()) {
+ MOZ_ASSERT(IsHTMLEditor());
+ // We're backspacing from the beginning of a node. Delete the last thing
+ // of previous editable content.
+ nsIContent* previousEditableContent = HTMLEditUtils::GetPreviousContent(
+ *point.GetContainer(), {WalkTreeOption::IgnoreNonEditableNode},
+ anonymousDivOrEditingHost);
+ if (!previousEditableContent) {
+ NS_WARNING("There was no editable content before the collapsed range");
+ return nullptr;
+ }
+
+ // There is an editable content, so delete its last child (if a text node,
+ // delete the last char). If it has no children, delete it.
+ if (previousEditableContent->IsText()) {
+ uint32_t length = previousEditableContent->Length();
+ // Bail out for empty text node.
+ // XXX Do we want to do something else?
+ // XXX If other browsers delete empty text node, we should follow it.
+ if (NS_WARN_IF(!length)) {
+ NS_WARNING("Previous editable content was an empty text node");
+ return nullptr;
+ }
+ RefPtr<DeleteTextTransaction> deleteTextTransaction =
+ DeleteTextTransaction::MaybeCreateForPreviousCharacter(
+ *this, *previousEditableContent->AsText(), length);
+ if (!deleteTextTransaction) {
+ NS_WARNING(
+ "DeleteTextTransaction::MaybeCreateForPreviousCharacter() failed");
+ return nullptr;
+ }
+ return deleteTextTransaction.forget();
+ }
+
+ if (IsHTMLEditor() &&
+ NS_WARN_IF(!HTMLEditUtils::IsRemovableNode(*previousEditableContent))) {
+ return nullptr;
+ }
+ RefPtr<DeleteNodeTransaction> deleteNodeTransaction =
+ DeleteNodeTransaction::MaybeCreate(*this, *previousEditableContent);
+ if (!deleteNodeTransaction) {
+ NS_WARNING("DeleteNodeTransaction::MaybeCreate() failed");
+ return nullptr;
+ }
+ return deleteNodeTransaction.forget();
+ }
+
+ if (aHowToHandleCollapsedRange == HowToHandleCollapsedRange::ExtendForward &&
+ point.IsEndOfContainer()) {
+ MOZ_ASSERT(IsHTMLEditor());
+ // We're deleting from the end of a node. Delete the first thing of
+ // next editable content.
+ nsIContent* nextEditableContent = HTMLEditUtils::GetNextContent(
+ *point.GetContainer(), {WalkTreeOption::IgnoreNonEditableNode},
+ anonymousDivOrEditingHost);
+ if (!nextEditableContent) {
+ NS_WARNING("There was no editable content after the collapsed range");
+ return nullptr;
+ }
+
+ // There is an editable content, so delete its first child (if a text node,
+ // delete the first char). If it has no children, delete it.
+ if (nextEditableContent->IsText()) {
+ uint32_t length = nextEditableContent->Length();
+ // Bail out for empty text node.
+ // XXX Do we want to do something else?
+ // XXX If other browsers delete empty text node, we should follow it.
+ if (!length) {
+ NS_WARNING("Next editable content was an empty text node");
+ return nullptr;
+ }
+ RefPtr<DeleteTextTransaction> deleteTextTransaction =
+ DeleteTextTransaction::MaybeCreateForNextCharacter(
+ *this, *nextEditableContent->AsText(), 0);
+ if (!deleteTextTransaction) {
+ NS_WARNING(
+ "DeleteTextTransaction::MaybeCreateForNextCharacter() failed");
+ return nullptr;
+ }
+ return deleteTextTransaction.forget();
+ }
+
+ if (IsHTMLEditor() &&
+ NS_WARN_IF(!HTMLEditUtils::IsRemovableNode(*nextEditableContent))) {
+ return nullptr;
+ }
+ RefPtr<DeleteNodeTransaction> deleteNodeTransaction =
+ DeleteNodeTransaction::MaybeCreate(*this, *nextEditableContent);
+ if (!deleteNodeTransaction) {
+ NS_WARNING("DeleteNodeTransaction::MaybeCreate() failed");
+ return nullptr;
+ }
+ return deleteNodeTransaction.forget();
+ }
+
+ if (point.IsInTextNode()) {
+ if (aHowToHandleCollapsedRange ==
+ HowToHandleCollapsedRange::ExtendBackward) {
+ RefPtr<DeleteTextTransaction> deleteTextTransaction =
+ DeleteTextTransaction::MaybeCreateForPreviousCharacter(
+ *this, *point.ContainerAs<Text>(), point.Offset());
+ NS_WARNING_ASSERTION(
+ deleteTextTransaction,
+ "DeleteTextTransaction::MaybeCreateForPreviousCharacter() failed");
+ return deleteTextTransaction.forget();
+ }
+ RefPtr<DeleteTextTransaction> deleteTextTransaction =
+ DeleteTextTransaction::MaybeCreateForNextCharacter(
+ *this, *point.ContainerAs<Text>(), point.Offset());
+ NS_WARNING_ASSERTION(
+ deleteTextTransaction,
+ "DeleteTextTransaction::MaybeCreateForNextCharacter() failed");
+ return deleteTextTransaction.forget();
+ }
+
+ nsIContent* editableContent = nullptr;
+ if (IsHTMLEditor()) {
+ editableContent =
+ aHowToHandleCollapsedRange == HowToHandleCollapsedRange::ExtendBackward
+ ? HTMLEditUtils::GetPreviousContent(
+ point, {WalkTreeOption::IgnoreNonEditableNode},
+ anonymousDivOrEditingHost)
+ : HTMLEditUtils::GetNextContent(
+ point, {WalkTreeOption::IgnoreNonEditableNode},
+ anonymousDivOrEditingHost);
+ if (!editableContent) {
+ NS_WARNING("There was no editable content around the collapsed range");
+ return nullptr;
+ }
+ while (editableContent && editableContent->IsCharacterData() &&
+ !editableContent->Length()) {
+ // Can't delete an empty text node (bug 762183)
+ editableContent =
+ aHowToHandleCollapsedRange ==
+ HowToHandleCollapsedRange::ExtendBackward
+ ? HTMLEditUtils::GetPreviousContent(
+ *editableContent, {WalkTreeOption::IgnoreNonEditableNode},
+ anonymousDivOrEditingHost)
+ : HTMLEditUtils::GetNextContent(
+ *editableContent, {WalkTreeOption::IgnoreNonEditableNode},
+ anonymousDivOrEditingHost);
+ }
+ if (!editableContent) {
+ NS_WARNING(
+ "There was no editable content which is not empty around the "
+ "collapsed range");
+ return nullptr;
+ }
+ } else {
+ MOZ_ASSERT(point.IsInTextNode());
+ editableContent = point.GetContainerAs<nsIContent>();
+ if (!editableContent) {
+ NS_WARNING("If there was no text node, should've been handled first");
+ return nullptr;
+ }
+ }
+
+ if (editableContent->IsText()) {
+ if (aHowToHandleCollapsedRange ==
+ HowToHandleCollapsedRange::ExtendBackward) {
+ RefPtr<DeleteTextTransaction> deleteTextTransaction =
+ DeleteTextTransaction::MaybeCreateForPreviousCharacter(
+ *this, *editableContent->AsText(), editableContent->Length());
+ NS_WARNING_ASSERTION(
+ deleteTextTransaction,
+ "DeleteTextTransaction::MaybeCreateForPreviousCharacter() failed");
+ return deleteTextTransaction.forget();
+ }
+
+ RefPtr<DeleteTextTransaction> deleteTextTransaction =
+ DeleteTextTransaction::MaybeCreateForNextCharacter(
+ *this, *editableContent->AsText(), 0);
+ NS_WARNING_ASSERTION(
+ deleteTextTransaction,
+ "DeleteTextTransaction::MaybeCreateForNextCharacter() failed");
+ return deleteTextTransaction.forget();
+ }
+
+ MOZ_ASSERT(IsHTMLEditor());
+ if (NS_WARN_IF(!HTMLEditUtils::IsRemovableNode(*editableContent))) {
+ return nullptr;
+ }
+ RefPtr<DeleteNodeTransaction> deleteNodeTransaction =
+ DeleteNodeTransaction::MaybeCreate(*this, *editableContent);
+ NS_WARNING_ASSERTION(deleteNodeTransaction,
+ "DeleteNodeTransaction::MaybeCreate() failed");
+ return deleteNodeTransaction.forget();
+}
+
+bool EditorBase::FlushPendingNotificationsIfToHandleDeletionWithFrameSelection(
+ nsIEditor::EDirection aDirectionAndAmount) const {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ if (NS_WARN_IF(Destroyed())) {
+ return false;
+ }
+ if (!EditorUtils::IsFrameSelectionRequiredToExtendSelection(
+ aDirectionAndAmount, SelectionRef())) {
+ return true;
+ }
+ // Although AutoRangeArray::ExtendAnchorFocusRangeFor() will use
+ // nsFrameSelection, if it still has dirty frame, nsFrameSelection doesn't
+ // extend selection since we block script.
+ if (RefPtr<PresShell> presShell = GetPresShell()) {
+ presShell->FlushPendingNotifications(FlushType::Layout);
+ if (NS_WARN_IF(Destroyed())) {
+ return false;
+ }
+ }
+ return true;
+}
+
+nsresult EditorBase::DeleteSelectionAsAction(
+ nsIEditor::EDirection aDirectionAndAmount,
+ nsIEditor::EStripWrappers aStripWrappers, nsIPrincipal* aPrincipal) {
+ MOZ_ASSERT(aStripWrappers == eStrip || aStripWrappers == eNoStrip);
+ // Showing this assertion is fine if this method is called by outside via
+ // mutation event listener or something. Otherwise, this is called by
+ // wrong method.
+ NS_ASSERTION(
+ !mPlaceholderBatch,
+ "Should be called only when this is the only edit action of the "
+ "operation unless mutation event listener nests some operations");
+
+ // If we're a TextEditor instance, we don't need to treat parent elements
+ // so that we can ignore aStripWrappers for skipping unnecessary cost.
+ if (IsTextEditor()) {
+ aStripWrappers = nsIEditor::eNoStrip;
+ }
+
+ EditAction editAction = EditAction::eDeleteSelection;
+ switch (aDirectionAndAmount) {
+ case nsIEditor::ePrevious:
+ editAction = EditAction::eDeleteBackward;
+ break;
+ case nsIEditor::eNext:
+ editAction = EditAction::eDeleteForward;
+ break;
+ case nsIEditor::ePreviousWord:
+ editAction = EditAction::eDeleteWordBackward;
+ break;
+ case nsIEditor::eNextWord:
+ editAction = EditAction::eDeleteWordForward;
+ break;
+ case nsIEditor::eToBeginningOfLine:
+ editAction = EditAction::eDeleteToBeginningOfSoftLine;
+ break;
+ case nsIEditor::eToEndOfLine:
+ editAction = EditAction::eDeleteToEndOfSoftLine;
+ break;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, editAction, aPrincipal);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ // If there is an existing selection when an extended delete is requested,
+ // platforms that use "caret-style" caret positioning collapse the
+ // selection to the start and then create a new selection.
+ // Platforms that use "selection-style" caret positioning just delete the
+ // existing selection without extending it.
+ if (!SelectionRef().IsCollapsed()) {
+ switch (aDirectionAndAmount) {
+ case eNextWord:
+ case ePreviousWord:
+ case eToBeginningOfLine:
+ case eToEndOfLine: {
+ if (mCaretStyle != 1) {
+ aDirectionAndAmount = eNone;
+ break;
+ }
+ ErrorResult error;
+ SelectionRef().CollapseToStart(error);
+ if (NS_WARN_IF(Destroyed())) {
+ error.SuppressException();
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_DESTROYED);
+ }
+ if (error.Failed()) {
+ NS_WARNING("Selection::CollapseToStart() failed");
+ editActionData.Abort();
+ return EditorBase::ToGenericNSResult(error.StealNSResult());
+ }
+ break;
+ }
+ default:
+ break;
+ }
+ }
+
+ // If Selection is still NOT collapsed, it does not important removing
+ // range of the operation since we'll remove the selected content. However,
+ // information of direction (backward or forward) may be important for
+ // web apps. E.g., web apps may want to mark selected range as "deleted"
+ // and move caret before or after the range. Therefore, we should forget
+ // only the range information but keep range information. See discussion
+ // of the spec issue for the detail:
+ // https://github.com/w3c/input-events/issues/82
+ if (!SelectionRef().IsCollapsed()) {
+ switch (editAction) {
+ case EditAction::eDeleteWordBackward:
+ case EditAction::eDeleteToBeginningOfSoftLine:
+ editActionData.UpdateEditAction(EditAction::eDeleteBackward);
+ break;
+ case EditAction::eDeleteWordForward:
+ case EditAction::eDeleteToEndOfSoftLine:
+ editActionData.UpdateEditAction(EditAction::eDeleteForward);
+ break;
+ default:
+ break;
+ }
+ }
+
+ editActionData.SetSelectionCreatedByDoubleclick(
+ SelectionRef().GetFrameSelection() &&
+ SelectionRef().GetFrameSelection()->IsDoubleClickSelection());
+
+ if (!FlushPendingNotificationsIfToHandleDeletionWithFrameSelection(
+ aDirectionAndAmount)) {
+ NS_WARNING("Flusing pending notifications caused destroying the editor");
+ editActionData.Abort();
+ return EditorBase::ToGenericNSResult(NS_ERROR_EDITOR_DESTROYED);
+ }
+
+ nsresult rv =
+ editActionData.MaybeDispatchBeforeInputEvent(aDirectionAndAmount);
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "MaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ // delete placeholder txns merge.
+ AutoPlaceholderBatch treatAsOneTransaction(*this, *nsGkAtoms::DeleteTxnName,
+ ScrollSelectionIntoView::Yes,
+ __FUNCTION__);
+ rv = DeleteSelectionAsSubAction(aDirectionAndAmount, aStripWrappers);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::DeleteSelectionAsSubAction() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::DeleteSelectionAsSubAction(
+ nsIEditor::EDirection aDirectionAndAmount,
+ nsIEditor::EStripWrappers aStripWrappers) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ // If handling edit action is for table editing, this may be called with
+ // selecting an any table element by the caller, but it's not usual work of
+ // this so that `MayEditActionDeleteSelection()` returns false.
+ MOZ_ASSERT(MayEditActionDeleteSelection(GetEditAction()) ||
+ IsEditActionTableEditing(GetEditAction()));
+ MOZ_ASSERT(mPlaceholderBatch);
+ MOZ_ASSERT(aStripWrappers == eStrip || aStripWrappers == eNoStrip);
+ NS_ASSERTION(IsHTMLEditor() || aStripWrappers == nsIEditor::eNoStrip,
+ "TextEditor does not support strip wrappers");
+
+ if (NS_WARN_IF(!mInitSucceeded)) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ IgnoredErrorResult ignoredError;
+ AutoEditSubActionNotifier startToHandleEditSubAction(
+ *this, EditSubAction::eDeleteSelectedContent, aDirectionAndAmount,
+ ignoredError);
+ if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ return ignoredError.StealNSResult();
+ }
+ NS_WARNING_ASSERTION(
+ !ignoredError.Failed(),
+ "TextEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored");
+
+ {
+ Result<EditActionResult, nsresult> result =
+ HandleDeleteSelection(aDirectionAndAmount, aStripWrappers);
+ if (MOZ_UNLIKELY(result.isErr())) {
+ NS_WARNING("TextEditor::HandleDeleteSelection() failed");
+ return result.unwrapErr();
+ }
+ if (result.inspect().Canceled()) {
+ return NS_OK;
+ }
+ }
+
+ // XXX This is odd. We just tries to remove empty text node here but we
+ // refer `Selection`. It may be modified by mutation event listeners
+ // so that we should remove the empty text node when we make it empty.
+ const auto atNewStartOfSelection =
+ GetFirstSelectionStartPoint<EditorDOMPoint>();
+ if (NS_WARN_IF(!atNewStartOfSelection.IsSet())) {
+ // XXX And also it seems that we don't need to return error here.
+ // Why don't we just ignore? `Selection::RemoveAllRanges()` may
+ // have been called by mutation event listeners.
+ return NS_ERROR_FAILURE;
+ }
+ if (IsHTMLEditor() && atNewStartOfSelection.IsInTextNode() &&
+ !atNewStartOfSelection.GetContainer()->Length()) {
+ nsresult rv = DeleteNodeWithTransaction(
+ MOZ_KnownLive(*atNewStartOfSelection.ContainerAs<Text>()));
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::DeleteNodeWithTransaction() failed");
+ return rv;
+ }
+ }
+
+ // XXX I don't think that this is necessary in anonymous `<div>` element of
+ // TextEditor since there should be at most one text node and at most
+ // one padding `<br>` element so that `<br>` element won't be before
+ // caret.
+ if (!TopLevelEditSubActionDataRef().mDidExplicitlySetInterLine) {
+ // We prevent the caret from sticking on the left of previous `<br>`
+ // element (i.e. the end of previous line) after this deletion. Bug 92124.
+ if (MOZ_UNLIKELY(NS_FAILED(SelectionRef().SetInterlinePosition(
+ InterlinePosition::StartOfNextLine)))) {
+ NS_WARNING(
+ "Selection::SetInterlinePosition(InterlinePosition::StartOfNextLine) "
+ "failed");
+ return NS_ERROR_FAILURE; // Don't need to return NS_ERROR_NOT_INITIALIZED
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult EditorBase::HandleDropEvent(DragEvent* aDropEvent) {
+ if (NS_WARN_IF(!aDropEvent)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ DebugOnly<nsresult> rvIgnored = CommitComposition();
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "EditorBase::CommitComposition() failed, but ignored");
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eDrop);
+ // We need to initialize data or dataTransfer later. Therefore, we cannot
+ // dispatch "beforeinput" event until then.
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ RefPtr<DataTransfer> dataTransfer = aDropEvent->GetDataTransfer();
+ if (NS_WARN_IF(!dataTransfer)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ nsCOMPtr<nsIDragSession> dragSession = nsContentUtils::GetDragSession();
+ if (NS_WARN_IF(!dragSession)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ nsCOMPtr<nsINode> sourceNode = dataTransfer->GetMozSourceNode();
+
+ // If there is no source document, then the drag was from another application
+ // or another process (such as an out of process subframe). The latter case is
+ // not currently handled below when checking for a move/copy and deleting the
+ // existing text.
+ RefPtr<Document> srcdoc;
+ if (sourceNode) {
+ srcdoc = sourceNode->OwnerDoc();
+ }
+
+ nsCOMPtr<nsIPrincipal> sourcePrincipal;
+ dragSession->GetTriggeringPrincipal(getter_AddRefs(sourcePrincipal));
+
+ if (nsContentUtils::CheckForSubFrameDrop(
+ dragSession, aDropEvent->WidgetEventPtr()->AsDragEvent())) {
+ // Don't allow drags from subframe documents with different origins than
+ // the drop destination.
+ if (IsSafeToInsertData(sourcePrincipal) == SafeToInsertData::No) {
+ return NS_OK;
+ }
+ }
+
+ // Current doc is destination
+ RefPtr<Document> document = GetDocument();
+ if (NS_WARN_IF(!document)) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ const uint32_t numItems = dataTransfer->MozItemCount();
+ if (NS_WARN_IF(!numItems)) {
+ return NS_ERROR_FAILURE; // Nothing to drop?
+ }
+
+ // We have to figure out whether to delete and relocate caret only once
+ // Parent and offset are under the mouse cursor.
+ int32_t dropOffset = -1;
+ nsCOMPtr<nsIContent> dropParentContent =
+ aDropEvent->GetRangeParentContentAndOffset(&dropOffset);
+ if (dropOffset < 0) {
+ NS_WARNING(
+ "DropEvent::GetRangeParentContentAndOffset() returned negative offset");
+ return NS_ERROR_FAILURE;
+ }
+ EditorDOMPoint droppedAt(dropParentContent,
+ AssertedCast<uint32_t>(dropOffset));
+ if (NS_WARN_IF(!droppedAt.IsInContentNode())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Check if dropping into a selected range. If so and the source comes from
+ // same document, jump through some hoops to determine if mouse is over
+ // selection (bail) and whether user wants to copy selection or delete it.
+ if (sourceNode && sourceNode->IsEditable() && srcdoc == document) {
+ bool isPointInSelection = EditorUtils::IsPointInSelection(
+ SelectionRef(), *droppedAt.GetContainer(), droppedAt.Offset());
+ if (isPointInSelection) {
+ // If source document and destination document is same and dropping
+ // into one of selected ranges, we don't need to do nothing.
+ // XXX If the source comes from outside of this editor, this check
+ // means that we don't allow to drop the item in the selected
+ // range. However, the selection is hidden until the <input> or
+ // <textarea> gets focus, therefore, this looks odd.
+ return NS_OK;
+ }
+ }
+
+ // Delete if user doesn't want to copy when user moves selected content
+ // to different place in same editor.
+ // XXX Do we need the check whether it's in same document or not?
+ RefPtr<EditorBase> editorToDeleteSelection;
+ if (sourceNode && sourceNode->IsEditable() && srcdoc == document) {
+ if ((dataTransfer->DropEffectInt() &
+ nsIDragService::DRAGDROP_ACTION_MOVE) &&
+ !(dataTransfer->DropEffectInt() &
+ nsIDragService::DRAGDROP_ACTION_COPY)) {
+ // If the source node is in native anonymous tree, it must be in
+ // <input> or <textarea> element. If so, its TextEditor can remove it.
+ if (sourceNode->IsInNativeAnonymousSubtree()) {
+ if (RefPtr textControlElement = TextControlElement::FromNodeOrNull(
+ sourceNode
+ ->GetClosestNativeAnonymousSubtreeRootParentOrHost())) {
+ editorToDeleteSelection = textControlElement->GetTextEditor();
+ }
+ }
+ // Otherwise, must be the content is in HTMLEditor.
+ else if (IsHTMLEditor()) {
+ editorToDeleteSelection = this;
+ } else {
+ editorToDeleteSelection =
+ nsContentUtils::GetHTMLEditor(srcdoc->GetPresContext());
+ }
+ }
+ // If the found editor isn't modifiable, we should not try to delete
+ // selection.
+ if (editorToDeleteSelection && !editorToDeleteSelection->IsModifiable()) {
+ editorToDeleteSelection = nullptr;
+ }
+ // If the found editor has collapsed selection, we need to delete nothing
+ // in the editor.
+ if (editorToDeleteSelection) {
+ if (Selection* selection = editorToDeleteSelection->GetSelection()) {
+ if (selection->IsCollapsed()) {
+ editorToDeleteSelection = nullptr;
+ }
+ }
+ }
+ }
+
+ if (IsInPlaintextMode()) {
+ for (nsIContent* content = droppedAt.ContainerAs<nsIContent>(); content;
+ content = content->GetParent()) {
+ nsCOMPtr<nsIFormControl> formControl(do_QueryInterface(content));
+ if (formControl && !formControl->AllowDrop()) {
+ // Don't allow dropping into a form control that doesn't allow being
+ // dropped into.
+ return NS_OK;
+ }
+ }
+ }
+
+ // Combine any deletion and drop insertion into one transaction.
+ AutoPlaceholderBatch treatAsOneTransaction(
+ *this, ScrollSelectionIntoView::Yes, __FUNCTION__);
+
+ // Don't dispatch "selectionchange" event until inserting all contents.
+ SelectionBatcher selectionBatcher(SelectionRef(), __FUNCTION__);
+
+ // Track dropped point with nsRange because we shouldn't insert the
+ // dropped content into different position even if some event listeners
+ // modify selection. Note that Chrome's behavior is really odd. So,
+ // we don't need to worry about web-compat about this.
+ IgnoredErrorResult ignoredError;
+ RefPtr<nsRange> rangeAtDropPoint =
+ nsRange::Create(droppedAt.ToRawRangeBoundary(),
+ droppedAt.ToRawRangeBoundary(), ignoredError);
+ if (NS_WARN_IF(ignoredError.Failed()) ||
+ NS_WARN_IF(!rangeAtDropPoint->IsPositioned())) {
+ editActionData.Abort();
+ return NS_ERROR_FAILURE;
+ }
+
+ // Remove selected contents first here because we need to fire a pair of
+ // "beforeinput" and "input" for deletion and web apps can cancel only
+ // this deletion. Note that callee may handle insertion asynchronously.
+ // Therefore, it is the best to remove selected content here.
+ if (editorToDeleteSelection) {
+ nsresult rv = editorToDeleteSelection->DeleteSelectionByDragAsAction(
+ mDispatchInputEvent);
+ if (NS_WARN_IF(Destroyed())) {
+ editActionData.Abort();
+ return NS_OK;
+ }
+ // Ignore the editor instance specific error if it's another editor.
+ if (this != editorToDeleteSelection &&
+ (rv == NS_ERROR_NOT_INITIALIZED || rv == NS_ERROR_EDITOR_DESTROYED)) {
+ rv = NS_OK;
+ }
+ // Don't cancel "insertFromDrop" even if "deleteByDrag" is canceled.
+ if (rv != NS_ERROR_EDITOR_ACTION_CANCELED && NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::DeleteSelectionByDragAsAction() failed");
+ editActionData.Abort();
+ return EditorBase::ToGenericNSResult(rv);
+ }
+ if (NS_WARN_IF(!rangeAtDropPoint->IsPositioned()) ||
+ NS_WARN_IF(!rangeAtDropPoint->GetStartContainer()->IsContent())) {
+ editActionData.Abort();
+ return NS_ERROR_FAILURE;
+ }
+ droppedAt = rangeAtDropPoint->StartRef();
+ MOZ_ASSERT(droppedAt.IsSetAndValid());
+ MOZ_ASSERT(droppedAt.IsInContentNode());
+ }
+
+ // Before inserting dropping content, we need to move focus for compatibility
+ // with Chrome and firing "beforeinput" event on new editing host.
+ RefPtr<Element> focusedElement, newFocusedElement;
+ if (IsTextEditor()) {
+ newFocusedElement = GetExposedRoot();
+ focusedElement = IsActiveInDOMWindow() ? newFocusedElement : nullptr;
+ }
+ // TODO: We need to add automated tests when dropping something into an
+ // editing host for contenteditable which is in a shadow DOM tree
+ // and its host which is in design mode.
+ else if (!AsHTMLEditor()->IsInDesignMode()) {
+ focusedElement = AsHTMLEditor()->ComputeEditingHost();
+ if (focusedElement &&
+ droppedAt.ContainerAs<nsIContent>()->IsInclusiveDescendantOf(
+ focusedElement)) {
+ newFocusedElement = focusedElement;
+ } else {
+ newFocusedElement = droppedAt.ContainerAs<nsIContent>()->GetEditingHost();
+ }
+ }
+ // Move selection right now. Note that this does not move focus because
+ // `Selection` moves focus with selection change only when the API caller is
+ // JS. And also this does not notify selection listeners (nor
+ // "selectionchange") since we created SelectionBatcher above.
+ ErrorResult error;
+ SelectionRef().SetStartAndEnd(droppedAt.ToRawRangeBoundary(),
+ droppedAt.ToRawRangeBoundary(), error);
+ if (error.Failed()) {
+ NS_WARNING("Selection::SetStartAndEnd() failed");
+ editActionData.Abort();
+ return error.StealNSResult();
+ }
+ if (NS_WARN_IF(Destroyed())) {
+ editActionData.Abort();
+ return NS_OK;
+ }
+ // Then, move focus if necessary. This must cause dispatching "blur" event
+ // and "focus" event.
+ if (newFocusedElement && focusedElement != newFocusedElement) {
+ RefPtr<nsFocusManager> fm = nsFocusManager::GetFocusManager();
+ DebugOnly<nsresult> rvIgnored = fm->SetFocus(newFocusedElement, 0);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "nsFocusManager::SetFocus() failed to set focus "
+ "to the element, but ignored");
+ if (NS_WARN_IF(Destroyed())) {
+ editActionData.Abort();
+ return NS_OK;
+ }
+ // "blur" or "focus" event listener may have changed the value.
+ // Let's keep using the original point.
+ if (NS_WARN_IF(!rangeAtDropPoint->IsPositioned()) ||
+ NS_WARN_IF(!rangeAtDropPoint->GetStartContainer()->IsContent())) {
+ return NS_ERROR_FAILURE;
+ }
+ droppedAt = rangeAtDropPoint->StartRef();
+ MOZ_ASSERT(droppedAt.IsSetAndValid());
+
+ // If focus is changed to different element and we're handling drop in
+ // contenteditable, we cannot handle it without focus. So, we should give
+ // it up.
+ if (IsHTMLEditor() && !AsHTMLEditor()->IsInDesignMode() &&
+ NS_WARN_IF(newFocusedElement != AsHTMLEditor()->ComputeEditingHost())) {
+ editActionData.Abort();
+ return NS_OK;
+ }
+ }
+
+ nsresult rv = InsertDroppedDataTransferAsAction(editActionData, *dataTransfer,
+ droppedAt, sourcePrincipal);
+ if (rv == NS_ERROR_EDITOR_DESTROYED ||
+ rv == NS_ERROR_EDITOR_ACTION_CANCELED) {
+ return EditorBase::ToGenericNSResult(rv);
+ }
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::InsertDroppedDataTransferAsAction() failed, but ignored");
+
+ rv = ScrollSelectionFocusIntoView();
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::ScrollSelectionFocusIntoView() failed");
+ return rv;
+}
+
+nsresult EditorBase::DeleteSelectionByDragAsAction(bool aDispatchInputEvent) {
+ // TODO: Move this method to `EditorBase`.
+ AutoRestore<bool> saveDispatchInputEvent(mDispatchInputEvent);
+ mDispatchInputEvent = aDispatchInputEvent;
+ // Even if we're handling "deleteByDrag" in same editor as "insertFromDrop",
+ // we need to recreate edit action data here because
+ // `AutoEditActionDataSetter` needs to manage event state separately.
+ bool requestedByAnotherEditor = GetEditAction() != EditAction::eDrop;
+ AutoEditActionDataSetter editActionData(*this, EditAction::eDeleteByDrag);
+ MOZ_ASSERT(!SelectionRef().IsCollapsed());
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return rv;
+ }
+ // But keep using placeholder transaction for "insertFromDrop" if there is.
+ Maybe<AutoPlaceholderBatch> treatAsOneTransaction;
+ if (requestedByAnotherEditor) {
+ treatAsOneTransaction.emplace(*this, ScrollSelectionIntoView::Yes,
+ __FUNCTION__);
+ }
+
+ // We may need to update the source node to dispatch "dragend" below.
+ // Chrome restricts the new target under the <body> here. Therefore, we
+ // should follow it here.
+ // https://source.chromium.org/chromium/chromium/src/+/main:third_party/blink/renderer/core/editing/editing_utilities.cc;l=254;drc=da35f4ed6398ae287d5adc828b9546eec95f668a
+ const RefPtr<Element> editingHost =
+ IsHTMLEditor() ? AsHTMLEditor()->ComputeEditingHost(
+ HTMLEditor::LimitInBodyElement::Yes)
+ : nullptr;
+
+ rv = DeleteSelectionAsSubAction(nsIEditor::eNone, IsTextEditor()
+ ? nsIEditor::eNoStrip
+ : nsIEditor::eStrip);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::DeleteSelectionAsSubAction(eNone) failed");
+ return rv;
+ }
+
+ if (!mDispatchInputEvent) {
+ return NS_OK;
+ }
+
+ if (treatAsOneTransaction.isNothing()) {
+ DispatchInputEvent();
+ }
+
+ if (NS_WARN_IF(Destroyed())) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+
+ // If we success everything here, we may need to retarget "dragend" event
+ // target for compatibility with the other browsers. They do this only when
+ // their builtin editor delete the source node from the document. Then,
+ // they retarget the source node to the editing host.
+ // https://source.chromium.org/chromium/chromium/src/+/main:third_party/blink/renderer/core/page/drag_controller.cc;l=724;drc=d9ba13b8cd8ac0faed7afc3d1f7e4b67ebac2a0b
+ if (editingHost) {
+ if (nsCOMPtr<nsIDragService> dragService =
+ do_GetService("@mozilla.org/widget/dragservice;1")) {
+ dragService->MaybeEditorDeletedSourceNode(editingHost);
+ }
+ }
+ return NS_WARN_IF(Destroyed()) ? NS_ERROR_EDITOR_DESTROYED : NS_OK;
+}
+
+nsresult EditorBase::DeleteSelectionWithTransaction(
+ nsIEditor::EDirection aDirectionAndAmount,
+ nsIEditor::EStripWrappers aStripWrappers) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(aStripWrappers == eStrip || aStripWrappers == eNoStrip);
+ if (NS_WARN_IF(Destroyed())) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+
+ AutoRangeArray rangesToDelete(SelectionRef());
+ if (NS_WARN_IF(rangesToDelete.Ranges().IsEmpty())) {
+ NS_ASSERTION(
+ false,
+ "For avoiding to throw incompatible exception for `execCommand`, fix "
+ "the caller");
+ return NS_ERROR_FAILURE;
+ }
+
+ if (IsTextEditor()) {
+ if (const Text* theTextNode = AsTextEditor()->GetTextNode()) {
+ rangesToDelete.EnsureRangesInTextNode(*theTextNode);
+ }
+ }
+
+ Result<CaretPoint, nsresult> caretPointOrError = DeleteRangesWithTransaction(
+ aDirectionAndAmount, aStripWrappers, rangesToDelete);
+ if (MOZ_UNLIKELY(caretPointOrError.isErr())) {
+ NS_WARNING("EditorBase::DeleteRangesWithTransaction() failed");
+ return caretPointOrError.unwrapErr();
+ }
+ nsresult rv = caretPointOrError.inspect().SuggestCaretPointTo(
+ *this, {SuggestCaret::OnlyIfHasSuggestion,
+ SuggestCaret::OnlyIfTransactionsAllowedToDoIt,
+ SuggestCaret::AndIgnoreTrivialError});
+ if (NS_FAILED(rv)) {
+ NS_WARNING("CaretPoint::SuggestCaretPointTo() failed");
+ }
+ NS_WARNING_ASSERTION(rv != NS_SUCCESS_EDITOR_BUT_IGNORED_TRIVIAL_ERROR,
+ "CaretPoint::SuggestCaretPointTo() failed, but ignored");
+ return NS_OK;
+}
+
+Result<CaretPoint, nsresult> EditorBase::DeleteRangesWithTransaction(
+ nsIEditor::EDirection aDirectionAndAmount,
+ nsIEditor::EStripWrappers aStripWrappers,
+ const AutoRangeArray& aRangesToDelete) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(!Destroyed());
+ MOZ_ASSERT(aStripWrappers == eStrip || aStripWrappers == eNoStrip);
+ MOZ_ASSERT(!aRangesToDelete.Ranges().IsEmpty());
+
+ HowToHandleCollapsedRange howToHandleCollapsedRange =
+ EditorBase::HowToHandleCollapsedRangeFor(aDirectionAndAmount);
+ if (NS_WARN_IF(aRangesToDelete.IsCollapsed() &&
+ howToHandleCollapsedRange ==
+ HowToHandleCollapsedRange::Ignore)) {
+ NS_ASSERTION(
+ false,
+ "For avoiding to throw incompatible exception for `execCommand`, fix "
+ "the caller");
+ return Err(NS_ERROR_FAILURE);
+ }
+
+ RefPtr<DeleteMultipleRangesTransaction> deleteSelectionTransaction =
+ CreateTransactionForDeleteSelection(howToHandleCollapsedRange,
+ aRangesToDelete);
+ if (MOZ_UNLIKELY(!deleteSelectionTransaction)) {
+ NS_WARNING("EditorBase::CreateTransactionForDeleteSelection() failed");
+ return Err(NS_ERROR_FAILURE);
+ }
+
+ // XXX This is odd, this assumes that there are no multiple collapsed
+ // ranges in `Selection`, but it's possible scenario.
+ // XXX This loop looks slow, but it's rarely so because of multiple
+ // selection is not used so many times.
+ nsCOMPtr<nsIContent> deleteContent;
+ uint32_t deleteCharOffset = 0;
+ for (const OwningNonNull<EditTransactionBase>& transactionBase :
+ Reversed(deleteSelectionTransaction->ChildTransactions())) {
+ if (DeleteTextTransaction* deleteTextTransaction =
+ transactionBase->GetAsDeleteTextTransaction()) {
+ deleteContent = deleteTextTransaction->GetText();
+ deleteCharOffset = deleteTextTransaction->Offset();
+ break;
+ }
+ if (DeleteNodeTransaction* deleteNodeTransaction =
+ transactionBase->GetAsDeleteNodeTransaction()) {
+ deleteContent = deleteNodeTransaction->GetContent();
+ break;
+ }
+ }
+
+ RefPtr<CharacterData> deleteCharData =
+ CharacterData::FromNodeOrNull(deleteContent);
+ IgnoredErrorResult ignoredError;
+ AutoEditSubActionNotifier startToHandleEditSubAction(
+ *this, EditSubAction::eDeleteSelectedContent, aDirectionAndAmount,
+ ignoredError);
+ if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ return Err(ignoredError.StealNSResult());
+ }
+ NS_WARNING_ASSERTION(
+ !ignoredError.Failed(),
+ "TextEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored");
+
+ if (IsHTMLEditor()) {
+ if (!deleteContent) {
+ // XXX We may remove multiple ranges in the following. Therefore,
+ // this must have a bug since we only add the first range into
+ // the changed range.
+ TopLevelEditSubActionDataRef().WillDeleteRange(
+ *this, aRangesToDelete.GetFirstRangeStartPoint<EditorRawDOMPoint>(),
+ aRangesToDelete.GetFirstRangeEndPoint<EditorRawDOMPoint>());
+ } else if (!deleteCharData) {
+ TopLevelEditSubActionDataRef().WillDeleteContent(*this, *deleteContent);
+ }
+ }
+
+ // Notify nsIEditActionListener::WillDelete[Selection|Text]
+ if (!mActionListeners.IsEmpty()) {
+ if (!deleteContent) {
+ MOZ_ASSERT(!aRangesToDelete.Ranges().IsEmpty());
+ AutoTArray<RefPtr<nsRange>, 8> rangesToDelete(
+ aRangesToDelete.CloneRanges<RefPtr>());
+ AutoActionListenerArray listeners(mActionListeners.Clone());
+ for (auto& listener : listeners) {
+ DebugOnly<nsresult> rvIgnored =
+ listener->WillDeleteRanges(rangesToDelete);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsIEditActionListener::WillDeleteRanges() failed, but ignored");
+ MOZ_DIAGNOSTIC_ASSERT(!Destroyed(),
+ "nsIEditActionListener::WillDeleteRanges() "
+ "must not destroy the editor");
+ }
+ } else if (deleteCharData) {
+ AutoActionListenerArray listeners(mActionListeners.Clone());
+ for (auto& listener : listeners) {
+ // XXX Why don't we notify listeners of actual length?
+ DebugOnly<nsresult> rvIgnored =
+ listener->WillDeleteText(deleteCharData, deleteCharOffset, 1);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsIEditActionListener::WillDeleteText() failed, but ignored");
+ MOZ_DIAGNOSTIC_ASSERT(!Destroyed(),
+ "nsIEditActionListener::WillDeleteText() must "
+ "not destroy the editor");
+ }
+ }
+ }
+
+ // Delete the specified amount
+ nsresult rv = DoTransactionInternal(deleteSelectionTransaction);
+ // I'm not sure whether we should keep notifying edit action listeners or
+ // stop doing it. For now, just keep traditional behavior.
+ bool destroyedByTransaction = Destroyed();
+ NS_WARNING_ASSERTION(destroyedByTransaction || NS_SUCCEEDED(rv),
+ "EditorBase::DoTransactionInternal() failed");
+
+ if (IsHTMLEditor() && deleteCharData) {
+ MOZ_ASSERT(deleteContent);
+ TopLevelEditSubActionDataRef().DidDeleteText(
+ *this, EditorRawDOMPoint(deleteContent));
+ }
+
+ if (mTextServicesDocument && NS_SUCCEEDED(rv) && deleteContent &&
+ !deleteCharData) {
+ RefPtr<TextServicesDocument> textServicesDocument = mTextServicesDocument;
+ textServicesDocument->DidDeleteContent(*deleteContent);
+ MOZ_ASSERT(
+ destroyedByTransaction || !Destroyed(),
+ "TextServicesDocument::DidDeleteContent() must not destroy the editor");
+ }
+
+ if (!mActionListeners.IsEmpty() && deleteContent && !deleteCharData) {
+ for (auto& listener : mActionListeners.Clone()) {
+ DebugOnly<nsresult> rvIgnored =
+ listener->DidDeleteNode(deleteContent, rv);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsIEditActionListener::DidDeleteNode() failed, but ignored");
+ MOZ_DIAGNOSTIC_ASSERT(
+ destroyedByTransaction || !Destroyed(),
+ "nsIEditActionListener::DidDeleteNode() must not destroy the editor");
+ }
+ }
+
+ if (NS_WARN_IF(destroyedByTransaction)) {
+ return Err(NS_ERROR_EDITOR_DESTROYED);
+ }
+ if (NS_FAILED(rv)) {
+ return Err(rv);
+ }
+
+ EditorDOMPoint pointToPutCaret =
+ deleteSelectionTransaction->SuggestPointToPutCaret();
+ if (IsHTMLEditor() && aStripWrappers == nsIEditor::eStrip) {
+ const nsCOMPtr<nsIContent> anchorContent =
+ pointToPutCaret.GetContainerAs<nsIContent>();
+ if (MOZ_LIKELY(anchorContent) &&
+ MOZ_LIKELY(HTMLEditUtils::IsSimplyEditableNode(*anchorContent)) &&
+ // FIXME: Perhaps, this should use `HTMLEditor::IsEmptyNode` instead.
+ !anchorContent->Length()) {
+ AutoTrackDOMPoint trackPoint(RangeUpdaterRef(), &pointToPutCaret);
+ nsresult rv =
+ MOZ_KnownLive(AsHTMLEditor())
+ ->RemoveEmptyInclusiveAncestorInlineElements(*anchorContent);
+ if (NS_FAILED(rv)) {
+ NS_WARNING(
+ "HTMLEditor::RemoveEmptyInclusiveAncestorInlineElements() "
+ "failed");
+ return Err(rv);
+ }
+ }
+ }
+
+ return CaretPoint(std::move(pointToPutCaret));
+}
+
+already_AddRefed<Element> EditorBase::CreateHTMLContent(
+ const nsAtom* aTag) const {
+ MOZ_ASSERT(aTag);
+
+ RefPtr<Document> document = GetDocument();
+ if (NS_WARN_IF(!document)) {
+ return nullptr;
+ }
+
+ // XXX Wallpaper over editor bug (editor tries to create elements with an
+ // empty nodename).
+ if (aTag == nsGkAtoms::_empty) {
+ NS_ERROR(
+ "Don't pass an empty tag to EditorBase::CreateHTMLContent, "
+ "check caller.");
+ return nullptr;
+ }
+
+ return document->CreateElem(nsDependentAtomString(aTag), nullptr,
+ kNameSpaceID_XHTML);
+}
+
+already_AddRefed<nsTextNode> EditorBase::CreateTextNode(
+ const nsAString& aData) const {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ Document* document = GetDocument();
+ if (NS_WARN_IF(!document)) {
+ return nullptr;
+ }
+ RefPtr<nsTextNode> text = document->CreateEmptyTextNode();
+ text->MarkAsMaybeModifiedFrequently();
+ if (IsPasswordEditor()) {
+ text->MarkAsMaybeMasked();
+ }
+ // Don't notify; this node is still being created.
+ DebugOnly<nsresult> rvIgnored = text->SetText(aData, false);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "Text::SetText() failed, but ignored");
+ return text.forget();
+}
+
+NS_IMETHODIMP EditorBase::SetAttributeOrEquivalent(Element* aElement,
+ const nsAString& aAttribute,
+ const nsAString& aValue,
+ bool aSuppressTransaction) {
+ if (NS_WARN_IF(!aElement)) {
+ return NS_ERROR_NULL_POINTER;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eSetAttribute);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ RefPtr<nsAtom> attribute = NS_Atomize(aAttribute);
+ rv = SetAttributeOrEquivalent(aElement, attribute, aValue,
+ aSuppressTransaction);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::SetAttributeOrEquivalent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+NS_IMETHODIMP EditorBase::RemoveAttributeOrEquivalent(
+ Element* aElement, const nsAString& aAttribute, bool aSuppressTransaction) {
+ if (NS_WARN_IF(!aElement)) {
+ return NS_ERROR_NULL_POINTER;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eRemoveAttribute);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ RefPtr<nsAtom> attribute = NS_Atomize(aAttribute);
+ rv = RemoveAttributeOrEquivalent(aElement, attribute, aSuppressTransaction);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::RemoveAttributeOrEquivalent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+void EditorBase::HandleKeyPressEventInReadOnlyMode(
+ WidgetKeyboardEvent& aKeyboardEvent) const {
+ MOZ_ASSERT(IsReadonly());
+ MOZ_ASSERT(aKeyboardEvent.mMessage == eKeyPress);
+
+ switch (aKeyboardEvent.mKeyCode) {
+ case NS_VK_BACK:
+ // If it's a `Backspace` key, let's consume it because it may be mapped
+ // to "Back" of the history navigation. So, it's possible that user
+ // tries to delete a character with `Backspace` even in the read-only
+ // editor.
+ aKeyboardEvent.PreventDefault();
+ break;
+ }
+ // XXX How about space key (page up and page down in browser navigation)?
+}
+
+nsresult EditorBase::HandleKeyPressEvent(WidgetKeyboardEvent* aKeyboardEvent) {
+ MOZ_ASSERT(!IsReadonly());
+ MOZ_ASSERT(aKeyboardEvent);
+ MOZ_ASSERT(aKeyboardEvent->mMessage == eKeyPress);
+
+ // NOTE: When you change this method, you should also change:
+ // * editor/libeditor/tests/test_texteditor_keyevent_handling.html
+ // * editor/libeditor/tests/test_htmleditor_keyevent_handling.html
+ //
+ // And also when you add new key handling, you need to change the subclass's
+ // HandleKeyPressEvent()'s switch statement.
+
+ switch (aKeyboardEvent->mKeyCode) {
+ case NS_VK_META:
+ case NS_VK_WIN:
+ case NS_VK_SHIFT:
+ case NS_VK_CONTROL:
+ case NS_VK_ALT:
+ MOZ_ASSERT_UNREACHABLE(
+ "eKeyPress event shouldn't be fired for modifier keys");
+ return NS_ERROR_UNEXPECTED;
+
+ case NS_VK_BACK: {
+ if (aKeyboardEvent->IsControl() || aKeyboardEvent->IsAlt() ||
+ aKeyboardEvent->IsMeta() || aKeyboardEvent->IsOS()) {
+ return NS_OK;
+ }
+ DebugOnly<nsresult> rvIgnored =
+ DeleteSelectionAsAction(nsIEditor::ePrevious, nsIEditor::eStrip);
+ aKeyboardEvent->PreventDefault();
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::DeleteSelectionAsAction() failed, but ignored");
+ return NS_OK;
+ }
+ case NS_VK_DELETE: {
+ // on certain platforms (such as windows) the shift key
+ // modifies what delete does (cmd_cut in this case).
+ // bailing here to allow the keybindings to do the cut.
+ if (aKeyboardEvent->IsShift() || aKeyboardEvent->IsControl() ||
+ aKeyboardEvent->IsAlt() || aKeyboardEvent->IsMeta() ||
+ aKeyboardEvent->IsOS()) {
+ return NS_OK;
+ }
+ DebugOnly<nsresult> rvIgnored =
+ DeleteSelectionAsAction(nsIEditor::eNext, nsIEditor::eStrip);
+ aKeyboardEvent->PreventDefault();
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "EditorBase::DeleteSelectionAsAction() failed, but ignored");
+ return NS_OK;
+ }
+ }
+ return NS_OK;
+}
+
+nsresult EditorBase::OnInputText(const nsAString& aStringToInsert) {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eInsertText);
+ MOZ_ASSERT(!aStringToInsert.IsVoid());
+ editActionData.SetData(aStringToInsert);
+ // FYI: For conforming to current UI Events spec, we should dispatch
+ // "beforeinput" event before "keypress" event, but here is in a
+ // "keypress" event listener. However, the other browsers dispatch
+ // "beforeinput" event after "keypress" event. Therefore, it makes
+ // sense to follow the other browsers. Spec issue:
+ // https://github.com/w3c/uievents/issues/220
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ AutoPlaceholderBatch treatAsOneTransaction(*this, *nsGkAtoms::TypingTxnName,
+ ScrollSelectionIntoView::Yes,
+ __FUNCTION__);
+ rv = InsertTextAsSubAction(aStringToInsert, SelectionHandling::Delete);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::InsertTextAsSubAction() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::ReplaceTextAsAction(
+ const nsAString& aString, nsRange* aReplaceRange,
+ AllowBeforeInputEventCancelable aAllowBeforeInputEventCancelable,
+ nsIPrincipal* aPrincipal) {
+ MOZ_ASSERT(aString.FindChar(nsCRT::CR) == kNotFound);
+ MOZ_ASSERT_IF(!aReplaceRange, IsTextEditor());
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eReplaceText,
+ aPrincipal);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+ if (aAllowBeforeInputEventCancelable == AllowBeforeInputEventCancelable::No) {
+ editActionData.MakeBeforeInputEventNonCancelable();
+ }
+
+ if (IsTextEditor()) {
+ editActionData.SetData(aString);
+ } else {
+ editActionData.InitializeDataTransfer(aString);
+ RefPtr<StaticRange> targetRange;
+ if (aReplaceRange) {
+ // Compute offset of the range before dispatching `beforeinput` event
+ // because it may be referred after the DOM tree is changed and the
+ // range may have not computed the offset yet.
+ targetRange = StaticRange::Create(
+ aReplaceRange->GetStartContainer(), aReplaceRange->StartOffset(),
+ aReplaceRange->GetEndContainer(), aReplaceRange->EndOffset(),
+ IgnoreErrors());
+ NS_WARNING_ASSERTION(targetRange && targetRange->IsPositioned(),
+ "StaticRange::Create() failed");
+ } else {
+ Element* editingHost = AsHTMLEditor()->ComputeEditingHost();
+ NS_WARNING_ASSERTION(editingHost,
+ "No active editing host, no target ranges");
+ if (editingHost) {
+ targetRange = StaticRange::Create(
+ editingHost, 0, editingHost, editingHost->Length(), IgnoreErrors());
+ NS_WARNING_ASSERTION(targetRange && targetRange->IsPositioned(),
+ "StaticRange::Create() failed");
+ }
+ }
+ if (targetRange && targetRange->IsPositioned()) {
+ editActionData.AppendTargetRange(*targetRange);
+ }
+ }
+
+ nsresult rv = editActionData.MaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "MaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ AutoPlaceholderBatch treatAsOneTransaction(
+ *this, ScrollSelectionIntoView::Yes, __FUNCTION__);
+
+ // This should emulates inserting text for better undo/redo behavior.
+ IgnoredErrorResult ignoredError;
+ AutoEditSubActionNotifier startToHandleEditSubAction(
+ *this, EditSubAction::eInsertText, nsIEditor::eNext, ignoredError);
+ if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ return EditorBase::ToGenericNSResult(ignoredError.StealNSResult());
+ }
+ NS_WARNING_ASSERTION(
+ !ignoredError.Failed(),
+ "TextEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored");
+
+ if (!aReplaceRange) {
+ // Use fast path if we're `TextEditor` because it may be in a hot path.
+ if (IsTextEditor()) {
+ nsresult rv = MOZ_KnownLive(AsTextEditor())->SetTextAsSubAction(aString);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "TextEditor::SetTextAsSubAction() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ MOZ_ASSERT_UNREACHABLE("Setting value of `HTMLEditor` isn't supported");
+ return EditorBase::ToGenericNSResult(NS_ERROR_FAILURE);
+ }
+
+ if (aString.IsEmpty() && aReplaceRange->Collapsed()) {
+ NS_WARNING("Setting value was empty and replaced range was empty");
+ return NS_OK;
+ }
+
+ // Note that do not notify selectionchange caused by selecting all text
+ // because it's preparation of our delete implementation so web apps
+ // shouldn't receive such selectionchange before the first mutation.
+ AutoUpdateViewBatch preventSelectionChangeEvent(*this, __FUNCTION__);
+
+ // Select the range but as far as possible, we should not create new range
+ // even if it's part of special Selection.
+ ErrorResult error;
+ SelectionRef().RemoveAllRanges(error);
+ if (error.Failed()) {
+ NS_WARNING("Selection::RemoveAllRanges() failed");
+ return error.StealNSResult();
+ }
+ SelectionRef().AddRangeAndSelectFramesAndNotifyListeners(*aReplaceRange,
+ error);
+ if (error.Failed()) {
+ NS_WARNING("Selection::AddRangeAndSelectFramesAndNotifyListeners() failed");
+ return error.StealNSResult();
+ }
+
+ rv = ReplaceSelectionAsSubAction(aString);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::ReplaceSelectionAsSubAction() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::ReplaceSelectionAsSubAction(const nsAString& aString) {
+ if (aString.IsEmpty()) {
+ nsresult rv = DeleteSelectionAsSubAction(
+ nsIEditor::eNone,
+ IsTextEditor() ? nsIEditor::eNoStrip : nsIEditor::eStrip);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::DeleteSelectionAsSubAction(eNone) failed");
+ return rv;
+ }
+
+ nsresult rv = InsertTextAsSubAction(aString, SelectionHandling::Delete);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::InsertTextAsSubAction() failed");
+ return rv;
+}
+
+nsresult EditorBase::HandleInlineSpellCheck(
+ const EditorDOMPoint& aPreviouslySelectedStart,
+ const AbstractRange* aRange) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ if (!mInlineSpellChecker) {
+ return NS_OK;
+ }
+ nsresult rv = mInlineSpellChecker->SpellCheckAfterEditorChange(
+ GetTopLevelEditSubAction(), SelectionRef(),
+ aPreviouslySelectedStart.GetContainer(),
+ aPreviouslySelectedStart.Offset(),
+ aRange ? aRange->GetStartContainer() : nullptr,
+ aRange ? aRange->StartOffset() : 0,
+ aRange ? aRange->GetEndContainer() : nullptr,
+ aRange ? aRange->EndOffset() : 0);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "mozInlineSpellChecker::SpellCheckAfterEditorChange() failed");
+ return rv;
+}
+
+Element* EditorBase::FindSelectionRoot(const nsINode& aNode) const {
+ return GetRoot();
+}
+
+void EditorBase::InitializeSelectionAncestorLimit(
+ nsIContent& aAncestorLimit) const {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ SelectionRef().SetAncestorLimiter(&aAncestorLimit);
+}
+
+nsresult EditorBase::InitializeSelection(
+ const nsINode& aOriginalEventTargetNode) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ nsCOMPtr<nsIContent> selectionRootContent =
+ FindSelectionRoot(aOriginalEventTargetNode);
+ if (!selectionRootContent) {
+ return NS_OK;
+ }
+
+ nsCOMPtr<nsISelectionController> selectionController =
+ GetSelectionController();
+ if (NS_WARN_IF(!selectionController)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Init the caret
+ RefPtr<nsCaret> caret = GetCaret();
+ if (NS_WARN_IF(!caret)) {
+ return NS_ERROR_FAILURE;
+ }
+ caret->SetSelection(&SelectionRef());
+ DebugOnly<nsresult> rvIgnored =
+ selectionController->SetCaretReadOnly(IsReadonly());
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsISelectionController::SetCaretReadOnly() failed, but ignored");
+ rvIgnored = selectionController->SetCaretEnabled(true);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsISelectionController::SetCaretEnabled() failed, but ignored");
+ // NOTE(emilio): It's important for this call to be after
+ // SetCaretEnabled(true), since that would override mIgnoreUserModify to true.
+ //
+ // Also, make sure to always ignore it for designMode, since that effectively
+ // overrides everything and we allow to edit stuff with
+ // contenteditable="false" subtrees in such a document.
+ caret->SetIgnoreUserModify(aOriginalEventTargetNode.IsInDesignMode());
+
+ // Init selection
+ rvIgnored =
+ selectionController->SetSelectionFlags(nsISelectionDisplay::DISPLAY_ALL);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsISelectionController::SetSelectionFlags() failed, but ignored");
+
+ selectionController->SelectionWillTakeFocus();
+
+ // If the computed selection root isn't root content, we should set it
+ // as selection ancestor limit. However, if that is root element, it means
+ // there is not limitation of the selection, then, we must set nullptr.
+ // NOTE: If we set a root element to the ancestor limit, some selection
+ // methods don't work fine.
+ if (selectionRootContent->GetParent()) {
+ InitializeSelectionAncestorLimit(*selectionRootContent);
+ } else {
+ SelectionRef().SetAncestorLimiter(nullptr);
+ }
+
+ // If there is composition when this is called, we may need to restore IME
+ // selection because if the editor is reframed, this already forgot IME
+ // selection and the transaction.
+ if (mComposition && mComposition->IsMovingToNewTextNode()) {
+ // We need to look for the new text node from current selection.
+ // XXX If selection is changed during reframe, this doesn't work well!
+ const nsRange* firstRange = SelectionRef().GetRangeAt(0);
+ if (NS_WARN_IF(!firstRange)) {
+ return NS_ERROR_FAILURE;
+ }
+ EditorRawDOMPoint atStartOfFirstRange(firstRange->StartRef());
+ EditorRawDOMPoint betterInsertionPoint =
+ FindBetterInsertionPoint(atStartOfFirstRange);
+ RefPtr<Text> textNode = betterInsertionPoint.GetContainerAs<Text>();
+ MOZ_ASSERT(textNode,
+ "There must be text node if composition string is not empty");
+ if (textNode) {
+ MOZ_ASSERT(textNode->Length() >= mComposition->XPEndOffsetInTextNode(),
+ "The text node must be different from the old text node");
+ RefPtr<TextRangeArray> ranges = mComposition->GetRanges();
+ DebugOnly<nsresult> rvIgnored = CompositionTransaction::SetIMESelection(
+ *this, textNode, mComposition->XPOffsetInTextNode(),
+ mComposition->XPLengthInTextNode(), ranges);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "CompositionTransaction::SetIMESelection() failed, but ignored");
+ mComposition->OnUpdateCompositionInEditor(
+ mComposition->String(), *textNode,
+ mComposition->XPOffsetInTextNode());
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult EditorBase::FinalizeSelection() {
+ nsCOMPtr<nsISelectionController> selectionController =
+ GetSelectionController();
+ if (NS_WARN_IF(!selectionController)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eNotEditing);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ SelectionRef().SetAncestorLimiter(nullptr);
+
+ if (NS_WARN_IF(!GetPresShell())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ if (RefPtr<nsCaret> caret = GetCaret()) {
+ caret->SetIgnoreUserModify(true);
+ DebugOnly<nsresult> rvIgnored = selectionController->SetCaretEnabled(false);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "nsISelectionController::SetCaretEnabled(false) failed, but ignored");
+ }
+
+ RefPtr<nsFocusManager> focusManager = nsFocusManager::GetFocusManager();
+ if (NS_WARN_IF(!focusManager)) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+ // TODO: Running script from here makes harder to handle blur events. We
+ // should do this asynchronously.
+ focusManager->UpdateCaretForCaretBrowsingMode();
+ if (nsCOMPtr<nsINode> node = do_QueryInterface(GetDOMEventTarget())) {
+ if (node->OwnerDoc()->GetUnretargetedFocusedContent() != node) {
+ selectionController->SelectionWillLoseFocus();
+ } else {
+ // We leave this selection as the focused one. When the focus returns, it
+ // either returns to us (nothing to do), or it returns to something else,
+ // and nsDocumentViewerFocusListener::HandleEvent fixes it up.
+ }
+ }
+ return NS_OK;
+}
+
+Element* EditorBase::GetExposedRoot() const {
+ Element* rootElement = GetRoot();
+ if (!rootElement || !rootElement->IsInNativeAnonymousSubtree()) {
+ return rootElement;
+ }
+ return Element::FromNodeOrNull(
+ rootElement->GetClosestNativeAnonymousSubtreeRootParentOrHost());
+}
+
+nsresult EditorBase::DetermineCurrentDirection() {
+ // Get the current root direction from its frame
+ Element* rootElement = GetExposedRoot();
+ if (NS_WARN_IF(!rootElement)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // If we don't have an explicit direction, determine our direction
+ // from the content's direction
+ if (!IsRightToLeft() && !IsLeftToRight()) {
+ nsIFrame* frameForRootElement = rootElement->GetPrimaryFrame();
+ if (NS_WARN_IF(!frameForRootElement)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Set the flag here, to enable us to use the same code path below.
+ // It will be flipped before returning from the function.
+ if (frameForRootElement->StyleVisibility()->mDirection ==
+ StyleDirection::Rtl) {
+ mFlags |= nsIEditor::eEditorRightToLeft;
+ } else {
+ mFlags |= nsIEditor::eEditorLeftToRight;
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult EditorBase::ToggleTextDirectionAsAction(nsIPrincipal* aPrincipal) {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eSetTextDirection,
+ aPrincipal);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ nsresult rv = DetermineCurrentDirection();
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::DetermineCurrentDirection() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ MOZ_ASSERT(IsRightToLeft() || IsLeftToRight());
+ // Note that we need to consider new direction before dispatching
+ // "beforeinput" event since "beforeinput" event listener may change it
+ // but not canceled.
+ TextDirection newDirection =
+ IsRightToLeft() ? TextDirection::eLTR : TextDirection::eRTL;
+ editActionData.SetData(IsRightToLeft() ? u"ltr"_ns : u"rtl"_ns);
+
+ // FYI: Oddly, Chrome does not dispatch beforeinput event in this case but
+ // dispatches input event.
+ rv = editActionData.MaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "MaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ rv = SetTextDirectionTo(newDirection);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::SetTextDirectionTo() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ editActionData.MarkAsHandled();
+
+ // XXX When we don't change the text direction, do we really need to
+ // dispatch input event?
+ DispatchInputEvent();
+
+ return NS_OK;
+}
+
+void EditorBase::SwitchTextDirectionTo(TextDirection aTextDirection) {
+ MOZ_ASSERT(aTextDirection == TextDirection::eLTR ||
+ aTextDirection == TextDirection::eRTL);
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eSetTextDirection);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return;
+ }
+
+ nsresult rv = DetermineCurrentDirection();
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ return;
+ }
+
+ editActionData.SetData(aTextDirection == TextDirection::eLTR ? u"ltr"_ns
+ : u"rtl"_ns);
+
+ // FYI: Oddly, Chrome does not dispatch beforeinput event in this case but
+ // dispatches input event.
+ rv = editActionData.MaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "MaybeDispatchBeforeInputEvent() failed");
+ return;
+ }
+
+ if ((aTextDirection == TextDirection::eLTR && IsRightToLeft()) ||
+ (aTextDirection == TextDirection::eRTL && IsLeftToRight())) {
+ // Do it only when the direction is still different from the original
+ // new direction. Note that "beforeinput" event listener may have already
+ // changed the direction here, but they may not cancel the event.
+ nsresult rv = SetTextDirectionTo(aTextDirection);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("EditorBase::SetTextDirectionTo() failed");
+ return;
+ }
+ }
+
+ editActionData.MarkAsHandled();
+
+ // XXX When we don't change the text direction, do we really need to
+ // dispatch input event?
+ DispatchInputEvent();
+}
+
+nsresult EditorBase::SetTextDirectionTo(TextDirection aTextDirection) {
+ Element* rootElement = GetExposedRoot();
+
+ if (aTextDirection == TextDirection::eLTR) {
+ NS_ASSERTION(!IsLeftToRight(), "Unexpected mutually exclusive flag");
+ mFlags &= ~nsIEditor::eEditorRightToLeft;
+ mFlags |= nsIEditor::eEditorLeftToRight;
+ nsresult rv = rootElement->SetAttr(kNameSpaceID_None, nsGkAtoms::dir,
+ u"ltr"_ns, true);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "Element::SetAttr(nsGkAtoms::dir, ltr) failed");
+ return rv;
+ }
+
+ if (aTextDirection == TextDirection::eRTL) {
+ NS_ASSERTION(!IsRightToLeft(), "Unexpected mutually exclusive flag");
+ mFlags |= nsIEditor::eEditorRightToLeft;
+ mFlags &= ~nsIEditor::eEditorLeftToRight;
+ nsresult rv = rootElement->SetAttr(kNameSpaceID_None, nsGkAtoms::dir,
+ u"rtl"_ns, true);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "Element::SetAttr(nsGkAtoms::dir, rtl) failed");
+ return rv;
+ }
+
+ return NS_OK;
+}
+
+Element* EditorBase::GetFocusedElement() const {
+ EventTarget* eventTarget = GetDOMEventTarget();
+ if (!eventTarget) {
+ return nullptr;
+ }
+
+ nsFocusManager* focusManager = nsFocusManager::GetFocusManager();
+ if (NS_WARN_IF(!focusManager)) {
+ return nullptr;
+ }
+
+ Element* focusedElement = focusManager->GetFocusedElement();
+ MOZ_ASSERT((focusedElement == eventTarget) ==
+ SameCOMIdentity(focusedElement, eventTarget));
+
+ return (focusedElement == eventTarget) ? focusedElement : nullptr;
+}
+
+bool EditorBase::IsActiveInDOMWindow() const {
+ EventTarget* piTarget = GetDOMEventTarget();
+ if (!piTarget) {
+ return false;
+ }
+
+ nsFocusManager* focusManager = nsFocusManager::GetFocusManager();
+ if (NS_WARN_IF(!focusManager)) {
+ return false; // Do we need to check the singleton instance??
+ }
+
+ Document* document = GetDocument();
+ if (NS_WARN_IF(!document)) {
+ return false;
+ }
+ nsPIDOMWindowOuter* ourWindow = document->GetWindow();
+ nsCOMPtr<nsPIDOMWindowOuter> win;
+ nsIContent* content = nsFocusManager::GetFocusedDescendant(
+ ourWindow, nsFocusManager::eOnlyCurrentWindow, getter_AddRefs(win));
+ return SameCOMIdentity(content, piTarget);
+}
+
+bool EditorBase::IsAcceptableInputEvent(WidgetGUIEvent* aGUIEvent) const {
+ // If the event is trusted, the event should always cause input.
+ if (NS_WARN_IF(!aGUIEvent)) {
+ return false;
+ }
+
+ // If this is dispatched by using cordinates but this editor doesn't have
+ // focus, we shouldn't handle it.
+ if (aGUIEvent->IsUsingCoordinates() && !GetFocusedElement()) {
+ return false;
+ }
+
+ // If a composition event isn't dispatched via widget, we need to ignore them
+ // since they cannot be managed by TextComposition. E.g., the event was
+ // created by chrome JS.
+ // Note that if we allow to handle such events, editor may be confused by
+ // strange event order.
+ bool needsWidget = false;
+ switch (aGUIEvent->mMessage) {
+ case eUnidentifiedEvent:
+ // If events are not created with proper event interface, their message
+ // are initialized with eUnidentifiedEvent. Let's ignore such event.
+ return false;
+ case eCompositionStart:
+ case eCompositionEnd:
+ case eCompositionUpdate:
+ case eCompositionChange:
+ case eCompositionCommitAsIs:
+ // Don't allow composition events whose internal event are not
+ // WidgetCompositionEvent.
+ if (!aGUIEvent->AsCompositionEvent()) {
+ return false;
+ }
+ needsWidget = true;
+ break;
+ default:
+ break;
+ }
+ if (needsWidget && !aGUIEvent->mWidget) {
+ return false;
+ }
+
+ // Accept all trusted events.
+ if (aGUIEvent->IsTrusted()) {
+ return true;
+ }
+
+ // Ignore untrusted mouse event.
+ // XXX Why are we handling other untrusted input events?
+ if (aGUIEvent->AsMouseEventBase()) {
+ return false;
+ }
+
+ // Otherwise, we shouldn't handle any input events when we're not an active
+ // element of the DOM window.
+ return IsActiveInDOMWindow();
+}
+
+nsresult EditorBase::FlushPendingSpellCheck() {
+ // If the spell check skip flag is still enabled from creation time,
+ // disable it because focused editors are allowed to spell check.
+ if (!ShouldSkipSpellCheck()) {
+ return NS_OK;
+ }
+ MOZ_ASSERT(!IsHTMLEditor(), "HTMLEditor should not has pending spell checks");
+ nsresult rv = RemoveFlags(nsIEditor::eEditorSkipSpellCheck);
+ if (NS_WARN_IF(Destroyed())) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "EditorBase::RemoveFlags(nsIEditor::eEditorSkipSpellCheck) failed");
+ return rv;
+}
+
+bool EditorBase::CanKeepHandlingFocusEvent(
+ const nsINode& aOriginalEventTargetNode) const {
+ if (MOZ_UNLIKELY(!IsListeningToEvents() || Destroyed())) {
+ return false;
+ }
+
+ nsFocusManager* focusManager = nsFocusManager::GetFocusManager();
+ if (MOZ_UNLIKELY(!focusManager)) {
+ return false;
+ }
+
+ // If the event target is document mode, we only need to handle the focus
+ // event when the document is still in designMode. Otherwise, the
+ // mode has been disabled by somebody while we're handling the focus event.
+ if (aOriginalEventTargetNode.IsDocument()) {
+ return IsHTMLEditor() && aOriginalEventTargetNode.IsInDesignMode();
+ }
+ MOZ_ASSERT(aOriginalEventTargetNode.IsContent());
+
+ // If nobody has focus, the focus event target has been blurred by somebody
+ // else. So the editor shouldn't initialize itself to start to handle
+ // anything.
+ if (!focusManager->GetFocusedElement()) {
+ return false;
+ }
+ const nsIContent* exposedTargetContent =
+ aOriginalEventTargetNode.AsContent()
+ ->FindFirstNonChromeOnlyAccessContent();
+ const nsIContent* exposedFocusedContent =
+ focusManager->GetFocusedElement()->FindFirstNonChromeOnlyAccessContent();
+ return exposedTargetContent && exposedFocusedContent &&
+ exposedTargetContent == exposedFocusedContent;
+}
+
+nsresult EditorBase::OnFocus(const nsINode& aOriginalEventTargetNode) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ InitializeSelection(aOriginalEventTargetNode);
+ mSpellCheckerDictionaryUpdated = false;
+ if (mInlineSpellChecker && CanEnableSpellCheck()) {
+ DebugOnly<nsresult> rvIgnored =
+ mInlineSpellChecker->UpdateCurrentDictionary();
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "mozInlineSpellCHecker::UpdateCurrentDictionary() failed, but ignored");
+ mSpellCheckerDictionaryUpdated = true;
+ }
+ // XXX Why don't we stop handling focus with the spell checker immediately
+ // after calling InitializeSelection?
+ if (MOZ_UNLIKELY(!CanKeepHandlingFocusEvent(aOriginalEventTargetNode))) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+
+ const RefPtr<Element> focusedElement = GetFocusedElement();
+ RefPtr<nsPresContext> presContext =
+ focusedElement ? focusedElement->GetPresContext(
+ Element::PresContextFor::eForComposedDoc)
+ : GetPresContext();
+ if (NS_WARN_IF(!presContext)) {
+ return NS_ERROR_FAILURE;
+ }
+ IMEStateManager::OnFocusInEditor(*presContext, focusedElement, *this);
+
+ return NS_OK;
+}
+
+void EditorBase::HideCaret(bool aHide) {
+ if (mHidingCaret == aHide) {
+ return;
+ }
+
+ RefPtr<nsCaret> caret = GetCaret();
+ if (NS_WARN_IF(!caret)) {
+ return;
+ }
+
+ mHidingCaret = aHide;
+ if (aHide) {
+ caret->AddForceHide();
+ } else {
+ caret->RemoveForceHide();
+ }
+}
+
+NS_IMETHODIMP EditorBase::Unmask(uint32_t aStart, int64_t aEnd,
+ uint32_t aTimeout, uint8_t aArgc) {
+ if (NS_WARN_IF(!IsPasswordEditor())) {
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+ if (NS_WARN_IF(aArgc >= 1 && aStart == UINT32_MAX) ||
+ NS_WARN_IF(aArgc >= 2 && aEnd == 0) ||
+ NS_WARN_IF(aArgc >= 2 && aEnd > 0 && aStart >= aEnd)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eHidePassword);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ uint32_t start = aArgc < 1 ? 0 : aStart;
+ uint32_t length = aArgc < 2 || aEnd < 0 ? UINT32_MAX : aEnd - start;
+ uint32_t timeout = aArgc < 3 ? 0 : aTimeout;
+ nsresult rv = MOZ_KnownLive(AsTextEditor())
+ ->SetUnmaskRangeAndNotify(start, length, timeout);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("TextEditor::SetUnmaskRangeAndNotify() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ // Flush pending layout right now since the caller may access us before
+ // doing it.
+ if (RefPtr<PresShell> presShell = GetPresShell()) {
+ presShell->FlushPendingNotifications(FlushType::Layout);
+ }
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::Mask() {
+ if (NS_WARN_IF(!IsPasswordEditor())) {
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eHidePassword);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ nsresult rv = MOZ_KnownLive(AsTextEditor())->MaskAllCharactersAndNotify();
+ if (NS_FAILED(rv)) {
+ NS_WARNING("TextEditor::MaskAllCharactersAndNotify() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ // Flush pending layout right now since the caller may access us before
+ // doing it.
+ if (RefPtr<PresShell> presShell = GetPresShell()) {
+ presShell->FlushPendingNotifications(FlushType::Layout);
+ }
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetUnmaskedStart(uint32_t* aResult) {
+ if (NS_WARN_IF(!IsPasswordEditor())) {
+ *aResult = 0;
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+ *aResult =
+ AsTextEditor()->IsAllMasked() ? 0 : AsTextEditor()->UnmaskedStart();
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetUnmaskedEnd(uint32_t* aResult) {
+ if (NS_WARN_IF(!IsPasswordEditor())) {
+ *aResult = 0;
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+ *aResult = AsTextEditor()->IsAllMasked() ? 0 : AsTextEditor()->UnmaskedEnd();
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetAutoMaskingEnabled(bool* aResult) {
+ if (NS_WARN_IF(!IsPasswordEditor())) {
+ *aResult = false;
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+ *aResult = AsTextEditor()->IsMaskingPassword();
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::GetPasswordMask(nsAString& aPasswordMask) {
+ aPasswordMask.Assign(TextEditor::PasswordMask());
+ return NS_OK;
+}
+
+template <typename PT, typename CT>
+EditorBase::AutoCaretBidiLevelManager::AutoCaretBidiLevelManager(
+ const EditorBase& aEditorBase, nsIEditor::EDirection aDirectionAndAmount,
+ const EditorDOMPointBase<PT, CT>& aPointAtCaret) {
+ MOZ_ASSERT(aEditorBase.IsEditActionDataAvailable());
+
+ nsPresContext* presContext = aEditorBase.GetPresContext();
+ if (NS_WARN_IF(!presContext)) {
+ mFailed = true;
+ return;
+ }
+
+ if (!presContext->BidiEnabled()) {
+ return; // Perform the deletion
+ }
+
+ if (!aPointAtCaret.IsInContentNode()) {
+ mFailed = true;
+ return;
+ }
+
+ // XXX Not sure whether this requires strong reference here.
+ RefPtr<nsFrameSelection> frameSelection =
+ aEditorBase.SelectionRef().GetFrameSelection();
+ if (NS_WARN_IF(!frameSelection)) {
+ mFailed = true;
+ return;
+ }
+
+ nsPrevNextBidiLevels levels = frameSelection->GetPrevNextBidiLevels(
+ aPointAtCaret.template ContainerAs<nsIContent>(), aPointAtCaret.Offset(),
+ true);
+
+ mozilla::intl::BidiEmbeddingLevel levelBefore = levels.mLevelBefore;
+ mozilla::intl::BidiEmbeddingLevel levelAfter = levels.mLevelAfter;
+
+ mozilla::intl::BidiEmbeddingLevel currentCaretLevel =
+ frameSelection->GetCaretBidiLevel();
+
+ mozilla::intl::BidiEmbeddingLevel levelOfDeletion;
+ levelOfDeletion = (nsIEditor::eNext == aDirectionAndAmount ||
+ nsIEditor::eNextWord == aDirectionAndAmount)
+ ? levelAfter
+ : levelBefore;
+
+ if (currentCaretLevel == levelOfDeletion) {
+ return; // Perform the deletion
+ }
+
+ // Set the bidi level of the caret to that of the
+ // character that will be (or would have been) deleted
+ mNewCaretBidiLevel = Some(levelOfDeletion);
+ mCanceled =
+ !StaticPrefs::bidi_edit_delete_immediately() && levelBefore != levelAfter;
+}
+
+void EditorBase::AutoCaretBidiLevelManager::MaybeUpdateCaretBidiLevel(
+ const EditorBase& aEditorBase) const {
+ MOZ_ASSERT(!mFailed);
+ if (mNewCaretBidiLevel.isNothing()) {
+ return;
+ }
+ RefPtr<nsFrameSelection> frameSelection =
+ aEditorBase.SelectionRef().GetFrameSelection();
+ MOZ_ASSERT(frameSelection);
+ frameSelection->SetCaretBidiLevelAndMaybeSchedulePaint(
+ mNewCaretBidiLevel.value());
+}
+
+void EditorBase::UndefineCaretBidiLevel() const {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ /**
+ * After inserting text the caret Bidi level must be set to the level of the
+ * inserted text.This is difficult, because we cannot know what the level is
+ * until after the Bidi algorithm is applied to the whole paragraph.
+ *
+ * So we set the caret Bidi level to UNDEFINED here, and the caret code will
+ * set it correctly later
+ */
+ nsFrameSelection* frameSelection = SelectionRef().GetFrameSelection();
+ if (frameSelection) {
+ frameSelection->UndefineCaretBidiLevel();
+ }
+}
+
+NS_IMETHODIMP EditorBase::GetTextLength(uint32_t* aCount) {
+ return NS_ERROR_NOT_IMPLEMENTED;
+}
+
+NS_IMETHODIMP EditorBase::GetWrapWidth(int32_t* aWrapColumn) {
+ if (NS_WARN_IF(!aWrapColumn)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aWrapColumn = WrapWidth();
+ return NS_OK;
+}
+
+//
+// See if the style value includes this attribute, and if it does,
+// cut out everything from the attribute to the next semicolon.
+//
+static void CutStyle(const char* stylename, nsString& styleValue) {
+ // Find the current wrapping type:
+ int32_t styleStart = styleValue.LowerCaseFindASCII(stylename);
+ if (styleStart >= 0) {
+ int32_t styleEnd = styleValue.Find(u";", styleStart);
+ if (styleEnd > styleStart) {
+ styleValue.Cut(styleStart, styleEnd - styleStart + 1);
+ } else {
+ styleValue.Cut(styleStart, styleValue.Length() - styleStart);
+ }
+ }
+}
+
+NS_IMETHODIMP EditorBase::SetWrapWidth(int32_t aWrapColumn) {
+ AutoEditActionDataSetter editActionData(*this, EditAction::eSetWrapWidth);
+ if (NS_WARN_IF(!editActionData.CanHandle())) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ SetWrapColumn(aWrapColumn);
+
+ // Make sure we're a plaintext editor, otherwise we shouldn't
+ // do the rest of this.
+ if (!IsInPlaintextMode()) {
+ return NS_OK;
+ }
+
+ // Ought to set a style sheet here ...
+ // Probably should keep around an mPlaintextStyleSheet for this purpose.
+ RefPtr<Element> rootElement = GetRoot();
+ if (NS_WARN_IF(!rootElement)) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ // Get the current style for this root element:
+ nsAutoString styleValue;
+ rootElement->GetAttr(kNameSpaceID_None, nsGkAtoms::style, styleValue);
+
+ // We'll replace styles for these values:
+ CutStyle("white-space", styleValue);
+ CutStyle("width", styleValue);
+ CutStyle("font-family", styleValue);
+
+ // If we have other style left, trim off any existing semicolons
+ // or white-space, then add a known semicolon-space:
+ if (!styleValue.IsEmpty()) {
+ styleValue.Trim("; \t", false, true);
+ styleValue.AppendLiteral("; ");
+ }
+
+ // Make sure we have fixed-width font. This should be done for us,
+ // but it isn't, see bug 22502, so we have to add "font: -moz-fixed;".
+ // Only do this if we're wrapping.
+ if (IsWrapHackEnabled() && aWrapColumn >= 0) {
+ styleValue.AppendLiteral("font-family: -moz-fixed; ");
+ }
+
+ // and now we're ready to set the new white-space/wrapping style.
+ if (aWrapColumn > 0) {
+ // Wrap to a fixed column.
+ styleValue.AppendLiteral("white-space: pre-wrap; width: ");
+ styleValue.AppendInt(aWrapColumn);
+ styleValue.AppendLiteral("ch;");
+ } else if (!aWrapColumn) {
+ styleValue.AppendLiteral("white-space: pre-wrap;");
+ } else {
+ styleValue.AppendLiteral("white-space: pre;");
+ }
+
+ nsresult rv = rootElement->SetAttr(kNameSpaceID_None, nsGkAtoms::style,
+ styleValue, true);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "Element::SetAttr(nsGkAtoms::style) failed");
+ return rv;
+}
+
+NS_IMETHODIMP EditorBase::GetNewlineHandling(int32_t* aNewlineHandling) {
+ if (NS_WARN_IF(!aNewlineHandling)) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aNewlineHandling = mNewlineHandling;
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::SetNewlineHandling(int32_t aNewlineHandling) {
+ switch (aNewlineHandling) {
+ case nsIEditor::eNewlinesPasteIntact:
+ case nsIEditor::eNewlinesPasteToFirst:
+ case nsIEditor::eNewlinesReplaceWithSpaces:
+ case nsIEditor::eNewlinesStrip:
+ case nsIEditor::eNewlinesReplaceWithCommas:
+ case nsIEditor::eNewlinesStripSurroundingWhitespace:
+ mNewlineHandling = aNewlineHandling;
+ return NS_OK;
+ default:
+ NS_ERROR("SetNewlineHandling() is called with wrong value");
+ return NS_ERROR_INVALID_ARG;
+ }
+}
+
+bool EditorBase::IsSelectionRangeContainerNotContent() const {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+
+ // TODO: Make all callers use !AutoRangeArray::IsInContent() instead.
+ const uint32_t rangeCount = SelectionRef().RangeCount();
+ for (const uint32_t i : IntegerRange(rangeCount)) {
+ MOZ_ASSERT(SelectionRef().RangeCount() == rangeCount);
+ const nsRange* range = SelectionRef().GetRangeAt(i);
+ MOZ_ASSERT(range);
+ if (MOZ_UNLIKELY(!range) || MOZ_UNLIKELY(!range->GetStartContainer()) ||
+ MOZ_UNLIKELY(!range->GetStartContainer()->IsContent()) ||
+ MOZ_UNLIKELY(!range->GetEndContainer()) ||
+ MOZ_UNLIKELY(!range->GetEndContainer()->IsContent())) {
+ return true;
+ }
+ }
+ return false;
+}
+
+NS_IMETHODIMP EditorBase::InsertText(const nsAString& aStringToInsert) {
+ nsresult rv = InsertTextAsAction(aStringToInsert);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::InsertTextAsAction() failed");
+ return rv;
+}
+
+nsresult EditorBase::InsertTextAsAction(const nsAString& aStringToInsert,
+ nsIPrincipal* aPrincipal) {
+ // Showing this assertion is fine if this method is called by outside via
+ // mutation event listener or something. Otherwise, this is called by
+ // wrong method.
+ NS_ASSERTION(!mPlaceholderBatch,
+ "Should be called only when this is the only edit action of the "
+ "operation "
+ "unless mutation event listener nests some operations");
+
+ AutoEditActionDataSetter editActionData(*this, EditAction::eInsertText,
+ aPrincipal);
+ // Note that we don't need to replace native line breaks with XP line breaks
+ // here because Chrome does not do it.
+ MOZ_ASSERT(!aStringToInsert.IsVoid());
+ editActionData.SetData(aStringToInsert);
+ nsresult rv = editActionData.CanHandleAndMaybeDispatchBeforeInputEvent();
+ if (NS_FAILED(rv)) {
+ NS_WARNING_ASSERTION(rv == NS_ERROR_EDITOR_ACTION_CANCELED,
+ "CanHandleAndMaybeDispatchBeforeInputEvent() failed");
+ return EditorBase::ToGenericNSResult(rv);
+ }
+
+ nsString stringToInsert(aStringToInsert);
+ if (IsTextEditor()) {
+ nsContentUtils::PlatformToDOMLineBreaks(stringToInsert);
+ }
+ AutoPlaceholderBatch treatAsOneTransaction(
+ *this, ScrollSelectionIntoView::Yes, __FUNCTION__);
+ rv = InsertTextAsSubAction(stringToInsert, SelectionHandling::Delete);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv),
+ "EditorBase::InsertTextAsSubAction() failed");
+ return EditorBase::ToGenericNSResult(rv);
+}
+
+nsresult EditorBase::InsertTextAsSubAction(
+ const nsAString& aStringToInsert, SelectionHandling aSelectionHandling) {
+ MOZ_ASSERT(IsEditActionDataAvailable());
+ MOZ_ASSERT(mPlaceholderBatch);
+ MOZ_ASSERT(IsHTMLEditor() ||
+ aStringToInsert.FindChar(nsCRT::CR) == kNotFound);
+ MOZ_ASSERT_IF(aSelectionHandling == SelectionHandling::Ignore, mComposition);
+
+ if (NS_WARN_IF(!mInitSucceeded)) {
+ return NS_ERROR_NOT_INITIALIZED;
+ }
+
+ if (NS_WARN_IF(Destroyed())) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+
+ EditSubAction editSubAction = ShouldHandleIMEComposition()
+ ? EditSubAction::eInsertTextComingFromIME
+ : EditSubAction::eInsertText;
+
+ IgnoredErrorResult ignoredError;
+ AutoEditSubActionNotifier startToHandleEditSubAction(
+ *this, editSubAction, nsIEditor::eNext, ignoredError);
+ if (NS_WARN_IF(ignoredError.ErrorCodeIs(NS_ERROR_EDITOR_DESTROYED))) {
+ return ignoredError.StealNSResult();
+ }
+ NS_WARNING_ASSERTION(
+ !ignoredError.Failed(),
+ "TextEditor::OnStartToHandleTopLevelEditSubAction() failed, but ignored");
+
+ Result<EditActionResult, nsresult> result =
+ HandleInsertText(editSubAction, aStringToInsert, aSelectionHandling);
+ if (MOZ_UNLIKELY(result.isErr())) {
+ NS_WARNING("EditorBase::HandleInsertText() failed");
+ return result.unwrapErr();
+ }
+ return NS_OK;
+}
+
+NS_IMETHODIMP EditorBase::InsertLineBreak() { return NS_ERROR_NOT_IMPLEMENTED; }
+
+/*****************************************************************************
+ * mozilla::EditorBase::AutoEditActionDataSetter
+ *****************************************************************************/
+
+EditorBase::AutoEditActionDataSetter::AutoEditActionDataSetter(
+ const EditorBase& aEditorBase, EditAction aEditAction,
+ nsIPrincipal* aPrincipal /* = nullptr */)
+ : mEditorBase(const_cast<EditorBase&>(aEditorBase)),
+ mPrincipal(aPrincipal),
+ mParentData(aEditorBase.mEditActionData),
+ mData(VoidString()),
+ mRawEditAction(aEditAction),
+ mTopLevelEditSubAction(EditSubAction::eNone),
+ mAborted(false),
+ mHasTriedToDispatchBeforeInputEvent(false),
+ mBeforeInputEventCanceled(false),
+ mMakeBeforeInputEventNonCancelable(false),
+ mHasTriedToDispatchClipboardEvent(false),
+ mEditorWasDestroyedDuringHandlingEditAction(
+ mParentData &&
+ mParentData->mEditorWasDestroyedDuringHandlingEditAction),
+ mHandled(false) {
+ // If we're nested edit action, copies necessary data from the parent.
+ if (mParentData) {
+ mSelection = mParentData->mSelection;
+ MOZ_ASSERT(!mSelection ||
+ (mSelection->GetType() == SelectionType::eNormal));
+
+ // If we're not editing something, we should inherit the parent's edit
+ // action. This may occur if creator or its callee use public methods which
+ // just returns something.
+ if (IsEditActionInOrderToEditSomething(aEditAction)) {
+ mEditAction = aEditAction;
+ } else {
+ mEditAction = mParentData->mEditAction;
+ // If we inherit an edit action whose handler needs to dispatch a
+ // clipboard event, we should inherit the clipboard dispatching state
+ // too because this nest occurs by a clipboard event listener or
+ // a beforeinput/mutation event listener is important for checking
+ // whether we've already called `MaybeDispatchBeforeInputEvent()`
+ // property in some points. If the former case, not yet dispatching
+ // beforeinput event is okay (not fine).
+ mHasTriedToDispatchClipboardEvent =
+ mParentData->mHasTriedToDispatchClipboardEvent;
+ }
+ mTopLevelEditSubAction = mParentData->mTopLevelEditSubAction;
+
+ // Parent's mTopLevelEditSubActionData should be referred instead so that
+ // we don't need to set mTopLevelEditSubActionData.mSelectedRange nor
+ // mTopLevelEditActionData.mChangedRange here.
+
+ mDirectionOfTopLevelEditSubAction =
+ mParentData->mDirectionOfTopLevelEditSubAction;
+ } else {
+ mSelection = mEditorBase.GetSelection();
+ if (NS_WARN_IF(!mSelection)) {
+ return;
+ }
+
+ MOZ_ASSERT(mSelection->GetType() == SelectionType::eNormal);
+
+ mEditAction = aEditAction;
+ mDirectionOfTopLevelEditSubAction = eNone;
+ if (mEditorBase.IsHTMLEditor()) {
+ mTopLevelEditSubActionData.mSelectedRange =
+ mEditorBase.AsHTMLEditor()
+ ->GetSelectedRangeItemForTopLevelEditSubAction();
+ mTopLevelEditSubActionData.mChangedRange =
+ mEditorBase.AsHTMLEditor()->GetChangedRangeForTopLevelEditSubAction();
+ mTopLevelEditSubActionData.mCachedPendingStyles.emplace();
+ }
+ }
+ mEditorBase.mEditActionData = this;
+}
+
+EditorBase::AutoEditActionDataSetter::~AutoEditActionDataSetter() {
+ MOZ_ASSERT(mHasCanHandleChecked);
+
+ if (!mSelection || NS_WARN_IF(mEditorBase.mEditActionData != this)) {
+ return;
+ }
+ mEditorBase.mEditActionData = mParentData;
+
+ MOZ_ASSERT(
+ !mTopLevelEditSubActionData.mSelectedRange ||
+ (!mTopLevelEditSubActionData.mSelectedRange->mStartContainer &&
+ !mTopLevelEditSubActionData.mSelectedRange->mEndContainer),
+ "mTopLevelEditSubActionData.mSelectedRange should've been cleared");
+}
+
+void EditorBase::AutoEditActionDataSetter::UpdateSelectionCache(
+ Selection& aSelection) {
+ MOZ_ASSERT(aSelection.GetType() == SelectionType::eNormal);
+
+ if (mSelection == &aSelection) {
+ return;
+ }
+
+ AutoEditActionDataSetter& topLevelEditActionData =
+ [&]() -> AutoEditActionDataSetter& {
+ for (AutoEditActionDataSetter* editActionData = this;;
+ editActionData = editActionData->mParentData) {
+ if (!editActionData->mParentData) {
+ return *editActionData;
+ }
+ }
+ MOZ_ASSERT_UNREACHABLE("You do something wrong");
+ }();
+
+ // Keep grabbing the old selection in the top level edit action data until the
+ // all owners end handling it.
+ if (mSelection) {
+ topLevelEditActionData.mRetiredSelections.AppendElement(*mSelection);
+ }
+
+ // If the old selection is in batch, we should end the batch which
+ // `EditorBase::BeginUpdateViewBatch` started.
+ if (mEditorBase.mUpdateCount && mSelection) {
+ mSelection->EndBatchChanges(__FUNCTION__);
+ }
+
+ Selection* previousSelection = mSelection;
+ mSelection = &aSelection;
+ for (AutoEditActionDataSetter* parentActionData = mParentData;
+ parentActionData; parentActionData = parentActionData->mParentData) {
+ if (!parentActionData->mSelection) {
+ continue;
+ }
+ // Skip scanning mRetiredSelections if we've already handled the selection
+ // previous time.
+ if (parentActionData->mSelection != previousSelection) {
+ if (!topLevelEditActionData.mRetiredSelections.Contains(
+ OwningNonNull<Selection>(*parentActionData->mSelection))) {
+ topLevelEditActionData.mRetiredSelections.AppendElement(
+ *parentActionData->mSelection);
+ }
+ previousSelection = parentActionData->mSelection;
+ }
+ parentActionData->mSelection = &aSelection;
+ }
+
+ // Restart the batching in the new selection.
+ if (mEditorBase.mUpdateCount) {
+ aSelection.StartBatchChanges(__FUNCTION__);
+ }
+}
+
+void EditorBase::AutoEditActionDataSetter::SetColorData(
+ const nsAString& aData) {
+ MOZ_ASSERT(!HasTriedToDispatchBeforeInputEvent(),
+ "It's too late to set data since this may have already dispatched "
+ "a beforeinput event");
+
+ if (aData.IsEmpty()) {
+ // When removing color/background-color, let's use empty string.
+ mData.Truncate();
+ MOZ_ASSERT(!mData.IsVoid());
+ return;
+ }
+
+ DebugOnly<bool> validColorValue = HTMLEditUtils::GetNormalizedCSSColorValue(
+ aData, HTMLEditUtils::ZeroAlphaColor::RGBAValue, mData);
+ MOZ_ASSERT_IF(validColorValue, !mData.IsVoid());
+}
+
+void EditorBase::AutoEditActionDataSetter::InitializeDataTransfer(
+ DataTransfer* aDataTransfer) {
+ MOZ_ASSERT(aDataTransfer);
+ MOZ_ASSERT(aDataTransfer->IsReadOnly());
+ MOZ_ASSERT(!HasTriedToDispatchBeforeInputEvent(),
+ "It's too late to set dataTransfer since this may have already "
+ "dispatched a beforeinput event");
+
+ mDataTransfer = aDataTransfer;
+}
+
+void EditorBase::AutoEditActionDataSetter::InitializeDataTransfer(
+ nsITransferable* aTransferable) {
+ MOZ_ASSERT(aTransferable);
+ MOZ_ASSERT(!HasTriedToDispatchBeforeInputEvent(),
+ "It's too late to set dataTransfer since this may have already "
+ "dispatched a beforeinput event");
+
+ Document* document = mEditorBase.GetDocument();
+ nsIGlobalObject* scopeObject =
+ document ? document->GetScopeObject() : nullptr;
+ mDataTransfer = new DataTransfer(scopeObject, eEditorInput, aTransferable);
+}
+
+void EditorBase::AutoEditActionDataSetter::InitializeDataTransfer(
+ const nsAString& aString) {
+ MOZ_ASSERT(!HasTriedToDispatchBeforeInputEvent(),
+ "It's too late to set dataTransfer since this may have already "
+ "dispatched a beforeinput event");
+ Document* document = mEditorBase.GetDocument();
+ nsIGlobalObject* scopeObject =
+ document ? document->GetScopeObject() : nullptr;
+ mDataTransfer = new DataTransfer(scopeObject, eEditorInput, aString);
+}
+
+void EditorBase::AutoEditActionDataSetter::InitializeDataTransferWithClipboard(
+ SettingDataTransfer aSettingDataTransfer, int32_t aClipboardType) {
+ MOZ_ASSERT(!HasTriedToDispatchBeforeInputEvent(),
+ "It's too late to set dataTransfer since this may have already "
+ "dispatched a beforeinput event");
+
+ Document* document = mEditorBase.GetDocument();
+ nsIGlobalObject* scopeObject =
+ document ? document->GetScopeObject() : nullptr;
+ // mDataTransfer will be used for eEditorInput event, but we can keep
+ // using ePaste and ePasteNoFormatting here. If we need to use eEditorInput,
+ // we need to create eEditorInputNoFormatting or something...
+ mDataTransfer =
+ new DataTransfer(scopeObject,
+ aSettingDataTransfer == SettingDataTransfer::eWithFormat
+ ? ePaste
+ : ePasteNoFormatting,
+ true /* is external */, aClipboardType);
+}
+
+void EditorBase::AutoEditActionDataSetter::AppendTargetRange(
+ StaticRange& aTargetRange) {
+ mTargetRanges.AppendElement(aTargetRange);
+}
+
+bool EditorBase::AutoEditActionDataSetter::IsBeforeInputEventEnabled() const {
+ if (!StaticPrefs::dom_input_events_beforeinput_enabled()) {
+ return false;
+ }
+
+ // Don't dispatch "beforeinput" event when the editor user makes us stop
+ // dispatching input event.
+ if (mEditorBase.IsSuppressingDispatchingInputEvent()) {
+ return false;
+ }
+
+ // If mPrincipal has set, it means that we're handling an edit action
+ // which is requested by JS. If it's not chrome script, we shouldn't
+ // dispatch "beforeinput" event.
+ if (mPrincipal && !mPrincipal->IsSystemPrincipal()) {
+ // But if it's content script of an addon, `execCommand` calls are a
+ // part of browser's default action from point of view of web apps.
+ // Therefore, we should dispatch `beforeinput` event.
+ // https://github.com/w3c/input-events/issues/91
+ if (!mPrincipal->GetIsAddonOrExpandedAddonPrincipal()) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+nsresult EditorBase::AutoEditActionDataSetter::MaybeFlushPendingNotifications()
+ const {
+ MOZ_ASSERT(CanHandle());
+ if (!MayEditActionRequireLayout(mRawEditAction)) {
+ return NS_SUCCESS_DOM_NO_OPERATION;
+ }
+ OwningNonNull<EditorBase> editorBase = mEditorBase;
+ RefPtr<PresShell> presShell = editorBase->GetPresShell();
+ if (MOZ_UNLIKELY(NS_WARN_IF(!presShell))) {
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+ presShell->FlushPendingNotifications(FlushType::Layout);
+ if (MOZ_UNLIKELY(NS_WARN_IF(editorBase->Destroyed()))) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+ return NS_OK;
+}
+
+nsresult EditorBase::AutoEditActionDataSetter::MaybeDispatchBeforeInputEvent(
+ nsIEditor::EDirection aDeleteDirectionAndAmount /* = nsIEditor::eNone */) {
+ MOZ_ASSERT(!HasTriedToDispatchBeforeInputEvent(),
+ "We've already handled beforeinput event");
+ MOZ_ASSERT(CanHandle());
+ MOZ_ASSERT_IF(IsBeforeInputEventEnabled(),
+ ShouldAlreadyHaveHandledBeforeInputEventDispatching());
+ MOZ_ASSERT_IF(!MayEditActionDeleteAroundCollapsedSelection(mEditAction),
+ aDeleteDirectionAndAmount == nsIEditor::eNone);
+
+ mHasTriedToDispatchBeforeInputEvent = true;
+
+ if (!IsBeforeInputEventEnabled()) {
+ return NS_OK;
+ }
+
+ // If we're called from OnCompositionEnd(), we shouldn't dispatch
+ // "beforeinput" event since the preceding OnCompositionChange() call has
+ // already dispatched "beforeinput" event for this.
+ if (mEditAction == EditAction::eCommitComposition ||
+ mEditAction == EditAction::eCancelComposition) {
+ return NS_OK;
+ }
+
+ RefPtr<Element> targetElement = mEditorBase.GetInputEventTargetElement();
+ if (!targetElement) {
+ // If selection is not in editable element and it is outside of any
+ // editing hosts, there may be no target element to dispatch `beforeinput`
+ // event. In this case, the caller shouldn't keep handling the edit
+ // action since web apps cannot override it with `beforeinput` event
+ // listener, but for backward compatibility, we should return a special
+ // success code instead of error.
+ return NS_OK;
+ }
+ OwningNonNull<EditorBase> editorBase = mEditorBase;
+ EditorInputType inputType = ToInputType(mEditAction);
+ if (editorBase->IsHTMLEditor() && mTargetRanges.IsEmpty()) {
+ // If the edit action will delete selected ranges, compute the range
+ // strictly.
+ if (MayEditActionDeleteAroundCollapsedSelection(mEditAction) ||
+ (!editorBase->SelectionRef().IsCollapsed() &&
+ MayEditActionDeleteSelection(mEditAction))) {
+ if (!editorBase
+ ->FlushPendingNotificationsIfToHandleDeletionWithFrameSelection(
+ aDeleteDirectionAndAmount)) {
+ NS_WARNING(
+ "Flusing pending notifications caused destroying the editor");
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+
+ AutoRangeArray rangesToDelete(editorBase->SelectionRef());
+ if (!rangesToDelete.Ranges().IsEmpty()) {
+ nsresult rv = MOZ_KnownLive(editorBase->AsHTMLEditor())
+ ->ComputeTargetRanges(aDeleteDirectionAndAmount,
+ rangesToDelete);
+ if (rv == NS_ERROR_EDITOR_DESTROYED) {
+ NS_WARNING("HTMLEditor::ComputeTargetRanges() destroyed the editor");
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+ if (rv == NS_ERROR_EDITOR_NO_EDITABLE_RANGE) {
+ // For now, keep dispatching `beforeinput` event even if no selection
+ // range can be editable.
+ rv = NS_OK;
+ }
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "HTMLEditor::ComputeTargetRanges() failed, but ignored");
+ for (auto& range : rangesToDelete.Ranges()) {
+ RefPtr<StaticRange> staticRange =
+ StaticRange::Create(range, IgnoreErrors());
+ if (NS_WARN_IF(!staticRange)) {
+ continue;
+ }
+ AppendTargetRange(*staticRange);
+ }
+ }
+ }
+ // Otherwise, just set target ranges to selection ranges.
+ else if (MayHaveTargetRangesOnHTMLEditor(inputType)) {
+ if (uint32_t rangeCount = editorBase->SelectionRef().RangeCount()) {
+ mTargetRanges.SetCapacity(rangeCount);
+ for (const uint32_t i : IntegerRange(rangeCount)) {
+ MOZ_ASSERT(editorBase->SelectionRef().RangeCount() == rangeCount);
+ const nsRange* range = editorBase->SelectionRef().GetRangeAt(i);
+ MOZ_ASSERT(range);
+ MOZ_ASSERT(range->IsPositioned());
+ if (MOZ_UNLIKELY(NS_WARN_IF(!range)) ||
+ MOZ_UNLIKELY(NS_WARN_IF(!range->IsPositioned()))) {
+ continue;
+ }
+ // Now, we need to fix the offset of target range because it may
+ // be referred after modifying the DOM tree and range boundaries
+ // of `range` may have not computed offset yet.
+ RefPtr<StaticRange> targetRange = StaticRange::Create(
+ range->GetStartContainer(), range->StartOffset(),
+ range->GetEndContainer(), range->EndOffset(), IgnoreErrors());
+ if (NS_WARN_IF(!targetRange) ||
+ NS_WARN_IF(!targetRange->IsPositioned())) {
+ continue;
+ }
+ mTargetRanges.AppendElement(std::move(targetRange));
+ }
+ }
+ }
+ }
+ nsEventStatus status = nsEventStatus_eIgnore;
+ InputEventOptions::NeverCancelable neverCancelable =
+ mMakeBeforeInputEventNonCancelable
+ ? InputEventOptions::NeverCancelable::Yes
+ : InputEventOptions::NeverCancelable::No;
+ nsresult rv = nsContentUtils::DispatchInputEvent(
+ targetElement, eEditorBeforeInput, inputType, editorBase,
+ mDataTransfer
+ ? InputEventOptions(mDataTransfer, std::move(mTargetRanges),
+ neverCancelable)
+ : InputEventOptions(mData, std::move(mTargetRanges), neverCancelable),
+ &status);
+ if (NS_WARN_IF(mEditorBase.Destroyed())) {
+ return NS_ERROR_EDITOR_DESTROYED;
+ }
+ if (NS_FAILED(rv)) {
+ NS_WARNING("nsContentUtils::DispatchInputEvent() failed");
+ return rv;
+ }
+ mBeforeInputEventCanceled = status == nsEventStatus_eConsumeNoDefault;
+ if (mBeforeInputEventCanceled && mEditorBase.IsHTMLEditor()) {
+ mEditorBase.AsHTMLEditor()->mHasBeforeInputBeenCanceled = true;
+ }
+ return mBeforeInputEventCanceled ? NS_ERROR_EDITOR_ACTION_CANCELED : NS_OK;
+}
+
+/*****************************************************************************
+ * mozilla::EditorBase::TopLevelEditSubActionData
+ *****************************************************************************/
+
+nsresult EditorBase::TopLevelEditSubActionData::AddNodeToChangedRange(
+ const HTMLEditor& aHTMLEditor, nsINode& aNode) {
+ EditorRawDOMPoint startPoint(&aNode);
+ EditorRawDOMPoint endPoint(&aNode);
+ DebugOnly<bool> advanced = endPoint.AdvanceOffset();
+ NS_WARNING_ASSERTION(advanced, "Failed to set endPoint to next to aNode");
+ nsresult rv = AddRangeToChangedRange(aHTMLEditor, startPoint, endPoint);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "TopLevelEditSubActionData::AddRangeToChangedRange() failed");
+ return rv;
+}
+
+nsresult EditorBase::TopLevelEditSubActionData::AddPointToChangedRange(
+ const HTMLEditor& aHTMLEditor, const EditorRawDOMPoint& aPoint) {
+ nsresult rv = AddRangeToChangedRange(aHTMLEditor, aPoint, aPoint);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rv),
+ "TopLevelEditSubActionData::AddRangeToChangedRange() failed");
+ return rv;
+}
+
+nsresult EditorBase::TopLevelEditSubActionData::AddRangeToChangedRange(
+ const HTMLEditor& aHTMLEditor, const EditorRawDOMPoint& aStart,
+ const EditorRawDOMPoint& aEnd) {
+ if (NS_WARN_IF(!aStart.IsSet()) || NS_WARN_IF(!aEnd.IsSet())) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (!aHTMLEditor.IsDescendantOfRoot(aStart.GetContainer()) ||
+ (aStart.GetContainer() != aEnd.GetContainer() &&
+ !aHTMLEditor.IsDescendantOfRoot(aEnd.GetContainer()))) {
+ return NS_OK;
+ }
+
+ // If mChangedRange hasn't been set, we can just set it to `aStart` and
+ // `aEnd`.
+ if (!mChangedRange->IsPositioned()) {
+ nsresult rv = mChangedRange->SetStartAndEnd(aStart.ToRawRangeBoundary(),
+ aEnd.ToRawRangeBoundary());
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "nsRange::SetStartAndEnd() failed");
+ return rv;
+ }
+
+ Maybe<int32_t> relation =
+ mChangedRange->StartRef().IsSet()
+ ? nsContentUtils::ComparePoints(mChangedRange->StartRef(),
+ aStart.ToRawRangeBoundary())
+ : Some(1);
+ if (NS_WARN_IF(!relation)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // If aStart is before start of mChangedRange, reset the start.
+ if (*relation > 0) {
+ ErrorResult error;
+ mChangedRange->SetStart(aStart.ToRawRangeBoundary(), error);
+ if (error.Failed()) {
+ NS_WARNING("nsRange::SetStart() failed");
+ return error.StealNSResult();
+ }
+ }
+
+ relation = mChangedRange->EndRef().IsSet()
+ ? nsContentUtils::ComparePoints(mChangedRange->EndRef(),
+ aEnd.ToRawRangeBoundary())
+ : Some(1);
+ if (NS_WARN_IF(!relation)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // If aEnd is after end of mChangedRange, reset the end.
+ if (*relation < 0) {
+ ErrorResult error;
+ mChangedRange->SetEnd(aEnd.ToRawRangeBoundary(), error);
+ if (error.Failed()) {
+ NS_WARNING("nsRange::SetEnd() failed");
+ return error.StealNSResult();
+ }
+ }
+
+ return NS_OK;
+}
+
+void EditorBase::TopLevelEditSubActionData::DidCreateElement(
+ EditorBase& aEditorBase, Element& aNewElement) {
+ MOZ_ASSERT(aEditorBase.AsHTMLEditor());
+
+ if (!aEditorBase.mInitSucceeded || aEditorBase.Destroyed()) {
+ return; // We have not been initialized yet or already been destroyed.
+ }
+
+ if (!aEditorBase.EditSubActionDataRef().mAdjustChangedRangeFromListener) {
+ return; // Temporarily disabled by edit sub-action handler.
+ }
+
+ DebugOnly<nsresult> rvIgnored =
+ AddNodeToChangedRange(*aEditorBase.AsHTMLEditor(), aNewElement);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "TopLevelEditSubActionData::AddNodeToChangedRange() failed, but ignored");
+}
+
+void EditorBase::TopLevelEditSubActionData::DidInsertContent(
+ EditorBase& aEditorBase, nsIContent& aNewContent) {
+ MOZ_ASSERT(aEditorBase.AsHTMLEditor());
+
+ if (!aEditorBase.mInitSucceeded || aEditorBase.Destroyed()) {
+ return; // We have not been initialized yet or already been destroyed.
+ }
+
+ if (!aEditorBase.EditSubActionDataRef().mAdjustChangedRangeFromListener) {
+ return; // Temporarily disabled by edit sub-action handler.
+ }
+
+ DebugOnly<nsresult> rvIgnored =
+ AddNodeToChangedRange(*aEditorBase.AsHTMLEditor(), aNewContent);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "TopLevelEditSubActionData::AddNodeToChangedRange() failed, but ignored");
+}
+
+void EditorBase::TopLevelEditSubActionData::WillDeleteContent(
+ EditorBase& aEditorBase, nsIContent& aRemovingContent) {
+ MOZ_ASSERT(aEditorBase.AsHTMLEditor());
+
+ if (!aEditorBase.mInitSucceeded || aEditorBase.Destroyed()) {
+ return; // We have not been initialized yet or already been destroyed.
+ }
+
+ if (!aEditorBase.EditSubActionDataRef().mAdjustChangedRangeFromListener) {
+ return; // Temporarily disabled by edit sub-action handler.
+ }
+
+ DebugOnly<nsresult> rvIgnored =
+ AddNodeToChangedRange(*aEditorBase.AsHTMLEditor(), aRemovingContent);
+ NS_WARNING_ASSERTION(
+ NS_SUCCEEDED(rvIgnored),
+ "TopLevelEditSubActionData::AddNodeToChangedRange() failed, but ignored");
+}
+
+void EditorBase::TopLevelEditSubActionData::DidSplitContent(
+ EditorBase& aEditorBase, nsIContent& aSplitContent, nsIContent& aNewContent,
+ SplitNodeDirection aSplitNodeDirection) {
+ MOZ_ASSERT(aEditorBase.AsHTMLEditor());
+
+ if (!aEditorBase.mInitSucceeded || aEditorBase.Destroyed()) {
+ return; // We have not been initialized yet or already been destroyed.
+ }
+
+ if (!aEditorBase.EditSubActionDataRef().mAdjustChangedRangeFromListener) {
+ return; // Temporarily disabled by edit sub-action handler.
+ }
+
+ DebugOnly<nsresult> rvIgnored =
+ aSplitNodeDirection == SplitNodeDirection::LeftNodeIsNewOne
+ ? AddRangeToChangedRange(*aEditorBase.AsHTMLEditor(),
+ EditorRawDOMPoint(&aNewContent, 0),
+ EditorRawDOMPoint(&aSplitContent, 0))
+ : AddRangeToChangedRange(*aEditorBase.AsHTMLEditor(),
+ EditorRawDOMPoint::AtEndOf(aSplitContent),
+ EditorRawDOMPoint::AtEndOf(aNewContent));
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "TopLevelEditSubActionData::AddRangeToChangedRange() "
+ "failed, but ignored");
+}
+
+void EditorBase::TopLevelEditSubActionData::DidJoinContents(
+ EditorBase& aEditorBase, const EditorRawDOMPoint& aJoinedPoint) {
+ MOZ_ASSERT(aEditorBase.AsHTMLEditor());
+
+ if (!aEditorBase.mInitSucceeded || aEditorBase.Destroyed()) {
+ return; // We have not been initialized yet or already been destroyed.
+ }
+
+ if (!aEditorBase.EditSubActionDataRef().mAdjustChangedRangeFromListener) {
+ return; // Temporarily disabled by edit sub-action handler.
+ }
+
+ DebugOnly<nsresult> rvIgnored =
+ AddPointToChangedRange(*aEditorBase.AsHTMLEditor(), aJoinedPoint);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "TopLevelEditSubActionData::AddPointToChangedRange() "
+ "failed, but ignored");
+}
+
+void EditorBase::TopLevelEditSubActionData::DidInsertText(
+ EditorBase& aEditorBase, const EditorRawDOMPoint& aInsertionBegin,
+ const EditorRawDOMPoint& aInsertionEnd) {
+ MOZ_ASSERT(aEditorBase.AsHTMLEditor());
+
+ if (!aEditorBase.mInitSucceeded || aEditorBase.Destroyed()) {
+ return; // We have not been initialized yet or already been destroyed.
+ }
+
+ if (!aEditorBase.EditSubActionDataRef().mAdjustChangedRangeFromListener) {
+ return; // Temporarily disabled by edit sub-action handler.
+ }
+
+ DebugOnly<nsresult> rvIgnored = AddRangeToChangedRange(
+ *aEditorBase.AsHTMLEditor(), aInsertionBegin, aInsertionEnd);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "TopLevelEditSubActionData::AddRangeToChangedRange() "
+ "failed, but ignored");
+}
+
+void EditorBase::TopLevelEditSubActionData::DidDeleteText(
+ EditorBase& aEditorBase, const EditorRawDOMPoint& aStartInTextNode) {
+ MOZ_ASSERT(aEditorBase.AsHTMLEditor());
+
+ if (!aEditorBase.mInitSucceeded || aEditorBase.Destroyed()) {
+ return; // We have not been initialized yet or already been destroyed.
+ }
+
+ if (!aEditorBase.EditSubActionDataRef().mAdjustChangedRangeFromListener) {
+ return; // Temporarily disabled by edit sub-action handler.
+ }
+
+ DebugOnly<nsresult> rvIgnored =
+ AddPointToChangedRange(*aEditorBase.AsHTMLEditor(), aStartInTextNode);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "TopLevelEditSubActionData::AddPointToChangedRange() "
+ "failed, but ignored");
+}
+
+void EditorBase::TopLevelEditSubActionData::WillDeleteRange(
+ EditorBase& aEditorBase, const EditorRawDOMPoint& aStart,
+ const EditorRawDOMPoint& aEnd) {
+ MOZ_ASSERT(aEditorBase.AsHTMLEditor());
+ MOZ_ASSERT(aStart.IsSet());
+ MOZ_ASSERT(aEnd.IsSet());
+
+ if (!aEditorBase.mInitSucceeded || aEditorBase.Destroyed()) {
+ return; // We have not been initialized yet or already been destroyed.
+ }
+
+ if (!aEditorBase.EditSubActionDataRef().mAdjustChangedRangeFromListener) {
+ return; // Temporarily disabled by edit sub-action handler.
+ }
+
+ // XXX Looks like that this is wrong. We delete multiple selection ranges
+ // once, but this adds only first range into the changed range.
+ // Anyway, we should take the range as an argument.
+ DebugOnly<nsresult> rvIgnored =
+ AddRangeToChangedRange(*aEditorBase.AsHTMLEditor(), aStart, aEnd);
+ NS_WARNING_ASSERTION(NS_SUCCEEDED(rvIgnored),
+ "TopLevelEditSubActionData::AddRangeToChangedRange() "
+ "failed, but ignored");
+}
+
+nsPIDOMWindowOuter* EditorBase::GetWindow() const {
+ return mDocument ? mDocument->GetWindow() : nullptr;
+}
+
+nsPIDOMWindowInner* EditorBase::GetInnerWindow() const {
+ return mDocument ? mDocument->GetInnerWindow() : nullptr;
+}
+
+PresShell* EditorBase::GetPresShell() const {
+ return mDocument ? mDocument->GetPresShell() : nullptr;
+}
+
+nsPresContext* EditorBase::GetPresContext() const {
+ PresShell* presShell = GetPresShell();
+ return presShell ? presShell->GetPresContext() : nullptr;
+}
+
+already_AddRefed<nsCaret> EditorBase::GetCaret() const {
+ PresShell* presShell = GetPresShell();
+ if (NS_WARN_IF(!presShell)) {
+ return nullptr;
+ }
+ return presShell->GetCaret();
+}
+
+nsISelectionController* EditorBase::GetSelectionController() const {
+ if (mSelectionController) {
+ return mSelectionController;
+ }
+ if (!mDocument) {
+ return nullptr;
+ }
+ return mDocument->GetPresShell();
+}
+
+} // namespace mozilla