1 /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* vim: set ts=8 sts=2 et sw=2 tw=80: */
3 /* This Source Code Form is subject to the terms of the Mozilla Public
4 * License, v. 2.0. If a copy of the MPL was not distributed with this
5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
7 #ifndef widget_windows_WinWindowOcclusionTracker_h
8 #define widget_windows_WinWindowOcclusionTracker_h
12 #include <unordered_map>
13 #include <unordered_set>
16 #include "nsIWeakReferenceUtils.h"
17 #include "mozilla/Monitor.h"
18 #include "mozilla/StaticPtr.h"
19 #include "mozilla/widget/WindowOcclusionState.h"
21 #include "nsThreadUtils.h"
24 struct IVirtualDesktopManager
;
25 class WinWindowOcclusionTrackerTest
;
26 class WinWindowOcclusionTrackerInteractiveTest
;
36 class OcclusionUpdateRunnable
;
37 class SerializedTaskDispatcher
;
38 class UpdateOcclusionStateRunnable
;
40 // This class handles window occlusion tracking by using HWND.
41 // Implementation is borrowed from chromium's NativeWindowOcclusionTrackerWin.
42 class WinWindowOcclusionTracker final
{
44 NS_INLINE_DECL_THREADSAFE_REFCOUNTING(WinWindowOcclusionTracker
)
46 /// Can only be called from the main thread.
47 static WinWindowOcclusionTracker
* Get();
49 /// Can only be called from the main thread.
52 /// Can only be called from the main thread.
53 static void ShutDown();
55 /// Can be called from any thread.
56 static MessageLoop
* OcclusionCalculatorLoop();
58 /// Can be called from any thread.
59 static bool IsInWinWindowOcclusionThread();
61 // Enables notifying to widget via NotifyOcclusionState() when the occlusion
62 // state has been computed.
63 void Enable(nsBaseWidget
* aWindow
, HWND aHwnd
);
65 // Disables notifying to widget via NotifyOcclusionState() when the occlusion
66 // state has been computed.
67 void Disable(nsBaseWidget
* aWindow
, HWND aHwnd
);
69 // Called when widget's visibility is changed
70 void OnWindowVisibilityChanged(nsBaseWidget
* aWindow
, bool aVisible
);
72 SerializedTaskDispatcher
* GetSerializedTaskDispatcher() {
73 return mSerializedTaskDispatcher
;
76 void TriggerCalculation();
78 void DumpOccludingWindows(HWND aHWnd
);
81 friend class ::WinWindowOcclusionTrackerTest
;
82 friend class ::WinWindowOcclusionTrackerInteractiveTest
;
84 explicit WinWindowOcclusionTracker(UniquePtr
<base::Thread
> aThread
);
85 virtual ~WinWindowOcclusionTracker();
87 // This class computes the occlusion state of the tracked windows.
88 // It runs on a separate thread, and notifies the main thread of
89 // the occlusion state of the tracked windows.
90 class WindowOcclusionCalculator
{
91 NS_INLINE_DECL_THREADSAFE_REFCOUNTING(WindowOcclusionCalculator
)
93 // Creates WindowOcclusionCalculator instance.
94 static void CreateInstance();
96 // Clear WindowOcclusionCalculator instance.
97 static void ClearInstance();
99 // Returns existing WindowOcclusionCalculator instance.
100 static WindowOcclusionCalculator
* GetInstance() { return sCalculator
; }
105 void EnableOcclusionTrackingForWindow(HWND hwnd
);
106 void DisableOcclusionTrackingForWindow(HWND hwnd
);
108 // If a window becomes visible, makes sure event hooks are registered.
109 void HandleVisibilityChanged(bool aVisible
);
111 void HandleTriggerCalculation();
114 WindowOcclusionCalculator();
115 ~WindowOcclusionCalculator();
117 // Registers event hooks, if not registered.
118 void MaybeRegisterEventHooks();
120 // This is the callback registered to get notified of various Windows
121 // events, like window moving/resizing.
122 static void CALLBACK
EventHookCallback(HWINEVENTHOOK aWinEventHook
,
123 DWORD aEvent
, HWND aHwnd
,
124 LONG aIdObject
, LONG aIdChild
,
128 // EnumWindows callback used to iterate over all hwnds to determine
129 // occlusion status of all tracked root windows. Also builds up
130 // |current_pids_with_visible_windows_| and registers event hooks for newly
131 // discovered processes with visible hwnds.
133 ComputeNativeWindowOcclusionStatusCallback(HWND hwnd
, LPARAM lParam
);
135 // EnumWindows callback used to update the list of process ids with
136 // visible hwnds, |pids_for_location_change_hook_|.
137 static BOOL CALLBACK
UpdateVisibleWindowProcessIdsCallback(HWND aHwnd
,
140 // Determines which processes owning visible application windows to set the
141 // EVENT_OBJECT_LOCATIONCHANGE event hook for and stores the pids in
142 // |pids_for_location_change_hook_|.
143 void UpdateVisibleWindowProcessIds();
145 // Computes the native window occlusion status for all tracked root gecko
146 // windows in |root_window_hwnds_occlusion_state_| and notifies them if
147 // their occlusion status has changed.
148 void ComputeNativeWindowOcclusionStatus();
150 // Schedules an occlusion calculation , if one isn't already scheduled.
151 void ScheduleOcclusionCalculationIfNeeded();
153 // Registers a global event hook (not per process) for the events in the
154 // range from |event_min| to |event_max|, inclusive.
155 void RegisterGlobalEventHook(DWORD aEventMin
, DWORD aEventMax
);
157 // Registers the EVENT_OBJECT_LOCATIONCHANGE event hook for the process with
158 // passed id. The process has one or more visible, opaque windows.
159 void RegisterEventHookForProcess(DWORD aPid
);
161 // Registers/Unregisters the event hooks necessary for occlusion tracking
162 // via calls to RegisterEventHook. These event hooks are disabled when all
163 // tracked windows are minimized.
164 void RegisterEventHooks();
165 void UnregisterEventHooks();
167 // EnumWindows callback for occlusion calculation. Returns true to
168 // continue enumeration, false otherwise. Currently, always returns
169 // true because this function also updates currentPidsWithVisibleWindows,
170 // and needs to see all HWNDs.
171 bool ProcessComputeNativeWindowOcclusionStatusCallback(
172 HWND aHwnd
, std::unordered_set
<DWORD
>* aCurrentPidsWithVisibleWindows
);
174 // Processes events sent to OcclusionEventHookCallback.
175 // It generally triggers scheduling of the occlusion calculation, but
176 // ignores certain events in order to not calculate occlusion more than
178 void ProcessEventHookCallback(HWINEVENTHOOK aWinEventHook
, DWORD aEvent
,
179 HWND aHwnd
, LONG aIdObject
, LONG aIdChild
);
181 // EnumWindows callback for determining which processes to set the
182 // EVENT_OBJECT_LOCATIONCHANGE event hook for. We set that event hook for
183 // processes hosting fully visible, opaque windows.
184 void ProcessUpdateVisibleWindowProcessIdsCallback(HWND aHwnd
);
186 // Returns true if the window is visible, fully opaque, and on the current
187 // virtual desktop, false otherwise.
188 bool WindowCanOccludeOtherWindowsOnCurrentVirtualDesktop(
189 HWND aHwnd
, LayoutDeviceIntRect
* aWindowRect
);
191 // Returns true if aHwnd is definitely on the current virtual desktop,
192 // false if it's definitely not on the current virtual desktop, and Nothing
193 // if we we can't tell for sure.
194 Maybe
<bool> IsWindowOnCurrentVirtualDesktop(HWND aHwnd
);
196 static StaticRefPtr
<WindowOcclusionCalculator
> sCalculator
;
198 // Map of root app window hwnds and their occlusion state. This contains
199 // both visible and hidden windows.
200 // It is accessed from WinWindowOcclusionTracker::UpdateOcclusionState()
201 // without using mutex. The access is safe by using
202 // SerializedTaskDispatcher.
203 std::unordered_map
<HWND
, OcclusionState
> mRootWindowHwndsOcclusionState
;
205 // Values returned by SetWinEventHook are stored so that hooks can be
206 // unregistered when necessary.
207 std::vector
<HWINEVENTHOOK
> mGlobalEventHooks
;
209 // Map from process id to EVENT_OBJECT_LOCATIONCHANGE event hook.
210 std::unordered_map
<DWORD
, HWINEVENTHOOK
> mProcessEventHooks
;
212 // Pids of processes for which the EVENT_OBJECT_LOCATIONCHANGE event hook is
214 std::unordered_set
<DWORD
> mPidsForLocationChangeHook
;
216 // Used as a timer to delay occlusion update.
217 RefPtr
<CancelableRunnable
> mOcclusionUpdateRunnable
;
219 // Used to determine if a window is occluded. As we iterate through the
220 // hwnds in z-order, we subtract each opaque window's rect from
221 // mUnoccludedDesktopRegion. When we get to a root window, we subtract
222 // it from mUnoccludedDesktopRegion, and if mUnoccludedDesktopRegion
223 // doesn't change, the root window was already occluded.
224 LayoutDeviceIntRegion mUnoccludedDesktopRegion
;
226 // Keeps track of how many root windows we need to compute the occlusion
227 // state of in a call to ComputeNativeWindowOcclusionStatus. Once we've
228 // determined the state of all root windows, we can stop subtracting
229 // windows from mUnoccludedDesktopRegion;.
230 int mNumRootWindowsWithUnknownOcclusionState
;
232 // This is true if the task bar thumbnails or the alt tab thumbnails are
234 bool mShowingThumbnails
= false;
236 // Used to keep track of the window that's currently moving. That window
237 // is ignored for calculation occlusion so that tab dragging won't
238 // ignore windows occluded by the dragged window.
239 HWND mMovingWindow
= 0;
241 // Only used on Win10+.
242 RefPtr
<IVirtualDesktopManager
> mVirtualDesktopManager
;
244 // Used to serialize tasks related to mRootWindowHwndsOcclusionState.
245 RefPtr
<SerializedTaskDispatcher
> mSerializedTaskDispatcher
;
247 // This is an alias to the singleton WinWindowOcclusionTracker mMonitor,
248 // and is used in ShutDown().
251 friend class OcclusionUpdateRunnable
;
254 static BOOL CALLBACK
DumpOccludingWindowsCallback(HWND aHWnd
, LPARAM aLParam
);
256 // Returns true if we are interested in |hwnd| for purposes of occlusion
257 // calculation. We are interested in |hwnd| if it is a window that is
258 // visible, opaque, bounded, and not a popup or floating window. If we are
259 // interested in |hwnd|, stores the window rectangle in |window_rect|.
260 static bool IsWindowVisibleAndFullyOpaque(HWND aHwnd
,
261 LayoutDeviceIntRect
* aWindowRect
);
265 static void CallUpdateOcclusionState(
266 std::unordered_map
<HWND
, OcclusionState
>* aMap
, bool aShowAllWindows
);
268 // Updates root windows occclusion state. If aShowAllWindows is true,
269 // all non-hidden windows will be marked visible. This is used to force
270 // rendering of thumbnails.
271 void UpdateOcclusionState(std::unordered_map
<HWND
, OcclusionState
>* aMap
,
272 bool aShowAllWindows
);
275 // This is called with session changed notifications. If the screen is locked
276 // by the current session, it marks app windows as occluded.
277 void OnSessionChange(WPARAM aStatusCode
);
279 // This is called when the display is put to sleep. If the display is sleeping
280 // it marks app windows as occluded.
281 void OnDisplayStateChanged(bool aDisplayOn
);
284 // Marks all root windows as either occluded, or if hwnd IsIconic, hidden.
285 void MarkNonIconicWindowsOccluded();
287 static StaticRefPtr
<WinWindowOcclusionTracker
> sTracker
;
289 // "WinWindowOcclusionCalc" thread.
290 UniquePtr
<base::Thread
> mThread
;
293 // Has ShutDown been called on us? We might have survived if our thread join
295 bool mHasAttemptedShutdown
= false;
297 // Map of HWND to widget. Maintained on main thread, and used to send
298 // occlusion state notifications to Windows from
299 // mRootWindowHwndsOcclusionState.
300 std::unordered_map
<HWND
, nsWeakPtr
> mHwndRootWindowMap
;
302 // This is set by UpdateOcclusionState(). It is currently only used by tests.
303 int mNumVisibleRootWindows
= 0;
305 // If the screen is locked, windows are considered occluded.
306 bool mScreenLocked
= false;
308 // If the display is off, windows are considered occluded.
309 bool mDisplayOn
= true;
311 // Used to serialize tasks related to mRootWindowHwndsOcclusionState.
312 RefPtr
<SerializedTaskDispatcher
> mSerializedTaskDispatcher
;
314 friend class OcclusionUpdateRunnable
;
315 friend class UpdateOcclusionStateRunnable
;
318 } // namespace widget
319 } // namespace mozilla
321 #endif // widget_windows_WinWindowOcclusionTracker_h