/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* 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/. */ #ifndef mozilla_dom_workers_serviceworkerevents_h__ #define mozilla_dom_workers_serviceworkerevents_h__ #include "mozilla/dom/Event.h" #include "mozilla/dom/ExtendableEventBinding.h" #include "mozilla/dom/ExtendableMessageEventBinding.h" #include "mozilla/dom/FetchEventBinding.h" #include "mozilla/dom/File.h" #include "mozilla/dom/Promise.h" #include "mozilla/dom/Response.h" #include "mozilla/dom/workers/bindings/ServiceWorker.h" #include "mozilla/dom/workers/Workers.h" #include "nsProxyRelease.h" #include "nsContentUtils.h" class nsIInterceptedChannel; namespace mozilla { namespace dom { class Blob; class MessagePort; class Request; class ResponseOrPromise; struct PushEventInit; } // namespace dom } // namespace mozilla BEGIN_WORKERS_NAMESPACE class ServiceWorkerRegistrationInfo; class CancelChannelRunnable final : public Runnable { nsMainThreadPtrHandle mChannel; nsMainThreadPtrHandle mRegistration; const nsresult mStatus; public: CancelChannelRunnable(nsMainThreadPtrHandle& aChannel, nsMainThreadPtrHandle& aRegistration, nsresult aStatus); NS_IMETHOD Run() override; }; class ExtendableEvent : public Event { protected: nsTArray> mPromises; explicit ExtendableEvent(mozilla::dom::EventTarget* aOwner); ~ExtendableEvent() {} public: NS_DECL_ISUPPORTS_INHERITED NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(ExtendableEvent, Event) NS_FORWARD_TO_EVENT virtual JSObject* WrapObjectInternal(JSContext* aCx, JS::Handle aGivenProto) override { return mozilla::dom::ExtendableEventBinding::Wrap(aCx, this, aGivenProto); } static already_AddRefed Constructor(mozilla::dom::EventTarget* aOwner, const nsAString& aType, const EventInit& aOptions) { RefPtr e = new ExtendableEvent(aOwner); bool trusted = e->Init(aOwner); e->InitEvent(aType, aOptions.mBubbles, aOptions.mCancelable); e->SetTrusted(trusted); e->SetComposed(aOptions.mComposed); return e.forget(); } static already_AddRefed Constructor(const GlobalObject& aGlobal, const nsAString& aType, const EventInit& aOptions, ErrorResult& aRv) { nsCOMPtr target = do_QueryInterface(aGlobal.GetAsSupports()); return Constructor(target, aType, aOptions); } void WaitUntil(JSContext* aCx, Promise& aPromise, ErrorResult& aRv); already_AddRefed GetPromise(); virtual ExtendableEvent* AsExtendableEvent() override { return this; } }; class FetchEvent final : public ExtendableEvent { nsMainThreadPtrHandle mChannel; nsMainThreadPtrHandle mRegistration; RefPtr mRequest; nsCString mScriptSpec; nsCString mPreventDefaultScriptSpec; nsString mClientId; uint32_t mPreventDefaultLineNumber; uint32_t mPreventDefaultColumnNumber; bool mIsReload; bool mWaitToRespond; protected: explicit FetchEvent(EventTarget* aOwner); ~FetchEvent(); public: NS_DECL_ISUPPORTS_INHERITED NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(FetchEvent, ExtendableEvent) // Note, we cannot use NS_FORWARD_TO_EVENT because we want a different // PreventDefault(JSContext*) override. NS_FORWARD_NSIDOMEVENT(Event::) virtual JSObject* WrapObjectInternal(JSContext* aCx, JS::Handle aGivenProto) override { return FetchEventBinding::Wrap(aCx, this, aGivenProto); } void PostInit(nsMainThreadPtrHandle& aChannel, nsMainThreadPtrHandle& aRegistration, const nsACString& aScriptSpec); static already_AddRefed Constructor(const GlobalObject& aGlobal, const nsAString& aType, const FetchEventInit& aOptions, ErrorResult& aRv); bool WaitToRespond() const { return mWaitToRespond; } Request* Request_() const { MOZ_ASSERT(mRequest); return mRequest; } void GetClientId(nsAString& aClientId) const { aClientId = mClientId; } bool IsReload() const { return mIsReload; } void RespondWith(JSContext* aCx, Promise& aArg, ErrorResult& aRv); already_AddRefed ForwardTo(const nsAString& aUrl); already_AddRefed Default(); void PreventDefault(JSContext* aCx) override; void ReportCanceled(); }; class PushMessageData final : public nsISupports, public nsWrapperCache { public: NS_DECL_CYCLE_COLLECTING_ISUPPORTS NS_DECL_CYCLE_COLLECTION_SCRIPT_HOLDER_CLASS(PushMessageData) virtual JSObject* WrapObject(JSContext* aCx, JS::Handle aGivenProto) override; nsISupports* GetParentObject() const { return mOwner; } void Json(JSContext* cx, JS::MutableHandle aRetval, ErrorResult& aRv); void Text(nsAString& aData); void ArrayBuffer(JSContext* cx, JS::MutableHandle aRetval, ErrorResult& aRv); already_AddRefed Blob(ErrorResult& aRv); PushMessageData(nsISupports* aOwner, nsTArray&& aBytes); private: nsCOMPtr mOwner; nsTArray mBytes; nsString mDecodedText; ~PushMessageData(); nsresult EnsureDecodedText(); uint8_t* GetContentsCopy(); }; class PushEvent final : public ExtendableEvent { RefPtr mData; protected: explicit PushEvent(mozilla::dom::EventTarget* aOwner); ~PushEvent() {} public: NS_DECL_ISUPPORTS_INHERITED NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(PushEvent, ExtendableEvent) NS_FORWARD_TO_EVENT virtual JSObject* WrapObjectInternal(JSContext* aCx, JS::Handle aGivenProto) override; static already_AddRefed Constructor(mozilla::dom::EventTarget* aOwner, const nsAString& aType, const PushEventInit& aOptions, ErrorResult& aRv); static already_AddRefed Constructor(const GlobalObject& aGlobal, const nsAString& aType, const PushEventInit& aOptions, ErrorResult& aRv) { nsCOMPtr owner = do_QueryInterface(aGlobal.GetAsSupports()); return Constructor(owner, aType, aOptions, aRv); } PushMessageData* GetData() const { return mData; } }; class ExtendableMessageEvent final : public ExtendableEvent { JS::Heap mData; nsString mOrigin; nsString mLastEventId; RefPtr mClient; RefPtr mServiceWorker; RefPtr mMessagePort; nsTArray> mPorts; protected: explicit ExtendableMessageEvent(EventTarget* aOwner); ~ExtendableMessageEvent(); public: NS_DECL_ISUPPORTS_INHERITED NS_DECL_CYCLE_COLLECTION_SCRIPT_HOLDER_CLASS_INHERITED(ExtendableMessageEvent, ExtendableEvent) NS_FORWARD_TO_EVENT virtual JSObject* WrapObjectInternal( JSContext* aCx, JS::Handle aGivenProto) override { return mozilla::dom::ExtendableMessageEventBinding::Wrap(aCx, this, aGivenProto); } static already_AddRefed Constructor(mozilla::dom::EventTarget* aOwner, const nsAString& aType, const ExtendableMessageEventInit& aOptions, ErrorResult& aRv); static already_AddRefed Constructor(const GlobalObject& aGlobal, const nsAString& aType, const ExtendableMessageEventInit& aOptions, ErrorResult& aRv); void GetData(JSContext* aCx, JS::MutableHandle aData, ErrorResult& aRv); void GetSource(Nullable& aValue) const; NS_IMETHOD GetOrigin(nsAString& aOrigin) { aOrigin = mOrigin; return NS_OK; } NS_IMETHOD GetLastEventId(nsAString& aLastEventId) { aLastEventId = mLastEventId; return NS_OK; } void GetPorts(nsTArray>& aPorts); }; END_WORKERS_NAMESPACE #endif /* mozilla_dom_workers_serviceworkerevents_h__ */