/* -*- 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 "License"); you may not use this file except in * compliance with the License. You may obtain a copy of the License at * http://www.mozilla.org/NPL/ * * Software distributed under the License is distributed on an "AS IS" * basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See * the License for the specific language governing rights and limitations * under the License. * * The Original Code is Mozilla Communicator client code. * * The Initial Developer of the Original Code is Netscape Communications * Corporation. Portions created by Netscape are Copyright (C) 1998 * Netscape Communications Corporation. All Rights Reserved. */ #include "nsIDOMComment.h" #include "nsGenericDOMDataNode.h" #include "nsIScriptObjectOwner.h" #include "nsIDOMEventReceiver.h" #include "nsIContent.h" #include "nsFrame.h" #include "nsLayoutAtoms.h" #include "nsIDOMSelection.h" #include "nsXIFConverter.h" #include "nsIDocument.h" #include "nsIEnumerator.h" #include "nsCOMPtr.h" #include "nsIDOMRange.h" static NS_DEFINE_IID(kIDOMCommentIID, NS_IDOMCOMMENT_IID); static NS_DEFINE_IID(kIEnumeratorIID, NS_IENUMERATOR_IID); static NS_DEFINE_IID(kITextContentIID, NS_ITEXT_CONTENT_IID); class nsCommentNode : public nsIDOMComment, public nsIScriptObjectOwner, public nsIDOMEventReceiver, public nsIContent, public nsITextContent { public: nsCommentNode(); virtual ~nsCommentNode(); // nsISupports NS_DECL_ISUPPORTS // nsIDOMNode NS_IMPL_IDOMNODE_USING_GENERIC_DOM_DATA(mInner) // nsIDOMCharacterData NS_IMPL_IDOMCHARACTERDATA_USING_GENERIC_DOM_DATA(mInner) // nsIDOMComment // nsIScriptObjectOwner NS_IMPL_ISCRIPTOBJECTOWNER_USING_GENERIC_DOM_DATA(mInner) // nsIDOMEventReceiver NS_IMPL_IDOMEVENTRECEIVER_USING_GENERIC_DOM_DATA(mInner) // nsIContent //NS_IMPL_ICONTENT_USING_GENERIC_DOM_DATA(mInner) NS_IMETHOD GetDocument(nsIDocument*& aResult) const { return mInner.GetDocument(aResult); } NS_IMETHOD SetDocument(nsIDocument* aDocument, PRBool aDeep) { return mInner.SetDocument(aDocument, aDeep); } NS_IMETHOD GetParent(nsIContent*& aResult) const { return mInner.GetParent(aResult); } NS_IMETHOD SetParent(nsIContent* aParent) { return mInner.SetParent(aParent); } NS_IMETHOD CanContainChildren(PRBool& aResult) const { return mInner.CanContainChildren(aResult); } NS_IMETHOD ChildCount(PRInt32& aResult) const { return mInner.ChildCount(aResult); } NS_IMETHOD ChildAt(PRInt32 aIndex, nsIContent*& aResult) const { return mInner.ChildAt(aIndex, aResult); } NS_IMETHOD IndexOf(nsIContent* aPossibleChild, PRInt32& aResult) const { return mInner.IndexOf(aPossibleChild, aResult); } NS_IMETHOD InsertChildAt(nsIContent* aKid, PRInt32 aIndex, PRBool aNotify) { return mInner.InsertChildAt(aKid, aIndex, aNotify); } NS_IMETHOD ReplaceChildAt(nsIContent* aKid, PRInt32 aIndex, PRBool aNotify) { return mInner.ReplaceChildAt(aKid, aIndex, aNotify); } NS_IMETHOD AppendChildTo(nsIContent* aKid, PRBool aNotify) { return mInner.AppendChildTo(aKid, aNotify); } NS_IMETHOD RemoveChildAt(PRInt32 aIndex, PRBool aNotify) { return mInner.RemoveChildAt(aIndex, aNotify); } NS_IMETHOD IsSynthetic(PRBool& aResult) { return mInner.IsSynthetic(aResult); } NS_IMETHOD GetNameSpaceID(PRInt32& aID) const { return mInner.GetNameSpaceID(aID); } NS_IMETHOD GetTag(nsIAtom*& aResult) const; NS_IMETHOD ParseAttributeString(const nsString& aStr, nsIAtom*& aName, PRInt32& aNameSpaceID) { return mInner.ParseAttributeString(aStr, aName, aNameSpaceID); } NS_IMETHOD GetNameSpacePrefixFromId(PRInt32 aNameSpaceID, nsIAtom*& aPrefix) { return mInner.GetNameSpacePrefixFromId(aNameSpaceID, aPrefix); } NS_IMETHOD GetAttribute(PRInt32 aNameSpaceID, nsIAtom *aAttribute, nsString &aResult) const { return mInner.GetAttribute(aNameSpaceID, aAttribute, aResult); } NS_IMETHOD SetAttribute(PRInt32 aNameSpaceID, nsIAtom* aAttribute, const nsString& aValue, PRBool aNotify) { return mInner.SetAttribute(aNameSpaceID, aAttribute, aValue, aNotify); } NS_IMETHOD UnsetAttribute(PRInt32 aNameSpaceID, nsIAtom* aAttribute, PRBool aNotify) { return mInner.UnsetAttribute(aNameSpaceID, aAttribute, aNotify); } NS_IMETHOD GetAttributeNameAt(PRInt32 aIndex, PRInt32& aNameSpaceID, nsIAtom*& aName) const { return mInner.GetAttributeNameAt(aIndex, aNameSpaceID, aName); } NS_IMETHOD GetAttributeCount(PRInt32& aResult) const { return mInner.GetAttributeCount(aResult); } NS_IMETHOD List(FILE* out, PRInt32 aIndent) const; NS_IMETHOD BeginConvertToXIF(nsXIFConverter& aConverter) const { return mInner.BeginConvertToXIF(aConverter); } NS_IMETHOD ConvertContentToXIF(nsXIFConverter& aConverter) const; NS_IMETHOD FinishConvertToXIF(nsXIFConverter& aConverter) const { return mInner.FinishConvertToXIF(aConverter); } NS_IMETHOD HandleDOMEvent(nsIPresContext& aPresContext, nsEvent* aEvent, nsIDOMEvent** aDOMEvent, PRUint32 aFlags, nsEventStatus& aEventStatus); NS_IMETHOD GetContentID(PRUint32* aID) { *aID = mContentID; return NS_OK; } NS_IMETHOD SetContentID(PRUint32 aID) { mContentID = aID; return NS_OK; } NS_IMETHOD RangeAdd(nsIDOMRange& aRange){ return mInner.RangeAdd(aRange); } NS_IMETHOD RangeRemove(nsIDOMRange& aRange){ return mInner.RangeRemove(aRange); } NS_IMETHOD GetRangeList(nsVoidArray*& aResult) const { return mInner.GetRangeList(aResult); } NS_IMETHOD SizeOf(nsISizeOfHandler* aSizer, PRUint32* aResult) const { if (!aResult) { return NS_ERROR_NULL_POINTER; } #ifdef DEBUG *aResult = sizeof(*this); #else *aResult = 0; #endif return NS_OK; } NS_IMETHOD GetText(const nsTextFragment*& aFragmentsResult, PRInt32& aNumFragmentsResult) { return mInner.GetText(aFragmentsResult, aNumFragmentsResult); } NS_IMETHOD GetTextLength(PRInt32* aLengthResult) { return mInner.GetTextLength(aLengthResult); } NS_IMETHOD CopyText(nsString& aResult) { return mInner.CopyText(aResult); } NS_IMETHOD SetText(const PRUnichar* aBuffer, PRInt32 aLength, PRBool aNotify); NS_IMETHOD SetText(const char* aBuffer, PRInt32 aLength, PRBool aNotify); NS_IMETHOD IsOnlyWhitespace(PRBool* aResult) { return mInner.IsOnlyWhitespace(aResult); } protected: nsGenericDOMDataNode mInner; PRUint32 mContentID; }; nsresult NS_NewCommentNode(nsIContent** aInstancePtrResult); nsresult NS_NewCommentNode(nsIContent** aInstancePtrResult) { NS_PRECONDITION(nsnull != aInstancePtrResult, "null ptr"); if (nsnull == aInstancePtrResult) { return NS_ERROR_NULL_POINTER; } nsIContent* it = new nsCommentNode(); if (nsnull == it) { return NS_ERROR_OUT_OF_MEMORY; } return it->QueryInterface(kIContentIID, (void **) aInstancePtrResult); } nsCommentNode::nsCommentNode() { NS_INIT_REFCNT(); mInner.Init(this); mContentID = 0; } nsCommentNode::~nsCommentNode() { } NS_IMPL_ADDREF(nsCommentNode) NS_IMPL_RELEASE(nsCommentNode) NS_IMETHODIMP nsCommentNode::QueryInterface(REFNSIID aIID, void** aInstancePtr) { NS_IMPL_DOM_DATA_QUERY_INTERFACE(aIID, aInstancePtr, this) if (aIID.Equals(kIDOMCommentIID)) { nsIDOMComment* tmp = this; *aInstancePtr = (void*) tmp; NS_ADDREF_THIS(); return NS_OK; } if (aIID.Equals(kITextContentIID)) { nsITextContent* tmp = this; *aInstancePtr = (void*) tmp; NS_ADDREF_THIS(); return NS_OK; } return NS_NOINTERFACE; } NS_IMETHODIMP nsCommentNode::GetTag(nsIAtom*& aResult) const { aResult = nsLayoutAtoms::commentTagName; NS_ADDREF(aResult); return NS_OK; } NS_IMETHODIMP nsCommentNode::GetNodeName(nsString& aNodeName) { aNodeName.SetString("#comment"); return NS_OK; } NS_IMETHODIMP nsCommentNode::GetNodeType(PRUint16* aNodeType) { *aNodeType = (PRUint16)nsIDOMNode::COMMENT_NODE; return NS_OK; } NS_IMETHODIMP nsCommentNode::CloneNode(PRBool aDeep, nsIDOMNode** aReturn) { nsresult result = NS_OK; nsCommentNode* it = new nsCommentNode(); if (nsnull == it) { return NS_ERROR_OUT_OF_MEMORY; } // XXX Increment the ref count before calling any // methods. If they do a QI and then a Release() // the instance will be deleted. result = it->QueryInterface(kIDOMNodeIID, (void**) aReturn); if (NS_FAILED(result)) { return result; } nsAutoString data; result = GetData(data); if (NS_FAILED(result)) { NS_RELEASE(*aReturn); return result; } result = it->SetData(data); if (NS_FAILED(result)) { NS_RELEASE(*aReturn); return result; } return result; } NS_IMETHODIMP nsCommentNode::List(FILE* out, PRInt32 aIndent) const { NS_PRECONDITION(nsnull != mInner.mDocument, "bad content"); PRInt32 indx; for (indx = aIndent; --indx >= 0; ) fputs(" ", out); fprintf(out, "Comment refcount=%d\n", out); return NS_OK; } NS_IMETHODIMP nsCommentNode::HandleDOMEvent(nsIPresContext& aPresContext, nsEvent* aEvent, nsIDOMEvent** aDOMEvent, PRUint32 aFlags, nsEventStatus& aEventStatus) { return mInner.HandleDOMEvent(aPresContext, aEvent, aDOMEvent, aFlags, aEventStatus); } nsresult NS_NewCommentFrame(nsIFrame*& aResult); nsresult NS_NewCommentFrame(nsIFrame*& aResult) { nsIFrame* frame; NS_NewEmptyFrame(&frame); if (nsnull == frame) { return NS_ERROR_OUT_OF_MEMORY; } aResult = frame; return NS_OK; } /** * Translate the content object into the (XIF) XML Interchange Format * XIF is an intermediate form of the content model, the buffer * will then be parsed into any number of formats including HTML, TXT, etc. */ nsresult nsCommentNode::ConvertContentToXIF(nsXIFConverter& aConverter) const { const nsIContent* content = this; nsIDOMSelection* sel = aConverter.GetSelection(); nsIDocument* document; nsresult res; res = GetDocument(document); if (!NS_SUCCEEDED(res)) return res; const nsTextFragment* textFrag; PRInt32 numFragments; // XXX This method is const, but GetText() isn't, // XXX so cast away the constness of mInner: nsGenericDOMDataNode* inner = (nsGenericDOMDataNode*)&mInner; res = inner->GetText(textFrag, numFragments); if (!NS_SUCCEEDED(res)) return res; #ifdef DEBUG_akkana if (numFragments == 0) printf("numFragments is zero! Go figure!\n"); #endif if (sel != nsnull && document->IsInSelection(sel,content)) { nsIEnumerator *enumerator; if (NS_SUCCEEDED(sel->QueryInterface(kIEnumeratorIID, (void **)&enumerator))) { for (enumerator->First();NS_OK != enumerator->IsDone(); enumerator->Next()) { nsIDOMRange* range = nsnull; if (NS_SUCCEEDED(enumerator->CurrentItem((nsISupports**)&range))) { nsCOMPtr startNode; nsCOMPtr endNode; PRInt32 startOffset = 0; PRInt32 endOffset = 0; range->GetStartParent(getter_AddRefs(startNode)); range->GetEndParent(getter_AddRefs(endNode)); range->GetStartOffset(&startOffset); range->GetEndOffset(&endOffset); nsCOMPtr startContent; nsCOMPtr endContent; startContent = do_QueryInterface(startNode); endContent = do_QueryInterface(endNode); nsString buffer; textFrag->AppendTo(buffer); if (startContent.get() == content || endContent.get() == content) { // NOTE: ORDER MATTERS! // This must go before the Cut if (endContent.get() == content) buffer.Truncate(endOffset); // This must go after the Trunctate if (startContent.get() == content) buffer.Cut(0,startOffset); } aConverter.AddContentComment(buffer); } } } } else { nsString buffer; textFrag->AppendTo(buffer); aConverter.AddContentComment(buffer); } NS_IF_RELEASE(document); // XXX Possible mem leak: Do we need to delete textFrag? return NS_OK; } #if 0 nsresult nsCommentNode::BeginConvertToXIF(nsXIFConverter& aConverter) const { return NS_OK; } nsresult nsCommentNode::FinishConvertToXIF(nsXIFConverter& aConverter) const { return NS_OK; } #endif // This would ideally be done by the parser, but for the sake // of "genericity" it's being done in the comment content code static void StripCommentDelimiters(nsString& aCommentString) { PRInt32 offset; static char* kCommentStart = ""; static char* kCommentAlternateEnd = "--!>"; static char kMinus = '-'; offset = aCommentString.Find(kCommentStart); offset += strlen(kCommentStart); if (-1 != offset) { // Take up to 2 '-' characters if (kMinus == aCommentString.CharAt(offset)) { offset++; if (kMinus == aCommentString.CharAt(offset)) { offset++; } } aCommentString.Cut(0, offset); } offset = aCommentString.RFind(kCommentEnd); if (-1 != offset) { // Take up to 1 more '-' if (kMinus == aCommentString.CharAt(offset-1)) { offset--; } aCommentString.Cut(offset, aCommentString.Length()-offset); } else { offset = aCommentString.RFind(kCommentAlternateEnd); if (-1 != offset) { aCommentString.Cut(offset, aCommentString.Length()-offset); } } } NS_IMETHODIMP nsCommentNode::SetText(const PRUnichar* aBuffer, PRInt32 aLength, PRBool aNotify) { nsAutoString str(aBuffer); StripCommentDelimiters(str); return mInner.SetText(str.GetUnicode(), str.Length(), aNotify); } NS_IMETHODIMP nsCommentNode::SetText(const char* aBuffer, PRInt32 aLength, PRBool aNotify) { nsAutoString str(aBuffer); StripCommentDelimiters(str); return mInner.SetText(str.GetUnicode(), str.Length(), aNotify); }