/* -*- Mode: c++; c-basic-offset: 2; tab-width: 20; indent-tabs-mode: nil; -*- * vim: set sw=2 ts=4 expandtab: * 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 NSWINDOW_H_ #define NSWINDOW_H_ #include "nsBaseWidget.h" #include "gfxPoint.h" #include "nsIIdleServiceInternal.h" #include "nsTArray.h" #include "EventDispatcher.h" #include "GeneratedJNIWrappers.h" #include "mozilla/EventForwards.h" #include "mozilla/MozPromise.h" #include "mozilla/Mutex.h" #include "mozilla/StaticPtr.h" #include "mozilla/TextRange.h" #include "mozilla/UniquePtr.h" struct ANPEvent; namespace mozilla { class WidgetTouchEvent; namespace layers { class CompositorBridgeChild; class LayerManager; class APZCTreeManager; class UiCompositorControllerChild; } // namespace layers namespace widget { class GeckoEditableSupport; } // namespace widget namespace ipc { class Shmem; } // namespace ipc namespace a11y { class SessionAccessibility; } } // namespace mozilla class nsWindow final : public nsBaseWidget { private: virtual ~nsWindow(); public: using nsBaseWidget::GetLayerManager; nsWindow(); NS_INLINE_DECL_REFCOUNTING_INHERITED(nsWindow, nsBaseWidget) static void InitNatives(); void SetScreenId(uint32_t aScreenId) { mScreenId = aScreenId; } void OnGeckoViewReady(); RefPtr> OnLoadRequest( nsIURI* aUri, int32_t aWindowType, int32_t aFlags, nsIPrincipal* aTriggeringPrincipal, bool aHasUserGesture); private: uint32_t mScreenId; // An Event subclass that guards against stale events. template class WindowEvent; public: // Smart pointer for holding a pointer back to the nsWindow inside a native // object class. The nsWindow pointer is automatically cleared when the // nsWindow is destroyed, and a WindowPtr::Locked class is provided // for thread-safe access to the nsWindow pointer off of the Gecko thread. template class WindowPtr; // Smart pointer for holding a pointer to a native object class. The // pointer is automatically cleared when the object is destroyed. template class NativePtr final { friend WindowPtr; friend nsWindow; static const char sName[]; WindowPtr* mPtr; Impl* mImpl; mozilla::Mutex mImplLock; NativePtr() : mPtr(nullptr), mImpl(nullptr), mImplLock(sName) {} ~NativePtr() { MOZ_ASSERT(!mPtr); } public: class Locked; operator Impl*() const { MOZ_ASSERT(NS_IsMainThread()); return mImpl; } Impl* operator->() const { return operator Impl*(); } template void Attach(const mozilla::jni::LocalRef& aInstance, nsWindow* aWindow, Args&&... aArgs); template void Detach(const mozilla::jni::Ref& aInstance); }; template class WindowPtr final { friend NativePtr; NativePtr* mPtr; nsWindow* mWindow; mozilla::Mutex mWindowLock; public: class Locked final : private mozilla::MutexAutoLock { nsWindow* const mWindow; public: explicit Locked(WindowPtr& aPtr) : mozilla::MutexAutoLock(aPtr.mWindowLock), mWindow(aPtr.mWindow) {} operator nsWindow*() const { return mWindow; } nsWindow* operator->() const { return mWindow; } }; WindowPtr(NativePtr* aPtr, nsWindow* aWindow); ~WindowPtr() { MOZ_ASSERT(NS_IsMainThread()); if (!mPtr) { return; } mPtr->mPtr = nullptr; mPtr->mImpl = nullptr; } operator nsWindow*() const { MOZ_ASSERT(NS_IsMainThread()); return mWindow; } nsWindow* operator->() const { return operator nsWindow*(); } }; private: class AndroidView final : public nsIAndroidView { virtual ~AndroidView() {} public: const RefPtr mEventDispatcher{ new mozilla::widget::EventDispatcher()}; AndroidView() {} NS_DECL_ISUPPORTS NS_DECL_NSIANDROIDVIEW NS_FORWARD_NSIANDROIDEVENTDISPATCHER(mEventDispatcher->) mozilla::java::GeckoBundle::GlobalRef mInitData; }; RefPtr mAndroidView; class LayerViewSupport; // Object that implements native LayerView calls. // Owned by the Java LayerView instance. NativePtr mLayerViewSupport; class NPZCSupport; // Object that implements native NativePanZoomController calls. // Owned by the Java NativePanZoomController instance. NativePtr mNPZCSupport; // Object that implements native GeckoEditable calls. // Strong referenced by the Java instance. NativePtr mEditableSupport; mozilla::jni::Object::GlobalRef mEditableParent; // Object that implements native SessionAccessibility calls. // Strong referenced by the Java instance. NativePtr mSessionAccessibility; class GeckoViewSupport; // Object that implements native GeckoView calls and associated states. // nullptr for nsWindows that were not opened from GeckoView. // Because other objects get destroyed in the mGeckOViewSupport destructor, // keep it last in the list, so its destructor is called first. mozilla::UniquePtr mGeckoViewSupport; mozilla::Atomic mContentDocumentDisplayed; public: static already_AddRefed From(nsPIDOMWindowOuter* aDOMWindow); static already_AddRefed From(nsIWidget* aWidget); static nsWindow* TopWindow(); static mozilla::Modifiers GetModifiers(int32_t aMetaState); static mozilla::TimeStamp GetEventTimeStamp(int64_t aEventTime); void OnSizeChanged(const mozilla::gfx::IntSize& aSize); void InitEvent(mozilla::WidgetGUIEvent& event, LayoutDeviceIntPoint* aPoint = 0); void UpdateOverscrollVelocity(const float aX, const float aY); void UpdateOverscrollOffset(const float aX, const float aY); mozilla::widget::EventDispatcher* GetEventDispatcher() const { if (mAndroidView) { return mAndroidView->mEventDispatcher; } return nullptr; } void NotifyDisablingWebRender(); // // nsIWidget // using nsBaseWidget::Create; // for Create signature not overridden here [[nodiscard]] virtual nsresult Create(nsIWidget* aParent, nsNativeWidget aNativeParent, const LayoutDeviceIntRect& aRect, nsWidgetInitData* aInitData) override; virtual void Destroy() override; virtual nsresult ConfigureChildren( const nsTArray&) override; virtual void SetParent(nsIWidget* aNewParent) override; virtual nsIWidget* GetParent(void) override; virtual float GetDPI() override; virtual double GetDefaultScaleInternal() override; virtual void Show(bool aState) override; virtual bool IsVisible() const override; virtual void ConstrainPosition(bool aAllowSlop, int32_t* aX, int32_t* aY) override; virtual void Move(double aX, double aY) override; virtual void Resize(double aWidth, double aHeight, bool aRepaint) override; virtual void Resize(double aX, double aY, double aWidth, double aHeight, bool aRepaint) override; void SetZIndex(int32_t aZIndex) override; virtual void SetSizeMode(nsSizeMode aMode) override; virtual void Enable(bool aState) override; virtual bool IsEnabled() const override; virtual void Invalidate(const LayoutDeviceIntRect& aRect) override; virtual void SetFocus(Raise, mozilla::dom::CallerType aCallerType) override; virtual LayoutDeviceIntRect GetScreenBounds() override; virtual LayoutDeviceIntPoint WidgetToScreenOffset() override; virtual nsresult DispatchEvent(mozilla::WidgetGUIEvent* aEvent, nsEventStatus& aStatus) override; nsEventStatus DispatchEvent(mozilla::WidgetGUIEvent* aEvent); virtual already_AddRefed GetWidgetScreen() override; virtual nsresult MakeFullScreen(bool aFullScreen, nsIScreen* aTargetScreen = nullptr) override; void SetCursor(nsCursor aDefaultCursor, imgIContainer* aImageCursor, uint32_t aHotspotX, uint32_t aHotspotY) override {} void* GetNativeData(uint32_t aDataType) override; void SetNativeData(uint32_t aDataType, uintptr_t aVal) override; virtual nsresult SetTitle(const nsAString& aTitle) override { return NS_OK; } [[nodiscard]] virtual nsresult GetAttention(int32_t aCycleCount) override { return NS_ERROR_NOT_IMPLEMENTED; } TextEventDispatcherListener* GetNativeTextEventDispatcherListener() override; virtual void SetInputContext(const InputContext& aContext, const InputContextAction& aAction) override; virtual InputContext GetInputContext() override; LayerManager* GetLayerManager( PLayerTransactionChild* aShadowManager = nullptr, LayersBackend aBackendHint = mozilla::layers::LayersBackend::LAYERS_NONE, LayerManagerPersistence aPersistence = LAYER_MANAGER_CURRENT) override; virtual bool NeedsPaint() override; virtual bool WidgetPaintsBackground() override; virtual uint32_t GetMaxTouchPoints() const override; void UpdateZoomConstraints( const uint32_t& aPresShellId, const ScrollableLayerGuid::ViewID& aViewId, const mozilla::Maybe& aConstraints) override; nsresult SynthesizeNativeTouchPoint(uint32_t aPointerId, TouchPointerState aPointerState, LayoutDeviceIntPoint aPoint, double aPointerPressure, uint32_t aPointerOrientation, nsIObserver* aObserver) override; nsresult SynthesizeNativeMouseEvent(LayoutDeviceIntPoint aPoint, uint32_t aNativeMessage, uint32_t aModifierFlags, nsIObserver* aObserver) override; nsresult SynthesizeNativeMouseMove(LayoutDeviceIntPoint aPoint, nsIObserver* aObserver) override; mozilla::layers::CompositorBridgeChild* GetCompositorBridgeChild() const; void SetContentDocumentDisplayed(bool aDisplayed); bool IsContentDocumentDisplayed(); // Call this function when the users activity is the direct cause of an // event (like a keypress or mouse click). void UserActivity(); mozilla::jni::Object::Ref& GetEditableParent() { return mEditableParent; } mozilla::a11y::SessionAccessibility* GetSessionAccessibility() { return mSessionAccessibility; } void RecvToolbarAnimatorMessageFromCompositor(int32_t aMessage) override; void UpdateRootFrameMetrics(const ScreenPoint& aScrollOffset, const CSSToScreenScale& aZoom) override; void RecvScreenPixels(mozilla::ipc::Shmem&& aMem, const ScreenIntSize& aSize) override; void UpdateDynamicToolbarMaxHeight(mozilla::ScreenIntCoord aHeight) override; mozilla::ScreenIntCoord GetDynamicToolbarMaxHeight() const override { return mDynamicToolbarMaxHeight; } nsresult SetPrefersReducedMotionOverrideForTest(bool aValue) override; nsresult ResetPrefersReducedMotionOverrideForTest() override; void UpdateDynamicToolbarOffset(mozilla::ScreenIntCoord aOffset); virtual mozilla::ScreenIntMargin GetSafeAreaInsets() const override; void UpdateSafeAreaInsets(const mozilla::ScreenIntMargin& aSafeAreaInsets); protected: void BringToFront(); nsWindow* FindTopLevel(); bool IsTopLevel(); void ConfigureAPZControllerThread() override; void DispatchHitTest(const mozilla::WidgetTouchEvent& aEvent); already_AddRefed CreateRootContentController() override; bool mIsVisible; nsTArray mChildren; nsWindow* mParent; nsCOMPtr mIdleService; mozilla::ScreenIntCoord mDynamicToolbarMaxHeight; mozilla::ScreenIntMargin mSafeAreaInsets; bool mIsFullScreen; bool mIsDisablingWebRender; bool UseExternalCompositingSurface() const override { return true; } static void DumpWindows(); static void DumpWindows(const nsTArray& wins, int indent = 0); static void LogWindow(nsWindow* win, int index, int indent); private: void CreateLayerManager(); void RedrawAll(); mozilla::layers::LayersId GetRootLayerId() const; RefPtr GetUiCompositorControllerChild(); }; // Explicit template declarations to make clang be quiet. template <> const char nsWindow::NativePtr::sName[]; template <> const char nsWindow::NativePtr::sName[]; template <> const char nsWindow::NativePtr::sName[]; template <> const char nsWindow::NativePtr::sName[]; template nsWindow::WindowPtr::WindowPtr(NativePtr* aPtr, nsWindow* aWindow) : mPtr(aPtr), mWindow(aWindow), mWindowLock(NativePtr::sName) { MOZ_ASSERT(NS_IsMainThread()); if (mPtr) { mPtr->mPtr = this; } } #endif /* NSWINDOW_H_ */