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/display/display_controller.h"
6 #include "ash/display/display_manager.h"
7 #include "ash/root_window_controller.h"
8 #include "ash/screen_util.h"
10 #include "ash/shell_window_ids.h"
11 #include "ash/system/tray/system_tray.h"
12 #include "ash/test/ash_test_base.h"
13 #include "ash/wm/coordinate_conversion.h"
14 #include "ash/wm/window_properties.h"
15 #include "ash/wm/window_util.h"
16 #include "base/strings/string_util.h"
17 #include "base/strings/utf_string_conversions.h"
18 #include "ui/aura/client/capture_client.h"
19 #include "ui/aura/client/focus_client.h"
20 #include "ui/aura/test/event_generator.h"
21 #include "ui/aura/test/test_windows.h"
22 #include "ui/aura/test/window_test_api.h"
23 #include "ui/aura/window.h"
24 #include "ui/aura/window_event_dispatcher.h"
25 #include "ui/base/cursor/cursor.h"
26 #include "ui/events/event_handler.h"
27 #include "ui/gfx/display.h"
28 #include "ui/gfx/screen.h"
29 #include "ui/views/controls/textfield/textfield.h"
30 #include "ui/views/widget/widget.h"
31 #include "ui/views/widget/widget_delegate.h"
32 #include "ui/wm/public/activation_client.h"
37 void SetSecondaryDisplayLayout(DisplayLayout::Position position
) {
38 DisplayLayout layout
=
39 Shell::GetInstance()->display_manager()->GetCurrentDisplayLayout();
40 layout
.position
= position
;
41 Shell::GetInstance()->display_manager()->
42 SetLayoutForCurrentDisplays(layout
);
45 class ModalWidgetDelegate
: public views::WidgetDelegateView
{
47 ModalWidgetDelegate() {}
48 virtual ~ModalWidgetDelegate() {}
50 // Overridden from views::WidgetDelegate:
51 virtual views::View
* GetContentsView() OVERRIDE
{
54 virtual ui::ModalType
GetModalType() const OVERRIDE
{
55 return ui::MODAL_TYPE_SYSTEM
;
59 DISALLOW_COPY_AND_ASSIGN(ModalWidgetDelegate
);
62 // An event handler which moves the target window to the secondary root window
63 // at pre-handle phase of a mouse release event.
64 class MoveWindowByClickEventHandler
: public ui::EventHandler
{
66 explicit MoveWindowByClickEventHandler(aura::Window
* target
)
68 virtual ~MoveWindowByClickEventHandler() {}
71 // ui::EventHandler overrides:
72 virtual void OnMouseEvent(ui::MouseEvent
* event
) OVERRIDE
{
73 if (event
->type() == ui::ET_MOUSE_RELEASED
) {
74 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
75 DCHECK_LT(1u, root_windows
.size());
76 root_windows
[1]->AddChild(target_
);
80 aura::Window
* target_
;
81 DISALLOW_COPY_AND_ASSIGN(MoveWindowByClickEventHandler
);
84 // An event handler which records the event's locations.
85 class EventLocationRecordingEventHandler
: public ui::EventHandler
{
87 explicit EventLocationRecordingEventHandler() {
90 virtual ~EventLocationRecordingEventHandler() {}
92 std::string
GetLocationsAndReset() {
94 location_
.ToString() + " " + root_location_
.ToString();
100 // ui::EventHandler overrides:
101 virtual void OnMouseEvent(ui::MouseEvent
* event
) OVERRIDE
{
102 if (event
->type() == ui::ET_MOUSE_MOVED
||
103 event
->type() == ui::ET_MOUSE_DRAGGED
) {
104 location_
= event
->location();
105 root_location_
= event
->root_location();
110 location_
.SetPoint(-999, -999);
111 root_location_
.SetPoint(-999, -999);
114 gfx::Point root_location_
;
115 gfx::Point location_
;
117 DISALLOW_COPY_AND_ASSIGN(EventLocationRecordingEventHandler
);
120 class EventLocationHandler
: public ui::EventHandler
{
122 EventLocationHandler() {}
123 virtual ~EventLocationHandler() {}
125 const gfx::Point
& press_location() const { return press_location_
; }
126 const gfx::Point
& release_location() const { return release_location_
; }
130 virtual void OnMouseEvent(ui::MouseEvent
* event
) OVERRIDE
{
131 if (event
->type() == ui::ET_MOUSE_PRESSED
)
132 press_location_
= event
->location();
133 else if (event
->type() == ui::ET_MOUSE_RELEASED
)
134 release_location_
= event
->location();
137 gfx::Point press_location_
;
138 gfx::Point release_location_
;
140 DISALLOW_COPY_AND_ASSIGN(EventLocationHandler
);
145 class ExtendedDesktopTest
: public test::AshTestBase
{
147 views::Widget
* CreateTestWidget(const gfx::Rect
& bounds
) {
148 return CreateTestWidgetWithParentAndContext(
149 NULL
, CurrentContext(), bounds
, false);
152 views::Widget
* CreateTestWidgetWithParent(views::Widget
* parent
,
153 const gfx::Rect
& bounds
,
156 return CreateTestWidgetWithParentAndContext(parent
, NULL
, bounds
, child
);
159 views::Widget
* CreateTestWidgetWithParentAndContext(views::Widget
* parent
,
160 gfx::NativeView context
,
161 const gfx::Rect
& bounds
,
163 views::Widget::InitParams
params(views::Widget::InitParams::TYPE_WINDOW
);
165 params
.parent
= parent
->GetNativeView();
166 params
.context
= context
;
167 params
.bounds
= bounds
;
168 params
.child
= child
;
169 views::Widget
* widget
= new views::Widget
;
170 widget
->Init(params
);
176 // Test conditions that root windows in extended desktop mode
178 TEST_F(ExtendedDesktopTest
, Basic
) {
179 if (!SupportsMultipleDisplays())
182 UpdateDisplay("1000x600,600x400");
183 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
185 // All root windows must have the root window controller.
186 ASSERT_EQ(2U, root_windows
.size());
187 for (aura::Window::Windows::const_iterator iter
= root_windows
.begin();
188 iter
!= root_windows
.end(); ++iter
) {
189 EXPECT_TRUE(GetRootWindowController(*iter
) != NULL
);
191 // Make sure root windows share the same controllers.
192 EXPECT_EQ(aura::client::GetFocusClient(root_windows
[0]),
193 aura::client::GetFocusClient(root_windows
[1]));
194 EXPECT_EQ(aura::client::GetActivationClient(root_windows
[0]),
195 aura::client::GetActivationClient(root_windows
[1]));
196 EXPECT_EQ(aura::client::GetCaptureClient(root_windows
[0]),
197 aura::client::GetCaptureClient(root_windows
[1]));
200 TEST_F(ExtendedDesktopTest
, Activation
) {
201 if (!SupportsMultipleDisplays())
204 UpdateDisplay("1000x600,600x400");
205 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
207 views::Widget
* widget_on_1st
= CreateTestWidget(gfx::Rect(10, 10, 100, 100));
208 views::Widget
* widget_on_2nd
=
209 CreateTestWidget(gfx::Rect(1200, 10, 100, 100));
210 EXPECT_EQ(root_windows
[0], widget_on_1st
->GetNativeView()->GetRootWindow());
211 EXPECT_EQ(root_windows
[1], widget_on_2nd
->GetNativeView()->GetRootWindow());
213 EXPECT_EQ(widget_on_2nd
->GetNativeView(),
214 aura::client::GetFocusClient(root_windows
[0])->GetFocusedWindow());
215 EXPECT_TRUE(wm::IsActiveWindow(widget_on_2nd
->GetNativeView()));
217 aura::test::EventGenerator
& event_generator(GetEventGenerator());
218 // Clicking a window changes the active window and active root window.
219 event_generator
.MoveMouseToCenterOf(widget_on_1st
->GetNativeView());
220 event_generator
.ClickLeftButton();
222 EXPECT_EQ(widget_on_1st
->GetNativeView(),
223 aura::client::GetFocusClient(root_windows
[0])->GetFocusedWindow());
224 EXPECT_TRUE(wm::IsActiveWindow(widget_on_1st
->GetNativeView()));
226 event_generator
.MoveMouseToCenterOf(widget_on_2nd
->GetNativeView());
227 event_generator
.ClickLeftButton();
229 EXPECT_EQ(widget_on_2nd
->GetNativeView(),
230 aura::client::GetFocusClient(root_windows
[0])->GetFocusedWindow());
231 EXPECT_TRUE(wm::IsActiveWindow(widget_on_2nd
->GetNativeView()));
234 TEST_F(ExtendedDesktopTest
, SystemModal
) {
235 if (!SupportsMultipleDisplays())
238 UpdateDisplay("1000x600,600x400");
239 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
241 views::Widget
* widget_on_1st
= CreateTestWidget(gfx::Rect(10, 10, 100, 100));
242 EXPECT_TRUE(wm::IsActiveWindow(widget_on_1st
->GetNativeView()));
243 EXPECT_EQ(root_windows
[0], widget_on_1st
->GetNativeView()->GetRootWindow());
244 EXPECT_EQ(root_windows
[0], Shell::GetTargetRootWindow());
246 // Open system modal. Make sure it's on 2nd root window and active.
247 views::Widget
* modal_widget
= views::Widget::CreateWindowWithContextAndBounds(
248 new ModalWidgetDelegate(),
250 gfx::Rect(1200, 100, 100, 100));
251 modal_widget
->Show();
252 EXPECT_TRUE(wm::IsActiveWindow(modal_widget
->GetNativeView()));
253 EXPECT_EQ(root_windows
[1], modal_widget
->GetNativeView()->GetRootWindow());
254 EXPECT_EQ(root_windows
[1], Shell::GetTargetRootWindow());
256 aura::test::EventGenerator
& event_generator(GetEventGenerator());
258 // Clicking a widget on widget_on_1st display should not change activation.
259 event_generator
.MoveMouseToCenterOf(widget_on_1st
->GetNativeView());
260 event_generator
.ClickLeftButton();
261 EXPECT_TRUE(wm::IsActiveWindow(modal_widget
->GetNativeView()));
262 EXPECT_EQ(root_windows
[1], Shell::GetTargetRootWindow());
264 // Close system modal and so clicking a widget should work now.
265 modal_widget
->Close();
266 event_generator
.MoveMouseToCenterOf(widget_on_1st
->GetNativeView());
267 event_generator
.ClickLeftButton();
268 EXPECT_TRUE(wm::IsActiveWindow(widget_on_1st
->GetNativeView()));
269 EXPECT_EQ(root_windows
[0], Shell::GetTargetRootWindow());
272 TEST_F(ExtendedDesktopTest
, TestCursor
) {
273 if (!SupportsMultipleDisplays())
276 UpdateDisplay("1000x600,600x400");
277 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
278 aura::WindowTreeHost
* host0
= root_windows
[0]->GetHost();
279 aura::WindowTreeHost
* host1
= root_windows
[1]->GetHost();
280 EXPECT_EQ(ui::kCursorPointer
, host0
->last_cursor().native_type());
281 EXPECT_EQ(ui::kCursorPointer
, host1
->last_cursor().native_type());
282 Shell::GetInstance()->cursor_manager()->SetCursor(ui::kCursorCopy
);
283 EXPECT_EQ(ui::kCursorCopy
, host0
->last_cursor().native_type());
284 EXPECT_EQ(ui::kCursorCopy
, host1
->last_cursor().native_type());
287 TEST_F(ExtendedDesktopTest
, TestCursorLocation
) {
288 if (!SupportsMultipleDisplays())
291 UpdateDisplay("1000x600,600x400");
292 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
293 aura::test::WindowTestApi
root_window0_test_api(root_windows
[0]);
294 aura::test::WindowTestApi
root_window1_test_api(root_windows
[1]);
296 root_windows
[0]->MoveCursorTo(gfx::Point(10, 10));
297 EXPECT_EQ("10,10", Shell::GetScreen()->GetCursorScreenPoint().ToString());
298 EXPECT_TRUE(root_window0_test_api
.ContainsMouse());
299 EXPECT_FALSE(root_window1_test_api
.ContainsMouse());
300 root_windows
[1]->MoveCursorTo(gfx::Point(10, 20));
301 EXPECT_EQ("1010,20", Shell::GetScreen()->GetCursorScreenPoint().ToString());
302 EXPECT_FALSE(root_window0_test_api
.ContainsMouse());
303 EXPECT_TRUE(root_window1_test_api
.ContainsMouse());
304 root_windows
[0]->MoveCursorTo(gfx::Point(20, 10));
305 EXPECT_EQ("20,10", Shell::GetScreen()->GetCursorScreenPoint().ToString());
306 EXPECT_TRUE(root_window0_test_api
.ContainsMouse());
307 EXPECT_FALSE(root_window1_test_api
.ContainsMouse());
310 TEST_F(ExtendedDesktopTest
, GetRootWindowAt
) {
311 if (!SupportsMultipleDisplays())
314 UpdateDisplay("700x500,500x500");
315 SetSecondaryDisplayLayout(DisplayLayout::LEFT
);
316 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
318 EXPECT_EQ(root_windows
[1], wm::GetRootWindowAt(gfx::Point(-400, 100)));
319 EXPECT_EQ(root_windows
[1], wm::GetRootWindowAt(gfx::Point(-1, 100)));
320 EXPECT_EQ(root_windows
[0], wm::GetRootWindowAt(gfx::Point(0, 300)));
321 EXPECT_EQ(root_windows
[0], wm::GetRootWindowAt(gfx::Point(700, 300)));
324 EXPECT_EQ(root_windows
[0], wm::GetRootWindowAt(gfx::Point(0, 0)));
326 // Out of range point should return the nearest root window
327 EXPECT_EQ(root_windows
[1], wm::GetRootWindowAt(gfx::Point(-600, 0)));
328 EXPECT_EQ(root_windows
[0], wm::GetRootWindowAt(gfx::Point(701, 100)));
331 TEST_F(ExtendedDesktopTest
, GetRootWindowMatching
) {
332 if (!SupportsMultipleDisplays())
335 UpdateDisplay("700x500,500x500");
336 SetSecondaryDisplayLayout(DisplayLayout::LEFT
);
338 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
341 EXPECT_EQ(root_windows
[1],
342 wm::GetRootWindowMatching(gfx::Rect(-300, 10, 50, 50)));
343 EXPECT_EQ(root_windows
[0],
344 wm::GetRootWindowMatching(gfx::Rect(100, 10, 50, 50)));
346 // Intersecting rect.
347 EXPECT_EQ(root_windows
[1],
348 wm::GetRootWindowMatching(gfx::Rect(-200, 0, 300, 300)));
349 EXPECT_EQ(root_windows
[0],
350 wm::GetRootWindowMatching(gfx::Rect(-100, 0, 300, 300)));
353 EXPECT_EQ(root_windows
[0],
354 wm::GetRootWindowMatching(gfx::Rect(0, 0, 0, 0)));
355 EXPECT_EQ(root_windows
[0],
356 wm::GetRootWindowMatching(gfx::Rect(0, 0, 1, 1)));
359 EXPECT_EQ(root_windows
[1],
360 wm::GetRootWindowMatching(gfx::Rect(-400, 100, 0, 0)));
361 EXPECT_EQ(root_windows
[0],
362 wm::GetRootWindowMatching(gfx::Rect(100, 100, 0, 0)));
364 // Out of range rect should return the primary root window.
365 EXPECT_EQ(root_windows
[0],
366 wm::GetRootWindowMatching(gfx::Rect(-600, -300, 50, 50)));
367 EXPECT_EQ(root_windows
[0],
368 wm::GetRootWindowMatching(gfx::Rect(0, 1000, 50, 50)));
371 TEST_F(ExtendedDesktopTest
, Capture
) {
372 if (!SupportsMultipleDisplays())
375 UpdateDisplay("1000x600,600x400");
376 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
378 aura::test::EventCountDelegate r1_d1
;
379 aura::test::EventCountDelegate r1_d2
;
380 aura::test::EventCountDelegate r2_d1
;
382 scoped_ptr
<aura::Window
> r1_w1(aura::test::CreateTestWindowWithDelegate(
383 &r1_d1
, 0, gfx::Rect(10, 10, 100, 100), root_windows
[0]));
384 scoped_ptr
<aura::Window
> r1_w2(aura::test::CreateTestWindowWithDelegate(
385 &r1_d2
, 0, gfx::Rect(10, 100, 100, 100), root_windows
[0]));
386 scoped_ptr
<aura::Window
> r2_w1(aura::test::CreateTestWindowWithDelegate(
387 &r2_d1
, 0, gfx::Rect(10, 10, 100, 100), root_windows
[1]));
391 EXPECT_EQ(r1_w1
.get(),
392 aura::client::GetCaptureWindow(r2_w1
->GetRootWindow()));
394 aura::test::EventGenerator
& generator
= GetEventGenerator();
395 generator
.MoveMouseToCenterOf(r2_w1
.get());
396 // |r1_w1| will receive the events because it has capture.
397 EXPECT_EQ("1 1 0", r1_d1
.GetMouseMotionCountsAndReset());
398 EXPECT_EQ("0 0 0", r1_d2
.GetMouseMotionCountsAndReset());
399 EXPECT_EQ("0 0 0", r2_d1
.GetMouseMotionCountsAndReset());
401 generator
.ClickLeftButton();
402 EXPECT_EQ("0 0 0", r2_d1
.GetMouseMotionCountsAndReset());
403 EXPECT_EQ("0 0", r2_d1
.GetMouseButtonCountsAndReset());
404 // The mouse is outside. On chromeos, the mouse is warped to the
405 // dest root window, but it's not implemented on Win yet, so
406 // no mouse move event on Win.
407 EXPECT_EQ("0 0 0", r1_d1
.GetMouseMotionCountsAndReset());
408 EXPECT_EQ("1 1", r1_d1
.GetMouseButtonCountsAndReset());
410 generator
.MoveMouseTo(15, 15);
411 EXPECT_EQ("0 1 0", r1_d1
.GetMouseMotionCountsAndReset());
412 EXPECT_EQ("0 0 0", r1_d2
.GetMouseMotionCountsAndReset());
415 EXPECT_EQ(r1_w2
.get(),
416 aura::client::GetCaptureWindow(r2_w1
->GetRootWindow()));
417 generator
.MoveMouseBy(10, 10);
418 // |r1_w2| has the capture. So it will receive the mouse-move event.
419 EXPECT_EQ("0 0 0", r1_d1
.GetMouseMotionCountsAndReset());
420 EXPECT_EQ("0 1 0", r1_d2
.GetMouseMotionCountsAndReset());
421 EXPECT_EQ("0 0 0", r2_d1
.GetMouseMotionCountsAndReset());
423 generator
.ClickLeftButton();
424 EXPECT_EQ("0 0 0", r2_d1
.GetMouseMotionCountsAndReset());
425 EXPECT_EQ("0 0", r2_d1
.GetMouseButtonCountsAndReset());
426 EXPECT_EQ("0 0 0", r1_d2
.GetMouseMotionCountsAndReset());
427 EXPECT_EQ("1 1", r1_d2
.GetMouseButtonCountsAndReset());
429 r1_w2
->ReleaseCapture();
430 EXPECT_EQ(NULL
, aura::client::GetCaptureWindow(r2_w1
->GetRootWindow()));
432 generator
.MoveMouseToCenterOf(r2_w1
.get());
433 generator
.ClickLeftButton();
434 EXPECT_EQ("1 1 0", r2_d1
.GetMouseMotionCountsAndReset());
435 EXPECT_EQ("1 1", r2_d1
.GetMouseButtonCountsAndReset());
436 // Make sure the mouse_moved_handler_ is properly reset.
437 EXPECT_EQ("0 0 0", r1_d2
.GetMouseMotionCountsAndReset());
438 EXPECT_EQ("0 0", r1_d2
.GetMouseButtonCountsAndReset());
441 TEST_F(ExtendedDesktopTest
, CaptureEventLocation
) {
442 if (!SupportsMultipleDisplays())
445 UpdateDisplay("1000x600,600x400");
446 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
448 aura::test::EventCountDelegate r1_d1
;
449 aura::test::EventCountDelegate r1_d2
;
450 aura::test::EventCountDelegate r2_d1
;
452 scoped_ptr
<aura::Window
> r1_w1(aura::test::CreateTestWindowWithDelegate(
453 &r1_d1
, 0, gfx::Rect(10, 10, 100, 100), root_windows
[0]));
454 scoped_ptr
<aura::Window
> r1_w2(aura::test::CreateTestWindowWithDelegate(
455 &r1_d2
, 0, gfx::Rect(10, 100, 100, 100), root_windows
[0]));
456 scoped_ptr
<aura::Window
> r2_w1(aura::test::CreateTestWindowWithDelegate(
457 &r2_d1
, 0, gfx::Rect(10, 10, 100, 100), root_windows
[1]));
461 aura::test::EventGenerator
& generator
= GetEventGenerator();
462 generator
.MoveMouseToCenterOf(r2_w1
.get());
463 EXPECT_EQ(gfx::Point(1060, 60).ToString(),
464 generator
.current_location().ToString());
466 EventLocationHandler location_handler
;
467 r1_w1
->AddPreTargetHandler(&location_handler
);
468 generator
.ClickLeftButton();
469 r1_w1
->RemovePreTargetHandler(&location_handler
);
470 EXPECT_EQ(gfx::Point(1050, 50).ToString(),
471 location_handler
.press_location().ToString());
472 EXPECT_EQ(gfx::Point(1050, 50).ToString(),
473 location_handler
.release_location().ToString());
476 TEST_F(ExtendedDesktopTest
, CaptureEventLocationHighDPI
) {
477 if (!SupportsMultipleDisplays())
480 UpdateDisplay("1000x600*2,600x400");
481 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
483 aura::test::EventCountDelegate r1_d1
;
484 aura::test::EventCountDelegate r1_d2
;
485 aura::test::EventCountDelegate r2_d1
;
487 scoped_ptr
<aura::Window
> r1_w1(aura::test::CreateTestWindowWithDelegate(
488 &r1_d1
, 0, gfx::Rect(10, 10, 100, 100), root_windows
[0]));
489 scoped_ptr
<aura::Window
> r1_w2(aura::test::CreateTestWindowWithDelegate(
490 &r1_d2
, 0, gfx::Rect(10, 100, 100, 100), root_windows
[0]));
491 scoped_ptr
<aura::Window
> r2_w1(aura::test::CreateTestWindowWithDelegate(
492 &r2_d1
, 0, gfx::Rect(10, 10, 100, 100), root_windows
[1]));
496 aura::test::EventGenerator
& generator
= GetEventGenerator();
497 generator
.MoveMouseToCenterOf(r2_w1
.get());
498 EXPECT_EQ(gfx::Point(560, 60).ToString(),
499 generator
.current_location().ToString());
501 EventLocationHandler location_handler
;
502 r1_w1
->AddPreTargetHandler(&location_handler
);
503 generator
.ClickLeftButton();
504 r1_w1
->RemovePreTargetHandler(&location_handler
);
505 EXPECT_EQ(gfx::Point(550, 50).ToString(),
506 location_handler
.press_location().ToString());
507 EXPECT_EQ(gfx::Point(550, 50).ToString(),
508 location_handler
.release_location().ToString());
511 TEST_F(ExtendedDesktopTest
, CaptureEventLocationHighDPI_2
) {
512 if (!SupportsMultipleDisplays())
515 UpdateDisplay("1000x600,600x400*2");
516 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
518 aura::test::EventCountDelegate r1_d1
;
519 aura::test::EventCountDelegate r1_d2
;
520 aura::test::EventCountDelegate r2_d1
;
522 scoped_ptr
<aura::Window
> r1_w1(aura::test::CreateTestWindowWithDelegate(
523 &r1_d1
, 0, gfx::Rect(10, 10, 100, 100), root_windows
[0]));
524 scoped_ptr
<aura::Window
> r1_w2(aura::test::CreateTestWindowWithDelegate(
525 &r1_d2
, 0, gfx::Rect(10, 100, 100, 100), root_windows
[0]));
526 scoped_ptr
<aura::Window
> r2_w1(aura::test::CreateTestWindowWithDelegate(
527 &r2_d1
, 0, gfx::Rect(10, 10, 100, 100), root_windows
[1]));
531 aura::test::EventGenerator
& generator
= GetEventGenerator();
532 generator
.MoveMouseToCenterOf(r2_w1
.get());
533 EXPECT_EQ(gfx::Point(1060, 60).ToString(),
534 generator
.current_location().ToString());
536 EventLocationHandler location_handler
;
537 r1_w1
->AddPreTargetHandler(&location_handler
);
538 generator
.ClickLeftButton();
539 r1_w1
->RemovePreTargetHandler(&location_handler
);
540 // Event-generator dispatches the event in the primary root-window's coord
541 // space. Since the location is (1060, 60), it goes to the secondary
542 // root-window as (30, 30) since the secondary root-window has a device scale
544 EXPECT_EQ(gfx::Point(1020, 20).ToString(),
545 location_handler
.press_location().ToString());
546 EXPECT_EQ(gfx::Point(1020, 20).ToString(),
547 location_handler
.release_location().ToString());
550 TEST_F(ExtendedDesktopTest
, MoveWindow
) {
551 if (!SupportsMultipleDisplays())
554 UpdateDisplay("1000x600,600x400");
555 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
556 views::Widget
* d1
= CreateTestWidget(gfx::Rect(10, 10, 100, 100));
558 EXPECT_EQ(root_windows
[0], d1
->GetNativeView()->GetRootWindow());
560 d1
->SetBounds(gfx::Rect(1010, 10, 100, 100));
561 EXPECT_EQ("1010,10 100x100",
562 d1
->GetWindowBoundsInScreen().ToString());
564 EXPECT_EQ(root_windows
[1], d1
->GetNativeView()->GetRootWindow());
566 d1
->SetBounds(gfx::Rect(10, 10, 100, 100));
567 EXPECT_EQ("10,10 100x100",
568 d1
->GetWindowBoundsInScreen().ToString());
570 EXPECT_EQ(root_windows
[0], d1
->GetNativeView()->GetRootWindow());
572 // Make sure the bounds which doesn't fit to the root window
574 d1
->SetBounds(gfx::Rect(1560, 30, 100, 100));
575 EXPECT_EQ(root_windows
[1], d1
->GetNativeView()->GetRootWindow());
576 EXPECT_EQ("1560,30 100x100",
577 d1
->GetWindowBoundsInScreen().ToString());
579 // Setting outside of root windows will be moved to primary root window.
580 // TODO(oshima): This one probably should pick the closest root window.
581 d1
->SetBounds(gfx::Rect(200, 10, 100, 100));
582 EXPECT_EQ(root_windows
[0], d1
->GetNativeView()->GetRootWindow());
585 // Verifies if the mouse event arrives to the window even when the window
586 // moves to another root in a pre-target handler. See: crbug.com/157583
587 TEST_F(ExtendedDesktopTest
, MoveWindowByMouseClick
) {
588 if (!SupportsMultipleDisplays())
591 UpdateDisplay("1000x600,600x400");
593 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
594 aura::test::EventCountDelegate delegate
;
595 scoped_ptr
<aura::Window
> window(aura::test::CreateTestWindowWithDelegate(
596 &delegate
, 0, gfx::Rect(10, 10, 100, 100), root_windows
[0]));
597 MoveWindowByClickEventHandler
event_handler(window
.get());
598 window
->AddPreTargetHandler(&event_handler
);
600 aura::test::EventGenerator
& event_generator(GetEventGenerator());
602 event_generator
.MoveMouseToCenterOf(window
.get());
603 event_generator
.ClickLeftButton();
604 // Both mouse pressed and released arrive at the window and its delegate.
605 EXPECT_EQ("1 1", delegate
.GetMouseButtonCountsAndReset());
606 // Also event_handler moves the window to another root at mouse release.
607 EXPECT_EQ(root_windows
[1], window
->GetRootWindow());
610 TEST_F(ExtendedDesktopTest
, MoveWindowToDisplay
) {
611 if (!SupportsMultipleDisplays())
614 UpdateDisplay("1000x1000,1000x1000");
615 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
617 gfx::Display display0
= Shell::GetScreen()->GetDisplayMatching(
618 root_windows
[0]->GetBoundsInScreen());
619 gfx::Display display1
= Shell::GetScreen()->GetDisplayMatching(
620 root_windows
[1]->GetBoundsInScreen());
621 EXPECT_NE(display0
.id(), display1
.id());
623 views::Widget
* d1
= CreateTestWidget(gfx::Rect(10, 10, 1000, 100));
624 EXPECT_EQ(root_windows
[0], d1
->GetNativeView()->GetRootWindow());
626 // Move the window where the window spans both root windows. Since the second
627 // parameter is |display1|, the window should be shown on the secondary root.
628 d1
->GetNativeWindow()->SetBoundsInScreen(gfx::Rect(500, 10, 1000, 100),
630 EXPECT_EQ("500,10 1000x100",
631 d1
->GetWindowBoundsInScreen().ToString());
632 EXPECT_EQ(root_windows
[1], d1
->GetNativeView()->GetRootWindow());
634 // Move to the primary root.
635 d1
->GetNativeWindow()->SetBoundsInScreen(gfx::Rect(500, 10, 1000, 100),
637 EXPECT_EQ("500,10 1000x100",
638 d1
->GetWindowBoundsInScreen().ToString());
639 EXPECT_EQ(root_windows
[0], d1
->GetNativeView()->GetRootWindow());
642 TEST_F(ExtendedDesktopTest
, MoveWindowWithTransient
) {
643 if (!SupportsMultipleDisplays())
646 UpdateDisplay("1000x600,600x400");
647 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
648 views::Widget
* w1
= CreateTestWidget(gfx::Rect(10, 10, 100, 100));
649 views::Widget
* w1_t1
= CreateTestWidgetWithParent(
650 w1
, gfx::Rect(50, 50, 50, 50), false /* transient */);
651 // Transient child of the transient child.
652 views::Widget
* w1_t11
= CreateTestWidgetWithParent(
653 w1_t1
, gfx::Rect(1200, 70, 35, 35), false /* transient */);
655 views::Widget
* w11
= CreateTestWidgetWithParent(
656 w1
, gfx::Rect(10, 10, 40, 40), true /* child */);
657 views::Widget
* w11_t1
= CreateTestWidgetWithParent(
658 w1
, gfx::Rect(1300, 100, 80, 80), false /* transient */);
660 EXPECT_EQ(root_windows
[0], w1
->GetNativeView()->GetRootWindow());
661 EXPECT_EQ(root_windows
[0], w11
->GetNativeView()->GetRootWindow());
662 EXPECT_EQ(root_windows
[0], w1_t1
->GetNativeView()->GetRootWindow());
663 EXPECT_EQ(root_windows
[0], w1_t11
->GetNativeView()->GetRootWindow());
664 EXPECT_EQ(root_windows
[0], w11_t1
->GetNativeView()->GetRootWindow());
665 EXPECT_EQ("50,50 50x50",
666 w1_t1
->GetWindowBoundsInScreen().ToString());
667 EXPECT_EQ("1200,70 35x35",
668 w1_t11
->GetWindowBoundsInScreen().ToString());
669 EXPECT_EQ("20,20 40x40",
670 w11
->GetWindowBoundsInScreen().ToString());
671 EXPECT_EQ("1300,100 80x80",
672 w11_t1
->GetWindowBoundsInScreen().ToString());
674 w1
->SetBounds(gfx::Rect(1100, 10, 100, 100));
676 EXPECT_EQ(root_windows
[1], w1_t1
->GetNativeView()->GetRootWindow());
677 EXPECT_EQ(root_windows
[1], w1_t1
->GetNativeView()->GetRootWindow());
678 EXPECT_EQ(root_windows
[1], w1_t11
->GetNativeView()->GetRootWindow());
679 EXPECT_EQ(root_windows
[1], w11
->GetNativeView()->GetRootWindow());
680 EXPECT_EQ(root_windows
[1], w11_t1
->GetNativeView()->GetRootWindow());
682 EXPECT_EQ("1110,20 40x40",
683 w11
->GetWindowBoundsInScreen().ToString());
684 // Transient window's screen bounds stays the same.
685 EXPECT_EQ("50,50 50x50",
686 w1_t1
->GetWindowBoundsInScreen().ToString());
687 EXPECT_EQ("1200,70 35x35",
688 w1_t11
->GetWindowBoundsInScreen().ToString());
689 EXPECT_EQ("1300,100 80x80",
690 w11_t1
->GetWindowBoundsInScreen().ToString());
692 // Transient window doesn't move between root window unless
693 // its transient parent moves.
694 w1_t1
->SetBounds(gfx::Rect(10, 50, 50, 50));
695 EXPECT_EQ(root_windows
[1], w1_t1
->GetNativeView()->GetRootWindow());
696 EXPECT_EQ("10,50 50x50",
697 w1_t1
->GetWindowBoundsInScreen().ToString());
700 // Test if the Window::ConvertPointToTarget works across root windows.
701 // TODO(oshima): Move multiple display suport and this test to aura.
702 TEST_F(ExtendedDesktopTest
, ConvertPoint
) {
703 if (!SupportsMultipleDisplays())
705 gfx::Screen
* screen
= Shell::GetScreen();
706 UpdateDisplay("1000x600,600x400");
707 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
708 gfx::Display display_1
= screen
->GetDisplayNearestWindow(root_windows
[0]);
709 EXPECT_EQ("0,0", display_1
.bounds().origin().ToString());
710 gfx::Display display_2
= screen
->GetDisplayNearestWindow(root_windows
[1]);
711 EXPECT_EQ("1000,0", display_2
.bounds().origin().ToString());
714 CreateTestWidget(gfx::Rect(10, 10, 100, 100))->GetNativeView();
716 CreateTestWidget(gfx::Rect(1020, 20, 100, 100))->GetNativeView();
717 EXPECT_EQ(root_windows
[0], d1
->GetRootWindow());
718 EXPECT_EQ(root_windows
[1], d2
->GetRootWindow());
720 // Convert point in the Root2's window to the Root1's window Coord.
722 aura::Window::ConvertPointToTarget(root_windows
[1], root_windows
[0], &p
);
723 EXPECT_EQ("1000,0", p
.ToString());
725 aura::Window::ConvertPointToTarget(d2
, d1
, &p
);
726 EXPECT_EQ("1010,10", p
.ToString());
728 // Convert point in the Root1's window to the Root2's window Coord.
730 aura::Window::ConvertPointToTarget(root_windows
[0], root_windows
[1], &p
);
731 EXPECT_EQ("-1000,0", p
.ToString());
733 aura::Window::ConvertPointToTarget(d1
, d2
, &p
);
734 EXPECT_EQ("-1010,-10", p
.ToString());
736 // Move the 2nd display to the bottom and test again.
737 SetSecondaryDisplayLayout(DisplayLayout::BOTTOM
);
739 display_2
= screen
->GetDisplayNearestWindow(root_windows
[1]);
740 EXPECT_EQ("0,600", display_2
.bounds().origin().ToString());
742 // Convert point in Root2's window to Root1's window Coord.
744 aura::Window::ConvertPointToTarget(root_windows
[1], root_windows
[0], &p
);
745 EXPECT_EQ("0,600", p
.ToString());
747 aura::Window::ConvertPointToTarget(d2
, d1
, &p
);
748 EXPECT_EQ("10,610", p
.ToString());
750 // Convert point in Root1's window to Root2's window Coord.
752 aura::Window::ConvertPointToTarget(root_windows
[0], root_windows
[1], &p
);
753 EXPECT_EQ("0,-600", p
.ToString());
755 aura::Window::ConvertPointToTarget(d1
, d2
, &p
);
756 EXPECT_EQ("-10,-610", p
.ToString());
759 TEST_F(ExtendedDesktopTest
, OpenSystemTray
) {
760 if (!SupportsMultipleDisplays())
763 UpdateDisplay("500x600,600x400");
764 SystemTray
* tray
= ash::Shell::GetInstance()->GetPrimarySystemTray();
765 ASSERT_FALSE(tray
->HasSystemBubble());
767 aura::test::EventGenerator
& event_generator(GetEventGenerator());
769 // Opens the tray by a dummy click event and makes sure that adding/removing
770 // displays doesn't break anything.
771 event_generator
.MoveMouseToCenterOf(tray
->GetWidget()->GetNativeWindow());
772 event_generator
.ClickLeftButton();
773 EXPECT_TRUE(tray
->HasSystemBubble());
775 UpdateDisplay("500x600");
776 EXPECT_TRUE(tray
->HasSystemBubble());
777 UpdateDisplay("500x600,600x400");
778 EXPECT_TRUE(tray
->HasSystemBubble());
780 // Closes the tray and again makes sure that adding/removing displays doesn't
782 event_generator
.ClickLeftButton();
783 RunAllPendingInMessageLoop();
785 EXPECT_FALSE(tray
->HasSystemBubble());
787 UpdateDisplay("500x600");
788 EXPECT_FALSE(tray
->HasSystemBubble());
789 UpdateDisplay("500x600,600x400");
790 EXPECT_FALSE(tray
->HasSystemBubble());
793 TEST_F(ExtendedDesktopTest
, StayInSameRootWindow
) {
794 if (!SupportsMultipleDisplays())
797 UpdateDisplay("100x100,200x200");
798 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
799 views::Widget
* w1
= CreateTestWidget(gfx::Rect(10, 10, 50, 50));
800 EXPECT_EQ(root_windows
[0], w1
->GetNativeView()->GetRootWindow());
801 w1
->SetBounds(gfx::Rect(150, 10, 50, 50));
802 EXPECT_EQ(root_windows
[1], w1
->GetNativeView()->GetRootWindow());
804 // The widget stays in the same root if kStayInSameRootWindowKey is set to
806 w1
->GetNativeView()->SetProperty(kStayInSameRootWindowKey
, true);
807 w1
->SetBounds(gfx::Rect(10, 10, 50, 50));
808 EXPECT_EQ(root_windows
[1], w1
->GetNativeView()->GetRootWindow());
810 // The widget should now move to the 1st root window without the property.
811 w1
->GetNativeView()->ClearProperty(kStayInSameRootWindowKey
);
812 w1
->SetBounds(gfx::Rect(10, 10, 50, 50));
813 EXPECT_EQ(root_windows
[0], w1
->GetNativeView()->GetRootWindow());
815 // a window in SettingsBubbleContainer and StatusContainer should
816 // not move to another root window regardles of the bounds specified.
817 aura::Window
* settings_bubble_container
=
818 Shell::GetPrimaryRootWindowController()->GetContainer(
819 kShellWindowId_SettingBubbleContainer
);
820 aura::Window
* window
= aura::test::CreateTestWindowWithId(
821 100, settings_bubble_container
);
822 window
->SetBoundsInScreen(gfx::Rect(150, 10, 50, 50),
823 ScreenUtil::GetSecondaryDisplay());
824 EXPECT_EQ(root_windows
[0], window
->GetRootWindow());
826 aura::Window
* status_container
=
827 Shell::GetPrimaryRootWindowController()->GetContainer(
828 kShellWindowId_StatusContainer
);
829 window
= aura::test::CreateTestWindowWithId(100, status_container
);
830 window
->SetBoundsInScreen(gfx::Rect(150, 10, 50, 50),
831 ScreenUtil::GetSecondaryDisplay());
832 EXPECT_EQ(root_windows
[0], window
->GetRootWindow());
835 TEST_F(ExtendedDesktopTest
, KeyEventsOnLockScreen
) {
836 if (!SupportsMultipleDisplays())
839 UpdateDisplay("100x100,200x200");
840 aura::Window::Windows root_windows
= Shell::GetAllRootWindows();
842 // Create normal windows on both displays.
843 views::Widget
* widget1
= CreateTestWidget(
844 Shell::GetScreen()->GetPrimaryDisplay().bounds());
846 EXPECT_EQ(root_windows
[0], widget1
->GetNativeView()->GetRootWindow());
847 views::Widget
* widget2
= CreateTestWidget(
848 ScreenUtil::GetSecondaryDisplay().bounds());
850 EXPECT_EQ(root_windows
[1], widget2
->GetNativeView()->GetRootWindow());
852 // Create a LockScreen window.
853 views::Widget
* lock_widget
= CreateTestWidget(
854 Shell::GetScreen()->GetPrimaryDisplay().bounds());
855 views::Textfield
* textfield
= new views::Textfield
;
856 lock_widget
->client_view()->AddChildView(textfield
);
858 ash::Shell::GetContainer(Shell::GetPrimaryRootWindow(),
859 ash::kShellWindowId_LockScreenContainer
)
860 ->AddChild(lock_widget
->GetNativeView());
862 textfield
->RequestFocus();
864 aura::client::FocusClient
* focus_client
=
865 aura::client::GetFocusClient(root_windows
[0]);
866 EXPECT_EQ(lock_widget
->GetNativeView(), focus_client
->GetFocusedWindow());
868 // The lock window should get events on both root windows.
869 aura::test::EventGenerator
& event_generator(GetEventGenerator());
871 event_generator
.set_current_target(root_windows
[0]);
872 event_generator
.PressKey(ui::VKEY_A
, 0);
873 event_generator
.ReleaseKey(ui::VKEY_A
, 0);
874 EXPECT_EQ(lock_widget
->GetNativeView(), focus_client
->GetFocusedWindow());
875 EXPECT_EQ("a", base::UTF16ToASCII(textfield
->text()));
877 event_generator
.set_current_target(root_windows
[1]);
878 event_generator
.PressKey(ui::VKEY_B
, 0);
879 event_generator
.ReleaseKey(ui::VKEY_B
, 0);
880 EXPECT_EQ(lock_widget
->GetNativeView(), focus_client
->GetFocusedWindow());
881 EXPECT_EQ("ab", base::UTF16ToASCII(textfield
->text()));
883 // Deleting 2nd display. The lock window still should get the events.
884 UpdateDisplay("100x100");
885 event_generator
.PressKey(ui::VKEY_C
, 0);
886 event_generator
.ReleaseKey(ui::VKEY_C
, 0);
887 EXPECT_EQ(lock_widget
->GetNativeView(), focus_client
->GetFocusedWindow());
888 EXPECT_EQ("abc", base::UTF16ToASCII(textfield
->text()));
890 // Creating 2nd display again, and lock window still should get events
891 // on both root windows.
892 UpdateDisplay("100x100,200x200");
893 root_windows
= Shell::GetAllRootWindows();
894 event_generator
.set_current_target(root_windows
[0]);
895 event_generator
.PressKey(ui::VKEY_D
, 0);
896 event_generator
.ReleaseKey(ui::VKEY_D
, 0);
897 EXPECT_EQ(lock_widget
->GetNativeView(), focus_client
->GetFocusedWindow());
898 EXPECT_EQ("abcd", base::UTF16ToASCII(textfield
->text()));
900 event_generator
.set_current_target(root_windows
[1]);
901 event_generator
.PressKey(ui::VKEY_E
, 0);
902 event_generator
.ReleaseKey(ui::VKEY_E
, 0);
903 EXPECT_EQ(lock_widget
->GetNativeView(), focus_client
->GetFocusedWindow());
904 EXPECT_EQ("abcde", base::UTF16ToASCII(textfield
->text()));
907 TEST_F(ExtendedDesktopTest
, PassiveGrab
) {
908 if (!SupportsMultipleDisplays())
911 EventLocationRecordingEventHandler event_handler
;
912 ash::Shell::GetInstance()->AddPreTargetHandler(&event_handler
);
914 UpdateDisplay("300x300,200x200");
916 views::Widget
* widget
= CreateTestWidget(gfx::Rect(50, 50, 200, 200));
918 ASSERT_EQ("50,50 200x200", widget
->GetWindowBoundsInScreen().ToString());
920 aura::test::EventGenerator
& generator(GetEventGenerator());
921 generator
.MoveMouseTo(150, 150);
922 EXPECT_EQ("100,100 150,150", event_handler
.GetLocationsAndReset());
924 generator
.PressLeftButton();
925 generator
.MoveMouseTo(400, 150);
927 EXPECT_EQ("350,100 400,150", event_handler
.GetLocationsAndReset());
929 generator
.ReleaseLeftButton();
930 EXPECT_EQ("-999,-999 -999,-999", event_handler
.GetLocationsAndReset());
932 generator
.MoveMouseTo(400, 150);
933 EXPECT_EQ("100,150 100,150", event_handler
.GetLocationsAndReset());
935 ash::Shell::GetInstance()->RemovePreTargetHandler(&event_handler
);