diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-28 14:29:10 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-28 14:29:10 +0000 |
commit | 2aa4a82499d4becd2284cdb482213d541b8804dd (patch) | |
tree | b80bf8bf13c3766139fbacc530efd0dd9d54394c /dom/events/DOMEventTargetHelper.cpp | |
parent | Initial commit. (diff) | |
download | firefox-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.cpp | 357 |
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 |