mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
8a42231835
--HG-- rename : editor/libeditor/text/nsEditorEventListeners.cpp => editor/libeditor/base/nsEditorEventListener.cpp rename : editor/libeditor/text/nsEditorEventListeners.h => editor/libeditor/base/nsEditorEventListener.h rename : editor/libeditor/html/nsHTMLEditorMouseListener.cpp => editor/libeditor/html/nsHTMLEditorEventListener.cpp rename : editor/libeditor/html/nsHTMLEditorMouseListener.h => editor/libeditor/html/nsHTMLEditorEventListener.h
5375 lines
156 KiB
C++
5375 lines
156 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is mozilla.org code.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Netscape Communications Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 1998
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Pierre Phaneuf <pp@ludusdesign.com>
|
|
* Daniel Glazman <glazman@netscape.com>
|
|
* Masayuki Nakano <masayuki@d-toybox.com>
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either of the GNU General Public License Version 2 or later (the "GPL"),
|
|
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "pratom.h"
|
|
#include "nsIDOMDocument.h"
|
|
#include "nsIDOMHTMLDocument.h"
|
|
#include "nsIDOMHTMLElement.h"
|
|
#include "nsIDOMNSHTMLElement.h"
|
|
#include "nsIDOMEventTarget.h"
|
|
#include "nsPIDOMEventTarget.h"
|
|
#include "nsIEventListenerManager.h"
|
|
#include "nsIPrefBranch.h"
|
|
#include "nsIPrefService.h"
|
|
#include "nsUnicharUtils.h"
|
|
#include "nsReadableUtils.h"
|
|
|
|
#include "nsIDOMText.h"
|
|
#include "nsIDOMElement.h"
|
|
#include "nsIDOMAttr.h"
|
|
#include "nsIDOMNode.h"
|
|
#include "nsIDOMDocumentFragment.h"
|
|
#include "nsIDOMNamedNodeMap.h"
|
|
#include "nsIDOMNodeList.h"
|
|
#include "nsIDOMRange.h"
|
|
#include "nsIDOMEventListener.h"
|
|
#include "nsIDOMEventGroup.h"
|
|
#include "nsIDOMMouseListener.h"
|
|
#include "nsIDOMFocusListener.h"
|
|
#include "nsIDOMTextListener.h"
|
|
#include "nsIDOMCompositionListener.h"
|
|
#include "nsIDOMHTMLBRElement.h"
|
|
#include "nsIDocument.h"
|
|
#include "nsITransactionManager.h"
|
|
#include "nsIAbsorbingTransaction.h"
|
|
#include "nsIPresShell.h"
|
|
#include "nsIViewManager.h"
|
|
#include "nsISelection.h"
|
|
#include "nsISelectionPrivate.h"
|
|
#include "nsISelectionController.h"
|
|
#include "nsIEnumerator.h"
|
|
#include "nsIAtom.h"
|
|
#include "nsCaret.h"
|
|
#include "nsIWidget.h"
|
|
#include "nsIPlaintextEditor.h"
|
|
#include "nsGUIEvent.h" // nsTextEventReply
|
|
|
|
#include "nsIFrame.h" // Needed by IME code
|
|
|
|
#include "nsICSSStyleSheet.h"
|
|
|
|
#include "nsIContent.h"
|
|
#include "nsServiceManagerUtils.h"
|
|
|
|
// transactions the editor knows how to build
|
|
#include "EditAggregateTxn.h"
|
|
#include "PlaceholderTxn.h"
|
|
#include "ChangeAttributeTxn.h"
|
|
#include "CreateElementTxn.h"
|
|
#include "InsertElementTxn.h"
|
|
#include "DeleteElementTxn.h"
|
|
#include "InsertTextTxn.h"
|
|
#include "DeleteTextTxn.h"
|
|
#include "DeleteRangeTxn.h"
|
|
#include "SplitElementTxn.h"
|
|
#include "JoinElementTxn.h"
|
|
#include "nsStyleSheetTxns.h"
|
|
#include "IMETextTxn.h"
|
|
#include "nsString.h"
|
|
|
|
#include "nsEditor.h"
|
|
#include "nsEditorUtils.h"
|
|
#include "nsEditorEventListener.h"
|
|
#include "nsISelectionDisplay.h"
|
|
#include "nsIInlineSpellChecker.h"
|
|
#include "nsINameSpaceManager.h"
|
|
#include "nsIHTMLDocument.h"
|
|
#include "nsIParserService.h"
|
|
|
|
#include "nsITransferable.h"
|
|
|
|
#define NS_ERROR_EDITOR_NO_SELECTION NS_ERROR_GENERATE_FAILURE(NS_ERROR_MODULE_EDITOR,1)
|
|
#define NS_ERROR_EDITOR_NO_TEXTNODE NS_ERROR_GENERATE_FAILURE(NS_ERROR_MODULE_EDITOR,2)
|
|
|
|
#ifdef NS_DEBUG_EDITOR
|
|
static PRBool gNoisy = PR_FALSE;
|
|
#endif
|
|
|
|
|
|
// Defined in nsEditorRegistration.cpp
|
|
extern nsIParserService *sParserService;
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
// nsEditor: base editor class implementation
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
nsEditor::nsEditor()
|
|
: mModCount(0)
|
|
, mPresShellWeak(nsnull)
|
|
, mViewManager(nsnull)
|
|
, mUpdateCount(0)
|
|
, mSpellcheckCheckboxState(eTriUnset)
|
|
, mPlaceHolderTxn(nsnull)
|
|
, mPlaceHolderName(nsnull)
|
|
, mPlaceHolderBatch(0)
|
|
, mSelState(nsnull)
|
|
, mSavedSel()
|
|
, mRangeUpdater()
|
|
, mAction(nsnull)
|
|
, mDirection(eNone)
|
|
, mIMETextNode(nsnull)
|
|
, mIMETextOffset(0)
|
|
, mIMEBufferLength(0)
|
|
, mInIMEMode(PR_FALSE)
|
|
, mIsIMEComposing(PR_FALSE)
|
|
, mShouldTxnSetSelection(PR_TRUE)
|
|
, mDidPreDestroy(PR_FALSE)
|
|
, mDocDirtyState(-1)
|
|
, mDocWeak(nsnull)
|
|
, mPhonetic(nsnull)
|
|
{
|
|
//initialize member variables here
|
|
}
|
|
|
|
nsEditor::~nsEditor()
|
|
{
|
|
mTxnMgr = nsnull;
|
|
|
|
delete mPhonetic;
|
|
|
|
NS_IF_RELEASE(mViewManager);
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(nsEditor)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(nsEditor)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mRootElement)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mInlineSpellChecker)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mTxnMgr)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mIMETextRangeList)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mIMETextNode)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMARRAY(mActionListeners)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMARRAY(mEditorObservers)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMARRAY(mDocStateListeners)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mEventTarget)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMPTR(mEventListener)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(nsEditor)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mRootElement)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mInlineSpellChecker)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mTxnMgr)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mIMETextRangeList)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mIMETextNode)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMARRAY(mActionListeners)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMARRAY(mEditorObservers)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMARRAY(mDocStateListeners)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mEventTarget)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMPTR(mEventListener)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(nsEditor)
|
|
NS_INTERFACE_MAP_ENTRY(nsIMutationObserver)
|
|
NS_INTERFACE_MAP_ENTRY(nsIPhonetic)
|
|
NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference)
|
|
NS_INTERFACE_MAP_ENTRY(nsIEditorIMESupport)
|
|
NS_INTERFACE_MAP_ENTRY(nsIEditor)
|
|
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIEditor)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF_AMBIGUOUS(nsEditor, nsIEditor)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE_AMBIGUOUS(nsEditor, nsIEditor)
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark nsIEditorMethods
|
|
#pragma mark -
|
|
#endif
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::Init(nsIDOMDocument *aDoc, nsIPresShell* aPresShell, nsIContent *aRoot, nsISelectionController *aSelCon, PRUint32 aFlags)
|
|
{
|
|
NS_PRECONDITION(nsnull!=aDoc && nsnull!=aPresShell, "bad arg");
|
|
if ((nsnull==aDoc) || (nsnull==aPresShell))
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
mFlags = aFlags;
|
|
mDocWeak = do_GetWeakReference(aDoc); // weak reference to doc
|
|
mPresShellWeak = do_GetWeakReference(aPresShell); // weak reference to pres shell
|
|
mSelConWeak = do_GetWeakReference(aSelCon); // weak reference to selectioncontroller
|
|
nsCOMPtr<nsIPresShell> ps = do_QueryReferent(mPresShellWeak);
|
|
if (!ps) return NS_ERROR_NOT_INITIALIZED;
|
|
|
|
//set up root element if we are passed one.
|
|
if (aRoot)
|
|
mRootElement = do_QueryInterface(aRoot);
|
|
|
|
nsCOMPtr<nsINode> document = do_QueryInterface(aDoc);
|
|
document->AddMutationObserver(this);
|
|
|
|
// Set up the DTD
|
|
// XXX - in the long run we want to get this from the document, but there
|
|
// is no way to do that right now. So we leave it null here and set
|
|
// up a nav html dtd in nsHTMLEditor::Init
|
|
|
|
mViewManager = ps->GetViewManager();
|
|
if (!mViewManager) {return NS_ERROR_NULL_POINTER;}
|
|
NS_ADDREF(mViewManager);
|
|
|
|
mUpdateCount=0;
|
|
|
|
/* initialize IME stuff */
|
|
mIMETextNode = nsnull;
|
|
mIMETextOffset = 0;
|
|
mIMEBufferLength = 0;
|
|
|
|
/* Show the caret */
|
|
aSelCon->SetCaretReadOnly(PR_FALSE);
|
|
aSelCon->SetDisplaySelection(nsISelectionController::SELECTION_ON);
|
|
|
|
aSelCon->SetSelectionFlags(nsISelectionDisplay::DISPLAY_ALL);//we want to see all the selection reflected to user
|
|
|
|
#if 1
|
|
// THIS BLOCK CAUSES ASSERTIONS because sometimes we don't yet have
|
|
// a moz-br but we do have a presshell.
|
|
|
|
// Set the selection to the beginning:
|
|
|
|
//hack to get around this for now.
|
|
nsCOMPtr<nsIPresShell> shell = do_QueryReferent(mSelConWeak);
|
|
if (shell)
|
|
BeginningOfDocument();
|
|
#endif
|
|
|
|
NS_POSTCONDITION(mDocWeak && mPresShellWeak, "bad state");
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::PostCreate()
|
|
{
|
|
// Set up spellchecking
|
|
nsresult rv = SyncRealTimeSpell();
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// Set up listeners
|
|
rv = CreateEventListeners();
|
|
if (NS_FAILED(rv))
|
|
{
|
|
RemoveEventListeners();
|
|
|
|
return rv;
|
|
}
|
|
|
|
rv = InstallEventListeners();
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// nuke the modification count, so the doc appears unmodified
|
|
// do this before we notify listeners
|
|
ResetModificationCount();
|
|
|
|
// update the UI with our state
|
|
NotifyDocumentListeners(eDocumentCreated);
|
|
NotifyDocumentListeners(eDocumentStateChanged);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::CreateEventListeners()
|
|
{
|
|
NS_ENSURE_TRUE(!mEventListener, NS_ERROR_ALREADY_INITIALIZED);
|
|
mEventListener = do_QueryInterface(
|
|
static_cast<nsIDOMKeyListener*>(new nsEditorEventListener(this)));
|
|
NS_ENSURE_TRUE(mEventListener, NS_ERROR_OUT_OF_MEMORY);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::InstallEventListeners()
|
|
{
|
|
NS_ENSURE_TRUE(mDocWeak && mPresShellWeak && mEventListener,
|
|
NS_ERROR_NOT_INITIALIZED);
|
|
|
|
nsCOMPtr<nsPIDOMEventTarget> piTarget = GetPIDOMEventTarget();
|
|
|
|
if (!piTarget) {
|
|
RemoveEventListeners();
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
// register the event listeners with the listener manager
|
|
nsCOMPtr<nsIDOMEventGroup> sysGroup;
|
|
piTarget->GetSystemEventGroup(getter_AddRefs(sysGroup));
|
|
nsIEventListenerManager* elmP = piTarget->GetListenerManager(PR_TRUE);
|
|
|
|
if (sysGroup && elmP)
|
|
{
|
|
rv = elmP->AddEventListenerByType(mEventListener,
|
|
NS_LITERAL_STRING("keypress"),
|
|
NS_EVENT_FLAG_BUBBLE |
|
|
NS_PRIV_EVENT_UNTRUSTED_PERMITTED,
|
|
sysGroup);
|
|
NS_ASSERTION(NS_SUCCEEDED(rv),
|
|
"failed to register key listener in system group");
|
|
}
|
|
|
|
rv |= piTarget->AddEventListenerByIID(mEventListener,
|
|
NS_GET_IID(nsIDOMMouseListener));
|
|
|
|
if (elmP) {
|
|
// Focus event doesn't bubble so adding the listener to capturing phase.
|
|
// Make sure this works after bug 235441 gets fixed.
|
|
rv |= elmP->AddEventListenerByIID(mEventListener,
|
|
NS_GET_IID(nsIDOMFocusListener),
|
|
NS_EVENT_FLAG_CAPTURE);
|
|
}
|
|
|
|
rv |= piTarget->AddEventListenerByIID(mEventListener,
|
|
NS_GET_IID(nsIDOMTextListener));
|
|
|
|
rv |= piTarget->AddEventListenerByIID(mEventListener,
|
|
NS_GET_IID(nsIDOMCompositionListener));
|
|
|
|
nsCOMPtr<nsIDOMEventTarget> target(do_QueryInterface(piTarget));
|
|
if (target) {
|
|
// See bug 455215, we cannot use the standard dragstart event yet
|
|
rv |= target->AddEventListener(NS_LITERAL_STRING("draggesture"),
|
|
mEventListener, PR_FALSE);
|
|
rv |= target->AddEventListener(NS_LITERAL_STRING("dragenter"),
|
|
mEventListener, PR_FALSE);
|
|
rv |= target->AddEventListener(NS_LITERAL_STRING("dragover"),
|
|
mEventListener, PR_FALSE);
|
|
rv |= target->AddEventListener(NS_LITERAL_STRING("dragleave"),
|
|
mEventListener, PR_FALSE);
|
|
rv |= target->AddEventListener(NS_LITERAL_STRING("drop"),
|
|
mEventListener, PR_FALSE);
|
|
}
|
|
|
|
if (NS_FAILED(rv))
|
|
{
|
|
NS_ERROR("failed to register some event listeners");
|
|
|
|
RemoveEventListeners();
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
void
|
|
nsEditor::RemoveEventListeners()
|
|
{
|
|
if (!mDocWeak || !mEventListener)
|
|
{
|
|
return;
|
|
}
|
|
|
|
nsCOMPtr<nsPIDOMEventTarget> piTarget = GetPIDOMEventTarget();
|
|
|
|
if (piTarget)
|
|
{
|
|
// unregister the event listeners with the DOM event target
|
|
nsCOMPtr<nsIEventListenerManager> elmP =
|
|
piTarget->GetListenerManager(PR_TRUE);
|
|
nsCOMPtr<nsIDOMEventGroup> sysGroup;
|
|
piTarget->GetSystemEventGroup(getter_AddRefs(sysGroup));
|
|
if (sysGroup && elmP)
|
|
{
|
|
elmP->RemoveEventListenerByType(mEventListener,
|
|
NS_LITERAL_STRING("keypress"),
|
|
NS_EVENT_FLAG_BUBBLE |
|
|
NS_PRIV_EVENT_UNTRUSTED_PERMITTED,
|
|
sysGroup);
|
|
}
|
|
|
|
piTarget->RemoveEventListenerByIID(mEventListener,
|
|
NS_GET_IID(nsIDOMMouseListener));
|
|
|
|
elmP->RemoveEventListenerByIID(mEventListener,
|
|
NS_GET_IID(nsIDOMFocusListener),
|
|
NS_EVENT_FLAG_CAPTURE);
|
|
|
|
piTarget->RemoveEventListenerByIID(mEventListener,
|
|
NS_GET_IID(nsIDOMTextListener));
|
|
|
|
piTarget->RemoveEventListenerByIID(mEventListener,
|
|
NS_GET_IID(nsIDOMCompositionListener));
|
|
|
|
nsCOMPtr<nsIDOMEventTarget> target(do_QueryInterface(piTarget));
|
|
if (target) {
|
|
target->RemoveEventListener(NS_LITERAL_STRING("draggesture"),
|
|
mEventListener, PR_FALSE);
|
|
target->RemoveEventListener(NS_LITERAL_STRING("dragenter"),
|
|
mEventListener, PR_FALSE);
|
|
target->RemoveEventListener(NS_LITERAL_STRING("dragover"),
|
|
mEventListener, PR_FALSE);
|
|
target->RemoveEventListener(NS_LITERAL_STRING("dragleave"),
|
|
mEventListener, PR_FALSE);
|
|
target->RemoveEventListener(NS_LITERAL_STRING("drop"),
|
|
mEventListener, PR_FALSE);
|
|
}
|
|
}
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::GetDesiredSpellCheckState()
|
|
{
|
|
// Check user override on this element
|
|
if (mSpellcheckCheckboxState != eTriUnset) {
|
|
return (mSpellcheckCheckboxState == eTriTrue);
|
|
}
|
|
|
|
// Check user preferences
|
|
nsresult rv;
|
|
nsCOMPtr<nsIPrefBranch> prefBranch =
|
|
do_GetService(NS_PREFSERVICE_CONTRACTID, &rv);
|
|
PRInt32 spellcheckLevel = 1;
|
|
if (NS_SUCCEEDED(rv) && prefBranch) {
|
|
prefBranch->GetIntPref("layout.spellcheckDefault", &spellcheckLevel);
|
|
}
|
|
|
|
if (spellcheckLevel == 0) {
|
|
return PR_FALSE; // Spellchecking forced off globally
|
|
}
|
|
|
|
// Check for password/readonly/disabled, which are not spellchecked
|
|
// regardless of DOM
|
|
PRUint32 flags;
|
|
if (NS_SUCCEEDED(GetFlags(&flags)) &&
|
|
flags & (nsIPlaintextEditor::eEditorPasswordMask |
|
|
nsIPlaintextEditor::eEditorReadonlyMask |
|
|
nsIPlaintextEditor::eEditorDisabledMask)) {
|
|
return PR_FALSE;
|
|
}
|
|
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
rv = GetPresShell(getter_AddRefs(presShell));
|
|
if (NS_SUCCEEDED(rv)) {
|
|
nsPresContext* context = presShell->GetPresContext();
|
|
if (context && !context->IsDynamic()) {
|
|
return PR_FALSE;
|
|
}
|
|
}
|
|
|
|
// Check DOM state
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(GetRoot());
|
|
if (!content) {
|
|
return PR_FALSE;
|
|
}
|
|
|
|
if (content->IsRootOfNativeAnonymousSubtree()) {
|
|
content = content->GetParent();
|
|
}
|
|
|
|
nsCOMPtr<nsIDOMNSHTMLElement> element = do_QueryInterface(content);
|
|
if (!element) {
|
|
return PR_FALSE;
|
|
}
|
|
|
|
PRBool enable;
|
|
element->GetSpellcheck(&enable);
|
|
|
|
return enable;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::PreDestroy(PRBool aDestroyingFrames)
|
|
{
|
|
if (mDidPreDestroy)
|
|
return NS_OK;
|
|
|
|
// 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 NULL could free the
|
|
// object that is still in use! It will be freed when the editor is
|
|
// destroyed.
|
|
if (mInlineSpellChecker)
|
|
mInlineSpellChecker->Cleanup(aDestroyingFrames);
|
|
|
|
// tell our listeners that the doc is going away
|
|
NotifyDocumentListeners(eDocumentToBeDestroyed);
|
|
|
|
nsCOMPtr<nsINode> document = do_QueryReferent(mDocWeak);
|
|
if (document)
|
|
document->RemoveMutationObserver(this);
|
|
|
|
// Unregister event listeners
|
|
RemoveEventListeners();
|
|
mActionListeners.Clear();
|
|
mEditorObservers.Clear();
|
|
mDocStateListeners.Clear();
|
|
mInlineSpellChecker = nsnull;
|
|
|
|
mDidPreDestroy = PR_TRUE;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetFlags(PRUint32 *aFlags)
|
|
{
|
|
*aFlags = mFlags;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::SetFlags(PRUint32 aFlags)
|
|
{
|
|
mFlags = aFlags;
|
|
|
|
// Changing the flags can change whether spellchecking is on, so re-sync it
|
|
SyncRealTimeSpell();
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetIsDocumentEditable(PRBool *aIsDocumentEditable)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aIsDocumentEditable);
|
|
nsCOMPtr<nsIDOMDocument> doc;
|
|
GetDocument(getter_AddRefs(doc));
|
|
*aIsDocumentEditable = doc ? PR_TRUE : PR_FALSE;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetDocument(nsIDOMDocument **aDoc)
|
|
{
|
|
if (!aDoc)
|
|
return NS_ERROR_NULL_POINTER;
|
|
*aDoc = nsnull; // init out param
|
|
NS_PRECONDITION(mDocWeak, "bad state, mDocWeak weak pointer not initialized");
|
|
nsCOMPtr<nsIDOMDocument> doc = do_QueryReferent(mDocWeak);
|
|
if (!doc) return NS_ERROR_NOT_INITIALIZED;
|
|
NS_ADDREF(*aDoc = doc);
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
nsresult
|
|
nsEditor::GetPresShell(nsIPresShell **aPS)
|
|
{
|
|
if (!aPS)
|
|
return NS_ERROR_NULL_POINTER;
|
|
*aPS = nsnull; // init out param
|
|
NS_PRECONDITION(mPresShellWeak, "bad state, null mPresShellWeak");
|
|
nsCOMPtr<nsIPresShell> ps = do_QueryReferent(mPresShellWeak);
|
|
if (!ps) return NS_ERROR_NOT_INITIALIZED;
|
|
NS_ADDREF(*aPS = ps);
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
/* attribute string contentsMIMEType; */
|
|
NS_IMETHODIMP
|
|
nsEditor::GetContentsMIMEType(char * *aContentsMIMEType)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aContentsMIMEType);
|
|
*aContentsMIMEType = ToNewCString(mContentMIMEType);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::SetContentsMIMEType(const char * aContentsMIMEType)
|
|
{
|
|
mContentMIMEType.Assign(aContentsMIMEType ? aContentsMIMEType : "");
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetSelectionController(nsISelectionController **aSel)
|
|
{
|
|
if (!aSel)
|
|
return NS_ERROR_NULL_POINTER;
|
|
*aSel = nsnull; // init out param
|
|
NS_PRECONDITION(mSelConWeak, "bad state, null mSelConWeak");
|
|
nsCOMPtr<nsISelectionController> selCon = do_QueryReferent(mSelConWeak);
|
|
if (!selCon) return NS_ERROR_NOT_INITIALIZED;
|
|
NS_ADDREF(*aSel = selCon);
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DeleteSelection(EDirection aAction)
|
|
{
|
|
return DeleteSelectionImpl(aAction);
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetSelection(nsISelection **aSelection)
|
|
{
|
|
if (!aSelection)
|
|
return NS_ERROR_NULL_POINTER;
|
|
*aSelection = nsnull;
|
|
nsCOMPtr<nsISelectionController> selcon = do_QueryReferent(mSelConWeak);
|
|
if (!selcon) return NS_ERROR_NOT_INITIALIZED;
|
|
return selcon->GetSelection(nsISelectionController::SELECTION_NORMAL, aSelection); // does an addref
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DoTransaction(nsITransaction *aTxn)
|
|
{
|
|
#ifdef NS_DEBUG_EDITOR
|
|
if (gNoisy) { printf("Editor::DoTransaction ----------\n"); }
|
|
#endif
|
|
|
|
nsresult result = NS_OK;
|
|
|
|
if (mPlaceHolderBatch && !mPlaceHolderTxn)
|
|
{
|
|
// it's pretty darn amazing how many different types of pointers
|
|
// this transaction goes through here. I bet this is a record.
|
|
|
|
// We start off with an EditTxn since that's what the factory returns.
|
|
nsRefPtr<EditTxn> editTxn = new PlaceholderTxn();
|
|
if (!editTxn) { return NS_ERROR_OUT_OF_MEMORY; }
|
|
|
|
// Then we QI to an nsIAbsorbingTransaction to get at placeholder functionality
|
|
nsCOMPtr<nsIAbsorbingTransaction> plcTxn;
|
|
editTxn->QueryInterface(NS_GET_IID(nsIAbsorbingTransaction), getter_AddRefs(plcTxn));
|
|
// have to use line above instead of "plcTxn = do_QueryInterface(editTxn);"
|
|
// due to our broken interface model for transactions.
|
|
|
|
// save off weak reference to placeholder txn
|
|
mPlaceHolderTxn = do_GetWeakReference(plcTxn);
|
|
plcTxn->Init(mPlaceHolderName, mSelState, this);
|
|
mSelState = nsnull; // placeholder txn took ownership of this pointer
|
|
|
|
// finally we QI to an nsITransaction since that's what DoTransaction() expects
|
|
nsCOMPtr<nsITransaction> theTxn = do_QueryInterface(plcTxn);
|
|
DoTransaction(theTxn); // we will recurse, but will not hit this case in the nested call
|
|
|
|
if (mTxnMgr)
|
|
{
|
|
nsCOMPtr<nsITransaction> topTxn;
|
|
result = mTxnMgr->PeekUndoStack(getter_AddRefs(topTxn));
|
|
if (NS_FAILED(result)) return result;
|
|
if (topTxn)
|
|
{
|
|
plcTxn = do_QueryInterface(topTxn);
|
|
if (plcTxn)
|
|
{
|
|
// there is a palceholder 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.
|
|
mPlaceHolderTxn = do_GetWeakReference(plcTxn);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aTxn)
|
|
{
|
|
// 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 nsAutoUpdateViewBatch 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
|
|
nsCOMPtr<nsISelection>selection;
|
|
result = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(result)) { return result; }
|
|
if (!selection) { return NS_ERROR_NULL_POINTER; }
|
|
nsCOMPtr<nsISelectionPrivate>selPrivate(do_QueryInterface(selection));
|
|
|
|
selPrivate->StartBatchChanges();
|
|
|
|
if (mTxnMgr) {
|
|
result = mTxnMgr->DoTransaction(aTxn);
|
|
}
|
|
else {
|
|
result = aTxn->DoTransaction();
|
|
}
|
|
if (NS_SUCCEEDED(result)) {
|
|
result = DoAfterDoTransaction(aTxn);
|
|
}
|
|
|
|
selPrivate->EndBatchChanges(); // no need to check result here, don't lose result of operation
|
|
}
|
|
|
|
NS_POSTCONDITION((NS_SUCCEEDED(result)), "transaction did not execute properly");
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::EnableUndo(PRBool aEnable)
|
|
{
|
|
nsresult result=NS_OK;
|
|
|
|
if (PR_TRUE==aEnable)
|
|
{
|
|
if (!mTxnMgr)
|
|
{
|
|
mTxnMgr = do_CreateInstance(NS_TRANSACTIONMANAGER_CONTRACTID, &result);
|
|
if (NS_FAILED(result) || !mTxnMgr) {
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
}
|
|
}
|
|
mTxnMgr->SetMaxTransactionCount(-1);
|
|
}
|
|
else
|
|
{ // disable the transaction manager if it is enabled
|
|
if (mTxnMgr)
|
|
{
|
|
mTxnMgr->Clear();
|
|
mTxnMgr->SetMaxTransactionCount(0);
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetTransactionManager(nsITransactionManager* *aTxnManager)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aTxnManager);
|
|
|
|
*aTxnManager = NULL;
|
|
if (!mTxnMgr)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
NS_ADDREF(*aTxnManager = mTxnMgr);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::SetTransactionManager(nsITransactionManager *aTxnManager)
|
|
{
|
|
NS_ENSURE_TRUE(aTxnManager, NS_ERROR_FAILURE);
|
|
|
|
mTxnMgr = aTxnManager;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::Undo(PRUint32 aCount)
|
|
{
|
|
#ifdef NS_DEBUG_EDITOR
|
|
if (gNoisy) { printf("Editor::Undo ----------\n"); }
|
|
#endif
|
|
|
|
nsresult result = NS_OK;
|
|
ForceCompositionEnd();
|
|
|
|
PRBool hasTxnMgr, hasTransaction = PR_FALSE;
|
|
CanUndo(&hasTxnMgr, &hasTransaction);
|
|
if (!hasTransaction)
|
|
return result;
|
|
|
|
nsAutoRules beginRulesSniffing(this, kOpUndo, nsIEditor::eNone);
|
|
|
|
if ((nsITransactionManager *)nsnull!=mTxnMgr.get())
|
|
{
|
|
PRUint32 i=0;
|
|
for ( ; i<aCount; i++)
|
|
{
|
|
result = mTxnMgr->UndoTransaction();
|
|
|
|
if (NS_SUCCEEDED(result))
|
|
result = DoAfterUndoTransaction();
|
|
|
|
if (NS_FAILED(result))
|
|
break;
|
|
}
|
|
}
|
|
|
|
NotifyEditorObservers();
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::CanUndo(PRBool *aIsEnabled, PRBool *aCanUndo)
|
|
{
|
|
if (!aIsEnabled || !aCanUndo)
|
|
return NS_ERROR_NULL_POINTER;
|
|
*aIsEnabled = ((PRBool)((nsITransactionManager *)0!=mTxnMgr.get()));
|
|
if (*aIsEnabled)
|
|
{
|
|
PRInt32 numTxns=0;
|
|
mTxnMgr->GetNumberOfUndoItems(&numTxns);
|
|
*aCanUndo = ((PRBool)(0!=numTxns));
|
|
}
|
|
else {
|
|
*aCanUndo = PR_FALSE;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::Redo(PRUint32 aCount)
|
|
{
|
|
#ifdef NS_DEBUG_EDITOR
|
|
if (gNoisy) { printf("Editor::Redo ----------\n"); }
|
|
#endif
|
|
|
|
nsresult result = NS_OK;
|
|
|
|
PRBool hasTxnMgr, hasTransaction = PR_FALSE;
|
|
CanRedo(&hasTxnMgr, &hasTransaction);
|
|
if (!hasTransaction)
|
|
return result;
|
|
|
|
nsAutoRules beginRulesSniffing(this, kOpRedo, nsIEditor::eNone);
|
|
|
|
if ((nsITransactionManager *)nsnull!=mTxnMgr.get())
|
|
{
|
|
PRUint32 i=0;
|
|
for ( ; i<aCount; i++)
|
|
{
|
|
result = mTxnMgr->RedoTransaction();
|
|
|
|
if (NS_SUCCEEDED(result))
|
|
result = DoAfterRedoTransaction();
|
|
|
|
if (NS_FAILED(result))
|
|
break;
|
|
}
|
|
}
|
|
|
|
NotifyEditorObservers();
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::CanRedo(PRBool *aIsEnabled, PRBool *aCanRedo)
|
|
{
|
|
if (!aIsEnabled || !aCanRedo)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aIsEnabled = ((PRBool)((nsITransactionManager *)0!=mTxnMgr.get()));
|
|
if (*aIsEnabled)
|
|
{
|
|
PRInt32 numTxns=0;
|
|
mTxnMgr->GetNumberOfRedoItems(&numTxns);
|
|
*aCanRedo = ((PRBool)(0!=numTxns));
|
|
}
|
|
else {
|
|
*aCanRedo = PR_FALSE;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::BeginTransaction()
|
|
{
|
|
BeginUpdateViewBatch();
|
|
|
|
if ((nsITransactionManager *)nsnull!=mTxnMgr.get())
|
|
{
|
|
mTxnMgr->BeginBatch();
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::EndTransaction()
|
|
{
|
|
if ((nsITransactionManager *)nsnull!=mTxnMgr.get())
|
|
{
|
|
mTxnMgr->EndBatch();
|
|
}
|
|
|
|
EndUpdateViewBatch();
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
// These two routines are similar to the above, but do not use
|
|
// the transaction managers batching feature. Instead we use
|
|
// a placeholder transaction to wrap up any further transaction
|
|
// while the batch is open. The advantage of this is that
|
|
// placeholder transactions can later merge, if needed. Merging
|
|
// is unavailable between transaction manager batches.
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::BeginPlaceHolderTransaction(nsIAtom *aName)
|
|
{
|
|
NS_PRECONDITION(mPlaceHolderBatch >= 0, "negative placeholder batch count!");
|
|
if (!mPlaceHolderBatch)
|
|
{
|
|
// time to turn on the batch
|
|
BeginUpdateViewBatch();
|
|
mPlaceHolderTxn = nsnull;
|
|
mPlaceHolderName = aName;
|
|
nsCOMPtr<nsISelection> selection;
|
|
nsresult res = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(res)) return res;
|
|
mSelState = new nsSelectionState();
|
|
if (!mSelState)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
mSelState->SaveSelection(selection);
|
|
}
|
|
mPlaceHolderBatch++;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::EndPlaceHolderTransaction()
|
|
{
|
|
NS_PRECONDITION(mPlaceHolderBatch > 0, "zero or negative placeholder batch count when ending batch!");
|
|
if (mPlaceHolderBatch == 1)
|
|
{
|
|
nsCOMPtr<nsISelection>selection;
|
|
GetSelection(getter_AddRefs(selection));
|
|
|
|
nsCOMPtr<nsISelectionPrivate>selPrivate(do_QueryInterface(selection));
|
|
|
|
// By making the assumption that no reflow happens during the calls
|
|
// to EndUpdateViewBatch and ScrollSelectionIntoView, 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.
|
|
if (selPrivate) {
|
|
selPrivate->SetCanCacheFrameOffset(PR_TRUE);
|
|
}
|
|
|
|
// time to turn off the batch
|
|
EndUpdateViewBatch();
|
|
// make sure selection is in view
|
|
|
|
// After ScrollSelectionIntoView(), the pending notifications might be
|
|
// flushed and PresShell/PresContext/Frames may be dead. See bug 418470.
|
|
ScrollSelectionIntoView(PR_FALSE);
|
|
|
|
// cached for frame offset are Not available now
|
|
if (selPrivate) {
|
|
selPrivate->SetCanCacheFrameOffset(PR_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.
|
|
delete mSelState;
|
|
mSelState = nsnull;
|
|
}
|
|
if (mPlaceHolderTxn) // we might have never made a placeholder if no action took place
|
|
{
|
|
nsCOMPtr<nsIAbsorbingTransaction> plcTxn = do_QueryReferent(mPlaceHolderTxn);
|
|
if (plcTxn)
|
|
{
|
|
plcTxn->EndPlaceHolderBatch();
|
|
}
|
|
else
|
|
{
|
|
// in the future we will check to make sure undo is off here,
|
|
// since that is the only known case where the placeholdertxn would disappear on us.
|
|
// For now just removing the assert.
|
|
}
|
|
// notify editor observers of action unless it is uncommitted IME
|
|
if (!mInIMEMode) NotifyEditorObservers();
|
|
}
|
|
}
|
|
mPlaceHolderBatch--;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::ShouldTxnSetSelection(PRBool *aResult)
|
|
{
|
|
if (!aResult) return NS_ERROR_NULL_POINTER;
|
|
*aResult = mShouldTxnSetSelection;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::SetShouldTxnSetSelection(PRBool aShould)
|
|
{
|
|
mShouldTxnSetSelection = aShould;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetDocumentIsEmpty(PRBool *aDocumentIsEmpty)
|
|
{
|
|
*aDocumentIsEmpty = PR_TRUE;
|
|
|
|
nsIDOMElement *rootElement = GetRoot();
|
|
if (!rootElement)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
PRBool hasChildNodes;
|
|
nsresult res = rootElement->HasChildNodes(&hasChildNodes);
|
|
|
|
*aDocumentIsEmpty = !hasChildNodes;
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
// XXX: the rule system should tell us which node to select all on (ie, the root, or the body)
|
|
NS_IMETHODIMP nsEditor::SelectAll()
|
|
{
|
|
if (!mDocWeak || !mPresShellWeak) { return NS_ERROR_NOT_INITIALIZED; }
|
|
ForceCompositionEnd();
|
|
|
|
nsCOMPtr<nsISelectionController> selCon = do_QueryReferent(mSelConWeak);
|
|
if (!selCon) return NS_ERROR_NOT_INITIALIZED;
|
|
nsCOMPtr<nsISelection> selection;
|
|
nsresult result = selCon->GetSelection(nsISelectionController::SELECTION_NORMAL, getter_AddRefs(selection));
|
|
if (NS_SUCCEEDED(result) && selection)
|
|
{
|
|
result = SelectEntireDocument(selection);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
NS_IMETHODIMP nsEditor::BeginningOfDocument()
|
|
{
|
|
if (!mDocWeak || !mPresShellWeak) { return NS_ERROR_NOT_INITIALIZED; }
|
|
|
|
// get the selection
|
|
nsCOMPtr<nsISelection> selection;
|
|
nsresult result = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(result))
|
|
return result;
|
|
if (!selection)
|
|
return NS_ERROR_NOT_INITIALIZED;
|
|
|
|
// get the root element
|
|
nsIDOMElement *rootElement = GetRoot();
|
|
if (!rootElement)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
// find first editable thingy
|
|
nsCOMPtr<nsIDOMNode> firstNode;
|
|
result = GetFirstEditableNode(rootElement, address_of(firstNode));
|
|
if (firstNode)
|
|
{
|
|
// if firstNode is text, set selection to beginning of the text node
|
|
if (IsTextNode(firstNode))
|
|
{
|
|
result = selection->Collapse(firstNode, 0);
|
|
}
|
|
else
|
|
{ // otherwise, it's a leaf node and we set the selection just in front of it
|
|
nsCOMPtr<nsIDOMNode> parentNode;
|
|
result = firstNode->GetParentNode(getter_AddRefs(parentNode));
|
|
if (NS_FAILED(result)) { return result; }
|
|
if (!parentNode) { return NS_ERROR_NULL_POINTER; }
|
|
PRInt32 offsetInParent;
|
|
result = nsEditor::GetChildOffset(firstNode, parentNode, offsetInParent);
|
|
if (NS_FAILED(result)) return result;
|
|
result = selection->Collapse(parentNode, offsetInParent);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// just the root node, set selection to inside the root
|
|
result = selection->Collapse(rootElement, 0);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::EndOfDocument()
|
|
{
|
|
if (!mDocWeak || !mPresShellWeak) { return NS_ERROR_NOT_INITIALIZED; }
|
|
nsresult res;
|
|
|
|
// get selection
|
|
nsCOMPtr<nsISelection> selection;
|
|
res = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(res)) return res;
|
|
if (!selection) return NS_ERROR_NULL_POINTER;
|
|
|
|
// get the root element
|
|
nsIDOMElement *rootElement = GetRoot();
|
|
if (!rootElement)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
// get the length of the rot element
|
|
PRUint32 len;
|
|
res = GetLengthOfDOMNode(rootElement, len);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
// set the selection to after the last child of the root element
|
|
return selection->Collapse(rootElement, (PRInt32)len);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetDocumentModified(PRBool *outDocModified)
|
|
{
|
|
if (!outDocModified)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
PRInt32 modCount = 0;
|
|
GetModificationCount(&modCount);
|
|
|
|
*outDocModified = (modCount != 0);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetDocumentCharacterSet(nsACString &characterSet)
|
|
{
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
|
|
nsresult rv = GetPresShell(getter_AddRefs(presShell));
|
|
if (NS_SUCCEEDED(rv))
|
|
{
|
|
nsIDocument *doc = presShell->GetDocument();
|
|
if (doc) {
|
|
characterSet = doc->GetDocumentCharacterSet();
|
|
return NS_OK;
|
|
}
|
|
rv = NS_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::SetDocumentCharacterSet(const nsACString& characterSet)
|
|
{
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
nsresult rv = GetPresShell(getter_AddRefs(presShell));
|
|
if (NS_SUCCEEDED(rv))
|
|
{
|
|
nsIDocument *doc = presShell->GetDocument();
|
|
if (doc) {
|
|
doc->SetDocumentCharacterSet(characterSet);
|
|
return NS_OK;
|
|
}
|
|
rv = NS_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
//
|
|
// Get an appropriate wrap width for saving this document.
|
|
// This class just uses a pref; subclasses are expected to
|
|
// override if they know more about the document.
|
|
//
|
|
NS_IMETHODIMP
|
|
nsEditor::GetWrapWidth(PRInt32 *aWrapColumn)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aWrapColumn);
|
|
*aWrapColumn = 72;
|
|
|
|
nsresult rv;
|
|
nsCOMPtr<nsIPrefBranch> prefBranch =
|
|
do_GetService(NS_PREFSERVICE_CONTRACTID, &rv);
|
|
if (NS_SUCCEEDED(rv) && prefBranch)
|
|
(void) prefBranch->GetIntPref("editor.htmlWrapColumn", aWrapColumn);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::Cut()
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CanCut(PRBool *aCanCut)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::Copy()
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CanCopy(PRBool *aCanCut)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::Paste(PRInt32 aSelectionType)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::PasteTransferable(nsITransferable *aTransferable)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CanPaste(PRInt32 aSelectionType, PRBool *aCanPaste)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CanPasteTransferable(nsITransferable *aTransferable, PRBool *aCanPaste)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CanDrag(nsIDOMEvent *aEvent, PRBool *aCanDrag)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DoDrag(nsIDOMEvent *aEvent)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::InsertFromDrop(nsIDOMEvent *aEvent)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::SetAttribute(nsIDOMElement *aElement, const nsAString & aAttribute, const nsAString & aValue)
|
|
{
|
|
nsRefPtr<ChangeAttributeTxn> txn;
|
|
nsresult result = CreateTxnForSetAttribute(aElement, aAttribute, aValue,
|
|
getter_AddRefs(txn));
|
|
if (NS_SUCCEEDED(result)) {
|
|
result = DoTransaction(txn);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetAttributeValue(nsIDOMElement *aElement,
|
|
const nsAString & aAttribute,
|
|
nsAString & aResultValue,
|
|
PRBool *aResultIsSet)
|
|
{
|
|
if (!aResultIsSet)
|
|
return NS_ERROR_NULL_POINTER;
|
|
*aResultIsSet=PR_FALSE;
|
|
nsresult result=NS_OK;
|
|
if (aElement)
|
|
{
|
|
nsCOMPtr<nsIDOMAttr> attNode;
|
|
result = aElement->GetAttributeNode(aAttribute, getter_AddRefs(attNode));
|
|
if ((NS_SUCCEEDED(result)) && attNode)
|
|
{
|
|
attNode->GetSpecified(aResultIsSet);
|
|
attNode->GetValue(aResultValue);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::RemoveAttribute(nsIDOMElement *aElement, const nsAString& aAttribute)
|
|
{
|
|
nsRefPtr<ChangeAttributeTxn> txn;
|
|
nsresult result = CreateTxnForRemoveAttribute(aElement, aAttribute,
|
|
getter_AddRefs(txn));
|
|
if (NS_SUCCEEDED(result)) {
|
|
result = DoTransaction(txn);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::MarkNodeDirty(nsIDOMNode* aNode)
|
|
{
|
|
// mark the node dirty.
|
|
nsCOMPtr<nsIDOMElement> element (do_QueryInterface(aNode));
|
|
if (element)
|
|
element->SetAttribute(NS_LITERAL_STRING("_moz_dirty"), EmptyString());
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsEditor::GetInlineSpellChecker(PRBool autoCreate,
|
|
nsIInlineSpellChecker ** aInlineSpellChecker)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aInlineSpellChecker);
|
|
|
|
if (mDidPreDestroy) {
|
|
// Don't allow people to get or create the spell checker once the editor
|
|
// is going away.
|
|
*aInlineSpellChecker = nsnull;
|
|
return autoCreate ? NS_ERROR_NOT_AVAILABLE : NS_OK;
|
|
}
|
|
|
|
nsresult rv;
|
|
if (!mInlineSpellChecker && autoCreate) {
|
|
mInlineSpellChecker = do_CreateInstance(MOZ_INLINESPELLCHECKER_CONTRACTID, &rv);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
|
|
if (mInlineSpellChecker) {
|
|
rv = mInlineSpellChecker->Init(this);
|
|
if (NS_FAILED(rv))
|
|
mInlineSpellChecker = nsnull;
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
|
|
NS_IF_ADDREF(*aInlineSpellChecker = mInlineSpellChecker);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsEditor::SyncRealTimeSpell()
|
|
{
|
|
PRBool enable = GetDesiredSpellCheckState();
|
|
|
|
nsCOMPtr<nsIInlineSpellChecker> spellChecker;
|
|
GetInlineSpellChecker(enable, getter_AddRefs(spellChecker));
|
|
|
|
if (spellChecker) {
|
|
spellChecker->SetEnableRealTimeSpell(enable);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsEditor::SetSpellcheckUserOverride(PRBool enable)
|
|
{
|
|
mSpellcheckCheckboxState = enable ? eTriTrue : eTriFalse;
|
|
|
|
return SyncRealTimeSpell();
|
|
}
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark main node manipulation routines
|
|
#pragma mark -
|
|
#endif
|
|
|
|
NS_IMETHODIMP nsEditor::CreateNode(const nsAString& aTag,
|
|
nsIDOMNode * aParent,
|
|
PRInt32 aPosition,
|
|
nsIDOMNode ** aNewNode)
|
|
{
|
|
PRInt32 i;
|
|
|
|
nsAutoRules beginRulesSniffing(this, kOpCreateNode, nsIEditor::eNext);
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillCreateNode(aTag, aParent, aPosition);
|
|
|
|
nsRefPtr<CreateElementTxn> txn;
|
|
nsresult result = CreateTxnForCreateElement(aTag, aParent, aPosition,
|
|
getter_AddRefs(txn));
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
result = DoTransaction(txn);
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
result = txn->GetNewNode(aNewNode);
|
|
NS_ASSERTION((NS_SUCCEEDED(result)), "GetNewNode can't fail if txn::DoTransaction succeeded.");
|
|
}
|
|
}
|
|
|
|
mRangeUpdater.SelAdjCreateNode(aParent, aPosition);
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->DidCreateNode(aTag, *aNewNode, aParent, aPosition, result);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::InsertNode(nsIDOMNode * aNode,
|
|
nsIDOMNode * aParent,
|
|
PRInt32 aPosition)
|
|
{
|
|
PRInt32 i;
|
|
nsAutoRules beginRulesSniffing(this, kOpInsertNode, nsIEditor::eNext);
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillInsertNode(aNode, aParent, aPosition);
|
|
|
|
nsRefPtr<InsertElementTxn> txn;
|
|
nsresult result = CreateTxnForInsertElement(aNode, aParent, aPosition,
|
|
getter_AddRefs(txn));
|
|
if (NS_SUCCEEDED(result)) {
|
|
result = DoTransaction(txn);
|
|
}
|
|
|
|
mRangeUpdater.SelAdjInsertNode(aParent, aPosition);
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->DidInsertNode(aNode, aParent, aPosition, result);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::SplitNode(nsIDOMNode * aNode,
|
|
PRInt32 aOffset,
|
|
nsIDOMNode **aNewLeftNode)
|
|
{
|
|
PRInt32 i;
|
|
nsAutoRules beginRulesSniffing(this, kOpSplitNode, nsIEditor::eNext);
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillSplitNode(aNode, aOffset);
|
|
|
|
nsRefPtr<SplitElementTxn> txn;
|
|
nsresult result = CreateTxnForSplitNode(aNode, aOffset, getter_AddRefs(txn));
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
result = DoTransaction(txn);
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
result = txn->GetNewNode(aNewLeftNode);
|
|
NS_ASSERTION((NS_SUCCEEDED(result)), "result must succeeded for GetNewNode");
|
|
}
|
|
}
|
|
|
|
mRangeUpdater.SelAdjSplitNode(aNode, aOffset, *aNewLeftNode);
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
{
|
|
nsIDOMNode *ptr = *aNewLeftNode;
|
|
mActionListeners[i]->DidSplitNode(aNode, aOffset, ptr, result);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::JoinNodes(nsIDOMNode * aLeftNode,
|
|
nsIDOMNode * aRightNode,
|
|
nsIDOMNode * aParent)
|
|
{
|
|
PRInt32 i, offset;
|
|
nsAutoRules beginRulesSniffing(this, kOpJoinNode, nsIEditor::ePrevious);
|
|
|
|
// remember some values; later used for saved selection updating.
|
|
// find the offset between the nodes to be joined.
|
|
nsresult result = GetChildOffset(aRightNode, aParent, offset);
|
|
if (NS_FAILED(result)) return result;
|
|
// find the number of children of the lefthand node
|
|
PRUint32 oldLeftNodeLen;
|
|
result = GetLengthOfDOMNode(aLeftNode, oldLeftNodeLen);
|
|
if (NS_FAILED(result)) return result;
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillJoinNodes(aLeftNode, aRightNode, aParent);
|
|
|
|
nsRefPtr<JoinElementTxn> txn;
|
|
result = CreateTxnForJoinNode(aLeftNode, aRightNode, getter_AddRefs(txn));
|
|
if (NS_SUCCEEDED(result)) {
|
|
result = DoTransaction(txn);
|
|
}
|
|
|
|
mRangeUpdater.SelAdjJoinNodes(aLeftNode, aRightNode, aParent, offset, (PRInt32)oldLeftNodeLen);
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->DidJoinNodes(aLeftNode, aRightNode, aParent, result);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::DeleteNode(nsIDOMNode * aElement)
|
|
{
|
|
PRInt32 i, offset;
|
|
nsCOMPtr<nsIDOMNode> parent;
|
|
nsAutoRules beginRulesSniffing(this, kOpCreateNode, nsIEditor::ePrevious);
|
|
|
|
// save node location for selection updating code.
|
|
nsresult result = GetNodeLocation(aElement, address_of(parent), &offset);
|
|
if (NS_FAILED(result)) return result;
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillDeleteNode(aElement);
|
|
|
|
nsRefPtr<DeleteElementTxn> txn;
|
|
result = CreateTxnForDeleteElement(aElement, getter_AddRefs(txn));
|
|
if (NS_SUCCEEDED(result)) {
|
|
result = DoTransaction(txn);
|
|
}
|
|
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->DidDeleteNode(aElement, result);
|
|
|
|
return result;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// ReplaceContainer: replace inNode with a new node (outNode) which is contructed
|
|
// to be of type aNodeType. Put inNodes children into outNode.
|
|
// Callers responsibility to make sure inNode's children can
|
|
// go in outNode.
|
|
nsresult
|
|
nsEditor::ReplaceContainer(nsIDOMNode *inNode,
|
|
nsCOMPtr<nsIDOMNode> *outNode,
|
|
const nsAString &aNodeType,
|
|
const nsAString *aAttribute,
|
|
const nsAString *aValue,
|
|
PRBool aCloneAttributes)
|
|
{
|
|
if (!inNode || !outNode)
|
|
return NS_ERROR_NULL_POINTER;
|
|
nsCOMPtr<nsIDOMNode> parent;
|
|
PRInt32 offset;
|
|
nsresult res = GetNodeLocation(inNode, address_of(parent), &offset);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
// create new container
|
|
nsCOMPtr<nsIContent> newContent;
|
|
|
|
//new call to use instead to get proper HTML element, bug# 39919
|
|
res = CreateHTMLContent(aNodeType, getter_AddRefs(newContent));
|
|
nsCOMPtr<nsIDOMElement> elem = do_QueryInterface(newContent);
|
|
if (NS_FAILED(res)) return res;
|
|
*outNode = do_QueryInterface(elem);
|
|
|
|
// set attribute if needed
|
|
if (aAttribute && aValue && !aAttribute->IsEmpty())
|
|
{
|
|
res = elem->SetAttribute(*aAttribute, *aValue);
|
|
if (NS_FAILED(res)) return res;
|
|
}
|
|
if (aCloneAttributes)
|
|
{
|
|
nsCOMPtr<nsIDOMNode>newNode = do_QueryInterface(elem);
|
|
res = CloneAttributes(newNode, inNode);
|
|
if (NS_FAILED(res)) return res;
|
|
}
|
|
|
|
// notify our internal selection state listener
|
|
// (Note: A nsAutoSelectionReset object must be created
|
|
// before calling this to initialize mRangeUpdater)
|
|
nsAutoReplaceContainerSelNotify selStateNotify(mRangeUpdater, inNode, *outNode);
|
|
{
|
|
nsAutoTxnsConserveSelection conserveSelection(this);
|
|
nsCOMPtr<nsIDOMNode> child;
|
|
PRBool bHasMoreChildren;
|
|
inNode->HasChildNodes(&bHasMoreChildren);
|
|
while (bHasMoreChildren)
|
|
{
|
|
inNode->GetFirstChild(getter_AddRefs(child));
|
|
res = DeleteNode(child);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
res = InsertNode(child, *outNode, -1);
|
|
if (NS_FAILED(res)) return res;
|
|
inNode->HasChildNodes(&bHasMoreChildren);
|
|
}
|
|
}
|
|
// insert new container into tree
|
|
res = InsertNode( *outNode, parent, offset);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
// delete old container
|
|
return DeleteNode(inNode);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// RemoveContainer: remove inNode, reparenting its children into their
|
|
// the parent of inNode
|
|
//
|
|
nsresult
|
|
nsEditor::RemoveContainer(nsIDOMNode *inNode)
|
|
{
|
|
if (!inNode)
|
|
return NS_ERROR_NULL_POINTER;
|
|
nsCOMPtr<nsIDOMNode> parent;
|
|
PRInt32 offset;
|
|
|
|
nsresult res = GetNodeLocation(inNode, address_of(parent), &offset);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
// loop through the child nodes of inNode and promote them
|
|
// into inNode's parent.
|
|
PRBool bHasMoreChildren;
|
|
inNode->HasChildNodes(&bHasMoreChildren);
|
|
nsCOMPtr<nsIDOMNodeList> nodeList;
|
|
res = inNode->GetChildNodes(getter_AddRefs(nodeList));
|
|
if (NS_FAILED(res)) return res;
|
|
if (!nodeList) return NS_ERROR_NULL_POINTER;
|
|
PRUint32 nodeOrigLen;
|
|
nodeList->GetLength(&nodeOrigLen);
|
|
|
|
// notify our internal selection state listener
|
|
nsAutoRemoveContainerSelNotify selNotify(mRangeUpdater, inNode, parent, offset, nodeOrigLen);
|
|
|
|
nsCOMPtr<nsIDOMNode> child;
|
|
while (bHasMoreChildren)
|
|
{
|
|
inNode->GetLastChild(getter_AddRefs(child));
|
|
res = DeleteNode(child);
|
|
if (NS_FAILED(res)) return res;
|
|
res = InsertNode(child, parent, offset);
|
|
if (NS_FAILED(res)) return res;
|
|
inNode->HasChildNodes(&bHasMoreChildren);
|
|
}
|
|
return DeleteNode(inNode);
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// InsertContainerAbove: insert a new parent for inNode, returned in outNode,
|
|
// which is contructed to be of type aNodeType. outNode becomes
|
|
// a child of inNode's earlier parent.
|
|
// Callers responsibility to make sure inNode's can be child
|
|
// of outNode, and outNode can be child of old parent.
|
|
nsresult
|
|
nsEditor::InsertContainerAbove( nsIDOMNode *inNode,
|
|
nsCOMPtr<nsIDOMNode> *outNode,
|
|
const nsAString &aNodeType,
|
|
const nsAString *aAttribute,
|
|
const nsAString *aValue)
|
|
{
|
|
if (!inNode || !outNode)
|
|
return NS_ERROR_NULL_POINTER;
|
|
nsCOMPtr<nsIDOMNode> parent;
|
|
PRInt32 offset;
|
|
nsresult res = GetNodeLocation(inNode, address_of(parent), &offset);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
// create new container
|
|
nsCOMPtr<nsIContent> newContent;
|
|
|
|
//new call to use instead to get proper HTML element, bug# 39919
|
|
res = CreateHTMLContent(aNodeType, getter_AddRefs(newContent));
|
|
nsCOMPtr<nsIDOMElement> elem = do_QueryInterface(newContent);
|
|
if (NS_FAILED(res)) return res;
|
|
*outNode = do_QueryInterface(elem);
|
|
|
|
// set attribute if needed
|
|
if (aAttribute && aValue && !aAttribute->IsEmpty())
|
|
{
|
|
res = elem->SetAttribute(*aAttribute, *aValue);
|
|
if (NS_FAILED(res)) return res;
|
|
}
|
|
|
|
// notify our internal selection state listener
|
|
nsAutoInsertContainerSelNotify selNotify(mRangeUpdater);
|
|
|
|
// put inNode in new parent, outNode
|
|
res = DeleteNode(inNode);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
{
|
|
nsAutoTxnsConserveSelection conserveSelection(this);
|
|
res = InsertNode(inNode, *outNode, 0);
|
|
if (NS_FAILED(res)) return res;
|
|
}
|
|
|
|
// put new parent in doc
|
|
return InsertNode(*outNode, parent, offset);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// MoveNode: move aNode to {aParent,aOffset}
|
|
nsresult
|
|
nsEditor::MoveNode(nsIDOMNode *aNode, nsIDOMNode *aParent, PRInt32 aOffset)
|
|
{
|
|
if (!aNode || !aParent)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
nsCOMPtr<nsIDOMNode> oldParent;
|
|
PRInt32 oldOffset;
|
|
nsresult res = GetNodeLocation(aNode, address_of(oldParent), &oldOffset);
|
|
|
|
if (aOffset == -1)
|
|
{
|
|
PRUint32 unsignedOffset;
|
|
// magic value meaning "move to end of aParent"
|
|
res = GetLengthOfDOMNode(aParent, unsignedOffset);
|
|
if (NS_FAILED(res)) return res;
|
|
aOffset = (PRInt32)unsignedOffset;
|
|
}
|
|
|
|
// don't do anything if it's already in right place
|
|
if ((aParent == oldParent.get()) && (oldOffset == aOffset)) return NS_OK;
|
|
|
|
// notify our internal selection state listener
|
|
nsAutoMoveNodeSelNotify selNotify(mRangeUpdater, oldParent, oldOffset, aParent, aOffset);
|
|
|
|
// need to adjust aOffset if we are moving aNode further along in its current parent
|
|
if ((aParent == oldParent.get()) && (oldOffset < aOffset))
|
|
{
|
|
aOffset--; // this is because when we delete aNode, it will make the offsets after it off by one
|
|
}
|
|
|
|
// put aNode in new parent
|
|
res = DeleteNode(aNode);
|
|
if (NS_FAILED(res)) return res;
|
|
return InsertNode(aNode, aParent, aOffset);
|
|
}
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark editor observer maintainance
|
|
#pragma mark -
|
|
#endif
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::AddEditorObserver(nsIEditorObserver *aObserver)
|
|
{
|
|
// we don't keep ownership of the observers. They must
|
|
// remove themselves as observers before they are destroyed.
|
|
|
|
if (!aObserver)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
// Make sure the listener isn't already on the list
|
|
if (mEditorObservers.IndexOf(aObserver) == -1)
|
|
{
|
|
if (!mEditorObservers.AppendObject(aObserver))
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::RemoveEditorObserver(nsIEditorObserver *aObserver)
|
|
{
|
|
if (!aObserver)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
if (!mEditorObservers.RemoveObject(aObserver))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void nsEditor::NotifyEditorObservers(void)
|
|
{
|
|
for (PRInt32 i = 0; i < mEditorObservers.Count(); i++)
|
|
mEditorObservers[i]->EditAction();
|
|
}
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark action listener maintainance
|
|
#pragma mark -
|
|
#endif
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::AddEditActionListener(nsIEditActionListener *aListener)
|
|
{
|
|
if (!aListener)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
// Make sure the listener isn't already on the list
|
|
if (mActionListeners.IndexOf(aListener) == -1)
|
|
{
|
|
if (!mActionListeners.AppendObject(aListener))
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::RemoveEditActionListener(nsIEditActionListener *aListener)
|
|
{
|
|
if (!aListener)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
if (!mActionListeners.RemoveObject(aListener))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark docstate listener maintainance
|
|
#pragma mark -
|
|
#endif
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::AddDocumentStateListener(nsIDocumentStateListener *aListener)
|
|
{
|
|
if (!aListener)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
if (mDocStateListeners.IndexOf(aListener) == -1)
|
|
{
|
|
if (!mDocStateListeners.AppendObject(aListener))
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::RemoveDocumentStateListener(nsIDocumentStateListener *aListener)
|
|
{
|
|
if (!aListener)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
if (!mDocStateListeners.RemoveObject(aListener))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark misc
|
|
#pragma mark -
|
|
#endif
|
|
|
|
NS_IMETHODIMP nsEditor::OutputToString(const nsAString& aFormatType,
|
|
PRUint32 aFlags,
|
|
nsAString& aOutputString)
|
|
{
|
|
// these should be implemented by derived classes.
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::OutputToStream(nsIOutputStream* aOutputStream,
|
|
const nsAString& aFormatType,
|
|
const nsACString& aCharsetOverride,
|
|
PRUint32 aFlags)
|
|
{
|
|
// these should be implemented by derived classes.
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DumpContentTree()
|
|
{
|
|
#ifdef DEBUG
|
|
nsCOMPtr<nsIContent> root = do_QueryInterface(mRootElement);
|
|
if (root) root->List(stdout);
|
|
#endif
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DebugDumpContent()
|
|
{
|
|
#ifdef DEBUG
|
|
nsCOMPtr<nsIDOMHTMLDocument> doc = do_QueryReferent(mDocWeak);
|
|
if (!doc) return NS_ERROR_NOT_INITIALIZED;
|
|
|
|
nsCOMPtr<nsIDOMHTMLElement>bodyElem;
|
|
doc->GetBody(getter_AddRefs(bodyElem));
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(bodyElem);
|
|
if (content)
|
|
content->List();
|
|
#endif
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DebugUnitTests(PRInt32 *outNumTests, PRInt32 *outNumTestsFailed)
|
|
{
|
|
#ifdef DEBUG
|
|
NS_NOTREACHED("This should never get called. Overridden by subclasses");
|
|
#endif
|
|
return NS_OK;
|
|
}
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark support for selection preservation
|
|
#pragma mark -
|
|
#endif
|
|
|
|
PRBool
|
|
nsEditor::ArePreservingSelection()
|
|
{
|
|
return !(mSavedSel.IsEmpty());
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::PreserveSelectionAcrossActions(nsISelection *aSel)
|
|
{
|
|
mSavedSel.SaveSelection(aSel);
|
|
mRangeUpdater.RegisterSelectionState(mSavedSel);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::RestorePreservedSelection(nsISelection *aSel)
|
|
{
|
|
if (mSavedSel.IsEmpty()) return NS_ERROR_FAILURE;
|
|
mSavedSel.RestoreSelection(aSel);
|
|
StopPreservingSelection();
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
nsEditor::StopPreservingSelection()
|
|
{
|
|
mRangeUpdater.DropSelectionState(mSavedSel);
|
|
mSavedSel.MakeEmpty();
|
|
}
|
|
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark nsIEditorIMESupport
|
|
#pragma mark -
|
|
#endif
|
|
|
|
//
|
|
// The BeingComposition method is called from the Editor Composition event listeners.
|
|
//
|
|
nsresult
|
|
nsEditor::QueryComposition(nsTextEventReply* aReply)
|
|
{
|
|
nsresult result;
|
|
nsCOMPtr<nsISelection> selection;
|
|
nsCOMPtr<nsISelectionController> selcon = do_QueryReferent(mSelConWeak);
|
|
if (selcon)
|
|
selcon->GetSelection(nsISelectionController::SELECTION_NORMAL, getter_AddRefs(selection));
|
|
|
|
if (!mPresShellWeak) return NS_ERROR_NOT_INITIALIZED;
|
|
nsCOMPtr<nsIPresShell> ps = do_QueryReferent(mPresShellWeak);
|
|
if (!ps) return NS_ERROR_NOT_INITIALIZED;
|
|
nsRefPtr<nsCaret> caretP;
|
|
result = ps->GetCaret(getter_AddRefs(caretP));
|
|
|
|
if (NS_SUCCEEDED(result) && caretP) {
|
|
if (aReply) {
|
|
caretP->SetCaretDOMSelection(selection);
|
|
|
|
// XXX_kin: BEGIN HACK! HACK! HACK!
|
|
// XXX_kin:
|
|
// XXX_kin: This is lame! The IME stuff needs caret coordinates
|
|
// XXX_kin: synchronously, but the editor could be using async
|
|
// XXX_kin: updates (reflows and paints) for performance reasons.
|
|
// XXX_kin: In order to give IME what it needs, we have to temporarily
|
|
// XXX_kin: switch to sync updating during this call so that the
|
|
// XXX_kin: nsAutoUpdateViewBatch can force sync reflows and paints
|
|
// XXX_kin: so that we get back accurate caret coordinates.
|
|
|
|
PRUint32 flags = 0;
|
|
|
|
if (NS_SUCCEEDED(GetFlags(&flags)) &&
|
|
(flags & nsIPlaintextEditor::eEditorUseAsyncUpdatesMask))
|
|
{
|
|
PRBool restoreFlags = PR_FALSE;
|
|
|
|
if (NS_SUCCEEDED(SetFlags(flags & (~nsIPlaintextEditor::eEditorUseAsyncUpdatesMask))))
|
|
{
|
|
// Scope the viewBatch within this |if| block so that we
|
|
// force synchronous reflows and paints before restoring
|
|
// our editor flags below.
|
|
|
|
nsAutoUpdateViewBatch viewBatch(this);
|
|
restoreFlags = PR_TRUE;
|
|
}
|
|
|
|
// Restore the previous set of flags!
|
|
|
|
if (restoreFlags)
|
|
SetFlags(flags);
|
|
}
|
|
|
|
|
|
// XXX_kin: END HACK! HACK! HACK!
|
|
|
|
nsIView *view = nsnull;
|
|
nsRect rect;
|
|
result =
|
|
caretP->GetCaretCoordinates(nsCaret::eRenderingViewCoordinates,
|
|
selection,
|
|
&rect,
|
|
&(aReply->mCursorIsCollapsed),
|
|
&view);
|
|
aReply->mCursorPosition =
|
|
rect.ToOutsidePixels(ps->GetPresContext()->AppUnitsPerDevPixel());
|
|
if (NS_SUCCEEDED(result) && view)
|
|
aReply->mReferenceWidget = view->GetWidget();
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::BeginComposition(nsTextEventReply* aReply)
|
|
{
|
|
#ifdef DEBUG_tague
|
|
printf("nsEditor::StartComposition\n");
|
|
#endif
|
|
nsresult ret = QueryComposition(aReply);
|
|
mInIMEMode = PR_TRUE;
|
|
if (mPhonetic)
|
|
mPhonetic->Truncate(0);
|
|
|
|
return ret;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::EndComposition(void)
|
|
{
|
|
if (!mInIMEMode) return NS_OK; // nothing to do
|
|
|
|
nsresult result = NS_OK;
|
|
|
|
// 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 (mTxnMgr)
|
|
{
|
|
nsCOMPtr<nsITransaction> txn;
|
|
result = mTxnMgr->PeekUndoStack(getter_AddRefs(txn));
|
|
nsCOMPtr<nsIAbsorbingTransaction> plcTxn = do_QueryInterface(txn);
|
|
if (plcTxn)
|
|
{
|
|
result = plcTxn->Commit();
|
|
}
|
|
}
|
|
|
|
/* reset the data we need to construct a transaction */
|
|
mIMETextNode = do_QueryInterface(nsnull);
|
|
mIMETextOffset = 0;
|
|
mIMEBufferLength = 0;
|
|
mInIMEMode = PR_FALSE;
|
|
mIsIMEComposing = PR_FALSE;
|
|
|
|
// notify editor observers of action
|
|
NotifyEditorObservers();
|
|
|
|
return result;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::SetCompositionString(const nsAString& aCompositionString, nsIPrivateTextRangeList* aTextRangeList,nsTextEventReply* aReply)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetPhonetic(nsAString& aPhonetic)
|
|
{
|
|
if (mPhonetic)
|
|
aPhonetic = *mPhonetic;
|
|
else
|
|
aPhonetic.Truncate(0);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
static nsresult
|
|
GetEditorContentWindow(nsIDOMElement *aRoot, nsIWidget **aResult)
|
|
{
|
|
if (!aRoot || !aResult)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aResult = 0;
|
|
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aRoot);
|
|
|
|
if (!content)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
// Not ref counted
|
|
nsIFrame *frame = content->GetPrimaryFrame();
|
|
|
|
if (!frame)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
*aResult = frame->GetWindow();
|
|
if (!*aResult)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
NS_ADDREF(*aResult);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::GetWidget(nsIWidget **aWidget)
|
|
{
|
|
if (!aWidget)
|
|
return NS_ERROR_NULL_POINTER;
|
|
*aWidget = nsnull;
|
|
|
|
nsCOMPtr<nsIWidget> widget;
|
|
nsresult res = GetEditorContentWindow(GetRoot(), getter_AddRefs(widget));
|
|
if (NS_FAILED(res))
|
|
return res;
|
|
if (!widget)
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
NS_ADDREF(*aWidget = widget);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::ForceCompositionEnd()
|
|
{
|
|
|
|
// We can test mInIMEMode and do some optimization for Mac and Window
|
|
// Howerver, since UNIX support over-the-spot, we cannot rely on that
|
|
// flag for Unix.
|
|
// We should use nsILookAndFeel to resolve this
|
|
|
|
#if defined(XP_MAC) || defined(XP_MACOSX) || defined(XP_WIN) || defined(XP_OS2)
|
|
if(! mInIMEMode)
|
|
return NS_OK;
|
|
#endif
|
|
|
|
#ifdef XP_UNIX
|
|
if(mFlags & nsIPlaintextEditor::eEditorPasswordMask)
|
|
return NS_OK;
|
|
#endif
|
|
|
|
nsCOMPtr<nsIWidget> widget;
|
|
nsresult res = GetWidget(getter_AddRefs(widget));
|
|
if (NS_FAILED(res))
|
|
return res;
|
|
|
|
if (widget) {
|
|
res = widget->ResetInputState();
|
|
if (NS_FAILED(res))
|
|
return res;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetPreferredIMEState(PRUint32 *aState)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aState);
|
|
*aState = nsIContent::IME_STATUS_ENABLE;
|
|
|
|
PRUint32 flags;
|
|
nsresult rv = GetFlags(&flags);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
if (flags & (nsIPlaintextEditor::eEditorReadonlyMask |
|
|
nsIPlaintextEditor::eEditorDisabledMask)) {
|
|
*aState = nsIContent::IME_STATUS_DISABLE;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(GetRoot());
|
|
NS_ENSURE_TRUE(content, NS_ERROR_FAILURE);
|
|
|
|
nsIFrame* frame = content->GetPrimaryFrame();
|
|
NS_ENSURE_TRUE(frame, NS_ERROR_FAILURE);
|
|
|
|
switch (frame->GetStyleUIReset()->mIMEMode) {
|
|
case NS_STYLE_IME_MODE_AUTO:
|
|
if (flags & (nsIPlaintextEditor::eEditorPasswordMask))
|
|
*aState = nsIContent::IME_STATUS_PASSWORD;
|
|
break;
|
|
case NS_STYLE_IME_MODE_DISABLED:
|
|
// we should use password state for |ime-mode: disabled;|.
|
|
*aState = nsIContent::IME_STATUS_PASSWORD;
|
|
break;
|
|
case NS_STYLE_IME_MODE_ACTIVE:
|
|
*aState |= nsIContent::IME_STATUS_OPEN;
|
|
break;
|
|
case NS_STYLE_IME_MODE_INACTIVE:
|
|
*aState |= nsIContent::IME_STATUS_CLOSE;
|
|
break;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetComposing(PRBool* aResult)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aResult);
|
|
*aResult = IsIMEComposing();
|
|
return NS_OK;
|
|
}
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark public nsEditor methods
|
|
#pragma mark -
|
|
#endif
|
|
/* Non-interface, public methods */
|
|
|
|
|
|
void
|
|
nsEditor::ContentAppended(nsIDocument *aDocument, nsIContent* aContainer,
|
|
PRInt32 aNewIndexInContainer)
|
|
{
|
|
ContentInserted(aDocument, aContainer, nsnull, aNewIndexInContainer);
|
|
}
|
|
|
|
void
|
|
nsEditor::ContentInserted(nsIDocument *aDocument, nsIContent* aContainer,
|
|
nsIContent* aChild, PRInt32 aIndexInContainer)
|
|
{
|
|
// XXX If we need aChild then nsEditor::ContentAppended should start passing
|
|
// in the child.
|
|
if (!mRootElement)
|
|
{
|
|
// Need to remove the event listeners first because BeginningOfDocument
|
|
// could set a new root (and event target) and we won't be able to remove
|
|
// them from the old event target then.
|
|
RemoveEventListeners();
|
|
BeginningOfDocument();
|
|
InstallEventListeners();
|
|
SyncRealTimeSpell();
|
|
}
|
|
}
|
|
|
|
void
|
|
nsEditor::ContentRemoved(nsIDocument *aDocument, nsIContent* aContainer,
|
|
nsIContent* aChild, PRInt32 aIndexInContainer)
|
|
{
|
|
nsCOMPtr<nsIDOMHTMLElement> elem = do_QueryInterface(aChild);
|
|
if (elem == mRootElement)
|
|
{
|
|
RemoveEventListeners();
|
|
mRootElement = nsnull;
|
|
mEventTarget = nsnull;
|
|
InstallEventListeners();
|
|
}
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::GetRootElement(nsIDOMElement **aRootElement)
|
|
{
|
|
if (!aRootElement)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
if (mRootElement)
|
|
{
|
|
// if we have cached the body element, use that
|
|
*aRootElement = mRootElement;
|
|
NS_ADDREF(*aRootElement);
|
|
return NS_OK;
|
|
}
|
|
|
|
*aRootElement = nsnull;
|
|
|
|
NS_PRECONDITION(mDocWeak, "bad state, null mDocWeak");
|
|
nsCOMPtr<nsIDOMHTMLDocument> htmlDoc = do_QueryReferent(mDocWeak);
|
|
if (!htmlDoc) {
|
|
return NS_ERROR_NOT_INITIALIZED;
|
|
}
|
|
|
|
// Use the HTML documents body element as the editor root if we didn't
|
|
// get a root element during initialization.
|
|
|
|
nsCOMPtr<nsIDOMHTMLElement> bodyElement;
|
|
nsresult rv = htmlDoc->GetBody(getter_AddRefs(bodyElement));
|
|
if (NS_SUCCEEDED(rv) && bodyElement)
|
|
{
|
|
mRootElement = bodyElement;
|
|
}
|
|
else
|
|
{
|
|
// If the document isn't HTML's or there is no HTML body element,
|
|
// we should use the document root element instead.
|
|
nsCOMPtr<nsIDOMDocument> doc = do_QueryReferent(mDocWeak);
|
|
NS_ENSURE_TRUE(doc, NS_ERROR_NOT_INITIALIZED);
|
|
|
|
rv = doc->GetDocumentElement(getter_AddRefs(mRootElement));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
NS_ENSURE_TRUE(mRootElement, NS_ERROR_NULL_POINTER);
|
|
}
|
|
|
|
*aRootElement = mRootElement;
|
|
NS_ADDREF(*aRootElement);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
/** All editor operations which alter the doc should be prefaced
|
|
* with a call to StartOperation, naming the action and direction */
|
|
NS_IMETHODIMP
|
|
nsEditor::StartOperation(PRInt32 opID, nsIEditor::EDirection aDirection)
|
|
{
|
|
mAction = opID;
|
|
mDirection = aDirection;
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
/** All editor operations which alter the doc should be followed
|
|
* with a call to EndOperation */
|
|
NS_IMETHODIMP
|
|
nsEditor::EndOperation()
|
|
{
|
|
mAction = nsnull;
|
|
mDirection = eNone;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CloneAttribute(const nsAString & aAttribute,
|
|
nsIDOMNode *aDestNode, nsIDOMNode *aSourceNode)
|
|
{
|
|
if (!aDestNode || !aSourceNode)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
nsCOMPtr<nsIDOMElement> destElement = do_QueryInterface(aDestNode);
|
|
nsCOMPtr<nsIDOMElement> sourceElement = do_QueryInterface(aSourceNode);
|
|
if (!destElement || !sourceElement)
|
|
return NS_ERROR_NO_INTERFACE;
|
|
|
|
nsAutoString attrValue;
|
|
PRBool isAttrSet;
|
|
nsresult rv = GetAttributeValue(sourceElement,
|
|
aAttribute,
|
|
attrValue,
|
|
&isAttrSet);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
if (isAttrSet)
|
|
rv = SetAttribute(destElement, aAttribute, attrValue);
|
|
else
|
|
rv = RemoveAttribute(destElement, aAttribute);
|
|
|
|
return rv;
|
|
}
|
|
|
|
// Objects must be DOM elements
|
|
NS_IMETHODIMP
|
|
nsEditor::CloneAttributes(nsIDOMNode *aDestNode, nsIDOMNode *aSourceNode)
|
|
{
|
|
if (!aDestNode || !aSourceNode)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
nsCOMPtr<nsIDOMElement> destElement = do_QueryInterface(aDestNode);
|
|
nsCOMPtr<nsIDOMElement> sourceElement = do_QueryInterface(aSourceNode);
|
|
if (!destElement || !sourceElement)
|
|
return NS_ERROR_NO_INTERFACE;
|
|
|
|
nsCOMPtr<nsIDOMNamedNodeMap> sourceAttributes;
|
|
sourceElement->GetAttributes(getter_AddRefs(sourceAttributes));
|
|
nsCOMPtr<nsIDOMNamedNodeMap> destAttributes;
|
|
destElement->GetAttributes(getter_AddRefs(destAttributes));
|
|
if (!sourceAttributes || !destAttributes)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsAutoEditBatch beginBatching(this);
|
|
|
|
// Use transaction system for undo only if destination
|
|
// is already in the document
|
|
nsIDOMElement *rootElement = GetRoot();
|
|
if (!rootElement)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
PRBool destInBody = PR_TRUE;
|
|
nsCOMPtr<nsIDOMNode> rootNode = do_QueryInterface(rootElement);
|
|
nsCOMPtr<nsIDOMNode> p = aDestNode;
|
|
while (p && p != rootNode)
|
|
{
|
|
nsCOMPtr<nsIDOMNode> tmp;
|
|
if (NS_FAILED(p->GetParentNode(getter_AddRefs(tmp))) || !tmp)
|
|
{
|
|
destInBody = PR_FALSE;
|
|
break;
|
|
}
|
|
p = tmp;
|
|
}
|
|
|
|
PRUint32 sourceCount;
|
|
sourceAttributes->GetLength(&sourceCount);
|
|
PRUint32 i, destCount;
|
|
destAttributes->GetLength(&destCount);
|
|
nsCOMPtr<nsIDOMNode> attrNode;
|
|
|
|
// Clear existing attributes
|
|
for (i = 0; i < destCount; i++)
|
|
{
|
|
// always remove item number 0 (first item in list)
|
|
if( NS_SUCCEEDED(destAttributes->Item(0, getter_AddRefs(attrNode))) && attrNode)
|
|
{
|
|
nsCOMPtr<nsIDOMAttr> destAttribute = do_QueryInterface(attrNode);
|
|
if (destAttribute)
|
|
{
|
|
nsAutoString str;
|
|
if (NS_SUCCEEDED(destAttribute->GetName(str)))
|
|
{
|
|
if (destInBody)
|
|
RemoveAttribute(destElement, str);
|
|
else
|
|
destElement->RemoveAttribute(str);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
nsresult result = NS_OK;
|
|
|
|
// Set just the attributes that the source element has
|
|
for (i = 0; i < sourceCount; i++)
|
|
{
|
|
if( NS_SUCCEEDED(sourceAttributes->Item(i, getter_AddRefs(attrNode))) && attrNode)
|
|
{
|
|
nsCOMPtr<nsIDOMAttr> sourceAttribute = do_QueryInterface(attrNode);
|
|
if (sourceAttribute)
|
|
{
|
|
nsAutoString sourceAttrName;
|
|
if (NS_SUCCEEDED(sourceAttribute->GetName(sourceAttrName)))
|
|
{
|
|
nsAutoString sourceAttrValue;
|
|
/*
|
|
Presence of an attribute in the named node map indicates that it was set on the
|
|
element even if it has no value.
|
|
*/
|
|
if (NS_SUCCEEDED(sourceAttribute->GetValue(sourceAttrValue)))
|
|
{
|
|
if (destInBody) {
|
|
result = SetAttributeOrEquivalent(destElement, sourceAttrName, sourceAttrValue, PR_FALSE);
|
|
}
|
|
else {
|
|
// the element is not inserted in the document yet, we don't want to put a
|
|
// transaction on the UndoStack
|
|
result = SetAttributeOrEquivalent(destElement, sourceAttrName, sourceAttrValue, PR_TRUE);
|
|
}
|
|
} else {
|
|
// Do we ever get here?
|
|
#if DEBUG_cmanske
|
|
printf("Attribute in sourceAttribute has empty value in nsEditor::CloneAttributes()\n");
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark Protected and static methods
|
|
#pragma mark -
|
|
#endif
|
|
|
|
NS_IMETHODIMP nsEditor::ScrollSelectionIntoView(PRBool aScrollToAnchor)
|
|
{
|
|
nsCOMPtr<nsISelectionController> selCon;
|
|
if (NS_SUCCEEDED(GetSelectionController(getter_AddRefs(selCon))) && selCon)
|
|
{
|
|
PRInt16 region = nsISelectionController::SELECTION_FOCUS_REGION;
|
|
|
|
if (aScrollToAnchor)
|
|
region = nsISelectionController::SELECTION_ANCHOR_REGION;
|
|
|
|
PRBool syncScroll = PR_TRUE;
|
|
PRUint32 flags = 0;
|
|
|
|
if (NS_SUCCEEDED(GetFlags(&flags)))
|
|
{
|
|
// If the editor is relying on asynchronous reflows, we have
|
|
// to use asynchronous requests to scroll, so that the scrolling happens
|
|
// after reflow requests are processed.
|
|
// XXXbz why not just always do async scroll?
|
|
syncScroll = !(flags & nsIPlaintextEditor::eEditorUseAsyncUpdatesMask);
|
|
}
|
|
|
|
// After ScrollSelectionIntoView(), the pending notifications might be
|
|
// flushed and PresShell/PresContext/Frames may be dead. See bug 418470.
|
|
selCon->ScrollSelectionIntoView(nsISelectionController::SELECTION_NORMAL,
|
|
region, syncScroll);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsEditor::InsertTextImpl(const nsAString& aStringToInsert,
|
|
nsCOMPtr<nsIDOMNode> *aInOutNode,
|
|
PRInt32 *aInOutOffset,
|
|
nsIDOMDocument *aDoc)
|
|
{
|
|
// NOTE: caller *must* have already used nsAutoTxnsConserveSelection stack-based
|
|
// class to turn off txn selection updating. Caller also turned on rules sniffing
|
|
// if desired.
|
|
|
|
if (!aInOutNode || !*aInOutNode || !aInOutOffset || !aDoc) return NS_ERROR_NULL_POINTER;
|
|
if (!mInIMEMode && aStringToInsert.IsEmpty()) return NS_OK;
|
|
nsCOMPtr<nsIDOMText> nodeAsText = do_QueryInterface(*aInOutNode);
|
|
PRInt32 offset = *aInOutOffset;
|
|
nsresult res;
|
|
if (mInIMEMode)
|
|
{
|
|
if (!nodeAsText)
|
|
{
|
|
// create a text node
|
|
res = aDoc->CreateTextNode(EmptyString(), getter_AddRefs(nodeAsText));
|
|
if (NS_FAILED(res)) return res;
|
|
if (!nodeAsText) return NS_ERROR_NULL_POINTER;
|
|
nsCOMPtr<nsIDOMNode> newNode = do_QueryInterface(nodeAsText);
|
|
// then we insert it into the dom tree
|
|
res = InsertNode(newNode, *aInOutNode, offset);
|
|
if (NS_FAILED(res)) return res;
|
|
offset = 0;
|
|
}
|
|
res = InsertTextIntoTextNodeImpl(aStringToInsert, nodeAsText, offset);
|
|
if (NS_FAILED(res)) return res;
|
|
}
|
|
else
|
|
{
|
|
if (nodeAsText)
|
|
{
|
|
// we are inserting text into an existing text node.
|
|
res = InsertTextIntoTextNodeImpl(aStringToInsert, nodeAsText, offset);
|
|
if (NS_FAILED(res)) return res;
|
|
*aInOutOffset += aStringToInsert.Length();
|
|
}
|
|
else
|
|
{
|
|
// we are inserting text into a non-text node
|
|
// first we have to create a textnode (this also populates it with the text)
|
|
res = aDoc->CreateTextNode(aStringToInsert, getter_AddRefs(nodeAsText));
|
|
if (NS_FAILED(res)) return res;
|
|
if (!nodeAsText) return NS_ERROR_NULL_POINTER;
|
|
nsCOMPtr<nsIDOMNode> newNode = do_QueryInterface(nodeAsText);
|
|
// then we insert it into the dom tree
|
|
res = InsertNode(newNode, *aInOutNode, offset);
|
|
if (NS_FAILED(res)) return res;
|
|
*aInOutNode = newNode;
|
|
*aInOutOffset = aStringToInsert.Length();
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
|
|
nsresult nsEditor::InsertTextIntoTextNodeImpl(const nsAString& aStringToInsert,
|
|
nsIDOMCharacterData *aTextNode,
|
|
PRInt32 aOffset,
|
|
PRBool aSuppressIME)
|
|
{
|
|
nsRefPtr<EditTxn> txn;
|
|
nsresult result = NS_OK;
|
|
PRBool isIMETransaction = PR_FALSE;
|
|
// aSuppressIME is used when editor must insert text, yet this text is not
|
|
// part of current ime operation. example: adjusting whitespace around an ime insertion.
|
|
if (mIMETextRangeList && mInIMEMode && !aSuppressIME)
|
|
{
|
|
if (!mIMETextNode)
|
|
{
|
|
mIMETextNode = aTextNode;
|
|
mIMETextOffset = aOffset;
|
|
}
|
|
PRUint16 len ;
|
|
len = mIMETextRangeList->GetLength();
|
|
if (len > 0)
|
|
{
|
|
nsCOMPtr<nsIPrivateTextRange> range;
|
|
for (PRUint16 i = 0; i < len; i++)
|
|
{
|
|
range = mIMETextRangeList->Item(i);
|
|
if (range)
|
|
{
|
|
PRUint16 type;
|
|
result = range->GetRangeType(&type);
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
if (type == nsIPrivateTextRange::TEXTRANGE_RAWINPUT)
|
|
{
|
|
PRUint16 start, end;
|
|
result = range->GetRangeStart(&start);
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
result = range->GetRangeEnd(&end);
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
if (!mPhonetic)
|
|
mPhonetic = new nsString();
|
|
if (mPhonetic)
|
|
{
|
|
nsAutoString tmp(aStringToInsert);
|
|
tmp.Mid(*mPhonetic, start, end-start);
|
|
}
|
|
}
|
|
}
|
|
} // if
|
|
}
|
|
} // if
|
|
} // for
|
|
} // if
|
|
|
|
nsRefPtr<IMETextTxn> imeTxn;
|
|
result = CreateTxnForIMEText(aStringToInsert, getter_AddRefs(imeTxn));
|
|
txn = imeTxn;
|
|
isIMETransaction = PR_TRUE;
|
|
}
|
|
else
|
|
{
|
|
nsRefPtr<InsertTextTxn> insertTxn;
|
|
result = CreateTxnForInsertText(aStringToInsert, aTextNode, aOffset,
|
|
getter_AddRefs(insertTxn));
|
|
txn = insertTxn;
|
|
}
|
|
if (NS_FAILED(result)) return result;
|
|
|
|
// let listeners know what's up
|
|
PRInt32 i;
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillInsertText(aTextNode, aOffset, aStringToInsert);
|
|
|
|
// XXX we may not need these view batches anymore. This is handled at a higher level now I believe
|
|
BeginUpdateViewBatch();
|
|
result = DoTransaction(txn);
|
|
EndUpdateViewBatch();
|
|
|
|
mRangeUpdater.SelAdjInsertText(aTextNode, aOffset, aStringToInsert);
|
|
|
|
// let listeners know what happened
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->DidInsertText(aTextNode, aOffset, aStringToInsert, result);
|
|
|
|
// 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 (isIMETransaction)
|
|
{
|
|
PRUint32 len;
|
|
mIMETextNode->GetLength(&len);
|
|
if (!len)
|
|
{
|
|
DeleteNode(mIMETextNode);
|
|
mIMETextNode = nsnull;
|
|
static_cast<IMETextTxn*>(txn.get())->MarkFixed(); // mark the ime txn "fixed"
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::SelectEntireDocument(nsISelection *aSelection)
|
|
{
|
|
if (!aSelection) { return NS_ERROR_NULL_POINTER; }
|
|
|
|
nsIDOMElement *rootElement = GetRoot();
|
|
if (!rootElement) { return NS_ERROR_NOT_INITIALIZED; }
|
|
|
|
return aSelection->SelectAllChildren(rootElement);
|
|
}
|
|
|
|
|
|
nsresult nsEditor::GetFirstEditableNode(nsIDOMNode *aRoot, nsCOMPtr<nsIDOMNode> *outFirstNode)
|
|
{
|
|
if (!aRoot || !outFirstNode) return NS_ERROR_NULL_POINTER;
|
|
nsresult rv = NS_OK;
|
|
*outFirstNode = nsnull;
|
|
|
|
nsCOMPtr<nsIDOMNode> node = GetLeftmostChild(aRoot);
|
|
if (node && !IsEditable(node))
|
|
{
|
|
nsCOMPtr<nsIDOMNode> next;
|
|
rv = GetNextNode(node, PR_TRUE, address_of(next));
|
|
node = next;
|
|
}
|
|
|
|
if (node != aRoot)
|
|
*outFirstNode = node;
|
|
|
|
return rv;
|
|
}
|
|
|
|
#ifdef XXX_DEAD_CODE
|
|
// jfrancis wants to keep this method around for reference
|
|
nsresult nsEditor::GetLastEditableNode(nsIDOMNode *aRoot, nsCOMPtr<nsIDOMNode> *outLastNode)
|
|
{
|
|
if (!aRoot || !outLastNode) return NS_ERROR_NULL_POINTER;
|
|
nsresult rv = NS_OK;
|
|
*outLastNode = nsnull;
|
|
|
|
nsCOMPtr<nsIDOMNode> node = GetRightmostChild(aRoot);
|
|
if (node && !IsEditable(node))
|
|
{
|
|
nsCOMPtr<nsIDOMNode> next;
|
|
rv = GetPriorNode(node, PR_TRUE, address_of(next));
|
|
node = next;
|
|
}
|
|
|
|
if (node != aRoot)
|
|
*outLastNode = node;
|
|
|
|
return rv;
|
|
}
|
|
#endif
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::NotifyDocumentListeners(TDocumentListenerNotification aNotificationType)
|
|
{
|
|
PRInt32 numListeners = mDocStateListeners.Count();
|
|
if (!numListeners)
|
|
return NS_OK; // maybe there just aren't any.
|
|
|
|
nsCOMArray<nsIDocumentStateListener> listeners(mDocStateListeners);
|
|
nsresult rv = NS_OK;
|
|
PRInt32 i;
|
|
|
|
switch (aNotificationType)
|
|
{
|
|
case eDocumentCreated:
|
|
for (i = 0; i < numListeners;i++)
|
|
{
|
|
rv = listeners[i]->NotifyDocumentCreated();
|
|
if (NS_FAILED(rv))
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case eDocumentToBeDestroyed:
|
|
for (i = 0; i < numListeners;i++)
|
|
{
|
|
rv = listeners[i]->NotifyDocumentWillBeDestroyed();
|
|
if (NS_FAILED(rv))
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case eDocumentStateChanged:
|
|
{
|
|
PRBool docIsDirty;
|
|
rv = GetDocumentModified(&docIsDirty);
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
if (docIsDirty == mDocDirtyState)
|
|
return NS_OK;
|
|
|
|
mDocDirtyState = (PRInt8)docIsDirty;
|
|
|
|
for (i = 0; i < numListeners;i++)
|
|
{
|
|
rv = listeners[i]->NotifyDocumentStateChanged(mDocDirtyState);
|
|
if (NS_FAILED(rv))
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
NS_NOTREACHED("Unknown notification");
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::CreateTxnForInsertText(const nsAString & aStringToInsert,
|
|
nsIDOMCharacterData *aTextNode,
|
|
PRInt32 aOffset,
|
|
InsertTextTxn ** aTxn)
|
|
{
|
|
if (!aTextNode || !aTxn) return NS_ERROR_NULL_POINTER;
|
|
nsresult result;
|
|
|
|
*aTxn = new InsertTextTxn();
|
|
if (!*aTxn) return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
result = (*aTxn)->Init(aTextNode, aOffset, aStringToInsert, this);
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::DeleteText(nsIDOMCharacterData *aElement,
|
|
PRUint32 aOffset,
|
|
PRUint32 aLength)
|
|
{
|
|
nsRefPtr<DeleteTextTxn> txn;
|
|
nsresult result = CreateTxnForDeleteText(aElement, aOffset, aLength,
|
|
getter_AddRefs(txn));
|
|
nsAutoRules beginRulesSniffing(this, kOpDeleteText, nsIEditor::ePrevious);
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
// let listeners know what's up
|
|
PRInt32 i;
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillDeleteText(aElement, aOffset, aLength);
|
|
|
|
result = DoTransaction(txn);
|
|
|
|
// let listeners know what happened
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->DidDeleteText(aElement, aOffset, aLength, result);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::CreateTxnForDeleteText(nsIDOMCharacterData *aElement,
|
|
PRUint32 aOffset,
|
|
PRUint32 aLength,
|
|
DeleteTextTxn **aTxn)
|
|
{
|
|
if (!aElement)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aTxn = new DeleteTextTxn();
|
|
if (!*aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
return (*aTxn)->Init(this, aElement, aOffset, aLength, &mRangeUpdater);
|
|
}
|
|
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::CreateTxnForSplitNode(nsIDOMNode *aNode,
|
|
PRUint32 aOffset,
|
|
SplitElementTxn **aTxn)
|
|
{
|
|
if (!aNode)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aTxn = new SplitElementTxn();
|
|
if (!*aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
|
|
return (*aTxn)->Init(this, aNode, aOffset);
|
|
}
|
|
|
|
NS_IMETHODIMP nsEditor::CreateTxnForJoinNode(nsIDOMNode *aLeftNode,
|
|
nsIDOMNode *aRightNode,
|
|
JoinElementTxn **aTxn)
|
|
{
|
|
if (!aLeftNode || !aRightNode)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aTxn = new JoinElementTxn();
|
|
if (!*aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
|
|
return (*aTxn)->Init(this, aLeftNode, aRightNode);
|
|
}
|
|
|
|
|
|
// END nsEditor core implementation
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark nsEditor public static helper methods
|
|
#pragma mark -
|
|
#endif
|
|
|
|
// BEGIN nsEditor public helper methods
|
|
|
|
nsresult
|
|
nsEditor::SplitNodeImpl(nsIDOMNode * aExistingRightNode,
|
|
PRInt32 aOffset,
|
|
nsIDOMNode* aNewLeftNode,
|
|
nsIDOMNode* aParent)
|
|
{
|
|
#ifdef NS_DEBUG_EDITOR
|
|
if (gNoisy) { printf("SplitNodeImpl: left=%p, right=%p, offset=%d\n", (void*)aNewLeftNode, (void*)aExistingRightNode, aOffset); }
|
|
#endif
|
|
|
|
NS_ASSERTION(((nsnull!=aExistingRightNode) &&
|
|
(nsnull!=aNewLeftNode) &&
|
|
(nsnull!=aParent)),
|
|
"null arg");
|
|
nsresult result;
|
|
if ((nsnull!=aExistingRightNode) &&
|
|
(nsnull!=aNewLeftNode) &&
|
|
(nsnull!=aParent))
|
|
{
|
|
// get selection
|
|
nsCOMPtr<nsISelection> selection;
|
|
result = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(result)) return result;
|
|
if (!selection) return NS_ERROR_NULL_POINTER;
|
|
|
|
// remember some selection points
|
|
nsCOMPtr<nsIDOMNode> selStartNode, selEndNode;
|
|
PRInt32 selStartOffset, selEndOffset;
|
|
result = GetStartNodeAndOffset(selection, address_of(selStartNode), &selStartOffset);
|
|
if (NS_FAILED(result)) selStartNode = nsnull; // if selection is cleared, remember that
|
|
result = GetEndNodeAndOffset(selection, address_of(selEndNode), &selEndOffset);
|
|
if (NS_FAILED(result)) selStartNode = nsnull; // if selection is cleared, remember that
|
|
|
|
nsCOMPtr<nsIDOMNode> resultNode;
|
|
result = aParent->InsertBefore(aNewLeftNode, aExistingRightNode, getter_AddRefs(resultNode));
|
|
//printf(" after insert\n"); content->List(); // DEBUG
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
// split the children between the 2 nodes
|
|
// at this point, aExistingRightNode has all the children
|
|
// move all the children whose index is < aOffset to aNewLeftNode
|
|
if (0<=aOffset) // don't bother unless we're going to move at least one child
|
|
{
|
|
// if it's a text node, just shuffle around some text
|
|
nsCOMPtr<nsIDOMCharacterData> rightNodeAsText( do_QueryInterface(aExistingRightNode) );
|
|
nsCOMPtr<nsIDOMCharacterData> leftNodeAsText( do_QueryInterface(aNewLeftNode) );
|
|
if (leftNodeAsText && rightNodeAsText)
|
|
{
|
|
// fix right node
|
|
nsAutoString leftText;
|
|
rightNodeAsText->SubstringData(0, aOffset, leftText);
|
|
rightNodeAsText->DeleteData(0, aOffset);
|
|
// fix left node
|
|
leftNodeAsText->SetData(leftText);
|
|
// moose
|
|
}
|
|
else
|
|
{ // otherwise it's an interior node, so shuffle around the children
|
|
// go through list backwards so deletes don't interfere with the iteration
|
|
nsCOMPtr<nsIDOMNodeList> childNodes;
|
|
result = aExistingRightNode->GetChildNodes(getter_AddRefs(childNodes));
|
|
if ((NS_SUCCEEDED(result)) && (childNodes))
|
|
{
|
|
PRInt32 i=aOffset-1;
|
|
for ( ; ((NS_SUCCEEDED(result)) && (0<=i)); i--)
|
|
{
|
|
nsCOMPtr<nsIDOMNode> childNode;
|
|
result = childNodes->Item(i, getter_AddRefs(childNode));
|
|
if ((NS_SUCCEEDED(result)) && (childNode))
|
|
{
|
|
result = aExistingRightNode->RemoveChild(childNode, getter_AddRefs(resultNode));
|
|
//printf(" after remove\n"); content->List(); // DEBUG
|
|
if (NS_SUCCEEDED(result))
|
|
{
|
|
nsCOMPtr<nsIDOMNode> firstChild;
|
|
aNewLeftNode->GetFirstChild(getter_AddRefs(firstChild));
|
|
result = aNewLeftNode->InsertBefore(childNode, firstChild, getter_AddRefs(resultNode));
|
|
//printf(" after append\n"); content->List(); // DEBUG
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// handle selection
|
|
if (GetShouldTxnSetSelection())
|
|
{
|
|
// editor wants us to set selection at split point
|
|
selection->Collapse(aNewLeftNode, aOffset);
|
|
}
|
|
else if (selStartNode)
|
|
{
|
|
// else adjust the selection if needed. if selStartNode is null, then there was no selection.
|
|
// HACK: this is overly simplified - multi-range selections need more work than this
|
|
if (selStartNode.get() == aExistingRightNode)
|
|
{
|
|
if (selStartOffset < aOffset)
|
|
{
|
|
selStartNode = aNewLeftNode;
|
|
}
|
|
else
|
|
{
|
|
selStartOffset -= aOffset;
|
|
}
|
|
}
|
|
if (selEndNode.get() == aExistingRightNode)
|
|
{
|
|
if (selEndOffset < aOffset)
|
|
{
|
|
selEndNode = aNewLeftNode;
|
|
}
|
|
else
|
|
{
|
|
selEndOffset -= aOffset;
|
|
}
|
|
}
|
|
selection->Collapse(selStartNode,selStartOffset);
|
|
selection->Extend(selEndNode,selEndOffset);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
result = NS_ERROR_INVALID_ARG;
|
|
|
|
return result;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::JoinNodesImpl(nsIDOMNode * aNodeToKeep,
|
|
nsIDOMNode * aNodeToJoin,
|
|
nsIDOMNode * aParent,
|
|
PRBool aNodeToKeepIsFirst)
|
|
{
|
|
NS_ASSERTION(aNodeToKeep && aNodeToJoin && aParent, "null arg");
|
|
nsresult result;
|
|
if (aNodeToKeep && aNodeToJoin && aParent)
|
|
{
|
|
// get selection
|
|
nsCOMPtr<nsISelection> selection;
|
|
GetSelection(getter_AddRefs(selection));
|
|
if (!selection) return NS_ERROR_NULL_POINTER;
|
|
|
|
// remember some selection points
|
|
nsCOMPtr<nsIDOMNode> selStartNode, selEndNode;
|
|
PRInt32 selStartOffset, selEndOffset, joinOffset, keepOffset;
|
|
result = GetStartNodeAndOffset(selection, address_of(selStartNode), &selStartOffset);
|
|
if (NS_FAILED(result)) selStartNode = nsnull;
|
|
result = GetEndNodeAndOffset(selection, address_of(selEndNode), &selEndOffset);
|
|
// Joe or Kin should comment here on why the following line is not a copy/paste error
|
|
if (NS_FAILED(result)) selStartNode = nsnull;
|
|
|
|
nsCOMPtr<nsIDOMNode> leftNode;
|
|
if (aNodeToKeepIsFirst)
|
|
leftNode = aNodeToKeep;
|
|
else
|
|
leftNode = aNodeToJoin;
|
|
|
|
PRUint32 firstNodeLength;
|
|
result = GetLengthOfDOMNode(leftNode, firstNodeLength);
|
|
if (NS_FAILED(result)) return result;
|
|
nsCOMPtr<nsIDOMNode> parent;
|
|
result = GetNodeLocation(aNodeToJoin, address_of(parent), &joinOffset);
|
|
if (NS_FAILED(result)) return result;
|
|
result = GetNodeLocation(aNodeToKeep, address_of(parent), &keepOffset);
|
|
if (NS_FAILED(result)) return result;
|
|
|
|
// if selection endpoint is between the nodes, remember it as being
|
|
// in the one that is going away instead. This simplifies later selection
|
|
// adjustment logic at end of this method.
|
|
if (selStartNode)
|
|
{
|
|
if (selStartNode == parent)
|
|
{
|
|
if (aNodeToKeepIsFirst)
|
|
{
|
|
if ((selStartOffset > keepOffset) && (selStartOffset <= joinOffset))
|
|
{
|
|
selStartNode = aNodeToJoin;
|
|
selStartOffset = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((selStartOffset > joinOffset) && (selStartOffset <= keepOffset))
|
|
{
|
|
selStartNode = aNodeToJoin;
|
|
selStartOffset = firstNodeLength;
|
|
}
|
|
}
|
|
}
|
|
if (selEndNode == parent)
|
|
{
|
|
if (aNodeToKeepIsFirst)
|
|
{
|
|
if ((selEndOffset > keepOffset) && (selEndOffset <= joinOffset))
|
|
{
|
|
selEndNode = aNodeToJoin;
|
|
selEndOffset = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((selEndOffset > joinOffset) && (selEndOffset <= keepOffset))
|
|
{
|
|
selEndNode = aNodeToJoin;
|
|
selEndOffset = firstNodeLength;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// ok, ready to do join now.
|
|
// if it's a text node, just shuffle around some text
|
|
nsCOMPtr<nsIDOMCharacterData> keepNodeAsText( do_QueryInterface(aNodeToKeep) );
|
|
nsCOMPtr<nsIDOMCharacterData> joinNodeAsText( do_QueryInterface(aNodeToJoin) );
|
|
if (keepNodeAsText && joinNodeAsText)
|
|
{
|
|
nsAutoString rightText;
|
|
nsAutoString leftText;
|
|
if (aNodeToKeepIsFirst)
|
|
{
|
|
keepNodeAsText->GetData(leftText);
|
|
joinNodeAsText->GetData(rightText);
|
|
}
|
|
else
|
|
{
|
|
keepNodeAsText->GetData(rightText);
|
|
joinNodeAsText->GetData(leftText);
|
|
}
|
|
leftText += rightText;
|
|
keepNodeAsText->SetData(leftText);
|
|
}
|
|
else
|
|
{ // otherwise it's an interior node, so shuffle around the children
|
|
nsCOMPtr<nsIDOMNodeList> childNodes;
|
|
result = aNodeToJoin->GetChildNodes(getter_AddRefs(childNodes));
|
|
if ((NS_SUCCEEDED(result)) && (childNodes))
|
|
{
|
|
PRInt32 i; // must be signed int!
|
|
PRUint32 childCount=0;
|
|
nsCOMPtr<nsIDOMNode> firstNode; //only used if aNodeToKeepIsFirst is false
|
|
childNodes->GetLength(&childCount);
|
|
if (!aNodeToKeepIsFirst)
|
|
{ // remember the first child in aNodeToKeep, we'll insert all the children of aNodeToJoin in front of it
|
|
result = aNodeToKeep->GetFirstChild(getter_AddRefs(firstNode));
|
|
// GetFirstChild returns nsnull firstNode if aNodeToKeep has no children, that's ok.
|
|
}
|
|
nsCOMPtr<nsIDOMNode> resultNode;
|
|
// have to go through the list backwards to keep deletes from interfering with iteration
|
|
nsCOMPtr<nsIDOMNode> previousChild;
|
|
for (i=childCount-1; ((NS_SUCCEEDED(result)) && (0<=i)); i--)
|
|
{
|
|
nsCOMPtr<nsIDOMNode> childNode;
|
|
result = childNodes->Item(i, getter_AddRefs(childNode));
|
|
if ((NS_SUCCEEDED(result)) && (childNode))
|
|
{
|
|
if (aNodeToKeepIsFirst)
|
|
{ // append children of aNodeToJoin
|
|
//was result = aNodeToKeep->AppendChild(childNode, getter_AddRefs(resultNode));
|
|
result = aNodeToKeep->InsertBefore(childNode, previousChild, getter_AddRefs(resultNode));
|
|
previousChild = do_QueryInterface(childNode);
|
|
}
|
|
else
|
|
{ // prepend children of aNodeToJoin
|
|
result = aNodeToKeep->InsertBefore(childNode, firstNode, getter_AddRefs(resultNode));
|
|
firstNode = do_QueryInterface(childNode);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (!childNodes) {
|
|
result = NS_ERROR_NULL_POINTER;
|
|
}
|
|
}
|
|
if (NS_SUCCEEDED(result))
|
|
{ // delete the extra node
|
|
nsCOMPtr<nsIDOMNode> resultNode;
|
|
result = aParent->RemoveChild(aNodeToJoin, getter_AddRefs(resultNode));
|
|
|
|
if (GetShouldTxnSetSelection())
|
|
{
|
|
// editor wants us to set selection at join point
|
|
selection->Collapse(aNodeToKeep, firstNodeLength);
|
|
}
|
|
else if (selStartNode)
|
|
{
|
|
// and adjust the selection if needed
|
|
// HACK: this is overly simplified - multi-range selections need more work than this
|
|
PRBool bNeedToAdjust = PR_FALSE;
|
|
|
|
// check to see if we joined nodes where selection starts
|
|
if (selStartNode.get() == aNodeToJoin)
|
|
{
|
|
bNeedToAdjust = PR_TRUE;
|
|
selStartNode = aNodeToKeep;
|
|
if (aNodeToKeepIsFirst)
|
|
{
|
|
selStartOffset += firstNodeLength;
|
|
}
|
|
}
|
|
else if ((selStartNode.get() == aNodeToKeep) && !aNodeToKeepIsFirst)
|
|
{
|
|
bNeedToAdjust = PR_TRUE;
|
|
selStartOffset += firstNodeLength;
|
|
}
|
|
|
|
// check to see if we joined nodes where selection ends
|
|
if (selEndNode.get() == aNodeToJoin)
|
|
{
|
|
bNeedToAdjust = PR_TRUE;
|
|
selEndNode = aNodeToKeep;
|
|
if (aNodeToKeepIsFirst)
|
|
{
|
|
selEndOffset += firstNodeLength;
|
|
}
|
|
}
|
|
else if ((selEndNode.get() == aNodeToKeep) && !aNodeToKeepIsFirst)
|
|
{
|
|
bNeedToAdjust = PR_TRUE;
|
|
selEndOffset += firstNodeLength;
|
|
}
|
|
|
|
// adjust selection if needed
|
|
if (bNeedToAdjust)
|
|
{
|
|
selection->Collapse(selStartNode,selStartOffset);
|
|
selection->Extend(selEndNode,selEndOffset);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
result = NS_ERROR_INVALID_ARG;
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
nsresult
|
|
nsEditor::GetChildOffset(nsIDOMNode *aChild, nsIDOMNode *aParent, PRInt32 &aOffset)
|
|
{
|
|
NS_ASSERTION((aChild && aParent), "bad args");
|
|
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aParent);
|
|
nsCOMPtr<nsIContent> cChild = do_QueryInterface(aChild);
|
|
if (!cChild || !content)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
aOffset = content->IndexOf(cChild);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::GetNodeLocation(nsIDOMNode *inChild, nsCOMPtr<nsIDOMNode> *outParent, PRInt32 *outOffset)
|
|
{
|
|
NS_ASSERTION((inChild && outParent && outOffset), "bad args");
|
|
nsresult result = NS_ERROR_NULL_POINTER;
|
|
if (inChild && outParent && outOffset)
|
|
{
|
|
result = inChild->GetParentNode(getter_AddRefs(*outParent));
|
|
if ((NS_SUCCEEDED(result)) && (*outParent))
|
|
{
|
|
result = GetChildOffset(inChild, *outParent, *outOffset);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
// returns the number of things inside aNode.
|
|
// If aNode is text, returns number of characters. If not, returns number of children nodes.
|
|
nsresult
|
|
nsEditor::GetLengthOfDOMNode(nsIDOMNode *aNode, PRUint32 &aCount)
|
|
{
|
|
aCount = 0;
|
|
if (!aNode) { return NS_ERROR_NULL_POINTER; }
|
|
nsresult result=NS_OK;
|
|
nsCOMPtr<nsIDOMCharacterData>nodeAsChar = do_QueryInterface(aNode);
|
|
if (nodeAsChar) {
|
|
nodeAsChar->GetLength(&aCount);
|
|
}
|
|
else
|
|
{
|
|
PRBool hasChildNodes;
|
|
aNode->HasChildNodes(&hasChildNodes);
|
|
if (hasChildNodes)
|
|
{
|
|
nsCOMPtr<nsIDOMNodeList>nodeList;
|
|
result = aNode->GetChildNodes(getter_AddRefs(nodeList));
|
|
if (NS_SUCCEEDED(result) && nodeList) {
|
|
nodeList->GetLength(&aCount);
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
nsresult
|
|
nsEditor::GetPriorNode(nsIDOMNode *aParentNode,
|
|
PRInt32 aOffset,
|
|
PRBool aEditableNode,
|
|
nsCOMPtr<nsIDOMNode> *aResultNode,
|
|
PRBool bNoBlockCrossing)
|
|
{
|
|
// just another version of GetPriorNode that takes a {parent, offset}
|
|
// instead of a node
|
|
if (!aParentNode || !aResultNode) { return NS_ERROR_NULL_POINTER; }
|
|
*aResultNode = nsnull;
|
|
|
|
// if we are at beginning of node, or it is a textnode, then just look before it
|
|
if (!aOffset || IsTextNode(aParentNode))
|
|
{
|
|
if (bNoBlockCrossing && IsBlockNode(aParentNode))
|
|
{
|
|
// if we aren't allowed to cross blocks, don't look before this block
|
|
return NS_OK;
|
|
}
|
|
return GetPriorNode(aParentNode, aEditableNode, aResultNode, bNoBlockCrossing);
|
|
}
|
|
|
|
// else look before the child at 'aOffset'
|
|
nsCOMPtr<nsIDOMNode> child = GetChildAt(aParentNode, aOffset);
|
|
if (child)
|
|
return GetPriorNode(child, aEditableNode, aResultNode, bNoBlockCrossing);
|
|
|
|
// unless there isn't one, in which case we are at the end of the node
|
|
// and want the deep-right child.
|
|
*aResultNode = GetRightmostChild(aParentNode, bNoBlockCrossing);
|
|
if (!*aResultNode || !aEditableNode || IsEditable(*aResultNode))
|
|
return NS_OK;
|
|
|
|
// restart the search from the non-editable node we just found
|
|
nsCOMPtr<nsIDOMNode> notEditableNode = do_QueryInterface(*aResultNode);
|
|
return GetPriorNode(notEditableNode, aEditableNode, aResultNode, bNoBlockCrossing);
|
|
}
|
|
|
|
|
|
nsresult
|
|
nsEditor::GetNextNode(nsIDOMNode *aParentNode,
|
|
PRInt32 aOffset,
|
|
PRBool aEditableNode,
|
|
nsCOMPtr<nsIDOMNode> *aResultNode,
|
|
PRBool bNoBlockCrossing)
|
|
{
|
|
// just another version of GetNextNode that takes a {parent, offset}
|
|
// instead of a node
|
|
if (!aParentNode || !aResultNode) { return NS_ERROR_NULL_POINTER; }
|
|
|
|
*aResultNode = nsnull;
|
|
|
|
// if aParentNode is a text node, use it's location instead
|
|
if (IsTextNode(aParentNode))
|
|
{
|
|
nsCOMPtr<nsIDOMNode> parent;
|
|
nsEditor::GetNodeLocation(aParentNode, address_of(parent), &aOffset);
|
|
aParentNode = parent;
|
|
aOffset++; // _after_ the text node
|
|
}
|
|
// look at the child at 'aOffset'
|
|
nsCOMPtr<nsIDOMNode> child = GetChildAt(aParentNode, aOffset);
|
|
if (child)
|
|
{
|
|
if (bNoBlockCrossing && IsBlockNode(child))
|
|
{
|
|
*aResultNode = child; // return this block
|
|
return NS_OK;
|
|
}
|
|
*aResultNode = GetLeftmostChild(child, bNoBlockCrossing);
|
|
if (!*aResultNode)
|
|
{
|
|
*aResultNode = child;
|
|
return NS_OK;
|
|
}
|
|
if (!IsDescendantOfBody(*aResultNode))
|
|
{
|
|
*aResultNode = nsnull;
|
|
return NS_OK;
|
|
}
|
|
|
|
if (!aEditableNode || IsEditable(*aResultNode))
|
|
return NS_OK;
|
|
|
|
// restart the search from the non-editable node we just found
|
|
nsCOMPtr<nsIDOMNode> notEditableNode = do_QueryInterface(*aResultNode);
|
|
return GetNextNode(notEditableNode, aEditableNode, aResultNode, bNoBlockCrossing);
|
|
}
|
|
|
|
// unless there isn't one, in which case we are at the end of the node
|
|
// and want the next one.
|
|
if (bNoBlockCrossing && IsBlockNode(aParentNode))
|
|
{
|
|
// don't cross out of parent block
|
|
return NS_OK;
|
|
}
|
|
return GetNextNode(aParentNode, aEditableNode, aResultNode, bNoBlockCrossing);
|
|
}
|
|
|
|
|
|
nsresult
|
|
nsEditor::GetPriorNode(nsIDOMNode *aCurrentNode,
|
|
PRBool aEditableNode,
|
|
nsCOMPtr<nsIDOMNode> *aResultNode,
|
|
PRBool bNoBlockCrossing)
|
|
{
|
|
nsresult result;
|
|
if (!aCurrentNode || !aResultNode) { return NS_ERROR_NULL_POINTER; }
|
|
|
|
*aResultNode = nsnull; // init out-param
|
|
|
|
if (IsRootNode(aCurrentNode))
|
|
{
|
|
// Don't allow traversal above the root node! This helps
|
|
// prevent us from accidentally editing browser content
|
|
// when the editor is in a text widget.
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIDOMNode> candidate;
|
|
result = GetPriorNodeImpl(aCurrentNode, aEditableNode, address_of(candidate), bNoBlockCrossing);
|
|
if (NS_FAILED(result)) return result;
|
|
|
|
if (!candidate)
|
|
{
|
|
// we could not find a prior node. return null.
|
|
return NS_OK;
|
|
}
|
|
else if (!aEditableNode) *aResultNode = candidate;
|
|
else if (IsEditable(candidate)) *aResultNode = candidate;
|
|
else
|
|
{ // restart the search from the non-editable node we just found
|
|
nsCOMPtr<nsIDOMNode> notEditableNode = do_QueryInterface(candidate);
|
|
return GetPriorNode(notEditableNode, aEditableNode, aResultNode, bNoBlockCrossing);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::GetPriorNodeImpl(nsIDOMNode *aCurrentNode,
|
|
PRBool aEditableNode,
|
|
nsCOMPtr<nsIDOMNode> *aResultNode,
|
|
PRBool bNoBlockCrossing)
|
|
{
|
|
// called only by GetPriorNode so we don't need to check params.
|
|
|
|
// if aCurrentNode has a left sibling, return that sibling's rightmost child (or itself if it has no children)
|
|
nsCOMPtr<nsIDOMNode> prevSibling;
|
|
nsresult result = aCurrentNode->GetPreviousSibling(getter_AddRefs(prevSibling));
|
|
if ((NS_SUCCEEDED(result)) && prevSibling)
|
|
{
|
|
if (bNoBlockCrossing && IsBlockNode(prevSibling))
|
|
{
|
|
// don't look inside prevsib, since it is a block
|
|
*aResultNode = prevSibling;
|
|
return NS_OK;
|
|
}
|
|
*aResultNode = GetRightmostChild(prevSibling, bNoBlockCrossing);
|
|
if (!*aResultNode)
|
|
{
|
|
*aResultNode = prevSibling;
|
|
return NS_OK;
|
|
}
|
|
if (!IsDescendantOfBody(*aResultNode))
|
|
{
|
|
*aResultNode = nsnull;
|
|
return NS_OK;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// otherwise, walk up the parent tree until there is a child that comes before
|
|
// the ancestor of aCurrentNode. Then return that node's rightmost child
|
|
nsCOMPtr<nsIDOMNode> parent = do_QueryInterface(aCurrentNode);
|
|
nsCOMPtr<nsIDOMNode> node, notEditableNode;
|
|
do {
|
|
node = parent;
|
|
result = node->GetParentNode(getter_AddRefs(parent));
|
|
if ((NS_SUCCEEDED(result)) && parent)
|
|
{
|
|
if (!IsDescendantOfBody(parent))
|
|
{
|
|
*aResultNode = nsnull;
|
|
return NS_OK;
|
|
}
|
|
if ((bNoBlockCrossing && IsBlockNode(parent)) || IsRootNode(parent))
|
|
{
|
|
// we are at front of block or root, do not step out
|
|
*aResultNode = nsnull;
|
|
return NS_OK;
|
|
}
|
|
result = parent->GetPreviousSibling(getter_AddRefs(node));
|
|
if ((NS_SUCCEEDED(result)) && node)
|
|
{
|
|
if (bNoBlockCrossing && IsBlockNode(node))
|
|
{
|
|
// prev sibling is a block, do not step into it
|
|
*aResultNode = node;
|
|
return NS_OK;
|
|
}
|
|
*aResultNode = GetRightmostChild(node, bNoBlockCrossing);
|
|
if (!*aResultNode) *aResultNode = node;
|
|
return NS_OK;
|
|
}
|
|
}
|
|
} while ((NS_SUCCEEDED(result)) && parent && !*aResultNode);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::GetNextNode(nsIDOMNode *aCurrentNode,
|
|
PRBool aEditableNode,
|
|
nsCOMPtr<nsIDOMNode> *aResultNode,
|
|
PRBool bNoBlockCrossing)
|
|
{
|
|
if (!aCurrentNode || !aResultNode) { return NS_ERROR_NULL_POINTER; }
|
|
|
|
*aResultNode = nsnull; // init out-param
|
|
|
|
if (IsRootNode(aCurrentNode))
|
|
{
|
|
// Don't allow traversal above the root node! This helps
|
|
// prevent us from accidentally editing browser content
|
|
// when the editor is in a text widget.
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIDOMNode> candidate;
|
|
nsresult result = GetNextNodeImpl(aCurrentNode, aEditableNode,
|
|
address_of(candidate), bNoBlockCrossing);
|
|
if (NS_FAILED(result)) return result;
|
|
|
|
if (!candidate)
|
|
{
|
|
// we could not find a next node. return null.
|
|
*aResultNode = nsnull;
|
|
return NS_OK;
|
|
}
|
|
else if (!aEditableNode) *aResultNode = candidate;
|
|
else if (IsEditable(candidate)) *aResultNode = candidate;
|
|
else
|
|
{ // restart the search from the non-editable node we just found
|
|
nsCOMPtr<nsIDOMNode> notEditableNode = do_QueryInterface(candidate);
|
|
return GetNextNode(notEditableNode, aEditableNode, aResultNode, bNoBlockCrossing);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
nsresult
|
|
nsEditor::GetNextNodeImpl(nsIDOMNode *aCurrentNode,
|
|
PRBool aEditableNode,
|
|
nsCOMPtr<nsIDOMNode> *aResultNode,
|
|
PRBool bNoBlockCrossing)
|
|
{
|
|
// called only by GetNextNode so we don't need to check params.
|
|
|
|
// if aCurrentNode has a right sibling, return that sibling's leftmost child (or itself if it has no children)
|
|
nsCOMPtr<nsIDOMNode> nextSibling;
|
|
nsresult result = aCurrentNode->GetNextSibling(getter_AddRefs(nextSibling));
|
|
if ((NS_SUCCEEDED(result)) && nextSibling)
|
|
{
|
|
if (bNoBlockCrossing && IsBlockNode(nextSibling))
|
|
{
|
|
// next sibling is a block, do not step into it
|
|
*aResultNode = nextSibling;
|
|
return NS_OK;
|
|
}
|
|
*aResultNode = GetLeftmostChild(nextSibling, bNoBlockCrossing);
|
|
if (!*aResultNode)
|
|
{
|
|
*aResultNode = nextSibling;
|
|
return NS_OK;
|
|
}
|
|
if (!IsDescendantOfBody(*aResultNode))
|
|
{
|
|
*aResultNode = nsnull;
|
|
return NS_OK;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// otherwise, walk up the parent tree until there is a child that comes after
|
|
// the ancestor of aCurrentNode. Then return that node's leftmost child
|
|
nsCOMPtr<nsIDOMNode> parent(do_QueryInterface(aCurrentNode));
|
|
nsCOMPtr<nsIDOMNode> node, notEditableNode;
|
|
do {
|
|
node = parent;
|
|
result = node->GetParentNode(getter_AddRefs(parent));
|
|
if ((NS_SUCCEEDED(result)) && parent)
|
|
{
|
|
if (!IsDescendantOfBody(parent))
|
|
{
|
|
*aResultNode = nsnull;
|
|
return NS_OK;
|
|
}
|
|
if ((bNoBlockCrossing && IsBlockNode(parent)) || IsRootNode(parent))
|
|
{
|
|
// we are at end of block or root, do not step out
|
|
*aResultNode = nsnull;
|
|
return NS_OK;
|
|
}
|
|
result = parent->GetNextSibling(getter_AddRefs(node));
|
|
if ((NS_SUCCEEDED(result)) && node)
|
|
{
|
|
if (bNoBlockCrossing && IsBlockNode(node))
|
|
{
|
|
// next sibling is a block, do not step into it
|
|
*aResultNode = node;
|
|
return NS_OK;
|
|
}
|
|
*aResultNode = GetLeftmostChild(node, bNoBlockCrossing);
|
|
if (!*aResultNode) *aResultNode = node;
|
|
return NS_OK;
|
|
}
|
|
}
|
|
} while ((NS_SUCCEEDED(result)) && parent);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
nsCOMPtr<nsIDOMNode>
|
|
nsEditor::GetRightmostChild(nsIDOMNode *aCurrentNode,
|
|
PRBool bNoBlockCrossing)
|
|
{
|
|
if (!aCurrentNode) return nsnull;
|
|
nsCOMPtr<nsIDOMNode> resultNode, temp=aCurrentNode;
|
|
PRBool hasChildren;
|
|
aCurrentNode->HasChildNodes(&hasChildren);
|
|
while (hasChildren)
|
|
{
|
|
temp->GetLastChild(getter_AddRefs(resultNode));
|
|
if (resultNode)
|
|
{
|
|
if (bNoBlockCrossing && IsBlockNode(resultNode))
|
|
return resultNode;
|
|
resultNode->HasChildNodes(&hasChildren);
|
|
temp = resultNode;
|
|
}
|
|
else
|
|
hasChildren = PR_FALSE;
|
|
}
|
|
|
|
return resultNode;
|
|
}
|
|
|
|
nsCOMPtr<nsIDOMNode>
|
|
nsEditor::GetLeftmostChild(nsIDOMNode *aCurrentNode,
|
|
PRBool bNoBlockCrossing)
|
|
{
|
|
if (!aCurrentNode) return nsnull;
|
|
nsCOMPtr<nsIDOMNode> resultNode, temp=aCurrentNode;
|
|
PRBool hasChildren;
|
|
aCurrentNode->HasChildNodes(&hasChildren);
|
|
while (hasChildren)
|
|
{
|
|
temp->GetFirstChild(getter_AddRefs(resultNode));
|
|
if (resultNode)
|
|
{
|
|
if (bNoBlockCrossing && IsBlockNode(resultNode))
|
|
return resultNode;
|
|
resultNode->HasChildNodes(&hasChildren);
|
|
temp = resultNode;
|
|
}
|
|
else
|
|
hasChildren = PR_FALSE;
|
|
}
|
|
|
|
return resultNode;
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::IsBlockNode(nsIDOMNode *aNode)
|
|
{
|
|
// stub to be overridden in nsHTMLEditor.
|
|
// screwing around with the class heirarchy here in order
|
|
// to not duplicate the code in GetNextNode/GetPrevNode
|
|
// across both nsEditor/nsHTMLEditor.
|
|
return PR_FALSE;
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::CanContainTag(nsIDOMNode* aParent, const nsAString &aChildTag)
|
|
{
|
|
nsCOMPtr<nsIDOMElement> parentElement = do_QueryInterface(aParent);
|
|
if (!parentElement) return PR_FALSE;
|
|
|
|
nsAutoString parentStringTag;
|
|
parentElement->GetTagName(parentStringTag);
|
|
return TagCanContainTag(parentStringTag, aChildTag);
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::TagCanContain(const nsAString &aParentTag, nsIDOMNode* aChild)
|
|
{
|
|
nsAutoString childStringTag;
|
|
|
|
if (IsTextNode(aChild))
|
|
{
|
|
childStringTag.AssignLiteral("#text");
|
|
}
|
|
else
|
|
{
|
|
nsCOMPtr<nsIDOMElement> childElement = do_QueryInterface(aChild);
|
|
if (!childElement) return PR_FALSE;
|
|
childElement->GetTagName(childStringTag);
|
|
}
|
|
return TagCanContainTag(aParentTag, childStringTag);
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::TagCanContainTag(const nsAString &aParentTag, const nsAString &aChildTag)
|
|
{
|
|
return PR_TRUE;
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::IsRootNode(nsIDOMNode *inNode)
|
|
{
|
|
if (!inNode)
|
|
return PR_FALSE;
|
|
|
|
nsIDOMElement *rootElement = GetRoot();
|
|
|
|
nsCOMPtr<nsIDOMNode> rootNode = do_QueryInterface(rootElement);
|
|
|
|
return inNode == rootNode;
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::IsDescendantOfBody(nsIDOMNode *inNode)
|
|
{
|
|
if (!inNode) return PR_FALSE;
|
|
nsIDOMElement *rootElement = GetRoot();
|
|
if (!rootElement) return PR_FALSE;
|
|
nsCOMPtr<nsIDOMNode> root = do_QueryInterface(rootElement);
|
|
|
|
if (inNode == root.get()) return PR_TRUE;
|
|
|
|
nsCOMPtr<nsIDOMNode> parent, node = do_QueryInterface(inNode);
|
|
|
|
do
|
|
{
|
|
node->GetParentNode(getter_AddRefs(parent));
|
|
if (parent == root) return PR_TRUE;
|
|
node = parent;
|
|
} while (parent);
|
|
|
|
return PR_FALSE;
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::IsContainer(nsIDOMNode *aNode)
|
|
{
|
|
return aNode ? PR_TRUE : PR_FALSE;
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::IsTextInDirtyFrameVisible(nsIDOMNode *aNode)
|
|
{
|
|
// virtual method
|
|
//
|
|
// If this is a simple non-html editor,
|
|
// the best we can do is to assume it's visible.
|
|
|
|
return PR_TRUE;
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::IsEditable(nsIDOMNode *aNode)
|
|
{
|
|
if (!aNode) return PR_FALSE;
|
|
|
|
if (IsMozEditorBogusNode(aNode) || !IsModifiableNode(aNode)) return PR_FALSE;
|
|
|
|
// see if it has a frame. If so, we'll edit it.
|
|
// special case for textnodes: frame must have width.
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aNode);
|
|
if (content)
|
|
{
|
|
nsIFrame *resultFrame = content->GetPrimaryFrame();
|
|
if (!resultFrame) // if it has no frame, it is not editable
|
|
return PR_FALSE;
|
|
NS_ASSERTION(content->IsNodeOfType(nsINode::eTEXT) ||
|
|
content->IsNodeOfType(nsINode::eELEMENT),
|
|
"frame for non element-or-text?");
|
|
if (!content->IsNodeOfType(nsINode::eTEXT))
|
|
return PR_TRUE; // not a text node; has a frame
|
|
|
|
// test the textframe and all its non-fluid continuations
|
|
while (resultFrame) {
|
|
if (resultFrame->GetStateBits() & NS_FRAME_IS_DIRTY) // we can only trust width data for undirty frames
|
|
{
|
|
// In the past a comment said:
|
|
// "assume all text nodes with dirty frames are editable"
|
|
// Nowadays we use a virtual function, that assumes TRUE
|
|
// in the simple editor world,
|
|
// and uses enhanced logic to find out in the HTML world.
|
|
return IsTextInDirtyFrameVisible(aNode);
|
|
}
|
|
if (resultFrame->GetSize().width > 0)
|
|
return PR_TRUE; // text node has width
|
|
resultFrame = resultFrame->GetNextContinuation();
|
|
}
|
|
}
|
|
return PR_FALSE; // didn't pass any editability test
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::IsMozEditorBogusNode(nsIDOMNode *aNode)
|
|
{
|
|
if (!aNode)
|
|
return PR_FALSE;
|
|
|
|
nsCOMPtr<nsIDOMElement>element = do_QueryInterface(aNode);
|
|
if (element)
|
|
{
|
|
nsAutoString val;
|
|
(void)element->GetAttribute(kMOZEditorBogusNodeAttr, val);
|
|
if (val.Equals(kMOZEditorBogusNodeValue)) {
|
|
return PR_TRUE;
|
|
}
|
|
}
|
|
|
|
return PR_FALSE;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::CountEditableChildren(nsIDOMNode *aNode, PRUint32 &outCount)
|
|
{
|
|
outCount = 0;
|
|
if (!aNode) { return NS_ERROR_NULL_POINTER; }
|
|
nsresult res=NS_OK;
|
|
PRBool hasChildNodes;
|
|
aNode->HasChildNodes(&hasChildNodes);
|
|
if (hasChildNodes)
|
|
{
|
|
nsCOMPtr<nsIDOMNodeList>nodeList;
|
|
res = aNode->GetChildNodes(getter_AddRefs(nodeList));
|
|
if (NS_SUCCEEDED(res) && nodeList)
|
|
{
|
|
PRUint32 i;
|
|
PRUint32 len;
|
|
nodeList->GetLength(&len);
|
|
for (i=0 ; i<len; i++)
|
|
{
|
|
nsCOMPtr<nsIDOMNode> child;
|
|
res = nodeList->Item((PRInt32)i, getter_AddRefs(child));
|
|
if ((NS_SUCCEEDED(res)) && (child))
|
|
{
|
|
if (IsEditable(child))
|
|
{
|
|
outCount++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (!nodeList)
|
|
res = NS_ERROR_NULL_POINTER;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
//END nsEditor static utility methods
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::IncrementModificationCount(PRInt32 inNumMods)
|
|
{
|
|
PRUint32 oldModCount = mModCount;
|
|
|
|
mModCount += inNumMods;
|
|
|
|
if ((oldModCount == 0 && mModCount != 0)
|
|
|| (oldModCount != 0 && mModCount == 0))
|
|
NotifyDocumentListeners(eDocumentStateChanged);
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::GetModificationCount(PRInt32 *outModCount)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(outModCount);
|
|
*outModCount = mModCount;
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::ResetModificationCount()
|
|
{
|
|
PRBool doNotify = (mModCount != 0);
|
|
|
|
mModCount = 0;
|
|
|
|
if (doNotify)
|
|
NotifyDocumentListeners(eDocumentStateChanged);
|
|
return NS_OK;
|
|
}
|
|
|
|
//END nsEditor Private methods
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// GetTag: digs out the atom for the tag of this node
|
|
//
|
|
nsIAtom *
|
|
nsEditor::GetTag(nsIDOMNode *aNode)
|
|
{
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aNode);
|
|
|
|
if (!content)
|
|
{
|
|
NS_ASSERTION(aNode, "null node passed to nsEditor::Tag()");
|
|
|
|
return nsnull;
|
|
}
|
|
|
|
return content->Tag();
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// GetTagString: digs out string for the tag of this node
|
|
//
|
|
nsresult
|
|
nsEditor::GetTagString(nsIDOMNode *aNode, nsAString& outString)
|
|
{
|
|
if (!aNode)
|
|
{
|
|
NS_NOTREACHED("null node passed to nsEditor::GetTag()");
|
|
return NS_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
nsIAtom *atom = GetTag(aNode);
|
|
if (!atom)
|
|
{
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
atom->ToString(outString);
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// NodesSameType: do these nodes have the same tag?
|
|
//
|
|
PRBool
|
|
nsEditor::NodesSameType(nsIDOMNode *aNode1, nsIDOMNode *aNode2)
|
|
{
|
|
if (!aNode1 || !aNode2)
|
|
{
|
|
NS_NOTREACHED("null node passed to nsEditor::NodesSameType()");
|
|
return PR_FALSE;
|
|
}
|
|
|
|
return GetTag(aNode1) == GetTag(aNode2);
|
|
}
|
|
|
|
|
|
// IsTextOrElementNode: true if node of dom type element or text
|
|
//
|
|
PRBool
|
|
nsEditor::IsTextOrElementNode(nsIDOMNode *aNode)
|
|
{
|
|
if (!aNode)
|
|
{
|
|
NS_NOTREACHED("null node passed to IsTextOrElementNode()");
|
|
return PR_FALSE;
|
|
}
|
|
|
|
PRUint16 nodeType;
|
|
aNode->GetNodeType(&nodeType);
|
|
return ((nodeType == nsIDOMNode::ELEMENT_NODE) || (nodeType == nsIDOMNode::TEXT_NODE));
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// IsTextNode: true if node of dom type text
|
|
//
|
|
PRBool
|
|
nsEditor::IsTextNode(nsIDOMNode *aNode)
|
|
{
|
|
if (!aNode)
|
|
{
|
|
NS_NOTREACHED("null node passed to IsTextNode()");
|
|
return PR_FALSE;
|
|
}
|
|
|
|
PRUint16 nodeType;
|
|
aNode->GetNodeType(&nodeType);
|
|
return (nodeType == nsIDOMNode::TEXT_NODE);
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// GetIndexOf: returns the position index of the node in the parent
|
|
//
|
|
PRInt32
|
|
nsEditor::GetIndexOf(nsIDOMNode *parent, nsIDOMNode *child)
|
|
{
|
|
nsCOMPtr<nsINode> parentNode = do_QueryInterface(parent);
|
|
NS_PRECONDITION(parentNode, "null parentNode in nsEditor::GetIndexOf");
|
|
NS_PRECONDITION(parentNode->IsNodeOfType(nsINode::eCONTENT) ||
|
|
parentNode->IsNodeOfType(nsINode::eDOCUMENT),
|
|
"The parent node must be an element node or a document node");
|
|
|
|
nsCOMPtr<nsIContent> cChild = do_QueryInterface(child);
|
|
NS_PRECONDITION(cChild, "null content in nsEditor::GetIndexOf");
|
|
|
|
return parentNode->IndexOf(cChild);
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// GetChildAt: returns the node at this position index in the parent
|
|
//
|
|
nsCOMPtr<nsIDOMNode>
|
|
nsEditor::GetChildAt(nsIDOMNode *aParent, PRInt32 aOffset)
|
|
{
|
|
nsCOMPtr<nsIDOMNode> resultNode;
|
|
|
|
nsCOMPtr<nsIContent> parent = do_QueryInterface(aParent);
|
|
|
|
if (!parent)
|
|
return resultNode;
|
|
|
|
resultNode = do_QueryInterface(parent->GetChildAt(aOffset));
|
|
|
|
return resultNode;
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// GetStartNodeAndOffset: returns whatever the start parent & offset is of
|
|
// the first range in the selection.
|
|
nsresult
|
|
nsEditor::GetStartNodeAndOffset(nsISelection *aSelection,
|
|
nsCOMPtr<nsIDOMNode> *outStartNode,
|
|
PRInt32 *outStartOffset)
|
|
{
|
|
if (!outStartNode || !outStartOffset || !aSelection)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
// brade: set outStartNode to null or ?
|
|
|
|
nsCOMPtr<nsISelectionPrivate>selPrivate(do_QueryInterface(aSelection));
|
|
nsCOMPtr<nsIEnumerator> enumerator;
|
|
nsresult result = selPrivate->GetEnumerator(getter_AddRefs(enumerator));
|
|
if (NS_FAILED(result) || !enumerator)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
enumerator->First();
|
|
nsCOMPtr<nsISupports> currentItem;
|
|
if (NS_FAILED(enumerator->CurrentItem(getter_AddRefs(currentItem))))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsCOMPtr<nsIDOMRange> range( do_QueryInterface(currentItem) );
|
|
if (!range)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
if (NS_FAILED(range->GetStartContainer(getter_AddRefs(*outStartNode))))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
if (NS_FAILED(range->GetStartOffset(outStartOffset)))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// GetEndNodeAndOffset: returns whatever the end parent & offset is of
|
|
// the first range in the selection.
|
|
nsresult
|
|
nsEditor::GetEndNodeAndOffset(nsISelection *aSelection,
|
|
nsCOMPtr<nsIDOMNode> *outEndNode,
|
|
PRInt32 *outEndOffset)
|
|
{
|
|
if (!outEndNode || !outEndOffset)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
nsCOMPtr<nsISelectionPrivate>selPrivate(do_QueryInterface(aSelection));
|
|
nsCOMPtr<nsIEnumerator> enumerator;
|
|
nsresult result = selPrivate->GetEnumerator(getter_AddRefs(enumerator));
|
|
if (NS_FAILED(result) || !enumerator)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
enumerator->First();
|
|
nsCOMPtr<nsISupports> currentItem;
|
|
if (NS_FAILED(enumerator->CurrentItem(getter_AddRefs(currentItem))))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsCOMPtr<nsIDOMRange> range( do_QueryInterface(currentItem) );
|
|
if (!range)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
if (NS_FAILED(range->GetEndContainer(getter_AddRefs(*outEndNode))))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
if (NS_FAILED(range->GetEndOffset(outEndOffset)))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// IsPreformatted: checks the style info for the node for the preformatted
|
|
// text style.
|
|
nsresult
|
|
nsEditor::IsPreformatted(nsIDOMNode *aNode, PRBool *aResult)
|
|
{
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aNode);
|
|
|
|
if (!aResult || !content) return NS_ERROR_NULL_POINTER;
|
|
|
|
nsCOMPtr<nsIPresShell> ps = do_QueryReferent(mPresShellWeak);
|
|
if (!ps) return NS_ERROR_NOT_INITIALIZED;
|
|
|
|
nsIFrame *frame = content->GetPrimaryFrame();
|
|
|
|
NS_ASSERTION(frame, "no frame, see bug #188946");
|
|
if (!frame)
|
|
{
|
|
// Consider nodes without a style context to be NOT preformatted:
|
|
// For instance, this is true of JS tags inside the body (which show
|
|
// up as #text nodes but have no style context).
|
|
*aResult = PR_FALSE;
|
|
return NS_OK;
|
|
}
|
|
|
|
const nsStyleText* styleText = frame->GetStyleText();
|
|
|
|
*aResult = styleText->WhiteSpaceIsSignificant();
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// SplitNodeDeep: this splits a node "deeply", splitting children as
|
|
// appropriate. The place to split is represented by
|
|
// a dom point at {splitPointParent, splitPointOffset}.
|
|
// That dom point must be inside aNode, which is the node to
|
|
// split. outOffset is set to the offset in the parent of aNode where
|
|
// the split terminates - where you would want to insert
|
|
// a new element, for instance, if that's why you were splitting
|
|
// the node.
|
|
//
|
|
nsresult
|
|
nsEditor::SplitNodeDeep(nsIDOMNode *aNode,
|
|
nsIDOMNode *aSplitPointParent,
|
|
PRInt32 aSplitPointOffset,
|
|
PRInt32 *outOffset,
|
|
PRBool aNoEmptyContainers,
|
|
nsCOMPtr<nsIDOMNode> *outLeftNode,
|
|
nsCOMPtr<nsIDOMNode> *outRightNode)
|
|
{
|
|
if (!aNode || !aSplitPointParent || !outOffset) return NS_ERROR_NULL_POINTER;
|
|
nsCOMPtr<nsIDOMNode> tempNode, parentNode;
|
|
PRInt32 offset = aSplitPointOffset;
|
|
nsresult res;
|
|
|
|
if (outLeftNode) *outLeftNode = nsnull;
|
|
if (outRightNode) *outRightNode = nsnull;
|
|
|
|
nsCOMPtr<nsIDOMNode> nodeToSplit = do_QueryInterface(aSplitPointParent);
|
|
while (nodeToSplit)
|
|
{
|
|
// need to insert rules code call here to do things like
|
|
// not split a list if you are after the last <li> or before the first, etc.
|
|
// for now we just have some smarts about unneccessarily splitting
|
|
// textnodes, which should be universal enough to put straight in
|
|
// this nsEditor routine.
|
|
|
|
nsCOMPtr<nsIDOMCharacterData> nodeAsText = do_QueryInterface(nodeToSplit);
|
|
PRUint32 len;
|
|
PRBool bDoSplit = PR_FALSE;
|
|
res = GetLengthOfDOMNode(nodeToSplit, len);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
if (!(aNoEmptyContainers || nodeAsText) || (offset && (offset != (PRInt32)len)))
|
|
{
|
|
bDoSplit = PR_TRUE;
|
|
res = SplitNode(nodeToSplit, offset, getter_AddRefs(tempNode));
|
|
if (NS_FAILED(res)) return res;
|
|
if (outRightNode) *outRightNode = nodeToSplit;
|
|
if (outLeftNode) *outLeftNode = tempNode;
|
|
}
|
|
|
|
res = nodeToSplit->GetParentNode(getter_AddRefs(parentNode));
|
|
if (NS_FAILED(res)) return res;
|
|
if (!parentNode) return NS_ERROR_FAILURE;
|
|
|
|
if (!bDoSplit && offset) // must be "end of text node" case, we didn't split it, just move past it
|
|
{
|
|
offset = GetIndexOf(parentNode, nodeToSplit) +1;
|
|
if (outLeftNode) *outLeftNode = nodeToSplit;
|
|
}
|
|
else
|
|
{
|
|
offset = GetIndexOf(parentNode, nodeToSplit);
|
|
if (outRightNode) *outRightNode = nodeToSplit;
|
|
}
|
|
|
|
if (nodeToSplit.get() == aNode) // we split all the way up to (and including) aNode; we're done
|
|
break;
|
|
|
|
nodeToSplit = parentNode;
|
|
}
|
|
|
|
if (!nodeToSplit)
|
|
{
|
|
NS_NOTREACHED("null node obtained in nsEditor::SplitNodeDeep()");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
*outOffset = offset;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// JoinNodeDeep: this joins two like nodes "deeply", joining children as
|
|
// appropriate.
|
|
nsresult
|
|
nsEditor::JoinNodeDeep(nsIDOMNode *aLeftNode,
|
|
nsIDOMNode *aRightNode,
|
|
nsCOMPtr<nsIDOMNode> *aOutJoinNode,
|
|
PRInt32 *outOffset)
|
|
{
|
|
if (!aLeftNode || !aRightNode || !aOutJoinNode || !outOffset) return NS_ERROR_NULL_POINTER;
|
|
|
|
// while the rightmost children and their descendants of the left node
|
|
// match the leftmost children and their descendants of the right node
|
|
// join them up. Can you say that three times fast?
|
|
|
|
nsCOMPtr<nsIDOMNode> leftNodeToJoin = do_QueryInterface(aLeftNode);
|
|
nsCOMPtr<nsIDOMNode> rightNodeToJoin = do_QueryInterface(aRightNode);
|
|
nsCOMPtr<nsIDOMNode> parentNode,tmp;
|
|
nsresult res = NS_OK;
|
|
|
|
rightNodeToJoin->GetParentNode(getter_AddRefs(parentNode));
|
|
|
|
while (leftNodeToJoin && rightNodeToJoin && parentNode &&
|
|
NodesSameType(leftNodeToJoin, rightNodeToJoin))
|
|
{
|
|
// adjust out params
|
|
PRUint32 length;
|
|
if (IsTextNode(leftNodeToJoin))
|
|
{
|
|
nsCOMPtr<nsIDOMCharacterData>nodeAsText;
|
|
nodeAsText = do_QueryInterface(leftNodeToJoin);
|
|
nodeAsText->GetLength(&length);
|
|
}
|
|
else
|
|
{
|
|
res = GetLengthOfDOMNode(leftNodeToJoin, length);
|
|
if (NS_FAILED(res)) return res;
|
|
}
|
|
|
|
*aOutJoinNode = rightNodeToJoin;
|
|
*outOffset = length;
|
|
|
|
// do the join
|
|
res = JoinNodes(leftNodeToJoin, rightNodeToJoin, parentNode);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
if (IsTextNode(parentNode)) // we've joined all the way down to text nodes, we're done!
|
|
return NS_OK;
|
|
|
|
else
|
|
{
|
|
// get new left and right nodes, and begin anew
|
|
parentNode = rightNodeToJoin;
|
|
leftNodeToJoin = GetChildAt(parentNode, length-1);
|
|
rightNodeToJoin = GetChildAt(parentNode, length);
|
|
|
|
// skip over non-editable nodes
|
|
while (leftNodeToJoin && !IsEditable(leftNodeToJoin))
|
|
{
|
|
leftNodeToJoin->GetPreviousSibling(getter_AddRefs(tmp));
|
|
leftNodeToJoin = tmp;
|
|
}
|
|
if (!leftNodeToJoin) break;
|
|
|
|
while (rightNodeToJoin && !IsEditable(rightNodeToJoin))
|
|
{
|
|
rightNodeToJoin->GetNextSibling(getter_AddRefs(tmp));
|
|
rightNodeToJoin = tmp;
|
|
}
|
|
if (!rightNodeToJoin) break;
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
nsresult nsEditor::BeginUpdateViewBatch()
|
|
{
|
|
NS_PRECONDITION(mUpdateCount >= 0, "bad state");
|
|
|
|
|
|
if (0 == mUpdateCount)
|
|
{
|
|
// Turn off selection updates and notifications.
|
|
|
|
nsCOMPtr<nsISelection> selection;
|
|
GetSelection(getter_AddRefs(selection));
|
|
|
|
if (selection)
|
|
{
|
|
nsCOMPtr<nsISelectionPrivate> selPrivate(do_QueryInterface(selection));
|
|
selPrivate->StartBatchChanges();
|
|
}
|
|
|
|
// Turn off view updating.
|
|
mBatch.BeginUpdateViewBatch(mViewManager);
|
|
}
|
|
|
|
mUpdateCount++;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
nsresult nsEditor::EndUpdateViewBatch()
|
|
{
|
|
NS_PRECONDITION(mUpdateCount > 0, "bad state");
|
|
|
|
if (mUpdateCount <= 0)
|
|
{
|
|
mUpdateCount = 0;
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
mUpdateCount--;
|
|
|
|
if (0 == mUpdateCount)
|
|
{
|
|
// Hide the caret with an StCaretHider. By the time it goes out
|
|
// of scope and tries to show the caret, reflow and selection changed
|
|
// notifications should've happened so the caret should have enough info
|
|
// to draw at the correct position.
|
|
|
|
nsRefPtr<nsCaret> caret;
|
|
nsCOMPtr<nsIPresShell> presShell;
|
|
GetPresShell(getter_AddRefs(presShell));
|
|
|
|
if (presShell)
|
|
presShell->GetCaret(getter_AddRefs(caret));
|
|
|
|
StCaretHider caretHider(caret);
|
|
|
|
PRUint32 flags = 0;
|
|
|
|
GetFlags(&flags);
|
|
|
|
// Turn view updating back on.
|
|
if (mViewManager)
|
|
{
|
|
PRUint32 updateFlag = NS_VMREFRESH_IMMEDIATE;
|
|
|
|
// If we're doing async updates, use NS_VMREFRESH_DEFERRED here, so that
|
|
// the reflows we caused will get processed before the invalidates.
|
|
if (flags & nsIPlaintextEditor::eEditorUseAsyncUpdatesMask) {
|
|
updateFlag = NS_VMREFRESH_DEFERRED;
|
|
} else if (presShell) {
|
|
// Flush out layout. Need to do this because if we have no invalidates
|
|
// to flush the viewmanager code won't flush our reflow here, and we
|
|
// have selection code that does sync caret scrolling in this case.
|
|
presShell->FlushPendingNotifications(Flush_Layout);
|
|
}
|
|
mBatch.EndUpdateViewBatch(updateFlag);
|
|
}
|
|
|
|
// Turn selection updating and notifications back on.
|
|
|
|
nsCOMPtr<nsISelection>selection;
|
|
GetSelection(getter_AddRefs(selection));
|
|
|
|
if (selection) {
|
|
nsCOMPtr<nsISelectionPrivate>selPrivate(do_QueryInterface(selection));
|
|
selPrivate->EndBatchChanges();
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::GetShouldTxnSetSelection()
|
|
{
|
|
return mShouldTxnSetSelection;
|
|
}
|
|
|
|
|
|
#ifdef XP_MAC
|
|
#pragma mark -
|
|
#pragma mark protected nsEditor methods
|
|
#pragma mark -
|
|
#endif
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DeleteSelectionImpl(nsIEditor::EDirection aAction)
|
|
{
|
|
nsCOMPtr<nsISelection>selection;
|
|
nsresult res = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(res)) return res;
|
|
nsRefPtr<EditAggregateTxn> txn;
|
|
nsCOMPtr<nsIDOMNode> deleteNode;
|
|
PRInt32 deleteCharOffset = 0, deleteCharLength = 0;
|
|
res = CreateTxnForDeleteSelection(aAction, getter_AddRefs(txn),
|
|
getter_AddRefs(deleteNode),
|
|
&deleteCharOffset, &deleteCharLength);
|
|
nsCOMPtr<nsIDOMCharacterData> deleteCharData(do_QueryInterface(deleteNode));
|
|
|
|
if (NS_SUCCEEDED(res))
|
|
{
|
|
nsAutoRules beginRulesSniffing(this, kOpDeleteSelection, aAction);
|
|
PRInt32 i;
|
|
// Notify nsIEditActionListener::WillDelete[Selection|Text|Node]
|
|
if (!deleteNode)
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillDeleteSelection(selection);
|
|
else if (deleteCharData)
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillDeleteText(deleteCharData, deleteCharOffset, 1);
|
|
else
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->WillDeleteNode(deleteNode);
|
|
|
|
// Delete the specified amount
|
|
res = DoTransaction(txn);
|
|
|
|
// Notify nsIEditActionListener::DidDelete[Selection|Text|Node]
|
|
if (!deleteNode)
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->DidDeleteSelection(selection);
|
|
else if (deleteCharData)
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->DidDeleteText(deleteCharData, deleteCharOffset, 1, res);
|
|
else
|
|
for (i = 0; i < mActionListeners.Count(); i++)
|
|
mActionListeners[i]->DidDeleteNode(deleteNode, res);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
// XXX: error handling in this routine needs to be cleaned up!
|
|
NS_IMETHODIMP
|
|
nsEditor::DeleteSelectionAndCreateNode(const nsAString& aTag,
|
|
nsIDOMNode ** aNewNode)
|
|
{
|
|
nsCOMPtr<nsIDOMNode> parentSelectedNode;
|
|
PRInt32 offsetOfNewNode;
|
|
nsresult result = DeleteSelectionAndPrepareToCreateNode(parentSelectedNode,
|
|
offsetOfNewNode);
|
|
if (NS_FAILED(result))
|
|
return result;
|
|
|
|
nsCOMPtr<nsIDOMNode> newNode;
|
|
result = CreateNode(aTag, parentSelectedNode, offsetOfNewNode,
|
|
getter_AddRefs(newNode));
|
|
// XXX: ERROR_HANDLING check result, and make sure aNewNode is set correctly in success/failure cases
|
|
*aNewNode = newNode;
|
|
NS_IF_ADDREF(*aNewNode);
|
|
|
|
// we want the selection to be just after the new node
|
|
nsCOMPtr<nsISelection> selection;
|
|
result = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(result)) return result;
|
|
if (!selection) return NS_ERROR_NULL_POINTER;
|
|
return selection->Collapse(parentSelectedNode, offsetOfNewNode+1);
|
|
}
|
|
|
|
|
|
/* Non-interface, protected methods */
|
|
|
|
nsresult
|
|
nsEditor::GetIMEBufferLength(PRInt32* length)
|
|
{
|
|
*length = mIMEBufferLength;
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
nsEditor::SetIsIMEComposing(){
|
|
// We set mIsIMEComposing according to mIMETextRangeList.
|
|
nsCOMPtr<nsIPrivateTextRange> rangePtr;
|
|
PRUint16 listlen, type;
|
|
|
|
mIsIMEComposing = PR_FALSE;
|
|
listlen = mIMETextRangeList->GetLength();
|
|
|
|
for (PRUint16 i = 0; i < listlen; i++)
|
|
{
|
|
rangePtr = mIMETextRangeList->Item(i);
|
|
if (!rangePtr) continue;
|
|
nsresult result = rangePtr->GetRangeType(&type);
|
|
if (NS_FAILED(result)) continue;
|
|
if ( type == nsIPrivateTextRange::TEXTRANGE_RAWINPUT ||
|
|
type == nsIPrivateTextRange::TEXTRANGE_CONVERTEDTEXT ||
|
|
type == nsIPrivateTextRange::TEXTRANGE_SELECTEDRAWTEXT ||
|
|
type == nsIPrivateTextRange::TEXTRANGE_SELECTEDCONVERTEDTEXT )
|
|
{
|
|
mIsIMEComposing = PR_TRUE;
|
|
#ifdef DEBUG_IME
|
|
printf("nsEditor::mIsIMEComposing = PR_TRUE\n");
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
PRBool
|
|
nsEditor::IsIMEComposing() {
|
|
return mIsIMEComposing;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DeleteSelectionAndPrepareToCreateNode(nsCOMPtr<nsIDOMNode> &parentSelectedNode, PRInt32& offsetOfNewNode)
|
|
{
|
|
nsresult result=NS_ERROR_NOT_INITIALIZED;
|
|
nsCOMPtr<nsISelection> selection;
|
|
result = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(result)) return result;
|
|
if (!selection) return NS_ERROR_NULL_POINTER;
|
|
|
|
PRBool collapsed;
|
|
result = selection->GetIsCollapsed(&collapsed);
|
|
if (NS_SUCCEEDED(result) && !collapsed)
|
|
{
|
|
result = DeleteSelection(nsIEditor::eNone);
|
|
if (NS_FAILED(result)) {
|
|
return result;
|
|
}
|
|
// get the new selection
|
|
result = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(result)) {
|
|
return result;
|
|
}
|
|
#ifdef NS_DEBUG
|
|
nsCOMPtr<nsIDOMNode>testSelectedNode;
|
|
nsresult debugResult = selection->GetAnchorNode(getter_AddRefs(testSelectedNode));
|
|
// no selection is ok.
|
|
// if there is a selection, it must be collapsed
|
|
if (testSelectedNode)
|
|
{
|
|
PRBool testCollapsed;
|
|
debugResult = selection->GetIsCollapsed(&testCollapsed);
|
|
NS_ASSERTION((NS_SUCCEEDED(result)), "couldn't get a selection after deletion");
|
|
NS_ASSERTION(testCollapsed, "selection not reset after deletion");
|
|
}
|
|
#endif
|
|
}
|
|
// split the selected node
|
|
PRInt32 offsetOfSelectedNode;
|
|
result = selection->GetAnchorNode(getter_AddRefs(parentSelectedNode));
|
|
if (NS_SUCCEEDED(result) && NS_SUCCEEDED(selection->GetAnchorOffset(&offsetOfSelectedNode)) && parentSelectedNode)
|
|
{
|
|
nsCOMPtr<nsIDOMNode> selectedNode;
|
|
PRUint32 selectedNodeContentCount=0;
|
|
nsCOMPtr<nsIDOMCharacterData>selectedParentNodeAsText;
|
|
selectedParentNodeAsText = do_QueryInterface(parentSelectedNode);
|
|
|
|
offsetOfNewNode = offsetOfSelectedNode;
|
|
|
|
/* if the selection is a text node, split the text node if necessary
|
|
and compute where to put the new node
|
|
*/
|
|
if (selectedParentNodeAsText)
|
|
{
|
|
PRInt32 indexOfTextNodeInParent;
|
|
selectedNode = do_QueryInterface(parentSelectedNode);
|
|
selectedNode->GetParentNode(getter_AddRefs(parentSelectedNode));
|
|
selectedParentNodeAsText->GetLength(&selectedNodeContentCount);
|
|
GetChildOffset(selectedNode, parentSelectedNode, indexOfTextNodeInParent);
|
|
|
|
if ((offsetOfSelectedNode!=0) && (((PRUint32)offsetOfSelectedNode)!=selectedNodeContentCount))
|
|
{
|
|
nsCOMPtr<nsIDOMNode> newSiblingNode;
|
|
result = SplitNode(selectedNode, offsetOfSelectedNode, getter_AddRefs(newSiblingNode));
|
|
// now get the node's offset in it's parent, and insert the new tag there
|
|
if (NS_SUCCEEDED(result)) {
|
|
result = GetChildOffset(selectedNode, parentSelectedNode, offsetOfNewNode);
|
|
}
|
|
}
|
|
else
|
|
{ // determine where to insert the new node
|
|
if (0==offsetOfSelectedNode) {
|
|
offsetOfNewNode = indexOfTextNodeInParent; // insert new node as previous sibling to selection parent
|
|
}
|
|
else { // insert new node as last child
|
|
GetChildOffset(selectedNode, parentSelectedNode, offsetOfNewNode);
|
|
offsetOfNewNode++; // offsets are 0-based, and we need the index of the new node
|
|
}
|
|
}
|
|
}
|
|
// Here's where the new node was inserted
|
|
}
|
|
#ifdef DEBUG
|
|
else {
|
|
printf("InsertLineBreak into an empty document is not yet supported\n");
|
|
}
|
|
#endif
|
|
return result;
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DoAfterDoTransaction(nsITransaction *aTxn)
|
|
{
|
|
nsresult rv = NS_OK;
|
|
|
|
PRBool isTransientTransaction;
|
|
rv = aTxn->GetIsTransient(&isTransientTransaction);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
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.
|
|
PRInt32 modCount;
|
|
GetModificationCount(&modCount);
|
|
if (modCount < 0)
|
|
modCount = -modCount;
|
|
|
|
rv = IncrementModificationCount(1); // don't count transient transactions
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DoAfterUndoTransaction()
|
|
{
|
|
nsresult rv = NS_OK;
|
|
|
|
rv = IncrementModificationCount(-1); // all undoable transactions are non-transient
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::DoAfterRedoTransaction()
|
|
{
|
|
return IncrementModificationCount(1); // all redoable transactions are non-transient
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CreateTxnForSetAttribute(nsIDOMElement *aElement,
|
|
const nsAString& aAttribute,
|
|
const nsAString& aValue,
|
|
ChangeAttributeTxn ** aTxn)
|
|
{
|
|
if (!aElement)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aTxn = new ChangeAttributeTxn();
|
|
if (!*aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
return (*aTxn)->Init(this, aElement, aAttribute, aValue, PR_FALSE);
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CreateTxnForRemoveAttribute(nsIDOMElement *aElement,
|
|
const nsAString& aAttribute,
|
|
ChangeAttributeTxn ** aTxn)
|
|
{
|
|
if (!aElement)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aTxn = new ChangeAttributeTxn();
|
|
if (!*aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
|
|
return (*aTxn)->Init(this, aElement, aAttribute, EmptyString(), PR_TRUE);
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::CreateTxnForCreateElement(const nsAString& aTag,
|
|
nsIDOMNode *aParent,
|
|
PRInt32 aPosition,
|
|
CreateElementTxn ** aTxn)
|
|
{
|
|
if (!aParent)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aTxn = new CreateElementTxn();
|
|
if (!*aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
|
|
return (*aTxn)->Init(this, aTag, aParent, aPosition);
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP nsEditor::CreateTxnForInsertElement(nsIDOMNode * aNode,
|
|
nsIDOMNode * aParent,
|
|
PRInt32 aPosition,
|
|
InsertElementTxn ** aTxn)
|
|
{
|
|
if (!aNode || !aParent)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aTxn = new InsertElementTxn();
|
|
if (!*aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
|
|
return (*aTxn)->Init(aNode, aParent, aPosition, this);
|
|
}
|
|
|
|
NS_IMETHODIMP nsEditor::CreateTxnForDeleteElement(nsIDOMNode * aElement,
|
|
DeleteElementTxn ** aTxn)
|
|
{
|
|
if (!aElement)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*aTxn = new DeleteElementTxn();
|
|
if (!*aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
|
|
return (*aTxn)->Init(this, aElement, &mRangeUpdater);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CreateTxnForIMEText(const nsAString& aStringToInsert,
|
|
IMETextTxn ** aTxn)
|
|
{
|
|
NS_ASSERTION(aTxn, "illegal value- null ptr- aTxn");
|
|
|
|
*aTxn = new IMETextTxn();
|
|
if (!*aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
|
|
return (*aTxn)->Init(mIMETextNode, mIMETextOffset, mIMEBufferLength,
|
|
mIMETextRangeList, aStringToInsert, mSelConWeak);
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CreateTxnForAddStyleSheet(nsICSSStyleSheet* aSheet, AddStyleSheetTxn* *aTxn)
|
|
{
|
|
*aTxn = new AddStyleSheetTxn();
|
|
if (! *aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
|
|
return (*aTxn)->Init(this, aSheet);
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CreateTxnForRemoveStyleSheet(nsICSSStyleSheet* aSheet, RemoveStyleSheetTxn* *aTxn)
|
|
{
|
|
*aTxn = new RemoveStyleSheetTxn();
|
|
if (! *aTxn)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
NS_ADDREF(*aTxn);
|
|
|
|
return (*aTxn)->Init(this, aSheet);
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::CreateTxnForDeleteSelection(nsIEditor::EDirection aAction,
|
|
EditAggregateTxn ** aTxn,
|
|
nsIDOMNode ** aNode,
|
|
PRInt32 *aOffset,
|
|
PRInt32 *aLength)
|
|
{
|
|
if (!aTxn)
|
|
return NS_ERROR_NULL_POINTER;
|
|
*aTxn = nsnull;
|
|
|
|
nsCOMPtr<nsISelectionController> selCon = do_QueryReferent(mSelConWeak);
|
|
if (!selCon) return NS_ERROR_NOT_INITIALIZED;
|
|
nsCOMPtr<nsISelection> selection;
|
|
nsresult result = selCon->GetSelection(nsISelectionController::SELECTION_NORMAL,
|
|
getter_AddRefs(selection));
|
|
if ((NS_SUCCEEDED(result)) && selection)
|
|
{
|
|
// Check whether the selection is collapsed and we should do nothing:
|
|
PRBool isCollapsed;
|
|
result = (selection->GetIsCollapsed(&isCollapsed));
|
|
if (NS_SUCCEEDED(result) && isCollapsed && aAction == eNone)
|
|
return NS_OK;
|
|
|
|
// allocate the out-param transaction
|
|
*aTxn = new EditAggregateTxn();
|
|
if (!*aTxn) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
NS_ADDREF(*aTxn);
|
|
|
|
nsCOMPtr<nsISelectionPrivate>selPrivate(do_QueryInterface(selection));
|
|
nsCOMPtr<nsIEnumerator> enumerator;
|
|
result = selPrivate->GetEnumerator(getter_AddRefs(enumerator));
|
|
if (NS_SUCCEEDED(result) && enumerator)
|
|
{
|
|
for (enumerator->First(); NS_OK!=enumerator->IsDone(); enumerator->Next())
|
|
{
|
|
nsCOMPtr<nsISupports> currentItem;
|
|
result = enumerator->CurrentItem(getter_AddRefs(currentItem));
|
|
if ((NS_SUCCEEDED(result)) && (currentItem))
|
|
{
|
|
nsCOMPtr<nsIDOMRange> range( do_QueryInterface(currentItem) );
|
|
range->GetCollapsed(&isCollapsed);
|
|
if (!isCollapsed)
|
|
{
|
|
nsRefPtr<DeleteRangeTxn> txn = new DeleteRangeTxn();
|
|
if (txn)
|
|
{
|
|
txn->Init(this, range, &mRangeUpdater);
|
|
(*aTxn)->AppendChild(txn);
|
|
}
|
|
else
|
|
result = NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
// Same with range as with selection; if it is collapsed and action
|
|
// is eNone, do nothing.
|
|
else if (aAction != eNone)
|
|
{ // we have an insertion point. delete the thing in front of it or behind it, depending on aAction
|
|
result = CreateTxnForDeleteInsertionPoint(range, aAction, *aTxn, aNode, aOffset, aLength);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// if we didn't build the transaction correctly, destroy the out-param transaction so we don't leak it.
|
|
if (NS_FAILED(result))
|
|
{
|
|
NS_IF_RELEASE(*aTxn);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::CreateTxnForDeleteCharacter(nsIDOMCharacterData *aData,
|
|
PRUint32 aOffset,
|
|
nsIEditor::EDirection aDirection,
|
|
DeleteTextTxn **aTxn)
|
|
{
|
|
NS_ASSERTION(aDirection == eNext || aDirection == ePrevious,
|
|
"invalid direction");
|
|
nsAutoString data;
|
|
aData->GetData(data);
|
|
PRUint32 segOffset, segLength = 1;
|
|
if (aDirection == eNext) {
|
|
segOffset = aOffset;
|
|
if (NS_IS_HIGH_SURROGATE(data[segOffset]) &&
|
|
segOffset + 1 < data.Length() &&
|
|
NS_IS_LOW_SURROGATE(data[segOffset+1])) {
|
|
// delete both halves of the surrogate pair
|
|
++segLength;
|
|
}
|
|
} else {
|
|
segOffset = aOffset - 1;
|
|
if (NS_IS_LOW_SURROGATE(data[segOffset]) &&
|
|
segOffset > 0 &&
|
|
NS_IS_HIGH_SURROGATE(data[segOffset-1])) {
|
|
++segLength;
|
|
--segOffset;
|
|
}
|
|
}
|
|
return CreateTxnForDeleteText(aData, segOffset, segLength, aTxn);
|
|
}
|
|
|
|
//XXX: currently, this doesn't handle edge conditions because GetNext/GetPrior are not implemented
|
|
NS_IMETHODIMP
|
|
nsEditor::CreateTxnForDeleteInsertionPoint(nsIDOMRange *aRange,
|
|
nsIEditor::EDirection aAction,
|
|
EditAggregateTxn *aTxn,
|
|
nsIDOMNode **aNode,
|
|
PRInt32 *aOffset,
|
|
PRInt32 *aLength)
|
|
{
|
|
NS_ASSERTION(aAction == eNext || aAction == ePrevious, "invalid action");
|
|
|
|
// get the node and offset of the insertion point
|
|
nsCOMPtr<nsIDOMNode> node;
|
|
nsresult result = aRange->GetStartContainer(getter_AddRefs(node));
|
|
if (NS_FAILED(result))
|
|
return result;
|
|
|
|
PRInt32 offset;
|
|
result = aRange->GetStartOffset(&offset);
|
|
if (NS_FAILED(result))
|
|
return result;
|
|
|
|
// determine if the insertion point is at the beginning, middle, or end of the node
|
|
nsCOMPtr<nsIDOMCharacterData> nodeAsText = do_QueryInterface(node);
|
|
|
|
PRUint32 count=0;
|
|
|
|
if (nodeAsText)
|
|
nodeAsText->GetLength(&count);
|
|
else
|
|
{
|
|
// get the child list and count
|
|
nsCOMPtr<nsIDOMNodeList>childList;
|
|
result = node->GetChildNodes(getter_AddRefs(childList));
|
|
if ((NS_SUCCEEDED(result)) && childList)
|
|
childList->GetLength(&count);
|
|
}
|
|
|
|
PRBool isFirst = (0 == offset);
|
|
PRBool isLast = (count == (PRUint32)offset);
|
|
|
|
// XXX: if isFirst && isLast, 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
|
|
if ((ePrevious==aAction) && (PR_TRUE==isFirst))
|
|
{ // we're backspacing from the beginning of the node. Delete the first thing to our left
|
|
nsCOMPtr<nsIDOMNode> priorNode;
|
|
result = GetPriorNode(node, PR_TRUE, address_of(priorNode));
|
|
if ((NS_SUCCEEDED(result)) && priorNode)
|
|
{ // there is a priorNode, so delete it's last child (if text content, delete the last char.)
|
|
// if it has no children, delete it
|
|
nsCOMPtr<nsIDOMCharacterData> priorNodeAsText = do_QueryInterface(priorNode);
|
|
if (priorNodeAsText)
|
|
{
|
|
PRUint32 length=0;
|
|
priorNodeAsText->GetLength(&length);
|
|
if (0<length)
|
|
{
|
|
DeleteTextTxn *txn;
|
|
result = CreateTxnForDeleteCharacter(priorNodeAsText, length,
|
|
ePrevious, &txn);
|
|
if (NS_SUCCEEDED(result)) {
|
|
aTxn->AppendChild(txn);
|
|
NS_ADDREF(*aNode = priorNode);
|
|
*aOffset = txn->GetOffset();
|
|
*aLength = txn->GetNumCharsToDelete();
|
|
NS_RELEASE(txn);
|
|
}
|
|
}
|
|
else
|
|
{ // XXX: can you have an empty text node? If so, what do you do?
|
|
printf("ERROR: found a text node with 0 characters\n");
|
|
result = NS_ERROR_UNEXPECTED;
|
|
}
|
|
}
|
|
else
|
|
{ // priorNode is not text, so tell it's parent to delete it
|
|
DeleteElementTxn *txn;
|
|
result = CreateTxnForDeleteElement(priorNode, &txn);
|
|
if (NS_SUCCEEDED(result)) {
|
|
aTxn->AppendChild(txn);
|
|
NS_RELEASE(txn);
|
|
NS_ADDREF(*aNode = priorNode);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if ((nsIEditor::eNext==aAction) && (PR_TRUE==isLast))
|
|
{ // we're deleting from the end of the node. Delete the first thing to our right
|
|
nsCOMPtr<nsIDOMNode> nextNode;
|
|
result = GetNextNode(node, PR_TRUE, address_of(nextNode));
|
|
if ((NS_SUCCEEDED(result)) && nextNode)
|
|
{ // there is a nextNode, so delete it's first child (if text content, delete the first char.)
|
|
// if it has no children, delete it
|
|
nsCOMPtr<nsIDOMCharacterData> nextNodeAsText = do_QueryInterface(nextNode);
|
|
if (nextNodeAsText)
|
|
{
|
|
PRUint32 length=0;
|
|
nextNodeAsText->GetLength(&length);
|
|
if (0<length)
|
|
{
|
|
DeleteTextTxn *txn;
|
|
result = CreateTxnForDeleteCharacter(nextNodeAsText, 0, eNext, &txn);
|
|
if (NS_SUCCEEDED(result)) {
|
|
aTxn->AppendChild(txn);
|
|
NS_ADDREF(*aNode = nextNode);
|
|
*aOffset = txn->GetOffset();
|
|
*aLength = txn->GetNumCharsToDelete();
|
|
NS_RELEASE(txn);
|
|
}
|
|
}
|
|
else
|
|
{ // XXX: can you have an empty text node? If so, what do you do?
|
|
printf("ERROR: found a text node with 0 characters\n");
|
|
result = NS_ERROR_UNEXPECTED;
|
|
}
|
|
}
|
|
else
|
|
{ // nextNode is not text, so tell it's parent to delete it
|
|
DeleteElementTxn *txn;
|
|
result = CreateTxnForDeleteElement(nextNode, &txn);
|
|
if (NS_SUCCEEDED(result)) {
|
|
aTxn->AppendChild(txn);
|
|
NS_RELEASE(txn);
|
|
NS_ADDREF(*aNode = nextNode);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (nodeAsText)
|
|
{ // we have text, so delete a char at the proper offset
|
|
nsRefPtr<DeleteTextTxn> txn;
|
|
result = CreateTxnForDeleteCharacter(nodeAsText, offset, aAction,
|
|
getter_AddRefs(txn));
|
|
if (NS_SUCCEEDED(result)) {
|
|
aTxn->AppendChild(txn);
|
|
NS_ADDREF(*aNode = node);
|
|
*aOffset = txn->GetOffset();
|
|
*aLength = txn->GetNumCharsToDelete();
|
|
}
|
|
}
|
|
else
|
|
{ // we're either deleting a node or some text, need to dig into the next/prev node to find out
|
|
nsCOMPtr<nsIDOMNode> selectedNode;
|
|
if (ePrevious==aAction)
|
|
{
|
|
result = GetPriorNode(node, offset, PR_TRUE, address_of(selectedNode));
|
|
}
|
|
else if (eNext==aAction)
|
|
{
|
|
result = GetNextNode(node, offset, PR_TRUE, address_of(selectedNode));
|
|
}
|
|
if (NS_FAILED(result)) { return result; }
|
|
if (selectedNode)
|
|
{
|
|
nsCOMPtr<nsIDOMCharacterData> selectedNodeAsText =
|
|
do_QueryInterface(selectedNode);
|
|
if (selectedNodeAsText)
|
|
{ // we are deleting from a text node, so do a text deletion
|
|
PRUint32 position = 0; // default for forward delete
|
|
if (ePrevious==aAction)
|
|
{
|
|
selectedNodeAsText->GetLength(&position);
|
|
}
|
|
nsRefPtr<DeleteTextTxn> delTextTxn;
|
|
result = CreateTxnForDeleteCharacter(selectedNodeAsText, position,
|
|
aAction,
|
|
getter_AddRefs(delTextTxn));
|
|
if (NS_FAILED(result)) { return result; }
|
|
if (!delTextTxn) { return NS_ERROR_NULL_POINTER; }
|
|
aTxn->AppendChild(delTextTxn);
|
|
NS_ADDREF(*aNode = selectedNode);
|
|
*aOffset = delTextTxn->GetOffset();
|
|
*aLength = delTextTxn->GetNumCharsToDelete();
|
|
}
|
|
else
|
|
{
|
|
nsRefPtr<DeleteElementTxn> delElementTxn;
|
|
result = CreateTxnForDeleteElement(selectedNode,
|
|
getter_AddRefs(delElementTxn));
|
|
if (NS_FAILED(result)) { return result; }
|
|
if (!delElementTxn) { return NS_ERROR_NULL_POINTER; }
|
|
aTxn->AppendChild(delElementTxn);
|
|
NS_ADDREF(*aNode = selectedNode);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::CreateRange(nsIDOMNode *aStartParent, PRInt32 aStartOffset,
|
|
nsIDOMNode *aEndParent, PRInt32 aEndOffset,
|
|
nsIDOMRange **aRange)
|
|
{
|
|
nsresult result;
|
|
result = CallCreateInstance("@mozilla.org/content/range;1", aRange);
|
|
if (NS_FAILED(result))
|
|
return result;
|
|
|
|
if (!*aRange)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
result = (*aRange)->SetStart(aStartParent, aStartOffset);
|
|
|
|
if (NS_SUCCEEDED(result))
|
|
result = (*aRange)->SetEnd(aEndParent, aEndOffset);
|
|
|
|
if (NS_FAILED(result))
|
|
{
|
|
NS_RELEASE((*aRange));
|
|
*aRange = 0;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::AppendNodeToSelectionAsRange(nsIDOMNode *aNode)
|
|
{
|
|
if (!aNode) return NS_ERROR_NULL_POINTER;
|
|
nsCOMPtr<nsISelection> selection;
|
|
nsresult res = GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(res)) return res;
|
|
if(!selection) return NS_ERROR_FAILURE;
|
|
|
|
nsCOMPtr<nsIDOMNode> parentNode;
|
|
res = aNode->GetParentNode(getter_AddRefs(parentNode));
|
|
if (NS_FAILED(res)) return res;
|
|
if (!parentNode) return NS_ERROR_NULL_POINTER;
|
|
|
|
PRInt32 offset;
|
|
res = GetChildOffset(aNode, parentNode, offset);
|
|
if (NS_FAILED(res)) return res;
|
|
|
|
nsCOMPtr<nsIDOMRange> range;
|
|
res = CreateRange(parentNode, offset, parentNode, offset+1, getter_AddRefs(range));
|
|
if (NS_FAILED(res)) return res;
|
|
if (!range) return NS_ERROR_NULL_POINTER;
|
|
|
|
return selection->AddRange(range);
|
|
}
|
|
|
|
nsresult nsEditor::ClearSelection()
|
|
{
|
|
nsCOMPtr<nsISelection> selection;
|
|
nsresult res = nsEditor::GetSelection(getter_AddRefs(selection));
|
|
if (NS_FAILED(res)) return res;
|
|
if (!selection) return NS_ERROR_FAILURE;
|
|
return selection->RemoveAllRanges();
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::CreateHTMLContent(const nsAString& aTag, nsIContent** aContent)
|
|
{
|
|
nsCOMPtr<nsIDOMDocument> tempDoc;
|
|
GetDocument(getter_AddRefs(tempDoc));
|
|
|
|
nsCOMPtr<nsIDocument> doc = do_QueryInterface(tempDoc);
|
|
if (!doc)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
// XXX Wallpaper over editor bug (editor tries to create elements with an
|
|
// empty nodename).
|
|
if (aTag.IsEmpty()) {
|
|
NS_ERROR("Don't pass an empty tag to nsEditor::CreateHTMLContent, "
|
|
"check caller.");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsCOMPtr<nsIAtom> tag = do_GetAtom(aTag);
|
|
if (!tag)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
return doc->CreateElem(tag, nsnull, kNameSpaceID_XHTML, PR_FALSE, aContent);
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::SetAttributeOrEquivalent(nsIDOMElement * aElement,
|
|
const nsAString & aAttribute,
|
|
const nsAString & aValue,
|
|
PRBool aSuppressTransaction)
|
|
{
|
|
return SetAttribute(aElement, aAttribute, aValue);
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::RemoveAttributeOrEquivalent(nsIDOMElement * aElement,
|
|
const nsAString & aAttribute,
|
|
PRBool aSuppressTransaction)
|
|
{
|
|
return RemoveAttribute(aElement, aAttribute);
|
|
}
|
|
|
|
nsresult
|
|
nsEditor::HandleInlineSpellCheck(PRInt32 action,
|
|
nsISelection *aSelection,
|
|
nsIDOMNode *previousSelectedNode,
|
|
PRInt32 previousSelectedOffset,
|
|
nsIDOMNode *aStartNode,
|
|
PRInt32 aStartOffset,
|
|
nsIDOMNode *aEndNode,
|
|
PRInt32 aEndOffset)
|
|
{
|
|
return mInlineSpellChecker ? mInlineSpellChecker->SpellCheckAfterEditorChange(action,
|
|
aSelection,
|
|
previousSelectedNode,
|
|
previousSelectedOffset,
|
|
aStartNode,
|
|
aStartOffset,
|
|
aEndNode,
|
|
aEndOffset) : NS_OK;
|
|
}
|
|
|
|
already_AddRefed<nsPIDOMEventTarget>
|
|
nsEditor::GetPIDOMEventTarget()
|
|
{
|
|
nsPIDOMEventTarget* piTarget = mEventTarget;
|
|
if (piTarget)
|
|
{
|
|
NS_ADDREF(piTarget);
|
|
return piTarget;
|
|
}
|
|
|
|
nsIDOMElement *rootElement = GetRoot();
|
|
|
|
// Now hack to make sure we are not anonymous content.
|
|
// If we are grab the parent of root element for our observer.
|
|
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(rootElement);
|
|
|
|
if (content && content->IsRootOfNativeAnonymousSubtree())
|
|
{
|
|
mEventTarget = do_QueryInterface(content->GetParent());
|
|
piTarget = mEventTarget;
|
|
NS_IF_ADDREF(piTarget);
|
|
}
|
|
else
|
|
{
|
|
// Don't use getDocument here, because we have no way of knowing
|
|
// if Init() was ever called. So we need to get the document
|
|
// ourselves, if it exists.
|
|
if (mDocWeak)
|
|
{
|
|
CallQueryReferent(mDocWeak.get(), &piTarget);
|
|
}
|
|
else
|
|
{
|
|
NS_ERROR("not initialized yet");
|
|
}
|
|
}
|
|
|
|
return piTarget;
|
|
}
|
|
|
|
nsIDOMElement *
|
|
nsEditor::GetRoot()
|
|
{
|
|
if (!mRootElement)
|
|
{
|
|
nsCOMPtr<nsIDOMElement> root;
|
|
|
|
// Let GetRootElement() do the work
|
|
GetRootElement(getter_AddRefs(root));
|
|
}
|
|
|
|
return mRootElement;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEditor::SwitchTextDirection()
|
|
{
|
|
// Get the current root direction from its frame
|
|
nsIDOMElement *rootElement = GetRoot();
|
|
|
|
nsresult rv;
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(rootElement, &rv);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
nsIFrame *frame = content->GetPrimaryFrame();
|
|
if (!frame)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
// Apply the opposite direction
|
|
if (frame->GetStyleVisibility()->mDirection == NS_STYLE_DIRECTION_RTL)
|
|
rv = rootElement->SetAttribute(NS_LITERAL_STRING("dir"), NS_LITERAL_STRING("ltr"));
|
|
else
|
|
rv = rootElement->SetAttribute(NS_LITERAL_STRING("dir"), NS_LITERAL_STRING("rtl"));
|
|
|
|
return rv;
|
|
}
|
|
|
|
#if DEBUG_JOE
|
|
void
|
|
nsEditor::DumpNode(nsIDOMNode *aNode, PRInt32 indent)
|
|
{
|
|
PRInt32 i;
|
|
for (i=0; i<indent; i++)
|
|
printf(" ");
|
|
|
|
nsCOMPtr<nsIDOMElement> element = do_QueryInterface(aNode);
|
|
nsCOMPtr<nsIDOMDocumentFragment> docfrag = do_QueryInterface(aNode);
|
|
|
|
if (element || docfrag)
|
|
{
|
|
if (element)
|
|
{
|
|
nsAutoString tag;
|
|
element->GetTagName(tag);
|
|
printf("<%s>\n", NS_LossyConvertUTF16toASCII(tag).get());
|
|
}
|
|
else
|
|
{
|
|
printf("<document fragment>\n");
|
|
}
|
|
nsCOMPtr<nsIDOMNodeList> childList;
|
|
aNode->GetChildNodes(getter_AddRefs(childList));
|
|
if (!childList) return NS_ERROR_NULL_POINTER;
|
|
PRUint32 numChildren;
|
|
childList->GetLength(&numChildren);
|
|
nsCOMPtr<nsIDOMNode> child, tmp;
|
|
aNode->GetFirstChild(getter_AddRefs(child));
|
|
for (i=0; i<numChildren; i++)
|
|
{
|
|
DumpNode(child, indent+1);
|
|
child->GetNextSibling(getter_AddRefs(tmp));
|
|
child = tmp;
|
|
}
|
|
}
|
|
else if (IsTextNode(aNode))
|
|
{
|
|
nsCOMPtr<nsIDOMCharacterData> textNode = do_QueryInterface(aNode);
|
|
nsAutoString str;
|
|
textNode->GetData(str);
|
|
nsCAutoString cstr;
|
|
LossyCopyUTF16toASCII(str, cstr);
|
|
cstr.ReplaceChar('\n', ' ');
|
|
printf("<textnode> %s\n", cstr.get());
|
|
}
|
|
}
|
|
#endif
|
|
|
|
PRBool
|
|
nsEditor::IsModifiableNode(nsIDOMNode *aNode)
|
|
{
|
|
return PR_TRUE;
|
|
}
|