2007-03-22 10:30:00 -07:00
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2012-05-21 04:12:37 -07:00
/* This Source Code Form is subject to the terms of the Mozilla Public
* License , v . 2.0 . If a copy of the MPL was not distributed with this
* file , You can obtain one at http : //mozilla.org/MPL/2.0/. */
2007-03-22 10:30:00 -07:00
2011-10-10 22:50:08 -07:00
# include "mozilla/Util.h"
2007-03-22 10:30:00 -07:00
# include "nsCOMPtr.h"
# include "nsNetUtil.h"
# include "nsXBLService.h"
# include "nsXBLWindowKeyHandler.h"
# include "nsIInputStream.h"
# include "nsINameSpaceManager.h"
# include "nsHashtable.h"
# include "nsIURI.h"
# include "nsIDOMElement.h"
# include "nsIURL.h"
# include "nsIChannel.h"
# include "nsXPIDLString.h"
# include "plstr.h"
# include "nsIContent.h"
# include "nsIDocument.h"
# include "nsIXMLContentSink.h"
# include "nsContentCID.h"
2013-03-20 09:22:26 -07:00
# include "mozilla/dom/XMLDocument.h"
2007-03-22 10:30:00 -07:00
# include "nsGkAtoms.h"
# include "nsIMemory.h"
# include "nsIObserverService.h"
# include "nsIDOMNodeList.h"
# include "nsXBLContentSink.h"
# include "nsXBLBinding.h"
# include "nsXBLPrototypeBinding.h"
2010-07-14 18:53:11 -07:00
# include "nsXBLDocumentInfo.h"
2007-03-22 10:30:00 -07:00
# include "nsCRT.h"
# include "nsContentUtils.h"
# include "nsSyncLoadService.h"
# include "nsContentPolicyUtils.h"
2009-03-20 01:15:35 -07:00
# include "nsTArray.h"
2012-07-27 07:03:27 -07:00
# include "nsError.h"
2007-03-22 10:30:00 -07:00
# include "nsIPresShell.h"
# include "nsIDocumentObserver.h"
# include "nsFrameManager.h"
# include "nsStyleContext.h"
# include "nsIScriptSecurityManager.h"
# include "nsIScriptError.h"
2011-11-03 13:39:08 -07:00
# include "nsXBLSerialize.h"
2013-04-05 17:44:26 -07:00
# include "nsDOMEvent.h"
2007-03-22 10:30:00 -07:00
# ifdef MOZ_XUL
# include "nsXULPrototypeCache.h"
# endif
2011-06-24 16:12:34 -07:00
# include "nsIDOMEventListener.h"
2011-05-28 16:42:57 -07:00
# include "mozilla/Preferences.h"
2011-07-20 12:18:54 -07:00
# include "mozilla/dom/Element.h"
2012-06-18 19:30:09 -07:00
# include "mozilla/Attributes.h"
2011-05-28 16:42:57 -07:00
using namespace mozilla ;
2013-04-05 17:44:15 -07:00
using namespace mozilla : : dom ;
2007-03-22 10:30:00 -07:00
2007-04-01 05:19:44 -07:00
# define NS_MAX_XBL_BINDING_RECURSION 20
2012-07-30 07:20:58 -07:00
nsXBLService * nsXBLService : : gInstance = nullptr ;
2012-05-23 11:46:04 -07:00
2011-09-28 23:19:26 -07:00
static bool
2007-03-22 10:30:00 -07:00
IsAncestorBinding ( nsIDocument * aDocument ,
nsIURI * aChildBindingURI ,
nsIContent * aChild )
{
NS_ASSERTION ( aDocument , " expected a document " ) ;
NS_ASSERTION ( aChildBindingURI , " expected a binding URI " ) ;
NS_ASSERTION ( aChild , " expected a child content " ) ;
2012-08-22 08:56:38 -07:00
uint32_t bindingRecursion = 0 ;
2008-07-22 21:50:20 -07:00
for ( nsIContent * bindingParent = aChild - > GetBindingParent ( ) ;
bindingParent ;
bindingParent = bindingParent - > GetBindingParent ( ) ) {
2013-07-17 09:05:03 -07:00
nsXBLBinding * binding = bindingParent - > GetXBLBinding ( ) ;
2007-03-22 10:30:00 -07:00
if ( ! binding ) {
continue ;
}
2008-09-21 16:40:02 -07:00
2009-11-03 13:45:10 -08:00
if ( binding - > PrototypeBinding ( ) - > CompareBindingURI ( aChildBindingURI ) ) {
2007-04-01 05:19:44 -07:00
+ + bindingRecursion ;
if ( bindingRecursion < NS_MAX_XBL_BINDING_RECURSION ) {
continue ;
}
2012-09-01 19:35:17 -07:00
nsAutoCString spec ;
2007-03-22 10:30:00 -07:00
aChildBindingURI - > GetSpec ( spec ) ;
NS_ConvertUTF8toUTF16 bindingURI ( spec ) ;
const PRUnichar * params [ ] = { bindingURI . get ( ) } ;
2011-12-15 06:47:03 -08:00
nsContentUtils : : ReportToConsole ( nsIScriptError : : warningFlag ,
" XBL " , aDocument ,
nsContentUtils : : eXBL_PROPERTIES ,
2007-04-01 05:19:44 -07:00
" TooDeepBindingRecursion " ,
2011-12-15 06:47:03 -08:00
params , ArrayLength ( params ) ) ;
2011-10-17 07:59:28 -07:00
return true ;
2007-03-22 10:30:00 -07:00
}
}
2011-10-17 07:59:28 -07:00
return false ;
2007-03-22 10:30:00 -07:00
}
// Individual binding requests.
class nsXBLBindingRequest
{
public :
nsCOMPtr < nsIURI > mBindingURI ;
nsCOMPtr < nsIContent > mBoundElement ;
void DocumentLoaded ( nsIDocument * aBindingDoc )
{
// We only need the document here to cause frame construction, so
// we need the current doc, not the owner doc.
nsIDocument * doc = mBoundElement - > GetCurrentDoc ( ) ;
if ( ! doc )
return ;
// Get the binding.
2011-09-28 23:19:26 -07:00
bool ready = false ;
2012-05-23 11:46:04 -07:00
nsXBLService : : GetInstance ( ) - > BindingReady ( mBoundElement , mBindingURI , & ready ) ;
2007-03-22 10:30:00 -07:00
if ( ! ready )
return ;
// If |mBoundElement| 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.
2010-06-25 06:59:57 -07:00
nsIPresShell * shell = doc - > GetShell ( ) ;
2007-03-22 10:30:00 -07:00
if ( shell ) {
2009-12-24 13:20:05 -08:00
nsIFrame * childFrame = mBoundElement - > GetPrimaryFrame ( ) ;
2007-03-22 10:30:00 -07:00
if ( ! childFrame ) {
// Check to see if it's in the undisplayed content map.
nsStyleContext * sc =
shell - > FrameManager ( ) - > GetUndisplayedContent ( mBoundElement ) ;
if ( ! sc ) {
shell - > RecreateFramesFor ( mBoundElement ) ;
}
}
}
}
nsXBLBindingRequest ( nsIURI * aURI , nsIContent * aBoundElement )
: mBindingURI ( aURI ) ,
mBoundElement ( aBoundElement )
{
}
} ;
// nsXBLStreamListener, a helper class used for
// asynchronous parsing of URLs
/* Header file */
2012-06-18 19:30:09 -07:00
class nsXBLStreamListener MOZ_FINAL : public nsIStreamListener ,
public nsIDOMEventListener
2007-03-22 10:30:00 -07:00
{
public :
NS_DECL_ISUPPORTS
NS_DECL_NSISTREAMLISTENER
NS_DECL_NSIREQUESTOBSERVER
2011-06-24 16:12:34 -07:00
NS_DECL_NSIDOMEVENTLISTENER
2007-03-22 10:30:00 -07:00
2012-05-23 11:46:04 -07:00
nsXBLStreamListener ( nsIDocument * aBoundDocument ,
2009-01-14 03:24:26 -08:00
nsIXMLContentSink * aSink ,
nsIDocument * aBindingDocument ) ;
2007-11-07 16:05:03 -08:00
~ nsXBLStreamListener ( ) ;
2007-04-23 07:21:53 -07:00
void AddRequest ( nsXBLBindingRequest * aRequest ) { mBindingRequests . AppendElement ( aRequest ) ; }
2011-09-28 23:19:26 -07:00
bool HasRequest ( nsIURI * aURI , nsIContent * aBoundElement ) ;
2007-03-22 10:30:00 -07:00
private :
nsCOMPtr < nsIStreamListener > mInner ;
2009-03-20 01:15:35 -07:00
nsAutoTArray < nsXBLBindingRequest * , 8 > mBindingRequests ;
2007-03-22 10:30:00 -07:00
2009-01-14 03:24:26 -08:00
nsCOMPtr < nsIWeakReference > mBoundDocument ;
nsCOMPtr < nsIXMLContentSink > mSink ; // Only set until OnStartRequest
nsCOMPtr < nsIDocument > mBindingDocument ; // Only set until OnStartRequest
2007-03-22 10:30:00 -07:00
} ;
/* Implementation file */
2011-06-24 16:12:34 -07:00
NS_IMPL_ISUPPORTS3 ( nsXBLStreamListener ,
nsIStreamListener ,
nsIRequestObserver ,
nsIDOMEventListener )
2007-03-22 10:30:00 -07:00
2012-05-23 11:46:04 -07:00
nsXBLStreamListener : : nsXBLStreamListener ( nsIDocument * aBoundDocument ,
2009-01-14 03:24:26 -08:00
nsIXMLContentSink * aSink ,
nsIDocument * aBindingDocument )
: mSink ( aSink ) , mBindingDocument ( aBindingDocument )
2007-03-22 10:30:00 -07:00
{
/* member initializers and constructor code */
2009-01-14 03:24:26 -08:00
mBoundDocument = do_GetWeakReference ( aBoundDocument ) ;
2007-11-07 16:05:03 -08:00
}
nsXBLStreamListener : : ~ nsXBLStreamListener ( )
{
2012-08-22 08:56:38 -07:00
for ( uint32_t i = 0 ; i < mBindingRequests . Length ( ) ; i + + ) {
2009-03-20 01:15:35 -07:00
nsXBLBindingRequest * req = mBindingRequests . ElementAt ( i ) ;
2013-03-05 04:37:47 -08:00
delete req ;
2007-11-07 16:05:03 -08:00
}
2007-03-22 10:30:00 -07:00
}
NS_IMETHODIMP
2012-09-05 19:41:02 -07:00
nsXBLStreamListener : : OnDataAvailable ( nsIRequest * request , nsISupports * aCtxt ,
2013-03-05 04:37:47 -08:00
nsIInputStream * aInStr ,
2012-09-05 19:41:02 -07:00
uint64_t aSourceOffset , uint32_t aCount )
2007-03-22 10:30:00 -07:00
{
if ( mInner )
return mInner - > OnDataAvailable ( request , aCtxt , aInStr , aSourceOffset , aCount ) ;
return NS_ERROR_FAILURE ;
}
NS_IMETHODIMP
nsXBLStreamListener : : OnStartRequest ( nsIRequest * request , nsISupports * aCtxt )
{
2009-01-14 03:24:26 -08:00
// Make sure we don't hold on to the sink and binding document past this point
nsCOMPtr < nsIXMLContentSink > sink ;
mSink . swap ( sink ) ;
nsCOMPtr < nsIDocument > doc ;
mBindingDocument . swap ( doc ) ;
nsCOMPtr < nsIChannel > channel = do_QueryInterface ( request ) ;
NS_ENSURE_TRUE ( channel , NS_ERROR_UNEXPECTED ) ;
nsCOMPtr < nsILoadGroup > group ;
request - > GetLoadGroup ( getter_AddRefs ( group ) ) ;
nsresult rv = doc - > StartDocumentLoad ( " loadAsInteractiveData " ,
channel ,
group ,
2012-07-30 07:20:58 -07:00
nullptr ,
2009-01-14 03:24:26 -08:00
getter_AddRefs ( mInner ) ,
2011-10-17 07:59:28 -07:00
true ,
2009-01-14 03:24:26 -08:00
sink ) ;
NS_ENSURE_SUCCESS ( rv , rv ) ;
// Make sure to add ourselves as a listener after StartDocumentLoad,
// since that resets the event listners on the document.
2013-04-05 17:44:15 -07:00
doc - > AddEventListener ( NS_LITERAL_STRING ( " load " ) , this , false ) ;
2009-01-14 03:24:26 -08:00
return mInner - > OnStartRequest ( request , aCtxt ) ;
2007-03-22 10:30:00 -07:00
}
NS_IMETHODIMP
nsXBLStreamListener : : OnStopRequest ( nsIRequest * request , nsISupports * aCtxt , nsresult aStatus )
{
nsresult rv = NS_OK ;
if ( mInner ) {
rv = mInner - > OnStopRequest ( request , aCtxt , aStatus ) ;
}
2007-11-07 16:05:03 -08:00
// Don't hold onto the inner listener; holding onto it can create a cycle
// with the document
2012-07-30 07:20:58 -07:00
mInner = nullptr ;
2007-03-22 10:30:00 -07:00
return rv ;
}
2011-09-28 23:19:26 -07:00
bool
2007-03-22 10:30:00 -07:00
nsXBLStreamListener : : HasRequest ( nsIURI * aURI , nsIContent * aElt )
{
// XXX Could be more efficient.
2012-08-22 08:56:38 -07:00
uint32_t count = mBindingRequests . Length ( ) ;
for ( uint32_t i = 0 ; i < count ; i + + ) {
2009-03-20 01:15:35 -07:00
nsXBLBindingRequest * req = mBindingRequests . ElementAt ( i ) ;
2011-09-28 23:19:26 -07:00
bool eq ;
2007-03-22 10:30:00 -07:00
if ( req - > mBoundElement = = aElt & &
NS_SUCCEEDED ( req - > mBindingURI - > Equals ( aURI , & eq ) ) & & eq )
2011-10-17 07:59:28 -07:00
return true ;
2007-03-22 10:30:00 -07:00
}
2011-10-17 07:59:28 -07:00
return false ;
2007-03-22 10:30:00 -07:00
}
nsresult
2011-06-24 16:12:34 -07:00
nsXBLStreamListener : : HandleEvent ( nsIDOMEvent * aEvent )
2007-03-22 10:30:00 -07:00
{
nsresult rv = NS_OK ;
2012-08-22 08:56:38 -07:00
uint32_t i ;
uint32_t count = mBindingRequests . Length ( ) ;
2007-11-07 16:05:03 -08:00
// Get the binding document; note that we don't hold onto it in this object
// to avoid creating a cycle
2013-04-05 17:44:26 -07:00
nsDOMEvent * event = aEvent - > InternalDOMEvent ( ) ;
EventTarget * target = event - > GetCurrentTarget ( ) ;
2007-11-07 16:05:03 -08:00
nsCOMPtr < nsIDocument > bindingDocument = do_QueryInterface ( target ) ;
NS_ASSERTION ( bindingDocument , " Event not targeted at document?! " ) ;
2007-03-22 10:30:00 -07:00
// See if we're still alive.
2009-01-14 03:24:26 -08:00
nsCOMPtr < nsIDocument > doc ( do_QueryReferent ( mBoundDocument ) ) ;
2007-03-22 10:30:00 -07:00
if ( ! doc ) {
NS_WARNING ( " XBL load did not complete until after document went away! Modal dialog bug? \n " ) ;
}
else {
// We have to do a flush prior to notification of the document load.
// This has to happen since the HTML content sink can be holding on
// to notifications related to our children (e.g., if you bind to the
// <body> tag) that result in duplication of content.
// We need to get the sink's notifications flushed and then make the binding
// ready.
if ( count > 0 ) {
2009-03-20 01:15:35 -07:00
nsXBLBindingRequest * req = mBindingRequests . ElementAt ( 0 ) ;
2007-03-22 10:30:00 -07:00
nsIDocument * document = req - > mBoundElement - > GetCurrentDoc ( ) ;
if ( document )
document - > FlushPendingNotifications ( Flush_ContentAndNotify ) ;
}
// Remove ourselves from the set of pending docs.
nsBindingManager * bindingManager = doc - > BindingManager ( ) ;
2007-11-07 16:05:03 -08:00
nsIURI * documentURI = bindingDocument - > GetDocumentURI ( ) ;
2007-03-22 10:30:00 -07:00
bindingManager - > RemoveLoadingDocListener ( documentURI ) ;
2010-04-30 06:12:05 -07:00
if ( ! bindingDocument - > GetRootElement ( ) ) {
2010-05-04 21:28:19 -07:00
// FIXME: How about an error console warning?
2012-04-08 18:37:41 -07:00
NS_WARNING ( " XBL doc with no root element - this usually shouldn't happen " ) ;
2007-03-22 10:30:00 -07:00
return NS_ERROR_FAILURE ;
}
// Put our doc info in the doc table.
2007-11-07 16:05:03 -08:00
nsBindingManager * xblDocBindingManager = bindingDocument - > BindingManager ( ) ;
2010-07-14 18:53:11 -07:00
nsRefPtr < nsXBLDocumentInfo > info =
2007-03-22 10:30:00 -07:00
xblDocBindingManager - > GetXBLDocumentInfo ( documentURI ) ;
xblDocBindingManager - > RemoveXBLDocumentInfo ( info ) ; // Break the self-imposed cycle.
if ( ! info ) {
2011-11-03 13:39:07 -07:00
if ( nsXBLService : : IsChromeOrResourceURI ( documentURI ) ) {
2010-05-04 21:28:19 -07:00
NS_WARNING ( " An XBL file is malformed. Did you forget the XBL namespace on the bindings tag? " ) ;
}
2011-12-15 06:47:03 -08:00
nsContentUtils : : ReportToConsole ( nsIScriptError : : warningFlag ,
2012-07-30 07:20:58 -07:00
" XBL " , nullptr ,
2011-12-15 06:47:03 -08:00
nsContentUtils : : eXBL_PROPERTIES ,
2010-05-04 21:28:19 -07:00
" MalformedXBL " ,
2012-07-30 07:20:58 -07:00
nullptr , 0 , documentURI ) ;
2007-03-22 10:30:00 -07:00
return NS_ERROR_FAILURE ;
}
// If the doc is a chrome URI, then we put it into the XUL cache.
# ifdef MOZ_XUL
2011-11-03 13:39:07 -07:00
if ( nsXBLService : : IsChromeOrResourceURI ( documentURI ) ) {
2007-03-22 10:30:00 -07:00
nsXULPrototypeCache * cache = nsXULPrototypeCache : : GetInstance ( ) ;
if ( cache & & cache - > IsEnabled ( ) )
cache - > PutXBLDocumentInfo ( info ) ;
}
# endif
bindingManager - > PutXBLDocumentInfo ( info ) ;
// Notify all pending requests that their bindings are
// ready and can be installed.
for ( i = 0 ; i < count ; i + + ) {
2009-03-20 01:15:35 -07:00
nsXBLBindingRequest * req = mBindingRequests . ElementAt ( i ) ;
2007-11-07 16:05:03 -08:00
req - > DocumentLoaded ( bindingDocument ) ;
2007-03-22 10:30:00 -07:00
}
}
2011-10-17 07:59:28 -07:00
target - > RemoveEventListener ( NS_LITERAL_STRING ( " load " ) , this , false ) ;
2007-03-22 10:30:00 -07:00
return rv ;
}
// Implementation /////////////////////////////////////////////////////////////////
// Static member variable initialization
2011-09-28 23:19:26 -07:00
bool nsXBLService : : gAllowDataURIs = false ;
2008-04-28 16:56:07 -07:00
2012-07-30 07:20:58 -07:00
nsHashtable * nsXBLService : : gClassTable = nullptr ;
2007-03-22 10:30:00 -07:00
JSCList nsXBLService : : gClassLRUList = JS_INIT_STATIC_CLIST ( & nsXBLService : : gClassLRUList ) ;
2012-08-22 08:56:38 -07:00
uint32_t nsXBLService : : gClassLRUListLength = 0 ;
uint32_t nsXBLService : : gClassLRUListQuota = 64 ;
2007-03-22 10:30:00 -07:00
// Implement our nsISupports methods
2012-05-23 11:46:04 -07:00
NS_IMPL_ISUPPORTS2 ( nsXBLService , nsIObserver , nsISupportsWeakReference )
void
nsXBLService : : Init ( )
{
gInstance = new nsXBLService ( ) ;
NS_ADDREF ( gInstance ) ;
// Register the first (and only) nsXBLService as a memory pressure observer
// so it can flush the LRU list in low-memory situations.
nsCOMPtr < nsIObserverService > os = mozilla : : services : : GetObserverService ( ) ;
if ( os )
os - > AddObserver ( gInstance , " memory-pressure " , true ) ;
}
2007-03-22 10:30:00 -07:00
// Constructors/Destructors
nsXBLService : : nsXBLService ( void )
{
2012-05-23 11:46:04 -07:00
gClassTable = new nsHashtable ( ) ;
2011-05-28 16:42:57 -07:00
Preferences : : AddBoolVarCache ( & gAllowDataURIs , " layout.debug.enable_data_xbl " ) ;
2007-03-22 10:30:00 -07:00
}
nsXBLService : : ~ nsXBLService ( void )
{
2012-05-23 11:46:04 -07:00
// Walk the LRU list removing and deleting the nsXBLJSClasses.
FlushMemory ( ) ;
// Any straggling nsXBLJSClass instances held by unfinalized JS objects
// created for bindings will be deleted when those objects are finalized
// (and not put on gClassLRUList, because length >= quota).
gClassLRUListLength = gClassLRUListQuota = 0 ;
// At this point, the only hash table entries should be for referenced
// XBL class structs held by unfinalized JS binding objects.
delete gClassTable ;
2012-07-30 07:20:58 -07:00
gClassTable = nullptr ;
2007-03-22 10:30:00 -07:00
}
2011-11-03 13:39:07 -07:00
// static
bool
nsXBLService : : IsChromeOrResourceURI ( nsIURI * aURI )
{
bool isChrome = false ;
bool isResource = false ;
if ( NS_SUCCEEDED ( aURI - > SchemeIs ( " chrome " , & isChrome ) ) & &
NS_SUCCEEDED ( aURI - > SchemeIs ( " resource " , & isResource ) ) )
return ( isChrome | | isResource ) ;
return false ;
}
2007-03-22 10:30:00 -07:00
// This function loads a particular XBL file and installs all of the bindings
// onto the element.
2012-05-23 11:46:04 -07:00
nsresult
2007-07-18 14:56:57 -07:00
nsXBLService : : LoadBindings ( nsIContent * aContent , nsIURI * aURL ,
2013-01-24 09:45:50 -08:00
nsIPrincipal * aOriginPrincipal ,
2011-09-28 23:19:26 -07:00
nsXBLBinding * * aBinding , bool * aResolveStyle )
2007-07-18 14:56:57 -07:00
{
NS_PRECONDITION ( aOriginPrincipal , " Must have an origin principal " ) ;
2012-07-30 07:20:58 -07:00
* aBinding = nullptr ;
2011-10-17 07:59:28 -07:00
* aResolveStyle = false ;
2007-03-22 10:30:00 -07:00
nsresult rv ;
2011-10-18 03:53:36 -07:00
nsCOMPtr < nsIDocument > document = aContent - > OwnerDoc ( ) ;
2007-03-22 10:30:00 -07:00
2012-09-01 19:35:17 -07:00
nsAutoCString urlspec ;
2008-11-13 16:00:11 -08:00
if ( nsContentUtils : : GetWrapperSafeScriptFilename ( document , aURL , urlspec ) ) {
// Block an attempt to load a binding that has special wrapper
// automation needs.
return NS_OK ;
}
2013-07-17 09:05:03 -07:00
nsXBLBinding * binding = aContent - > GetXBLBinding ( ) ;
2013-01-24 09:45:50 -08:00
if ( binding ) {
2013-07-11 08:58:28 -07:00
if ( binding - > MarkedForDeath ( ) ) {
FlushStyleBindings ( aContent ) ;
binding = nullptr ;
}
else {
// See if the URIs match.
if ( binding - > PrototypeBinding ( ) - > CompareBindingURI ( aURL ) )
return NS_OK ;
FlushStyleBindings ( aContent ) ;
binding = nullptr ;
2007-03-22 10:30:00 -07:00
}
}
2011-09-28 23:19:26 -07:00
bool ready ;
2007-03-22 10:30:00 -07:00
nsRefPtr < nsXBLBinding > newBinding ;
2011-10-17 07:59:28 -07:00
if ( NS_FAILED ( rv = GetBinding ( aContent , aURL , false , aOriginPrincipal ,
2007-07-18 14:56:57 -07:00
& ready , getter_AddRefs ( newBinding ) ) ) ) {
2007-03-22 10:30:00 -07:00
return rv ;
}
if ( ! newBinding ) {
# ifdef DEBUG
2012-09-01 19:35:17 -07:00
nsAutoCString spec ;
2007-03-22 10:30:00 -07:00
aURL - > GetSpec ( spec ) ;
2012-09-01 19:35:17 -07:00
nsAutoCString str ( NS_LITERAL_CSTRING ( " Failed to locate XBL binding. XBL is now using id instead of name to reference bindings. Make sure you have switched over. The invalid binding name is: " ) + spec ) ;
2007-03-22 10:30:00 -07:00
NS_ERROR ( str . get ( ) ) ;
# endif
return NS_OK ;
}
if ( : : IsAncestorBinding ( document , aURL , aContent ) ) {
return NS_ERROR_ILLEGAL_VALUE ;
}
2013-01-24 09:45:50 -08:00
// We loaded a style binding. It goes on the end.
if ( binding ) {
// Get the last binding that is in the append layer.
binding - > RootBinding ( ) - > SetBaseBinding ( newBinding ) ;
2007-03-22 10:30:00 -07:00
}
else {
2013-01-24 09:45:50 -08:00
// Install the binding on the content node.
2013-07-17 09:05:03 -07:00
aContent - > SetXBLBinding ( newBinding ) ;
2007-03-22 10:30:00 -07:00
}
2009-11-18 07:14:14 -08:00
{
nsAutoScriptBlocker scriptBlocker ;
2007-03-22 10:30:00 -07:00
2009-11-18 07:14:14 -08:00
// Set the binding's bound element.
newBinding - > SetBoundElement ( aContent ) ;
2007-03-22 10:30:00 -07:00
2009-11-18 07:14:14 -08:00
// Tell the binding to build the anonymous content.
newBinding - > GenerateAnonymousContent ( ) ;
2008-03-14 16:08:57 -07:00
2009-11-18 07:14:14 -08:00
// Tell the binding to install event handlers
newBinding - > InstallEventHandlers ( ) ;
2008-03-14 16:08:57 -07:00
2009-11-18 07:14:14 -08:00
// Set up our properties
rv = newBinding - > InstallImplementation ( ) ;
NS_ENSURE_SUCCESS ( rv , rv ) ;
// Figure out if we have any scoped sheets. If so, we do a second resolve.
* aResolveStyle = newBinding - > HasStyleSheets ( ) ;
2007-03-22 10:30:00 -07:00
2009-11-18 07:14:14 -08:00
newBinding . swap ( * aBinding ) ;
}
2007-10-26 17:14:43 -07:00
2007-03-22 10:30:00 -07:00
return NS_OK ;
}
nsresult
nsXBLService : : FlushStyleBindings ( nsIContent * aContent )
{
2011-10-18 03:53:36 -07:00
nsCOMPtr < nsIDocument > document = aContent - > OwnerDoc ( ) ;
2007-03-22 10:30:00 -07:00
2013-07-17 09:05:03 -07:00
nsXBLBinding * binding = aContent - > GetXBLBinding ( ) ;
2007-03-22 10:30:00 -07:00
if ( binding ) {
2013-07-11 08:58:28 -07:00
// Clear out the script references.
binding - > ChangeDocument ( document , nullptr ) ;
2007-03-22 10:30:00 -07:00
2013-07-17 09:05:03 -07:00
aContent - > SetXBLBinding ( nullptr ) ; // Flush old style bindings
2007-03-22 10:30:00 -07:00
}
return NS_OK ;
}
//
// AttachGlobalKeyHandler
//
// Creates a new key handler and prepares to listen to key events on the given
// event receiver (either a document or an content node). If the receiver is content,
// then extra work needs to be done to hook it up to the document (XXX WHY??)
//
2012-05-23 11:46:04 -07:00
nsresult
2013-04-05 17:44:26 -07:00
nsXBLService : : AttachGlobalKeyHandler ( EventTarget * aTarget )
2007-03-22 10:30:00 -07:00
{
// check if the receiver is a content node (not a document), and hook
// it to the document if that is the case.
2013-04-05 17:44:26 -07:00
nsCOMPtr < EventTarget > piTarget = aTarget ;
2007-05-14 02:11:38 -07:00
nsCOMPtr < nsIContent > contentNode ( do_QueryInterface ( aTarget ) ) ;
2007-03-22 10:30:00 -07:00
if ( contentNode ) {
// Only attach if we're really in a document
nsCOMPtr < nsIDocument > doc = contentNode - > GetCurrentDoc ( ) ;
if ( doc )
2011-06-23 19:18:02 -07:00
piTarget = doc ; // We're a XUL keyset. Attach to our document.
2007-03-22 10:30:00 -07:00
}
2011-06-23 19:18:02 -07:00
2011-10-17 07:59:28 -07:00
nsEventListenerManager * manager = piTarget - > GetListenerManager ( true ) ;
2013-04-05 17:44:26 -07:00
2011-06-23 19:18:02 -07:00
if ( ! piTarget | | ! manager )
2007-03-22 10:30:00 -07:00
return NS_ERROR_FAILURE ;
2008-08-06 07:32:09 -07:00
// the listener already exists, so skip this
if ( contentNode & & contentNode - > GetProperty ( nsGkAtoms : : listener ) )
return NS_OK ;
2013-04-14 11:27:33 -07:00
2007-03-22 10:30:00 -07:00
nsCOMPtr < nsIDOMElement > elt ( do_QueryInterface ( contentNode ) ) ;
// Create the key handler
2013-04-14 11:27:33 -07:00
nsRefPtr < nsXBLWindowKeyHandler > handler =
NS_NewXBLWindowKeyHandler ( elt , piTarget ) ;
2007-03-22 10:30:00 -07:00
// listen to these events
2011-06-23 19:18:02 -07:00
manager - > AddEventListenerByType ( handler , NS_LITERAL_STRING ( " keydown " ) ,
2012-12-15 17:26:05 -08:00
dom : : TrustedEventsAtSystemGroupBubble ( ) ) ;
2011-06-23 19:18:02 -07:00
manager - > AddEventListenerByType ( handler , NS_LITERAL_STRING ( " keyup " ) ,
2012-12-15 17:26:05 -08:00
dom : : TrustedEventsAtSystemGroupBubble ( ) ) ;
2011-06-23 19:18:02 -07:00
manager - > AddEventListenerByType ( handler , NS_LITERAL_STRING ( " keypress " ) ,
2012-12-15 17:26:05 -08:00
dom : : TrustedEventsAtSystemGroupBubble ( ) ) ;
2007-03-22 10:30:00 -07:00
2008-08-06 07:32:09 -07:00
if ( contentNode )
2013-04-14 11:27:33 -07:00
return contentNode - > SetProperty ( nsGkAtoms : : listener , handler . forget ( ) . get ( ) ,
2011-10-17 07:59:28 -07:00
nsPropertyTable : : SupportsDtorFunc , true ) ;
2008-08-06 07:32:09 -07:00
2013-04-14 11:27:33 -07:00
// The reference to the handler will be maintained by the event target,
2008-08-06 07:32:09 -07:00
// and, if there is a content node, the property.
return NS_OK ;
}
//
// DetachGlobalKeyHandler
//
// Removes a key handler added by DeatchGlobalKeyHandler.
//
2012-05-23 11:46:04 -07:00
nsresult
2013-04-05 17:44:26 -07:00
nsXBLService : : DetachGlobalKeyHandler ( EventTarget * aTarget )
2008-08-06 07:32:09 -07:00
{
2013-04-05 17:44:26 -07:00
nsCOMPtr < EventTarget > piTarget = aTarget ;
2008-08-06 07:32:09 -07:00
nsCOMPtr < nsIContent > contentNode ( do_QueryInterface ( aTarget ) ) ;
if ( ! contentNode ) // detaching is only supported for content nodes
return NS_ERROR_FAILURE ;
// Only attach if we're really in a document
nsCOMPtr < nsIDocument > doc = contentNode - > GetCurrentDoc ( ) ;
if ( doc )
piTarget = do_QueryInterface ( doc ) ;
2011-06-23 19:18:02 -07:00
2011-10-17 07:59:28 -07:00
nsEventListenerManager * manager = piTarget - > GetListenerManager ( true ) ;
2013-04-05 17:44:26 -07:00
2011-06-23 19:18:02 -07:00
if ( ! piTarget | | ! manager )
2008-08-06 07:32:09 -07:00
return NS_ERROR_FAILURE ;
nsIDOMEventListener * handler =
static_cast < nsIDOMEventListener * > ( contentNode - > GetProperty ( nsGkAtoms : : listener ) ) ;
if ( ! handler )
return NS_ERROR_FAILURE ;
2011-06-23 19:18:02 -07:00
manager - > RemoveEventListenerByType ( handler , NS_LITERAL_STRING ( " keydown " ) ,
2012-12-15 17:26:05 -08:00
dom : : TrustedEventsAtSystemGroupBubble ( ) ) ;
2011-06-23 19:18:02 -07:00
manager - > RemoveEventListenerByType ( handler , NS_LITERAL_STRING ( " keyup " ) ,
2012-12-15 17:26:05 -08:00
dom : : TrustedEventsAtSystemGroupBubble ( ) ) ;
2011-06-23 19:18:02 -07:00
manager - > RemoveEventListenerByType ( handler , NS_LITERAL_STRING ( " keypress " ) ,
2012-12-15 17:26:05 -08:00
dom : : TrustedEventsAtSystemGroupBubble ( ) ) ;
2008-08-06 07:32:09 -07:00
contentNode - > DeleteProperty ( nsGkAtoms : : listener ) ;
2007-03-22 10:30:00 -07:00
return NS_OK ;
}
NS_IMETHODIMP
nsXBLService : : Observe ( nsISupports * aSubject , const char * aTopic , const PRUnichar * aSomeData )
{
if ( nsCRT : : strcmp ( aTopic , " memory-pressure " ) = = 0 )
FlushMemory ( ) ;
return NS_OK ;
}
nsresult
nsXBLService : : FlushMemory ( )
{
while ( ! JS_CLIST_IS_EMPTY ( & gClassLRUList ) ) {
JSCList * lru = gClassLRUList . next ;
2007-07-08 00:08:04 -07:00
nsXBLJSClass * c = static_cast < nsXBLJSClass * > ( lru ) ;
2007-03-22 10:30:00 -07:00
JS_REMOVE_AND_INIT_LINK ( lru ) ;
delete c ;
gClassLRUListLength - - ;
}
return NS_OK ;
}
// Internal helper methods ////////////////////////////////////////////////////////////////
2012-05-23 11:46:04 -07:00
nsresult
nsXBLService : : BindingReady ( nsIContent * aBoundElement ,
nsIURI * aURI ,
bool * aIsReady )
2007-03-22 10:30:00 -07:00
{
2007-07-18 14:56:57 -07:00
// Don't do a security check here; we know this binding is set to go.
2012-07-30 07:20:58 -07:00
return GetBinding ( aBoundElement , aURI , true , nullptr , aIsReady , nullptr ) ;
2007-03-22 10:30:00 -07:00
}
nsresult
nsXBLService : : GetBinding ( nsIContent * aBoundElement , nsIURI * aURI ,
2011-09-28 23:19:26 -07:00
bool aPeekOnly , nsIPrincipal * aOriginPrincipal ,
bool * aIsReady , nsXBLBinding * * aResult )
2007-03-22 10:30:00 -07:00
{
// More than 6 binding URIs are rare, see bug 55070 comment 18.
2007-12-12 20:17:14 -08:00
nsAutoTArray < nsIURI * , 6 > uris ;
2007-07-18 14:56:57 -07:00
return GetBinding ( aBoundElement , aURI , aPeekOnly , aOriginPrincipal , aIsReady ,
aResult , uris ) ;
2007-03-22 10:30:00 -07:00
}
nsresult
nsXBLService : : GetBinding ( nsIContent * aBoundElement , nsIURI * aURI ,
2011-09-28 23:19:26 -07:00
bool aPeekOnly , nsIPrincipal * aOriginPrincipal ,
bool * aIsReady , nsXBLBinding * * aResult ,
2007-03-22 10:30:00 -07:00
nsTArray < nsIURI * > & aDontExtendURIs )
{
NS_ASSERTION ( aPeekOnly | | aResult ,
" Must have non-null out param if not just peeking to see "
" whether the binding is ready " ) ;
if ( aResult )
2012-07-30 07:20:58 -07:00
* aResult = nullptr ;
2007-03-22 10:30:00 -07:00
if ( ! aURI )
return NS_ERROR_FAILURE ;
2012-09-01 19:35:17 -07:00
nsAutoCString ref ;
2011-05-21 18:12:46 -07:00
aURI - > GetRef ( ref ) ;
2007-03-22 10:30:00 -07:00
2011-10-18 03:53:36 -07:00
nsCOMPtr < nsIDocument > boundDocument = aBoundElement - > OwnerDoc ( ) ;
2007-03-22 10:30:00 -07:00
2010-07-14 18:53:11 -07:00
nsRefPtr < nsXBLDocumentInfo > docInfo ;
2007-07-18 14:56:57 -07:00
nsresult rv = LoadBindingDocumentInfo ( aBoundElement , boundDocument , aURI ,
aOriginPrincipal ,
2011-10-17 07:59:28 -07:00
false , getter_AddRefs ( docInfo ) ) ;
2007-07-18 14:56:57 -07:00
NS_ENSURE_SUCCESS ( rv , rv ) ;
2007-03-22 10:30:00 -07:00
if ( ! docInfo )
return NS_ERROR_FAILURE ;
2010-07-14 18:55:54 -07:00
nsXBLPrototypeBinding * protoBinding = docInfo - > GetPrototypeBinding ( ref ) ;
2007-03-22 10:30:00 -07:00
2012-05-09 15:57:44 -07:00
if ( ! protoBinding ) {
# ifdef DEBUG
2012-09-01 19:35:17 -07:00
nsAutoCString uriSpec ;
2012-05-09 15:57:44 -07:00
aURI - > GetSpec ( uriSpec ) ;
2012-09-01 19:35:17 -07:00
nsAutoCString doc ;
2012-05-09 15:57:44 -07:00
boundDocument - > GetDocumentURI ( ) - > GetSpec ( doc ) ;
2012-09-01 19:35:17 -07:00
nsAutoCString message ( " Unable to locate an XBL binding for URI " ) ;
2012-05-09 15:57:44 -07:00
message + = uriSpec ;
message + = " in document " ;
message + = doc ;
NS_WARNING ( message . get ( ) ) ;
# endif
2007-03-22 10:30:00 -07:00
return NS_ERROR_FAILURE ;
2012-05-09 15:57:44 -07:00
}
2007-03-22 10:30:00 -07:00
2009-11-03 13:45:10 -08:00
NS_ENSURE_TRUE ( aDontExtendURIs . AppendElement ( protoBinding - > BindingURI ( ) ) ,
NS_ERROR_OUT_OF_MEMORY ) ;
nsCOMPtr < nsIURI > altBindingURI = protoBinding - > AlternateBindingURI ( ) ;
if ( altBindingURI ) {
NS_ENSURE_TRUE ( aDontExtendURIs . AppendElement ( altBindingURI ) ,
NS_ERROR_OUT_OF_MEMORY ) ;
}
2007-03-22 10:30:00 -07:00
// Our prototype binding must have all its resources loaded.
2011-09-28 23:19:26 -07:00
bool ready = protoBinding - > LoadResources ( ) ;
2007-03-22 10:30:00 -07:00
if ( ! ready ) {
// Add our bound element to the protos list of elts that should
// be notified when the stylesheets and scripts finish loading.
protoBinding - > AddResourceListener ( aBoundElement ) ;
return NS_ERROR_FAILURE ; // The binding isn't ready yet.
}
2011-11-03 13:39:07 -07:00
rv = protoBinding - > ResolveBaseBinding ( ) ;
NS_ENSURE_SUCCESS ( rv , rv ) ;
nsIURI * baseBindingURI ;
2007-03-22 10:30:00 -07:00
nsXBLPrototypeBinding * baseProto = protoBinding - > GetBasePrototype ( ) ;
if ( baseProto ) {
2011-11-03 13:39:07 -07:00
baseBindingURI = baseProto - > BindingURI ( ) ;
2007-03-22 10:30:00 -07:00
}
2011-11-03 13:39:07 -07:00
else {
baseBindingURI = protoBinding - > GetBaseBindingURI ( ) ;
if ( baseBindingURI ) {
2012-08-22 08:56:38 -07:00
uint32_t count = aDontExtendURIs . Length ( ) ;
for ( uint32_t index = 0 ; index < count ; + + index ) {
2011-11-03 13:39:07 -07:00
bool equal ;
rv = aDontExtendURIs [ index ] - > Equals ( baseBindingURI , & equal ) ;
2007-03-22 10:30:00 -07:00
NS_ENSURE_SUCCESS ( rv , rv ) ;
2011-11-03 13:39:07 -07:00
if ( equal ) {
2012-09-01 19:35:17 -07:00
nsAutoCString spec , basespec ;
2011-11-03 13:39:07 -07:00
protoBinding - > BindingURI ( ) - > GetSpec ( spec ) ;
NS_ConvertUTF8toUTF16 protoSpec ( spec ) ;
baseBindingURI - > GetSpec ( basespec ) ;
NS_ConvertUTF8toUTF16 baseSpecUTF16 ( basespec ) ;
const PRUnichar * params [ ] = { protoSpec . get ( ) , baseSpecUTF16 . get ( ) } ;
2011-12-15 06:47:03 -08:00
nsContentUtils : : ReportToConsole ( nsIScriptError : : warningFlag ,
2012-07-30 07:20:58 -07:00
" XBL " , nullptr ,
2011-12-15 06:47:03 -08:00
nsContentUtils : : eXBL_PROPERTIES ,
2011-11-03 13:39:07 -07:00
" CircularExtendsBinding " ,
2011-12-15 06:47:03 -08:00
params , ArrayLength ( params ) ,
boundDocument - > GetDocumentURI ( ) ) ;
2011-11-03 13:39:07 -07:00
return NS_ERROR_ILLEGAL_VALUE ;
2007-03-22 10:30:00 -07:00
}
}
}
}
2011-11-03 13:39:07 -07:00
nsRefPtr < nsXBLBinding > baseBinding ;
if ( baseBindingURI ) {
nsCOMPtr < nsIContent > child = protoBinding - > GetBindingElement ( ) ;
rv = GetBinding ( aBoundElement , baseBindingURI , aPeekOnly ,
child - > NodePrincipal ( ) , aIsReady ,
getter_AddRefs ( baseBinding ) , aDontExtendURIs ) ;
if ( NS_FAILED ( rv ) )
return rv ; // We aren't ready yet.
}
2011-10-17 07:59:28 -07:00
* aIsReady = true ;
2011-11-03 13:39:07 -07:00
2007-03-22 10:30:00 -07:00
if ( ! aPeekOnly ) {
// Make a new binding
nsXBLBinding * newBinding = new nsXBLBinding ( protoBinding ) ;
NS_ENSURE_TRUE ( newBinding , NS_ERROR_OUT_OF_MEMORY ) ;
2011-11-03 13:39:07 -07:00
if ( baseBinding ) {
if ( ! baseProto ) {
protoBinding - > SetBasePrototype ( baseBinding - > PrototypeBinding ( ) ) ;
}
newBinding - > SetBaseBinding ( baseBinding ) ;
}
2007-03-22 10:30:00 -07:00
NS_ADDREF ( * aResult = newBinding ) ;
}
return NS_OK ;
}
2011-09-28 23:19:26 -07:00
static bool SchemeIs ( nsIURI * aURI , const char * aScheme )
2009-03-10 01:35:00 -07:00
{
nsCOMPtr < nsIURI > baseURI = NS_GetInnermostURI ( aURI ) ;
2011-10-17 07:59:28 -07:00
NS_ENSURE_TRUE ( baseURI , false ) ;
2009-03-10 01:35:00 -07:00
2011-09-28 23:19:26 -07:00
bool isScheme = false ;
2009-03-10 01:35:00 -07:00
return NS_SUCCEEDED ( baseURI - > SchemeIs ( aScheme , & isScheme ) ) & & isScheme ;
}
2011-09-28 23:19:26 -07:00
static bool
2010-08-19 16:12:46 -07:00
IsSystemOrChromeURLPrincipal ( nsIPrincipal * aPrincipal )
{
if ( nsContentUtils : : IsSystemPrincipal ( aPrincipal ) ) {
2011-10-17 07:59:28 -07:00
return true ;
2010-08-19 16:12:46 -07:00
}
nsCOMPtr < nsIURI > uri ;
aPrincipal - > GetURI ( getter_AddRefs ( uri ) ) ;
2011-10-17 07:59:28 -07:00
NS_ENSURE_TRUE ( uri , false ) ;
2010-08-19 16:12:46 -07:00
2011-09-28 23:19:26 -07:00
bool isChrome = false ;
2010-08-19 16:12:46 -07:00
return NS_SUCCEEDED ( uri - > SchemeIs ( " chrome " , & isChrome ) ) & & isChrome ;
}
2012-05-23 11:46:04 -07:00
nsresult
2007-03-22 10:30:00 -07:00
nsXBLService : : LoadBindingDocumentInfo ( nsIContent * aBoundElement ,
nsIDocument * aBoundDocument ,
nsIURI * aBindingURI ,
2007-07-18 14:56:57 -07:00
nsIPrincipal * aOriginPrincipal ,
2011-09-28 23:19:26 -07:00
bool aForceSyncLoad ,
2010-07-14 18:53:11 -07:00
nsXBLDocumentInfo * * aResult )
2007-03-22 10:30:00 -07:00
{
NS_PRECONDITION ( aBindingURI , " Must have a binding URI " ) ;
2007-07-18 14:56:57 -07:00
NS_PRECONDITION ( ! aOriginPrincipal | | aBoundDocument ,
" If we're doing a security check, we better have a document! " ) ;
2007-03-22 10:30:00 -07:00
2007-06-12 14:56:06 -07:00
nsresult rv ;
2007-07-18 14:56:57 -07:00
if ( aOriginPrincipal ) {
// Security check - Enforce same-origin policy, except to chrome.
// We have to be careful to not pass aContent as the context here.
// Otherwise, if there is a JS-implemented content policy, we will attempt
// to wrap the content node, which will try to load XBL bindings for it, if
// any. Since we're not done loading this binding yet, that will reenter
// this method and we'll end up creating a binding and then immediately
// clobbering it in our table. That makes things very confused, leading to
// misbehavior and crashes.
2007-06-12 14:56:06 -07:00
rv = nsContentUtils : :
2007-07-18 14:56:57 -07:00
CheckSecurityBeforeLoad ( aBindingURI , aOriginPrincipal ,
2007-06-12 14:56:06 -07:00
nsIScriptSecurityManager : : ALLOW_CHROME ,
2008-04-28 16:56:07 -07:00
gAllowDataURIs ,
2007-06-17 06:50:50 -07:00
nsIContentPolicy : : TYPE_XBL ,
2007-06-12 14:56:06 -07:00
aBoundDocument ) ;
2010-09-15 15:53:01 -07:00
NS_ENSURE_SUCCESS ( rv , NS_ERROR_XBL_BLOCKED ) ;
2009-03-09 23:15:06 -07:00
2010-08-19 16:12:46 -07:00
if ( ! IsSystemOrChromeURLPrincipal ( aOriginPrincipal ) ) {
// Also make sure that we're same-origin with the bound document
// except if the stylesheet has the system principal.
if ( ! ( gAllowDataURIs & & SchemeIs ( aBindingURI , " data " ) ) & &
! SchemeIs ( aBindingURI , " chrome " ) ) {
rv = aBoundDocument - > NodePrincipal ( ) - > CheckMayLoad ( aBindingURI ,
2012-08-20 11:34:33 -07:00
true , false ) ;
2010-09-15 15:53:01 -07:00
NS_ENSURE_SUCCESS ( rv , NS_ERROR_XBL_BLOCKED ) ;
2010-08-19 16:12:46 -07:00
}
2009-03-09 23:15:06 -07:00
2010-08-19 16:12:46 -07:00
// Finally check if this document is allowed to use XBL at all.
NS_ENSURE_TRUE ( aBoundDocument - > AllowXULXBL ( ) ,
2010-09-15 15:53:01 -07:00
NS_ERROR_XBL_BLOCKED ) ;
2009-03-09 23:15:06 -07:00
}
2007-06-12 14:56:06 -07:00
}
2007-03-22 10:30:00 -07:00
2012-07-30 07:20:58 -07:00
* aResult = nullptr ;
2010-07-14 18:53:11 -07:00
nsRefPtr < nsXBLDocumentInfo > info ;
2007-03-22 10:30:00 -07:00
nsCOMPtr < nsIURI > documentURI ;
2011-05-21 18:12:46 -07:00
rv = aBindingURI - > CloneIgnoringRef ( getter_AddRefs ( documentURI ) ) ;
2007-03-22 10:30:00 -07:00
NS_ENSURE_SUCCESS ( rv , rv ) ;
# ifdef MOZ_XUL
// We've got a file. Check our XBL document cache.
nsXULPrototypeCache * cache = nsXULPrototypeCache : : GetInstance ( ) ;
2011-09-28 23:19:26 -07:00
bool useXULCache = cache & & cache - > IsEnabled ( ) ;
2007-03-22 10:30:00 -07:00
if ( useXULCache ) {
// The first line of defense is the chrome cache.
// This cache crosses the entire product, so that any XBL bindings that are
// part of chrome will be reused across all XUL documents.
info = cache - > GetXBLDocumentInfo ( documentURI ) ;
}
# endif
if ( ! info ) {
// The second line of defense is the binding manager's document table.
2012-07-30 07:20:58 -07:00
nsBindingManager * bindingManager = nullptr ;
2007-03-22 10:30:00 -07:00
if ( aBoundDocument ) {
bindingManager = aBoundDocument - > BindingManager ( ) ;
info = bindingManager - > GetXBLDocumentInfo ( documentURI ) ;
2013-08-02 02:26:21 -07:00
if ( aBoundDocument - > IsStaticDocument ( ) & &
IsChromeOrResourceURI ( aBindingURI ) ) {
aForceSyncLoad = true ;
}
2007-03-22 10:30:00 -07:00
}
2012-07-30 07:20:58 -07:00
nsINodeInfo * ni = nullptr ;
2007-03-22 10:30:00 -07:00
if ( aBoundElement )
ni = aBoundElement - > NodeInfo ( ) ;
if ( ! info & & bindingManager & &
( ! ni | | ! ( ni - > Equals ( nsGkAtoms : : scrollbar , kNameSpaceID_XUL ) | |
ni - > Equals ( nsGkAtoms : : thumb , kNameSpaceID_XUL ) | |
( ( ni - > Equals ( nsGkAtoms : : input ) | |
ni - > Equals ( nsGkAtoms : : select ) ) & &
2009-08-24 13:02:07 -07:00
aBoundElement - > IsHTML ( ) ) ) ) & & ! aForceSyncLoad ) {
2007-03-22 10:30:00 -07:00
// The third line of defense is to investigate whether or not the
// document is currently being loaded asynchronously. If so, there's no
// document yet, but we need to glom on our request so that it will be
// processed whenever the doc does finish loading.
nsCOMPtr < nsIStreamListener > listener ;
if ( bindingManager )
listener = bindingManager - > GetLoadingDocListener ( documentURI ) ;
if ( listener ) {
nsXBLStreamListener * xblListener =
2007-07-08 00:08:04 -07:00
static_cast < nsXBLStreamListener * > ( listener . get ( ) ) ;
2007-03-22 10:30:00 -07:00
// Create a new load observer.
if ( ! xblListener - > HasRequest ( aBindingURI , aBoundElement ) ) {
2013-03-05 04:37:47 -08:00
nsXBLBindingRequest * req = new nsXBLBindingRequest ( aBindingURI , aBoundElement ) ;
2007-03-22 10:30:00 -07:00
xblListener - > AddRequest ( req ) ;
}
return NS_OK ;
}
}
2011-11-03 13:39:08 -07:00
# ifdef MOZ_XUL
// Next, look in the startup cache
bool useStartupCache = useXULCache & & IsChromeOrResourceURI ( documentURI ) ;
if ( ! info & & useStartupCache ) {
rv = nsXBLDocumentInfo : : ReadPrototypeBindings ( documentURI , getter_AddRefs ( info ) ) ;
if ( NS_SUCCEEDED ( rv ) ) {
cache - > PutXBLDocumentInfo ( info ) ;
if ( bindingManager ) {
// Cache it in our binding manager's document table.
bindingManager - > PutXBLDocumentInfo ( info ) ;
}
}
}
# endif
2007-03-22 10:30:00 -07:00
if ( ! info ) {
// Finally, if all lines of defense fail, we go and fetch the binding
// document.
// Always load chrome synchronously
2011-09-28 23:19:26 -07:00
bool chrome ;
2007-03-22 10:30:00 -07:00
if ( NS_SUCCEEDED ( documentURI - > SchemeIs ( " chrome " , & chrome ) ) & & chrome )
2011-10-17 07:59:28 -07:00
aForceSyncLoad = true ;
2007-03-22 10:30:00 -07:00
nsCOMPtr < nsIDocument > document ;
FetchBindingDocument ( aBoundElement , aBoundDocument , documentURI ,
aBindingURI , aForceSyncLoad , getter_AddRefs ( document ) ) ;
2011-11-03 13:39:08 -07:00
2007-03-22 10:30:00 -07:00
if ( document ) {
nsBindingManager * xblDocBindingManager = document - > BindingManager ( ) ;
info = xblDocBindingManager - > GetXBLDocumentInfo ( documentURI ) ;
if ( ! info ) {
NS_ERROR ( " An XBL file is malformed. Did you forget the XBL namespace on the bindings tag? " ) ;
return NS_ERROR_FAILURE ;
}
xblDocBindingManager - > RemoveXBLDocumentInfo ( info ) ; // Break the self-imposed cycle.
// If the doc is a chrome URI, then we put it into the XUL cache.
# ifdef MOZ_XUL
2011-11-03 13:39:08 -07:00
if ( useStartupCache ) {
2007-03-22 10:30:00 -07:00
cache - > PutXBLDocumentInfo ( info ) ;
2011-11-03 13:39:08 -07:00
// now write the bindings into the startup cache
info - > WritePrototypeBindings ( ) ;
2007-03-22 10:30:00 -07:00
}
# endif
if ( bindingManager ) {
// Also put it in our binding manager's document table.
bindingManager - > PutXBLDocumentInfo ( info ) ;
}
}
}
}
2012-03-22 13:46:03 -07:00
info . forget ( aResult ) ;
2007-03-22 10:30:00 -07:00
return NS_OK ;
}
nsresult
nsXBLService : : FetchBindingDocument ( nsIContent * aBoundElement , nsIDocument * aBoundDocument ,
nsIURI * aDocumentURI , nsIURI * aBindingURI ,
2011-09-28 23:19:26 -07:00
bool aForceSyncLoad , nsIDocument * * aResult )
2007-03-22 10:30:00 -07:00
{
nsresult rv = NS_OK ;
2012-07-30 07:20:58 -07:00
// Initialize our out pointer to nullptr
* aResult = nullptr ;
2007-03-22 10:30:00 -07:00
2009-01-13 14:53:04 -08:00
// Now we have to synchronously load the binding file.
// Create an XML content sink and a parser.
2007-03-22 10:30:00 -07:00
nsCOMPtr < nsILoadGroup > loadGroup ;
if ( aBoundDocument )
loadGroup = aBoundDocument - > GetDocumentLoadGroup ( ) ;
// We really shouldn't have to force a sync load for anything here... could
// we get away with not doing that? Not sure.
if ( IsChromeOrResourceURI ( aDocumentURI ) )
2011-10-17 07:59:28 -07:00
aForceSyncLoad = true ;
2007-03-22 10:30:00 -07:00
// Create document and contentsink and set them up.
nsCOMPtr < nsIDocument > doc ;
rv = NS_NewXMLDocument ( getter_AddRefs ( doc ) ) ;
NS_ENSURE_SUCCESS ( rv , rv ) ;
nsCOMPtr < nsIXMLContentSink > xblSink ;
2012-07-30 07:20:58 -07:00
rv = NS_NewXBLContentSink ( getter_AddRefs ( xblSink ) , doc , aDocumentURI , nullptr ) ;
2007-03-22 10:30:00 -07:00
NS_ENSURE_SUCCESS ( rv , rv ) ;
// Open channel
nsCOMPtr < nsIChannel > channel ;
2012-07-30 07:20:58 -07:00
rv = NS_NewChannel ( getter_AddRefs ( channel ) , aDocumentURI , nullptr , loadGroup ) ;
2007-03-22 10:30:00 -07:00
NS_ENSURE_SUCCESS ( rv , rv ) ;
2008-09-07 18:13:02 -07:00
nsCOMPtr < nsIInterfaceRequestor > sameOriginChecker = nsContentUtils : : GetSameOriginChecker ( ) ;
2007-06-12 14:56:06 -07:00
NS_ENSURE_TRUE ( sameOriginChecker , NS_ERROR_OUT_OF_MEMORY ) ;
channel - > SetNotificationCallbacks ( sameOriginChecker ) ;
2007-03-22 10:30:00 -07:00
if ( ! aForceSyncLoad ) {
// We can be asynchronous
2009-01-14 03:24:26 -08:00
nsXBLStreamListener * xblListener =
2012-05-23 11:46:04 -07:00
new nsXBLStreamListener ( aBoundDocument , xblSink , doc ) ;
2007-03-22 10:30:00 -07:00
NS_ENSURE_TRUE ( xblListener , NS_ERROR_OUT_OF_MEMORY ) ;
// Add ourselves to the list of loading docs.
nsBindingManager * bindingManager ;
if ( aBoundDocument )
bindingManager = aBoundDocument - > BindingManager ( ) ;
else
2012-07-30 07:20:58 -07:00
bindingManager = nullptr ;
2007-03-22 10:30:00 -07:00
if ( bindingManager )
bindingManager - > PutLoadingDocListener ( aDocumentURI , xblListener ) ;
// Add our request.
2013-03-05 04:37:47 -08:00
nsXBLBindingRequest * req = new nsXBLBindingRequest ( aBindingURI ,
aBoundElement ) ;
2007-03-22 10:30:00 -07:00
xblListener - > AddRequest ( req ) ;
// Now kick off the async read.
2012-07-30 07:20:58 -07:00
rv = channel - > AsyncOpen ( xblListener , nullptr ) ;
2008-12-12 11:46:59 -08:00
if ( NS_FAILED ( rv ) ) {
// Well, we won't be getting a load. Make sure to clean up our stuff!
if ( bindingManager ) {
bindingManager - > RemoveLoadingDocListener ( aDocumentURI ) ;
}
}
2007-03-22 10:30:00 -07:00
return NS_OK ;
}
2009-01-14 03:24:26 -08:00
nsCOMPtr < nsIStreamListener > listener ;
rv = doc - > StartDocumentLoad ( " loadAsInteractiveData " ,
channel ,
loadGroup ,
2012-07-30 07:20:58 -07:00
nullptr ,
2009-01-14 03:24:26 -08:00
getter_AddRefs ( listener ) ,
2011-10-17 07:59:28 -07:00
true ,
2009-01-14 03:24:26 -08:00
xblSink ) ;
NS_ENSURE_SUCCESS ( rv , rv ) ;
2007-03-22 10:30:00 -07:00
// Now do a blocking synchronous parse of the file.
nsCOMPtr < nsIInputStream > in ;
rv = channel - > Open ( getter_AddRefs ( in ) ) ;
NS_ENSURE_SUCCESS ( rv , rv ) ;
rv = nsSyncLoadService : : PushSyncStreamToListener ( in , listener , channel ) ;
NS_ENSURE_SUCCESS ( rv , rv ) ;
doc . swap ( * aResult ) ;
return NS_OK ;
}