2007-03-22 10:30:00 -07:00
|
|
|
/* -*- 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):
|
|
|
|
*
|
|
|
|
* 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 ***** */
|
|
|
|
|
|
|
|
#ifndef nsTextEditRules_h__
|
|
|
|
#define nsTextEditRules_h__
|
|
|
|
|
|
|
|
#include "nsCOMPtr.h"
|
|
|
|
|
|
|
|
#include "nsPlaintextEditor.h"
|
|
|
|
#include "nsIDOMNode.h"
|
|
|
|
|
|
|
|
#include "nsEditRules.h"
|
2009-11-02 07:37:25 -08:00
|
|
|
#include "nsITimer.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/** Object that encapsulates HTML text-specific editing rules.
|
|
|
|
*
|
|
|
|
* To be a good citizen, edit rules must live by these restrictions:
|
|
|
|
* 1. All data manipulation is through the editor.
|
|
|
|
* Content nodes in the document tree must <B>not</B> be manipulated directly.
|
|
|
|
* Content nodes in document fragments that are not part of the document itself
|
|
|
|
* may be manipulated at will. Operations on document fragments must <B>not</B>
|
|
|
|
* go through the editor.
|
|
|
|
* 2. Selection must not be explicitly set by the rule method.
|
|
|
|
* Any manipulation of Selection must be done by the editor.
|
|
|
|
*/
|
2009-11-02 07:37:25 -08:00
|
|
|
class nsTextEditRules : public nsIEditRules, public nsITimerCallback
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
public:
|
2009-11-02 07:37:25 -08:00
|
|
|
NS_DECL_NSITIMERCALLBACK
|
2009-05-08 21:59:25 -07:00
|
|
|
NS_DECL_CYCLE_COLLECTING_ISUPPORTS
|
2009-11-02 07:37:25 -08:00
|
|
|
NS_DECL_CYCLE_COLLECTION_CLASS_AMBIGUOUS(nsTextEditRules, nsIEditRules)
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
nsTextEditRules();
|
|
|
|
virtual ~nsTextEditRules();
|
|
|
|
|
|
|
|
// nsIEditRules methods
|
2010-04-11 19:35:18 -07:00
|
|
|
NS_IMETHOD Init(nsPlaintextEditor *aEditor);
|
2009-05-08 21:59:24 -07:00
|
|
|
NS_IMETHOD DetachEditor();
|
2007-03-22 10:30:00 -07:00
|
|
|
NS_IMETHOD BeforeEdit(PRInt32 action, nsIEditor::EDirection aDirection);
|
|
|
|
NS_IMETHOD AfterEdit(PRInt32 action, nsIEditor::EDirection aDirection);
|
|
|
|
NS_IMETHOD WillDoAction(nsISelection *aSelection, nsRulesInfo *aInfo, PRBool *aCancel, PRBool *aHandled);
|
|
|
|
NS_IMETHOD DidDoAction(nsISelection *aSelection, nsRulesInfo *aInfo, nsresult aResult);
|
|
|
|
NS_IMETHOD DocumentIsEmpty(PRBool *aDocumentIsEmpty);
|
2010-11-11 13:40:52 -08:00
|
|
|
NS_IMETHOD DocumentModified();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// nsTextEditRules action id's
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
kDefault = 0,
|
|
|
|
// any editor that has a txn mgr
|
|
|
|
kUndo = 1000,
|
|
|
|
kRedo = 1001,
|
|
|
|
// text actions
|
|
|
|
kInsertText = 2000,
|
|
|
|
kInsertTextIME = 2001,
|
|
|
|
kDeleteSelection = 2002,
|
|
|
|
kSetTextProperty = 2003,
|
|
|
|
kRemoveTextProperty = 2004,
|
|
|
|
kOutputText = 2005,
|
|
|
|
// html only action
|
|
|
|
kInsertBreak = 3000,
|
|
|
|
kMakeList = 3001,
|
|
|
|
kIndent = 3002,
|
|
|
|
kOutdent = 3003,
|
|
|
|
kAlign = 3004,
|
|
|
|
kMakeBasicBlock = 3005,
|
|
|
|
kRemoveList = 3006,
|
|
|
|
kMakeDefListItem = 3007,
|
|
|
|
kInsertElement = 3008,
|
|
|
|
kLoadHTML = 3013,
|
|
|
|
kSetAbsolutePosition = 3015,
|
|
|
|
kRemoveAbsolutePosition = 3016,
|
|
|
|
kDecreaseZIndex = 3017,
|
|
|
|
kIncreaseZIndex = 3018
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
|
|
|
nsresult ResetIMETextPWBuf();
|
|
|
|
|
2010-02-01 10:12:31 -08:00
|
|
|
/**
|
|
|
|
* Handles the newline characters either according to aNewLineHandling
|
|
|
|
* or to the default system prefs if aNewLineHandling is negative.
|
|
|
|
*
|
|
|
|
* @param aString the string to be modified in place.
|
|
|
|
* @param aNewLineHandling determine the desired type of newline handling:
|
|
|
|
* * negative values:
|
|
|
|
* handle newlines according to platform defaults.
|
|
|
|
* * nsIPlaintextEditor::eNewlinesReplaceWithSpaces:
|
|
|
|
* replace newlines with spaces.
|
|
|
|
* * nsIPlaintextEditor::eNewlinesStrip:
|
|
|
|
* remove newlines from the string.
|
|
|
|
* * nsIPlaintextEditor::eNewlinesReplaceWithCommas:
|
|
|
|
* replace newlines with commas.
|
|
|
|
* * nsIPlaintextEditor::eNewlinesStripSurroundingWhitespace:
|
|
|
|
* collapse newlines and surrounding whitespace characters and
|
|
|
|
* remove them from the string.
|
|
|
|
* * nsIPlaintextEditor::eNewlinesPasteIntact:
|
|
|
|
* only remove the leading and trailing newlines.
|
|
|
|
* * nsIPlaintextEditor::eNewlinesPasteToFirst or any other value:
|
|
|
|
* remove the first newline and all characters following it.
|
|
|
|
*/
|
|
|
|
static void HandleNewLines(nsString &aString, PRInt32 aNewLineHandling);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Prepare a string buffer for being displayed as the contents of a password
|
|
|
|
* field. This function uses the platform-specific character for representing
|
|
|
|
* characters entered into password fields.
|
|
|
|
*
|
|
|
|
* @param aOutString the output string. When this function returns,
|
|
|
|
* aOutString will contain aLength password characters.
|
|
|
|
* @param aLength the number of password characters that aOutString should
|
|
|
|
* contain.
|
|
|
|
*/
|
|
|
|
static nsresult FillBufWithPWChars(nsAString *aOutString, PRInt32 aLength);
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
protected:
|
|
|
|
|
|
|
|
// nsTextEditRules implementation methods
|
|
|
|
nsresult WillInsertText( PRInt32 aAction,
|
|
|
|
nsISelection *aSelection,
|
|
|
|
PRBool *aCancel,
|
|
|
|
PRBool *aHandled,
|
|
|
|
const nsAString *inString,
|
|
|
|
nsAString *outString,
|
|
|
|
PRInt32 aMaxLength);
|
|
|
|
nsresult DidInsertText(nsISelection *aSelection, nsresult aResult);
|
|
|
|
nsresult GetTopEnclosingPre(nsIDOMNode *aNode, nsIDOMNode** aOutPreNode);
|
|
|
|
|
2010-08-26 22:36:09 -07:00
|
|
|
nsresult WillInsertBreak(nsISelection *aSelection, PRBool *aCancel,
|
|
|
|
PRBool *aHandled, PRInt32 aMaxLength);
|
2007-03-22 10:30:00 -07:00
|
|
|
nsresult DidInsertBreak(nsISelection *aSelection, nsresult aResult);
|
|
|
|
|
|
|
|
nsresult WillInsert(nsISelection *aSelection, PRBool *aCancel);
|
|
|
|
nsresult DidInsert(nsISelection *aSelection, nsresult aResult);
|
|
|
|
|
|
|
|
nsresult WillDeleteSelection(nsISelection *aSelection,
|
|
|
|
nsIEditor::EDirection aCollapsedAction,
|
|
|
|
PRBool *aCancel,
|
|
|
|
PRBool *aHandled);
|
|
|
|
nsresult DidDeleteSelection(nsISelection *aSelection,
|
|
|
|
nsIEditor::EDirection aCollapsedAction,
|
|
|
|
nsresult aResult);
|
|
|
|
|
|
|
|
nsresult WillSetTextProperty(nsISelection *aSelection, PRBool *aCancel, PRBool *aHandled);
|
|
|
|
nsresult DidSetTextProperty(nsISelection *aSelection, nsresult aResult);
|
|
|
|
|
|
|
|
nsresult WillRemoveTextProperty(nsISelection *aSelection, PRBool *aCancel, PRBool *aHandled);
|
|
|
|
nsresult DidRemoveTextProperty(nsISelection *aSelection, nsresult aResult);
|
|
|
|
|
|
|
|
nsresult WillUndo(nsISelection *aSelection, PRBool *aCancel, PRBool *aHandled);
|
|
|
|
nsresult DidUndo(nsISelection *aSelection, nsresult aResult);
|
|
|
|
|
|
|
|
nsresult WillRedo(nsISelection *aSelection, PRBool *aCancel, PRBool *aHandled);
|
|
|
|
nsresult DidRedo(nsISelection *aSelection, nsresult aResult);
|
|
|
|
|
|
|
|
/** called prior to nsIEditor::OutputToString
|
|
|
|
* @param aSelection
|
|
|
|
* @param aInFormat the format requested for the output, a MIME type
|
|
|
|
* @param aOutText the string to use for output, if aCancel is set to true
|
|
|
|
* @param aOutCancel if set to PR_TRUE, the caller should cancel the operation
|
|
|
|
* and use aOutText as the result.
|
|
|
|
*/
|
|
|
|
nsresult WillOutputText(nsISelection *aSelection,
|
|
|
|
const nsAString *aInFormat,
|
|
|
|
nsAString *aOutText,
|
|
|
|
PRBool *aOutCancel,
|
|
|
|
PRBool *aHandled);
|
|
|
|
|
|
|
|
nsresult DidOutputText(nsISelection *aSelection, nsresult aResult);
|
|
|
|
|
|
|
|
|
|
|
|
// helper functions
|
|
|
|
|
|
|
|
/** creates a trailing break in the text doc if there is not one already */
|
|
|
|
nsresult CreateTrailingBRIfNeeded();
|
|
|
|
|
|
|
|
/** creates a bogus text node if the document has no editable content */
|
|
|
|
nsresult CreateBogusNodeIfNeeded(nsISelection *aSelection);
|
|
|
|
|
|
|
|
/** returns a truncated insertion string if insertion would place us
|
|
|
|
over aMaxLength */
|
|
|
|
nsresult TruncateInsertionIfNeeded(nsISelection *aSelection,
|
|
|
|
const nsAString *aInString,
|
|
|
|
nsAString *aOutString,
|
2010-08-26 22:36:09 -07:00
|
|
|
PRInt32 aMaxLength,
|
|
|
|
PRBool *aTruncated);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/** Remove IME composition text from password buffer */
|
|
|
|
nsresult RemoveIMETextFromPWBuf(PRUint32 &aStart, nsAString *aIMEString);
|
|
|
|
|
|
|
|
nsresult CreateMozBR(nsIDOMNode *inParent, PRInt32 inOffset, nsCOMPtr<nsIDOMNode> *outBRNode);
|
|
|
|
|
|
|
|
nsresult CheckBidiLevelForDeletion(nsISelection *aSelection,
|
|
|
|
nsIDOMNode *aSelNode,
|
|
|
|
PRInt32 aSelOffset,
|
|
|
|
nsIEditor::EDirection aAction,
|
|
|
|
PRBool *aCancel);
|
|
|
|
|
2009-11-02 07:37:25 -08:00
|
|
|
nsresult HideLastPWInput();
|
|
|
|
|
2010-09-01 15:06:52 -07:00
|
|
|
nsresult CollapseSelectionToTrailingBRIfNeeded(nsISelection *aSelection);
|
|
|
|
|
2010-04-11 19:35:18 -07:00
|
|
|
PRBool IsPasswordEditor() const
|
|
|
|
{
|
|
|
|
return mEditor ? mEditor->IsPasswordEditor() : PR_FALSE;
|
|
|
|
}
|
|
|
|
PRBool IsSingleLineEditor() const
|
|
|
|
{
|
|
|
|
return mEditor ? mEditor->IsSingleLineEditor() : PR_FALSE;
|
|
|
|
}
|
|
|
|
PRBool IsPlaintextEditor() const
|
|
|
|
{
|
|
|
|
return mEditor ? mEditor->IsPlaintextEditor() : PR_FALSE;
|
|
|
|
}
|
|
|
|
PRBool IsReadonly() const
|
|
|
|
{
|
|
|
|
return mEditor ? mEditor->IsReadonly() : PR_FALSE;
|
|
|
|
}
|
|
|
|
PRBool IsDisabled() const
|
|
|
|
{
|
|
|
|
return mEditor ? mEditor->IsDisabled() : PR_FALSE;
|
|
|
|
}
|
|
|
|
PRBool IsMailEditor() const
|
|
|
|
{
|
|
|
|
return mEditor ? mEditor->IsMailEditor() : PR_FALSE;
|
|
|
|
}
|
|
|
|
PRBool DontEchoPassword() const
|
|
|
|
{
|
|
|
|
return mEditor ? mEditor->DontEchoPassword() : PR_FALSE;
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
// data members
|
|
|
|
nsPlaintextEditor *mEditor; // note that we do not refcount the editor
|
|
|
|
nsString mPasswordText; // a buffer we use to store the real value of password editors
|
|
|
|
nsString mPasswordIMEText; // a buffer we use to track the IME composition string
|
|
|
|
PRUint32 mPasswordIMEIndex;
|
|
|
|
nsCOMPtr<nsIDOMNode> mBogusNode; // magic node acts as placeholder in empty doc
|
|
|
|
nsCOMPtr<nsIDOMNode> mCachedSelectionNode; // cached selected node
|
|
|
|
PRInt32 mCachedSelectionOffset; // cached selected offset
|
|
|
|
PRUint32 mActionNesting;
|
|
|
|
PRPackedBool mLockRulesSniffing;
|
|
|
|
PRPackedBool mDidExplicitlySetInterline;
|
|
|
|
PRPackedBool mDeleteBidiImmediately; // in bidirectional text, delete
|
|
|
|
// characters not visually
|
|
|
|
// adjacent to the caret without
|
|
|
|
// moving the caret first.
|
|
|
|
PRInt32 mTheAction; // the top level editor action
|
2009-11-02 07:37:25 -08:00
|
|
|
nsCOMPtr<nsITimer> mTimer;
|
|
|
|
PRUint32 mLastStart, mLastLength;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
// friends
|
|
|
|
friend class nsAutoLockRulesSniffing;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class nsTextRulesInfo : public nsRulesInfo
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
nsTextRulesInfo(int aAction) :
|
|
|
|
nsRulesInfo(aAction),
|
|
|
|
inString(0),
|
|
|
|
outString(0),
|
|
|
|
outputFormat(0),
|
|
|
|
maxLength(-1),
|
|
|
|
collapsedAction(nsIEditor::eNext),
|
|
|
|
bOrdered(PR_FALSE),
|
|
|
|
entireList(PR_FALSE),
|
|
|
|
bulletType(0),
|
|
|
|
alignType(0),
|
|
|
|
blockType(0),
|
|
|
|
insertElement(0)
|
2007-04-23 07:21:53 -07:00
|
|
|
{}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2007-04-23 07:21:53 -07:00
|
|
|
virtual ~nsTextRulesInfo() {}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// kInsertText
|
|
|
|
const nsAString *inString;
|
|
|
|
nsAString *outString;
|
|
|
|
const nsAString *outputFormat;
|
|
|
|
PRInt32 maxLength;
|
|
|
|
|
|
|
|
// kDeleteSelection
|
|
|
|
nsIEditor::EDirection collapsedAction;
|
|
|
|
|
|
|
|
// kMakeList
|
|
|
|
PRBool bOrdered;
|
|
|
|
PRBool entireList;
|
|
|
|
const nsAString *bulletType;
|
|
|
|
|
|
|
|
// kAlign
|
|
|
|
const nsAString *alignType;
|
|
|
|
|
|
|
|
// kMakeBasicBlock
|
|
|
|
const nsAString *blockType;
|
|
|
|
|
|
|
|
// kInsertElement
|
|
|
|
const nsIDOMElement* insertElement;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************************
|
|
|
|
* stack based helper class for StartOperation()/EndOperation() sandwich.
|
|
|
|
* this class sets a bool letting us know to ignore any rules sniffing
|
|
|
|
* that tries to occur reentrantly.
|
|
|
|
*/
|
|
|
|
class nsAutoLockRulesSniffing
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
nsAutoLockRulesSniffing(nsTextEditRules *rules) : mRules(rules)
|
|
|
|
{if (mRules) mRules->mLockRulesSniffing = PR_TRUE;}
|
|
|
|
~nsAutoLockRulesSniffing()
|
|
|
|
{if (mRules) mRules->mLockRulesSniffing = PR_FALSE;}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
nsTextEditRules *mRules;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************************
|
|
|
|
* stack based helper class for turning on/off the edit listener.
|
|
|
|
*/
|
|
|
|
class nsAutoLockListener
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
nsAutoLockListener(PRPackedBool *enabled) : mEnabled(enabled)
|
|
|
|
{if (mEnabled) { mOldState=*mEnabled; *mEnabled = PR_FALSE;}}
|
|
|
|
~nsAutoLockListener()
|
|
|
|
{if (mEnabled) *mEnabled = mOldState;}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
PRPackedBool *mEnabled;
|
|
|
|
PRPackedBool mOldState;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
nsresult NS_NewTextEditRules(nsIEditRules** aInstancePtrResult);
|
|
|
|
|
|
|
|
#endif //nsTextEditRules_h__
|
|
|
|
|