summaryrefslogtreecommitdiffstats
path: root/dom/events/DOMEventTargetHelper.cpp
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
commit2aa4a82499d4becd2284cdb482213d541b8804dd (patch)
treeb80bf8bf13c3766139fbacc530efd0dd9d54394c /dom/events/DOMEventTargetHelper.cpp
parentInitial commit. (diff)
downloadfirefox-upstream.tar.xz
firefox-upstream.zip
Adding upstream version 86.0.1.upstream/86.0.1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'dom/events/DOMEventTargetHelper.cpp')
-rw-r--r--dom/events/DOMEventTargetHelper.cpp357
1 files changed, 357 insertions, 0 deletions
diff --git a/dom/events/DOMEventTargetHelper.cpp b/dom/events/DOMEventTargetHelper.cpp
new file mode 100644
index 0000000000..1e32e826a5
--- /dev/null
+++ b/dom/events/DOMEventTargetHelper.cpp
@@ -0,0 +1,357 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* 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 "nsContentUtils.h"
+#include "mozilla/dom/Document.h"
+#include "mozilla/Sprintf.h"
+#include "mozilla/dom/Event.h"
+#include "mozilla/DOMEventTargetHelper.h"
+#include "mozilla/EventDispatcher.h"
+#include "mozilla/EventListenerManager.h"
+#include "mozilla/EventListenerManager.h"
+#include "mozilla/Likely.h"
+#include "MainThreadUtils.h"
+
+namespace mozilla {
+
+using namespace dom;
+
+NS_IMPL_CYCLE_COLLECTION_CLASS(DOMEventTargetHelper)
+
+NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN(DOMEventTargetHelper)
+ NS_IMPL_CYCLE_COLLECTION_TRACE_PRESERVED_WRAPPER
+NS_IMPL_CYCLE_COLLECTION_TRACE_END
+
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INTERNAL(DOMEventTargetHelper)
+ if (MOZ_UNLIKELY(cb.WantDebugInfo())) {
+ char name[512];
+ nsAutoString uri;
+ if (tmp->mOwnerWindow && tmp->mOwnerWindow->GetExtantDoc()) {
+ Unused << tmp->mOwnerWindow->GetExtantDoc()->GetDocumentURI(uri);
+ }
+
+ nsXPCOMCycleCollectionParticipant* participant = nullptr;
+ CallQueryInterface(tmp, &participant);
+
+ SprintfLiteral(name, "%s %s", participant->ClassName(),
+ NS_ConvertUTF16toUTF8(uri).get());
+ cb.DescribeRefCountedNode(tmp->mRefCnt.get(), name);
+ } else {
+ NS_IMPL_CYCLE_COLLECTION_DESCRIBE(DOMEventTargetHelper, tmp->mRefCnt.get())
+ }
+
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mListenerManager)
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
+
+NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(DOMEventTargetHelper)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK_PRESERVED_WRAPPER
+ if (tmp->mListenerManager) {
+ tmp->mListenerManager->Disconnect();
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mListenerManager)
+ }
+ tmp->MaybeDontKeepAlive();
+NS_IMPL_CYCLE_COLLECTION_UNLINK_END
+
+NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_BEGIN(DOMEventTargetHelper)
+ bool hasLiveWrapper = tmp->HasKnownLiveWrapper();
+ if (hasLiveWrapper || tmp->IsCertainlyAliveForCC()) {
+ if (tmp->mListenerManager) {
+ tmp->mListenerManager->MarkForCC();
+ }
+ if (!hasLiveWrapper && tmp->PreservingWrapper()) {
+ tmp->MarkWrapperLive();
+ }
+ return true;
+ }
+NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_END
+
+NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_IN_CC_BEGIN(DOMEventTargetHelper)
+ return tmp->HasKnownLiveWrapperAndDoesNotNeedTracing(tmp);
+NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_IN_CC_END
+
+NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_THIS_BEGIN(DOMEventTargetHelper)
+ return tmp->HasKnownLiveWrapper();
+NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_THIS_END
+
+NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(DOMEventTargetHelper)
+ NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
+ NS_INTERFACE_MAP_ENTRY(nsISupports)
+ NS_INTERFACE_MAP_ENTRY(dom::EventTarget)
+ NS_INTERFACE_MAP_ENTRY(DOMEventTargetHelper)
+NS_INTERFACE_MAP_END
+
+NS_IMPL_CYCLE_COLLECTING_ADDREF(DOMEventTargetHelper)
+NS_IMPL_CYCLE_COLLECTING_RELEASE_WITH_LAST_RELEASE(DOMEventTargetHelper,
+ LastRelease())
+
+DOMEventTargetHelper::DOMEventTargetHelper()
+ : mParentObject(nullptr),
+ mOwnerWindow(nullptr),
+ mHasOrHasHadOwnerWindow(false),
+ mIsKeptAlive(false) {}
+
+DOMEventTargetHelper::DOMEventTargetHelper(nsPIDOMWindowInner* aWindow)
+ : mParentObject(nullptr),
+ mOwnerWindow(nullptr),
+ mHasOrHasHadOwnerWindow(false),
+ mIsKeptAlive(false) {
+ nsIGlobalObject* global = aWindow ? aWindow->AsGlobal() : nullptr;
+ BindToOwner(global);
+}
+
+DOMEventTargetHelper::DOMEventTargetHelper(nsIGlobalObject* aGlobalObject)
+ : mParentObject(nullptr),
+ mOwnerWindow(nullptr),
+ mHasOrHasHadOwnerWindow(false),
+ mIsKeptAlive(false) {
+ BindToOwner(aGlobalObject);
+}
+
+DOMEventTargetHelper::DOMEventTargetHelper(DOMEventTargetHelper* aOther)
+ : mParentObject(nullptr),
+ mOwnerWindow(nullptr),
+ mHasOrHasHadOwnerWindow(false),
+ mIsKeptAlive(false) {
+ if (!aOther) {
+ BindToOwner(static_cast<nsIGlobalObject*>(nullptr));
+ return;
+ }
+ BindToOwner(aOther->GetParentObject());
+ mHasOrHasHadOwnerWindow = aOther->HasOrHasHadOwner();
+}
+
+DOMEventTargetHelper::~DOMEventTargetHelper() {
+ if (mParentObject) {
+ mParentObject->RemoveEventTargetObject(this);
+ }
+ if (mListenerManager) {
+ mListenerManager->Disconnect();
+ }
+ ReleaseWrapper(this);
+}
+
+void DOMEventTargetHelper::DisconnectFromOwner() {
+ if (mParentObject) {
+ mParentObject->RemoveEventTargetObject(this);
+ }
+ mOwnerWindow = nullptr;
+ mParentObject = nullptr;
+ // Event listeners can't be handled anymore, so we can release them here.
+ if (mListenerManager) {
+ mListenerManager->Disconnect();
+ mListenerManager = nullptr;
+ }
+
+ MaybeDontKeepAlive();
+}
+
+nsPIDOMWindowInner* DOMEventTargetHelper::GetWindowIfCurrent() const {
+ if (NS_FAILED(CheckCurrentGlobalCorrectness())) {
+ return nullptr;
+ }
+
+ return GetOwner();
+}
+
+Document* DOMEventTargetHelper::GetDocumentIfCurrent() const {
+ nsPIDOMWindowInner* win = GetWindowIfCurrent();
+ if (!win) {
+ return nullptr;
+ }
+
+ return win->GetDoc();
+}
+
+bool DOMEventTargetHelper::ComputeDefaultWantsUntrusted(ErrorResult& aRv) {
+ bool wantsUntrusted;
+ nsresult rv = WantsUntrusted(&wantsUntrusted);
+ if (NS_FAILED(rv)) {
+ aRv.Throw(rv);
+ return false;
+ }
+ return wantsUntrusted;
+}
+
+bool DOMEventTargetHelper::DispatchEvent(Event& aEvent, CallerType aCallerType,
+ ErrorResult& aRv) {
+ nsEventStatus status = nsEventStatus_eIgnore;
+ nsresult rv = EventDispatcher::DispatchDOMEvent(this, nullptr, &aEvent,
+ nullptr, &status);
+ bool retval = !aEvent.DefaultPrevented(aCallerType);
+ if (NS_FAILED(rv)) {
+ aRv.Throw(rv);
+ }
+ return retval;
+}
+
+nsresult DOMEventTargetHelper::DispatchTrustedEvent(
+ const nsAString& aEventName) {
+ RefPtr<Event> event = NS_NewDOMEvent(this, nullptr, nullptr);
+ event->InitEvent(aEventName, false, false);
+
+ return DispatchTrustedEvent(event);
+}
+
+nsresult DOMEventTargetHelper::DispatchTrustedEvent(Event* event) {
+ event->SetTrusted(true);
+
+ ErrorResult rv;
+ DispatchEvent(*event, rv);
+ return rv.StealNSResult();
+}
+
+void DOMEventTargetHelper::GetEventTargetParent(
+ EventChainPreVisitor& aVisitor) {
+ aVisitor.mCanHandle = true;
+ aVisitor.SetParentTarget(nullptr, false);
+}
+
+nsresult DOMEventTargetHelper::PostHandleEvent(
+ EventChainPostVisitor& aVisitor) {
+ return NS_OK;
+}
+
+EventListenerManager* DOMEventTargetHelper::GetOrCreateListenerManager() {
+ if (!mListenerManager) {
+ mListenerManager = new EventListenerManager(this);
+ }
+
+ return mListenerManager;
+}
+
+EventListenerManager* DOMEventTargetHelper::GetExistingListenerManager() const {
+ return mListenerManager;
+}
+
+nsresult DOMEventTargetHelper::WantsUntrusted(bool* aRetVal) {
+ nsresult rv = CheckCurrentGlobalCorrectness();
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ nsCOMPtr<Document> doc = GetDocumentIfCurrent();
+ // We can let listeners on workers to always handle all the events.
+ *aRetVal = (doc && !nsContentUtils::IsChromeDoc(doc)) || !NS_IsMainThread();
+ return rv;
+}
+
+void DOMEventTargetHelper::EventListenerAdded(nsAtom* aType) {
+ MaybeUpdateKeepAlive();
+}
+
+void DOMEventTargetHelper::EventListenerRemoved(nsAtom* aType) {
+ MaybeUpdateKeepAlive();
+}
+
+void DOMEventTargetHelper::KeepAliveIfHasListenersFor(const nsAString& aType) {
+ mKeepingAliveTypes.mStrings.AppendElement(aType);
+ MaybeUpdateKeepAlive();
+}
+
+void DOMEventTargetHelper::KeepAliveIfHasListenersFor(nsAtom* aType) {
+ mKeepingAliveTypes.mAtoms.AppendElement(aType);
+ MaybeUpdateKeepAlive();
+}
+
+void DOMEventTargetHelper::IgnoreKeepAliveIfHasListenersFor(
+ const nsAString& aType) {
+ mKeepingAliveTypes.mStrings.RemoveElement(aType);
+ MaybeUpdateKeepAlive();
+}
+
+void DOMEventTargetHelper::IgnoreKeepAliveIfHasListenersFor(nsAtom* aType) {
+ mKeepingAliveTypes.mAtoms.RemoveElement(aType);
+ MaybeUpdateKeepAlive();
+}
+
+void DOMEventTargetHelper::MaybeUpdateKeepAlive() {
+ bool shouldBeKeptAlive = false;
+
+ if (NS_SUCCEEDED(CheckCurrentGlobalCorrectness())) {
+ if (!mKeepingAliveTypes.mAtoms.IsEmpty()) {
+ for (uint32_t i = 0; i < mKeepingAliveTypes.mAtoms.Length(); ++i) {
+ if (HasListenersFor(mKeepingAliveTypes.mAtoms[i])) {
+ shouldBeKeptAlive = true;
+ break;
+ }
+ }
+ }
+
+ if (!shouldBeKeptAlive && !mKeepingAliveTypes.mStrings.IsEmpty()) {
+ for (uint32_t i = 0; i < mKeepingAliveTypes.mStrings.Length(); ++i) {
+ if (HasListenersFor(mKeepingAliveTypes.mStrings[i])) {
+ shouldBeKeptAlive = true;
+ break;
+ }
+ }
+ }
+ }
+
+ if (shouldBeKeptAlive == mIsKeptAlive) {
+ return;
+ }
+
+ mIsKeptAlive = shouldBeKeptAlive;
+ if (mIsKeptAlive) {
+ AddRef();
+ } else {
+ Release();
+ }
+}
+
+void DOMEventTargetHelper::MaybeDontKeepAlive() {
+ if (mIsKeptAlive) {
+ mIsKeptAlive = false;
+ Release();
+ }
+}
+
+void DOMEventTargetHelper::BindToOwner(nsIGlobalObject* aOwner) {
+ MOZ_ASSERT(!mParentObject);
+
+ if (aOwner) {
+ mParentObject = aOwner;
+ aOwner->AddEventTargetObject(this);
+ // Let's cache the result of this QI for fast access and off main thread
+ // usage
+ mOwnerWindow =
+ nsCOMPtr<nsPIDOMWindowInner>(do_QueryInterface(aOwner)).get();
+ if (mOwnerWindow) {
+ mHasOrHasHadOwnerWindow = true;
+ }
+ }
+}
+
+nsresult DOMEventTargetHelper::CheckCurrentGlobalCorrectness() const {
+ NS_ENSURE_STATE(!mHasOrHasHadOwnerWindow || mOwnerWindow);
+
+ // Main-thread.
+ if (mOwnerWindow && !mOwnerWindow->IsCurrentInnerWindow()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (NS_IsMainThread()) {
+ return NS_OK;
+ }
+
+ if (!mParentObject) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (mParentObject->IsDying()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ return NS_OK;
+}
+
+bool DOMEventTargetHelper::HasListenersFor(const nsAString& aType) const {
+ return mListenerManager && mListenerManager->HasListenersFor(aType);
+}
+
+bool DOMEventTargetHelper::HasListenersFor(nsAtom* aTypeWithOn) const {
+ return mListenerManager && mListenerManager->HasListenersFor(aTypeWithOn);
+}
+
+} // namespace mozilla