/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: NPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Netscape Public License * Version 1.1 (the "License"); you may not use this file except in * compliance with the License. You may obtain a copy of the License at * http://www.mozilla.org/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 the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the NPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the NPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include "nsDOMAttributeMap.h" #include "nsDOMAttribute.h" #include "nsGenericElement.h" #include "nsIContent.h" #include "nsINameSpaceManager.h" #include "nsDOMError.h" #include "nsContentUtils.h" //---------------------------------------------------------------------- nsDOMAttributeMap::nsDOMAttributeMap(nsIContent* aContent) : mContent(aContent) { // We don't add a reference to our content. If it goes away, // we'll be told to drop our reference } nsDOMAttributeMap::~nsDOMAttributeMap() { } void nsDOMAttributeMap::DropReference() { mContent = nsnull; } // QueryInterface implementation for nsDOMAttributeMap NS_INTERFACE_MAP_BEGIN(nsDOMAttributeMap) NS_INTERFACE_MAP_ENTRY(nsISupports) NS_INTERFACE_MAP_ENTRY(nsIDOMNamedNodeMap) NS_INTERFACE_MAP_ENTRY_CONTENT_CLASSINFO(NamedNodeMap) NS_INTERFACE_MAP_END NS_IMPL_ADDREF(nsDOMAttributeMap) NS_IMPL_RELEASE(nsDOMAttributeMap) nsresult nsDOMAttributeMap::GetNamedItem(const nsAString& aAttrName, nsIDOMNode** aAttribute) { NS_ENSURE_ARG_POINTER(aAttribute); *aAttribute = nsnull; nsresult rv = NS_OK; if (mContent) { nsCOMPtr ni; mContent->NormalizeAttrString(aAttrName, *getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); PRInt32 nsid; nsCOMPtr nameAtom; ni->GetNamespaceID(nsid); ni->GetNameAtom(*getter_AddRefs(nameAtom)); nsresult attrResult; nsAutoString value; attrResult = mContent->GetAttr(nsid, nameAtom, value); if (NS_CONTENT_ATTR_NOT_THERE != attrResult && NS_SUCCEEDED(attrResult)) { nsDOMAttribute* domAttribute; domAttribute = new nsDOMAttribute(mContent, ni, value); NS_ENSURE_TRUE(domAttribute, NS_ERROR_OUT_OF_MEMORY); rv = CallQueryInterface(domAttribute, aAttribute); } } return rv; } nsresult nsDOMAttributeMap::SetNamedItem(nsIDOMNode *aNode, nsIDOMNode **aReturn) { NS_ENSURE_ARG_POINTER(aReturn); if (!aNode) { return NS_ERROR_NULL_POINTER; } nsresult rv = NS_OK; *aReturn = nsnull; if (mContent) { // XXX should check same-origin between mContent and aNode however // nsContentUtils::CheckSameOrigin can't deal with attributenodes yet nsCOMPtr attribute(do_QueryInterface(aNode)); if (!attribute) { return NS_ERROR_DOM_HIERARCHY_REQUEST_ERR; } nsAutoString name, value; attribute->GetName(name); nsCOMPtr ni; mContent->NormalizeAttrString(name, *getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); nsCOMPtr nameAtom; PRInt32 nsid; ni->GetNamespaceID(nsid); ni->GetNameAtom(*getter_AddRefs(nameAtom)); nsresult attrResult = mContent->GetAttr(nsid, nameAtom, value); if (NS_CONTENT_ATTR_NOT_THERE != attrResult && NS_SUCCEEDED(attrResult)) { nsDOMAttribute* domAttribute; // We pass a null content here since the attr node we return isn't // tied to this content anymore. domAttribute = new nsDOMAttribute(nsnull, ni, value); if (!domAttribute) { return NS_ERROR_OUT_OF_MEMORY; } rv = CallQueryInterface(domAttribute, aReturn); } attribute->GetValue(value); rv = mContent->SetAttr(ni, value, PR_TRUE); } return rv; } NS_IMETHODIMP nsDOMAttributeMap::RemoveNamedItem(const nsAString& aName, nsIDOMNode** aReturn) { NS_ENSURE_ARG_POINTER(aReturn); *aReturn = nsnull; nsresult rv = NS_OK; if (mContent) { nsCOMPtr ni; mContent->NormalizeAttrString(aName, *getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); nsCOMPtr nameAtom; PRInt32 nsid; ni->GetNamespaceID(nsid); ni->GetNameAtom(*getter_AddRefs(nameAtom)); nsCOMPtr attribute; nsresult attrResult; nsAutoString value; attrResult = mContent->GetAttr(nsid, nameAtom, value); if (NS_CONTENT_ATTR_NOT_THERE != attrResult && NS_SUCCEEDED(attrResult)) { nsDOMAttribute* domAttribute; domAttribute = new nsDOMAttribute(nsnull, ni, value); if (!domAttribute) { return NS_ERROR_OUT_OF_MEMORY; } rv = CallQueryInterface(domAttribute, aReturn); } else { return NS_ERROR_DOM_NOT_FOUND_ERR; } rv = mContent->UnsetAttr(nsid, nameAtom, PR_TRUE); } return rv; } nsresult nsDOMAttributeMap::Item(PRUint32 aIndex, nsIDOMNode** aReturn) { NS_ENSURE_ARG_POINTER(aReturn); PRInt32 nameSpaceID; nsCOMPtr nameAtom, prefix; nsresult rv = NS_OK; if (mContent && NS_SUCCEEDED(mContent->GetAttrNameAt(aIndex, nameSpaceID, *getter_AddRefs(nameAtom), *getter_AddRefs(prefix)))) { nsAutoString value, name; mContent->GetAttr(nameSpaceID, nameAtom, value); nsCOMPtr ni; mContent->GetNodeInfo(*getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); nsCOMPtr nimgr; ni->GetNodeInfoManager(*getter_AddRefs(nimgr)); NS_ENSURE_TRUE(nimgr, NS_ERROR_FAILURE); nimgr->GetNodeInfo(nameAtom, prefix, nameSpaceID, *getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); nsDOMAttribute* domAttribute = new nsDOMAttribute(mContent, ni, value); NS_ENSURE_TRUE(domAttribute, NS_ERROR_OUT_OF_MEMORY); rv = CallQueryInterface(domAttribute, aReturn); } else { *aReturn = nsnull; } return rv; } nsresult nsDOMAttributeMap::GetLength(PRUint32 *aLength) { NS_ENSURE_ARG_POINTER(aLength); PRInt32 n; nsresult rv = NS_OK; if (nsnull != mContent) { rv = mContent->GetAttrCount(n); *aLength = PRUint32(n); } else { *aLength = 0; } return rv; } nsresult nsDOMAttributeMap::GetNamedItemNS(const nsAString& aNamespaceURI, const nsAString& aLocalName, nsIDOMNode** aReturn) { NS_ENSURE_ARG_POINTER(aReturn); *aReturn = nsnull; nsresult rv = NS_OK; if (mContent) { nsCOMPtr nameAtom = do_GetAtom(aLocalName); PRInt32 nameSpaceID = kNameSpaceID_None; nsCOMPtr prefix; nsCOMPtr ni; mContent->GetNodeInfo(*getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); nsCOMPtr nimgr; ni->GetNodeInfoManager(*getter_AddRefs(nimgr)); NS_ENSURE_TRUE(nimgr, NS_ERROR_FAILURE); if (aNamespaceURI.Length()) { nsContentUtils::GetNSManagerWeakRef()->GetNameSpaceID(aNamespaceURI, nameSpaceID); if (nameSpaceID == kNameSpaceID_Unknown) return NS_OK; } nsresult attrResult; nsAutoString value; attrResult = mContent->GetAttr(nameSpaceID, nameAtom, *getter_AddRefs(prefix), value); if (NS_CONTENT_ATTR_NOT_THERE != attrResult && NS_SUCCEEDED(attrResult)) { nimgr->GetNodeInfo(nameAtom, prefix, nameSpaceID, *getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); nsDOMAttribute* domAttribute; domAttribute = new nsDOMAttribute(mContent, ni, value); NS_ENSURE_TRUE(domAttribute, NS_ERROR_OUT_OF_MEMORY); rv = CallQueryInterface(domAttribute, aReturn); } } return rv; } nsresult nsDOMAttributeMap::SetNamedItemNS(nsIDOMNode* aArg, nsIDOMNode** aReturn) { NS_ENSURE_ARG_POINTER(aReturn); nsresult rv = NS_OK; *aReturn = nsnull; if (mContent) { // XXX should check same-origin between mContent and aNode however // nsContentUtils::CheckSameOrigin can't deal with attributenodes yet nsCOMPtr attribute(do_QueryInterface(aArg)); if (!attribute) { return NS_ERROR_DOM_HIERARCHY_REQUEST_ERR; } nsAutoString name, nsURI, value; nsCOMPtr nameAtom; PRInt32 nameSpaceID; attribute->GetName(name); attribute->GetPrefix(name); attribute->GetNamespaceURI(nsURI); nsCOMPtr ni; mContent->GetNodeInfo(*getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); nsCOMPtr nimgr; ni->GetNodeInfoManager(*getter_AddRefs(nimgr)); NS_ENSURE_TRUE(nimgr, NS_ERROR_FAILURE); nimgr->GetNodeInfo(name, nsURI, *getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); ni->GetNameAtom(*getter_AddRefs(nameAtom)); ni->GetNamespaceID(nameSpaceID); nsresult attrResult = mContent->GetAttr(nameSpaceID, nameAtom, value); if (NS_CONTENT_ATTR_NOT_THERE != attrResult && NS_SUCCEEDED(attrResult)) { nsDOMAttribute* domAttribute; // We pass a null content here since the attr node we return isn't // tied to this content anymore. domAttribute = new nsDOMAttribute(nsnull, ni, value); if (!domAttribute) { return NS_ERROR_OUT_OF_MEMORY; } rv = CallQueryInterface(domAttribute, aReturn); } attribute->GetValue(value); rv = mContent->SetAttr(ni, value, PR_TRUE); } return rv; } nsresult nsDOMAttributeMap::RemoveNamedItemNS(const nsAString& aNamespaceURI, const nsAString& aLocalName, nsIDOMNode** aReturn) { NS_ENSURE_ARG_POINTER(aReturn); *aReturn = nsnull; nsresult rv = NS_OK; if (mContent) { nsCOMPtr nameAtom = do_GetAtom(aLocalName); PRInt32 nameSpaceID = kNameSpaceID_None; nsCOMPtr attribute; nsCOMPtr prefix; nsCOMPtr ni; mContent->GetNodeInfo(*getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); nsCOMPtr nimgr; ni->GetNodeInfoManager(*getter_AddRefs(nimgr)); NS_ENSURE_TRUE(nimgr, NS_ERROR_FAILURE); if (aNamespaceURI.Length()) { nsContentUtils::GetNSManagerWeakRef()->GetNameSpaceID(aNamespaceURI, nameSpaceID); if (nameSpaceID == kNameSpaceID_Unknown) return NS_ERROR_DOM_NOT_FOUND_ERR; } nsresult attrResult; nsAutoString value; attrResult = mContent->GetAttr(nameSpaceID, nameAtom, *getter_AddRefs(prefix), value); if (NS_CONTENT_ATTR_NOT_THERE != attrResult && NS_SUCCEEDED(attrResult)) { nimgr->GetNodeInfo(nameAtom, prefix, nameSpaceID, *getter_AddRefs(ni)); NS_ENSURE_TRUE(ni, NS_ERROR_FAILURE); nsDOMAttribute* domAttribute; domAttribute = new nsDOMAttribute(nsnull, ni, value); if (!domAttribute) { return NS_ERROR_OUT_OF_MEMORY; } rv = CallQueryInterface(domAttribute, aReturn); } else { return NS_ERROR_DOM_NOT_FOUND_ERR; } rv = mContent->UnsetAttr(nameSpaceID, nameAtom, PR_TRUE); } return rv; }