| // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "ash/public/cpp/shell_window_ids.h" | 
 | #include "ash/root_window_controller.h" | 
 | #include "ash/shell.h" | 
 | #include "ash/system/status_area_widget.h" | 
 | #include "ash/system/status_area_widget_test_helper.h" | 
 | #include "ash/system/unified/unified_system_tray.h" | 
 | #include "ash/test/ash_test_base.h" | 
 | #include "ash/window_factory.h" | 
 | #include "ash/wm/desks/desks_util.h" | 
 | #include "ash/wm/window_properties.h" | 
 | #include "ash/wm/window_util.h" | 
 | #include "base/run_loop.h" | 
 | #include "base/strings/string_util.h" | 
 | #include "base/strings/utf_string_conversions.h" | 
 | #include "ui/aura/client/capture_client.h" | 
 | #include "ui/aura/client/focus_client.h" | 
 | #include "ui/aura/test/test_windows.h" | 
 | #include "ui/aura/test/window_test_api.h" | 
 | #include "ui/aura/window.h" | 
 | #include "ui/aura/window_event_dispatcher.h" | 
 | #include "ui/base/cursor/cursor.h" | 
 | #include "ui/display/display.h" | 
 | #include "ui/display/display_layout.h" | 
 | #include "ui/display/manager/display_manager.h" | 
 | #include "ui/display/screen.h" | 
 | #include "ui/events/event_handler.h" | 
 | #include "ui/events/test/event_generator.h" | 
 | #include "ui/views/controls/textfield/textfield.h" | 
 | #include "ui/views/widget/widget.h" | 
 | #include "ui/views/widget/widget_delegate.h" | 
 | #include "ui/wm/core/window_util.h" | 
 | #include "ui/wm/public/activation_client.h" | 
 |  | 
 | namespace ash { | 
 | namespace { | 
 |  | 
 | void SetSecondaryDisplayLayout(display::DisplayPlacement::Position position) { | 
 |   std::unique_ptr<display::DisplayLayout> layout = | 
 |       Shell::Get()->display_manager()->GetCurrentDisplayLayout().Copy(); | 
 |   layout->placement_list[0].position = position; | 
 |   Shell::Get()->display_manager()->SetLayoutForCurrentDisplays( | 
 |       std::move(layout)); | 
 | } | 
 |  | 
 | class ModalWidgetDelegate : public views::WidgetDelegateView { | 
 |  public: | 
 |   ModalWidgetDelegate() = default; | 
 |   ~ModalWidgetDelegate() override = default; | 
 |  | 
 |   // Overridden from views::WidgetDelegate: | 
 |   ui::ModalType GetModalType() const override { return ui::MODAL_TYPE_SYSTEM; } | 
 |  | 
 |  private: | 
 |   DISALLOW_COPY_AND_ASSIGN(ModalWidgetDelegate); | 
 | }; | 
 |  | 
 | // An event handler which moves the target window to the secondary root window | 
 | // at pre-handle phase of a mouse release event. | 
 | class MoveWindowByClickEventHandler : public ui::EventHandler { | 
 |  public: | 
 |   explicit MoveWindowByClickEventHandler(aura::Window* target) | 
 |       : target_(target) {} | 
 |   ~MoveWindowByClickEventHandler() override = default; | 
 |  | 
 |  private: | 
 |   // ui::EventHandler overrides: | 
 |   void OnMouseEvent(ui::MouseEvent* event) override { | 
 |     if (event->type() == ui::ET_MOUSE_RELEASED) { | 
 |       aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |       DCHECK_LT(1u, root_windows.size()); | 
 |       root_windows[1]->AddChild(target_); | 
 |     } | 
 |   } | 
 |  | 
 |   aura::Window* target_; | 
 |   DISALLOW_COPY_AND_ASSIGN(MoveWindowByClickEventHandler); | 
 | }; | 
 |  | 
 | // An event handler which records the event's locations. | 
 | class EventLocationRecordingEventHandler : public ui::EventHandler { | 
 |  public: | 
 |   EventLocationRecordingEventHandler() { Reset(); } | 
 |   ~EventLocationRecordingEventHandler() override = default; | 
 |  | 
 |   // |location_| is relative to the target window. | 
 |   std::string GetLocation() const { return location_.ToString(); } | 
 |  | 
 |   // |root_location_| is relative to the display where the event started. | 
 |   std::string GetRootLocation() const { return root_location_.ToString(); } | 
 |  | 
 |   void Reset() { | 
 |     location_.SetPoint(-999, -999); | 
 |     root_location_.SetPoint(-999, -999); | 
 |   } | 
 |  | 
 |  private: | 
 |   // ui::EventHandler overrides: | 
 |   void OnMouseEvent(ui::MouseEvent* event) override { | 
 |     if (event->type() == ui::ET_MOUSE_MOVED || | 
 |         event->type() == ui::ET_MOUSE_DRAGGED) { | 
 |       location_ = event->location(); | 
 |       root_location_ = event->root_location(); | 
 |     } | 
 |   } | 
 |  | 
 |   gfx::Point root_location_; | 
 |   gfx::Point location_; | 
 |  | 
 |   DISALLOW_COPY_AND_ASSIGN(EventLocationRecordingEventHandler); | 
 | }; | 
 |  | 
 | class EventLocationHandler : public ui::EventHandler { | 
 |  public: | 
 |   EventLocationHandler() = default; | 
 |   ~EventLocationHandler() override = default; | 
 |  | 
 |   const gfx::Point& press_location() const { return press_location_; } | 
 |   const gfx::Point& release_location() const { return release_location_; } | 
 |  | 
 |  private: | 
 |   // ui::EventHandler: | 
 |   void OnMouseEvent(ui::MouseEvent* event) override { | 
 |     if (event->type() == ui::ET_MOUSE_PRESSED) | 
 |       press_location_ = event->location(); | 
 |     else if (event->type() == ui::ET_MOUSE_RELEASED) | 
 |       release_location_ = event->location(); | 
 |   } | 
 |  | 
 |   gfx::Point press_location_; | 
 |   gfx::Point release_location_; | 
 |  | 
 |   DISALLOW_COPY_AND_ASSIGN(EventLocationHandler); | 
 | }; | 
 |  | 
 | }  // namespace | 
 |  | 
 | class ExtendedDesktopTest : public AshTestBase { | 
 |  public: | 
 |   ExtendedDesktopTest() = default; | 
 |   ~ExtendedDesktopTest() override = default; | 
 |  | 
 |   void SetUp() override { | 
 |     AshTestBase::SetUp(); | 
 |  | 
 |     // Shell hides the cursor by default; show it for these tests. | 
 |     Shell::Get()->cursor_manager()->ShowCursor(); | 
 |   } | 
 |  | 
 |   // TODO(jamescook): Switch to AshTestBase::CreateTestWidget(). | 
 |   views::Widget* CreateTestWidget(const gfx::Rect& bounds) { | 
 |     views::Widget::InitParams params(views::Widget::InitParams::TYPE_WINDOW); | 
 |     params.bounds = bounds; | 
 |     views::Widget* widget = new views::Widget; | 
 |     params.context = CurrentContext(); | 
 |     widget->Init(std::move(params)); | 
 |     widget->Show(); | 
 |     return widget; | 
 |   } | 
 |  | 
 |  protected: | 
 |   bool IsBubbleShown() { | 
 |     return GetPrimaryUnifiedSystemTray()->IsBubbleShown(); | 
 |   } | 
 |  | 
 |   gfx::Rect GetSystemTrayBoundsInScreen() { | 
 |     return GetPrimaryUnifiedSystemTray()->GetBoundsInScreen(); | 
 |   } | 
 |  | 
 |  private: | 
 |   DISALLOW_COPY_AND_ASSIGN(ExtendedDesktopTest); | 
 | }; | 
 |  | 
 | // Test conditions that root windows in extended desktop mode must satisfy. | 
 | TEST_F(ExtendedDesktopTest, Basic) { | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   // All root windows must have a root window controller. | 
 |   ASSERT_EQ(2U, root_windows.size()); | 
 |   EXPECT_TRUE(RootWindowController::ForWindow(root_windows[0])); | 
 |   EXPECT_TRUE(RootWindowController::ForWindow(root_windows[1])); | 
 |   // Make sure root windows share the same controllers. | 
 |   EXPECT_EQ(aura::client::GetFocusClient(root_windows[0]), | 
 |             aura::client::GetFocusClient(root_windows[1])); | 
 |   EXPECT_EQ(::wm::GetActivationClient(root_windows[0]), | 
 |             ::wm::GetActivationClient(root_windows[1])); | 
 |   EXPECT_EQ(aura::client::GetCaptureClient(root_windows[0]), | 
 |             aura::client::GetCaptureClient(root_windows[1])); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, Activation) { | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   views::Widget* widget_on_1st = CreateTestWidget(gfx::Rect(10, 10, 100, 100)); | 
 |   views::Widget* widget_on_2nd = | 
 |       CreateTestWidget(gfx::Rect(1200, 10, 100, 100)); | 
 |   EXPECT_EQ(root_windows[0], widget_on_1st->GetNativeView()->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[1], widget_on_2nd->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   EXPECT_EQ(widget_on_2nd->GetNativeView(), | 
 |             aura::client::GetFocusClient(root_windows[0])->GetFocusedWindow()); | 
 |   EXPECT_TRUE(wm::IsActiveWindow(widget_on_2nd->GetNativeView())); | 
 |  | 
 |   ui::test::EventGenerator* event_generator = GetEventGenerator(); | 
 |   // Clicking a window changes the active window and active root window. | 
 |   event_generator->MoveMouseToCenterOf(widget_on_1st->GetNativeView()); | 
 |   event_generator->ClickLeftButton(); | 
 |  | 
 |   EXPECT_EQ(widget_on_1st->GetNativeView(), | 
 |             aura::client::GetFocusClient(root_windows[0])->GetFocusedWindow()); | 
 |   EXPECT_TRUE(wm::IsActiveWindow(widget_on_1st->GetNativeView())); | 
 |  | 
 |   event_generator->MoveMouseToCenterOf(widget_on_2nd->GetNativeView()); | 
 |   event_generator->ClickLeftButton(); | 
 |  | 
 |   EXPECT_EQ(widget_on_2nd->GetNativeView(), | 
 |             aura::client::GetFocusClient(root_windows[0])->GetFocusedWindow()); | 
 |   EXPECT_TRUE(wm::IsActiveWindow(widget_on_2nd->GetNativeView())); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, SystemModal) { | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   views::Widget* widget_on_1st = CreateTestWidget(gfx::Rect(10, 10, 100, 100)); | 
 |   EXPECT_TRUE(wm::IsActiveWindow(widget_on_1st->GetNativeView())); | 
 |   EXPECT_EQ(root_windows[0], widget_on_1st->GetNativeView()->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[0], Shell::GetRootWindowForNewWindows()); | 
 |  | 
 |   // Open system modal. Make sure it's on 2nd root window and active. | 
 |   views::Widget* modal_widget = views::Widget::CreateWindowWithContextAndBounds( | 
 |       new ModalWidgetDelegate(), CurrentContext(), | 
 |       gfx::Rect(1200, 100, 100, 100)); | 
 |   modal_widget->Show(); | 
 |   EXPECT_TRUE(wm::IsActiveWindow(modal_widget->GetNativeView())); | 
 |   EXPECT_EQ(root_windows[1], modal_widget->GetNativeView()->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[1], Shell::GetRootWindowForNewWindows()); | 
 |  | 
 |   ui::test::EventGenerator* event_generator = GetEventGenerator(); | 
 |  | 
 |   // Clicking a widget on widget_on_1st display should not change activation. | 
 |   event_generator->MoveMouseToCenterOf(widget_on_1st->GetNativeView()); | 
 |   event_generator->ClickLeftButton(); | 
 |   EXPECT_TRUE(wm::IsActiveWindow(modal_widget->GetNativeView())); | 
 |   EXPECT_EQ(root_windows[1], Shell::GetRootWindowForNewWindows()); | 
 |  | 
 |   // Close system modal and so clicking a widget should work now. | 
 |   modal_widget->Close(); | 
 |   event_generator->MoveMouseToCenterOf(widget_on_1st->GetNativeView()); | 
 |   event_generator->ClickLeftButton(); | 
 |   EXPECT_TRUE(wm::IsActiveWindow(widget_on_1st->GetNativeView())); | 
 |   EXPECT_EQ(root_windows[0], Shell::GetRootWindowForNewWindows()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, TestCursor) { | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |   aura::WindowTreeHost* host0 = root_windows[0]->GetHost(); | 
 |   aura::WindowTreeHost* host1 = root_windows[1]->GetHost(); | 
 |   EXPECT_EQ(ui::CursorType::kPointer, host0->last_cursor().native_type()); | 
 |   EXPECT_EQ(ui::CursorType::kNull, host1->last_cursor().native_type()); | 
 |   Shell::Get()->cursor_manager()->SetCursor(ui::CursorType::kCopy); | 
 |   EXPECT_EQ(ui::CursorType::kCopy, host0->last_cursor().native_type()); | 
 |   EXPECT_EQ(ui::CursorType::kCopy, host1->last_cursor().native_type()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, TestCursorLocation) { | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |   aura::test::WindowTestApi root_window0_test_api(root_windows[0]); | 
 |   aura::test::WindowTestApi root_window1_test_api(root_windows[1]); | 
 |  | 
 |   root_windows[0]->MoveCursorTo(gfx::Point(10, 10)); | 
 |   EXPECT_EQ("10,10", | 
 |             display::Screen::GetScreen()->GetCursorScreenPoint().ToString()); | 
 |   EXPECT_TRUE(root_window0_test_api.ContainsMouse()); | 
 |   EXPECT_FALSE(root_window1_test_api.ContainsMouse()); | 
 |   root_windows[1]->MoveCursorTo(gfx::Point(10, 20)); | 
 |   EXPECT_EQ("1010,20", | 
 |             display::Screen::GetScreen()->GetCursorScreenPoint().ToString()); | 
 |   EXPECT_FALSE(root_window0_test_api.ContainsMouse()); | 
 |   EXPECT_TRUE(root_window1_test_api.ContainsMouse()); | 
 |   root_windows[0]->MoveCursorTo(gfx::Point(20, 10)); | 
 |   EXPECT_EQ("20,10", | 
 |             display::Screen::GetScreen()->GetCursorScreenPoint().ToString()); | 
 |   EXPECT_TRUE(root_window0_test_api.ContainsMouse()); | 
 |   EXPECT_FALSE(root_window1_test_api.ContainsMouse()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, GetRootWindowAt) { | 
 |   UpdateDisplay("700x500,500x500"); | 
 |   SetSecondaryDisplayLayout(display::DisplayPlacement::LEFT); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   using window_util::GetRootWindowAt; | 
 |   EXPECT_EQ(root_windows[1], GetRootWindowAt(gfx::Point(-400, 100))); | 
 |   EXPECT_EQ(root_windows[1], GetRootWindowAt(gfx::Point(-1, 100))); | 
 |   EXPECT_EQ(root_windows[0], GetRootWindowAt(gfx::Point(0, 300))); | 
 |   EXPECT_EQ(root_windows[0], GetRootWindowAt(gfx::Point(700, 300))); | 
 |  | 
 |   // Zero origin. | 
 |   EXPECT_EQ(root_windows[0], GetRootWindowAt(gfx::Point(0, 0))); | 
 |  | 
 |   // Out of range point should return the nearest root window | 
 |   EXPECT_EQ(root_windows[1], GetRootWindowAt(gfx::Point(-600, 0))); | 
 |   EXPECT_EQ(root_windows[0], GetRootWindowAt(gfx::Point(701, 100))); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, GetRootWindowMatching) { | 
 |   UpdateDisplay("700x500,500x500"); | 
 |   SetSecondaryDisplayLayout(display::DisplayPlacement::LEFT); | 
 |  | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   // Containing rect. | 
 |   using window_util::GetRootWindowMatching; | 
 |   EXPECT_EQ(root_windows[1], | 
 |             GetRootWindowMatching(gfx::Rect(-300, 10, 50, 50))); | 
 |   EXPECT_EQ(root_windows[0], GetRootWindowMatching(gfx::Rect(100, 10, 50, 50))); | 
 |  | 
 |   // Intersecting rect. | 
 |   EXPECT_EQ(root_windows[1], | 
 |             GetRootWindowMatching(gfx::Rect(-200, 0, 300, 300))); | 
 |   EXPECT_EQ(root_windows[0], | 
 |             GetRootWindowMatching(gfx::Rect(-100, 0, 300, 300))); | 
 |  | 
 |   // Zero origin. | 
 |   EXPECT_EQ(root_windows[0], GetRootWindowMatching(gfx::Rect(0, 0, 0, 0))); | 
 |   EXPECT_EQ(root_windows[0], GetRootWindowMatching(gfx::Rect(0, 0, 1, 1))); | 
 |  | 
 |   // Empty rect. | 
 |   EXPECT_EQ(root_windows[1], GetRootWindowMatching(gfx::Rect(-400, 100, 0, 0))); | 
 |   EXPECT_EQ(root_windows[0], GetRootWindowMatching(gfx::Rect(100, 100, 0, 0))); | 
 |  | 
 |   // Out of range rect should return the primary root window. | 
 |   EXPECT_EQ(root_windows[0], | 
 |             GetRootWindowMatching(gfx::Rect(-600, -300, 50, 50))); | 
 |   EXPECT_EQ(root_windows[0], GetRootWindowMatching(gfx::Rect(0, 1000, 50, 50))); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, Capture) { | 
 |   // This test deals with input events but not visuals so don't throttle input | 
 |   // on visuals. | 
 |   aura::Env::GetInstance()->set_throttle_input_on_resize_for_testing(false); | 
 |  | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   aura::test::EventCountDelegate r1_d1; | 
 |   aura::test::EventCountDelegate r1_d2; | 
 |   aura::test::EventCountDelegate r2_d1; | 
 |  | 
 |   std::unique_ptr<aura::Window> r1_w1(aura::test::CreateTestWindowWithDelegate( | 
 |       &r1_d1, 0, gfx::Rect(10, 10, 100, 100), root_windows[0])); | 
 |   std::unique_ptr<aura::Window> r1_w2(aura::test::CreateTestWindowWithDelegate( | 
 |       &r1_d2, 0, gfx::Rect(10, 100, 100, 100), root_windows[0])); | 
 |   std::unique_ptr<aura::Window> r2_w1(aura::test::CreateTestWindowWithDelegate( | 
 |       &r2_d1, 0, gfx::Rect(10, 10, 100, 100), root_windows[1])); | 
 |  | 
 |   r1_w1->SetCapture(); | 
 |  | 
 |   EXPECT_EQ(r1_w1.get(), | 
 |             aura::client::GetCaptureWindow(r2_w1->GetRootWindow())); | 
 |  | 
 |   ui::test::EventGenerator* generator = GetEventGenerator(); | 
 |   generator->MoveMouseToCenterOf(r2_w1.get()); | 
 |   // |r1_w1| will receive the events because it has capture. | 
 |   EXPECT_EQ("1 1 0", r1_d1.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("0 0 0", r1_d2.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("0 0 0", r2_d1.GetMouseMotionCountsAndReset()); | 
 |  | 
 |   generator->ClickLeftButton(); | 
 |   EXPECT_EQ("0 0 0", r2_d1.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("0 0", r2_d1.GetMouseButtonCountsAndReset()); | 
 |   // The mouse is outside. On chromeos, the mouse is warped to the | 
 |   // dest root window, but it's not implemented on Win yet, so | 
 |   // no mouse move event on Win. | 
 |   EXPECT_EQ("0 0 0", r1_d1.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("1 1", r1_d1.GetMouseButtonCountsAndReset()); | 
 |  | 
 |   generator->MoveMouseTo(15, 15); | 
 |   EXPECT_EQ("0 1 0", r1_d1.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("0 0 0", r1_d2.GetMouseMotionCountsAndReset()); | 
 |  | 
 |   r1_w2->SetCapture(); | 
 |   EXPECT_EQ(r1_w2.get(), | 
 |             aura::client::GetCaptureWindow(r2_w1->GetRootWindow())); | 
 |   generator->MoveMouseBy(10, 10); | 
 |   // |r1_w2| has the capture. So it will receive the mouse-move event. | 
 |   EXPECT_EQ("0 0 0", r1_d1.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("0 1 0", r1_d2.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("0 0 0", r2_d1.GetMouseMotionCountsAndReset()); | 
 |  | 
 |   generator->ClickLeftButton(); | 
 |   EXPECT_EQ("0 0 0", r2_d1.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("0 0", r2_d1.GetMouseButtonCountsAndReset()); | 
 |   EXPECT_EQ("0 0 0", r1_d2.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("1 1", r1_d2.GetMouseButtonCountsAndReset()); | 
 |  | 
 |   r1_w2->ReleaseCapture(); | 
 |   EXPECT_EQ(nullptr, aura::client::GetCaptureWindow(r2_w1->GetRootWindow())); | 
 |  | 
 |   generator->MoveMouseToCenterOf(r2_w1.get()); | 
 |   generator->ClickLeftButton(); | 
 |   EXPECT_EQ("1 1 0", r2_d1.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("1 1", r2_d1.GetMouseButtonCountsAndReset()); | 
 |   // Make sure the mouse_moved_handler_ is properly reset. | 
 |   EXPECT_EQ("0 0 0", r1_d2.GetMouseMotionCountsAndReset()); | 
 |   EXPECT_EQ("0 0", r1_d2.GetMouseButtonCountsAndReset()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, CaptureEventLocation) { | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   aura::test::EventCountDelegate r1_d1; | 
 |   aura::test::EventCountDelegate r1_d2; | 
 |   aura::test::EventCountDelegate r2_d1; | 
 |  | 
 |   std::unique_ptr<aura::Window> r1_w1(aura::test::CreateTestWindowWithDelegate( | 
 |       &r1_d1, 0, gfx::Rect(10, 10, 100, 100), root_windows[0])); | 
 |   std::unique_ptr<aura::Window> r1_w2(aura::test::CreateTestWindowWithDelegate( | 
 |       &r1_d2, 0, gfx::Rect(10, 100, 100, 100), root_windows[0])); | 
 |   std::unique_ptr<aura::Window> r2_w1(aura::test::CreateTestWindowWithDelegate( | 
 |       &r2_d1, 0, gfx::Rect(10, 10, 100, 100), root_windows[1])); | 
 |  | 
 |   r1_w1->SetCapture(); | 
 |  | 
 |   ui::test::EventGenerator* generator = GetEventGenerator(); | 
 |   generator->MoveMouseToCenterOf(r2_w1.get()); | 
 |   EXPECT_EQ(gfx::Point(1060, 60).ToString(), | 
 |             generator->current_screen_location().ToString()); | 
 |  | 
 |   EventLocationHandler location_handler; | 
 |   r1_w1->AddPreTargetHandler(&location_handler); | 
 |   generator->ClickLeftButton(); | 
 |   r1_w1->RemovePreTargetHandler(&location_handler); | 
 |   EXPECT_EQ(gfx::Point(1050, 50).ToString(), | 
 |             location_handler.press_location().ToString()); | 
 |   EXPECT_EQ(gfx::Point(1050, 50).ToString(), | 
 |             location_handler.release_location().ToString()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, CaptureEventLocationHighDPI) { | 
 |   UpdateDisplay("1000x600*2,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   aura::test::EventCountDelegate r1_d1; | 
 |   aura::test::EventCountDelegate r1_d2; | 
 |   aura::test::EventCountDelegate r2_d1; | 
 |  | 
 |   std::unique_ptr<aura::Window> r1_w1(aura::test::CreateTestWindowWithDelegate( | 
 |       &r1_d1, 0, gfx::Rect(10, 10, 100, 100), root_windows[0])); | 
 |   std::unique_ptr<aura::Window> r1_w2(aura::test::CreateTestWindowWithDelegate( | 
 |       &r1_d2, 0, gfx::Rect(10, 100, 100, 100), root_windows[0])); | 
 |   std::unique_ptr<aura::Window> r2_w1(aura::test::CreateTestWindowWithDelegate( | 
 |       &r2_d1, 0, gfx::Rect(10, 10, 100, 100), root_windows[1])); | 
 |  | 
 |   r1_w1->SetCapture(); | 
 |  | 
 |   ui::test::EventGenerator* generator = GetEventGenerator(); | 
 |   generator->MoveMouseToCenterOf(r2_w1.get()); | 
 |   EXPECT_EQ(gfx::Point(560, 60).ToString(), | 
 |             generator->current_screen_location().ToString()); | 
 |  | 
 |   EventLocationHandler location_handler; | 
 |   r1_w1->AddPreTargetHandler(&location_handler); | 
 |   generator->ClickLeftButton(); | 
 |   r1_w1->RemovePreTargetHandler(&location_handler); | 
 |   EXPECT_EQ(gfx::Point(550, 50).ToString(), | 
 |             location_handler.press_location().ToString()); | 
 |   EXPECT_EQ(gfx::Point(550, 50).ToString(), | 
 |             location_handler.release_location().ToString()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, CaptureEventLocationHighDPI_2) { | 
 |   UpdateDisplay("1000x600,600x400*2"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   aura::test::EventCountDelegate r1_d1; | 
 |   aura::test::EventCountDelegate r1_d2; | 
 |   aura::test::EventCountDelegate r2_d1; | 
 |  | 
 |   std::unique_ptr<aura::Window> r1_w1(aura::test::CreateTestWindowWithDelegate( | 
 |       &r1_d1, 0, gfx::Rect(10, 10, 100, 100), root_windows[0])); | 
 |   std::unique_ptr<aura::Window> r1_w2(aura::test::CreateTestWindowWithDelegate( | 
 |       &r1_d2, 0, gfx::Rect(10, 100, 100, 100), root_windows[0])); | 
 |   std::unique_ptr<aura::Window> r2_w1(aura::test::CreateTestWindowWithDelegate( | 
 |       &r2_d1, 0, gfx::Rect(10, 10, 100, 100), root_windows[1])); | 
 |  | 
 |   r1_w1->SetCapture(); | 
 |  | 
 |   ui::test::EventGenerator* generator = GetEventGenerator(); | 
 |   generator->MoveMouseToCenterOf(r2_w1.get()); | 
 |   EXPECT_EQ(gfx::Point(1060, 60).ToString(), | 
 |             generator->current_screen_location().ToString()); | 
 |  | 
 |   EventLocationHandler location_handler; | 
 |   r1_w1->AddPreTargetHandler(&location_handler); | 
 |   generator->ClickLeftButton(); | 
 |   r1_w1->RemovePreTargetHandler(&location_handler); | 
 |   // Event-generator dispatches the event in the primary root-window's coord | 
 |   // space. Since the location is (1060, 60), it goes to the secondary | 
 |   // root-window as (30, 30) since the secondary root-window has a device scale | 
 |   // factor of 2. | 
 |   EXPECT_EQ(gfx::Point(1020, 20).ToString(), | 
 |             location_handler.press_location().ToString()); | 
 |   EXPECT_EQ(gfx::Point(1020, 20).ToString(), | 
 |             location_handler.release_location().ToString()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, MoveWindow) { | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |   views::Widget* d1 = CreateTestWidget(gfx::Rect(10, 10, 100, 100)); | 
 |  | 
 |   EXPECT_EQ(root_windows[0], d1->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   d1->SetBounds(gfx::Rect(1010, 10, 100, 100)); | 
 |   EXPECT_EQ("1010,10 100x100", d1->GetWindowBoundsInScreen().ToString()); | 
 |  | 
 |   EXPECT_EQ(root_windows[1], d1->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   d1->SetBounds(gfx::Rect(10, 10, 100, 100)); | 
 |   EXPECT_EQ("10,10 100x100", d1->GetWindowBoundsInScreen().ToString()); | 
 |  | 
 |   EXPECT_EQ(root_windows[0], d1->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   // Make sure the bounds which doesn't fit to the root window | 
 |   // works correctly. | 
 |   d1->SetBounds(gfx::Rect(1560, 30, 100, 100)); | 
 |   EXPECT_EQ(root_windows[1], d1->GetNativeView()->GetRootWindow()); | 
 |   EXPECT_EQ("1560,30 100x100", d1->GetWindowBoundsInScreen().ToString()); | 
 |  | 
 |   // Setting outside of root windows will be moved to primary root window. | 
 |   // TODO(oshima): This one probably should pick the closest root window. | 
 |   d1->SetBounds(gfx::Rect(200, 10, 100, 100)); | 
 |   EXPECT_EQ(root_windows[0], d1->GetNativeView()->GetRootWindow()); | 
 | } | 
 |  | 
 | // Verifies if the mouse event arrives to the window even when the window | 
 | // moves to another root in a pre-target handler.  See: crbug.com/157583 | 
 | TEST_F(ExtendedDesktopTest, MoveWindowByMouseClick) { | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |  | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |   aura::test::EventCountDelegate delegate; | 
 |   std::unique_ptr<aura::Window> window(aura::test::CreateTestWindowWithDelegate( | 
 |       &delegate, 0, gfx::Rect(10, 10, 100, 100), root_windows[0])); | 
 |   MoveWindowByClickEventHandler event_handler(window.get()); | 
 |   window->AddPreTargetHandler(&event_handler); | 
 |  | 
 |   ui::test::EventGenerator* event_generator = GetEventGenerator(); | 
 |  | 
 |   event_generator->MoveMouseToCenterOf(window.get()); | 
 |   event_generator->ClickLeftButton(); | 
 |   // Both mouse pressed and released arrive at the window and its delegate. | 
 |   EXPECT_EQ("1 1", delegate.GetMouseButtonCountsAndReset()); | 
 |   // Also event_handler moves the window to another root at mouse release. | 
 |   EXPECT_EQ(root_windows[1], window->GetRootWindow()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, MoveWindowToDisplay) { | 
 |   UpdateDisplay("1000x1000,1000x1000"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   display::Display display0 = display::Screen::GetScreen()->GetDisplayMatching( | 
 |       root_windows[0]->GetBoundsInScreen()); | 
 |   display::Display display1 = display::Screen::GetScreen()->GetDisplayMatching( | 
 |       root_windows[1]->GetBoundsInScreen()); | 
 |   EXPECT_NE(display0.id(), display1.id()); | 
 |  | 
 |   views::Widget* d1 = CreateTestWidget(gfx::Rect(10, 10, 1000, 100)); | 
 |   EXPECT_EQ(root_windows[0], d1->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   // Move the window where the window spans both root windows. Since the second | 
 |   // parameter is |display1|, the window should be shown on the secondary root. | 
 |   d1->GetNativeWindow()->SetBoundsInScreen(gfx::Rect(500, 10, 1000, 100), | 
 |                                            display1); | 
 |   EXPECT_EQ("500,10 1000x100", d1->GetWindowBoundsInScreen().ToString()); | 
 |   EXPECT_EQ(root_windows[1], d1->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   // Move to the primary root. | 
 |   d1->GetNativeWindow()->SetBoundsInScreen(gfx::Rect(500, 10, 1000, 100), | 
 |                                            display0); | 
 |   EXPECT_EQ("500,10 1000x100", d1->GetWindowBoundsInScreen().ToString()); | 
 |   EXPECT_EQ(root_windows[0], d1->GetNativeView()->GetRootWindow()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, MoveWindowWithTransient) { | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |   // Create and activate a normal window |w1|. | 
 |   aura::Window* w1 = | 
 |       CreateTestWindowInShellWithBounds(gfx::Rect(10, 10, 100, 100)); | 
 |   wm::ActivateWindow(w1); | 
 |   // |w1_t1| is a transient child window of |w1|. | 
 |   std::unique_ptr<aura::Window> w1_t1 = CreateChildWindow( | 
 |       w1, gfx::Rect(50, 50, 50, 50), desks_util::GetActiveDeskContainerId()); | 
 |   ::wm::AddTransientChild(w1, w1_t1.get()); | 
 |   // |w1_t11| is a transient child window of transient child window |w1_t1|. | 
 |   std::unique_ptr<aura::Window> w1_t11 = | 
 |       CreateChildWindow(w1_t1.get(), gfx::Rect(2, 7, 35, 35), | 
 |                         desks_util::GetActiveDeskContainerId()); | 
 |   ::wm::AddTransientChild(w1_t1.get(), w1_t11.get()); | 
 |  | 
 |   // |w11| is a non-transient child window of |w1|. | 
 |   std::unique_ptr<aura::Window> w11 = CreateChildWindow( | 
 |       w1, gfx::Rect(10, 10, 40, 40), desks_util::GetActiveDeskContainerId()); | 
 |   // |w11_t1| is a transient child window of |w11|. | 
 |   std::unique_ptr<aura::Window> w11_t1 = | 
 |       CreateChildWindow(w11.get(), gfx::Rect(30, 10, 80, 80), | 
 |                         desks_util::GetActiveDeskContainerId()); | 
 |   ::wm::AddTransientChild(w11.get(), w11_t1.get()); | 
 |  | 
 |   EXPECT_EQ(root_windows[0], w1->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[0], w1_t1->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[0], w1_t11->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[0], w11->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[0], w11_t1->GetRootWindow()); | 
 |   EXPECT_EQ("10,10 100x100", w1->GetBoundsInScreen().ToString()); | 
 |   EXPECT_EQ("60,60 50x50", w1_t1->GetBoundsInScreen().ToString()); | 
 |   EXPECT_EQ("62,67 35x35", w1_t11->GetBoundsInScreen().ToString()); | 
 |   EXPECT_EQ("20,20 40x40", w11->GetBoundsInScreen().ToString()); | 
 |   EXPECT_EQ("50,30 80x80", w11_t1->GetBoundsInScreen().ToString()); | 
 |  | 
 |   w1->SetBoundsInScreen(gfx::Rect(1200, 10, 100, 100), GetSecondaryDisplay()); | 
 |  | 
 |   EXPECT_EQ(root_windows[1], w1->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[1], w1_t1->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[1], w1_t11->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[1], w11->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[1], w11_t1->GetRootWindow()); | 
 |  | 
 |   EXPECT_EQ("1200,10 100x100", w1->GetBoundsInScreen().ToString()); | 
 |   EXPECT_EQ("1250,60 50x50", w1_t1->GetBoundsInScreen().ToString()); | 
 |   EXPECT_EQ("1252,67 35x35", w1_t11->GetBoundsInScreen().ToString()); | 
 |   EXPECT_EQ("1210,20 40x40", w11->GetBoundsInScreen().ToString()); | 
 |   EXPECT_EQ("1240,30 80x80", w11_t1->GetBoundsInScreen().ToString()); | 
 | } | 
 |  | 
 | // Test transient child is parented after its transient parent moved to another | 
 | // root window. | 
 | TEST_F(ExtendedDesktopTest, PostMoveParentTransientChild) { | 
 |   UpdateDisplay("600X400,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |   // Create and activate a normal window. | 
 |   aura::Window* window = | 
 |       CreateTestWindowInShellWithBounds(gfx::Rect(10, 10, 100, 100)); | 
 |   wm::ActivateWindow(window); | 
 |   // Create a transient child window of |window| without parenting to |window| | 
 |   // yet. | 
 |   std::unique_ptr<aura::Window> child = window_factory::NewWindow(); | 
 |   child->SetType(aura::client::WINDOW_TYPE_NORMAL); | 
 |   child->Init(ui::LAYER_TEXTURED); | 
 |   child->SetBounds(gfx::Rect(50, 50, 50, 50)); | 
 |   child->Show(); | 
 |   ::wm::AddTransientChild(window, child.get()); | 
 |  | 
 |   // Move |window| to another display and ensure that crash doesn't happen. | 
 |   window->SetBoundsInScreen(gfx::Rect(610, 10, 100, 100), | 
 |                             GetSecondaryDisplay()); | 
 |  | 
 |   // Parent |child| to |window| now. | 
 |   window->AddChild(child.get()); | 
 |  | 
 |   EXPECT_EQ(root_windows[1], window->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[1], child->GetRootWindow()); | 
 |   EXPECT_EQ("610,10 100x100", window->GetBoundsInScreen().ToString()); | 
 |   EXPECT_EQ("660,60 50x50", child->GetBoundsInScreen().ToString()); | 
 | } | 
 |  | 
 | // Test if the Window::ConvertPointToTarget works across root windows. | 
 | TEST_F(ExtendedDesktopTest, ConvertPoint) { | 
 |   display::Screen* screen = display::Screen::GetScreen(); | 
 |   UpdateDisplay("1000x600,600x400"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |   display::Display display_1 = screen->GetDisplayNearestWindow(root_windows[0]); | 
 |   EXPECT_EQ("0,0", display_1.bounds().origin().ToString()); | 
 |   display::Display display_2 = screen->GetDisplayNearestWindow(root_windows[1]); | 
 |   EXPECT_EQ("1000,0", display_2.bounds().origin().ToString()); | 
 |  | 
 |   aura::Window* d1 = | 
 |       CreateTestWidget(gfx::Rect(10, 10, 100, 100))->GetNativeView(); | 
 |   aura::Window* d2 = | 
 |       CreateTestWidget(gfx::Rect(1020, 20, 100, 100))->GetNativeView(); | 
 |   EXPECT_EQ(root_windows[0], d1->GetRootWindow()); | 
 |   EXPECT_EQ(root_windows[1], d2->GetRootWindow()); | 
 |  | 
 |   // Convert point in the Root2's window to the Root1's window Coord. | 
 |   gfx::Point p(0, 0); | 
 |   aura::Window::ConvertPointToTarget(root_windows[1], root_windows[0], &p); | 
 |   EXPECT_EQ("1000,0", p.ToString()); | 
 |   p.SetPoint(0, 0); | 
 |   aura::Window::ConvertPointToTarget(d2, d1, &p); | 
 |   EXPECT_EQ("1010,10", p.ToString()); | 
 |  | 
 |   // Convert point in the Root1's window to the Root2's window Coord. | 
 |   p.SetPoint(0, 0); | 
 |   aura::Window::ConvertPointToTarget(root_windows[0], root_windows[1], &p); | 
 |   EXPECT_EQ("-1000,0", p.ToString()); | 
 |   p.SetPoint(0, 0); | 
 |   aura::Window::ConvertPointToTarget(d1, d2, &p); | 
 |   EXPECT_EQ("-1010,-10", p.ToString()); | 
 |  | 
 |   // Move the 2nd display to the bottom and test again. | 
 |   SetSecondaryDisplayLayout(display::DisplayPlacement::BOTTOM); | 
 |  | 
 |   display_2 = screen->GetDisplayNearestWindow(root_windows[1]); | 
 |   EXPECT_EQ("0,600", display_2.bounds().origin().ToString()); | 
 |  | 
 |   // Convert point in Root2's window to Root1's window Coord. | 
 |   p.SetPoint(0, 0); | 
 |   aura::Window::ConvertPointToTarget(root_windows[1], root_windows[0], &p); | 
 |   EXPECT_EQ("0,600", p.ToString()); | 
 |   p.SetPoint(0, 0); | 
 |   aura::Window::ConvertPointToTarget(d2, d1, &p); | 
 |   EXPECT_EQ("10,610", p.ToString()); | 
 |  | 
 |   // Convert point in Root1's window to Root2's window Coord. | 
 |   p.SetPoint(0, 0); | 
 |   aura::Window::ConvertPointToTarget(root_windows[0], root_windows[1], &p); | 
 |   EXPECT_EQ("0,-600", p.ToString()); | 
 |   p.SetPoint(0, 0); | 
 |   aura::Window::ConvertPointToTarget(d1, d2, &p); | 
 |   EXPECT_EQ("-10,-610", p.ToString()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, OpenSystemTray) { | 
 |   // Two displays side by side and both are high enough for tray bubble. | 
 |   UpdateDisplay("500x600,600x700"); | 
 |   ASSERT_FALSE(IsBubbleShown()); | 
 |  | 
 |   ui::test::EventGenerator* event_generator = GetEventGenerator(); | 
 |  | 
 |   // Opens the tray by a dummy click event and makes sure that adding/removing | 
 |   // displays doesn't break anything. | 
 |   event_generator->MoveMouseTo(GetSystemTrayBoundsInScreen().CenterPoint()); | 
 |   event_generator->ClickLeftButton(); | 
 |   EXPECT_TRUE(IsBubbleShown()); | 
 |  | 
 |   // Verifies that the bubble is within the primary display. | 
 |   const gfx::Rect primary_display_bounds = GetPrimaryDisplay().bounds(); | 
 |   const gfx::Rect tray_bubble_bounds = | 
 |       GetPrimaryUnifiedSystemTray()->GetBubbleBoundsInScreen(); | 
 |   EXPECT_TRUE(primary_display_bounds.Contains(tray_bubble_bounds)) | 
 |       << "primary display bounds=" << primary_display_bounds.ToString() | 
 |       << ", tray bubble bounds=" << tray_bubble_bounds.ToString(); | 
 |  | 
 |   UpdateDisplay("500x600"); | 
 |   EXPECT_TRUE(IsBubbleShown()); | 
 |   UpdateDisplay("500x600,600x700"); | 
 |   EXPECT_TRUE(IsBubbleShown()); | 
 |  | 
 |   // Closes the tray and again makes sure that adding/removing displays doesn't | 
 |   // break anything. | 
 |   event_generator->ClickLeftButton(); | 
 |   base::RunLoop().RunUntilIdle(); | 
 |  | 
 |   EXPECT_FALSE(IsBubbleShown()); | 
 |  | 
 |   UpdateDisplay("500x600"); | 
 |   EXPECT_FALSE(IsBubbleShown()); | 
 |   UpdateDisplay("500x600,600x700"); | 
 |   EXPECT_FALSE(IsBubbleShown()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, StayInSameRootWindow) { | 
 |   UpdateDisplay("100x100,200x200"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |   views::Widget* w1 = CreateTestWidget(gfx::Rect(10, 10, 50, 50)); | 
 |   EXPECT_EQ(root_windows[0], w1->GetNativeView()->GetRootWindow()); | 
 |   w1->SetBounds(gfx::Rect(150, 10, 50, 50)); | 
 |   EXPECT_EQ(root_windows[1], w1->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   // The widget stays in the same root if kLockedToRootKey is set to true. | 
 |   w1->GetNativeView()->SetProperty(kLockedToRootKey, true); | 
 |   w1->SetBounds(gfx::Rect(10, 10, 50, 50)); | 
 |   EXPECT_EQ(root_windows[1], w1->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   // The widget should now move to the 1st root window without the property. | 
 |   w1->GetNativeView()->ClearProperty(kLockedToRootKey); | 
 |   w1->SetBounds(gfx::Rect(10, 10, 50, 50)); | 
 |   EXPECT_EQ(root_windows[0], w1->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   // a window in SettingsBubbleContainer and StatusContainer should | 
 |   // not move to another root window regardles of the bounds specified. | 
 |   aura::Window* settings_bubble_container = | 
 |       Shell::GetPrimaryRootWindowController()->GetContainer( | 
 |           kShellWindowId_SettingBubbleContainer); | 
 |   aura::Window* window = | 
 |       aura::test::CreateTestWindowWithId(100, settings_bubble_container); | 
 |   window->SetBoundsInScreen(gfx::Rect(150, 10, 50, 50), GetSecondaryDisplay()); | 
 |   EXPECT_EQ(root_windows[0], window->GetRootWindow()); | 
 |  | 
 |   aura::Window* status_container = | 
 |       Shell::GetPrimaryRootWindowController()->GetContainer( | 
 |           kShellWindowId_StatusContainer); | 
 |   window = aura::test::CreateTestWindowWithId(100, status_container); | 
 |   window->SetBoundsInScreen(gfx::Rect(150, 10, 50, 50), GetSecondaryDisplay()); | 
 |   EXPECT_EQ(root_windows[0], window->GetRootWindow()); | 
 | } | 
 |  | 
 | TEST_F(ExtendedDesktopTest, KeyEventsOnLockScreen) { | 
 |   UpdateDisplay("100x100,200x200"); | 
 |   aura::Window::Windows root_windows = Shell::GetAllRootWindows(); | 
 |  | 
 |   // Create normal windows on both displays. | 
 |   views::Widget* widget1 = CreateTestWidget( | 
 |       display::Screen::GetScreen()->GetPrimaryDisplay().bounds()); | 
 |   widget1->Show(); | 
 |   EXPECT_EQ(root_windows[0], widget1->GetNativeView()->GetRootWindow()); | 
 |   views::Widget* widget2 = CreateTestWidget(GetSecondaryDisplay().bounds()); | 
 |   widget2->Show(); | 
 |   EXPECT_EQ(root_windows[1], widget2->GetNativeView()->GetRootWindow()); | 
 |  | 
 |   // Create a LockScreen window. | 
 |   views::Widget* lock_widget = CreateTestWidget( | 
 |       display::Screen::GetScreen()->GetPrimaryDisplay().bounds()); | 
 |   views::Textfield* textfield = new views::Textfield; | 
 |   lock_widget->client_view()->AddChildView(textfield); | 
 |  | 
 |   ash::Shell::GetContainer(Shell::GetPrimaryRootWindow(), | 
 |                            ash::kShellWindowId_LockScreenContainer) | 
 |       ->AddChild(lock_widget->GetNativeView()); | 
 |   lock_widget->Show(); | 
 |   textfield->RequestFocus(); | 
 |  | 
 |   aura::client::FocusClient* focus_client = | 
 |       aura::client::GetFocusClient(root_windows[0]); | 
 |   EXPECT_EQ(lock_widget->GetNativeView(), focus_client->GetFocusedWindow()); | 
 |  | 
 |   // The lock window should get events on both root windows. | 
 |   ui::test::EventGenerator* event_generator = GetEventGenerator(); | 
 |  | 
 |   event_generator->set_current_target(root_windows[0]); | 
 |   event_generator->PressKey(ui::VKEY_A, 0); | 
 |   event_generator->ReleaseKey(ui::VKEY_A, 0); | 
 |   EXPECT_EQ(lock_widget->GetNativeView(), focus_client->GetFocusedWindow()); | 
 |   EXPECT_EQ("a", base::UTF16ToASCII(textfield->GetText())); | 
 |  | 
 |   event_generator->set_current_target(root_windows[1]); | 
 |   event_generator->PressKey(ui::VKEY_B, 0); | 
 |   event_generator->ReleaseKey(ui::VKEY_B, 0); | 
 |   EXPECT_EQ(lock_widget->GetNativeView(), focus_client->GetFocusedWindow()); | 
 |   EXPECT_EQ("ab", base::UTF16ToASCII(textfield->GetText())); | 
 |  | 
 |   // Deleting 2nd display. The lock window still should get the events. | 
 |   UpdateDisplay("100x100"); | 
 |   event_generator->set_current_target(root_windows[0]); | 
 |   event_generator->PressKey(ui::VKEY_C, 0); | 
 |   event_generator->ReleaseKey(ui::VKEY_C, 0); | 
 |   EXPECT_EQ(lock_widget->GetNativeView(), focus_client->GetFocusedWindow()); | 
 |   EXPECT_EQ("abc", base::UTF16ToASCII(textfield->GetText())); | 
 |  | 
 |   // Creating 2nd display again, and lock window still should get events | 
 |   // on both root windows. | 
 |   UpdateDisplay("100x100,200x200"); | 
 |   root_windows = Shell::GetAllRootWindows(); | 
 |   event_generator->set_current_target(root_windows[0]); | 
 |   event_generator->PressKey(ui::VKEY_D, 0); | 
 |   event_generator->ReleaseKey(ui::VKEY_D, 0); | 
 |   EXPECT_EQ(lock_widget->GetNativeView(), focus_client->GetFocusedWindow()); | 
 |   EXPECT_EQ("abcd", base::UTF16ToASCII(textfield->GetText())); | 
 |  | 
 |   event_generator->set_current_target(root_windows[1]); | 
 |   event_generator->PressKey(ui::VKEY_E, 0); | 
 |   event_generator->ReleaseKey(ui::VKEY_E, 0); | 
 |   EXPECT_EQ(lock_widget->GetNativeView(), focus_client->GetFocusedWindow()); | 
 |   EXPECT_EQ("abcde", base::UTF16ToASCII(textfield->GetText())); | 
 | } | 
 |  | 
 | // Verifies that clicking in the primary display and dragging to the secondary | 
 | // display correctly sets the event location (window local) and root_location | 
 | // (in screen coordinates) for the mouse event. https://crrev.com/11691010 | 
 | TEST_F(ExtendedDesktopTest, PassiveGrab) { | 
 |   // This test deals with input events but not visuals so don't throttle input | 
 |   // on visuals. | 
 |   aura::Env::GetInstance()->set_throttle_input_on_resize_for_testing(false); | 
 |   EventLocationRecordingEventHandler event_handler; | 
 |   Shell::Get()->AddPreTargetHandler(&event_handler); | 
 |  | 
 |   // Create large displays so the widget won't be moved after creation. | 
 |   // https://crbug.com/890633 | 
 |   UpdateDisplay("1000x1000,1000x1000"); | 
 |  | 
 |   views::Widget* widget = CreateTestWidget(gfx::Rect(50, 50, 200, 200)); | 
 |   widget->Show(); | 
 |   ASSERT_EQ("50,50 200x200", widget->GetWindowBoundsInScreen().ToString()); | 
 |  | 
 |   // Move the mouse to the center of the window. | 
 |   ui::test::EventGenerator* generator = GetEventGenerator(); | 
 |   generator->MoveMouseTo(150, 150); | 
 |  | 
 |   // Location is relative to the window. | 
 |   EXPECT_EQ("100,100", event_handler.GetLocation()); | 
 |  | 
 |   // Root location is relative to the display. | 
 |   EXPECT_EQ("150,150", event_handler.GetRootLocation()); | 
 |   event_handler.Reset(); | 
 |  | 
 |   // Drag the mouse to the secondary display. Does not drag the window. | 
 |   generator->PressLeftButton(); | 
 |   generator->MoveMouseBy(1000, 0); | 
 |  | 
 |   // Location is relative to the original window. | 
 |   EXPECT_EQ("1100,100", event_handler.GetLocation()); | 
 |  | 
 |   // Root location is relative to the original display. | 
 |   EXPECT_EQ("1150,150", event_handler.GetRootLocation()); | 
 |   event_handler.Reset(); | 
 |  | 
 |   // End the drag. | 
 |   generator->ReleaseLeftButton(); | 
 |   EXPECT_EQ("-999,-999", event_handler.GetRootLocation()); | 
 |   EXPECT_EQ("-999,-999", event_handler.GetLocation()); | 
 |   event_handler.Reset(); | 
 |  | 
 |   // Move the mouse to the top-left of the secondary display. | 
 |   generator->MoveMouseTo(1001, 1); | 
 |  | 
 |   // Location is relative to the new display. | 
 |   EXPECT_EQ("1,1", event_handler.GetLocation()); | 
 |  | 
 |   // Root location is relative to the new display. | 
 |   EXPECT_EQ("1,1", event_handler.GetRootLocation()); | 
 |   event_handler.Reset(); | 
 |  | 
 |   Shell::Get()->RemovePreTargetHandler(&event_handler); | 
 | } | 
 |  | 
 | }  // namespace ash |