/* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla 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/MPL/ * * 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.org code. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 2001 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Joe Hewitt (original author) * * 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 MPL, 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 MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include "inCSSValueSearch.h" #include "nsIComponentManager.h" #include "nsIServiceManager.h" #include "nsVoidArray.h" #include "nsReadableUtils.h" #include "nsIDOMDocumentStyle.h" #include "nsIDOM3Node.h" #include "nsIDOMStyleSheetList.h" #include "nsIDOMCSSStyleSheet.h" #include "nsIDOMCSSRuleList.h" #include "nsIDOMCSSStyleRule.h" #include "nsIDOMCSSStyleDeclaration.h" #include "nsIDOMCSSImportRule.h" #include "nsIDOMCSSMediaRule.h" #include "nsIURI.h" #include "nsNetUtil.h" static NS_DEFINE_CID(kInspectorCSSUtilsCID, NS_INSPECTORCSSUTILS_CID); /////////////////////////////////////////////////////////////////////////////// inCSSValueSearch::inCSSValueSearch() : mResults(nsnull), mProperties(nsnull), mResultCount(0), mPropertyCount(0), mIsActive(PR_FALSE), mHoldResults(PR_TRUE), mReturnRelativeURLs(PR_TRUE), mNormalizeChromeURLs(PR_FALSE) { mProperties = new nsCSSProperty[100]; mCSSUtils = do_GetService(kInspectorCSSUtilsCID); } inCSSValueSearch::~inCSSValueSearch() { delete[] mProperties; delete mResults; } NS_IMPL_ISUPPORTS2(inCSSValueSearch, inISearchProcess, inICSSValueSearch) /////////////////////////////////////////////////////////////////////////////// // inISearchProcess NS_IMETHODIMP inCSSValueSearch::GetIsActive(PRBool *aIsActive) { *aIsActive = mIsActive; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::GetResultCount(PRInt32 *aResultCount) { *aResultCount = mResultCount; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::GetHoldResults(PRBool *aHoldResults) { *aHoldResults = mHoldResults; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SetHoldResults(PRBool aHoldResults) { mHoldResults = aHoldResults; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SearchSync() { InitSearch(); nsCOMPtr baseURL; nsCOMPtr dom3Node = do_QueryInterface(mDocument); if (dom3Node) { nsAutoString uri; dom3Node->GetBaseURI(uri); NS_NewURI(getter_AddRefs(baseURL), uri); } nsCOMPtr doc = do_QueryInterface(mDocument); if (doc) { nsCOMPtr sheets; nsresult rv = doc->GetStyleSheets(getter_AddRefs(sheets)); NS_ENSURE_SUCCESS(rv, NS_OK); PRUint32 length; sheets->GetLength(&length); for (PRUint32 i = 0; i < length; ++i) { nsCOMPtr sheet; sheets->Item(i, getter_AddRefs(sheet)); nsCOMPtr cssSheet = do_QueryInterface(sheet); if (cssSheet) SearchStyleSheet(cssSheet, baseURL); } } // XXX would be nice to search inline style as well. return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SearchAsync(inISearchObserver *aObserver) { InitSearch(); mObserver = aObserver; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SearchStop() { KillSearch(inISearchObserver::IN_INTERRUPTED); return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SearchStep(PRBool* _retval) { return NS_OK; } NS_IMETHODIMP inCSSValueSearch::GetStringResultAt(PRInt32 aIndex, nsAString& _retval) { if (mHoldResults) { nsAutoString* result = (nsAutoString*)mResults->ElementAt(aIndex); _retval = *result; } else if (aIndex == mResultCount-1) { _retval = mLastResult; } else { return NS_ERROR_FAILURE; } return NS_OK; } NS_IMETHODIMP inCSSValueSearch::GetIntResultAt(PRInt32 aIndex, PRInt32 *_retval) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP inCSSValueSearch::GetUIntResultAt(PRInt32 aIndex, PRUint32 *_retval) { return NS_ERROR_NOT_IMPLEMENTED; } /////////////////////////////////////////////////////////////////////////////// // inICSSValueSearch NS_IMETHODIMP inCSSValueSearch::GetDocument(nsIDOMDocument** aDocument) { *aDocument = mDocument; NS_IF_ADDREF(*aDocument); return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SetDocument(nsIDOMDocument* aDocument) { mDocument = aDocument; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::GetBaseURL(PRUnichar** aBaseURL) { if (!(*aBaseURL = ToNewUnicode(mBaseURL))) return NS_ERROR_OUT_OF_MEMORY; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SetBaseURL(const PRUnichar* aBaseURL) { mBaseURL.Assign(aBaseURL); return NS_OK; } NS_IMETHODIMP inCSSValueSearch::GetReturnRelativeURLs(PRBool* aReturnRelativeURLs) { *aReturnRelativeURLs = mReturnRelativeURLs; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SetReturnRelativeURLs(PRBool aReturnRelativeURLs) { mReturnRelativeURLs = aReturnRelativeURLs; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::GetNormalizeChromeURLs(PRBool *aNormalizeChromeURLs) { *aNormalizeChromeURLs = mNormalizeChromeURLs; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SetNormalizeChromeURLs(PRBool aNormalizeChromeURLs) { mNormalizeChromeURLs = aNormalizeChromeURLs; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::AddPropertyCriteria(const PRUnichar *aPropName) { nsCSSProperty prop; mCSSUtils->LookupCSSProperty(nsDependentString(aPropName), &prop); mProperties[mPropertyCount] = prop; mPropertyCount++; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::GetTextCriteria(PRUnichar** aTextCriteria) { if (!(*aTextCriteria = ToNewUnicode(mTextCriteria))) return NS_ERROR_OUT_OF_MEMORY; return NS_OK; } NS_IMETHODIMP inCSSValueSearch::SetTextCriteria(const PRUnichar* aTextCriteria) { mTextCriteria.Assign(aTextCriteria); return NS_OK; } /////////////////////////////////////////////////////////////////////////////// // inCSSValueSearch nsresult inCSSValueSearch::InitSearch() { if (mHoldResults) { mResults = new nsVoidArray(); } mResultCount = 0; return NS_OK; } nsresult inCSSValueSearch::KillSearch(PRInt16 aResult) { mIsActive = PR_TRUE; mObserver->OnSearchEnd(this, aResult); return NS_OK; } nsresult inCSSValueSearch::SearchStyleSheet(nsIDOMCSSStyleSheet* aStyleSheet, nsIURI* aBaseURL) { nsCOMPtr baseURL; nsAutoString href; aStyleSheet->GetHref(href); if (href.IsEmpty()) baseURL = aBaseURL; else NS_NewURI(getter_AddRefs(baseURL), href, nsnull, aBaseURL); nsCOMPtr rules; nsresult rv = aStyleSheet->GetCssRules(getter_AddRefs(rules)); NS_ENSURE_SUCCESS(rv, rv); return SearchRuleList(rules, baseURL); } nsresult inCSSValueSearch::SearchRuleList(nsIDOMCSSRuleList* aRuleList, nsIURI* aBaseURL) { PRUint32 length; aRuleList->GetLength(&length); for (PRUint32 i = 0; i < length; ++i) { nsCOMPtr rule; aRuleList->Item(i, getter_AddRefs(rule)); PRUint16 type; rule->GetType(&type); switch (type) { case nsIDOMCSSRule::STYLE_RULE: { nsCOMPtr styleRule = do_QueryInterface(rule); SearchStyleRule(styleRule, aBaseURL); } break; case nsIDOMCSSRule::IMPORT_RULE: { nsCOMPtr importRule = do_QueryInterface(rule); nsCOMPtr childSheet; importRule->GetStyleSheet(getter_AddRefs(childSheet)); if (childSheet) SearchStyleSheet(childSheet, aBaseURL); } break; case nsIDOMCSSRule::MEDIA_RULE: { nsCOMPtr mediaRule = do_QueryInterface(rule); nsCOMPtr childRules; mediaRule->GetCssRules(getter_AddRefs(childRules)); SearchRuleList(childRules, aBaseURL); } break; default: // XXX handle nsIDOMCSSRule::PAGE_RULE if we ever support it break; } } return NS_OK; } nsresult inCSSValueSearch::SearchStyleRule(nsIDOMCSSStyleRule* aStyleRule, nsIURI* aBaseURL) { nsCOMPtr decl; nsresult rv = aStyleRule->GetStyle(getter_AddRefs(decl)); NS_ENSURE_SUCCESS(rv, rv); PRUint32 length; decl->GetLength(&length); nsAutoString property, value; for (PRUint32 i = 0; i < length; ++i) { decl->Item(i, property); // XXX This probably ought to use GetPropertyCSSValue if it were // implemented. decl->GetPropertyValue(property, value); SearchStyleValue(value, aBaseURL); } return NS_OK; } nsresult inCSSValueSearch::SearchStyleValue(const nsAFlatString& aValue, nsIURI* aBaseURL) { if (StringBeginsWith(aValue, NS_LITERAL_STRING("url(")) && StringEndsWith(aValue, NS_LITERAL_STRING(")"))) { const nsASingleFragmentString &url = Substring(aValue, 4, aValue.Length() - 5); // XXXldb Need to do more with |mReturnRelativeURLs|, perhaps? nsCOMPtr uri; nsresult rv = NS_NewURI(getter_AddRefs(uri), url, nsnull, aBaseURL); NS_ENSURE_SUCCESS(rv, rv); nsCAutoString spec; uri->GetSpec(spec); nsAutoString *result = new NS_ConvertUTF8toUTF16(spec); if (mReturnRelativeURLs) EqualizeURL(result); mResults->AppendElement(result); ++mResultCount; } return NS_OK; } nsresult inCSSValueSearch::EqualizeURL(nsAutoString* aURL) { if (mNormalizeChromeURLs) { if (aURL->Find("chrome://", PR_FALSE, 0, 1) >= 0) { PRUint32 len = aURL->Length(); PRUnichar* result = new PRUnichar[len-8]; const PRUnichar* src = aURL->get(); PRUint32 i = 9; PRUint32 milestone = 0; PRUint32 s = 0; while (i < len) { if (src[i] == '/') { milestone += 1; } if (milestone != 1) { result[i-9-s] = src[i]; } else { s++; } i++; } result[i-9-s] = 0; aURL->Assign(result); delete [] result; } } else { } return NS_OK; }