/* -*- 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 "CreateElementTxn.h" #include "nsEditor.h" #include "nsIDOMDocument.h" #include "nsIDOMNodeList.h" #include "nsISelection.h" #include "nsIDOMElement.h" #include "nsReadableUtils.h" #include "mozilla/dom/Element.h" #ifdef NS_DEBUG static bool gNoisy = false; #endif using namespace mozilla; CreateElementTxn::CreateElementTxn() : EditTxn() { } NS_IMPL_CYCLE_COLLECTION_CLASS(CreateElementTxn) NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(CreateElementTxn, EditTxn) NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mParent) NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mNewNode) NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mRefNode) NS_IMPL_CYCLE_COLLECTION_UNLINK_END NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(CreateElementTxn, EditTxn) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mParent) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mNewNode) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mRefNode) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END NS_IMPL_ADDREF_INHERITED(CreateElementTxn, EditTxn) NS_IMPL_RELEASE_INHERITED(CreateElementTxn, EditTxn) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(CreateElementTxn) NS_INTERFACE_MAP_END_INHERITING(EditTxn) NS_IMETHODIMP CreateElementTxn::Init(nsEditor *aEditor, const nsAString &aTag, nsIDOMNode *aParent, PRUint32 aOffsetInParent) { NS_ASSERTION(aEditor&&aParent, "null args"); if (!aEditor || !aParent) { return NS_ERROR_NULL_POINTER; } mEditor = aEditor; mTag = aTag; mParent = do_QueryInterface(aParent); mOffsetInParent = aOffsetInParent; return NS_OK; } NS_IMETHODIMP CreateElementTxn::DoTransaction(void) { #ifdef NS_DEBUG if (gNoisy) { char* nodename = ToNewCString(mTag); printf("Do Create Element parent = %p <%s>, offset = %d\n", static_cast(mParent.get()), nodename, mOffsetInParent); nsMemory::Free(nodename); } #endif NS_ASSERTION(mEditor && mParent, "bad state"); NS_ENSURE_TRUE(mEditor && mParent, NS_ERROR_NOT_INITIALIZED); nsCOMPtr newContent; //new call to use instead to get proper HTML element, bug# 39919 nsresult result = mEditor->CreateHTMLContent(mTag, getter_AddRefs(newContent)); NS_ENSURE_SUCCESS(result, result); NS_ENSURE_STATE(newContent); mNewNode = newContent->AsDOMNode(); // Try to insert formatting whitespace for the new node: mEditor->MarkNodeDirty(mNewNode); #ifdef NS_DEBUG if (gNoisy) { printf(" newNode = %p\n", static_cast(mNewNode.get())); } #endif // insert the new node if (CreateElementTxn::eAppend == PRInt32(mOffsetInParent)) { nsCOMPtr resultNode; return mParent->AppendChild(mNewNode, getter_AddRefs(resultNode)); } nsCOMPtr parent = do_QueryInterface(mParent); NS_ENSURE_STATE(parent); mOffsetInParent = NS_MIN(mOffsetInParent, parent->GetChildCount()); // note, it's ok for mRefNode to be null. that means append nsIContent* refNode = parent->GetChildAt(mOffsetInParent); mRefNode = refNode ? refNode->AsDOMNode() : nsnull; nsCOMPtr resultNode; result = mParent->InsertBefore(mNewNode, mRefNode, getter_AddRefs(resultNode)); NS_ENSURE_SUCCESS(result, result); // only set selection to insertion point if editor gives permission bool bAdjustSelection; mEditor->ShouldTxnSetSelection(&bAdjustSelection); if (!bAdjustSelection) { // do nothing - dom range gravity will adjust selection return NS_OK; } nsCOMPtr selection; result = mEditor->GetSelection(getter_AddRefs(selection)); NS_ENSURE_SUCCESS(result, result); NS_ENSURE_TRUE(selection, NS_ERROR_NULL_POINTER); nsCOMPtr parentContent = do_QueryInterface(mParent); NS_ENSURE_STATE(parentContent); result = selection->CollapseNative(parentContent, parentContent->IndexOf(newContent) + 1); NS_ASSERTION((NS_SUCCEEDED(result)), "selection could not be collapsed after insert."); return result; } NS_IMETHODIMP CreateElementTxn::UndoTransaction(void) { #ifdef NS_DEBUG if (gNoisy) { printf("Undo Create Element, mParent = %p, node = %p\n", static_cast(mParent.get()), static_cast(mNewNode.get())); } #endif NS_ASSERTION(mEditor && mParent, "bad state"); NS_ENSURE_TRUE(mEditor && mParent, NS_ERROR_NOT_INITIALIZED); nsCOMPtr resultNode; return mParent->RemoveChild(mNewNode, getter_AddRefs(resultNode)); } NS_IMETHODIMP CreateElementTxn::RedoTransaction(void) { #ifdef NS_DEBUG if (gNoisy) { printf("Redo Create Element\n"); } #endif NS_ASSERTION(mEditor && mParent, "bad state"); NS_ENSURE_TRUE(mEditor && mParent, NS_ERROR_NOT_INITIALIZED); // first, reset mNewNode so it has no attributes or content nsCOMPtrnodeAsText = do_QueryInterface(mNewNode); if (nodeAsText) { nodeAsText->SetData(EmptyString()); } // now, reinsert mNewNode nsCOMPtr resultNode; return mParent->InsertBefore(mNewNode, mRefNode, getter_AddRefs(resultNode)); } NS_IMETHODIMP CreateElementTxn::GetTxnDescription(nsAString& aString) { aString.AssignLiteral("CreateElementTxn: "); aString += mTag; return NS_OK; } NS_IMETHODIMP CreateElementTxn::GetNewNode(nsIDOMNode **aNewNode) { NS_ENSURE_TRUE(aNewNode, NS_ERROR_NULL_POINTER); NS_ENSURE_TRUE(mNewNode, NS_ERROR_NOT_INITIALIZED); *aNewNode = mNewNode; NS_ADDREF(*aNewNode); return NS_OK; }