/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* ***** 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 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): * Original Author: David W. Hyatt (hyatt@netscape.com) * * Alternatively, the contents of this file may be used under the terms of * either of 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 "nsICSSStyleSheet.h" #include "nsIStyleRuleProcessor.h" #include "nsIDocument.h" #include "nsIContent.h" #include "nsIPresShell.h" #include "nsIXBLService.h" #include "nsIServiceManager.h" #include "nsXBLResourceLoader.h" #include "nsXBLPrototypeResources.h" #include "nsIDocumentObserver.h" #include "imgILoader.h" #include "imgIRequest.h" #include "nsICSSLoader.h" #include "nsIXBLDocumentInfo.h" #include "nsIURI.h" #include "nsNetUtil.h" #include "nsGkAtoms.h" #include "nsFrameManager.h" #include "nsStyleContext.h" #include "nsXBLPrototypeBinding.h" #include "nsCSSRuleProcessor.h" #include "nsContentUtils.h" #include "nsIScriptSecurityManager.h" NS_IMPL_CYCLE_COLLECTION_CLASS(nsXBLResourceLoader) NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(nsXBLResourceLoader) NS_IMPL_CYCLE_COLLECTION_UNLINK_NSCOMARRAY(mBoundElements) NS_IMPL_CYCLE_COLLECTION_UNLINK_END NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(nsXBLResourceLoader) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_NSCOMARRAY(mBoundElements) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(nsXBLResourceLoader) NS_INTERFACE_MAP_ENTRY(nsICSSLoaderObserver) NS_INTERFACE_MAP_ENTRY(nsISupports) NS_INTERFACE_MAP_END NS_IMPL_CYCLE_COLLECTING_ADDREF(nsXBLResourceLoader) NS_IMPL_CYCLE_COLLECTING_RELEASE(nsXBLResourceLoader) nsXBLResourceLoader::nsXBLResourceLoader(nsXBLPrototypeBinding* aBinding, nsXBLPrototypeResources* aResources) :mBinding(aBinding), mResources(aResources), mResourceList(nsnull), mLastResource(nsnull), mLoadingResources(PR_FALSE), mInLoadResourcesFunc(PR_FALSE), mPendingSheets(0) { } nsXBLResourceLoader::~nsXBLResourceLoader() { delete mResourceList; } void nsXBLResourceLoader::LoadResources(PRBool* aResult) { mInLoadResourcesFunc = PR_TRUE; if (mLoadingResources) { *aResult = (mPendingSheets == 0); mInLoadResourcesFunc = PR_FALSE; return; } mLoadingResources = PR_TRUE; *aResult = PR_TRUE; // Declare our loaders. nsCOMPtr doc; mBinding->XBLDocumentInfo()->GetDocument(getter_AddRefs(doc)); nsICSSLoader* cssLoader = doc->CSSLoader(); nsIURI *docURL = doc->GetDocumentURI(); nsIPrincipal* docPrincipal = doc->NodePrincipal(); nsCOMPtr url; for (nsXBLResource* curr = mResourceList; curr; curr = curr->mNext) { if (curr->mSrc.IsEmpty()) continue; if (NS_FAILED(NS_NewURI(getter_AddRefs(url), curr->mSrc, doc->GetDocumentCharacterSet().get(), docURL))) continue; if (curr->mType == nsGkAtoms::image) { if (!nsContentUtils::CanLoadImage(url, doc, doc, docPrincipal)) { // We're not permitted to load this image, move on... continue; } // Now kick off the image load... // Passing NULL for pretty much everything -- cause we don't care! // XXX: initialDocumentURI is NULL! nsCOMPtr req; nsContentUtils::LoadImage(url, doc, docPrincipal, docURL, nsnull, nsIRequest::LOAD_BACKGROUND, getter_AddRefs(req)); } else if (curr->mType == nsGkAtoms::stylesheet) { // Kick off the load of the stylesheet. // Always load chrome synchronously // XXXbz should that still do a content policy check? PRBool chrome; nsresult rv; if (NS_SUCCEEDED(url->SchemeIs("chrome", &chrome)) && chrome) { rv = nsContentUtils::GetSecurityManager()-> CheckLoadURIWithPrincipal(docPrincipal, url, nsIScriptSecurityManager::ALLOW_CHROME); if (NS_SUCCEEDED(rv)) { nsCOMPtr sheet; rv = cssLoader->LoadSheetSync(url, getter_AddRefs(sheet)); NS_ASSERTION(NS_SUCCEEDED(rv), "Load failed!!!"); if (NS_SUCCEEDED(rv)) { rv = StyleSheetLoaded(sheet, PR_FALSE, NS_OK); NS_ASSERTION(NS_SUCCEEDED(rv), "Processing the style sheet failed!!!"); } } } else { rv = cssLoader->LoadSheet(url, docPrincipal, this); if (NS_SUCCEEDED(rv)) ++mPendingSheets; } } } *aResult = (mPendingSheets == 0); mInLoadResourcesFunc = PR_FALSE; // Destroy our resource list. delete mResourceList; mResourceList = nsnull; } // nsICSSLoaderObserver NS_IMETHODIMP nsXBLResourceLoader::StyleSheetLoaded(nsICSSStyleSheet* aSheet, PRBool aWasAlternate, nsresult aStatus) { if (!mResources) { // Our resources got destroyed -- just bail out return NS_OK; } mResources->mStyleSheetList.AppendObject(aSheet); if (!mInLoadResourcesFunc) mPendingSheets--; if (mPendingSheets == 0) { // All stylesheets are loaded. mResources->mRuleProcessor = new nsCSSRuleProcessor(mResources->mStyleSheetList); // XXX Check for mPendingScripts when scripts also come online. if (!mInLoadResourcesFunc) NotifyBoundElements(); } return NS_OK; } void nsXBLResourceLoader::AddResource(nsIAtom* aResourceType, const nsAString& aSrc) { nsXBLResource* res = new nsXBLResource(aResourceType, aSrc); if (!res) return; if (!mResourceList) mResourceList = res; else mLastResource->mNext = res; mLastResource = res; } void nsXBLResourceLoader::AddResourceListener(nsIContent* aBoundElement) { if (aBoundElement) { mBoundElements.AppendObject(aBoundElement); } } void nsXBLResourceLoader::NotifyBoundElements() { nsCOMPtr xblService(do_GetService("@mozilla.org/xbl;1")); nsIURI* bindingURI = mBinding->BindingURI(); PRUint32 eltCount = mBoundElements.Count(); for (PRUint32 j = 0; j < eltCount; j++) { nsCOMPtr content = mBoundElements.ObjectAt(j); PRBool ready = PR_FALSE; xblService->BindingReady(content, bindingURI, &ready); if (ready) { // We need the document to flush out frame construction and // such, so we want to use the current document. nsIDocument* doc = content->GetCurrentDoc(); if (doc) { // Flush first to make sure we can get the frame for content doc->FlushPendingNotifications(Flush_Frames); // If |content| is (in addition to having binding |mBinding|) // also a descendant of another element with binding |mBinding|, // then we might have just constructed it due to the // notification of its parent. (We can know about both if the // binding loads were triggered from the DOM rather than frame // construction.) So we have to check both whether the element // has a primary frame and whether it's in the undisplayed map // before sending a ContentInserted notification, or bad things // will happen. nsIPresShell *shell = doc->GetPrimaryShell(); if (shell) { nsIFrame* childFrame = shell->GetPrimaryFrameFor(content); if (!childFrame) { // Check to see if it's in the undisplayed content map. nsStyleContext* sc = shell->FrameManager()->GetUndisplayedContent(content); if (!sc) { shell->RecreateFramesFor(content); } } } // Flush again // XXXbz why is this needed? doc->FlushPendingNotifications(Flush_ContentAndNotify); } } } // Clear out the whole array. mBoundElements.Clear(); // Delete ourselves. NS_RELEASE(mResources->mLoader); }