1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "ash/wm/video_detector.h"
8 #include "ash/test/ash_test_base.h"
9 #include "ash/wm/window_state.h"
10 #include "ash/wm/wm_event.h"
11 #include "base/compiler_specific.h"
12 #include "base/memory/scoped_ptr.h"
13 #include "base/time/time.h"
14 #include "third_party/skia/include/core/SkColor.h"
15 #include "ui/aura/client/aura_constants.h"
16 #include "ui/aura/test/test_windows.h"
17 #include "ui/aura/window.h"
18 #include "ui/aura/window_event_dispatcher.h"
19 #include "ui/gfx/geometry/rect.h"
20 #include "ui/wm/public/window_types.h"
25 // Implementation that just counts the number of times we've been told that a
27 class TestVideoDetectorObserver
: public VideoDetectorObserver
{
29 TestVideoDetectorObserver() : num_invocations_(0),
31 num_not_fullscreens_(0) {}
33 int num_invocations() const { return num_invocations_
; }
34 int num_fullscreens() const { return num_fullscreens_
; }
35 int num_not_fullscreens() const { return num_not_fullscreens_
; }
39 num_not_fullscreens_
= 0;
42 // VideoDetectorObserver implementation.
43 void OnVideoDetected(bool is_fullscreen
) override
{
48 num_not_fullscreens_
++;
52 // Number of times that OnVideoDetected() has been called.
54 // Number of times that OnVideoDetected() has been called with is_fullscreen
57 // Number of times that OnVideoDetected() has been called with is_fullscreen
59 int num_not_fullscreens_
;
61 DISALLOW_COPY_AND_ASSIGN(TestVideoDetectorObserver
);
64 class VideoDetectorTest
: public AshTestBase
{
66 VideoDetectorTest() {}
67 ~VideoDetectorTest() override
{}
69 void SetUp() override
{
71 observer_
.reset(new TestVideoDetectorObserver
);
72 detector_
= Shell::GetInstance()->video_detector();
73 detector_
->AddObserver(observer_
.get());
75 now_
= base::TimeTicks::Now();
76 detector_
->set_now_for_test(now_
);
79 void TearDown() override
{
80 detector_
->RemoveObserver(observer_
.get());
81 AshTestBase::TearDown();
85 // Move |detector_|'s idea of the current time forward by |delta|.
86 void AdvanceTime(base::TimeDelta delta
) {
88 detector_
->set_now_for_test(now_
);
91 VideoDetector
* detector_
; // not owned
93 scoped_ptr
<TestVideoDetectorObserver
> observer_
;
98 DISALLOW_COPY_AND_ASSIGN(VideoDetectorTest
);
101 TEST_F(VideoDetectorTest
, Basic
) {
102 gfx::Rect
window_bounds(gfx::Point(), gfx::Size(1024, 768));
103 scoped_ptr
<aura::Window
> window(
104 CreateTestWindowInShell(SK_ColorRED
, 12345, window_bounds
));
106 // Send enough updates, but make them be too small to trigger detection.
107 gfx::Rect
update_region(
109 gfx::Size(VideoDetector::kMinUpdateWidth
- 1,
110 VideoDetector::kMinUpdateHeight
));
111 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
112 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
113 EXPECT_EQ(0, observer_
->num_invocations());
115 // Send not-quite-enough adaquately-sized updates.
116 observer_
->reset_stats();
117 AdvanceTime(base::TimeDelta::FromSeconds(2));
118 update_region
.set_size(
119 gfx::Size(VideoDetector::kMinUpdateWidth
,
120 VideoDetector::kMinUpdateHeight
));
121 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
- 1; ++i
)
122 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
123 EXPECT_EQ(0, observer_
->num_invocations());
125 // We should get notified after the next update, but not in response to
126 // additional updates.
127 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
128 EXPECT_EQ(1, observer_
->num_invocations());
129 EXPECT_EQ(0, observer_
->num_fullscreens());
130 EXPECT_EQ(1, observer_
->num_not_fullscreens());
131 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
132 EXPECT_EQ(1, observer_
->num_invocations());
133 EXPECT_EQ(0, observer_
->num_fullscreens());
134 EXPECT_EQ(1, observer_
->num_not_fullscreens());
136 // Spread out the frames over a longer period of time, but send enough
137 // over a one-second window that the observer should be notified.
138 observer_
->reset_stats();
139 AdvanceTime(base::TimeDelta::FromSeconds(2));
140 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
141 EXPECT_EQ(0, observer_
->num_invocations());
143 AdvanceTime(base::TimeDelta::FromMilliseconds(500));
144 const int kNumFrames
= VideoDetector::kMinFramesPerSecond
+ 1;
145 base::TimeDelta kInterval
=
146 base::TimeDelta::FromMilliseconds(1000 / kNumFrames
);
147 for (int i
= 0; i
< kNumFrames
; ++i
) {
148 AdvanceTime(kInterval
);
149 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
151 EXPECT_EQ(1, observer_
->num_invocations());
153 // Keep going and check that the observer is notified again.
154 for (int i
= 0; i
< kNumFrames
; ++i
) {
155 AdvanceTime(kInterval
);
156 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
158 EXPECT_EQ(2, observer_
->num_invocations());
160 // Send updates at a slower rate and check that the observer isn't notified.
161 base::TimeDelta kSlowInterval
= base::TimeDelta::FromMilliseconds(
162 1000 / (VideoDetector::kMinFramesPerSecond
- 2));
163 for (int i
= 0; i
< kNumFrames
; ++i
) {
164 AdvanceTime(kSlowInterval
);
165 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
167 EXPECT_EQ(2, observer_
->num_invocations());
170 TEST_F(VideoDetectorTest
, Shutdown
) {
171 gfx::Rect
window_bounds(gfx::Point(), gfx::Size(1024, 768));
172 scoped_ptr
<aura::Window
> window(
173 CreateTestWindowInShell(SK_ColorRED
, 12345, window_bounds
));
174 gfx::Rect
update_region(
176 gfx::Size(VideoDetector::kMinUpdateWidth
,
177 VideoDetector::kMinUpdateHeight
));
179 // It should not detect video during the shutdown.
180 Shell::GetInstance()->OnAppTerminating();
181 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
182 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
183 EXPECT_EQ(0, observer_
->num_invocations());
186 TEST_F(VideoDetectorTest
, WindowNotVisible
) {
187 gfx::Rect
window_bounds(gfx::Point(), gfx::Size(1024, 768));
188 scoped_ptr
<aura::Window
> window(
189 CreateTestWindowInShell(SK_ColorRED
, 12345, window_bounds
));
191 // Reparent the window to the root to make sure that visibility changes aren't
193 Shell::GetPrimaryRootWindow()->AddChild(window
.get());
195 // We shouldn't report video that's played in a hidden window.
197 gfx::Rect
update_region(
199 gfx::Size(VideoDetector::kMinUpdateWidth
,
200 VideoDetector::kMinUpdateHeight
));
201 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
202 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
203 EXPECT_EQ(0, observer_
->num_invocations());
205 // Make the window visible and send more updates.
206 observer_
->reset_stats();
207 AdvanceTime(base::TimeDelta::FromSeconds(2));
209 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
210 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
211 EXPECT_EQ(1, observer_
->num_invocations());
212 EXPECT_EQ(0, observer_
->num_fullscreens());
213 EXPECT_EQ(1, observer_
->num_not_fullscreens());
215 // We also shouldn't report video in a window that's fully offscreen.
216 observer_
->reset_stats();
217 AdvanceTime(base::TimeDelta::FromSeconds(2));
218 gfx::Rect
offscreen_bounds(
219 gfx::Point(Shell::GetPrimaryRootWindow()->bounds().width(), 0),
220 window_bounds
.size());
221 window
->SetBounds(offscreen_bounds
);
222 ASSERT_EQ(offscreen_bounds
, window
->bounds());
223 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
224 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
225 EXPECT_EQ(0, observer_
->num_invocations());
228 TEST_F(VideoDetectorTest
, MultipleWindows
) {
229 // Create two windows.
230 gfx::Rect
window_bounds(gfx::Point(), gfx::Size(1024, 768));
231 scoped_ptr
<aura::Window
> window1(
232 CreateTestWindowInShell(SK_ColorRED
, 12345, window_bounds
));
233 scoped_ptr
<aura::Window
> window2(
234 CreateTestWindowInShell(SK_ColorBLUE
, 23456, window_bounds
));
236 // Even if there's video playing in both, the observer should only receive a
237 // single notification.
238 gfx::Rect
update_region(
240 gfx::Size(VideoDetector::kMinUpdateWidth
,
241 VideoDetector::kMinUpdateHeight
));
242 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
243 detector_
->OnDelegatedFrameDamage(window1
.get(), update_region
);
244 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
245 detector_
->OnDelegatedFrameDamage(window2
.get(), update_region
);
246 EXPECT_EQ(1, observer_
->num_invocations());
247 EXPECT_EQ(0, observer_
->num_fullscreens());
248 EXPECT_EQ(1, observer_
->num_not_fullscreens());
251 // Test that the observer receives repeated notifications.
252 TEST_F(VideoDetectorTest
, RepeatedNotifications
) {
253 gfx::Rect
window_bounds(gfx::Point(), gfx::Size(1024, 768));
254 scoped_ptr
<aura::Window
> window(
255 CreateTestWindowInShell(SK_ColorRED
, 12345, window_bounds
));
257 gfx::Rect
update_region(
259 gfx::Size(VideoDetector::kMinUpdateWidth
,
260 VideoDetector::kMinUpdateHeight
));
261 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
262 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
263 EXPECT_EQ(1, observer_
->num_invocations());
264 EXPECT_EQ(0, observer_
->num_fullscreens());
265 EXPECT_EQ(1, observer_
->num_not_fullscreens());
266 // Let enough time pass that a second notification should be sent.
267 observer_
->reset_stats();
268 AdvanceTime(base::TimeDelta::FromSeconds(
269 static_cast<int64
>(VideoDetector::kNotifyIntervalSec
+ 1)));
270 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
271 detector_
->OnDelegatedFrameDamage(window
.get(), update_region
);
272 EXPECT_EQ(1, observer_
->num_invocations());
273 EXPECT_EQ(0, observer_
->num_fullscreens());
274 EXPECT_EQ(1, observer_
->num_not_fullscreens());
277 // Test that the observer receives a true value when the window is fullscreen.
278 TEST_F(VideoDetectorTest
, FullscreenWindow
) {
279 if (!SupportsMultipleDisplays())
282 UpdateDisplay("1024x768,1024x768");
284 const gfx::Rect
kLeftBounds(gfx::Point(), gfx::Size(1024, 768));
285 scoped_ptr
<aura::Window
> window(
286 CreateTestWindowInShell(SK_ColorRED
, 12345, kLeftBounds
));
287 wm::WindowState
* window_state
= wm::GetWindowState(window
.get());
288 const wm::WMEvent
toggle_fullscreen_event(wm::WM_EVENT_TOGGLE_FULLSCREEN
);
289 window_state
->OnWMEvent(&toggle_fullscreen_event
);
290 ASSERT_TRUE(window_state
->IsFullscreen());
292 const gfx::Rect
kUpdateRegion(
294 gfx::Size(VideoDetector::kMinUpdateWidth
,
295 VideoDetector::kMinUpdateHeight
));
296 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
297 detector_
->OnDelegatedFrameDamage(window
.get(), kUpdateRegion
);
298 EXPECT_EQ(1, observer_
->num_invocations());
299 EXPECT_EQ(1, observer_
->num_fullscreens());
300 EXPECT_EQ(0, observer_
->num_not_fullscreens());
302 // Make the first window non-fullscreen and open a second fullscreen window on
303 // a different desktop.
304 window_state
->OnWMEvent(&toggle_fullscreen_event
);
305 ASSERT_FALSE(window_state
->IsFullscreen());
306 const gfx::Rect
kRightBounds(gfx::Point(1024, 0), gfx::Size(1024, 768));
307 scoped_ptr
<aura::Window
> other_window(
308 CreateTestWindowInShell(SK_ColorBLUE
, 6789, kRightBounds
));
309 wm::WindowState
* other_window_state
= wm::GetWindowState(other_window
.get());
310 other_window_state
->OnWMEvent(&toggle_fullscreen_event
);
311 ASSERT_TRUE(other_window_state
->IsFullscreen());
313 // When video is detected in the first (now non-fullscreen) window, fullscreen
314 // video should still be reported due to the second window being fullscreen.
315 // This avoids situations where non-fullscreen video could be reported when
316 // multiple videos are playing in fullscreen and non-fullscreen windows.
317 observer_
->reset_stats();
318 AdvanceTime(base::TimeDelta::FromSeconds(2));
319 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
320 detector_
->OnDelegatedFrameDamage(window
.get(), kUpdateRegion
);
321 EXPECT_EQ(1, observer_
->num_invocations());
322 EXPECT_EQ(1, observer_
->num_fullscreens());
323 EXPECT_EQ(0, observer_
->num_not_fullscreens());
325 // Make the second window non-fullscreen and check that the next video report
326 // is non-fullscreen.
327 other_window_state
->OnWMEvent(&toggle_fullscreen_event
);
328 ASSERT_FALSE(other_window_state
->IsFullscreen());
329 observer_
->reset_stats();
330 AdvanceTime(base::TimeDelta::FromSeconds(2));
331 for (int i
= 0; i
< VideoDetector::kMinFramesPerSecond
; ++i
)
332 detector_
->OnDelegatedFrameDamage(window
.get(), kUpdateRegion
);
333 EXPECT_EQ(1, observer_
->num_invocations());
334 EXPECT_EQ(0, observer_
->num_fullscreens());
335 EXPECT_EQ(1, observer_
->num_not_fullscreens());