/* -*- 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 "InsertTextTransaction.h" #include "mozilla/EditorBase.h" // mEditorBase #include "mozilla/Logging.h" #include "mozilla/SelectionState.h" // RangeUpdater #include "mozilla/ToString.h" #include "mozilla/dom/Selection.h" // Selection local var #include "mozilla/dom/Text.h" // mTextNode #include "nsAString.h" // nsAString parameter #include "nsDebug.h" // for NS_ASSERTION, etc. #include "nsError.h" // for NS_OK, etc. #include "nsQueryObject.h" // for do_QueryObject namespace mozilla { using namespace dom; // static already_AddRefed InsertTextTransaction::Create( EditorBase& aEditorBase, const nsAString& aStringToInsert, const EditorDOMPointInText& aPointToInsert) { MOZ_ASSERT(aPointToInsert.IsSetAndValid()); RefPtr transaction = new InsertTextTransaction(aEditorBase, aStringToInsert, aPointToInsert); return transaction.forget(); } InsertTextTransaction::InsertTextTransaction( EditorBase& aEditorBase, const nsAString& aStringToInsert, const EditorDOMPointInText& aPointToInsert) : mTextNode(aPointToInsert.ContainerAs()), mOffset(aPointToInsert.Offset()), mStringToInsert(aStringToInsert), mEditorBase(&aEditorBase) {} std::ostream& operator<<(std::ostream& aStream, const InsertTextTransaction& aTransaction) { aStream << "{ mTextNode=" << aTransaction.mTextNode.get(); if (aTransaction.mTextNode) { aStream << " (" << *aTransaction.mTextNode << ")"; } aStream << ", mOffset=" << aTransaction.mOffset << ", mStringToInsert=\"" << NS_ConvertUTF16toUTF8(aTransaction.mStringToInsert).get() << "\"" << ", mEditorBase=" << aTransaction.mEditorBase.get() << " }"; return aStream; } NS_IMPL_CYCLE_COLLECTION_INHERITED(InsertTextTransaction, EditTransactionBase, mEditorBase, mTextNode) NS_IMPL_ADDREF_INHERITED(InsertTextTransaction, EditTransactionBase) NS_IMPL_RELEASE_INHERITED(InsertTextTransaction, EditTransactionBase) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(InsertTextTransaction) NS_INTERFACE_MAP_END_INHERITING(EditTransactionBase) NS_IMETHODIMP InsertTextTransaction::DoTransaction() { MOZ_LOG(GetLogModule(), LogLevel::Info, ("%p InsertTextTransaction::%s this=%s", this, __FUNCTION__, ToString(*this).c_str())); if (NS_WARN_IF(!mEditorBase) || NS_WARN_IF(!mTextNode)) { return NS_ERROR_NOT_AVAILABLE; } OwningNonNull editorBase = *mEditorBase; OwningNonNull textNode = *mTextNode; ErrorResult error; editorBase->DoInsertText(textNode, mOffset, mStringToInsert, error); if (error.Failed()) { NS_WARNING("EditorBase::DoInsertText() failed"); return error.StealNSResult(); } // Only set selection to insertion point if editor gives permission if (editorBase->AllowsTransactionsToChangeSelection()) { RefPtr selection = editorBase->GetSelection(); if (NS_WARN_IF(!selection)) { return NS_ERROR_FAILURE; } DebugOnly rvIgnored = editorBase->CollapseSelectionTo( EditorRawDOMPoint(textNode, mOffset + mStringToInsert.Length())); NS_ASSERTION(NS_SUCCEEDED(rvIgnored), "EditorBase::CollapseSelectionTo() failed, but ignored"); // Keep handling to adjust the ranges in the range updater even if the // editor is destroyed. } else { // Do nothing - DOM Range gravity will adjust selection } // XXX Other transactions do not do this but its callers do. // Why do this transaction do this by itself? editorBase->RangeUpdaterRef().SelAdjInsertText(textNode, mOffset, mStringToInsert.Length()); return MOZ_UNLIKELY(editorBase->Destroyed()) ? NS_ERROR_EDITOR_DESTROYED : NS_OK; } NS_IMETHODIMP InsertTextTransaction::UndoTransaction() { MOZ_LOG(GetLogModule(), LogLevel::Info, ("%p InsertTextTransaction::%s this=%s", this, __FUNCTION__, ToString(*this).c_str())); if (NS_WARN_IF(!mEditorBase) || NS_WARN_IF(!mTextNode)) { return NS_ERROR_NOT_INITIALIZED; } OwningNonNull editorBase = *mEditorBase; OwningNonNull textNode = *mTextNode; ErrorResult error; editorBase->DoDeleteText(textNode, mOffset, mStringToInsert.Length(), error); NS_WARNING_ASSERTION(!error.Failed(), "EditorBase::DoDeleteText() failed"); return error.StealNSResult(); } NS_IMETHODIMP InsertTextTransaction::RedoTransaction() { MOZ_LOG(GetLogModule(), LogLevel::Info, ("%p InsertTextTransaction::%s this=%s", this, __FUNCTION__, ToString(*this).c_str())); return DoTransaction(); } NS_IMETHODIMP InsertTextTransaction::Merge(nsITransaction* aOtherTransaction, bool* aDidMerge) { MOZ_LOG(GetLogModule(), LogLevel::Debug, ("%p InsertTextTransaction::%s(aOtherTransaction=%p) this=%s", this, __FUNCTION__, aOtherTransaction, ToString(*this).c_str())); if (NS_WARN_IF(!aOtherTransaction) || NS_WARN_IF(!aDidMerge)) { return NS_ERROR_INVALID_ARG; } // Set out param default value *aDidMerge = false; RefPtr otherTransactionBase = aOtherTransaction->GetAsEditTransactionBase(); if (!otherTransactionBase) { MOZ_LOG( GetLogModule(), LogLevel::Debug, ("%p InsertTextTransaction::%s(aOtherTransaction=%p) returned false", this, __FUNCTION__, aOtherTransaction)); return NS_OK; } // If aTransaction is a InsertTextTransaction, and if the selection hasn't // changed, then absorb it. InsertTextTransaction* otherInsertTextTransaction = otherTransactionBase->GetAsInsertTextTransaction(); if (!otherInsertTextTransaction || !IsSequentialInsert(*otherInsertTextTransaction)) { MOZ_LOG( GetLogModule(), LogLevel::Debug, ("%p InsertTextTransaction::%s(aOtherTransaction=%p) returned false", this, __FUNCTION__, aOtherTransaction)); return NS_OK; } nsAutoString otherData; otherInsertTextTransaction->GetData(otherData); mStringToInsert += otherData; *aDidMerge = true; MOZ_LOG(GetLogModule(), LogLevel::Debug, ("%p InsertTextTransaction::%s(aOtherTransaction=%p) returned true", this, __FUNCTION__, aOtherTransaction)); return NS_OK; } /* ============ private methods ================== */ void InsertTextTransaction::GetData(nsString& aResult) { aResult = mStringToInsert; } bool InsertTextTransaction::IsSequentialInsert( InsertTextTransaction& aOtherTransaction) { return aOtherTransaction.mTextNode == mTextNode && aOtherTransaction.mOffset == mOffset + mStringToInsert.Length(); } } // namespace mozilla