gecko-dev/editor/base/nsHTMLEditRules.cpp

927 lines
26 KiB
C++
Raw Normal View History

1999-03-29 06:21:01 +00:00
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL") you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
* Reserved.
*/
#include "nsHTMLEditRules.h"
#include "nsEditor.h"
#include "nsTextEditor.h"
1999-03-29 06:21:01 +00:00
#include "PlaceholderTxn.h"
#include "InsertTextTxn.h"
#include "nsIContent.h"
#include "nsIContentIterator.h"
1999-03-29 06:21:01 +00:00
#include "nsIDOMNode.h"
#include "nsIDOMText.h"
1999-03-29 06:21:01 +00:00
#include "nsIDOMElement.h"
#include "nsIDOMNodeList.h"
#include "nsIDOMSelection.h"
#include "nsIDOMRange.h"
#include "nsIDOMCharacterData.h"
#include "nsIEnumerator.h"
#include "nsIStyleContext.h"
#include "nsIPresShell.h"
#include "nsLayoutCID.h"
class nsIFrame;
1999-03-29 06:21:01 +00:00
//const static char* kMOZEditorBogusNodeAttr="MOZ_EDITOR_BOGUS_NODE";
//const static char* kMOZEditorBogusNodeValue="TRUE";
const unsigned char nbsp = 160;
1999-03-29 06:21:01 +00:00
static NS_DEFINE_IID(kPlaceholderTxnIID, PLACEHOLDER_TXN_IID);
static NS_DEFINE_CID(kCContentIteratorCID, NS_CONTENTITERATOR_CID);
1999-03-29 06:21:01 +00:00
1999-04-05 17:21:59 +00:00
/********************************************************
* Constructor/Destructor
********************************************************/
1999-03-29 06:21:01 +00:00
nsHTMLEditRules::nsHTMLEditRules()
{
}
nsHTMLEditRules::~nsHTMLEditRules()
{
}
1999-04-05 17:21:59 +00:00
/********************************************************
* Public methods
********************************************************/
NS_IMETHODIMP
1999-04-12 12:01:32 +00:00
nsHTMLEditRules::WillDoAction(nsIDOMSelection *aSelection,
nsRulesInfo *aInfo, PRBool *aCancel)
1999-04-05 17:21:59 +00:00
{
if (!aSelection || !aInfo || !aCancel)
1999-04-05 17:21:59 +00:00
return NS_ERROR_NULL_POINTER;
*aCancel = PR_FALSE;
1999-04-05 17:21:59 +00:00
1999-04-12 12:01:32 +00:00
// my kingdom for dynamic cast
nsTextRulesInfo *info = NS_STATIC_CAST(nsTextRulesInfo*, aInfo);
switch (info->action)
1999-04-05 17:21:59 +00:00
{
1999-04-12 12:01:32 +00:00
case kInsertText:
return WillInsertText(aSelection,
aCancel,
1999-04-26 14:08:52 +00:00
info->placeTxn,
1999-04-12 12:01:32 +00:00
info->inString,
info->outString,
info->typeInState,
info->maxLength);
1999-04-05 17:21:59 +00:00
case kInsertBreak:
return WillInsertBreak(aSelection, aCancel);
1999-04-26 14:08:52 +00:00
case kDeleteSelection:
return WillDeleteSelection(aSelection, info->collapsedAction, aCancel);
case kMakeList:
return WillMakeList(aSelection, aCancel);
case kIndent:
return WillIndent(aSelection, aCancel);
case kOutdent:
return WillOutdent(aSelection, aCancel);
case kAlign:
return WillAlign(aSelection, aCancel);
case kMakeHeader:
return WillMakeHeader(aSelection, aCancel);
case kMakeAddress:
return WillMakeAddress(aSelection, aCancel);
case kMakePRE:
return WillMakePRE(aSelection, aCancel);
1999-04-05 17:21:59 +00:00
}
1999-04-12 12:01:32 +00:00
return nsTextEditRules::WillDoAction(aSelection, aInfo, aCancel);
1999-04-05 17:21:59 +00:00
}
1999-05-26 22:44:08 +00:00
1999-04-05 17:21:59 +00:00
NS_IMETHODIMP
1999-04-12 12:01:32 +00:00
nsHTMLEditRules::DidDoAction(nsIDOMSelection *aSelection,
nsRulesInfo *aInfo, nsresult aResult)
1999-04-05 17:21:59 +00:00
{
1999-05-26 22:44:08 +00:00
// my kingdom for dynamic cast
nsTextRulesInfo *info = NS_STATIC_CAST(nsTextRulesInfo*, aInfo);
switch (info->action)
{
case kInsertBreak:
return NS_OK;
}
1999-04-12 12:01:32 +00:00
return nsTextEditRules::DidDoAction(aSelection, aInfo, aResult);
1999-04-05 17:21:59 +00:00
}
/********************************************************
1999-04-12 12:01:32 +00:00
* Protected rules methods
1999-04-05 17:21:59 +00:00
********************************************************/
1999-04-12 12:01:32 +00:00
nsresult
nsHTMLEditRules::WillInsertText(nsIDOMSelection *aSelection,
1999-04-12 12:01:32 +00:00
PRBool *aCancel,
PlaceholderTxn **aTxn,
const nsString *inString,
nsString *outString,
TypeInState typeInState,
PRInt32 aMaxLength)
1999-04-12 12:01:32 +00:00
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
// XXX - need to handle strings of length >1 with embedded tabs or spaces
1999-04-26 14:08:52 +00:00
// XXX - what about embedded returns?
1999-04-12 12:01:32 +00:00
// is it a tab?
if (*inString == "\t" )
return InsertTab(aSelection,aCancel,aTxn,outString);
// is it a space?
if (*inString == " ")
return InsertSpace(aSelection,aCancel,aTxn,outString);
// otherwise, return nsTextEditRules version
return nsTextEditRules::WillInsertText(aSelection,
aCancel,
aTxn,
inString,
outString,
typeInState,
aMaxLength);
1999-04-12 12:01:32 +00:00
}
nsresult
1999-03-29 06:21:01 +00:00
nsHTMLEditRules::WillInsertBreak(nsIDOMSelection *aSelection, PRBool *aCancel)
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
1999-04-26 14:08:52 +00:00
nsresult res;
// if the selection isn't collapsed, delete it.
PRBool bCollapsed;
res = aSelection->GetIsCollapsed(&bCollapsed);
if (NS_FAILED(res)) return res;
if (!bCollapsed)
{
res = mEditor->DeleteSelection(nsIEditor::eDoNothing);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
}
//smart splitting rules
nsCOMPtr<nsIDOMNode> node;
PRInt32 offset;
PRBool isPRE;
1999-05-28 21:17:30 +00:00
res = mEditor->GetStartNodeAndOffset(aSelection, &node, &offset);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
if (!node) return NS_ERROR_FAILURE;
1999-05-28 21:17:30 +00:00
res = mEditor->IsPreformatted(node,&isPRE);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
if (isPRE)
{
nsString theString = "\n";
*aCancel = PR_TRUE;
return mEditor->InsertText(theString);
}
1999-05-28 21:17:30 +00:00
nsCOMPtr<nsIDOMNode> blockParent = mEditor->GetBlockNodeParent(node);
1999-04-26 14:08:52 +00:00
if (!blockParent) return NS_ERROR_FAILURE;
// headers: put selection after the header
if (IsHeader(blockParent))
{
res = ReturnInHeader(aSelection, blockParent, node, offset);
*aCancel = PR_TRUE;
return NS_OK;
}
// paragraphs: special rules to look for <br>s
if (IsParagraph(blockParent))
{
res = ReturnInParagraph(aSelection, blockParent, node, offset, aCancel);
return NS_OK;
}
// list items: special rules to make new list items
if (IsListItem(blockParent))
{
res = ReturnInListItem(aSelection, blockParent, node, offset);
*aCancel = PR_TRUE;
return NS_OK;
}
1999-03-29 06:21:01 +00:00
return WillInsert(aSelection, aCancel);
}
1999-04-26 14:08:52 +00:00
1999-04-12 12:01:32 +00:00
nsresult
nsHTMLEditRules::WillDeleteSelection(nsIDOMSelection *aSelection, nsIEditor::ECollapsedSelectionAction aAction, PRBool *aCancel)
1999-04-26 14:08:52 +00:00
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
nsresult res = NS_OK;
PRBool bCollapsed;
res = aSelection->GetIsCollapsed(&bCollapsed);
if (NS_FAILED(res)) return res;
nsCOMPtr<nsIDOMNode> node;
PRInt32 offset;
1999-05-28 21:17:30 +00:00
res = mEditor->GetStartNodeAndOffset(aSelection, &node, &offset);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
if (!node) return NS_ERROR_FAILURE;
if (bCollapsed)
{
// easy case, in a text node:
1999-05-28 21:17:30 +00:00
if (mEditor->IsTextNode(node))
1999-03-29 06:21:01 +00:00
{
1999-04-26 14:08:52 +00:00
nsCOMPtr<nsIDOMText> textNode = do_QueryInterface(node);
PRUint32 strLength;
res = textNode->GetLength(&strLength);
if (NS_FAILED(res)) return res;
// at beginning of text node and backspaced?
if (!offset && (aAction == nsIEditor::eDeleteLeft))
1999-04-26 14:08:52 +00:00
{
nsCOMPtr<nsIDOMNode> priorNode;
1999-05-28 21:17:30 +00:00
res = mEditor->GetPriorNode(node, PR_TRUE, getter_AddRefs(priorNode));
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
// if there is no prior node then cancel the deletion
if (!priorNode)
{
*aCancel = PR_TRUE;
return res;
}
1999-04-26 14:08:52 +00:00
// block parents the same? use defaul deletion
1999-05-28 21:17:30 +00:00
if (mEditor->HasSameBlockNodeParent(node, priorNode)) return NS_OK;
1999-04-26 14:08:52 +00:00
// deleting across blocks
// are the blocks of same type?
1999-05-28 21:17:30 +00:00
nsCOMPtr<nsIDOMNode> leftParent = mEditor->GetBlockNodeParent(priorNode);
nsCOMPtr<nsIDOMNode> rightParent = mEditor->GetBlockNodeParent(node);
nsCOMPtr<nsIAtom> leftAtom = mEditor->GetTag(leftParent);
nsCOMPtr<nsIAtom> rightAtom = mEditor->GetTag(rightParent);
1999-04-26 14:08:52 +00:00
if (leftAtom.get() == rightAtom.get())
{
nsCOMPtr<nsIDOMNode> topParent;
leftParent->GetParentNode(getter_AddRefs(topParent));
if (IsParagraph(leftParent))
{
// join para's, insert break
*aCancel = PR_TRUE;
1999-05-28 21:17:30 +00:00
res = mEditor->JoinNodeDeep(leftParent,rightParent,aSelection);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
res = mEditor->InsertBreak();
return res;
}
if (IsListItem(leftParent) || IsHeader(leftParent))
{
// join blocks
*aCancel = PR_TRUE;
1999-05-28 21:17:30 +00:00
res = mEditor->JoinNodeDeep(leftParent,rightParent,aSelection);
1999-04-26 14:08:52 +00:00
return res;
}
}
// else blocks not same type, bail to default
return NS_OK;
}
// at end of text node and deleted?
if ((offset == strLength) && (aAction == nsIEditor::eDeleteRight))
1999-03-29 06:21:01 +00:00
{
1999-04-26 14:08:52 +00:00
nsCOMPtr<nsIDOMNode> nextNode;
1999-05-28 21:17:30 +00:00
res = mEditor->GetNextNode(node, PR_TRUE, getter_AddRefs(nextNode));
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
// if there is no next node then cancel the deletion
if (!nextNode)
{
*aCancel = PR_TRUE;
return res;
}
1999-04-26 14:08:52 +00:00
// block parents the same? use defaul deletion
1999-05-28 21:17:30 +00:00
if (mEditor->HasSameBlockNodeParent(node, nextNode)) return NS_OK;
1999-04-26 14:08:52 +00:00
// deleting across blocks
// are the blocks of same type?
1999-05-28 21:17:30 +00:00
nsCOMPtr<nsIDOMNode> leftParent = mEditor->GetBlockNodeParent(node);
nsCOMPtr<nsIDOMNode> rightParent = mEditor->GetBlockNodeParent(nextNode);
nsCOMPtr<nsIAtom> leftAtom = mEditor->GetTag(leftParent);
nsCOMPtr<nsIAtom> rightAtom = mEditor->GetTag(rightParent);
1999-04-26 14:08:52 +00:00
if (leftAtom.get() == rightAtom.get())
{
nsCOMPtr<nsIDOMNode> topParent;
leftParent->GetParentNode(getter_AddRefs(topParent));
if (IsParagraph(leftParent))
{
// join para's, insert break
*aCancel = PR_TRUE;
1999-05-28 21:17:30 +00:00
res = mEditor->JoinNodeDeep(leftParent,rightParent,aSelection);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
res = mEditor->InsertBreak();
return res;
}
if (IsListItem(leftParent) || IsHeader(leftParent))
{
// join blocks
*aCancel = PR_TRUE;
1999-05-28 21:17:30 +00:00
res = mEditor->JoinNodeDeep(leftParent,rightParent,aSelection);
1999-04-26 14:08:52 +00:00
return res;
}
}
// else blocks not same type, bail to default
return NS_OK;
1999-03-29 06:21:01 +00:00
}
1999-04-26 14:08:52 +00:00
// else do default
return NS_OK;
1999-03-29 06:21:01 +00:00
}
}
1999-04-26 14:08:52 +00:00
// else we have a non collapsed selection
// figure out if the enpoints are in nodes that can be merged
nsCOMPtr<nsIDOMNode> endNode;
PRInt32 endOffset;
1999-05-28 21:17:30 +00:00
res = mEditor->GetEndNodeAndOffset(aSelection, &endNode, &endOffset);
1999-04-26 14:08:52 +00:00
if (endNode.get() != node.get())
{
// block parents the same? use defaul deletion
1999-05-28 21:17:30 +00:00
if (mEditor->HasSameBlockNodeParent(node, endNode)) return NS_OK;
1999-04-26 14:08:52 +00:00
// deleting across blocks
// are the blocks of same type?
1999-05-28 21:17:30 +00:00
nsCOMPtr<nsIDOMNode> leftParent = mEditor->GetBlockNodeParent(node);
nsCOMPtr<nsIDOMNode> rightParent = mEditor->GetBlockNodeParent(endNode);
1999-04-26 14:08:52 +00:00
// are the blocks siblings?
nsCOMPtr<nsIDOMNode> leftBlockParent;
nsCOMPtr<nsIDOMNode> rightBlockParent;
leftParent->GetParentNode(getter_AddRefs(leftBlockParent));
rightParent->GetParentNode(getter_AddRefs(rightBlockParent));
// bail to default if blocks aren't siblings
if (leftBlockParent.get() != rightBlockParent.get()) return NS_OK;
1999-05-28 21:17:30 +00:00
nsCOMPtr<nsIAtom> leftAtom = mEditor->GetTag(leftParent);
nsCOMPtr<nsIAtom> rightAtom = mEditor->GetTag(rightParent);
1999-04-26 14:08:52 +00:00
if (leftAtom.get() == rightAtom.get())
{
nsCOMPtr<nsIDOMNode> topParent;
leftParent->GetParentNode(getter_AddRefs(topParent));
if (IsParagraph(leftParent))
{
// first delete the selection
*aCancel = PR_TRUE;
res = mEditor->nsEditor::DeleteSelection(aAction);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
// then join para's, insert break
1999-05-28 21:17:30 +00:00
res = mEditor->JoinNodeDeep(leftParent,rightParent,aSelection);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
1999-05-04 08:03:07 +00:00
//res = mEditor->InsertBreak();
// uhh, no, we don't want to have the <br> on a deleted selction across para's
1999-04-26 14:08:52 +00:00
return res;
}
if (IsListItem(leftParent) || IsHeader(leftParent))
{
// first delete the selection
*aCancel = PR_TRUE;
res = mEditor->nsEditor::DeleteSelection(aAction);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
// join blocks
1999-05-28 21:17:30 +00:00
res = mEditor->JoinNodeDeep(leftParent,rightParent,aSelection);
1999-04-26 14:08:52 +00:00
return res;
}
}
// else blocks not same type, bail to default
return NS_OK;
}
1999-04-12 12:01:32 +00:00
1999-04-26 14:08:52 +00:00
return res;
}
1999-04-12 12:01:32 +00:00
nsresult
nsHTMLEditRules::WillMakeList(nsIDOMSelection *aSelection, PRBool *aCancel)
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
nsresult res = NS_OK;
return res;
}
nsresult
nsHTMLEditRules::WillIndent(nsIDOMSelection *aSelection, PRBool *aCancel)
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
nsresult res = NS_OK;
return res;
}
nsresult
nsHTMLEditRules::WillOutdent(nsIDOMSelection *aSelection, PRBool *aCancel)
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
nsresult res = NS_OK;
return res;
}
nsresult
nsHTMLEditRules::WillAlign(nsIDOMSelection *aSelection, PRBool *aCancel)
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
nsresult res = NS_OK;
return res;
}
nsresult
nsHTMLEditRules::WillMakeHeader(nsIDOMSelection *aSelection, PRBool *aCancel)
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
nsresult res = NS_OK;
return res;
}
nsresult
nsHTMLEditRules::WillMakeAddress(nsIDOMSelection *aSelection, PRBool *aCancel)
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
nsresult res = NS_OK;
return res;
}
nsresult
nsHTMLEditRules::WillMakePRE(nsIDOMSelection *aSelection, PRBool *aCancel)
{
if (!aSelection || !aCancel) { return NS_ERROR_NULL_POINTER; }
// initialize out param
*aCancel = PR_FALSE;
nsresult res = NS_OK;
return res;
}
1999-04-12 12:01:32 +00:00
/********************************************************
* helper methods
********************************************************/
1999-04-26 14:08:52 +00:00
///////////////////////////////////////////////////////////////////////////
// IsHeader: true if node an html header
//
PRBool
nsHTMLEditRules::IsHeader(nsIDOMNode *node)
{
NS_PRECONDITION(node, "null parent passed to nsHTMLEditRules::IsHeader");
1999-05-28 21:17:30 +00:00
nsAutoString tag;
nsEditor::GetTagString(node,tag);
if ( (tag == "h1") ||
(tag == "h2") ||
(tag == "h3") ||
(tag == "h4") ||
(tag == "h5") ||
(tag == "h6") )
1999-04-26 14:08:52 +00:00
{
return PR_TRUE;
}
return PR_FALSE;
}
///////////////////////////////////////////////////////////////////////////
// IsParagraph: true if node an html paragraph
//
PRBool
nsHTMLEditRules::IsParagraph(nsIDOMNode *node)
{
NS_PRECONDITION(node, "null parent passed to nsHTMLEditRules::IsParagraph");
1999-05-28 21:17:30 +00:00
nsAutoString tag;
nsEditor::GetTagString(node,tag);
if (tag == "p")
1999-04-26 14:08:52 +00:00
{
return PR_TRUE;
}
return PR_FALSE;
}
///////////////////////////////////////////////////////////////////////////
// IsListItem: true if node an html list item
//
PRBool
nsHTMLEditRules::IsListItem(nsIDOMNode *node)
{
NS_PRECONDITION(node, "null parent passed to nsHTMLEditRules::IsListItem");
1999-05-28 21:17:30 +00:00
nsAutoString tag;
nsEditor::GetTagString(node,tag);
if (tag == "li")
1999-04-26 14:08:52 +00:00
{
return PR_TRUE;
}
return PR_FALSE;
}
///////////////////////////////////////////////////////////////////////////
1999-05-28 21:17:30 +00:00
// IsBreak: true if node an html break node
1999-04-26 14:08:52 +00:00
//
PRBool
nsHTMLEditRules::IsBreak(nsIDOMNode *node)
{
NS_PRECONDITION(node, "null parent passed to nsHTMLEditRules::IsBreak");
1999-05-28 21:17:30 +00:00
nsAutoString tag;
nsEditor::GetTagString(node,tag);
if (tag == "br")
1999-04-26 14:08:52 +00:00
{
return PR_TRUE;
}
return PR_FALSE;
}
///////////////////////////////////////////////////////////////////////////
// GetTabAsNBSPs: stuff the right number of nbsp's into outString
//
nsresult
nsHTMLEditRules::GetTabAsNBSPs(nsString *outString)
{
if (!outString) return NS_ERROR_NULL_POINTER;
// XXX - this should get the right number from prefs
*outString += nbsp; *outString += nbsp; *outString += nbsp; *outString += nbsp;
return NS_OK;
}
///////////////////////////////////////////////////////////////////////////
// GetTabAsNBSPsAndSpace: stuff the right number of nbsp's followed by a
// space into outString
nsresult
nsHTMLEditRules::GetTabAsNBSPsAndSpace(nsString *outString)
{
if (!outString) return NS_ERROR_NULL_POINTER;
// XXX - this should get the right number from prefs
*outString += nbsp; *outString += nbsp; *outString += nbsp; *outString += ' ';
return NS_OK;
}
/********************************************************
* main implemntation methods
********************************************************/
///////////////////////////////////////////////////////////////////////////
// InsertTab: top level logic for determining how to insert a tab
//
1999-04-12 12:01:32 +00:00
nsresult
nsHTMLEditRules::InsertTab(nsIDOMSelection *aSelection,
PRBool *aCancel,
PlaceholderTxn **aTxn,
nsString *outString)
{
nsCOMPtr<nsIDOMNode> parentNode;
PRInt32 offset;
PRBool isPRE;
PRBool isNextWhiteSpace;
PRBool isPrevWhiteSpace;
1999-05-28 21:17:30 +00:00
nsresult result = mEditor->GetStartNodeAndOffset(aSelection, &parentNode, &offset);
if (NS_FAILED(result)) return result;
if (!parentNode) return NS_ERROR_FAILURE;
1999-05-28 21:17:30 +00:00
result = mEditor->IsPreformatted(parentNode,&isPRE);
if (NS_FAILED(result)) return result;
1999-04-12 12:01:32 +00:00
if (isPRE)
{
*outString += '\t';
return NS_OK;
1999-04-12 12:01:32 +00:00
}
1999-05-28 21:17:30 +00:00
result = mEditor->IsNextCharWhitespace(parentNode, offset, &isNextWhiteSpace);
if (NS_FAILED(result)) return result;
1999-05-28 21:17:30 +00:00
result = mEditor->IsPrevCharWhitespace(parentNode, offset, &isPrevWhiteSpace);
if (NS_FAILED(result)) return result;
if (isPrevWhiteSpace)
1999-04-12 12:01:32 +00:00
{
// prev character is a whitespace; Need to
// insert nbsp's BEFORE the space
// XXX for now put tab in wrong place
if (isNextWhiteSpace)
{
GetTabAsNBSPs(outString);
return NS_OK;
}
GetTabAsNBSPsAndSpace(outString);
return NS_OK;
}
if (isNextWhiteSpace)
{
// character after us is ws. insert nbsps
GetTabAsNBSPs(outString);
return NS_OK;
1999-04-12 12:01:32 +00:00
}
// else we are in middle of a word; use n-1 nbsp's plus a space
GetTabAsNBSPsAndSpace(outString);
1999-04-12 12:01:32 +00:00
return NS_OK;
}
///////////////////////////////////////////////////////////////////////////
// InsertSpace: top level logic for determining how to insert a space
//
1999-04-12 12:01:32 +00:00
nsresult
nsHTMLEditRules::InsertSpace(nsIDOMSelection *aSelection,
PRBool *aCancel,
PlaceholderTxn **aTxn,
nsString *outString)
{
nsCOMPtr<nsIDOMNode> parentNode;
PRInt32 offset;
PRBool isPRE;
PRBool isNextWhiteSpace;
PRBool isPrevWhiteSpace;
1999-05-28 21:17:30 +00:00
nsresult result = mEditor->GetStartNodeAndOffset(aSelection, &parentNode, &offset);
if (NS_FAILED(result)) return result;
if (!parentNode) return NS_ERROR_FAILURE;
1999-05-28 21:17:30 +00:00
result = mEditor->IsPreformatted(parentNode,&isPRE);
if (NS_FAILED(result)) return result;
if (isPRE)
{
*outString += " ";
return NS_OK;
}
1999-05-28 21:17:30 +00:00
result = mEditor->IsNextCharWhitespace(parentNode, offset, &isNextWhiteSpace);
if (NS_FAILED(result)) return result;
1999-05-28 21:17:30 +00:00
result = mEditor->IsPrevCharWhitespace(parentNode, offset, &isPrevWhiteSpace);
if (NS_FAILED(result)) return result;
if (isPrevWhiteSpace)
{
// prev character is a whitespace; Need to
// insert nbsp BEFORE the space
// XXX for now put in wrong place
if (isNextWhiteSpace)
{
*outString += nbsp;
return NS_OK;
}
*outString += nbsp;
return NS_OK;
}
if (isNextWhiteSpace)
{
// character after us is ws. insert nbsp
*outString += nbsp;
return NS_OK;
}
// else just a space
*outString += " ";
1999-04-12 12:01:32 +00:00
return NS_OK;
}
1999-04-26 14:08:52 +00:00
///////////////////////////////////////////////////////////////////////////
// ReturnInHeader: do the right thing for returns pressed in headers
//
nsresult
nsHTMLEditRules::ReturnInHeader(nsIDOMSelection *aSelection,
nsIDOMNode *aHeader,
nsIDOMNode *aTextNode,
PRInt32 aOffset)
{
if (!aSelection || !aHeader || !aTextNode) return NS_ERROR_NULL_POINTER;
nsCOMPtr<nsIDOMNode> leftNode;
nsCOMPtr<nsIDOMNode> textNode = do_QueryInterface(aTextNode); // to hold a ref across the delete call
// split the node
nsresult res = mEditor->SplitNode(aTextNode, aOffset, getter_AddRefs(leftNode));
if (NS_FAILED(res)) return res;
// move the right node outside of the header, via deletion/insertion
// delete the right node
res = mEditor->DeleteNode(textNode);
if (NS_FAILED(res)) return res;
// insert the right node
nsCOMPtr<nsIDOMNode> p;
aHeader->GetParentNode(getter_AddRefs(p));
1999-05-28 21:17:30 +00:00
PRInt32 indx = mEditor->GetIndexOf(p,aHeader);
1999-04-26 14:08:52 +00:00
res = mEditor->InsertNode(textNode,p,indx+1);
if (NS_FAILED(res)) return res;
// merge text node with like sibling, if any
nsCOMPtr<nsIDOMNode> sibling;
textNode->GetNextSibling(getter_AddRefs(sibling));
if (sibling)
{
res = mEditor->JoinNodes(textNode,sibling,p);
if (NS_FAILED(res)) return res;
textNode = sibling; // sibling was the node kept by the join; remember it in "textNode"
}
// position selection before inserted node
res = aSelection->Collapse(textNode,0);
return res;
}
///////////////////////////////////////////////////////////////////////////
// ReturnInParagraph: do the right thing for returns pressed in paragraphs
//
nsresult
nsHTMLEditRules::ReturnInParagraph(nsIDOMSelection *aSelection,
nsIDOMNode *aHeader,
nsIDOMNode *aNode,
PRInt32 aOffset,
PRBool *aCancel)
{
if (!aSelection || !aHeader || !aNode || !aCancel) return NS_ERROR_NULL_POINTER;
*aCancel = PR_FALSE;
nsCOMPtr<nsIDOMNode> sibling;
nsresult res = NS_OK;
// easy case, in a text node:
1999-05-28 21:17:30 +00:00
if (mEditor->IsTextNode(aNode))
1999-04-26 14:08:52 +00:00
{
nsCOMPtr<nsIDOMText> textNode = do_QueryInterface(aNode);
PRUint32 strLength;
res = textNode->GetLength(&strLength);
if (NS_FAILED(res)) return res;
// at beginning of text node?
if (!aOffset)
{
// is there a BR prior to it?
aNode->GetPreviousSibling(getter_AddRefs(sibling));
if (!sibling)
{
// no previous sib, so
// just fall out to default of inserting a BR
return res;
}
if (IsBreak(sibling))
{
*aCancel = PR_TRUE;
// get rid of the break
res = mEditor->DeleteNode(sibling);
if (NS_FAILED(res)) return res;
// split the paragraph
1999-05-28 21:17:30 +00:00
res = mEditor->SplitNodeDeep( aHeader, aNode, aOffset);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
// position selection inside textnode
res = aSelection->Collapse(aNode,0);
}
// else just fall out to default of inserting a BR
return res;
}
// at end of text node?
if (aOffset == strLength)
{
// is there a BR after to it?
res = aNode->GetNextSibling(getter_AddRefs(sibling));
if (!sibling)
{
// no next sib, so
// just fall out to default of inserting a BR
return res;
}
if (IsBreak(sibling))
{
*aCancel = PR_TRUE;
// get rid of the break
res = mEditor->DeleteNode(sibling);
if (NS_FAILED(res)) return res;
// split the paragraph
1999-05-28 21:17:30 +00:00
res = mEditor->SplitNodeDeep( aHeader, aNode, aOffset);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
// position selection inside textnode
res = aSelection->Collapse(aNode,0);
}
// else just fall out to default of inserting a BR
return res;
}
// inside text node
// just fall out to default of inserting a BR
return res;
}
// not in a text node. are we next to BR's?
// XXX
return res;
}
///////////////////////////////////////////////////////////////////////////
// ReturnInListItem: do the right thing for returns pressed in list items
//
nsresult
nsHTMLEditRules::ReturnInListItem(nsIDOMSelection *aSelection,
nsIDOMNode *aListItem,
nsIDOMNode *aNode,
PRInt32 aOffset)
{
if (!aSelection || !aListItem || !aNode) return NS_ERROR_NULL_POINTER;
1999-05-28 21:17:30 +00:00
nsresult res = mEditor->SplitNodeDeep( aListItem, aNode, aOffset);
1999-04-26 14:08:52 +00:00
if (NS_FAILED(res)) return res;
res = aSelection->Collapse(aNode,0);
return res;
}
1999-04-12 12:01:32 +00:00