/* -*- 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 "nsContentList.h" #include "nsIContent.h" #include "nsIDOMNode.h" #include "nsIScriptGlobalObject.h" #include "nsIDocument.h" #include "nsINameSpaceManager.h" #include "nsIDOMScriptObjectFactory.h" #include "nsGenericElement.h" #include "nsLayoutAtoms.h" #include "nsHTMLAtoms.h" // XXX until atoms get factored into nsLayoutAtoms nsIAtom* nsContentList::gWildCardAtom = nsnull; nsContentList::nsContentList(nsIDocument *aDocument) { NS_INIT_REFCNT(); mScriptObject = nsnull; mFunc = nsnull; mMatchAtom = nsnull; mDocument = aDocument; mMatchAll = PR_FALSE; } nsContentList::nsContentList(nsIDocument *aDocument, nsIAtom* aMatchAtom, PRInt32 aMatchNameSpaceId, nsIContent* aRootContent) { mMatchAtom = aMatchAtom; NS_IF_ADDREF(mMatchAtom); if (nsnull == gWildCardAtom) { gWildCardAtom = NS_NewAtom("*"); } if (gWildCardAtom == mMatchAtom) { mMatchAll = PR_TRUE; } else { mMatchAll = PR_FALSE; } mMatchNameSpaceId = aMatchNameSpaceId; mFunc = nsnull; mData = nsnull; mRootContent = aRootContent; Init(aDocument); } nsContentList::nsContentList(nsIDocument *aDocument, nsContentListMatchFunc aFunc, void* aData, nsIContent* aRootContent) { mFunc = aFunc; mData = aData; mMatchAtom = nsnull; mRootContent = aRootContent; mMatchAll = PR_FALSE; Init(aDocument); } void nsContentList::Init(nsIDocument *aDocument) { NS_INIT_REFCNT(); mScriptObject = nsnull; // We don't reference count the reference to the document // If the document goes away first, we'll be informed and we // can drop our reference. // If we go away first, we'll get rid of ourselves from the // document's observer list. mDocument = aDocument; if (nsnull != mDocument) { mDocument->AddObserver(this); } PopulateSelf(); } nsContentList::~nsContentList() { if (nsnull != mDocument) { mDocument->RemoveObserver(this); } NS_IF_RELEASE(mMatchAtom); // XXX This is absolutely gross, since this is allocated // elsewhere. if (nsnull != mData) { delete mData; } } static NS_DEFINE_IID(kIDOMNodeListIID, NS_IDOMNODELIST_IID); static NS_DEFINE_IID(kIDOMHTMLCollectionIID, NS_IDOMHTMLCOLLECTION_IID); nsresult nsContentList::QueryInterface(REFNSIID aIID, void** aInstancePtr) { if (nsnull == aInstancePtr) { return NS_ERROR_NULL_POINTER; } if (aIID.Equals(kIDOMNodeListIID)) { *aInstancePtr = (void*)(nsIDOMNodeList*)this; NS_ADDREF_THIS(); return NS_OK; } if (aIID.Equals(kIDOMHTMLCollectionIID)) { *aInstancePtr = (void*)(nsIDOMHTMLCollection*)this; NS_ADDREF_THIS(); return NS_OK; } if (aIID.Equals(kIScriptObjectOwnerIID)) { *aInstancePtr = (void*)(nsIScriptObjectOwner*)this; NS_ADDREF_THIS(); return NS_OK; } if (aIID.Equals(kISupportsIID)) { *aInstancePtr = (void*)(nsISupports*)(nsIDOMNodeList*)this; NS_ADDREF_THIS(); return NS_OK; } return NS_NOINTERFACE; } NS_IMPL_ADDREF(nsContentList) NS_IMPL_RELEASE(nsContentList) NS_IMETHODIMP nsContentList::GetLength(PRUint32* aLength) { nsresult result = CheckDocumentExistence(); if (NS_OK == result) { *aLength = mContent.Count(); } return result; } NS_IMETHODIMP nsContentList::Item(PRUint32 aIndex, nsIDOMNode** aReturn) { nsresult result = CheckDocumentExistence(); if (NS_OK == result) { nsISupports *element = (nsISupports *)mContent.ElementAt(aIndex); if (nsnull != element) { result = element->QueryInterface(kIDOMNodeIID, (void **)aReturn); } else { *aReturn = nsnull; } } return result; } NS_IMETHODIMP nsContentList::NamedItem(const nsString& aName, nsIDOMNode** aReturn) { nsresult result = CheckDocumentExistence(); if (NS_OK == result) { PRInt32 i, count = mContent.Count(); for (i = 0; i < count; i++) { nsIContent *content = (nsIContent *)mContent.ElementAt(i); if (nsnull != content) { nsAutoString name; // XXX Should it be an EqualsIgnoreCase? if (((content->GetAttribute(kNameSpaceID_HTML, nsHTMLAtoms::name, name) == NS_CONTENT_ATTR_HAS_VALUE) && (aName.Equals(name))) || ((content->GetAttribute(kNameSpaceID_HTML, nsHTMLAtoms::id, name) == NS_CONTENT_ATTR_HAS_VALUE) && (aName.Equals(name)))) { return content->QueryInterface(kIDOMNodeIID, (void **)aReturn); } } } } *aReturn = nsnull; return result; } NS_IMETHODIMP nsContentList::GetScriptObject(nsIScriptContext *aContext, void** aScriptObject) { nsresult res = NS_OK; nsIScriptGlobalObject *global = aContext->GetGlobalObject(); if (nsnull == mScriptObject) { nsIDOMScriptObjectFactory *factory; res = nsGenericElement::GetScriptObjectFactory(&factory); if (NS_OK != res) { return res; } res = factory->NewScriptHTMLCollection(aContext, (nsISupports*)(nsIDOMHTMLCollection*)this, global, (void**)&mScriptObject); NS_RELEASE(factory); } *aScriptObject = mScriptObject; NS_RELEASE(global); return res; } NS_IMETHODIMP nsContentList::SetScriptObject(void *aScriptObject) { mScriptObject = aScriptObject; return NS_OK; } NS_IMETHODIMP nsContentList::ContentAppended(nsIDocument *aDocument, nsIContent* aContainer, PRInt32 aNewIndexInContainer) { PRInt32 i, count; aContainer->ChildCount(count); if ((count > 0) && IsDescendantOfRoot(aContainer)) { PRBool repopulate = PR_FALSE; for (i = aNewIndexInContainer; i <= count-1; i++) { nsIContent *content; aContainer->ChildAt(i, content); if (mMatchAll || MatchSelf(content)) { repopulate = PR_TRUE; } NS_RELEASE(content); } if (repopulate) { PopulateSelf(); } } return NS_OK; } NS_IMETHODIMP nsContentList::ContentInserted(nsIDocument *aDocument, nsIContent* aContainer, nsIContent* aChild, PRInt32 aIndexInContainer) { if (IsDescendantOfRoot(aContainer)) { if (mMatchAll || MatchSelf(aChild)) { PopulateSelf(); } } return NS_OK; } NS_IMETHODIMP nsContentList::ContentReplaced(nsIDocument *aDocument, nsIContent* aContainer, nsIContent* aOldChild, nsIContent* aNewChild, PRInt32 aIndexInContainer) { if (IsDescendantOfRoot(aContainer)) { if (mMatchAll || MatchSelf(aOldChild) || MatchSelf(aNewChild)) { PopulateSelf(); } } else if (ContainsRoot(aOldChild)) { DisconnectFromDocument(); } return NS_OK; } NS_IMETHODIMP nsContentList::ContentRemoved(nsIDocument *aDocument, nsIContent* aContainer, nsIContent* aChild, PRInt32 aIndexInContainer) { if (IsDescendantOfRoot(aContainer) && MatchSelf(aChild)) { PopulateSelf(); } else if (ContainsRoot(aChild)) { DisconnectFromDocument(); } return NS_OK; } NS_IMETHODIMP nsContentList::DocumentWillBeDestroyed(nsIDocument *aDocument) { if (nsnull != mDocument) { aDocument->RemoveObserver(this); mDocument = nsnull; } Reset(); return NS_OK; } // Returns whether the content element matches the // criterion nsresult nsContentList::Match(nsIContent *aContent, PRBool *aMatch) { if (nsnull != mMatchAtom) { nsIAtom *name = nsnull; aContent->GetTag(name); // If we have to match all, only do those that have // a tagName i.e. only the elements. if (mMatchAll && (nsLayoutAtoms::textTagName != name) && (nsLayoutAtoms::commentTagName != name)) { *aMatch = PR_TRUE; } // XXX We don't yet match on namespace. Maybe we should?? else if (name == mMatchAtom) { *aMatch = PR_TRUE; } else { *aMatch = PR_FALSE; } NS_IF_RELEASE(name); } else if (nsnull != mFunc) { *aMatch = (*mFunc)(aContent, mData); } else { *aMatch = PR_FALSE; } return NS_OK; } nsresult nsContentList::Add(nsIContent *aContent) { // Shouldn't hold a reference since we'll be // told when the content leaves the document or // the document will be destroyed. mContent.AppendElement(aContent); return NS_OK; } nsresult nsContentList::Remove(nsIContent *aContent) { mContent.RemoveElement(aContent); return NS_OK; } nsresult nsContentList::Reset() { mContent.Clear(); return NS_OK; } // If we were created outside the context of a document and we // have root content, then check if our content has been added // to a document yet. If so, we'll become an observer of the document. nsresult nsContentList::CheckDocumentExistence() { nsresult result = NS_OK; if ((nsnull == mDocument) && (nsnull != mRootContent)) { result = mRootContent->GetDocument(mDocument); if (nsnull != mDocument) { mDocument->AddObserver(this); PopulateSelf(); } } return result; } // Match recursively. See if anything in the subtree // matches the criterion. PRBool nsContentList::MatchSelf(nsIContent *aContent) { PRBool match; PRInt32 i, count; Match(aContent, &match); if (match) { return PR_TRUE; } aContent->ChildCount(count); for (i = 0; i < count; i++) { nsIContent *child; aContent->ChildAt(i, child); if (MatchSelf(child)) { NS_RELEASE(child); return PR_TRUE; } NS_RELEASE(child); } return PR_FALSE; } // Add all elements in this subtree that match to // our list. void nsContentList::PopulateWith(nsIContent *aContent, PRBool aIncludeRoot) { PRBool match; PRInt32 i, count; if (aIncludeRoot) { Match(aContent, &match); if (match) { Add(aContent); } } aContent->ChildCount(count); for (i = 0; i < count; i++) { nsIContent *child; aContent->ChildAt(i, child); PopulateWith(child, PR_TRUE); NS_RELEASE(child); } } // Clear out our old list and build up a new one void nsContentList::PopulateSelf() { Reset(); if (nsnull != mRootContent) { PopulateWith(mRootContent, PR_FALSE); } else if (nsnull != mDocument) { nsIContent *root; root = mDocument->GetRootContent(); PopulateWith(root, PR_TRUE); NS_RELEASE(root); } } // Is the specified element a descendant of the root? If there // is no root, then yes. Otherwise keep tracing up the tree from // the element till we find our root, or until we reach the // document root. PRBool nsContentList::IsDescendantOfRoot(nsIContent* aContainer) { if (nsnull == mRootContent) { return PR_TRUE; } else if (mRootContent == aContainer) { return PR_TRUE; } else if (nsnull == aContainer) { return PR_FALSE; } else { nsIContent* parent; PRBool ret; aContainer->GetParent(parent); ret = IsDescendantOfRoot(parent); NS_IF_RELEASE(parent); return ret; } } // Does this subtree contain the root? PRBool nsContentList::ContainsRoot(nsIContent* aContent) { if (nsnull == mRootContent) { return PR_FALSE; } else if (mRootContent == aContent) { return PR_TRUE; } else { PRInt32 i, count; aContent->ChildCount(count); for (i = 0; i < count; i++) { nsIContent *child; aContent->ChildAt(i, child); if (ContainsRoot(child)) { NS_RELEASE(child); return PR_TRUE; } NS_RELEASE(child); } return PR_FALSE; } } // Our root content has been disconnected from the // document, so stop observing. The list then becomes // a snapshot rather than a dynamic list. void nsContentList::DisconnectFromDocument() { if (nsnull != mDocument) { mDocument->RemoveObserver(this); mDocument = nsnull; } }