forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			167 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			167 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* vim: set ts=8 sts=2 et sw=2 tw=80: */
 | |
| /* 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/. */
 | |
| 
 | |
| #include "ExtensionEventManager.h"
 | |
| 
 | |
| #include "ExtensionAPIAddRemoveListener.h"
 | |
| 
 | |
| #include "mozilla/dom/ExtensionEventManagerBinding.h"
 | |
| #include "nsIGlobalObject.h"
 | |
| #include "ExtensionEventListener.h"
 | |
| 
 | |
| namespace mozilla {
 | |
| namespace extensions {
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_CLASS(ExtensionEventManager);
 | |
| NS_IMPL_CYCLE_COLLECTING_ADDREF(ExtensionEventManager);
 | |
| NS_IMPL_CYCLE_COLLECTING_RELEASE(ExtensionEventManager)
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(ExtensionEventManager)
 | |
|   tmp->mListeners.clear();
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mGlobal)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mExtensionBrowser)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK_PRESERVED_WRAPPER
 | |
| NS_IMPL_CYCLE_COLLECTION_UNLINK_END
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(ExtensionEventManager)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mGlobal)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mExtensionBrowser)
 | |
| NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN(ExtensionEventManager)
 | |
|   for (auto iter = tmp->mListeners.iter(); !iter.done(); iter.next()) {
 | |
|     aCallbacks.Trace(&iter.get().mutableKey(), "mListeners key", aClosure);
 | |
|   }
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRACE_PRESERVED_WRAPPER
 | |
| NS_IMPL_CYCLE_COLLECTION_TRACE_END
 | |
| 
 | |
| NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(ExtensionEventManager)
 | |
|   NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
 | |
|   NS_INTERFACE_MAP_ENTRY(nsISupports)
 | |
| NS_INTERFACE_MAP_END
 | |
| 
 | |
| ExtensionEventManager::ExtensionEventManager(
 | |
|     nsIGlobalObject* aGlobal, ExtensionBrowser* aExtensionBrowser,
 | |
|     const nsAString& aNamespace, const nsAString& aEventName,
 | |
|     const nsAString& aObjectType, const nsAString& aObjectId)
 | |
|     : mGlobal(aGlobal),
 | |
|       mExtensionBrowser(aExtensionBrowser),
 | |
|       mAPINamespace(aNamespace),
 | |
|       mEventName(aEventName),
 | |
|       mAPIObjectType(aObjectType),
 | |
|       mAPIObjectId(aObjectId) {
 | |
|   MOZ_DIAGNOSTIC_ASSERT(mGlobal);
 | |
|   MOZ_DIAGNOSTIC_ASSERT(mExtensionBrowser);
 | |
| 
 | |
|   mozilla::HoldJSObjects(this);
 | |
| }
 | |
| 
 | |
| ExtensionEventManager::~ExtensionEventManager() {
 | |
|   ReleaseListeners();
 | |
|   mozilla::DropJSObjects(this);
 | |
| };
 | |
| 
 | |
| void ExtensionEventManager::ReleaseListeners() {
 | |
|   if (mListeners.empty()) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   for (auto iter = mListeners.iter(); !iter.done(); iter.next()) {
 | |
|     iter.get().value()->Cleanup();
 | |
|   }
 | |
| 
 | |
|   mListeners.clear();
 | |
| }
 | |
| 
 | |
| JSObject* ExtensionEventManager::WrapObject(JSContext* aCx,
 | |
|                                             JS::Handle<JSObject*> aGivenProto) {
 | |
|   return dom::ExtensionEventManager_Binding::Wrap(aCx, this, aGivenProto);
 | |
| }
 | |
| 
 | |
| nsIGlobalObject* ExtensionEventManager::GetParentObject() const {
 | |
|   return mGlobal;
 | |
| }
 | |
| 
 | |
| void ExtensionEventManager::AddListener(
 | |
|     JSContext* aCx, dom::Function& aCallback,
 | |
|     const dom::Optional<JS::Handle<JSObject*>>& aOptions, ErrorResult& aRv) {
 | |
|   JS::Rooted<JSObject*> cb(aCx, aCallback.CallbackOrNull());
 | |
|   if (cb == nullptr) {
 | |
|     ThrowUnexpectedError(aCx, aRv);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RefPtr<ExtensionEventManager> self = this;
 | |
| 
 | |
|   IgnoredErrorResult rv;
 | |
|   RefPtr<ExtensionEventListener> wrappedCb = ExtensionEventListener::Create(
 | |
|       mGlobal, mExtensionBrowser, &aCallback,
 | |
|       [self = std::move(self)]() { self->ReleaseListeners(); }, rv);
 | |
| 
 | |
|   if (NS_WARN_IF(rv.Failed())) {
 | |
|     ThrowUnexpectedError(aCx, aRv);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RefPtr<ExtensionEventListener> storedWrapper = wrappedCb;
 | |
|   if (!mListeners.put(cb, std::move(storedWrapper))) {
 | |
|     ThrowUnexpectedError(aCx, aRv);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   auto request = SendAddListener(mEventName);
 | |
|   request->Run(mGlobal, aCx, {}, wrappedCb, aRv);
 | |
| 
 | |
|   if (!aRv.Failed() && mAPIObjectType.IsEmpty()) {
 | |
|     mExtensionBrowser->TrackWakeupEventListener(aCx, mAPINamespace, mEventName);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void ExtensionEventManager::RemoveListener(dom::Function& aCallback,
 | |
|                                            ErrorResult& aRv) {
 | |
|   dom::AutoJSAPI jsapi;
 | |
|   if (NS_WARN_IF(!jsapi.Init(mGlobal))) {
 | |
|     aRv.Throw(NS_ERROR_UNEXPECTED);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   JSContext* cx = jsapi.cx();
 | |
|   JS::Rooted<JSObject*> cb(cx, aCallback.CallbackOrNull());
 | |
|   const auto& ptr = mListeners.lookup(cb);
 | |
| 
 | |
|   // Return earlier if the listener wasn't found
 | |
|   if (!ptr) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RefPtr<ExtensionEventListener> wrappedCb = ptr->value();
 | |
|   auto request = SendRemoveListener(mEventName);
 | |
|   request->Run(mGlobal, cx, {}, wrappedCb, aRv);
 | |
| 
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (mAPIObjectType.IsEmpty()) {
 | |
|     mExtensionBrowser->UntrackWakeupEventListener(cx, mAPINamespace,
 | |
|                                                   mEventName);
 | |
|   }
 | |
| 
 | |
|   mListeners.remove(cb);
 | |
| 
 | |
|   wrappedCb->Cleanup();
 | |
| }
 | |
| 
 | |
| bool ExtensionEventManager::HasListener(dom::Function& aCallback,
 | |
|                                         ErrorResult& aRv) const {
 | |
|   return mListeners.has(aCallback.CallbackOrNull());
 | |
| }
 | |
| 
 | |
| bool ExtensionEventManager::HasListeners(ErrorResult& aRv) const {
 | |
|   return !mListeners.empty();
 | |
| }
 | |
| 
 | |
| }  // namespace extensions
 | |
| }  // namespace mozilla
 | 
