Fix for bug 237964 (Allow editable areas in browser (contentEditable)). r/sr=sicking.
git-svn-id: svn://10.0.0.236/trunk@228918 18797224-902f-48f8-a5cc-f745e15eee43
This commit is contained in:
parent
3a7754aab2
commit
9e04fca5dd
@ -63,9 +63,8 @@ class nsIDocShell;
|
|||||||
|
|
||||||
// IID for the nsIContent interface
|
// IID for the nsIContent interface
|
||||||
#define NS_ICONTENT_IID \
|
#define NS_ICONTENT_IID \
|
||||||
{ 0xb6408b0, 0x20c6, 0x4d60, \
|
{ 0x36b375cb, 0xf01e, 0x4c18, \
|
||||||
{ 0xb7, 0x2f, 0x90, 0xb7, 0x7a, 0x9d, 0xb9, 0xb6 } }
|
{ 0xbf, 0x9e, 0xba, 0xad, 0x77, 0x1d, 0xce, 0x22 } }
|
||||||
|
|
||||||
|
|
||||||
// hack to make egcs / gcc 2.95.2 happy
|
// hack to make egcs / gcc 2.95.2 happy
|
||||||
class nsIContent_base : public nsINode {
|
class nsIContent_base : public nsINode {
|
||||||
@ -715,10 +714,7 @@ public:
|
|||||||
*/
|
*/
|
||||||
// XXXbz this is PRInt32 because all the ESM content state APIs use
|
// XXXbz this is PRInt32 because all the ESM content state APIs use
|
||||||
// PRInt32. We should really use PRUint32 instead.
|
// PRInt32. We should really use PRUint32 instead.
|
||||||
virtual PRInt32 IntrinsicState() const
|
virtual PRInt32 IntrinsicState() const;
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* The default script type (language) ID for this content.
|
/* The default script type (language) ID for this content.
|
||||||
All content must support fetching the default script language.
|
All content must support fetching the default script language.
|
||||||
@ -790,6 +786,12 @@ public:
|
|||||||
*/
|
*/
|
||||||
virtual nsIAtom *GetClassAttributeName() const = 0;
|
virtual nsIAtom *GetClassAttributeName() const = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Should be called when the node can become editable or when it can stop
|
||||||
|
* being editable (for example when its contentEditable attribute changes,
|
||||||
|
* when it is moved into an editable parent, ...).
|
||||||
|
*/
|
||||||
|
virtual void UpdateEditableState();
|
||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
/**
|
/**
|
||||||
|
|||||||
@ -89,11 +89,13 @@ class nsNodeSupportsWeakRefTearoff;
|
|||||||
// Whether a binding manager may have a pointer to this
|
// Whether a binding manager may have a pointer to this
|
||||||
#define NODE_MAY_BE_IN_BINDING_MNGR 0x00000080U
|
#define NODE_MAY_BE_IN_BINDING_MNGR 0x00000080U
|
||||||
|
|
||||||
|
#define NODE_IS_EDITABLE 0x00000100U
|
||||||
|
|
||||||
// Four bits for the script-type ID
|
// Four bits for the script-type ID
|
||||||
#define NODE_SCRIPT_TYPE_OFFSET 8
|
#define NODE_SCRIPT_TYPE_OFFSET 9
|
||||||
|
|
||||||
// Remaining bits are node type specific.
|
// Remaining bits are node type specific.
|
||||||
#define NODE_TYPE_SPECIFIC_BITS_OFFSET 0x0c
|
#define NODE_TYPE_SPECIFIC_BITS_OFFSET 0x0d
|
||||||
|
|
||||||
// Useful macro for getting a node given an nsIContent and an nsIDocument
|
// Useful macro for getting a node given an nsIContent and an nsIDocument
|
||||||
// Returns the first argument cast to nsINode if it is non-null, otherwise
|
// Returns the first argument cast to nsINode if it is non-null, otherwise
|
||||||
@ -105,8 +107,8 @@ class nsNodeSupportsWeakRefTearoff;
|
|||||||
|
|
||||||
// IID for the nsINode interface
|
// IID for the nsINode interface
|
||||||
#define NS_INODE_IID \
|
#define NS_INODE_IID \
|
||||||
{ 0x22ab1440, 0xa6ee, 0x4da7, \
|
{ 0xd3e63f80, 0x9e98, 0x47d7, \
|
||||||
{ 0xbc, 0x3b, 0x94, 0x2e, 0x56, 0x0d, 0xdc, 0xe0 } }
|
{ 0xac, 0x8d, 0xad, 0x6f, 0x20, 0x6c, 0xe7, 0xc6 } }
|
||||||
|
|
||||||
// hack to make egcs / gcc 2.95.2 happy
|
// hack to make egcs / gcc 2.95.2 happy
|
||||||
class nsINode_base : public nsPIDOMEventTarget {
|
class nsINode_base : public nsPIDOMEventTarget {
|
||||||
@ -596,6 +598,16 @@ public:
|
|||||||
*flags &= ~aFlagsToUnset;
|
*flags &= ~aFlagsToUnset;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SetEditableFlag(PRBool aEditable)
|
||||||
|
{
|
||||||
|
if (aEditable) {
|
||||||
|
SetFlags(NODE_IS_EDITABLE);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
UnsetFlags(NODE_IS_EDITABLE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
// Override this function to create a custom slots class.
|
// Override this function to create a custom slots class.
|
||||||
|
|||||||
@ -41,6 +41,7 @@
|
|||||||
|
|
||||||
interface nsIDOMRange;
|
interface nsIDOMRange;
|
||||||
interface nsISelectionListener;
|
interface nsISelectionListener;
|
||||||
|
interface nsIContent;
|
||||||
|
|
||||||
%{C++
|
%{C++
|
||||||
class nsFrameSelection;
|
class nsFrameSelection;
|
||||||
@ -54,7 +55,7 @@ struct nsPoint;
|
|||||||
[ptr] native nsIPresShell(nsIPresShell);
|
[ptr] native nsIPresShell(nsIPresShell);
|
||||||
[ref] native nsPointRef(nsPoint);
|
[ref] native nsPointRef(nsPoint);
|
||||||
|
|
||||||
[scriptable, uuid(3225CA54-D7E1-4FF5-8EE9-091B0BFCDA1F)]
|
[scriptable, uuid(b416c692-eeb8-4186-addd-c444e81b68e5)]
|
||||||
interface nsISelectionPrivate : nsISupports
|
interface nsISelectionPrivate : nsISupports
|
||||||
{
|
{
|
||||||
const short ENDOFPRECEDINGLINE=0;
|
const short ENDOFPRECEDINGLINE=0;
|
||||||
@ -115,5 +116,7 @@ interface nsISelectionPrivate : nsISupports
|
|||||||
* Returnes a reference to the frame selection associated with this selection
|
* Returnes a reference to the frame selection associated with this selection
|
||||||
*/
|
*/
|
||||||
[noscript] nsFrameSelection getFrameSelection();
|
[noscript] nsFrameSelection getFrameSelection();
|
||||||
|
|
||||||
|
[noscript] void setAncestorLimiter(in nsIContent aContent);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@ -598,6 +598,8 @@ nsGenericDOMDataNode::BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
|||||||
|
|
||||||
nsNodeUtils::ParentChainChanged(this);
|
nsNodeUtils::ParentChainChanged(this);
|
||||||
|
|
||||||
|
UpdateEditableState();
|
||||||
|
|
||||||
NS_POSTCONDITION(aDocument == GetCurrentDoc(), "Bound to wrong document");
|
NS_POSTCONDITION(aDocument == GetCurrentDoc(), "Bound to wrong document");
|
||||||
NS_POSTCONDITION(aParent == GetParent(), "Bound to wrong parent");
|
NS_POSTCONDITION(aParent == GetParent(), "Bound to wrong parent");
|
||||||
NS_POSTCONDITION(aBindingParent == GetBindingParent(),
|
NS_POSTCONDITION(aBindingParent == GetBindingParent(),
|
||||||
|
|||||||
@ -300,6 +300,28 @@ nsIContent::SetNativeAnonymous(PRBool aAnonymous)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PRInt32
|
||||||
|
nsIContent::IntrinsicState() const
|
||||||
|
{
|
||||||
|
PRBool editable = HasFlag(NODE_IS_EDITABLE);
|
||||||
|
if (!editable) {
|
||||||
|
nsIDocument *doc = GetCurrentDoc();
|
||||||
|
if (doc) {
|
||||||
|
editable = doc->HasFlag(NODE_IS_EDITABLE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return editable ? NS_EVENT_STATE_MOZ_READWRITE : NS_EVENT_STATE_MOZ_READONLY;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
nsIContent::UpdateEditableState()
|
||||||
|
{
|
||||||
|
nsIContent *parent = GetParent();
|
||||||
|
|
||||||
|
SetEditableFlag(parent && parent->HasFlag(NODE_IS_EDITABLE));
|
||||||
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------------
|
//----------------------------------------------------------------------
|
||||||
|
|
||||||
nsChildContentList::~nsChildContentList()
|
nsChildContentList::~nsChildContentList()
|
||||||
@ -2002,6 +2024,8 @@ nsGenericElement::BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
UpdateEditableState();
|
||||||
|
|
||||||
// Now recurse into our kids
|
// Now recurse into our kids
|
||||||
PRUint32 i;
|
PRUint32 i;
|
||||||
// Don't call GetChildCount() here since that'll make XUL generate
|
// Don't call GetChildCount() here since that'll make XUL generate
|
||||||
|
|||||||
@ -208,6 +208,7 @@ GK_ATOM(commandupdater, "commandupdater")
|
|||||||
GK_ATOM(comment, "comment")
|
GK_ATOM(comment, "comment")
|
||||||
GK_ATOM(compact, "compact")
|
GK_ATOM(compact, "compact")
|
||||||
GK_ATOM(concat, "concat")
|
GK_ATOM(concat, "concat")
|
||||||
|
GK_ATOM(contenteditable, "contenteditable")
|
||||||
GK_ATOM(conditions, "conditions")
|
GK_ATOM(conditions, "conditions")
|
||||||
GK_ATOM(constructor, "constructor")
|
GK_ATOM(constructor, "constructor")
|
||||||
GK_ATOM(container, "container")
|
GK_ATOM(container, "container")
|
||||||
@ -410,6 +411,7 @@ GK_ATOM(indent, "indent")
|
|||||||
GK_ATOM(index, "index")
|
GK_ATOM(index, "index")
|
||||||
GK_ATOM(infer, "infer")
|
GK_ATOM(infer, "infer")
|
||||||
GK_ATOM(infinity, "infinity")
|
GK_ATOM(infinity, "infinity")
|
||||||
|
GK_ATOM(inherit, "inherit")
|
||||||
GK_ATOM(inherits, "inherits")
|
GK_ATOM(inherits, "inherits")
|
||||||
GK_ATOM(inheritstyle, "inheritstyle")
|
GK_ATOM(inheritstyle, "inheritstyle")
|
||||||
GK_ATOM(input, "input")
|
GK_ATOM(input, "input")
|
||||||
|
|||||||
@ -239,7 +239,9 @@ nsTextNode::List(FILE* out, PRInt32 aIndent) const
|
|||||||
PRInt32 index;
|
PRInt32 index;
|
||||||
for (index = aIndent; --index >= 0; ) fputs(" ", out);
|
for (index = aIndent; --index >= 0; ) fputs(" ", out);
|
||||||
|
|
||||||
fprintf(out, "Text@%p refcount=%d<", this, mRefCnt.get());
|
fprintf(out, "Text@%p", this);
|
||||||
|
fprintf(out, " intrinsicstate=[%08x]", IntrinsicState());
|
||||||
|
fprintf(out, " refcount=%d<", mRefCnt.get());
|
||||||
|
|
||||||
nsAutoString tmp;
|
nsAutoString tmp;
|
||||||
ToCString(tmp, 0, mText.GetLength());
|
ToCString(tmp, 0, mText.GetLength());
|
||||||
|
|||||||
@ -1102,18 +1102,12 @@ nsGenericHTMLElement::GetSpellcheck(PRBool* aSpellcheck)
|
|||||||
return NS_OK; // Not spellchecked by default
|
return NS_OK; // Not spellchecked by default
|
||||||
}
|
}
|
||||||
|
|
||||||
// Is this the actual body of the current document?
|
|
||||||
if (IsCurrentBodyElement()) {
|
if (IsCurrentBodyElement()) {
|
||||||
// Is designMode on?
|
nsCOMPtr<nsIHTMLDocument> doc = do_QueryInterface(GetCurrentDoc());
|
||||||
nsCOMPtr<nsIDOMNSHTMLDocument> nsHTMLDocument =
|
if (doc) {
|
||||||
do_QueryInterface(GetCurrentDoc());
|
*aSpellcheck = doc->IsEditingOn();
|
||||||
if (!nsHTMLDocument) {
|
|
||||||
return PR_FALSE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
nsAutoString designMode;
|
|
||||||
nsHTMLDocument->GetDesignMode(designMode);
|
|
||||||
*aSpellcheck = designMode.EqualsLiteral("on");
|
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1164,6 +1158,20 @@ nsGenericHTMLElement::InNavQuirksMode(nsIDocument* aDoc)
|
|||||||
return aDoc && aDoc->GetCompatibilityMode() == eCompatibility_NavQuirks;
|
return aDoc && aDoc->GetCompatibilityMode() == eCompatibility_NavQuirks;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
nsGenericHTMLElement::UpdateEditableState()
|
||||||
|
{
|
||||||
|
// XXX Should we do this only when in a document?
|
||||||
|
ContentEditableTristate value = GetContentEditableValue();
|
||||||
|
if (value != eInherit) {
|
||||||
|
SetEditableFlag(value);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsGenericElement::UpdateEditableState();
|
||||||
|
}
|
||||||
|
|
||||||
nsresult
|
nsresult
|
||||||
nsGenericHTMLElement::BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
nsGenericHTMLElement::BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
||||||
nsIContent* aBindingParent,
|
nsIContent* aBindingParent,
|
||||||
@ -1174,6 +1182,14 @@ nsGenericHTMLElement::BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
|||||||
aCompileEventHandlers);
|
aCompileEventHandlers);
|
||||||
NS_ENSURE_SUCCESS(rv, rv);
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
if (aDocument && HasFlag(NODE_IS_EDITABLE) &&
|
||||||
|
GetContentEditableValue() == eTrue) {
|
||||||
|
nsCOMPtr<nsIHTMLDocument> htmlDocument = do_QueryInterface(aDocument);
|
||||||
|
if (htmlDocument) {
|
||||||
|
htmlDocument->ChangeContentEditableCount(this, +1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// XXXbz if we already have a style attr parsed, this won't do
|
// XXXbz if we already have a style attr parsed, this won't do
|
||||||
// anything... need to fix that.
|
// anything... need to fix that.
|
||||||
ReparseStyleAttribute();
|
ReparseStyleAttribute();
|
||||||
@ -1190,6 +1206,19 @@ nsGenericHTMLElement::BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
|||||||
return rv;
|
return rv;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
nsGenericHTMLElement::UnbindFromTree(PRBool aDeep, PRBool aNullParent)
|
||||||
|
{
|
||||||
|
if (GetContentEditableValue() == eTrue) {
|
||||||
|
nsCOMPtr<nsIHTMLDocument> htmlDocument = do_QueryInterface(GetCurrentDoc());
|
||||||
|
if (htmlDocument) {
|
||||||
|
htmlDocument->ChangeContentEditableCount(this, -1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
nsGenericElement::UnbindFromTree(aDeep, aNullParent);
|
||||||
|
}
|
||||||
|
|
||||||
already_AddRefed<nsIDOMHTMLFormElement>
|
already_AddRefed<nsIDOMHTMLFormElement>
|
||||||
nsGenericHTMLElement::FindForm(nsIForm* aCurrentForm)
|
nsGenericHTMLElement::FindForm(nsIForm* aCurrentForm)
|
||||||
{
|
{
|
||||||
@ -1398,13 +1427,44 @@ nsGenericHTMLElement::GetEventListenerManagerForAttr(nsIEventListenerManager** a
|
|||||||
aDefer);
|
aDefer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nsresult
|
||||||
|
nsGenericHTMLElement::SetAttr(PRInt32 aNameSpaceID, nsIAtom* aName,
|
||||||
|
nsIAtom* aPrefix, const nsAString& aValue,
|
||||||
|
PRBool aNotify)
|
||||||
|
{
|
||||||
|
PRBool contentEditable = aNameSpaceID == kNameSpaceID_None &&
|
||||||
|
aName == nsGkAtoms::contenteditable;
|
||||||
|
PRInt32 change;
|
||||||
|
if (contentEditable) {
|
||||||
|
change = GetContentEditableValue() == eTrue ? -1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsresult rv = nsGenericElement::SetAttr(aNameSpaceID, aName, aPrefix, aValue,
|
||||||
|
aNotify);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
if (contentEditable) {
|
||||||
|
if (aValue.IsEmpty() || aValue.LowerCaseEqualsLiteral("true")) {
|
||||||
|
change += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
ChangeEditableState(change);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
nsresult
|
nsresult
|
||||||
nsGenericHTMLElement::UnsetAttr(PRInt32 aNameSpaceID, nsIAtom* aAttribute,
|
nsGenericHTMLElement::UnsetAttr(PRInt32 aNameSpaceID, nsIAtom* aAttribute,
|
||||||
PRBool aNotify)
|
PRBool aNotify)
|
||||||
{
|
{
|
||||||
// Check for event handlers
|
// Check for event handlers
|
||||||
if (aNameSpaceID == kNameSpaceID_None &&
|
if (aNameSpaceID == kNameSpaceID_None) {
|
||||||
nsContentUtils::IsEventAttributeName(aAttribute, EventNameType_HTML)) {
|
if (aAttribute == nsGkAtoms::contenteditable) {
|
||||||
|
ChangeEditableState(GetContentEditableValue() == eTrue ? -1 : 0);
|
||||||
|
}
|
||||||
|
else if (nsContentUtils::IsEventAttributeName(aAttribute,
|
||||||
|
EventNameType_HTML)) {
|
||||||
nsCOMPtr<nsIEventListenerManager> manager;
|
nsCOMPtr<nsIEventListenerManager> manager;
|
||||||
GetListenerManager(PR_FALSE, getter_AddRefs(manager));
|
GetListenerManager(PR_FALSE, getter_AddRefs(manager));
|
||||||
|
|
||||||
@ -1412,6 +1472,7 @@ nsGenericHTMLElement::UnsetAttr(PRInt32 aNameSpaceID, nsIAtom* aAttribute,
|
|||||||
manager->RemoveScriptEventListener(aAttribute);
|
manager->RemoveScriptEventListener(aAttribute);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return nsGenericElement::UnsetAttr(aNameSpaceID, aAttribute, aNotify);
|
return nsGenericElement::UnsetAttr(aNameSpaceID, aAttribute, aNotify);
|
||||||
}
|
}
|
||||||
@ -1580,6 +1641,11 @@ nsGenericHTMLElement::ParseAttribute(PRInt32 aNamespaceID,
|
|||||||
aResult.ParseAtom(aValue);
|
aResult.ParseAtom(aValue);
|
||||||
return PR_TRUE;
|
return PR_TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (aAttribute == nsGkAtoms::contenteditable) {
|
||||||
|
aResult.ParseAtom(aValue);
|
||||||
|
return PR_TRUE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return nsGenericElement::ParseAttribute(aNamespaceID, aAttribute, aValue,
|
return nsGenericElement::ParseAttribute(aNamespaceID, aAttribute, aValue,
|
||||||
@ -2021,6 +2087,24 @@ void
|
|||||||
nsGenericHTMLElement::MapCommonAttributesInto(const nsMappedAttributes* aAttributes,
|
nsGenericHTMLElement::MapCommonAttributesInto(const nsMappedAttributes* aAttributes,
|
||||||
nsRuleData* aData)
|
nsRuleData* aData)
|
||||||
{
|
{
|
||||||
|
if (aData->mSID == eStyleStruct_UserInterface) {
|
||||||
|
nsRuleDataUserInterface *ui = aData->mUserInterfaceData;
|
||||||
|
if (ui->mUserModify.GetUnit() == eCSSUnit_Null) {
|
||||||
|
const nsAttrValue* value =
|
||||||
|
aAttributes->GetAttr(nsGkAtoms::contenteditable);
|
||||||
|
if (value) {
|
||||||
|
if (value->Equals(nsGkAtoms::_empty, eCaseMatters) ||
|
||||||
|
value->Equals(nsGkAtoms::_true, eIgnoreCase)) {
|
||||||
|
ui->mUserModify.SetIntValue(NS_STYLE_USER_MODIFY_READ_WRITE,
|
||||||
|
eCSSUnit_Enumerated);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
ui->mUserModify.SetIntValue(NS_STYLE_USER_MODIFY_READ_ONLY,
|
||||||
|
eCSSUnit_Enumerated);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
if (aData->mSID == eStyleStruct_Visibility) {
|
if (aData->mSID == eStyleStruct_Visibility) {
|
||||||
const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::lang);
|
const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::lang);
|
||||||
if (value && value->Type() == nsAttrValue::eString) {
|
if (value && value->Type() == nsAttrValue::eString) {
|
||||||
@ -2030,10 +2114,34 @@ nsGenericHTMLElement::MapCommonAttributesInto(const nsMappedAttributes* aAttribu
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
nsGenericHTMLFormElement::UpdateEditableFormControlState()
|
||||||
|
{
|
||||||
|
ContentEditableTristate value = GetContentEditableValue();
|
||||||
|
if (value != eInherit) {
|
||||||
|
SetEditableFlag(value);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsIContent *parent = GetParent();
|
||||||
|
PRBool editable = parent && parent->HasFlag(NODE_IS_EDITABLE);
|
||||||
|
|
||||||
|
if (!editable) {
|
||||||
|
// If not contentEditable we still need to check the readonly attribute.
|
||||||
|
PRBool roState;
|
||||||
|
GetBoolAttr(nsGkAtoms::readonly, &roState);
|
||||||
|
|
||||||
|
editable = !roState;
|
||||||
|
}
|
||||||
|
|
||||||
|
SetEditableFlag(editable);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* static */ const nsGenericHTMLElement::MappedAttributeEntry
|
/* static */ const nsGenericHTMLElement::MappedAttributeEntry
|
||||||
nsGenericHTMLElement::sCommonAttributeMap[] = {
|
nsGenericHTMLElement::sCommonAttributeMap[] = {
|
||||||
|
{ &nsGkAtoms::contenteditable },
|
||||||
{ &nsGkAtoms::lang },
|
{ &nsGkAtoms::lang },
|
||||||
{ nsnull }
|
{ nsnull }
|
||||||
};
|
};
|
||||||
@ -2528,6 +2636,47 @@ nsGenericHTMLElement::GetURIListAttr(nsIAtom* aAttr, nsAString& aResult)
|
|||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nsresult
|
||||||
|
nsGenericHTMLElement::GetContentEditable(nsAString& aContentEditable)
|
||||||
|
{
|
||||||
|
ContentEditableTristate value = GetContentEditableValue();
|
||||||
|
|
||||||
|
if (value == eTrue) {
|
||||||
|
aContentEditable.AssignLiteral("true");
|
||||||
|
}
|
||||||
|
else if (value == eFalse) {
|
||||||
|
aContentEditable.AssignLiteral("false");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
aContentEditable.AssignLiteral("inherit");
|
||||||
|
}
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsresult
|
||||||
|
nsGenericHTMLElement::SetContentEditable(const nsAString& aContentEditable)
|
||||||
|
{
|
||||||
|
nsString contentEditable;
|
||||||
|
ToLowerCase(aContentEditable, contentEditable);
|
||||||
|
|
||||||
|
if (contentEditable.EqualsLiteral("inherit")) {
|
||||||
|
UnsetAttr(kNameSpaceID_None, nsGkAtoms::contenteditable, PR_TRUE);
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!contentEditable.EqualsLiteral("true") &&
|
||||||
|
!contentEditable.EqualsLiteral("false")) {
|
||||||
|
return NS_ERROR_DOM_SYNTAX_ERR;
|
||||||
|
}
|
||||||
|
|
||||||
|
SetAttr(kNameSpaceID_None, nsGkAtoms::contenteditable, contentEditable,
|
||||||
|
PR_TRUE);
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------------
|
//----------------------------------------------------------------------
|
||||||
|
|
||||||
NS_IMPL_INT_ATTR_DEFAULT_VALUE(nsGenericHTMLFrameElement, TabIndex, tabindex, 0)
|
NS_IMPL_INT_ATTR_DEFAULT_VALUE(nsGenericHTMLFrameElement, TabIndex, tabindex, 0)
|
||||||
@ -3153,11 +3302,20 @@ nsGenericHTMLElement::IsFocusable(PRInt32 *aTabIndex)
|
|||||||
PRInt32 tabIndex = 0; // Default value for non HTML elements with -moz-user-focus
|
PRInt32 tabIndex = 0; // Default value for non HTML elements with -moz-user-focus
|
||||||
GetTabIndex(&tabIndex);
|
GetTabIndex(&tabIndex);
|
||||||
|
|
||||||
|
PRBool disabled;
|
||||||
|
if (IsEditableRoot()) {
|
||||||
|
disabled = PR_FALSE;
|
||||||
|
if (!HasAttr(kNameSpaceID_None, nsGkAtoms::tabindex)) {
|
||||||
|
tabIndex = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
// Just check for disabled attribute on all HTML elements
|
// Just check for disabled attribute on all HTML elements
|
||||||
PRBool disabled = HasAttr(kNameSpaceID_None, nsGkAtoms::disabled);
|
disabled = HasAttr(kNameSpaceID_None, nsGkAtoms::disabled);
|
||||||
if (disabled) {
|
if (disabled) {
|
||||||
tabIndex = -1;
|
tabIndex = -1;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (aTabIndex) {
|
if (aTabIndex) {
|
||||||
*aTabIndex = tabIndex;
|
*aTabIndex = tabIndex;
|
||||||
@ -3768,3 +3926,93 @@ nsGenericHTMLElement::RecompileScriptEventListeners()
|
|||||||
AddScriptEventListener(attr, value, PR_TRUE);
|
AddScriptEventListener(attr, value, PR_TRUE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PRBool
|
||||||
|
nsGenericHTMLElement::IsEditableRoot() const
|
||||||
|
{
|
||||||
|
nsIDocument *document = GetCurrentDoc();
|
||||||
|
if (!document) {
|
||||||
|
return PR_FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (document->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return this == document->GetRootContent();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return PR_FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsIContent *parent = GetParent();
|
||||||
|
|
||||||
|
return !parent || !parent->HasFlag(NODE_IS_EDITABLE);
|
||||||
|
}
|
||||||
|
|
||||||
|
nsIContent*
|
||||||
|
nsGenericHTMLElement::FindEditableRoot()
|
||||||
|
{
|
||||||
|
nsIDocument *document = GetCurrentDoc();
|
||||||
|
if (!document) {
|
||||||
|
return nsnull;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (document->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return document->GetRootContent();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return nsnull;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsIContent *parent, *content = this;
|
||||||
|
while ((parent = content->GetParent()) && parent->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
content = parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
return content;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
MakeContentDescendantsEditable(nsIContent *aContent, nsIDocument *aDocument)
|
||||||
|
{
|
||||||
|
PRInt32 stateBefore = aContent->IntrinsicState();
|
||||||
|
|
||||||
|
aContent->UpdateEditableState();
|
||||||
|
|
||||||
|
if (aDocument && stateBefore != aContent->IntrinsicState()) {
|
||||||
|
aDocument->ContentStatesChanged(aContent, nsnull,
|
||||||
|
NS_EVENT_STATE_MOZ_READONLY |
|
||||||
|
NS_EVENT_STATE_MOZ_READWRITE);
|
||||||
|
}
|
||||||
|
|
||||||
|
PRUint32 i, n = aContent->GetChildCount();
|
||||||
|
for (i = 0; i < n; ++i) {
|
||||||
|
nsIContent *child = aContent->GetChildAt(i);
|
||||||
|
if (!child->HasAttr(kNameSpaceID_None, nsGkAtoms::contenteditable)) {
|
||||||
|
MakeContentDescendantsEditable(child, aDocument);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
nsGenericHTMLElement::ChangeEditableState(PRInt32 aChange)
|
||||||
|
{
|
||||||
|
nsIDocument* document = GetCurrentDoc();
|
||||||
|
if (!document) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (aChange != 0) {
|
||||||
|
nsCOMPtr<nsIHTMLDocument> htmlDocument =
|
||||||
|
do_QueryInterface(document);
|
||||||
|
if (htmlDocument) {
|
||||||
|
htmlDocument->ChangeContentEditableCount(this, aChange);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (document->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
document = nsnull;
|
||||||
|
}
|
||||||
|
|
||||||
|
MakeContentDescendantsEditable(this, document);
|
||||||
|
}
|
||||||
|
|||||||
@ -44,6 +44,7 @@
|
|||||||
#include "nsIFormControl.h"
|
#include "nsIFormControl.h"
|
||||||
#include "nsIDOMNSHTMLFrameElement.h"
|
#include "nsIDOMNSHTMLFrameElement.h"
|
||||||
#include "nsFrameLoader.h"
|
#include "nsFrameLoader.h"
|
||||||
|
#include "nsGkAtoms.h"
|
||||||
|
|
||||||
class nsIDOMAttr;
|
class nsIDOMAttr;
|
||||||
class nsIDOMEventListener;
|
class nsIDOMEventListener;
|
||||||
@ -165,6 +166,8 @@ public:
|
|||||||
NS_IMETHOD SetTabIndex(PRInt32 aTabIndex);
|
NS_IMETHOD SetTabIndex(PRInt32 aTabIndex);
|
||||||
NS_IMETHOD GetSpellcheck(PRBool* aSpellcheck);
|
NS_IMETHOD GetSpellcheck(PRBool* aSpellcheck);
|
||||||
NS_IMETHOD SetSpellcheck(PRBool aSpellcheck);
|
NS_IMETHOD SetSpellcheck(PRBool aSpellcheck);
|
||||||
|
nsresult GetContentEditable(nsAString &aContentEditable);
|
||||||
|
nsresult SetContentEditable(const nsAString &aContentEditable);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the frame's offset information for offsetTop/Left/Width/Height.
|
* Get the frame's offset information for offsetTop/Left/Width/Height.
|
||||||
@ -196,6 +199,16 @@ public:
|
|||||||
virtual nsresult BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
virtual nsresult BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
||||||
nsIContent* aBindingParent,
|
nsIContent* aBindingParent,
|
||||||
PRBool aCompileEventHandlers);
|
PRBool aCompileEventHandlers);
|
||||||
|
virtual void UnbindFromTree(PRBool aDeep = PR_TRUE,
|
||||||
|
PRBool aNullParent = PR_TRUE);
|
||||||
|
nsresult SetAttr(PRInt32 aNameSpaceID, nsIAtom* aName,
|
||||||
|
const nsAString& aValue, PRBool aNotify)
|
||||||
|
{
|
||||||
|
return SetAttr(aNameSpaceID, aName, nsnull, aValue, aNotify);
|
||||||
|
}
|
||||||
|
virtual nsresult SetAttr(PRInt32 aNameSpaceID, nsIAtom* aName,
|
||||||
|
nsIAtom* aPrefix, const nsAString& aValue,
|
||||||
|
PRBool aNotify);
|
||||||
virtual nsresult UnsetAttr(PRInt32 aNameSpaceID, nsIAtom* aName,
|
virtual nsresult UnsetAttr(PRInt32 aNameSpaceID, nsIAtom* aName,
|
||||||
PRBool aNotify);
|
PRBool aNotify);
|
||||||
virtual PRBool IsNodeOfType(PRUint32 aFlags) const;
|
virtual PRBool IsNodeOfType(PRUint32 aFlags) const;
|
||||||
@ -225,6 +238,8 @@ public:
|
|||||||
}
|
}
|
||||||
virtual nsMapRuleToAttributesFunc GetAttributeMappingFunction() const;
|
virtual nsMapRuleToAttributesFunc GetAttributeMappingFunction() const;
|
||||||
|
|
||||||
|
virtual void UpdateEditableState();
|
||||||
|
|
||||||
virtual const nsAttrValue* GetClasses() const;
|
virtual const nsAttrValue* GetClasses() const;
|
||||||
virtual nsIAtom *GetIDAttributeName() const;
|
virtual nsIAtom *GetIDAttributeName() const;
|
||||||
virtual nsIAtom *GetClassAttributeName() const;
|
virtual nsIAtom *GetClassAttributeName() const;
|
||||||
@ -768,6 +783,50 @@ protected:
|
|||||||
* spellchecking.
|
* spellchecking.
|
||||||
*/
|
*/
|
||||||
static void SyncEditorsOnSubtree(nsIContent* content);
|
static void SyncEditorsOnSubtree(nsIContent* content);
|
||||||
|
|
||||||
|
enum ContentEditableTristate {
|
||||||
|
eInherit = -1,
|
||||||
|
eFalse = 0,
|
||||||
|
eTrue = 1
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns eTrue if the element has a contentEditable attribute and its value
|
||||||
|
* is "true" or an empty string. Returns eFalse if the element has a
|
||||||
|
* contentEditable attribute and its value is "false". Otherwise returns
|
||||||
|
* eInherit.
|
||||||
|
*/
|
||||||
|
NS_HIDDEN_(ContentEditableTristate) GetContentEditableValue() const
|
||||||
|
{
|
||||||
|
static const nsIContent::AttrValuesArray values[] =
|
||||||
|
{ &nsGkAtoms::_false, &nsGkAtoms::_true, &nsGkAtoms::_empty, nsnull };
|
||||||
|
|
||||||
|
PRInt32 value = FindAttrValueIn(kNameSpaceID_None,
|
||||||
|
nsGkAtoms::contenteditable, values,
|
||||||
|
eIgnoreCase);
|
||||||
|
|
||||||
|
return value > 0 ? eTrue : (value == 0 ? eFalse : eInherit);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
/**
|
||||||
|
* Returns whether this element is an editable root. An editable root is
|
||||||
|
* defined as an element that is editable and whose parent is either a
|
||||||
|
* non-editable element or an editable document (so if the whole document is
|
||||||
|
* editable, then there is only one editable root, namely the
|
||||||
|
* documentElement).
|
||||||
|
*/
|
||||||
|
PRBool IsEditableRoot() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the first node amongst this node and its ancestors that is an
|
||||||
|
* editable root.
|
||||||
|
*
|
||||||
|
* @see IsEditableRoot for a definition of an editable root.
|
||||||
|
*/
|
||||||
|
nsIContent* FindEditableRoot();
|
||||||
|
|
||||||
|
void ChangeEditableState(PRInt32 aChange);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -817,6 +876,7 @@ public:
|
|||||||
virtual nsresult UnsetAttr(PRInt32 aNameSpaceID, nsIAtom* aName,
|
virtual nsresult UnsetAttr(PRInt32 aNameSpaceID, nsIAtom* aName,
|
||||||
PRBool aNotify);
|
PRBool aNotify);
|
||||||
virtual PRUint32 GetDesiredIMEState();
|
virtual PRUint32 GetDesiredIMEState();
|
||||||
|
virtual PRInt32 IntrinsicState() const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/**
|
/**
|
||||||
@ -838,7 +898,7 @@ protected:
|
|||||||
*/
|
*/
|
||||||
PRBool CanBeDisabled() const;
|
PRBool CanBeDisabled() const;
|
||||||
|
|
||||||
virtual PRInt32 IntrinsicState() const;
|
void UpdateEditableFormControlState();
|
||||||
|
|
||||||
void SetFocusAndScrollIntoView(nsPresContext* aPresContext);
|
void SetFocusAndScrollIntoView(nsPresContext* aPresContext);
|
||||||
|
|
||||||
|
|||||||
@ -238,6 +238,11 @@ public:
|
|||||||
|
|
||||||
virtual nsresult Clone(nsINodeInfo *aNodeInfo, nsINode **aResult) const;
|
virtual nsresult Clone(nsINodeInfo *aNodeInfo, nsINode **aResult) const;
|
||||||
|
|
||||||
|
virtual void UpdateEditableState()
|
||||||
|
{
|
||||||
|
return UpdateEditableFormControlState();
|
||||||
|
}
|
||||||
|
|
||||||
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED_NO_UNLINK(nsHTMLInputElement,
|
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED_NO_UNLINK(nsHTMLInputElement,
|
||||||
nsGenericHTMLFormElement)
|
nsGenericHTMLFormElement)
|
||||||
|
|
||||||
@ -601,6 +606,21 @@ nsHTMLInputElement::AfterSetAttr(PRInt32 aNameSpaceID, nsIAtom* aName,
|
|||||||
NS_EVENT_STATE_LOADING);
|
NS_EVENT_STATE_LOADING);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If readonly is changed for text and password we need to handle
|
||||||
|
// :read-only / :read-write
|
||||||
|
if (aNotify && aName == nsGkAtoms::readonly &&
|
||||||
|
(mType == NS_FORM_INPUT_TEXT || mType == NS_FORM_INPUT_PASSWORD)) {
|
||||||
|
UpdateEditableState();
|
||||||
|
|
||||||
|
nsIDocument* document = GetCurrentDoc();
|
||||||
|
if (document) {
|
||||||
|
mozAutoDocUpdate upd(document, UPDATE_CONTENT_STATE, PR_TRUE);
|
||||||
|
document->ContentStatesChanged(this, nsnull,
|
||||||
|
NS_EVENT_STATE_MOZ_READONLY |
|
||||||
|
NS_EVENT_STATE_MOZ_READWRITE);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return nsGenericHTMLFormElement::AfterSetAttr(aNameSpaceID, aName,
|
return nsGenericHTMLFormElement::AfterSetAttr(aNameSpaceID, aName,
|
||||||
|
|||||||
@ -165,6 +165,11 @@ public:
|
|||||||
nsIContent* aChild,
|
nsIContent* aChild,
|
||||||
PRInt32 aIndexInContainer);
|
PRInt32 aIndexInContainer);
|
||||||
|
|
||||||
|
virtual void UpdateEditableState()
|
||||||
|
{
|
||||||
|
return UpdateEditableFormControlState();
|
||||||
|
}
|
||||||
|
|
||||||
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(nsHTMLTextAreaElement,
|
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(nsHTMLTextAreaElement,
|
||||||
nsGenericHTMLFormElement)
|
nsGenericHTMLFormElement)
|
||||||
|
|
||||||
@ -202,6 +207,9 @@ protected:
|
|||||||
* parent; we should only respond to the change if aContent is non-anonymous.
|
* parent; we should only respond to the change if aContent is non-anonymous.
|
||||||
*/
|
*/
|
||||||
void ContentChanged(nsIContent* aContent);
|
void ContentChanged(nsIContent* aContent);
|
||||||
|
|
||||||
|
virtual nsresult AfterSetAttr(PRInt32 aNamespaceID, nsIAtom *aName,
|
||||||
|
const nsAString* aValue, PRBool aNotify);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -990,3 +998,23 @@ nsHTMLTextAreaElement::ContentChanged(nsIContent* aContent)
|
|||||||
Reset();
|
Reset();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nsresult
|
||||||
|
nsHTMLTextAreaElement::AfterSetAttr(PRInt32 aNameSpaceID, nsIAtom* aName,
|
||||||
|
const nsAString* aValue, PRBool aNotify)
|
||||||
|
{
|
||||||
|
if (aNotify && aNameSpaceID == kNameSpaceID_None &&
|
||||||
|
aName == nsGkAtoms::readonly) {
|
||||||
|
UpdateEditableState();
|
||||||
|
|
||||||
|
nsIDocument* document = GetCurrentDoc();
|
||||||
|
if (document) {
|
||||||
|
mozAutoDocUpdate upd(document, UPDATE_CONTENT_STATE, PR_TRUE);
|
||||||
|
document->ContentStatesChanged(this, nsnull,
|
||||||
|
NS_EVENT_STATE_MOZ_READONLY |
|
||||||
|
NS_EVENT_STATE_MOZ_READWRITE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nsGenericHTMLFormElement::AfterSetAttr(aNameSpaceID, aName, aValue,
|
||||||
|
aNotify);
|
||||||
|
}
|
||||||
|
|||||||
@ -76,6 +76,7 @@ REQUIRES = xpcom \
|
|||||||
composer \
|
composer \
|
||||||
editor \
|
editor \
|
||||||
plugin \
|
plugin \
|
||||||
|
txtsvc \
|
||||||
$(NULL)
|
$(NULL)
|
||||||
|
|
||||||
CPPSRCS = \
|
CPPSRCS = \
|
||||||
|
|||||||
@ -125,6 +125,7 @@
|
|||||||
#include "nsIMutableArray.h"
|
#include "nsIMutableArray.h"
|
||||||
#include "nsArrayUtils.h"
|
#include "nsArrayUtils.h"
|
||||||
#include "nsIEffectiveTLDService.h"
|
#include "nsIEffectiveTLDService.h"
|
||||||
|
#include "nsIEventStateManager.h"
|
||||||
|
|
||||||
#include "nsIPrompt.h"
|
#include "nsIPrompt.h"
|
||||||
//AHMED 12-2
|
//AHMED 12-2
|
||||||
@ -133,6 +134,11 @@
|
|||||||
#include "nsIEditingSession.h"
|
#include "nsIEditingSession.h"
|
||||||
#include "nsIEditor.h"
|
#include "nsIEditor.h"
|
||||||
#include "nsNodeInfoManager.h"
|
#include "nsNodeInfoManager.h"
|
||||||
|
#include "nsIEditor.h"
|
||||||
|
#include "nsIEditorDocShell.h"
|
||||||
|
#include "nsIEditorStyleSheets.h"
|
||||||
|
#include "nsIInlineSpellChecker.h"
|
||||||
|
#include "nsRange.h"
|
||||||
|
|
||||||
#define NS_MAX_DOCUMENT_WRITE_DEPTH 20
|
#define NS_MAX_DOCUMENT_WRITE_DEPTH 20
|
||||||
|
|
||||||
@ -1204,8 +1210,13 @@ nsHTMLDocument::EndLoad()
|
|||||||
mWriteState == eDocumentClosed, "EndLoad called early");
|
mWriteState == eDocumentClosed, "EndLoad called early");
|
||||||
mWriteState = eNotWriting;
|
mWriteState = eNotWriting;
|
||||||
|
|
||||||
|
PRBool turnOnEditing =
|
||||||
|
mParser && (HasFlag(NODE_IS_EDITABLE) || mContentEditableCount > 0);
|
||||||
// Note: nsDocument::EndLoad nulls out mParser.
|
// Note: nsDocument::EndLoad nulls out mParser.
|
||||||
nsDocument::EndLoad();
|
nsDocument::EndLoad();
|
||||||
|
if (turnOnEditing) {
|
||||||
|
EditingStateChanged();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
NS_IMETHODIMP
|
NS_IMETHODIMP
|
||||||
@ -2207,14 +2218,14 @@ nsHTMLDocument::OpenCommon(const nsACString& aContentType, PRBool aReplace)
|
|||||||
mRootContent = root;
|
mRootContent = root;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mEditingIsOn) {
|
if (IsEditingOn()) {
|
||||||
// Reset() blows away all event listeners in the document, and our
|
// Reset() blows away all event listeners in the document, and our
|
||||||
// editor relies heavily on those. Midas is turned on, to make it
|
// editor relies heavily on those. Midas is turned on, to make it
|
||||||
// work, re-initialize it to give it a chance to add its event
|
// work, re-initialize it to give it a chance to add its event
|
||||||
// listeners again.
|
// listeners again.
|
||||||
|
|
||||||
SetDesignMode(NS_LITERAL_STRING("off"));
|
TurnEditingOff();
|
||||||
SetDesignMode(NS_LITERAL_STRING("on"));
|
EditingStateChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Zap the old title -- otherwise it would hang around until document.close()
|
// Zap the old title -- otherwise it would hang around until document.close()
|
||||||
@ -3717,7 +3728,7 @@ nsHTMLDocument::GenerateParserKey(void)
|
|||||||
NS_IMETHODIMP
|
NS_IMETHODIMP
|
||||||
nsHTMLDocument::GetDesignMode(nsAString & aDesignMode)
|
nsHTMLDocument::GetDesignMode(nsAString & aDesignMode)
|
||||||
{
|
{
|
||||||
if (mEditingIsOn) {
|
if (HasFlag(NODE_IS_EDITABLE)) {
|
||||||
aDesignMode.AssignLiteral("on");
|
aDesignMode.AssignLiteral("on");
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@ -3726,9 +3737,152 @@ nsHTMLDocument::GetDesignMode(nsAString & aDesignMode)
|
|||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
NS_IMETHODIMP
|
nsresult
|
||||||
nsHTMLDocument::SetDesignMode(const nsAString & aDesignMode)
|
nsHTMLDocument::ChangeContentEditableCount(nsIContent *aElement,
|
||||||
|
PRInt32 aChange)
|
||||||
{
|
{
|
||||||
|
NS_ASSERTION(mContentEditableCount + aChange >= 0,
|
||||||
|
"Trying to decrement too much.");
|
||||||
|
|
||||||
|
mContentEditableCount += aChange;
|
||||||
|
|
||||||
|
if (mParser) {
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
EditingState oldState = mEditingState;
|
||||||
|
|
||||||
|
nsresult rv = EditingStateChanged();
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
if (oldState == mEditingState && mEditingState == eContentEditable) {
|
||||||
|
// We just changed the contentEditable state of a node, we need to reset
|
||||||
|
// the spellchecking state of that node.
|
||||||
|
nsCOMPtr<nsIDOMNode> node = do_QueryInterface(aElement);
|
||||||
|
if (node) {
|
||||||
|
nsPIDOMWindow *window = GetWindow();
|
||||||
|
if (!window)
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
nsIDocShell *docshell = window->GetDocShell();
|
||||||
|
if (!docshell)
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
nsCOMPtr<nsIEditorDocShell> editorDocShell =
|
||||||
|
do_QueryInterface(docshell, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIEditor> editor;
|
||||||
|
rv = editorDocShell->GetEditor(getter_AddRefs(editor));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIDOMRange> range;
|
||||||
|
rv = NS_NewRange(getter_AddRefs(range));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
rv = range->SelectNode(node);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIInlineSpellChecker> spellChecker;
|
||||||
|
rv = editor->GetInlineSpellChecker(PR_FALSE,
|
||||||
|
getter_AddRefs(spellChecker));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
if (spellChecker) {
|
||||||
|
rv = spellChecker->SpellCheckRange(range);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
NotifyEditableStateChange(nsINode *aNode, nsIDocument *aDocument,
|
||||||
|
PRBool aEditable)
|
||||||
|
{
|
||||||
|
PRUint32 i, n = aNode->GetChildCount();
|
||||||
|
for (i = 0; i < n; ++i) {
|
||||||
|
nsIContent *child = aNode->GetChildAt(i);
|
||||||
|
if (child->HasFlag(NODE_IS_EDITABLE) != aEditable) {
|
||||||
|
aDocument->ContentStatesChanged(child, nsnull,
|
||||||
|
NS_EVENT_STATE_MOZ_READONLY |
|
||||||
|
NS_EVENT_STATE_MOZ_READWRITE);
|
||||||
|
}
|
||||||
|
NotifyEditableStateChange(child, aDocument, aEditable);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
nsresult
|
||||||
|
nsHTMLDocument::TurnEditingOff()
|
||||||
|
{
|
||||||
|
NS_ASSERTION(mEditingState != eOff, "Editing is already off.");
|
||||||
|
|
||||||
|
nsPIDOMWindow *window = GetWindow();
|
||||||
|
if (!window)
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
nsIDocShell *docshell = window->GetDocShell();
|
||||||
|
if (!docshell)
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
nsresult rv;
|
||||||
|
nsCOMPtr<nsIEditorDocShell> editorDocShell =
|
||||||
|
do_QueryInterface(docshell, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIEditor> editor;
|
||||||
|
rv = editorDocShell->GetEditor(getter_AddRefs(editor));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIEditingSession> editSession = do_GetInterface(docshell, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
// turn editing off
|
||||||
|
rv = editSession->TearDownEditorOnWindow(window, PR_TRUE);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIEditorStyleSheets> editorss = do_QueryInterface(editor, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
if (!HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
editorss->RemoveOverrideStyleSheet(NS_LITERAL_STRING("resource:/res/contenteditable.css"));
|
||||||
|
editorss->RemoveOverrideStyleSheet(NS_LITERAL_STRING("resource:/res/designmode.css"));
|
||||||
|
|
||||||
|
rv = docshell->SetAllowJavascript(mScriptsEnabled);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
rv = docshell->SetAllowPlugins(mPluginsEnabled);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
}
|
||||||
|
|
||||||
|
mEditingState = eOff;
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsresult
|
||||||
|
nsHTMLDocument::EditingStateChanged()
|
||||||
|
{
|
||||||
|
if (mEditingState == eSettingUp) {
|
||||||
|
// XXX We shouldn't recurse.
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
PRBool designMode = HasFlag(NODE_IS_EDITABLE);
|
||||||
|
EditingState newState = designMode ? eDesignMode :
|
||||||
|
(mContentEditableCount > 0 ? eContentEditable : eOff);
|
||||||
|
if (mEditingState == newState) {
|
||||||
|
// No changes in editing mode.
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newState == eOff) {
|
||||||
|
// Editing is being turned off.
|
||||||
|
return TurnEditingOff();
|
||||||
|
}
|
||||||
|
|
||||||
// get editing session
|
// get editing session
|
||||||
nsPIDOMWindow *window = GetWindow();
|
nsPIDOMWindow *window = GetWindow();
|
||||||
if (!window)
|
if (!window)
|
||||||
@ -3738,6 +3892,128 @@ nsHTMLDocument::SetDesignMode(const nsAString & aDesignMode)
|
|||||||
if (!docshell)
|
if (!docshell)
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
nsresult rv;
|
||||||
|
nsCOMPtr<nsIEditingSession> editSession = do_GetInterface(docshell, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
PRBool makeWindowEditable = (mEditingState == eOff);
|
||||||
|
if (makeWindowEditable) {
|
||||||
|
// Editing is being turned on (through designMode or contentEditable)
|
||||||
|
// Turn on editor.
|
||||||
|
// XXX This can cause flushing which can change the editing state, so make
|
||||||
|
// sure to avoid recursing.
|
||||||
|
EditingState oldState = mEditingState;
|
||||||
|
mEditingState = eSettingUp;
|
||||||
|
|
||||||
|
rv = editSession->MakeWindowEditable(window, "html", PR_FALSE, PR_FALSE,
|
||||||
|
PR_TRUE);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
mEditingState = oldState;
|
||||||
|
}
|
||||||
|
|
||||||
|
// XXX Need to call TearDownEditorOnWindow for all failures.
|
||||||
|
nsCOMPtr<nsIEditorDocShell> editorDocShell =
|
||||||
|
do_QueryInterface(docshell, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIEditor> editor;
|
||||||
|
rv = editorDocShell->GetEditor(getter_AddRefs(editor));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIEditorStyleSheets> editorss = do_QueryInterface(editor, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
editorss->AddOverrideStyleSheet(NS_LITERAL_STRING("resource:/res/contenteditable.css"));
|
||||||
|
|
||||||
|
// Should we update the editable state of all the nodes in the document? We
|
||||||
|
// need to do this when the designMode value changes, as that overrides
|
||||||
|
// specific states on the elements.
|
||||||
|
PRBool updateState;
|
||||||
|
|
||||||
|
PRBool spellRecheckAll = PR_FALSE;
|
||||||
|
if (designMode) {
|
||||||
|
// designMode is being turned on (overrides contentEditable).
|
||||||
|
editorss->AddOverrideStyleSheet(NS_LITERAL_STRING("resource:/res/designmode.css"));
|
||||||
|
|
||||||
|
// Store scripting and plugins state.
|
||||||
|
PRBool tmp;
|
||||||
|
rv = docshell->GetAllowJavascript(&tmp);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
mScriptsEnabled = tmp;
|
||||||
|
|
||||||
|
rv = docshell->GetAllowPlugins(&tmp);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
mPluginsEnabled = tmp;
|
||||||
|
|
||||||
|
updateState = PR_TRUE;
|
||||||
|
spellRecheckAll = mEditingState == eContentEditable;
|
||||||
|
}
|
||||||
|
else if (mEditingState == eDesignMode) {
|
||||||
|
// designMode is being turned off (contentEditable is still on).
|
||||||
|
editorss->RemoveOverrideStyleSheet(NS_LITERAL_STRING("resource:/res/designmode.css"));
|
||||||
|
|
||||||
|
rv = docshell->SetAllowJavascript(mScriptsEnabled);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
rv = docshell->SetAllowPlugins(mPluginsEnabled);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
updateState = PR_TRUE;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// contentEditable is being turned on (and designMode is off).
|
||||||
|
updateState = PR_FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
mEditingState = newState;
|
||||||
|
|
||||||
|
if (makeWindowEditable) {
|
||||||
|
// Set the editor to not insert br's on return when in p
|
||||||
|
// elements by default.
|
||||||
|
// XXX Do we only want to do this for designMode?
|
||||||
|
PRBool unused;
|
||||||
|
rv = ExecCommand(NS_LITERAL_STRING("insertBrOnReturn"), PR_FALSE,
|
||||||
|
NS_LITERAL_STRING("false"), &unused);
|
||||||
|
|
||||||
|
if (NS_FAILED(rv)) {
|
||||||
|
// Editor setup failed. Editing is not on after all.
|
||||||
|
// XXX Should we reset the editable flag on nodes?
|
||||||
|
editSession->TearDownEditorOnWindow(window, PR_TRUE);
|
||||||
|
mEditingState = eOff;
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (updateState) {
|
||||||
|
mozAutoDocUpdate upd(this, UPDATE_CONTENT_STATE, PR_TRUE);
|
||||||
|
NotifyEditableStateChange(this, this, !designMode);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resync the editor's spellcheck state.
|
||||||
|
if (spellRecheckAll) {
|
||||||
|
nsCOMPtr<nsISelectionController> selcon;
|
||||||
|
nsresult rv = editor->GetSelectionController(getter_AddRefs(selcon));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsISelection> spellCheckSelection;
|
||||||
|
rv = selcon->GetSelection(nsISelectionController::SELECTION_SPELLCHECK,
|
||||||
|
getter_AddRefs(spellCheckSelection));
|
||||||
|
if (NS_SUCCEEDED(rv)) {
|
||||||
|
spellCheckSelection->RemoveAllRanges();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
editor->SyncRealTimeSpell();
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
NS_IMETHODIMP
|
||||||
|
nsHTMLDocument::SetDesignMode(const nsAString & aDesignMode)
|
||||||
|
{
|
||||||
nsresult rv = NS_OK;
|
nsresult rv = NS_OK;
|
||||||
|
|
||||||
if (!nsContentUtils::IsCallerTrustedForWrite()) {
|
if (!nsContentUtils::IsCallerTrustedForWrite()) {
|
||||||
@ -3751,53 +4027,14 @@ nsHTMLDocument::SetDesignMode(const nsAString & aDesignMode)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
nsCOMPtr<nsIEditingSession> editSession = do_GetInterface(docshell);
|
PRBool editableMode = HasFlag(NODE_IS_EDITABLE);
|
||||||
if (!editSession)
|
if (aDesignMode.LowerCaseEqualsASCII(editableMode ? "off" : "on")) {
|
||||||
return NS_ERROR_FAILURE;
|
SetEditableFlag(!editableMode);
|
||||||
|
|
||||||
if (aDesignMode.LowerCaseEqualsLiteral("on") && !mEditingIsOn) {
|
return EditingStateChanged();
|
||||||
rv = editSession->MakeWindowEditable(window, "html", PR_FALSE);
|
|
||||||
|
|
||||||
if (NS_SUCCEEDED(rv)) {
|
|
||||||
// now that we've successfully created the editor, we can
|
|
||||||
// reset our flag
|
|
||||||
mEditingIsOn = PR_TRUE;
|
|
||||||
|
|
||||||
// Set the editor to not insert br's on return when in p
|
|
||||||
// elements by default.
|
|
||||||
PRBool unused;
|
|
||||||
rv = ExecCommand(NS_LITERAL_STRING("insertBrOnReturn"), PR_FALSE,
|
|
||||||
NS_LITERAL_STRING("false"), &unused);
|
|
||||||
|
|
||||||
if (NS_FAILED(rv)) {
|
|
||||||
// Editor setup failed. Editing is is not on after all.
|
|
||||||
|
|
||||||
editSession->TearDownEditorOnWindow(window);
|
|
||||||
|
|
||||||
mEditingIsOn = PR_FALSE;
|
|
||||||
} else {
|
|
||||||
// Resync the editor's spellcheck state, since when the editor was
|
|
||||||
// created it asked us whether designMode was on, and we told it no.
|
|
||||||
// Note that reporting "yes" (by setting mEditingIsOn true before
|
|
||||||
// calling MakeWindowEditable()) exposed several crash bugs (see bugs
|
|
||||||
// 348497, 348981).
|
|
||||||
nsCOMPtr<nsIEditor> editor;
|
|
||||||
rv = editSession->GetEditorForWindow(window, getter_AddRefs(editor));
|
|
||||||
if (NS_SUCCEEDED(rv)) {
|
|
||||||
editor->SyncRealTimeSpell();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if (aDesignMode.LowerCaseEqualsLiteral("off") && mEditingIsOn) {
|
|
||||||
// turn editing off
|
|
||||||
rv = editSession->TearDownEditorOnWindow(window);
|
|
||||||
|
|
||||||
if (NS_SUCCEEDED(rv)) {
|
|
||||||
mEditingIsOn = PR_FALSE;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return rv;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
nsresult
|
nsresult
|
||||||
@ -3982,6 +4219,7 @@ nsHTMLDocument::ConvertToMidasInternalCommand(const nsAString & inCommandID,
|
|||||||
NS_ConvertUTF16toUTF8 convertedParam(inParam);
|
NS_ConvertUTF16toUTF8 convertedParam(inParam);
|
||||||
|
|
||||||
// check to see if we need to convert the parameter
|
// check to see if we need to convert the parameter
|
||||||
|
if (outCommandID.EqualsLiteral("cmd_paragraphState")) {
|
||||||
PRUint32 j;
|
PRUint32 j;
|
||||||
for (j = 0; j < MidasParamCount; ++j) {
|
for (j = 0; j < MidasParamCount; ++j) {
|
||||||
if (convertedParam.Equals(gMidasParamTable[j].incomingParamString,
|
if (convertedParam.Equals(gMidasParamTable[j].incomingParamString,
|
||||||
@ -3991,12 +4229,13 @@ nsHTMLDocument::ConvertToMidasInternalCommand(const nsAString & inCommandID,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// if we didn't convert the parameter, just
|
return j != MidasParamCount;
|
||||||
// pass through the parameter that was passed to us
|
}
|
||||||
if (j == MidasParamCount)
|
else {
|
||||||
outParam.Assign(convertedParam);
|
outParam.Assign(convertedParam);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
} // end else for useNewParam (do convert existing param)
|
} // end else for useNewParam (do convert existing param)
|
||||||
else {
|
else {
|
||||||
// reset results if the command is not found in our table
|
// reset results if the command is not found in our table
|
||||||
@ -4069,12 +4308,12 @@ nsHTMLDocument::ExecCommand(const nsAString & commandID,
|
|||||||
*_retval = PR_FALSE;
|
*_retval = PR_FALSE;
|
||||||
|
|
||||||
// if editing is not on, bail
|
// if editing is not on, bail
|
||||||
if (!mEditingIsOn)
|
if (!IsEditingOn())
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
// if they are requesting UI from us, let's fail since we have no UI
|
// if they are requesting UI from us, let's fail since we have no UI
|
||||||
if (doShowUI)
|
if (doShowUI)
|
||||||
return NS_ERROR_NOT_IMPLEMENTED;
|
return NS_OK;
|
||||||
|
|
||||||
nsresult rv = NS_OK;
|
nsresult rv = NS_OK;
|
||||||
|
|
||||||
@ -4105,7 +4344,7 @@ nsHTMLDocument::ExecCommand(const nsAString & commandID,
|
|||||||
PRBool isBool, boolVal;
|
PRBool isBool, boolVal;
|
||||||
if (!ConvertToMidasInternalCommand(commandID, value,
|
if (!ConvertToMidasInternalCommand(commandID, value,
|
||||||
cmdToDispatch, paramStr, isBool, boolVal))
|
cmdToDispatch, paramStr, isBool, boolVal))
|
||||||
return NS_ERROR_NOT_IMPLEMENTED;
|
return NS_OK;
|
||||||
|
|
||||||
if (!isBool && paramStr.IsEmpty()) {
|
if (!isBool && paramStr.IsEmpty()) {
|
||||||
rv = cmdMgr->DoCommand(cmdToDispatch.get(), nsnull, window);
|
rv = cmdMgr->DoCommand(cmdToDispatch.get(), nsnull, window);
|
||||||
@ -4144,7 +4383,7 @@ nsHTMLDocument::ExecCommandShowHelp(const nsAString & commandID,
|
|||||||
*_retval = PR_FALSE;
|
*_retval = PR_FALSE;
|
||||||
|
|
||||||
// if editing is not on, bail
|
// if editing is not on, bail
|
||||||
if (!mEditingIsOn)
|
if (!IsEditingOn())
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
return NS_ERROR_NOT_IMPLEMENTED;
|
return NS_ERROR_NOT_IMPLEMENTED;
|
||||||
@ -4159,7 +4398,7 @@ nsHTMLDocument::QueryCommandEnabled(const nsAString & commandID,
|
|||||||
*_retval = PR_FALSE;
|
*_retval = PR_FALSE;
|
||||||
|
|
||||||
// if editing is not on, bail
|
// if editing is not on, bail
|
||||||
if (!mEditingIsOn)
|
if (!IsEditingOn())
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
// get command manager and dispatch command to our window if it's acceptable
|
// get command manager and dispatch command to our window if it's acceptable
|
||||||
@ -4190,7 +4429,7 @@ nsHTMLDocument::QueryCommandIndeterm(const nsAString & commandID,
|
|||||||
*_retval = PR_FALSE;
|
*_retval = PR_FALSE;
|
||||||
|
|
||||||
// if editing is not on, bail
|
// if editing is not on, bail
|
||||||
if (!mEditingIsOn)
|
if (!IsEditingOn())
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
// get command manager and dispatch command to our window if it's acceptable
|
// get command manager and dispatch command to our window if it's acceptable
|
||||||
@ -4232,7 +4471,7 @@ nsHTMLDocument::QueryCommandState(const nsAString & commandID, PRBool *_retval)
|
|||||||
*_retval = PR_FALSE;
|
*_retval = PR_FALSE;
|
||||||
|
|
||||||
// if editing is not on, bail
|
// if editing is not on, bail
|
||||||
if (!mEditingIsOn)
|
if (!IsEditingOn())
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
// get command manager and dispatch command to our window if it's acceptable
|
// get command manager and dispatch command to our window if it's acceptable
|
||||||
@ -4294,7 +4533,7 @@ nsHTMLDocument::QueryCommandSupported(const nsAString & commandID,
|
|||||||
*_retval = PR_FALSE;
|
*_retval = PR_FALSE;
|
||||||
|
|
||||||
// if editing is not on, bail
|
// if editing is not on, bail
|
||||||
if (!mEditingIsOn)
|
if (!IsEditingOn())
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
return NS_ERROR_NOT_IMPLEMENTED;
|
return NS_ERROR_NOT_IMPLEMENTED;
|
||||||
@ -4308,7 +4547,7 @@ nsHTMLDocument::QueryCommandText(const nsAString & commandID,
|
|||||||
_retval.SetLength(0);
|
_retval.SetLength(0);
|
||||||
|
|
||||||
// if editing is not on, bail
|
// if editing is not on, bail
|
||||||
if (!mEditingIsOn)
|
if (!IsEditingOn())
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
return NS_ERROR_NOT_IMPLEMENTED;
|
return NS_ERROR_NOT_IMPLEMENTED;
|
||||||
@ -4322,7 +4561,7 @@ nsHTMLDocument::QueryCommandValue(const nsAString & commandID,
|
|||||||
_retval.SetLength(0);
|
_retval.SetLength(0);
|
||||||
|
|
||||||
// if editing is not on, bail
|
// if editing is not on, bail
|
||||||
if (!mEditingIsOn)
|
if (!IsEditingOn())
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
// get command manager and dispatch command to our window if it's acceptable
|
// get command manager and dispatch command to our window if it's acceptable
|
||||||
|
|||||||
@ -59,6 +59,8 @@
|
|||||||
|
|
||||||
#include "nsICommandManager.h"
|
#include "nsICommandManager.h"
|
||||||
|
|
||||||
|
class nsIEditor;
|
||||||
|
class nsIEditorDocShell;
|
||||||
class nsIParser;
|
class nsIParser;
|
||||||
class nsIURI;
|
class nsIURI;
|
||||||
class nsIMarkupDocumentViewer;
|
class nsIMarkupDocumentViewer;
|
||||||
@ -207,6 +209,13 @@ public:
|
|||||||
nsIContent** aResult);
|
nsIContent** aResult);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
nsresult ChangeContentEditableCount(nsIContent *aElement, PRInt32 aChange);
|
||||||
|
|
||||||
|
virtual PRBool IsEditingOn()
|
||||||
|
{
|
||||||
|
return mEditingState != eOff;
|
||||||
|
}
|
||||||
|
|
||||||
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED_NO_UNLINK(nsHTMLDocument, nsDocument)
|
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED_NO_UNLINK(nsHTMLDocument, nsDocument)
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@ -365,7 +374,20 @@ protected:
|
|||||||
PRBool& isBoolean,
|
PRBool& isBoolean,
|
||||||
PRBool& boolValue);
|
PRBool& boolValue);
|
||||||
nsCOMPtr<nsICommandManager> mMidasCommandManager;
|
nsCOMPtr<nsICommandManager> mMidasCommandManager;
|
||||||
PRBool mEditingIsOn;
|
|
||||||
|
nsresult TurnEditingOff();
|
||||||
|
nsresult EditingStateChanged();
|
||||||
|
|
||||||
|
PRUint32 mContentEditableCount;
|
||||||
|
enum EditingState {
|
||||||
|
eSettingUp = -1,
|
||||||
|
eOff = 0,
|
||||||
|
eDesignMode,
|
||||||
|
eContentEditable
|
||||||
|
};
|
||||||
|
EditingState mEditingState;
|
||||||
|
PRPackedBool mScriptsEnabled;
|
||||||
|
PRPackedBool mPluginsEnabled;
|
||||||
|
|
||||||
nsresult DoClipboardSecurityCheck(PRBool aPaste);
|
nsresult DoClipboardSecurityCheck(PRBool aPaste);
|
||||||
static jsval sCutCopyInternal_id;
|
static jsval sCutCopyInternal_id;
|
||||||
|
|||||||
@ -55,8 +55,8 @@ class nsIDOMHTMLBodyElement;
|
|||||||
class nsIScriptElement;
|
class nsIScriptElement;
|
||||||
|
|
||||||
#define NS_IHTMLDOCUMENT_IID \
|
#define NS_IHTMLDOCUMENT_IID \
|
||||||
{ 0xcfe72003, 0xcc90, 0x4624, \
|
{ 0xf6aa3582, 0x67c3, 0x4f42, \
|
||||||
{ 0xb4, 0x1b, 0xc3, 0x14, 0x1d, 0x31, 0x7a, 0x71 } }
|
{ 0xb6, 0xee, 0x89, 0x19, 0x24, 0x5c, 0x15, 0x89 } }
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -127,6 +127,23 @@ public:
|
|||||||
* the document that are of type nsIContent::eHTML_FORM_CONTROL).
|
* the document that are of type nsIContent::eHTML_FORM_CONTROL).
|
||||||
*/
|
*/
|
||||||
virtual nsContentList* GetFormControls() = 0;
|
virtual nsContentList* GetFormControls() = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Should be called when an element's editable changes as a result of
|
||||||
|
* changing its contentEditable attribute/property.
|
||||||
|
*
|
||||||
|
* @param aElement the element for which the contentEditable
|
||||||
|
* attribute/property was changed
|
||||||
|
* @param aChange +1 if the contentEditable attribute/property was changed to
|
||||||
|
* true, -1 if it was changed to false
|
||||||
|
*/
|
||||||
|
virtual nsresult ChangeContentEditableCount(nsIContent *aElement,
|
||||||
|
PRInt32 aChange) = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns whether the document is editable.
|
||||||
|
*/
|
||||||
|
virtual PRBool IsEditingOn() = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
NS_DEFINE_STATIC_IID_ACCESSOR(nsIHTMLDocument, NS_IHTMLDOCUMENT_IID)
|
NS_DEFINE_STATIC_IID_ACCESSOR(nsIHTMLDocument, NS_IHTMLDOCUMENT_IID)
|
||||||
|
|||||||
@ -1040,6 +1040,15 @@ nsXULElement::AfterSetAttr(PRInt32 aNamespaceID, nsIAtom* aName,
|
|||||||
HideWindowChrome(aValue && NS_LITERAL_STRING("true").Equals(*aValue));
|
HideWindowChrome(aValue && NS_LITERAL_STRING("true").Equals(*aValue));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// handle :read-only/:read-write
|
||||||
|
nsIDocument *document = GetCurrentDoc();
|
||||||
|
if (aName == nsGkAtoms::readonly && document) {
|
||||||
|
mozAutoDocUpdate upd(document, UPDATE_CONTENT_STATE, PR_TRUE);
|
||||||
|
document->ContentStatesChanged(this, nsnull,
|
||||||
|
NS_EVENT_STATE_MOZ_READONLY |
|
||||||
|
NS_EVENT_STATE_MOZ_READWRITE);
|
||||||
|
}
|
||||||
|
|
||||||
// XXX need to check if they're changing an event handler: if
|
// XXX need to check if they're changing an event handler: if
|
||||||
// so, then we need to unhook the old one. Or something.
|
// so, then we need to unhook the old one. Or something.
|
||||||
}
|
}
|
||||||
@ -2098,6 +2107,22 @@ nsXULElement::AddPopupListener(nsIAtom* aName)
|
|||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PRInt32
|
||||||
|
nsXULElement::IntrinsicState() const
|
||||||
|
{
|
||||||
|
PRInt32 state = nsGenericElement::IntrinsicState();
|
||||||
|
|
||||||
|
const nsIAtom* tag = Tag();
|
||||||
|
if (GetNameSpaceID() == kNameSpaceID_XUL &&
|
||||||
|
(tag == nsGkAtoms::textbox || tag == nsGkAtoms::textarea) &&
|
||||||
|
!HasAttr(kNameSpaceID_None, nsGkAtoms::readonly)) {
|
||||||
|
state |= NS_EVENT_STATE_MOZ_READWRITE;
|
||||||
|
state &= ~NS_EVENT_STATE_MOZ_READONLY;
|
||||||
|
}
|
||||||
|
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------------
|
//----------------------------------------------------------------------
|
||||||
|
|
||||||
nsGenericElement::nsAttrInfo
|
nsGenericElement::nsAttrInfo
|
||||||
|
|||||||
@ -558,6 +558,7 @@ public:
|
|||||||
NS_DECL_NSIDOMXULELEMENT
|
NS_DECL_NSIDOMXULELEMENT
|
||||||
|
|
||||||
virtual nsresult Clone(nsINodeInfo *aNodeInfo, nsINode **aResult) const;
|
virtual nsresult Clone(nsINodeInfo *aNodeInfo, nsINode **aResult) const;
|
||||||
|
virtual PRInt32 IntrinsicState() const;
|
||||||
|
|
||||||
nsresult GetStyle(nsIDOMCSSStyleDeclaration** aStyle);
|
nsresult GetStyle(nsIDOMCSSStyleDeclaration** aStyle);
|
||||||
|
|
||||||
|
|||||||
@ -73,7 +73,7 @@ nsDocShellEditorData::~nsDocShellEditorData()
|
|||||||
nsCOMPtr<nsIDOMWindow> domWindow = do_GetInterface(mDocShell);
|
nsCOMPtr<nsIDOMWindow> domWindow = do_GetInterface(mDocShell);
|
||||||
// This will eventually call nsDocShellEditorData::SetEditor(nsnull)
|
// This will eventually call nsDocShellEditorData::SetEditor(nsnull)
|
||||||
// which will call mEditorPreDestroy() and delete the editor
|
// which will call mEditorPreDestroy() and delete the editor
|
||||||
mEditingSession->TearDownEditorOnWindow(domWindow);
|
mEditingSession->TearDownEditorOnWindow(domWindow, PR_TRUE);
|
||||||
}
|
}
|
||||||
else if (mEditor) // Should never have this w/o nsEditingSession!
|
else if (mEditor) // Should never have this w/o nsEditingSession!
|
||||||
{
|
{
|
||||||
@ -104,6 +104,7 @@ nsDocShellEditorData::MakeEditable(PRBool inWaitForUriLoad /*, PRBool inEditable
|
|||||||
mEditor = nsnull;
|
mEditor = nsnull;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (inWaitForUriLoad)
|
||||||
mMakeEditable = PR_TRUE;
|
mMakeEditable = PR_TRUE;
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
@ -191,6 +192,8 @@ nsDocShellEditorData::SetEditor(nsIEditor *inEditor)
|
|||||||
}
|
}
|
||||||
|
|
||||||
mEditor = inEditor; // owning addref
|
mEditor = inEditor; // owning addref
|
||||||
|
if (!mEditor)
|
||||||
|
mMakeEditable = PR_FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
|
|||||||
@ -773,6 +773,10 @@ nsWebShell::OnLinkClick(nsIContent* aContent,
|
|||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (aContent->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
nsCOMPtr<nsIRunnable> ev =
|
nsCOMPtr<nsIRunnable> ev =
|
||||||
new OnLinkClickEvent(this, aContent, aURI, aTargetSpec,
|
new OnLinkClickEvent(this, aContent, aURI, aTargetSpec,
|
||||||
aPostDataStream, aHeadersDataStream);
|
aPostDataStream, aHeadersDataStream);
|
||||||
@ -800,6 +804,10 @@ nsWebShell::OnLinkClickSync(nsIContent *aContent,
|
|||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (aContent->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
// defer to an external protocol handler if necessary...
|
// defer to an external protocol handler if necessary...
|
||||||
nsCOMPtr<nsIExternalProtocolService> extProtService = do_GetService(NS_EXTERNALPROTOCOLSERVICE_CONTRACTID);
|
nsCOMPtr<nsIExternalProtocolService> extProtService = do_GetService(NS_EXTERNALPROTOCOLSERVICE_CONTRACTID);
|
||||||
@ -895,6 +903,10 @@ nsWebShell::OnOverLink(nsIContent* aContent,
|
|||||||
nsIURI* aURI,
|
nsIURI* aURI,
|
||||||
const PRUnichar* aTargetSpec)
|
const PRUnichar* aTargetSpec)
|
||||||
{
|
{
|
||||||
|
if (aContent->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
nsCOMPtr<nsIWebBrowserChrome2> browserChrome2 = do_GetInterface(mTreeOwner);
|
nsCOMPtr<nsIWebBrowserChrome2> browserChrome2 = do_GetInterface(mTreeOwner);
|
||||||
nsresult rv = NS_ERROR_FAILURE;
|
nsresult rv = NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
|||||||
@ -38,7 +38,7 @@
|
|||||||
|
|
||||||
#include "domstubs.idl"
|
#include "domstubs.idl"
|
||||||
|
|
||||||
[scriptable, uuid(b0a29b0a-ce2b-4cdf-b98a-4c7ed994d6e2)]
|
[scriptable, uuid(eac0a4ee-2e4f-403c-9b77-5cf32cfb42f7)]
|
||||||
interface nsIDOMNSHTMLElement : nsISupports
|
interface nsIDOMNSHTMLElement : nsISupports
|
||||||
{
|
{
|
||||||
readonly attribute long offsetTop;
|
readonly attribute long offsetTop;
|
||||||
@ -60,6 +60,8 @@ interface nsIDOMNSHTMLElement : nsISupports
|
|||||||
|
|
||||||
attribute long tabIndex;
|
attribute long tabIndex;
|
||||||
|
|
||||||
|
attribute DOMString contentEditable;
|
||||||
|
|
||||||
void blur();
|
void blur();
|
||||||
void focus();
|
void focus();
|
||||||
|
|
||||||
|
|||||||
@ -43,7 +43,7 @@
|
|||||||
|
|
||||||
interface nsIEditor;
|
interface nsIEditor;
|
||||||
|
|
||||||
[scriptable, uuid(d39fd2b4-3978-45d2-a4be-ba448171b61b)]
|
[scriptable, uuid(aee80d50-2065-4411-834d-0cadfb649a19)]
|
||||||
|
|
||||||
interface nsIEditingSession : nsISupports
|
interface nsIEditingSession : nsISupports
|
||||||
{
|
{
|
||||||
@ -68,8 +68,16 @@ interface nsIEditingSession : nsISupports
|
|||||||
* Make this window editable
|
* Make this window editable
|
||||||
* @param aWindow nsIDOMWindow, the window the embedder needs to make editable
|
* @param aWindow nsIDOMWindow, the window the embedder needs to make editable
|
||||||
* @param aEditorType string, "html" "htmlsimple" "text" "textsimple"
|
* @param aEditorType string, "html" "htmlsimple" "text" "textsimple"
|
||||||
|
* @param aMakeWholeDocumentEditable if PR_TRUE make the whole document in
|
||||||
|
* aWindow editable, otherwise it's the
|
||||||
|
* embedder who should make the document
|
||||||
|
* (or part of it) editable.
|
||||||
|
* @param aInteractive if PR_FALSE turn off scripting and plugins
|
||||||
*/
|
*/
|
||||||
void makeWindowEditable(in nsIDOMWindow window, in string aEditorType, in boolean doAfterUriLoad);
|
void makeWindowEditable(in nsIDOMWindow window, in string aEditorType,
|
||||||
|
in boolean doAfterUriLoad,
|
||||||
|
in boolean aMakeWholeDocumentEditable,
|
||||||
|
in boolean aInteractive);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test whether a specific window has had its editable flag set; it may have an editor
|
* Test whether a specific window has had its editable flag set; it may have an editor
|
||||||
@ -93,7 +101,7 @@ interface nsIEditingSession : nsISupports
|
|||||||
/**
|
/**
|
||||||
* Destroy editor and related support objects
|
* Destroy editor and related support objects
|
||||||
*/
|
*/
|
||||||
void tearDownEditorOnWindow(in nsIDOMWindow window);
|
void tearDownEditorOnWindow(in nsIDOMWindow window, in boolean aStopEditing);
|
||||||
|
|
||||||
void setEditorOnControllers(in nsIDOMWindow aWindow,
|
void setEditorOnControllers(in nsIDOMWindow aWindow,
|
||||||
in nsIEditor aEditor);
|
in nsIEditor aEditor);
|
||||||
|
|||||||
@ -95,6 +95,7 @@
|
|||||||
nsEditingSession::nsEditingSession()
|
nsEditingSession::nsEditingSession()
|
||||||
: mDoneSetup(PR_FALSE)
|
: mDoneSetup(PR_FALSE)
|
||||||
, mCanCreateEditor(PR_FALSE)
|
, mCanCreateEditor(PR_FALSE)
|
||||||
|
, mInteractive(PR_FALSE)
|
||||||
, mScriptsEnabled(PR_TRUE)
|
, mScriptsEnabled(PR_TRUE)
|
||||||
, mPluginsEnabled(PR_TRUE)
|
, mPluginsEnabled(PR_TRUE)
|
||||||
, mProgressListenerRegistered(PR_FALSE)
|
, mProgressListenerRegistered(PR_FALSE)
|
||||||
@ -128,14 +129,18 @@ NS_IMPL_ISUPPORTS3(nsEditingSession, nsIEditingSession, nsIWebProgressListener,
|
|||||||
|
|
||||||
aEditorType string, "html" "htmlsimple" "text" "textsimple"
|
aEditorType string, "html" "htmlsimple" "text" "textsimple"
|
||||||
void makeWindowEditable(in nsIDOMWindow aWindow, in string aEditorType,
|
void makeWindowEditable(in nsIDOMWindow aWindow, in string aEditorType,
|
||||||
in boolean aDoAfterUriLoad);
|
in boolean aDoAfterUriLoad,
|
||||||
|
in boolean aMakeWholeDocumentEditable,
|
||||||
|
in boolean aInteractive);
|
||||||
----------------------------------------------------------------------------*/
|
----------------------------------------------------------------------------*/
|
||||||
#define DEFAULT_EDITOR_TYPE "html"
|
#define DEFAULT_EDITOR_TYPE "html"
|
||||||
|
|
||||||
NS_IMETHODIMP
|
NS_IMETHODIMP
|
||||||
nsEditingSession::MakeWindowEditable(nsIDOMWindow *aWindow,
|
nsEditingSession::MakeWindowEditable(nsIDOMWindow *aWindow,
|
||||||
const char *aEditorType,
|
const char *aEditorType,
|
||||||
PRBool aDoAfterUriLoad)
|
PRBool aDoAfterUriLoad,
|
||||||
|
PRBool aMakeWholeDocumentEditable,
|
||||||
|
PRBool aInteractive)
|
||||||
{
|
{
|
||||||
mEditorType.Truncate();
|
mEditorType.Truncate();
|
||||||
mEditorFlags = 0;
|
mEditorFlags = 0;
|
||||||
@ -146,6 +151,20 @@ nsEditingSession::MakeWindowEditable(nsIDOMWindow *aWindow,
|
|||||||
if (!docShell) return NS_ERROR_FAILURE;
|
if (!docShell) return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
nsresult rv;
|
nsresult rv;
|
||||||
|
if (aMakeWholeDocumentEditable) {
|
||||||
|
nsCOMPtr<nsIDOMDocument> domDoc;
|
||||||
|
rv = aWindow->GetDocument(getter_AddRefs(domDoc));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIDocument> doc = do_QueryInterface(domDoc, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
doc->SetEditableFlag(PR_TRUE);
|
||||||
|
}
|
||||||
|
|
||||||
|
mInteractive = aInteractive;
|
||||||
|
|
||||||
|
if (!mInteractive) {
|
||||||
// Disable JavaScript in this document:
|
// Disable JavaScript in this document:
|
||||||
PRBool tmp;
|
PRBool tmp;
|
||||||
rv = docShell->GetAllowJavascript(&tmp);
|
rv = docShell->GetAllowJavascript(&tmp);
|
||||||
@ -164,9 +183,10 @@ nsEditingSession::MakeWindowEditable(nsIDOMWindow *aWindow,
|
|||||||
|
|
||||||
rv = docShell->SetAllowPlugins(PR_FALSE);
|
rv = docShell->SetAllowPlugins(PR_FALSE);
|
||||||
NS_ENSURE_SUCCESS(rv, rv);
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
}
|
||||||
|
|
||||||
// Always remove existing editor
|
// Always remove existing editor
|
||||||
TearDownEditorOnWindow(aWindow);
|
TearDownEditorOnWindow(aWindow, PR_FALSE);
|
||||||
|
|
||||||
// Tells embedder that startup is in progress
|
// Tells embedder that startup is in progress
|
||||||
mEditorStatus = eEditorCreationInProgress;
|
mEditorStatus = eEditorCreationInProgress;
|
||||||
@ -215,7 +235,7 @@ nsEditingSession::MakeWindowEditable(nsIDOMWindow *aWindow,
|
|||||||
// Since this is used only when editing an existing page,
|
// Since this is used only when editing an existing page,
|
||||||
// it IS ok to destroy current editor
|
// it IS ok to destroy current editor
|
||||||
if (NS_FAILED(rv))
|
if (NS_FAILED(rv))
|
||||||
TearDownEditorOnWindow(aWindow);
|
TearDownEditorOnWindow(aWindow, PR_FALSE);
|
||||||
}
|
}
|
||||||
return rv;
|
return rv;
|
||||||
}
|
}
|
||||||
@ -361,6 +381,10 @@ nsEditingSession::SetupEditorOnWindow(nsIDOMWindow *aWindow)
|
|||||||
needHTMLController = PR_TRUE;
|
needHTMLController = PR_TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (mInteractive) {
|
||||||
|
mEditorFlags |= nsIPlaintextEditor::eEditorAllowInteraction;
|
||||||
|
}
|
||||||
|
|
||||||
// make the UI state maintainer
|
// make the UI state maintainer
|
||||||
nsComposerCommandsUpdater *stateMaintainer;
|
nsComposerCommandsUpdater *stateMaintainer;
|
||||||
NS_NEWXPCOM(stateMaintainer, nsComposerCommandsUpdater);
|
NS_NEWXPCOM(stateMaintainer, nsComposerCommandsUpdater);
|
||||||
@ -390,6 +414,7 @@ nsEditingSession::SetupEditorOnWindow(nsIDOMWindow *aWindow)
|
|||||||
nsIDocShell *docShell = GetDocShellFromWindow(aWindow);
|
nsIDocShell *docShell = GetDocShellFromWindow(aWindow);
|
||||||
if (!docShell) return NS_ERROR_FAILURE;
|
if (!docShell) return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
if (!mInteractive) {
|
||||||
// Disable animation of images in this document:
|
// Disable animation of images in this document:
|
||||||
nsCOMPtr<nsIDOMWindowUtils> utils(do_GetInterface(aWindow));
|
nsCOMPtr<nsIDOMWindowUtils> utils(do_GetInterface(aWindow));
|
||||||
if (!utils) return NS_ERROR_FAILURE;
|
if (!utils) return NS_ERROR_FAILURE;
|
||||||
@ -397,6 +422,7 @@ nsEditingSession::SetupEditorOnWindow(nsIDOMWindow *aWindow)
|
|||||||
rv = utils->GetImageAnimationMode(&mImageAnimationMode);
|
rv = utils->GetImageAnimationMode(&mImageAnimationMode);
|
||||||
if (NS_FAILED(rv)) return rv;
|
if (NS_FAILED(rv)) return rv;
|
||||||
utils->SetImageAnimationMode(imgIContainer::kDontAnimMode);
|
utils->SetImageAnimationMode(imgIContainer::kDontAnimMode);
|
||||||
|
}
|
||||||
|
|
||||||
// create and set editor
|
// create and set editor
|
||||||
nsCOMPtr<nsIEditorDocShell> editorDocShell = do_QueryInterface(docShell, &rv);
|
nsCOMPtr<nsIEditorDocShell> editorDocShell = do_QueryInterface(docShell, &rv);
|
||||||
@ -480,10 +506,12 @@ nsEditingSession::SetupEditorOnWindow(nsIDOMWindow *aWindow)
|
|||||||
|
|
||||||
TearDownEditorOnWindow
|
TearDownEditorOnWindow
|
||||||
|
|
||||||
void tearDownEditorOnWindow (in nsIDOMWindow aWindow);
|
void tearDownEditorOnWindow (in nsIDOMWindow aWindow,
|
||||||
|
in boolean aStopEditing);
|
||||||
----------------------------------------------------------------------------*/
|
----------------------------------------------------------------------------*/
|
||||||
NS_IMETHODIMP
|
NS_IMETHODIMP
|
||||||
nsEditingSession::TearDownEditorOnWindow(nsIDOMWindow *aWindow)
|
nsEditingSession::TearDownEditorOnWindow(nsIDOMWindow *aWindow,
|
||||||
|
PRBool aStopEditing)
|
||||||
{
|
{
|
||||||
if (!mDoneSetup)
|
if (!mDoneSetup)
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
@ -501,23 +529,7 @@ nsEditingSession::TearDownEditorOnWindow(nsIDOMWindow *aWindow)
|
|||||||
|
|
||||||
mDoneSetup = PR_FALSE;
|
mDoneSetup = PR_FALSE;
|
||||||
|
|
||||||
nsCOMPtr<nsIDOMDocument> dom_doc;
|
if (aStopEditing) {
|
||||||
aWindow->GetDocument(getter_AddRefs(dom_doc));
|
|
||||||
|
|
||||||
nsCOMPtr<nsIDOMNSHTMLDocument> html_doc(do_QueryInterface(dom_doc));
|
|
||||||
PRBool isMidas = PR_FALSE;
|
|
||||||
|
|
||||||
if (html_doc) {
|
|
||||||
nsAutoString designMode;
|
|
||||||
html_doc->GetDesignMode(designMode);
|
|
||||||
|
|
||||||
isMidas = designMode.EqualsLiteral("on");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isMidas) {
|
|
||||||
// We're tearing down a midas editor, unregister callbacks since
|
|
||||||
// we're all done editing here.
|
|
||||||
|
|
||||||
nsCOMPtr<nsIWebProgress> webProgress = do_GetInterface(docShell);
|
nsCOMPtr<nsIWebProgress> webProgress = do_GetInterface(docShell);
|
||||||
if (webProgress) {
|
if (webProgress) {
|
||||||
webProgress->RemoveProgressListener(this);
|
webProgress->RemoveProgressListener(this);
|
||||||
@ -622,7 +634,7 @@ nsEditingSession::TearDownEditorOnWindow(nsIDOMWindow *aWindow)
|
|||||||
mHTMLCommandControllerId = 0;
|
mHTMLCommandControllerId = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isMidas) {
|
if (aStopEditing && !mInteractive) {
|
||||||
// Make things the way they were before we started editing.
|
// Make things the way they were before we started editing.
|
||||||
if (mScriptsEnabled) {
|
if (mScriptsEnabled) {
|
||||||
docShell->SetAllowJavascript(PR_TRUE);
|
docShell->SetAllowJavascript(PR_TRUE);
|
||||||
@ -972,7 +984,7 @@ nsEditingSession::StartDocumentLoad(nsIWebProgress *aWebProgress,
|
|||||||
aWebProgress->GetDOMWindow(getter_AddRefs(domWindow));
|
aWebProgress->GetDOMWindow(getter_AddRefs(domWindow));
|
||||||
if (domWindow)
|
if (domWindow)
|
||||||
{
|
{
|
||||||
TearDownEditorOnWindow(domWindow);
|
TearDownEditorOnWindow(domWindow, PR_FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (aIsToBeMadeEditable)
|
if (aIsToBeMadeEditable)
|
||||||
@ -1041,6 +1053,13 @@ nsEditingSession::EndDocumentLoad(nsIWebProgress *aWebProgress,
|
|||||||
editorDocShell->GetEditable(&makeEditable);
|
editorDocShell->GetEditable(&makeEditable);
|
||||||
|
|
||||||
if (makeEditable)
|
if (makeEditable)
|
||||||
|
{
|
||||||
|
// do we already have an editor here?
|
||||||
|
nsCOMPtr<nsIEditor> editor;
|
||||||
|
rv = editorDocShell->GetEditor(getter_AddRefs(editor));
|
||||||
|
if (NS_FAILED(rv))
|
||||||
|
return rv;
|
||||||
|
if (!editor)
|
||||||
{
|
{
|
||||||
mCanCreateEditor = PR_FALSE;
|
mCanCreateEditor = PR_FALSE;
|
||||||
rv = SetupEditorOnWindow(domWindow);
|
rv = SetupEditorOnWindow(domWindow);
|
||||||
@ -1064,6 +1083,17 @@ nsEditingSession::EndDocumentLoad(nsIWebProgress *aWebProgress,
|
|||||||
10, nsITimer::TYPE_ONE_SHOT);
|
10, nsITimer::TYPE_ONE_SHOT);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// XXX This should move somewhere else!
|
||||||
|
nsCOMPtr<nsIDOMDocument> domDoc;
|
||||||
|
rv = domWindow->GetDocument(getter_AddRefs(domDoc));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIDocument> doc = do_QueryInterface(domDoc, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
doc->SetEditableFlag(PR_TRUE);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return rv;
|
return rv;
|
||||||
}
|
}
|
||||||
@ -1144,7 +1174,7 @@ nsEditingSession::EndPageLoad(nsIWebProgress *aWebProgress,
|
|||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
// Shouldn't we do this when we want to edit sub-frames?
|
// Shouldn't we do this when we want to edit sub-frames?
|
||||||
return MakeWindowEditable(domWindow, "html", PR_FALSE);
|
return MakeWindowEditable(domWindow, "html", PR_FALSE, mInteractive);
|
||||||
#else
|
#else
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@ -129,6 +129,8 @@ protected:
|
|||||||
// before creating an editor
|
// before creating an editor
|
||||||
PRPackedBool mCanCreateEditor;
|
PRPackedBool mCanCreateEditor;
|
||||||
|
|
||||||
|
PRPackedBool mInteractive;
|
||||||
|
|
||||||
// True if scripts were enabled before the editor turned scripts
|
// True if scripts were enabled before the editor turned scripts
|
||||||
// off, otherwise false.
|
// off, otherwise false.
|
||||||
PRPackedBool mScriptsEnabled;
|
PRPackedBool mScriptsEnabled;
|
||||||
|
|||||||
@ -35,6 +35,10 @@
|
|||||||
*
|
*
|
||||||
* ***** END LICENSE BLOCK ***** */
|
* ***** END LICENSE BLOCK ***** */
|
||||||
|
|
||||||
|
*|* {
|
||||||
|
-moz-user-modify: read-write;
|
||||||
|
}
|
||||||
|
|
||||||
/* Styles to alter look of things in the Editor content window
|
/* Styles to alter look of things in the Editor content window
|
||||||
* that should NOT be removed when we display in completely WYSIWYG
|
* that should NOT be removed when we display in completely WYSIWYG
|
||||||
* "Browser Preview" mode.
|
* "Browser Preview" mode.
|
||||||
|
|||||||
@ -18,10 +18,12 @@ GetContentDOMWindow call. Then simply call
|
|||||||
nsIWebBrowser->do_GetInterface on the nsIWebBrowser to retrieve the
|
nsIWebBrowser->do_GetInterface on the nsIWebBrowser to retrieve the
|
||||||
nsIEditingSession from it. From there you call
|
nsIEditingSession from it. From there you call
|
||||||
editingSession->MakeWindowEditable(domWindow, editortype,
|
editingSession->MakeWindowEditable(domWindow, editortype,
|
||||||
PR_TRUE); The first parameter is the nsIDOMWindow you
|
PR_TRUE, PR_FALSE); The first parameter is the nsIDOMWindow
|
||||||
just retrieved, the second is the editor type you want to create and the
|
you just retrieved, the second is the editor type you want to create and the
|
||||||
third is whether you want the window editable immediately or when the
|
third is whether you want the window editable immediately or when the
|
||||||
document is done loading. In calling this method the editor is
|
document is done loading, the fourth is whether you want the editor to make
|
||||||
|
the whole document editable, the fifth is whether you want to turn of
|
||||||
|
scripts, plugins, ... In calling this method the editor is
|
||||||
created underneath and the event listeners etc. are all prepared.<br>
|
created underneath and the event listeners etc. are all prepared.<br>
|
||||||
</p>
|
</p>
|
||||||
<p><i> nsCOMPtr<nsIDOMWindow> domWindow;<br>
|
<p><i> nsCOMPtr<nsIDOMWindow> domWindow;<br>
|
||||||
@ -36,7 +38,8 @@ editingSession;<br>
|
|||||||
nsIWebBrowser->do_GetInterface(getter_AddRefs(editingSession));<br>
|
nsIWebBrowser->do_GetInterface(getter_AddRefs(editingSession));<br>
|
||||||
if (editingSession)<br>
|
if (editingSession)<br>
|
||||||
|
|
||||||
editingSession->MakeWindowEditable(domWindow, "html", PR_TRUE);</i></p>
|
editingSession->MakeWindowEditable(domWindow, "html", PR_TRUE,
|
||||||
|
PR_FALSE, PR_TRUE, PR_FALSE);</i></p>
|
||||||
<p>The valid editor types are:<br>
|
<p>The valid editor types are:<br>
|
||||||
</p>
|
</p>
|
||||||
<ul>
|
<ul>
|
||||||
|
|||||||
@ -62,7 +62,7 @@ typedef short EDirection;
|
|||||||
|
|
||||||
[ptr] native nsIPresShellPtr(nsIPresShell);
|
[ptr] native nsIPresShellPtr(nsIPresShell);
|
||||||
|
|
||||||
[scriptable, uuid(470e18e4-2e82-48de-8850-7474cdbbd97d)]
|
[scriptable, uuid(dc81f464-89dd-47bf-bf21-10df3b65b956)]
|
||||||
|
|
||||||
interface nsIEditor : nsISupports
|
interface nsIEditor : nsISupports
|
||||||
{
|
{
|
||||||
@ -562,4 +562,7 @@ interface nsIEditor : nsISupports
|
|||||||
|
|
||||||
/* Run unit tests. Noop in optimized builds */
|
/* Run unit tests. Noop in optimized builds */
|
||||||
void debugUnitTests(out long outNumTests, out long outNumTestsFailed);
|
void debugUnitTests(out long outNumTests, out long outNumTestsFailed);
|
||||||
|
|
||||||
|
/* checks if a node is read-only or not */
|
||||||
|
[notxpcom] boolean isModifiableNode(in nsIDOMNode aNode);
|
||||||
};
|
};
|
||||||
|
|||||||
@ -39,34 +39,23 @@
|
|||||||
|
|
||||||
interface nsIDOMKeyEvent;
|
interface nsIDOMKeyEvent;
|
||||||
|
|
||||||
[scriptable, uuid(28dbb4d0-5fea-43f4-aca7-344fc2ecc4f9)]
|
[scriptable, uuid(35d74f2b-3d03-43c5-8ace-9d90c3e31244)]
|
||||||
interface nsIPlaintextEditor : nsISupports
|
interface nsIPlaintextEditor : nsISupports
|
||||||
{
|
{
|
||||||
|
|
||||||
/* the bits in an editor behavior mask. */
|
// XXX Why aren't these in nsIEditor?
|
||||||
const short eEditorPlaintextBit = 0; /* only plain text entry is allowed via events */
|
const long eEditorPlaintextMask = 0x0001; /* only plain text entry is allowed via events */
|
||||||
const short eEditorSingleLineBit = 1; /* enter key and CR-LF handled specially */
|
const long eEditorSingleLineMask = 0x0002; /* enter key and CR-LF handled specially */
|
||||||
const short eEditorPasswordBit = 2; /* text is not entered into content, only a representative character */
|
const long eEditorPasswordMask = 0x0004; /* text is not entered into content, only a representative character */
|
||||||
const short eEditorReadonlyBit = 3; /* editing events are disabled. Editor may still accept focus. */
|
const long eEditorReadonlyMask = 0x0008; /* editing events are disabled. Editor may still accept focus. */
|
||||||
const short eEditorDisabledBit = 4; /* all events are disabled (like scrolling). Editor will not accept focus. */
|
const long eEditorDisabledMask = 0x0010; /* all events are disabled (like scrolling). Editor will not accept focus. */
|
||||||
const short eEditorFilterInputBit = 5; /* text input is limited to certain character types, use mFilter */
|
const long eEditorFilterInputMask = 0x0020; /* text input is limited to certain character types, use mFilter */
|
||||||
const short eEditorMailBit = 6; /* use mail-compose editting rules */
|
const long eEditorMailMask = 0x0040; /* use mail-compose editing rules */
|
||||||
const short eEditorUseAsyncUpdatesBit = 7; /* prevent immediate reflows and view refreshes */
|
const long eEditorUseAsyncUpdatesMask = 0x0080; /* prevent immediate reflows and view refreshes */
|
||||||
const short eEditorEnableWrapHackBit = 8; /* allow the editor to set font: monospace on the root node */
|
const long eEditorEnableWrapHackMask = 0x0100; /* allow the editor to set font: monospace on the root node */
|
||||||
const short eEditorWidgetBit = 9; /* bit for widgets */
|
const long eEditorWidgetMask = 0x0200; /* bit for widgets */
|
||||||
const short eEditorNoCSSBit = 10; /* this HTML editor should not create css styles */
|
const long eEditorNoCSSMask = 0x0400; /* this HTML editor should not create css styles */
|
||||||
|
const long eEditorAllowInteraction = 0x0800; /* */
|
||||||
const long eEditorPlaintextMask = 1;
|
|
||||||
const long eEditorSingleLineMask = 2;
|
|
||||||
const long eEditorPasswordMask = 4;
|
|
||||||
const long eEditorReadonlyMask = 8;
|
|
||||||
const long eEditorDisabledMask = 16;
|
|
||||||
const long eEditorFilterInputMask = 32;
|
|
||||||
const long eEditorMailMask = 64;
|
|
||||||
const long eEditorUseAsyncUpdatesMask = 128;
|
|
||||||
const long eEditorEnableWrapHackMask = 256;
|
|
||||||
const long eEditorWidgetMask = 512;
|
|
||||||
const long eEditorNoCSSMask = 1024;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The valid values for newlines handling.
|
* The valid values for newlines handling.
|
||||||
|
|||||||
@ -58,11 +58,21 @@ DeleteElementTxn::DeleteElementTxn()
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
NS_IMETHODIMP DeleteElementTxn::Init(nsIDOMNode *aElement,
|
NS_IMETHODIMP DeleteElementTxn::Init(nsIEditor *aEditor,
|
||||||
|
nsIDOMNode *aElement,
|
||||||
nsRangeUpdater *aRangeUpdater)
|
nsRangeUpdater *aRangeUpdater)
|
||||||
{
|
{
|
||||||
if (!aElement) return NS_ERROR_NULL_POINTER;
|
if (!aEditor || !aElement) return NS_ERROR_NULL_POINTER;
|
||||||
|
mEditor = aEditor;
|
||||||
mElement = do_QueryInterface(aElement);
|
mElement = do_QueryInterface(aElement);
|
||||||
|
nsresult result = mElement->GetParentNode(getter_AddRefs(mParent));
|
||||||
|
if (NS_FAILED(result)) { return result; }
|
||||||
|
|
||||||
|
// do nothing if the parent is read-only
|
||||||
|
if (mParent && !mEditor->IsModifiableNode(mParent)) {
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
mRangeUpdater = aRangeUpdater;
|
mRangeUpdater = aRangeUpdater;
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
@ -76,8 +86,6 @@ NS_IMETHODIMP DeleteElementTxn::DoTransaction(void)
|
|||||||
|
|
||||||
if (!mElement) return NS_ERROR_NOT_INITIALIZED;
|
if (!mElement) return NS_ERROR_NOT_INITIALIZED;
|
||||||
|
|
||||||
nsresult result = mElement->GetParentNode(getter_AddRefs(mParent));
|
|
||||||
if (NS_FAILED(result)) { return result; }
|
|
||||||
if (!mParent) { return NS_OK; } // this is a no-op, there's no parent to delete mElement from
|
if (!mParent) { return NS_OK; } // this is a no-op, there's no parent to delete mElement from
|
||||||
|
|
||||||
#ifdef NS_DEBUG
|
#ifdef NS_DEBUG
|
||||||
@ -105,7 +113,7 @@ NS_IMETHODIMP DeleteElementTxn::DoTransaction(void)
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
// remember which child mElement was (by remembering which child was next)
|
// remember which child mElement was (by remembering which child was next)
|
||||||
result = mElement->GetNextSibling(getter_AddRefs(mRefNode)); // can return null mRefNode
|
nsresult result = mElement->GetNextSibling(getter_AddRefs(mRefNode)); // can return null mRefNode
|
||||||
|
|
||||||
// give range updater a chance. SelAdjDeleteNode() needs to be called *before*
|
// give range updater a chance. SelAdjDeleteNode() needs to be called *before*
|
||||||
// we do the action, unlike some of the other nsRangeStore update methods.
|
// we do the action, unlike some of the other nsRangeStore update methods.
|
||||||
|
|||||||
@ -41,6 +41,7 @@
|
|||||||
#include "EditTxn.h"
|
#include "EditTxn.h"
|
||||||
|
|
||||||
#include "nsIDOMNode.h"
|
#include "nsIDOMNode.h"
|
||||||
|
#include "nsIEditor.h"
|
||||||
#include "nsCOMPtr.h"
|
#include "nsCOMPtr.h"
|
||||||
|
|
||||||
#define DELETE_ELEMENT_TXN_CID \
|
#define DELETE_ELEMENT_TXN_CID \
|
||||||
@ -62,7 +63,7 @@ public:
|
|||||||
/** initialize the transaction.
|
/** initialize the transaction.
|
||||||
* @param aElement the node to delete
|
* @param aElement the node to delete
|
||||||
*/
|
*/
|
||||||
NS_IMETHOD Init(nsIDOMNode *aElement, nsRangeUpdater *aRangeUpdater);
|
NS_IMETHOD Init(nsIEditor *aEditor, nsIDOMNode *aElement, nsRangeUpdater *aRangeUpdater);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
DeleteElementTxn();
|
DeleteElementTxn();
|
||||||
@ -83,6 +84,9 @@ protected:
|
|||||||
/** next sibling to remember for undo/redo purposes */
|
/** next sibling to remember for undo/redo purposes */
|
||||||
nsCOMPtr<nsIDOMNode> mRefNode;
|
nsCOMPtr<nsIDOMNode> mRefNode;
|
||||||
|
|
||||||
|
/** the editor for this transaction */
|
||||||
|
nsIEditor* mEditor;
|
||||||
|
|
||||||
/** range updater object */
|
/** range updater object */
|
||||||
nsRangeUpdater *mRangeUpdater;
|
nsRangeUpdater *mRangeUpdater;
|
||||||
|
|
||||||
|
|||||||
@ -88,6 +88,17 @@ NS_IMETHODIMP DeleteRangeTxn::Init(nsIEditor *aEditor,
|
|||||||
result = aRange->GetCommonAncestorContainer(getter_AddRefs(mCommonParent));
|
result = aRange->GetCommonAncestorContainer(getter_AddRefs(mCommonParent));
|
||||||
NS_ASSERTION((NS_SUCCEEDED(result)), "GetCommonParent failed.");
|
NS_ASSERTION((NS_SUCCEEDED(result)), "GetCommonParent failed.");
|
||||||
|
|
||||||
|
if (!mEditor->IsModifiableNode(mStartParent)) {
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mStartParent!=mEndParent &&
|
||||||
|
(!mEditor->IsModifiableNode(mEndParent) ||
|
||||||
|
!mEditor->IsModifiableNode(mCommonParent)))
|
||||||
|
{
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef NS_DEBUG
|
#ifdef NS_DEBUG
|
||||||
{
|
{
|
||||||
PRUint32 count;
|
PRUint32 count;
|
||||||
@ -236,7 +247,8 @@ DeleteRangeTxn::CreateTxnsToDeleteBetween(nsIDOMNode *aStartParent,
|
|||||||
numToDel = 1;
|
numToDel = 1;
|
||||||
else
|
else
|
||||||
numToDel = aEndOffset-aStartOffset;
|
numToDel = aEndOffset-aStartOffset;
|
||||||
txn->Init(mEditor, textNode, aStartOffset, numToDel, mRangeUpdater);
|
result = txn->Init(mEditor, textNode, aStartOffset, numToDel, mRangeUpdater);
|
||||||
|
if (NS_SUCCEEDED(result))
|
||||||
AppendChild(txn);
|
AppendChild(txn);
|
||||||
NS_RELEASE(txn);
|
NS_RELEASE(txn);
|
||||||
}
|
}
|
||||||
@ -265,7 +277,8 @@ DeleteRangeTxn::CreateTxnsToDeleteBetween(nsIDOMNode *aStartParent,
|
|||||||
if (NS_FAILED(result)) return result;
|
if (NS_FAILED(result)) return result;
|
||||||
if (!txn) return NS_ERROR_NULL_POINTER;
|
if (!txn) return NS_ERROR_NULL_POINTER;
|
||||||
|
|
||||||
txn->Init(child, mRangeUpdater);
|
result = txn->Init(mEditor, child, mRangeUpdater);
|
||||||
|
if (NS_SUCCEEDED(result))
|
||||||
AppendChild(txn);
|
AppendChild(txn);
|
||||||
NS_RELEASE(txn);
|
NS_RELEASE(txn);
|
||||||
}
|
}
|
||||||
@ -302,7 +315,8 @@ NS_IMETHODIMP DeleteRangeTxn::CreateTxnsToDeleteContent(nsIDOMNode *aParent,
|
|||||||
if (NS_FAILED(result)) return result;
|
if (NS_FAILED(result)) return result;
|
||||||
if (!txn) return NS_ERROR_NULL_POINTER;
|
if (!txn) return NS_ERROR_NULL_POINTER;
|
||||||
|
|
||||||
txn->Init(mEditor, textNode, start, numToDelete, mRangeUpdater);
|
result = txn->Init(mEditor, textNode, start, numToDelete, mRangeUpdater);
|
||||||
|
if (NS_SUCCEEDED(result))
|
||||||
AppendChild(txn);
|
AppendChild(txn);
|
||||||
NS_RELEASE(txn);
|
NS_RELEASE(txn);
|
||||||
}
|
}
|
||||||
@ -319,7 +333,7 @@ NS_IMETHODIMP DeleteRangeTxn::CreateTxnsToDeleteNodesBetween()
|
|||||||
nsresult result = iter->Init(mRange);
|
nsresult result = iter->Init(mRange);
|
||||||
if (NS_FAILED(result)) return result;
|
if (NS_FAILED(result)) return result;
|
||||||
|
|
||||||
while (!iter->IsDone())
|
while (!iter->IsDone() && NS_SUCCEEDED(result))
|
||||||
{
|
{
|
||||||
nsCOMPtr<nsIDOMNode> node = do_QueryInterface(iter->GetCurrentNode());
|
nsCOMPtr<nsIDOMNode> node = do_QueryInterface(iter->GetCurrentNode());
|
||||||
if (!node)
|
if (!node)
|
||||||
@ -330,7 +344,8 @@ NS_IMETHODIMP DeleteRangeTxn::CreateTxnsToDeleteNodesBetween()
|
|||||||
if (NS_FAILED(result)) return result;
|
if (NS_FAILED(result)) return result;
|
||||||
if (!txn) return NS_ERROR_NULL_POINTER;
|
if (!txn) return NS_ERROR_NULL_POINTER;
|
||||||
|
|
||||||
txn->Init(node, mRangeUpdater);
|
result = txn->Init(mEditor, node, mRangeUpdater);
|
||||||
|
if (NS_SUCCEEDED(result))
|
||||||
AppendChild(txn);
|
AppendChild(txn);
|
||||||
NS_RELEASE(txn);
|
NS_RELEASE(txn);
|
||||||
iter->Next();
|
iter->Next();
|
||||||
|
|||||||
@ -65,6 +65,11 @@ NS_IMETHODIMP DeleteTextTxn::Init(nsIEditor *aEditor,
|
|||||||
|
|
||||||
mEditor = aEditor;
|
mEditor = aEditor;
|
||||||
mElement = do_QueryInterface(aElement);
|
mElement = do_QueryInterface(aElement);
|
||||||
|
// do nothing if the node is read-only
|
||||||
|
if (!mEditor->IsModifiableNode(mElement)) {
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
mOffset = aOffset;
|
mOffset = aOffset;
|
||||||
mNumCharsToDelete = aNumCharsToDelete;
|
mNumCharsToDelete = aNumCharsToDelete;
|
||||||
NS_ASSERTION(0!=aNumCharsToDelete, "bad arg, numCharsToDelete");
|
NS_ASSERTION(0!=aNumCharsToDelete, "bad arg, numCharsToDelete");
|
||||||
|
|||||||
@ -57,6 +57,12 @@ NS_IMETHODIMP JoinElementTxn::Init(nsEditor *aEditor,
|
|||||||
if (!aEditor || !aLeftNode || !aRightNode) { return NS_ERROR_NULL_POINTER; }
|
if (!aEditor || !aLeftNode || !aRightNode) { return NS_ERROR_NULL_POINTER; }
|
||||||
mEditor = aEditor;
|
mEditor = aEditor;
|
||||||
mLeftNode = do_QueryInterface(aLeftNode);
|
mLeftNode = do_QueryInterface(aLeftNode);
|
||||||
|
nsCOMPtr<nsIDOMNode>leftParent;
|
||||||
|
nsresult result = mLeftNode->GetParentNode(getter_AddRefs(leftParent));
|
||||||
|
if (NS_FAILED(result)) return result;
|
||||||
|
if (!mEditor->IsModifiableNode(leftParent)) {
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
}
|
||||||
mRightNode = do_QueryInterface(aRightNode);
|
mRightNode = do_QueryInterface(aRightNode);
|
||||||
mOffset=0;
|
mOffset=0;
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
|
|||||||
@ -166,6 +166,7 @@ EDITOR_ATOM(cssWhitespace, "white-space")
|
|||||||
EDITOR_ATOM(cssWidth, "width")
|
EDITOR_ATOM(cssWidth, "width")
|
||||||
EDITOR_ATOM(cssZIndex, "z-index")
|
EDITOR_ATOM(cssZIndex, "z-index")
|
||||||
|
|
||||||
|
EDITOR_ATOM(cssMozUserModify, "-moz-user-modify")
|
||||||
EDITOR_ATOM(cssMozUserSelect, "-moz-user-select")
|
EDITOR_ATOM(cssMozUserSelect, "-moz-user-select")
|
||||||
EDITOR_ATOM(mozdirty, "_moz_dirty")
|
EDITOR_ATOM(mozdirty, "_moz_dirty")
|
||||||
|
|
||||||
|
|||||||
@ -3777,7 +3777,7 @@ nsEditor::IsEditable(nsIDOMNode *aNode)
|
|||||||
GetPresShell(getter_AddRefs(shell));
|
GetPresShell(getter_AddRefs(shell));
|
||||||
if (!shell) return PR_FALSE;
|
if (!shell) return PR_FALSE;
|
||||||
|
|
||||||
if (IsMozEditorBogusNode(aNode)) return PR_FALSE;
|
if (IsMozEditorBogusNode(aNode) || !IsModifiableNode(aNode)) return PR_FALSE;
|
||||||
|
|
||||||
// see if it has a frame. If so, we'll edit it.
|
// see if it has a frame. If so, we'll edit it.
|
||||||
// special case for textnodes: frame must have width.
|
// special case for textnodes: frame must have width.
|
||||||
@ -4734,7 +4734,7 @@ NS_IMETHODIMP nsEditor::CreateTxnForDeleteElement(nsIDOMNode * aElement,
|
|||||||
{
|
{
|
||||||
result = TransactionFactory::GetNewTransaction(DeleteElementTxn::GetCID(), (EditTxn **)aTxn);
|
result = TransactionFactory::GetNewTransaction(DeleteElementTxn::GetCID(), (EditTxn **)aTxn);
|
||||||
if (NS_SUCCEEDED(result)) {
|
if (NS_SUCCEEDED(result)) {
|
||||||
result = (*aTxn)->Init(aElement, &mRangeUpdater);
|
result = (*aTxn)->Init(this, aElement, &mRangeUpdater);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
@ -5355,3 +5355,9 @@ nsEditor::DumpNode(nsIDOMNode *aNode, PRInt32 indent)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
PRBool
|
||||||
|
nsEditor::IsModifiableNode(nsIDOMNode *aNode)
|
||||||
|
{
|
||||||
|
return PR_TRUE;
|
||||||
|
}
|
||||||
|
|||||||
@ -586,6 +586,51 @@ nsHTMLEditRules::WillDoAction(nsISelection *aSelection,
|
|||||||
// my kingdom for dynamic cast
|
// my kingdom for dynamic cast
|
||||||
nsTextRulesInfo *info = NS_STATIC_CAST(nsTextRulesInfo*, aInfo);
|
nsTextRulesInfo *info = NS_STATIC_CAST(nsTextRulesInfo*, aInfo);
|
||||||
|
|
||||||
|
// Deal with actions for which we don't need to check whether the selection is
|
||||||
|
// editable.
|
||||||
|
if (info->action == kOutputText) {
|
||||||
|
return nsTextEditRules::WillDoAction(aSelection, aInfo, aCancel, aHandled);
|
||||||
|
}
|
||||||
|
|
||||||
|
nsCOMPtr<nsIDOMRange> domRange;
|
||||||
|
nsresult rv = aSelection->GetRangeAt(0, getter_AddRefs(domRange));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIDOMNode> selStartNode;
|
||||||
|
rv = domRange->GetStartContainer(getter_AddRefs(selStartNode));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
if (!mHTMLEditor->IsModifiableNode(selStartNode))
|
||||||
|
{
|
||||||
|
*aCancel = PR_TRUE;
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsCOMPtr<nsIDOMNode> selEndNode;
|
||||||
|
rv = domRange->GetEndContainer(getter_AddRefs(selEndNode));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
if (selStartNode != selEndNode)
|
||||||
|
{
|
||||||
|
if (!mHTMLEditor->IsModifiableNode(selEndNode))
|
||||||
|
{
|
||||||
|
*aCancel = PR_TRUE;
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsCOMPtr<nsIRange> range = do_QueryInterface(domRange);
|
||||||
|
nsCOMPtr<nsIDOMNode> ancestor =
|
||||||
|
do_QueryInterface(range->GetCommonAncestor());
|
||||||
|
if (!mHTMLEditor->IsModifiableNode(ancestor))
|
||||||
|
{
|
||||||
|
*aCancel = PR_TRUE;
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
switch (info->action)
|
switch (info->action)
|
||||||
{
|
{
|
||||||
case kInsertText:
|
case kInsertText:
|
||||||
@ -1569,6 +1614,13 @@ nsHTMLEditRules::WillInsertBreak(nsISelection *aSelection, PRBool *aCancel, PRBo
|
|||||||
|
|
||||||
if (!blockParent) return NS_ERROR_FAILURE;
|
if (!blockParent) return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
// do nothing if the node is read-only
|
||||||
|
if (!mHTMLEditor->IsModifiableNode(blockParent))
|
||||||
|
{
|
||||||
|
*aCancel = PR_TRUE;
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
// if block is empty, populate with br.
|
// if block is empty, populate with br.
|
||||||
// (for example, imagine a div that contains the word "text". the user selects
|
// (for example, imagine a div that contains the word "text". the user selects
|
||||||
// "text" and types return. "text" is deleted leaving an empty block. we want
|
// "text" and types return. "text" is deleted leaving an empty block. we want
|
||||||
|
|||||||
@ -130,6 +130,7 @@
|
|||||||
#include "nsIView.h"
|
#include "nsIView.h"
|
||||||
#include "nsIWidget.h"
|
#include "nsIWidget.h"
|
||||||
#include "nsIParserService.h"
|
#include "nsIParserService.h"
|
||||||
|
#include "nsIEventStateManager.h"
|
||||||
|
|
||||||
// Some utilities to handle annoying overloading of "A" tag for link and named anchor
|
// Some utilities to handle annoying overloading of "A" tag for link and named anchor
|
||||||
static char hrefText[] = "href";
|
static char hrefText[] = "href";
|
||||||
@ -302,7 +303,7 @@ nsHTMLEditor::Init(nsIDOMDocument *aDoc, nsIPresShell *aPresShell,
|
|||||||
// disable links
|
// disable links
|
||||||
nsPresContext *context = aPresShell->GetPresContext();
|
nsPresContext *context = aPresShell->GetPresContext();
|
||||||
if (!context) return NS_ERROR_NULL_POINTER;
|
if (!context) return NS_ERROR_NULL_POINTER;
|
||||||
if (!(mFlags & eEditorPlaintextMask)) {
|
if (!(mFlags & (eEditorPlaintextMask | eEditorAllowInteraction))) {
|
||||||
mLinkHandler = context->GetLinkHandler();
|
mLinkHandler = context->GetLinkHandler();
|
||||||
|
|
||||||
context->SetLinkHandler(nsnull);
|
context->SetLinkHandler(nsnull);
|
||||||
@ -317,8 +318,10 @@ nsHTMLEditor::Init(nsIDOMDocument *aDoc, nsIPresShell *aPresShell,
|
|||||||
mSelectionListenerP = new ResizerSelectionListener(this);
|
mSelectionListenerP = new ResizerSelectionListener(this);
|
||||||
if (!mSelectionListenerP) {return NS_ERROR_NULL_POINTER;}
|
if (!mSelectionListenerP) {return NS_ERROR_NULL_POINTER;}
|
||||||
|
|
||||||
|
if (!(mFlags & eEditorAllowInteraction)) {
|
||||||
// ignore any errors from this in case the file is missing
|
// ignore any errors from this in case the file is missing
|
||||||
AddOverrideStyleSheet(NS_LITERAL_STRING("resource:/res/EditorOverride.css"));
|
AddOverrideStyleSheet(NS_LITERAL_STRING("resource:/res/EditorOverride.css"));
|
||||||
|
}
|
||||||
|
|
||||||
nsCOMPtr<nsISelection>selection;
|
nsCOMPtr<nsISelection>selection;
|
||||||
result = GetSelection(getter_AddRefs(selection));
|
result = GetSelection(getter_AddRefs(selection));
|
||||||
@ -3884,6 +3887,11 @@ nsHTMLEditor::GetEmbeddedObjects(nsISupportsArray** aNodeList)
|
|||||||
|
|
||||||
NS_IMETHODIMP nsHTMLEditor::DeleteNode(nsIDOMNode * aNode)
|
NS_IMETHODIMP nsHTMLEditor::DeleteNode(nsIDOMNode * aNode)
|
||||||
{
|
{
|
||||||
|
// do nothing if the node is read-only
|
||||||
|
if (!IsModifiableNode(aNode)) {
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
nsCOMPtr<nsIDOMNode> selectAllNode = FindUserSelectAllNode(aNode);
|
nsCOMPtr<nsIDOMNode> selectAllNode = FindUserSelectAllNode(aNode);
|
||||||
|
|
||||||
if (selectAllNode)
|
if (selectAllNode)
|
||||||
@ -3897,6 +3905,11 @@ NS_IMETHODIMP nsHTMLEditor::DeleteText(nsIDOMCharacterData *aTextNode,
|
|||||||
PRUint32 aOffset,
|
PRUint32 aOffset,
|
||||||
PRUint32 aLength)
|
PRUint32 aLength)
|
||||||
{
|
{
|
||||||
|
// do nothing if the node is read-only
|
||||||
|
if (!IsModifiableNode(aTextNode)) {
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
nsCOMPtr<nsIDOMNode> selectAllNode = FindUserSelectAllNode(aTextNode);
|
nsCOMPtr<nsIDOMNode> selectAllNode = FindUserSelectAllNode(aTextNode);
|
||||||
|
|
||||||
if (selectAllNode)
|
if (selectAllNode)
|
||||||
@ -3906,6 +3919,18 @@ NS_IMETHODIMP nsHTMLEditor::DeleteText(nsIDOMCharacterData *aTextNode,
|
|||||||
return nsEditor::DeleteText(aTextNode, aOffset, aLength);
|
return nsEditor::DeleteText(aTextNode, aOffset, aLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NS_IMETHODIMP nsHTMLEditor::InsertTextImpl(const nsAString& aStringToInsert,
|
||||||
|
nsCOMPtr<nsIDOMNode> *aInOutNode,
|
||||||
|
PRInt32 *aInOutOffset,
|
||||||
|
nsIDOMDocument *aDoc)
|
||||||
|
{
|
||||||
|
// do nothing if the node is read-only
|
||||||
|
if (!IsModifiableNode(*aInOutNode)) {
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return nsEditor::InsertTextImpl(aStringToInsert, aInOutNode, aInOutOffset, aDoc);
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef XP_MAC
|
#ifdef XP_MAC
|
||||||
#pragma mark -
|
#pragma mark -
|
||||||
@ -3947,6 +3972,14 @@ nsCOMPtr<nsIDOMNode> nsHTMLEditor::FindUserSelectAllNode(nsIDOMNode *aNode)
|
|||||||
return resultNode;
|
return resultNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NS_IMETHODIMP_(PRBool)
|
||||||
|
nsHTMLEditor::IsModifiableNode(nsIDOMNode *aNode)
|
||||||
|
{
|
||||||
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aNode);
|
||||||
|
|
||||||
|
return !content || !(content->IntrinsicState() & NS_EVENT_STATE_MOZ_READONLY);
|
||||||
|
}
|
||||||
|
|
||||||
static nsresult SetSelectionAroundHeadChildren(nsCOMPtr<nsISelection> aSelection, nsWeakPtr aDocWeak)
|
static nsresult SetSelectionAroundHeadChildren(nsCOMPtr<nsISelection> aSelection, nsWeakPtr aDocWeak)
|
||||||
{
|
{
|
||||||
nsresult res = NS_OK;
|
nsresult res = NS_OK;
|
||||||
@ -4204,6 +4237,54 @@ nsHTMLEditor::SelectEntireDocument(nsISelection *aSelection)
|
|||||||
return nsEditor::SelectEntireDocument(aSelection);
|
return nsEditor::SelectEntireDocument(aSelection);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static nsIContent*
|
||||||
|
FindEditableRoot(nsIContent *aContent)
|
||||||
|
{
|
||||||
|
nsIDocument *document = aContent->GetCurrentDoc();
|
||||||
|
if (!document || document->HasFlag(NODE_IS_EDITABLE) ||
|
||||||
|
!aContent->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return nsnull;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsIContent *parent, *content = aContent;
|
||||||
|
while ((parent = content->GetParent()) && parent->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
content = parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
return content;
|
||||||
|
}
|
||||||
|
|
||||||
|
NS_IMETHODIMP
|
||||||
|
nsHTMLEditor::SelectAll()
|
||||||
|
{
|
||||||
|
ForceCompositionEnd();
|
||||||
|
|
||||||
|
nsresult rv;
|
||||||
|
nsCOMPtr<nsISelectionController> selCon = do_QueryReferent(mSelConWeak, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsISelection> selection;
|
||||||
|
rv = selCon->GetSelection(nsISelectionController::SELECTION_NORMAL,
|
||||||
|
getter_AddRefs(selection));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIDOMNode> anchorNode;
|
||||||
|
rv = selection->GetAnchorNode(getter_AddRefs(anchorNode));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsCOMPtr<nsIContent> anchorContent = do_QueryInterface(anchorNode, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
nsIContent *rootContent = FindEditableRoot(anchorContent);
|
||||||
|
if (!rootContent) {
|
||||||
|
return SelectEntireDocument(selection);
|
||||||
|
}
|
||||||
|
|
||||||
|
nsCOMPtr<nsIDOMNode> rootElement = do_QueryInterface(rootContent, &rv);
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
|
||||||
|
return selection->SelectAllChildren(rootElement);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#ifdef XP_MAC
|
#ifdef XP_MAC
|
||||||
|
|||||||
@ -363,6 +363,13 @@ public:
|
|||||||
NS_IMETHODIMP DeleteText(nsIDOMCharacterData *aTextNode,
|
NS_IMETHODIMP DeleteText(nsIDOMCharacterData *aTextNode,
|
||||||
PRUint32 aOffset,
|
PRUint32 aOffset,
|
||||||
PRUint32 aLength);
|
PRUint32 aLength);
|
||||||
|
NS_IMETHOD InsertTextImpl(const nsAString& aStringToInsert,
|
||||||
|
nsCOMPtr<nsIDOMNode> *aInOutNode,
|
||||||
|
PRInt32 *aInOutOffset,
|
||||||
|
nsIDOMDocument *aDoc);
|
||||||
|
NS_IMETHOD_(PRBool) IsModifiableNode(nsIDOMNode *aNode);
|
||||||
|
|
||||||
|
NS_IMETHOD SelectAll();
|
||||||
|
|
||||||
/* ------------ nsICSSLoaderObserver -------------- */
|
/* ------------ nsICSSLoaderObserver -------------- */
|
||||||
NS_IMETHOD StyleSheetLoaded(nsICSSStyleSheet*aSheet, PRBool aWasAlternate,
|
NS_IMETHOD StyleSheetLoaded(nsICSSStyleSheet*aSheet, PRBool aWasAlternate,
|
||||||
|
|||||||
@ -68,6 +68,7 @@
|
|||||||
#include "nsEditorUtils.h"
|
#include "nsEditorUtils.h"
|
||||||
#include "nsIDOMEventTarget.h"
|
#include "nsIDOMEventTarget.h"
|
||||||
#include "nsIEventStateManager.h"
|
#include "nsIEventStateManager.h"
|
||||||
|
#include "nsISelectionPrivate.h"
|
||||||
|
|
||||||
//#define DEBUG_IME
|
//#define DEBUG_IME
|
||||||
|
|
||||||
@ -223,7 +224,8 @@ nsTextEditorKeyListener::KeyPress(nsIDOMEvent* aKeyEvent)
|
|||||||
case nsIDOMKeyEvent::DOM_VK_TAB:
|
case nsIDOMKeyEvent::DOM_VK_TAB:
|
||||||
if ((flags & nsIPlaintextEditor::eEditorSingleLineMask) ||
|
if ((flags & nsIPlaintextEditor::eEditorSingleLineMask) ||
|
||||||
(flags & nsIPlaintextEditor::eEditorPasswordMask) ||
|
(flags & nsIPlaintextEditor::eEditorPasswordMask) ||
|
||||||
(flags & nsIPlaintextEditor::eEditorWidgetMask))
|
(flags & nsIPlaintextEditor::eEditorWidgetMask) ||
|
||||||
|
(flags & nsIPlaintextEditor::eEditorAllowInteraction))
|
||||||
return NS_OK; // let it be used for focus switching
|
return NS_OK; // let it be used for focus switching
|
||||||
|
|
||||||
if (isAnyModifierKeyButShift)
|
if (isAnyModifierKeyButShift)
|
||||||
@ -1032,6 +1034,30 @@ IsTargetFocused(nsIDOMEventTarget* aTarget)
|
|||||||
return (focusedContent == content);
|
return (focusedContent == content);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static nsIContent*
|
||||||
|
FindEditableRoot(nsIContent *aContent)
|
||||||
|
{
|
||||||
|
nsIDocument *document = aContent->GetCurrentDoc();
|
||||||
|
if (!document) {
|
||||||
|
return nsnull;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (document->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return document->GetRootContent();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!aContent->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
return nsnull;
|
||||||
|
}
|
||||||
|
|
||||||
|
nsIContent *parent, *content = aContent;
|
||||||
|
while ((parent = content->GetParent()) && parent->HasFlag(NODE_IS_EDITABLE)) {
|
||||||
|
content = parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
return content;
|
||||||
|
}
|
||||||
|
|
||||||
nsresult
|
nsresult
|
||||||
nsTextEditorFocusListener::Focus(nsIDOMEvent* aEvent)
|
nsTextEditorFocusListener::Focus(nsIDOMEvent* aEvent)
|
||||||
{
|
{
|
||||||
@ -1059,18 +1085,38 @@ nsTextEditorFocusListener::Focus(nsIDOMEvent* aEvent)
|
|||||||
mEditor->GetFlags(&flags);
|
mEditor->GetFlags(&flags);
|
||||||
if (! (flags & nsIPlaintextEditor::eEditorDisabledMask))
|
if (! (flags & nsIPlaintextEditor::eEditorDisabledMask))
|
||||||
{ // only enable caret and selection if the editor is not disabled
|
{ // only enable caret and selection if the editor is not disabled
|
||||||
nsCOMPtr<nsIEditor>editor = do_QueryInterface(mEditor);
|
nsCOMPtr<nsIContent> content = do_QueryInterface(target);
|
||||||
if (editor)
|
|
||||||
{
|
nsIContent *editableRoot = content ? FindEditableRoot(content) : nsnull;
|
||||||
nsCOMPtr<nsISelectionController>selCon;
|
|
||||||
editor->GetSelectionController(getter_AddRefs(selCon));
|
nsCOMPtr<nsISelectionController> selCon;
|
||||||
if (selCon)
|
mEditor->GetSelectionController(getter_AddRefs(selCon));
|
||||||
|
nsCOMPtr<nsIPresShell> presShell = do_QueryInterface(selCon);
|
||||||
|
if (selCon && editableRoot)
|
||||||
{
|
{
|
||||||
|
nsCOMPtr<nsISelection> selection;
|
||||||
|
selCon->GetSelection(nsISelectionController::SELECTION_NORMAL,
|
||||||
|
getter_AddRefs(selection));
|
||||||
|
|
||||||
|
if (presShell && selection) {
|
||||||
|
nsCOMPtr<nsICaret> caret;
|
||||||
|
presShell->GetCaret(getter_AddRefs(caret));
|
||||||
|
if (caret) {
|
||||||
|
caret->SetCaretDOMSelection(selection);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const PRBool kIsReadonly = (flags & nsIPlaintextEditor::eEditorReadonlyMask) != 0;
|
const PRBool kIsReadonly = (flags & nsIPlaintextEditor::eEditorReadonlyMask) != 0;
|
||||||
selCon->SetCaretReadOnly(kIsReadonly);
|
selCon->SetCaretReadOnly(kIsReadonly);
|
||||||
selCon->SetCaretEnabled(PR_TRUE);
|
selCon->SetCaretEnabled(PR_TRUE);
|
||||||
selCon->SetDisplaySelection(nsISelectionController::SELECTION_ON);
|
selCon->SetDisplaySelection(nsISelectionController::SELECTION_ON);
|
||||||
selCon->RepaintSelection(nsISelectionController::SELECTION_NORMAL);
|
selCon->RepaintSelection(nsISelectionController::SELECTION_NORMAL);
|
||||||
|
|
||||||
|
nsCOMPtr<nsISelectionPrivate> selectionPrivate =
|
||||||
|
do_QueryInterface(selection);
|
||||||
|
if (selectionPrivate)
|
||||||
|
{
|
||||||
|
selectionPrivate->SetAncestorLimiter(editableRoot);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1105,6 +1151,16 @@ nsTextEditorFocusListener::Blur(nsIDOMEvent* aEvent)
|
|||||||
editor->GetSelectionController(getter_AddRefs(selCon));
|
editor->GetSelectionController(getter_AddRefs(selCon));
|
||||||
if (selCon)
|
if (selCon)
|
||||||
{
|
{
|
||||||
|
nsCOMPtr<nsISelection> selection;
|
||||||
|
selCon->GetSelection(nsISelectionController::SELECTION_NORMAL,
|
||||||
|
getter_AddRefs(selection));
|
||||||
|
|
||||||
|
nsCOMPtr<nsISelectionPrivate> selectionPrivate =
|
||||||
|
do_QueryInterface(selection);
|
||||||
|
if (selectionPrivate) {
|
||||||
|
selectionPrivate->SetAncestorLimiter(nsnull);
|
||||||
|
}
|
||||||
|
|
||||||
selCon->SetCaretEnabled(PR_FALSE);
|
selCon->SetCaretEnabled(PR_FALSE);
|
||||||
|
|
||||||
PRUint32 flags;
|
PRUint32 flags;
|
||||||
|
|||||||
@ -1296,7 +1296,9 @@ nsTextEditRules::CreateBogusNodeIfNeeded(nsISelection *aSelection)
|
|||||||
nsresult res = mBody->GetFirstChild(getter_AddRefs(bodyChild));
|
nsresult res = mBody->GetFirstChild(getter_AddRefs(bodyChild));
|
||||||
while ((NS_SUCCEEDED(res)) && bodyChild)
|
while ((NS_SUCCEEDED(res)) && bodyChild)
|
||||||
{
|
{
|
||||||
if (mEditor->IsMozEditorBogusNode(bodyChild) || mEditor->IsEditable(bodyChild))
|
if (mEditor->IsMozEditorBogusNode(bodyChild) ||
|
||||||
|
!mEditor->IsEditable(mBody) ||
|
||||||
|
mEditor->IsEditable(bodyChild))
|
||||||
{
|
{
|
||||||
needsBogusContent = PR_FALSE;
|
needsBogusContent = PR_FALSE;
|
||||||
break;
|
break;
|
||||||
|
|||||||
@ -54,6 +54,7 @@ const kDisplayModeTabIDS = ["NormalModeButton", "TagModeButton", "SourceModeButt
|
|||||||
const kNormalStyleSheet = "chrome://editor/content/EditorContent.css";
|
const kNormalStyleSheet = "chrome://editor/content/EditorContent.css";
|
||||||
const kAllTagsStyleSheet = "chrome://editor/content/EditorAllTags.css";
|
const kAllTagsStyleSheet = "chrome://editor/content/EditorAllTags.css";
|
||||||
const kParagraphMarksStyleSheet = "chrome://editor/content/EditorParagraphMarks.css";
|
const kParagraphMarksStyleSheet = "chrome://editor/content/EditorParagraphMarks.css";
|
||||||
|
const kContentEditableStyleSheet = "resource:/res/contenteditable.css";
|
||||||
|
|
||||||
const kTextMimeType = "text/plain";
|
const kTextMimeType = "text/plain";
|
||||||
const kHTMLMimeType = "text/html";
|
const kHTMLMimeType = "text/html";
|
||||||
@ -399,6 +400,10 @@ var gEditorDocumentObserver =
|
|||||||
|
|
||||||
// and extra styles for showing anchors, table borders, smileys, etc
|
// and extra styles for showing anchors, table borders, smileys, etc
|
||||||
editor.addOverrideStyleSheet(kNormalStyleSheet);
|
editor.addOverrideStyleSheet(kNormalStyleSheet);
|
||||||
|
|
||||||
|
// remove contenteditable stylesheets if they were applied by the
|
||||||
|
// editingSession
|
||||||
|
editor.removeOverrideStyleSheet(kContentEditableStyleSheet);
|
||||||
} catch (e) {}
|
} catch (e) {}
|
||||||
|
|
||||||
// Things for just the Web Composer application
|
// Things for just the Web Composer application
|
||||||
|
|||||||
@ -1291,7 +1291,8 @@ HRESULT CMozillaBrowser::SetEditorMode(BOOL bEnabled)
|
|||||||
if (NS_FAILED(rv))
|
if (NS_FAILED(rv))
|
||||||
return E_FAIL;
|
return E_FAIL;
|
||||||
|
|
||||||
rv = mEditingSession->MakeWindowEditable(domWindow, "html", PR_FALSE);
|
rv = mEditingSession->MakeWindowEditable(domWindow, "html", PR_FALSE,
|
||||||
|
PR_FALSE);
|
||||||
|
|
||||||
return S_OK;
|
return S_OK;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -101,7 +101,8 @@ void CnsIEditSession::MakeWinEditTest(PRBool afterUriLoad, PRInt16 displayMode)
|
|||||||
editingSession = GetEditSessionObject();
|
editingSession = GetEditSessionObject();
|
||||||
domWindow = GetTheDOMWindow(qaWebBrowser);
|
domWindow = GetTheDOMWindow(qaWebBrowser);
|
||||||
if (editingSession) {
|
if (editingSession) {
|
||||||
rv= editingSession->MakeWindowEditable(domWindow, "text", afterUriLoad);
|
rv= editingSession->MakeWindowEditable(domWindow, "text", afterUriLoad,
|
||||||
|
PR_TRUE, PR_FALSE);
|
||||||
RvTestResult(rv, "MakeWindowEditable() test", displayMode);
|
RvTestResult(rv, "MakeWindowEditable() test", displayMode);
|
||||||
if (displayMode == 1)
|
if (displayMode == 1)
|
||||||
RvTestResultDlg(rv, "MakeWindowEditable() test");
|
RvTestResultDlg(rv, "MakeWindowEditable() test");
|
||||||
@ -171,7 +172,7 @@ void CnsIEditSession::TearEditorWinTest(PRInt16 displayMode)
|
|||||||
editingSession = GetEditSessionObject();
|
editingSession = GetEditSessionObject();
|
||||||
domWindow = GetTheDOMWindow(qaWebBrowser);
|
domWindow = GetTheDOMWindow(qaWebBrowser);
|
||||||
if (editingSession) {
|
if (editingSession) {
|
||||||
rv = editingSession->TearDownEditorOnWindow(domWindow);
|
rv = editingSession->TearDownEditorOnWindow(domWindow, PR_FALSE);
|
||||||
RvTestResult(rv, "TearDownEditorOnWindow() test", displayMode);
|
RvTestResult(rv, "TearDownEditorOnWindow() test", displayMode);
|
||||||
if (displayMode == 1)
|
if (displayMode == 1)
|
||||||
RvTestResultDlg(rv, "TearDownEditorOnWindow() test");
|
RvTestResultDlg(rv, "TearDownEditorOnWindow() test");
|
||||||
|
|||||||
@ -84,7 +84,7 @@ void EditorFrame::MakeEditable()
|
|||||||
nsCOMPtr<nsIEditingSession> editingSession = do_GetInterface(mWebBrowser);
|
nsCOMPtr<nsIEditingSession> editingSession = do_GetInterface(mWebBrowser);
|
||||||
if (!editingSession)
|
if (!editingSession)
|
||||||
return;// NS_ERROR_FAILURE;
|
return;// NS_ERROR_FAILURE;
|
||||||
editingSession->MakeWindowEditable(domWindow, NULL, PR_TRUE);
|
editingSession->MakeWindowEditable(domWindow, NULL, PR_TRUE, PR_FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
nsresult EditorFrame::DoCommand(const char *aCommand, nsICommandParams *aCommandParams)
|
nsresult EditorFrame::DoCommand(const char *aCommand, nsICommandParams *aCommandParams)
|
||||||
|
|||||||
@ -93,6 +93,8 @@
|
|||||||
#include "nsString.h"
|
#include "nsString.h"
|
||||||
#include "nsThreadUtils.h"
|
#include "nsThreadUtils.h"
|
||||||
#include "nsUnicharUtils.h"
|
#include "nsUnicharUtils.h"
|
||||||
|
#include "nsIContent.h"
|
||||||
|
#include "nsIEventStateManager.h"
|
||||||
|
|
||||||
// Set to spew messages to the console about what is happening.
|
// Set to spew messages to the console about what is happening.
|
||||||
//#define DEBUG_INLINESPELL
|
//#define DEBUG_INLINESPELL
|
||||||
@ -1108,6 +1110,11 @@ mozInlineSpellChecker::SkipSpellCheckForNode(nsIEditor* aEditor,
|
|||||||
parent = nextParent;
|
parent = nextParent;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
// XXX Do we really want this for all read-write content?
|
||||||
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aNode);
|
||||||
|
*checkSpelling = content->IntrinsicState() & NS_EVENT_STATE_MOZ_READWRITE;
|
||||||
|
}
|
||||||
|
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -587,11 +587,7 @@ nsCaret::DrawAtPositionWithHint(nsIDOMNode* aNode,
|
|||||||
|
|
||||||
// now we have a frame, check whether it's appropriate to show the caret here
|
// now we have a frame, check whether it's appropriate to show the caret here
|
||||||
const nsStyleUserInterface* userinterface = theFrame->GetStyleUserInterface();
|
const nsStyleUserInterface* userinterface = theFrame->GetStyleUserInterface();
|
||||||
if (
|
if ((userinterface->mUserModify == NS_STYLE_USER_MODIFY_READ_ONLY) ||
|
||||||
#ifdef SUPPORT_USER_MODIFY
|
|
||||||
// editable content still defaults to NS_STYLE_USER_MODIFY_READ_ONLY at present. See bug 15284
|
|
||||||
(userinterface->mUserModify == NS_STYLE_USER_MODIFY_READ_ONLY) ||
|
|
||||||
#endif
|
|
||||||
(userinterface->mUserInput == NS_STYLE_USER_INPUT_NONE) ||
|
(userinterface->mUserInput == NS_STYLE_USER_INPUT_NONE) ||
|
||||||
(userinterface->mUserInput == NS_STYLE_USER_INPUT_DISABLED))
|
(userinterface->mUserInput == NS_STYLE_USER_INPUT_DISABLED))
|
||||||
{
|
{
|
||||||
|
|||||||
@ -5737,17 +5737,6 @@ nsIFrame::IsFocusable(PRInt32 *aTabIndex, PRBool aWithMouse)
|
|||||||
const nsStyleVisibility* vis = GetStyleVisibility();
|
const nsStyleVisibility* vis = GetStyleVisibility();
|
||||||
if (vis->mVisible != NS_STYLE_VISIBILITY_COLLAPSE &&
|
if (vis->mVisible != NS_STYLE_VISIBILITY_COLLAPSE &&
|
||||||
vis->mVisible != NS_STYLE_VISIBILITY_HIDDEN) {
|
vis->mVisible != NS_STYLE_VISIBILITY_HIDDEN) {
|
||||||
if (mContent->IsNodeOfType(nsINode::eHTML)) {
|
|
||||||
nsCOMPtr<nsISupports> container(PresContext()->GetContainer());
|
|
||||||
nsCOMPtr<nsIEditorDocShell> editorDocShell(do_QueryInterface(container));
|
|
||||||
if (editorDocShell) {
|
|
||||||
PRBool isEditable;
|
|
||||||
editorDocShell->GetEditable(&isEditable);
|
|
||||||
if (isEditable) {
|
|
||||||
return NS_OK; // Editor content is not focusable
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const nsStyleUserInterface* ui = GetStyleUserInterface();
|
const nsStyleUserInterface* ui = GetStyleUserInterface();
|
||||||
if (ui->mUserFocus != NS_STYLE_USER_FOCUS_IGNORE &&
|
if (ui->mUserFocus != NS_STYLE_USER_FOCUS_IGNORE &&
|
||||||
ui->mUserFocus != NS_STYLE_USER_FOCUS_NONE) {
|
ui->mUserFocus != NS_STYLE_USER_FOCUS_NONE) {
|
||||||
|
|||||||
@ -457,6 +457,9 @@ public:
|
|||||||
*/
|
*/
|
||||||
nsIContent* GetLimiter() { return mLimiter; }
|
nsIContent* GetLimiter() { return mLimiter; }
|
||||||
|
|
||||||
|
nsIContent* GetAncestorLimiter() { return mAncestorLimiter; }
|
||||||
|
void SetAncestorLimiter(nsIContent *aLimiter);
|
||||||
|
|
||||||
/** This will tell the frame selection that a double click has been pressed
|
/** This will tell the frame selection that a double click has been pressed
|
||||||
* so it can track abort future drags if inside the same selection
|
* so it can track abort future drags if inside the same selection
|
||||||
* @aDoubleDown has the double click down happened
|
* @aDoubleDown has the double click down happened
|
||||||
@ -631,6 +634,8 @@ private:
|
|||||||
PRInt32 mBatching;
|
PRInt32 mBatching;
|
||||||
|
|
||||||
nsIContent *mLimiter; //limit selection navigation to a child of this node.
|
nsIContent *mLimiter; //limit selection navigation to a child of this node.
|
||||||
|
nsIContent *mAncestorLimiter; // Limit selection navigation to a descendant of
|
||||||
|
// this node.
|
||||||
nsIPresShell *mShell;
|
nsIPresShell *mShell;
|
||||||
|
|
||||||
PRInt16 mSelectionChangeReason; // reason for notifications of selection changing
|
PRInt16 mSelectionChangeReason; // reason for notifications of selection changing
|
||||||
|
|||||||
@ -664,12 +664,22 @@ IsValidSelectionPoint(nsFrameSelection *aFrameSel, nsIContent *aContent)
|
|||||||
return PR_FALSE;
|
return PR_FALSE;
|
||||||
if (aFrameSel)
|
if (aFrameSel)
|
||||||
{
|
{
|
||||||
nsCOMPtr<nsIContent> tLimiter = aFrameSel->GetLimiter();
|
nsIContent *limiter = aFrameSel->GetLimiter();
|
||||||
if (tLimiter && tLimiter != aContent)
|
if (limiter)
|
||||||
{
|
{
|
||||||
if (tLimiter != aContent->GetParent()) //if newfocus == the limiter. that's ok. but if not there and not parent bad
|
if (limiter != aContent && limiter != aContent->GetParent()) //if newfocus == the limiter. that's ok. but if not there and not parent bad
|
||||||
return PR_FALSE; //not in the right content. tLimiter said so
|
return PR_FALSE; //not in the right content. tLimiter said so
|
||||||
}
|
}
|
||||||
|
limiter = aFrameSel->GetAncestorLimiter();
|
||||||
|
if (limiter)
|
||||||
|
{
|
||||||
|
nsIContent *content = aContent;
|
||||||
|
while (content && content != limiter)
|
||||||
|
{
|
||||||
|
content = content->GetParent();
|
||||||
|
}
|
||||||
|
return content != nsnull;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return PR_TRUE;
|
return PR_TRUE;
|
||||||
}
|
}
|
||||||
@ -825,6 +835,7 @@ nsFrameSelection::nsFrameSelection()
|
|||||||
mChangesDuringBatching = PR_FALSE;
|
mChangesDuringBatching = PR_FALSE;
|
||||||
mNotifyFrames = PR_TRUE;
|
mNotifyFrames = PR_TRUE;
|
||||||
mLimiter = nsnull; //no default limiter.
|
mLimiter = nsnull; //no default limiter.
|
||||||
|
mAncestorLimiter = nsnull;
|
||||||
|
|
||||||
mMouseDoubleDownState = PR_FALSE;
|
mMouseDoubleDownState = PR_FALSE;
|
||||||
|
|
||||||
@ -2220,8 +2231,10 @@ nsFrameSelection::HandleClick(nsIContent *aNewFocus,
|
|||||||
|
|
||||||
InvalidateDesiredX();
|
InvalidateDesiredX();
|
||||||
|
|
||||||
if (!aContinueSelection)
|
if (!aContinueSelection) {
|
||||||
mMaintainRange = nsnull;
|
mMaintainRange = nsnull;
|
||||||
|
mAncestorLimiter = nsnull;
|
||||||
|
}
|
||||||
|
|
||||||
mHint = HINT(aHint);
|
mHint = HINT(aHint);
|
||||||
// Don't take focus when dragging off of a table
|
// Don't take focus when dragging off of a table
|
||||||
@ -2786,6 +2799,9 @@ nsFrameSelection::SelectAll()
|
|||||||
{
|
{
|
||||||
rootContent = mLimiter;//addrefit
|
rootContent = mLimiter;//addrefit
|
||||||
}
|
}
|
||||||
|
else if (mAncestorLimiter) {
|
||||||
|
rootContent = mAncestorLimiter;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
nsIDocument *doc = mShell->GetDocument();
|
nsIDocument *doc = mShell->GetDocument();
|
||||||
@ -2797,7 +2813,7 @@ nsFrameSelection::SelectAll()
|
|||||||
}
|
}
|
||||||
PRInt32 numChildren = rootContent->GetChildCount();
|
PRInt32 numChildren = rootContent->GetChildCount();
|
||||||
PostReason(nsISelectionListener::NO_REASON);
|
PostReason(nsISelectionListener::NO_REASON);
|
||||||
return TakeFocus(mLimiter, 0, numChildren, PR_FALSE, PR_FALSE);
|
return TakeFocus(rootContent, 0, numChildren, PR_FALSE, PR_FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
//////////END FRAMESELECTION
|
//////////END FRAMESELECTION
|
||||||
@ -3823,6 +3839,23 @@ nsFrameSelection::CreateAndAddRange(nsIDOMNode *aParentNode, PRInt32 aOffset)
|
|||||||
|
|
||||||
// End of Table Selection
|
// End of Table Selection
|
||||||
|
|
||||||
|
void
|
||||||
|
nsFrameSelection::SetAncestorLimiter(nsIContent *aLimiter)
|
||||||
|
{
|
||||||
|
if (mAncestorLimiter != aLimiter) {
|
||||||
|
mAncestorLimiter = aLimiter;
|
||||||
|
PRInt8 index =
|
||||||
|
GetIndexFromSelectionType(nsISelectionController::SELECTION_NORMAL);
|
||||||
|
if (!IsValidSelectionPoint(this, mDomSelections[index]->FetchFocusNode())) {
|
||||||
|
ClearNormalSelection();
|
||||||
|
if (mAncestorLimiter) {
|
||||||
|
PostReason(nsISelectionListener::NO_REASON);
|
||||||
|
TakeFocus(mAncestorLimiter, 0, 0, PR_FALSE, PR_FALSE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//END nsFrameSelection methods
|
//END nsFrameSelection methods
|
||||||
|
|
||||||
|
|
||||||
@ -5291,6 +5324,13 @@ nsTypedSelection::GetFrameSelection(nsFrameSelection **aFrameSelection) {
|
|||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NS_IMETHODIMP
|
||||||
|
nsTypedSelection::SetAncestorLimiter(nsIContent *aContent)
|
||||||
|
{
|
||||||
|
mFrameSelection->SetAncestorLimiter(aContent);
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
nsresult
|
nsresult
|
||||||
nsTypedSelection::StartAutoScrollTimer(nsPresContext *aPresContext,
|
nsTypedSelection::StartAutoScrollTimer(nsPresContext *aPresContext,
|
||||||
nsIView *aView,
|
nsIView *aView,
|
||||||
|
|||||||
@ -178,6 +178,8 @@ _FILES = \
|
|||||||
viewsource.css \
|
viewsource.css \
|
||||||
arrow.gif \
|
arrow.gif \
|
||||||
arrowd.gif \
|
arrowd.gif \
|
||||||
|
contenteditable.css \
|
||||||
|
designmode.css \
|
||||||
$(NULL)
|
$(NULL)
|
||||||
|
|
||||||
GARBAGE += $(addprefix $(DIST)/bin/res/,$(_FILES))
|
GARBAGE += $(addprefix $(DIST)/bin/res/,$(_FILES))
|
||||||
|
|||||||
@ -134,6 +134,11 @@ input > .anonymous-div {
|
|||||||
ime-mode: inherit;
|
ime-mode: inherit;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
input:-moz-read-write,
|
||||||
|
textarea:-moz-read-write {
|
||||||
|
-moz-user-modify: read-write !important;
|
||||||
|
}
|
||||||
|
|
||||||
select {
|
select {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
border-color: ThreeDFace;
|
border-color: ThreeDFace;
|
||||||
|
|||||||
@ -64,7 +64,7 @@ find-menu appears in editor element which has had makeEditable() called but desi
|
|||||||
var webnav = gBrowser.webNavigation;
|
var webnav = gBrowser.webNavigation;
|
||||||
var edsession = webnav.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
|
var edsession = webnav.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
|
||||||
.getInterface(Components.interfaces.nsIEditingSession);
|
.getInterface(Components.interfaces.nsIEditingSession);
|
||||||
edsession.makeWindowEditable(gBrowser.contentWindow, "html", false);
|
edsession.makeWindowEditable(gBrowser.contentWindow, "html", false, true, false);
|
||||||
gBrowser.contentWindow.focus();
|
gBrowser.contentWindow.focus();
|
||||||
enterStringIntoEditor("'");
|
enterStringIntoEditor("'");
|
||||||
enterStringIntoEditor("/");
|
enterStringIntoEditor("/");
|
||||||
|
|||||||
@ -66,7 +66,7 @@
|
|||||||
<parameter name="waitForUrlLoad"/>
|
<parameter name="waitForUrlLoad"/>
|
||||||
<body>
|
<body>
|
||||||
<![CDATA[
|
<![CDATA[
|
||||||
this.editingSession.makeWindowEditable(this.contentWindow, editortype, waitForUrlLoad);
|
this.editingSession.makeWindowEditable(this.contentWindow, editortype, waitForUrlLoad, true, false);
|
||||||
this.setAttribute("editortype", editortype);
|
this.setAttribute("editortype", editortype);
|
||||||
|
|
||||||
this.docShell.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
|
this.docShell.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
|
||||||
|
|||||||
@ -66,7 +66,7 @@
|
|||||||
<parameter name="waitForUrlLoad"/>
|
<parameter name="waitForUrlLoad"/>
|
||||||
<body>
|
<body>
|
||||||
<![CDATA[
|
<![CDATA[
|
||||||
this.editingSession.makeWindowEditable(this.contentWindow, editortype, waitForUrlLoad);
|
this.editingSession.makeWindowEditable(this.contentWindow, editortype, waitForUrlLoad, true, false);
|
||||||
this.setAttribute("editortype", editortype);
|
this.setAttribute("editortype", editortype);
|
||||||
|
|
||||||
this.docShell.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
|
this.docShell.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user