/* -*- 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 "nsDOMCSSAttrDeclaration.h" #include "nsICSSDeclaration.h" #include "nsIDocument.h" #include "nsHTMLAtoms.h" #include "nsIHTMLContent.h" #include "nsIDOMMutationEvent.h" #include "nsHTMLValue.h" #include "nsICSSStyleRule.h" #include "nsINodeInfo.h" #include "nsICSSLoader.h" #include "nsICSSParser.h" #include "nsICSSDeclaration.h" #include "nsIURI.h" #include "nsINameSpaceManager.h" #include "nsIHTMLContentContainer.h" MOZ_DECL_CTOR_COUNTER(nsDOMCSSAttributeDeclaration) nsDOMCSSAttributeDeclaration::nsDOMCSSAttributeDeclaration(nsIHTMLContent *aContent) { MOZ_COUNT_CTOR(nsDOMCSSAttributeDeclaration); // This reference is not reference-counted. The content // object tells us when its about to go away. mContent = aContent; } nsDOMCSSAttributeDeclaration::~nsDOMCSSAttributeDeclaration() { MOZ_COUNT_DTOR(nsDOMCSSAttributeDeclaration); } NS_IMETHODIMP nsDOMCSSAttributeDeclaration::RemoveProperty(const nsAReadableString& aPropertyName, nsAWritableString& aReturn) { nsCOMPtr decl; nsresult rv = GetCSSDeclaration(getter_AddRefs(decl), PR_TRUE); if (NS_SUCCEEDED(rv) && decl && mContent) { nsCOMPtr doc; mContent->GetDocument(*getter_AddRefs(doc)); if (doc) { doc->BeginUpdate(); doc->AttributeWillChange(mContent, kNameSpaceID_None, nsHTMLAtoms::style); } PRInt32 hint; decl->GetStyleImpact(&hint); nsCSSProperty prop = nsCSSProps::LookupProperty(aPropertyName); nsCSSValue val; rv = decl->RemoveProperty(prop, val); if (NS_SUCCEEDED(rv)) { // We pass in eCSSProperty_UNKNOWN here so that we don't get the // property name in the return string. val.ToString(aReturn, eCSSProperty_UNKNOWN); } else { // If we tried to remove an invalid property or a property that wasn't // set we simply return success and an empty string rv = NS_OK; } if (doc) { doc->AttributeChanged(mContent, kNameSpaceID_None, nsHTMLAtoms::style, nsIDOMMutationEvent::MODIFICATION, hint); doc->EndUpdate(); } } return rv; } void nsDOMCSSAttributeDeclaration::DropReference() { mContent = nsnull; } nsresult nsDOMCSSAttributeDeclaration::GetCSSDeclaration(nsICSSDeclaration **aDecl, PRBool aAllocate) { nsHTMLValue val; nsIStyleRule* rule; nsICSSStyleRule* cssRule; nsresult result = NS_OK; *aDecl = nsnull; if (nsnull != mContent) { mContent->GetHTMLAttribute(nsHTMLAtoms::style, val); if (eHTMLUnit_ISupports == val.GetUnit()) { rule = (nsIStyleRule*) val.GetISupportsValue(); result = rule->QueryInterface(NS_GET_IID(nsICSSStyleRule), (void**)&cssRule); if (NS_OK == result) { *aDecl = cssRule->GetDeclaration(); NS_RELEASE(cssRule); } NS_RELEASE(rule); } else if (PR_TRUE == aAllocate) { result = NS_NewCSSDeclaration(aDecl); if (NS_OK == result) { result = NS_NewCSSStyleRule(&cssRule, nsCSSSelector()); if (NS_OK == result) { cssRule->SetDeclaration(*aDecl); cssRule->SetWeight(0x7fffffff); rule = (nsIStyleRule *)cssRule; result = mContent->SetHTMLAttribute(nsHTMLAtoms::style, nsHTMLValue(cssRule), PR_FALSE); NS_RELEASE(cssRule); } else { NS_RELEASE(*aDecl); } } } } return result; } nsresult nsDOMCSSAttributeDeclaration::SetCSSDeclaration(nsICSSDeclaration *aDecl) { nsHTMLValue val; nsIStyleRule* rule; nsICSSStyleRule* cssRule; nsresult result = NS_OK; if (nsnull != mContent) { mContent->GetHTMLAttribute(nsHTMLAtoms::style, val); if (eHTMLUnit_ISupports == val.GetUnit()) { rule = (nsIStyleRule*) val.GetISupportsValue(); result = rule->QueryInterface(NS_GET_IID(nsICSSStyleRule), (void**)&cssRule); if (NS_OK == result) { cssRule->SetDeclaration(aDecl); NS_RELEASE(cssRule); } NS_RELEASE(rule); } } return result; } /* * This is a utility function. It will only fail if it can't get a * parser. This means it can return NS_OK without aURI or aCSSLoader * being initialized */ nsresult nsDOMCSSAttributeDeclaration::GetCSSParsingEnvironment(nsIContent* aContent, nsIURI** aBaseURI, nsICSSLoader** aCSSLoader, nsICSSParser** aCSSParser) { NS_ASSERTION(aContent, "Something is severely broken -- there should be an nsIContent here!"); // null out the out params since some of them may not get initialized below *aBaseURI = nsnull; *aCSSLoader = nsnull; *aCSSParser = nsnull; nsCOMPtr nodeInfo; nsresult result = aContent->GetNodeInfo(*getter_AddRefs(nodeInfo)); if (NS_FAILED(result)) { return result; } nsCOMPtr doc; result = nodeInfo->GetDocument(*getter_AddRefs(doc)); if (NS_FAILED(result)) { return result; } if (doc) { doc->GetBaseURL(*aBaseURI); nsCOMPtr htmlContainer(do_QueryInterface(doc)); if (htmlContainer) { htmlContainer->GetCSSLoader(*aCSSLoader); } NS_ASSERTION(*aCSSLoader, "Document with no CSS loader!"); } if (*aCSSLoader) { result = (*aCSSLoader)->GetParserFor(nsnull, aCSSParser); } else { result = NS_NewCSSParser(aCSSParser); } if (NS_FAILED(result)) { return result; } // look up our namespace. If we're XHTML, we need to be case-sensitive // Otherwise, we should not be (*aCSSParser)->SetCaseSensitive(nodeInfo->NamespaceEquals(kNameSpaceID_XHTML)); return NS_OK; } nsresult nsDOMCSSAttributeDeclaration::ParsePropertyValue(const nsAReadableString& aPropName, const nsAReadableString& aPropValue) { nsCOMPtr decl; nsresult result = GetCSSDeclaration(getter_AddRefs(decl), PR_TRUE); if (!decl) { return result; } nsCOMPtr cssLoader; nsCOMPtr cssParser; nsCOMPtr baseURI; nsCOMPtr doc; result = mContent->GetDocument(*getter_AddRefs(doc)); if (NS_FAILED(result)) { return result; } result = GetCSSParsingEnvironment(mContent, getter_AddRefs(baseURI), getter_AddRefs(cssLoader), getter_AddRefs(cssParser)); if (NS_FAILED(result)) { return result; } PRInt32 hint; if (doc) { doc->BeginUpdate(); doc->AttributeWillChange(mContent, kNameSpaceID_None, nsHTMLAtoms::style); } result = cssParser->ParseProperty(aPropName, aPropValue, baseURI, decl, &hint); if (doc) { doc->AttributeChanged(mContent, kNameSpaceID_None, nsHTMLAtoms::style, nsIDOMMutationEvent::MODIFICATION, hint); doc->EndUpdate(); } if (cssLoader) { cssLoader->RecycleParser(cssParser); } return NS_OK; } nsresult nsDOMCSSAttributeDeclaration::ParseDeclaration(const nsAReadableString& aDecl, PRBool aParseOnlyOneDecl, PRBool aClearOldDecl) { nsCOMPtr decl; nsresult result = GetCSSDeclaration(getter_AddRefs(decl), PR_TRUE); if (decl) { nsCOMPtr cssLoader; nsCOMPtr cssParser; nsCOMPtr baseURI; nsCOMPtr doc; result = mContent->GetDocument(*getter_AddRefs(doc)); if (NS_FAILED(result)) { return result; } result = GetCSSParsingEnvironment(mContent, getter_AddRefs(baseURI), getter_AddRefs(cssLoader), getter_AddRefs(cssParser)); if (NS_SUCCEEDED(result)) { PRInt32 hint; if (doc) { doc->BeginUpdate(); doc->AttributeWillChange(mContent, kNameSpaceID_None, nsHTMLAtoms::style); } nsCOMPtr declClone; decl->Clone(*getter_AddRefs(declClone)); if (aClearOldDecl) { // This should be done with decl->Clear() once such a method exists. nsAutoString propName; PRUint32 count, i; decl->Count(&count); for (i = 0; i < count; i++) { decl->GetNthProperty(0, propName); nsCSSProperty prop = nsCSSProps::LookupProperty(propName); nsCSSValue val; decl->RemoveProperty(prop, val); } } result = cssParser->ParseAndAppendDeclaration(aDecl, baseURI, decl, aParseOnlyOneDecl, &hint); if (result == NS_CSS_PARSER_DROP_DECLARATION) { SetCSSDeclaration(declClone); result = NS_OK; } if (doc) { if (NS_SUCCEEDED(result) && result != NS_CSS_PARSER_DROP_DECLARATION) { doc->AttributeChanged(mContent, kNameSpaceID_None, nsHTMLAtoms::style, nsIDOMMutationEvent::MODIFICATION, hint); } doc->EndUpdate(); } if (cssLoader) { cssLoader->RecycleParser(cssParser); } } } return result; } nsresult nsDOMCSSAttributeDeclaration::GetParent(nsISupports **aParent) { NS_ENSURE_ARG_POINTER(aParent); *aParent = mContent; NS_IF_ADDREF(*aParent); return NS_OK; }