chromium/chrome/browser/ui/window_sizer/window_sizer_chromeos_unittest.cc

// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "chrome/browser/ui/window_sizer/window_sizer_chromeos.h"

#include "ash/public/cpp/window_properties.h"
#include "ash/root_window_controller.h"
#include "ash/shelf/shelf.h"
#include "ash/shell.h"
#include "base/command_line.h"
#include "base/memory/ptr_util.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/window_sizer/window_sizer_common_unittest.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/test/base/chrome_ash_test_base.h"
#include "chrome/test/base/test_browser_window_aura.h"
#include "chrome/test/base/testing_profile.h"
#include "content/public/test/render_view_test.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/aura/client/aura_constants.h"
#include "ui/aura/env.h"
#include "ui/aura/test/test_windows.h"
#include "ui/aura/window_event_dispatcher.h"
#include "ui/display/display.h"
#include "ui/display/manager/display_manager.h"
#include "ui/display/screen.h"
#include "ui/display/test/display_manager_test_api.h"
#include "ui/wm/public/activation_client.h"

class WindowSizerChromeOSTest : public ChromeAshTestBase {
 public:
  WindowSizerChromeOSTest() = default;
  WindowSizerChromeOSTest(const WindowSizerChromeOSTest&) = delete;
  WindowSizerChromeOSTest& operator=(const WindowSizerChromeOSTest&) = delete;
  ~WindowSizerChromeOSTest() override = default;

  // The window sizing code only works when the window hasn't yet been created.
  std::unique_ptr<Browser> CreateWindowlessBrowser(
      Browser::CreateParams params) {
    params.skip_window_init_for_testing = true;
    return std::unique_ptr<Browser>(Browser::Create(params));
  }

  // Similar to WindowSizerTestUtil::GetWindowBounds() but takes an existing
  // |display_id| instead of creating a TestScreen and new displays.
  // TODO(mek): Refactor this to use a builder pattern similar to what
  // WindowSizerTestUtil does.
  enum Source { DEFAULT, LAST_ACTIVE, PERSISTED, BOTH };
  void GetWindowBounds(const Browser* browser,
                       const gfx::Rect& passed_in,
                       int64_t display_id,
                       const gfx::Rect& bounds,
                       const gfx::Rect& work_area,
                       Source source,
                       gfx::Rect* out_bounds) {
    auto state_provider = std::make_unique<TestStateProvider>();
    if (source == PERSISTED) {
      state_provider->SetPersistentState(bounds, work_area,
                                         ui::SHOW_STATE_DEFAULT);
    } else {
      DCHECK_EQ(source, DEFAULT);
    }
    display::Screen::GetScreen()->SetDisplayForNewWindows(display_id);

    ui::WindowShowState ignored;
    WindowSizer::GetBrowserWindowBoundsAndShowState(
        std::move(state_provider), passed_in, browser, out_bounds, &ignored);
  }
  void GetWindowBounds(const Browser* browser,
                       const gfx::Rect& passed_in,
                       int64_t display_id,
                       gfx::Rect* out_bounds = nullptr) {
    GetWindowBounds(browser, passed_in, display_id, gfx::Rect(), gfx::Rect(),
                    PERSISTED, out_bounds);
  }

  // Returns browser window |out_bounds| and |out_show_state| for simulated
  // persisted and last-active window bounds, work area, show state, etc.
  static void GetBrowserWindowBoundsAndShowState(
      const gfx::Rect& bounds,
      const gfx::Rect& work_area,
      ui::WindowShowState show_state_persisted,
      ui::WindowShowState show_state_last,
      Source source,
      const Browser* browser,
      const gfx::Rect& passed_in,
      gfx::Rect* out_bounds,
      ui::WindowShowState* out_show_state) {
    DCHECK(out_show_state);
    auto provider = std::make_unique<TestStateProvider>();
    if (source == PERSISTED || source == BOTH)
      provider->SetPersistentState(bounds, work_area, show_state_persisted);
    if (source == LAST_ACTIVE || source == BOTH)
      provider->SetLastActiveState(bounds, show_state_last);

    WindowSizer::GetBrowserWindowBoundsAndShowState(
        std::move(provider), passed_in, browser, out_bounds, out_show_state);
  }

  // Returns browser window show state for simulated persisted and last-active
  // window bounds, work area, show state, etc.
  static ui::WindowShowState GetBrowserWindowShowState(
      ui::WindowShowState show_state_persisted,
      ui::WindowShowState show_state_last,
      Source source,
      const Browser* browser,
      const gfx::Rect& bounds,
      const gfx::Rect& work_area) {
    ui::WindowShowState out_show_state = ui::SHOW_STATE_DEFAULT;
    gfx::Rect ignored;
    GetBrowserWindowBoundsAndShowState(bounds, work_area, show_state_persisted,
                                       show_state_last, source, browser,
                                       gfx::Rect(), &ignored, &out_show_state);
    return out_show_state;
  }

 protected:
  TestingProfile profile_;
};

namespace {

// Shorten identifiers to improve line wrapping.
const int kDesktopBorderSize = WindowSizerChromeOS::kDesktopBorderSize;
const int kMaximumWindowWidth = WindowSizerChromeOS::kMaximumWindowWidth;
const int kWindowTilePixels = WindowSizer::kWindowTilePixels;

std::unique_ptr<Browser> CreateTestBrowser(aura::Window* window,
                                           const gfx::Rect& bounds,
                                           Browser::CreateParams* params) {
  if (!bounds.IsEmpty())
    window->SetBounds(bounds);
  std::unique_ptr<Browser> browser =
      chrome::CreateBrowserWithAuraTestWindowForParams(base::WrapUnique(window),
                                                       params);
  if (browser->is_type_normal()) {
    browser->window()->GetNativeWindow()->SetProperty(
        ash::kWindowPositionManagedTypeKey, true);
  }
  return browser;
}

}  // namespace

// WindowSizerChromeOSNoAshTest are the tests that does not require ash
// environment, thus it can use WindowSizerTestUtil that create a TestScreen
// inside.

// Test that the window is sized appropriately for the first run experience
// where the default window bounds calculation is invoked.
TEST(WindowSizerChromeOSNoAshTest, DefaultSizeCase) {
  {
    // 4:3 monitor case, 1024x768, no taskbar.
    gfx::Rect window_bounds =
        WindowSizerTestUtil().WithMonitorBounds(p1024x768).GetWindowBounds();
    EXPECT_EQ(
        gfx::Rect(kDesktopBorderSize, kDesktopBorderSize,
                  1024 - kDesktopBorderSize * 2, 768 - kDesktopBorderSize),
        window_bounds);
  }

  {
    // 4:3 monitor case, 1024x768, taskbar on bottom.
    gfx::Rect window_bounds = WindowSizerTestUtil()
                                  .WithMonitorBounds(p1024x768)
                                  .WithMonitorWorkArea(taskbar_bottom_work_area)
                                  .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(kDesktopBorderSize, kDesktopBorderSize,
                        1024 - kDesktopBorderSize * 2,
                        taskbar_bottom_work_area.height() - kDesktopBorderSize),
              window_bounds);
  }

  {
    // 4:3 monitor case, 1024x768, taskbar on right.
    gfx::Rect window_bounds = WindowSizerTestUtil()
                                  .WithMonitorBounds(p1024x768)
                                  .WithMonitorWorkArea(taskbar_right_work_area)
                                  .GetWindowBounds();
    EXPECT_EQ(
        gfx::Rect(kDesktopBorderSize, kDesktopBorderSize,
                  taskbar_right_work_area.width() - kDesktopBorderSize * 2,
                  768 - kDesktopBorderSize),
        window_bounds);
  }

  {
    // 4:3 monitor case, 1024x768, taskbar on left.
    gfx::Rect window_bounds = WindowSizerTestUtil()
                                  .WithMonitorBounds(p1024x768)
                                  .WithMonitorWorkArea(taskbar_left_work_area)
                                  .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(taskbar_left_work_area.x() + kDesktopBorderSize,
                        kDesktopBorderSize,
                        taskbar_left_work_area.width() - kDesktopBorderSize * 2,
                        taskbar_left_work_area.height() - kDesktopBorderSize),
              window_bounds);
  }

  {
    // 4:3 monitor case, 1024x768, taskbar on top.
    gfx::Rect window_bounds = WindowSizerTestUtil()
                                  .WithMonitorBounds(p1024x768)
                                  .WithMonitorWorkArea(taskbar_top_work_area)
                                  .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(kDesktopBorderSize,
                        taskbar_top_work_area.y() + kDesktopBorderSize,
                        1024 - kDesktopBorderSize * 2,
                        taskbar_top_work_area.height() - kDesktopBorderSize),
              window_bounds);
  }

  {
    // 4:3 monitor case, 1280x1024.
    gfx::Rect window_bounds =
        WindowSizerTestUtil().WithMonitorBounds(p1280x1024).GetWindowBounds();
    EXPECT_EQ(gfx::Rect((1280 - kMaximumWindowWidth) / 2, kDesktopBorderSize,
                        kMaximumWindowWidth, 1024 - kDesktopBorderSize),
              window_bounds);
  }

  {
    // 4:3 monitor case, 1600x1200.
    gfx::Rect window_bounds =
        WindowSizerTestUtil().WithMonitorBounds(p1600x1200).GetWindowBounds();
    EXPECT_EQ(gfx::Rect((1600 - kMaximumWindowWidth) / 2, kDesktopBorderSize,
                        kMaximumWindowWidth, 1200 - kDesktopBorderSize),
              window_bounds);
  }

  {
    // 16:10 monitor case, 1680x1050.
    gfx::Rect window_bounds =
        WindowSizerTestUtil().WithMonitorBounds(p1680x1050).GetWindowBounds();
    EXPECT_EQ(gfx::Rect((1680 - kMaximumWindowWidth) / 2, kDesktopBorderSize,
                        kMaximumWindowWidth, 1050 - kDesktopBorderSize),
              window_bounds);
  }

  {
    // 16:10 monitor case, 1920x1200.
    gfx::Rect window_bounds =
        WindowSizerTestUtil().WithMonitorBounds(p1920x1200).GetWindowBounds();
    EXPECT_EQ(gfx::Rect((1920 - kMaximumWindowWidth) / 2, kDesktopBorderSize,
                        kMaximumWindowWidth, 1200 - kDesktopBorderSize),
              window_bounds);
  }
}

// Test that the next opened window is positioned appropriately given the
// bounds of an existing window of the same type.
TEST(WindowSizerChromeOS2NoAshTest, LastWindowBoundsCase) {
  {
    // Normal, in the middle of the screen somewhere.
    gfx::Rect window_bounds =
        WindowSizerTestUtil()
            .WithMonitorBounds(p1024x768)
            .WithLastActiveBounds(
                gfx::Rect(kDesktopBorderSize, kDesktopBorderSize, 500, 400))
            .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(kWindowTilePixels + kDesktopBorderSize,
                        kWindowTilePixels + kDesktopBorderSize, 500, 400)
                  .ToString(),
              window_bounds.ToString());
  }

  {
    // Taskbar on top.
    gfx::Rect window_bounds =
        WindowSizerTestUtil()
            .WithMonitorBounds(p1024x768)
            .WithMonitorWorkArea(taskbar_top_work_area)
            .WithLastActiveBounds(
                gfx::Rect(kDesktopBorderSize, kDesktopBorderSize, 500, 400))
            .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(kWindowTilePixels + kDesktopBorderSize,
                        std::max(kWindowTilePixels + kDesktopBorderSize,
                                 34 /* toolbar height */),
                        500, 400)
                  .ToString(),
              window_bounds.ToString());
  }

  {
    // Too small to satisfy the minimum visibility condition.
    gfx::Rect window_bounds =
        WindowSizerTestUtil()
            .WithMonitorBounds(p1024x768)
            .WithLastActiveBounds(
                gfx::Rect(kDesktopBorderSize, kDesktopBorderSize, 29, 29))
            .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(kWindowTilePixels + kDesktopBorderSize,
                        kWindowTilePixels + kDesktopBorderSize, 30 /* not 29 */,
                        30 /* not 29 */)
                  .ToString(),
              window_bounds.ToString());
  }
}

TEST(WindowSizerChromeOSNoAshTest,
     LastWindowOffscreenWithNonAggressiveRepositioning) {
  {
    // Taskbar on left.
    gfx::Rect window_bounds =
        WindowSizerTestUtil()
            .WithMonitorBounds(p1024x768)
            .WithMonitorWorkArea(taskbar_left_work_area)
            .WithLastActiveBounds(
                gfx::Rect(kDesktopBorderSize, kDesktopBorderSize, 500, 400))
            .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(kWindowTilePixels + kDesktopBorderSize,
                        kWindowTilePixels + kDesktopBorderSize, 500, 400)
                  .ToString(),
              window_bounds.ToString());
  }

  {
    // Offset would put the new window offscreen at the bottom but the minimum
    // visibility condition is barely satisfied without relocation.
    gfx::Rect window_bounds =
        WindowSizerTestUtil()
            .WithMonitorBounds(p1024x768)
            .WithLastActiveBounds(gfx::Rect(10, 638, 500, 400))
            .GetWindowBounds();
    // The y position is the display height (768) minus the minimum visible
    // height (0.3f * 400).
    EXPECT_EQ(gfx::Rect(10 + kWindowTilePixels, 648, 500, 400).ToString(),
              window_bounds.ToString());
  }

  {
    // Offset would put the new window offscreen at the bottom and the minimum
    // visibility condition is satisfied by relocation.
    gfx::Rect window_bounds =
        WindowSizerTestUtil()
            .WithMonitorBounds(p1024x768)
            .WithLastActiveBounds(gfx::Rect(20, 639, 500, 400))
            .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(20 + kWindowTilePixels, 648, 500, 400).ToString(),
              window_bounds.ToString());
  }

  {
    // Offset would put the new window offscreen at the right but the minimum
    // visibility condition is barely satisfied without relocation.
    gfx::Rect window_bounds =
        WindowSizerTestUtil()
            .WithMonitorBounds(p1024x768)
            .WithLastActiveBounds(gfx::Rect(864, 10, 500, 400))
            .GetWindowBounds();
    // The x position is the display width (1024) minus the minimum visible
    // width (0.3f * 500).
    EXPECT_EQ(gfx::Rect(874, 10 + kWindowTilePixels, 500, 400).ToString(),
              window_bounds.ToString());
  }

  {
    // Offset would put the new window offscreen at the right and the minimum
    // visibility condition is satisfied by relocation.
    gfx::Rect window_bounds =
        WindowSizerTestUtil()
            .WithMonitorBounds(p1024x768)
            .WithLastActiveBounds(gfx::Rect(865, 20, 500, 400))
            .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(874, 20 + kWindowTilePixels, 500, 400).ToString(),
              window_bounds.ToString());
  }

  {
    // Offset would put the new window offscreen at the bottom right and the
    // minimum visibility condition is satisfied by relocation.
    gfx::Rect window_bounds =
        WindowSizerTestUtil()
            .WithMonitorBounds(p1024x768)
            .WithLastActiveBounds(gfx::Rect(865, 639, 500, 400))
            .GetWindowBounds();
    EXPECT_EQ(gfx::Rect(874, 648, 500, 400).ToString(),
              window_bounds.ToString());
  }
}

// Test the placement of newly created windows.
TEST_F(WindowSizerChromeOSTest, PlaceNewWindows) {
  UpdateDisplay("1600x1200");
  auto* shelf = ash::Shell::GetPrimaryRootWindowController()->shelf();
  shelf->SetAutoHideBehavior(ash::ShelfAutoHideBehavior::kAlways);

  int64_t display_id = display::Screen::GetScreen()->GetPrimaryDisplay().id();

  // Create a browser to pass into the WindowSizerTestUtil::GetWindowBounds
  // function.
  Browser::CreateParams native_params(&profile_, true);
  auto browser = CreateWindowlessBrowser(native_params);

  // Creating a popup handler here to make sure it does not interfere with the
  // existing windows.
  Browser::CreateParams params2(&profile_, true);
  std::unique_ptr<Browser> browser2 = (CreateTestBrowser(
      CreateTestWindowInShellWithId(0), gfx::Rect(16, 32, 640, 320), &params2));
  BrowserWindow* browser_window = browser2->window();

  // Creating a popup to make sure it does not interfere with the positioning.
  Browser::CreateParams params_popup(Browser::TYPE_POPUP, &profile_, true);
  std::unique_ptr<Browser> browser_popup(
      CreateTestBrowser(CreateTestWindowInShellWithId(1),
                        gfx::Rect(16, 32, 128, 256), &params_popup));

  browser_window->Show();

  // Make sure that popups do not get changed.
  {
    Browser::CreateParams params_new_popup(Browser::TYPE_POPUP, &profile_,
                                           true);
    auto new_popup = CreateWindowlessBrowser(params_new_popup);
    gfx::Rect window_bounds;
    GetWindowBounds(new_popup.get(), gfx::Rect(), display_id,
                    gfx::Rect(50, 100, 300, 150), bottom_s1600x1200, PERSISTED,
                    &window_bounds);
    EXPECT_EQ("50,100 300x150", window_bounds.ToString());
  }

  browser_window->Hide();
  {
    // If a window is there but not shown the persisted default should be used.
    gfx::Rect window_bounds;
    GetWindowBounds(browser.get(), gfx::Rect(), display_id,
                    gfx::Rect(50, 100, 300, 150), bottom_s1600x1200, PERSISTED,
                    &window_bounds);
    EXPECT_EQ("50,100 300x150", window_bounds.ToString());
  }

  {
    // If a window is there but not shown the default should be returned.
    gfx::Rect window_bounds;
    GetWindowBounds(browser.get(), gfx::Rect(), display_id, gfx::Rect(),
                    bottom_s1600x1200, DEFAULT, &window_bounds);
    // Note: We need to also take the defaults maximum width into account here
    // since that might get used if the resolution is too big.
    EXPECT_EQ(
        gfx::Rect(
            std::max(kDesktopBorderSize, (1600 - kMaximumWindowWidth) / 2),
            kDesktopBorderSize,
            std::min(kMaximumWindowWidth, 1600 - 2 * kDesktopBorderSize),
            1200 - kDesktopBorderSize)
            .ToString(),
        window_bounds.ToString());
  }
}

// Test the placement of newly created windows on an empty desktop.
// This test supplements "PlaceNewWindows" by testing the creation of a newly
// created browser window on an empty desktop.
TEST_F(WindowSizerChromeOSTest, PlaceNewBrowserWindowOnEmptyDesktop) {
  Browser::CreateParams native_params(&profile_, true);
  auto browser = CreateWindowlessBrowser(native_params);

  // A common screen size for Chrome OS devices where forced-maximized
  // windows are desirable.
  UpdateDisplay("1366x768");
  const gfx::Rect p1366x768(0, 0, 1366, 768);

  // If there is no previous state the window should get maximized if the
  // screen is less than or equal to our limit (1366 pixels width).
  gfx::Rect window_bounds;
  ui::WindowShowState out_show_state1 = ui::SHOW_STATE_DEFAULT;
  GetBrowserWindowBoundsAndShowState(
      gfx::Rect(),             // The (persisted) bounds.
      p1366x768,               // The overall work area.
      ui::SHOW_STATE_NORMAL,   // The persisted show state.
      ui::SHOW_STATE_DEFAULT,  // The last show state.
      DEFAULT,                 // No persisted values.
      browser.get(),           // Use this browser.
      gfx::Rect(),             // Don't request valid bounds.
      &window_bounds, &out_show_state1);
  EXPECT_EQ(ui::SHOW_STATE_MAXIMIZED, out_show_state1);

  // If there is a stored coordinate however, that should be taken instead.
  ui::WindowShowState out_show_state2 = ui::SHOW_STATE_DEFAULT;
  GetBrowserWindowBoundsAndShowState(
      gfx::Rect(50, 100, 300, 150),  // The (persisted) bounds.
      p1366x768,                     // The overall work area.
      ui::SHOW_STATE_NORMAL,         // The persisted show state.
      ui::SHOW_STATE_DEFAULT,        // The last show state.
      PERSISTED,                     // Set the persisted values.
      browser.get(),                 // Use this browser.
      gfx::Rect(),                   // Don't request valid bounds.
      &window_bounds, &out_show_state2);
  EXPECT_EQ(ui::SHOW_STATE_NORMAL, out_show_state2);
  EXPECT_EQ("50,100 300x150", window_bounds.ToString());
}

TEST_F(WindowSizerChromeOSTest, PlaceNewBrowserWindowOnLargeDesktop) {
  Browser::CreateParams native_params(&profile_, true);
  auto browser = CreateWindowlessBrowser(native_params);

  // A larger monitor should not trigger auto-maximize.
  UpdateDisplay("1600x1200");
  gfx::Rect window_bounds;
  ui::WindowShowState out_show_state = ui::SHOW_STATE_DEFAULT;
  GetBrowserWindowBoundsAndShowState(
      gfx::Rect(),             // The (persisted) bounds.
      p1600x1200,              // The overall work area.
      ui::SHOW_STATE_NORMAL,   // The persisted show state.
      ui::SHOW_STATE_DEFAULT,  // The last show state.
      DEFAULT,                 // No persisted values.
      browser.get(),           // Use this browser.
      gfx::Rect(),             // Don't request valid bounds.
      &window_bounds, &out_show_state);
  EXPECT_EQ(ui::SHOW_STATE_DEFAULT, out_show_state);
}

// Test the placement of newly created windows on multiple dislays.
TEST_F(WindowSizerChromeOSTest, PlaceNewWindowsOnMultipleDisplays) {
  UpdateDisplay("1600x1200,1600x1200");
  display::Display primary_display =
      display::Screen::GetScreen()->GetPrimaryDisplay();
  display::Display second_display =
      display::test::DisplayManagerTestApi(display_manager())
          .GetSecondaryDisplay();
  gfx::Rect primary_bounds = primary_display.bounds();
  gfx::Rect secondary_bounds = second_display.bounds();

  // Create browser windows that are used as reference.
  Browser::CreateParams params(&profile_, true);
  std::unique_ptr<Browser> browser(CreateTestBrowser(
      CreateTestWindowInShellWithId(0), gfx::Rect(10, 10, 200, 200), &params));
  BrowserWindow* browser_window = browser->window();
  gfx::NativeWindow native_window = browser_window->GetNativeWindow();
  browser_window->Show();
  EXPECT_EQ(native_window->GetRootWindow(),
            ash::Shell::GetRootWindowForNewWindows());

  Browser::CreateParams another_params(&profile_, true);
  std::unique_ptr<Browser> another_browser(
      CreateTestBrowser(CreateTestWindowInShellWithId(1),
                        gfx::Rect(400, 10, 300, 300), &another_params));
  BrowserWindow* another_browser_window = another_browser->window();
  gfx::NativeWindow another_native_window =
      another_browser_window->GetNativeWindow();
  another_browser_window->Show();

  // Creating a new window to verify the new placement.
  Browser::CreateParams new_params(&profile_, true);
  auto new_browser = CreateWindowlessBrowser(new_params);

  // Make sure the primary root is active.
  ASSERT_EQ(ash::Shell::GetPrimaryRootWindow(),
            ash::Shell::GetRootWindowForNewWindows());

  // First new window should be in the primary.
  {
    gfx::Rect window_bounds;
    GetWindowBounds(new_browser.get(), gfx::Rect(), primary_display.id(),
                    &window_bounds);
    // TODO(oshima): Use exact bounds when the window sizer includes the result
    // from RearrangeVisibleWindowOnShow.
    EXPECT_TRUE(primary_bounds.Contains(window_bounds));
  }

  // Move the window to the right side of the secondary display and create a new
  // window. It should be opened then on the secondary display.
  {
    browser_window->GetNativeWindow()->SetBoundsInScreen(
        gfx::Rect(secondary_bounds.CenterPoint().x() - 100, 10, 200, 200),
        second_display);
    wm::GetActivationClient(native_window->GetRootWindow())
        ->ActivateWindow(native_window);
    EXPECT_NE(ash::Shell::GetPrimaryRootWindow(),
              ash::Shell::GetRootWindowForNewWindows());
    gfx::Rect window_bounds;
    GetWindowBounds(new_browser.get(), gfx::Rect(), second_display.id(),
                    &window_bounds);
    // TODO(oshima): Use exact bounds when the window sizer includes the result
    // from RearrangeVisibleWindowOnShow.
    EXPECT_TRUE(secondary_bounds.Contains(window_bounds));
  }

  // Activate another window in the primary display and create a new window.
  // It should be created in the primary display.
  {
    wm::GetActivationClient(another_native_window->GetRootWindow())
        ->ActivateWindow(another_native_window);
    EXPECT_EQ(ash::Shell::GetPrimaryRootWindow(),
              ash::Shell::GetRootWindowForNewWindows());

    gfx::Rect window_bounds;
    GetWindowBounds(new_browser.get(), gfx::Rect(), primary_display.id(),
                    &window_bounds);
    // TODO(oshima): Use exact bounds when the window sizer includes the result
    // from RearrangeVisibleWindowOnShow.
    EXPECT_TRUE(primary_bounds.Contains(window_bounds));
  }
}

// Test that the show state is properly returned for non default cases.
TEST_F(WindowSizerChromeOSTest, TestShowState) {
  UpdateDisplay("1600x1200");

  // Creating a browser & window to play with.
  Browser::CreateParams params(Browser::TYPE_NORMAL, &profile_, true);
  auto browser = CreateWindowlessBrowser(params);

  // Create also a popup browser since that behaves different.
  Browser::CreateParams params_popup(Browser::TYPE_POPUP, &profile_, true);
  auto browser_popup = CreateWindowlessBrowser(params_popup);

  // Tabbed windows should retrieve the saved window state - since there is a
  // top window.
  EXPECT_EQ(ui::SHOW_STATE_MAXIMIZED,
            GetBrowserWindowShowState(ui::SHOW_STATE_MAXIMIZED,
                                      ui::SHOW_STATE_NORMAL, PERSISTED,
                                      browser.get(), p1600x1200, p1600x1200));
  // A window that is smaller than the whole work area is set to default state.
  EXPECT_EQ(ui::SHOW_STATE_DEFAULT,
            GetBrowserWindowShowState(ui::SHOW_STATE_DEFAULT,
                                      ui::SHOW_STATE_NORMAL, PERSISTED,
                                      browser.get(), p1280x1024, p1600x1200));
  // A window that is sized to occupy the whole work area is maximized.
  EXPECT_EQ(ui::SHOW_STATE_MAXIMIZED,
            GetBrowserWindowShowState(ui::SHOW_STATE_DEFAULT,
                                      ui::SHOW_STATE_NORMAL, PERSISTED,
                                      browser.get(), p1600x1200, p1600x1200));
  // Non tabbed windows should always follow the window saved visibility state.
  EXPECT_EQ(ui::SHOW_STATE_MAXIMIZED,
            GetBrowserWindowShowState(
                ui::SHOW_STATE_MAXIMIZED, ui::SHOW_STATE_NORMAL, BOTH,
                browser_popup.get(), p1600x1200, p1600x1200));
  // The non tabbed window will take the status of the last active of its kind.
  EXPECT_EQ(ui::SHOW_STATE_NORMAL,
            GetBrowserWindowShowState(
                ui::SHOW_STATE_DEFAULT, ui::SHOW_STATE_NORMAL, BOTH,
                browser_popup.get(), p1600x1200, p1600x1200));

  // A tabbed window should now take the top level window state.
  EXPECT_EQ(
      ui::SHOW_STATE_NORMAL,
      GetBrowserWindowShowState(ui::SHOW_STATE_MAXIMIZED, ui::SHOW_STATE_NORMAL,
                                BOTH, browser.get(), p1600x1200, p1600x1200));
  // Non tabbed windows should always follow the window saved visibility state.
  EXPECT_EQ(ui::SHOW_STATE_MAXIMIZED,
            GetBrowserWindowShowState(
                ui::SHOW_STATE_MAXIMIZED, ui::SHOW_STATE_MINIMIZED, BOTH,
                browser_popup.get(), p1600x1200, p1600x1200));
}

TEST_F(WindowSizerChromeOSTest, TestShowStateOnTinyScreen) {
  Browser::CreateParams params(Browser::TYPE_NORMAL, &profile_, true);
  auto browser = CreateWindowlessBrowser(params);

  // In smaller screen resolutions we default to maximized if there is no other
  // window visible.
  UpdateDisplay("640x480");
  const gfx::Rect tiny_screen(0, 0, 640, 480);
  EXPECT_EQ(ui::SHOW_STATE_MAXIMIZED,
            GetBrowserWindowShowState(ui::SHOW_STATE_MAXIMIZED,
                                      ui::SHOW_STATE_DEFAULT, BOTH,
                                      browser.get(), tiny_screen, tiny_screen));
}

// Test that the default show state override behavior is properly handled.
TEST_F(WindowSizerChromeOSTest, TestShowStateDefaults) {
  UpdateDisplay("1600x1200");
  // Creating a browser & window to play with.

  Browser::CreateParams params(Browser::TYPE_NORMAL, &profile_, true);
  auto browser = CreateWindowlessBrowser(params);

  // Create also a popup browser since that behaves slightly different for
  // defaults.
  Browser::CreateParams params_popup(Browser::TYPE_POPUP, &profile_, true);
  auto browser_popup = CreateWindowlessBrowser(params_popup);

  // Check that a browser creation state always get used if not given as
  // SHOW_STATE_DEFAULT.
  ui::WindowShowState window_show_state = GetBrowserWindowShowState(
      ui::SHOW_STATE_MAXIMIZED, ui::SHOW_STATE_MAXIMIZED, DEFAULT,
      browser.get(), p1600x1200, p1600x1200);
  EXPECT_EQ(window_show_state, ui::SHOW_STATE_DEFAULT);

  browser->set_initial_show_state(ui::SHOW_STATE_MINIMIZED);
  EXPECT_EQ(GetBrowserWindowShowState(ui::SHOW_STATE_MAXIMIZED,
                                      ui::SHOW_STATE_MAXIMIZED, BOTH,
                                      browser.get(), p1600x1200, p1600x1200),
            ui::SHOW_STATE_MINIMIZED);
  browser->set_initial_show_state(ui::SHOW_STATE_NORMAL);
  EXPECT_EQ(GetBrowserWindowShowState(ui::SHOW_STATE_MAXIMIZED,
                                      ui::SHOW_STATE_MAXIMIZED, BOTH,
                                      browser.get(), p1600x1200, p1600x1200),
            ui::SHOW_STATE_NORMAL);
  browser->set_initial_show_state(ui::SHOW_STATE_MAXIMIZED);
  EXPECT_EQ(
      GetBrowserWindowShowState(ui::SHOW_STATE_NORMAL, ui::SHOW_STATE_NORMAL,
                                BOTH, browser.get(), p1600x1200, p1600x1200),
      ui::SHOW_STATE_MAXIMIZED);

  // Check that setting the maximized command line option is forcing the
  // maximized state.
  base::CommandLine::ForCurrentProcess()->AppendSwitch(
      switches::kStartMaximized);

  browser->set_initial_show_state(ui::SHOW_STATE_NORMAL);
  EXPECT_EQ(
      GetBrowserWindowShowState(ui::SHOW_STATE_NORMAL, ui::SHOW_STATE_NORMAL,
                                BOTH, browser.get(), p1600x1200, p1600x1200),
      ui::SHOW_STATE_MAXIMIZED);

  // The popup should favor the initial show state over the command line.
  EXPECT_EQ(GetBrowserWindowShowState(
                ui::SHOW_STATE_NORMAL, ui::SHOW_STATE_NORMAL, BOTH,
                browser_popup.get(), p1600x1200, p1600x1200),
            ui::SHOW_STATE_NORMAL);
}

TEST_F(WindowSizerChromeOSTest, DefaultStateBecomesMaximized) {
  // Create a browser to pass into the WindowSizerTestUtil::GetWindowBounds
  // function.
  Browser::CreateParams native_params(&profile_, true);
  auto browser = CreateWindowlessBrowser(native_params);

  gfx::Rect display_bounds =
      display::Screen::GetScreen()->GetPrimaryDisplay().bounds();
  gfx::Rect specified_bounds = display_bounds;

  // Make a window bigger than the display work area.
  specified_bounds.Inset(-20);
  ui::WindowShowState show_state = ui::SHOW_STATE_DEFAULT;
  gfx::Rect bounds;
  WindowSizer::GetBrowserWindowBoundsAndShowState(
      specified_bounds, browser.get(), &bounds, &show_state);
  // The window should start maximized with its restore bounds shrunken.
  EXPECT_EQ(ui::SHOW_STATE_MAXIMIZED, show_state);
  EXPECT_NE(display_bounds.ToString(), bounds.ToString());
  EXPECT_TRUE(display_bounds.Contains(bounds));

  // Make a window smaller than the display work area.
  specified_bounds.Inset(100);
  show_state = ui::SHOW_STATE_DEFAULT;
  WindowSizer::GetBrowserWindowBoundsAndShowState(
      specified_bounds, browser.get(), &bounds, &show_state);
  // The window should start in default state.
  EXPECT_EQ(ui::SHOW_STATE_DEFAULT, show_state);
  EXPECT_EQ(specified_bounds.ToString(), bounds.ToString());
}

// Test that the target root window is used as the destination of
// the non browser window. This differ from PersistedBoundsCase
// in that this uses real ash shell implementations + StateProvider
// rather than mocks.
TEST_F(WindowSizerChromeOSTest, DefaultBoundsInTargetDisplay) {
  UpdateDisplay("500x400,600x500");

  {
    // By default windows are placed on the primary display.
    aura::Window* first_root = ash::Shell::GetAllRootWindows()[0];
    EXPECT_EQ(first_root, ash::Shell::GetRootWindowForNewWindows());
    gfx::Rect bounds;
    ui::WindowShowState show_state;
    WindowSizer::GetBrowserWindowBoundsAndShowState(gfx::Rect(), nullptr,
                                                    &bounds, &show_state);
    EXPECT_TRUE(first_root->GetBoundsInScreen().Contains(bounds));
  }

  {
    // When the second display is active new windows are placed there.
    aura::Window* second_root = ash::Shell::GetAllRootWindows()[1];
    int64_t second_display_id =
        display::test::DisplayManagerTestApi(display_manager())
            .GetSecondaryDisplay()
            .id();
    display::Screen::GetScreen()->SetDisplayForNewWindows(second_display_id);
    gfx::Rect bounds;
    ui::WindowShowState show_state;
    WindowSizer::GetBrowserWindowBoundsAndShowState(gfx::Rect(), nullptr,
                                                    &bounds, &show_state);
    EXPECT_TRUE(second_root->GetBoundsInScreen().Contains(bounds));
  }
}

TEST_F(WindowSizerChromeOSTest, TrustedPopupBehavior) {
  Browser::CreateParams trusted_popup_create_params(Browser::TYPE_POPUP,
                                                    &profile_, true);
  trusted_popup_create_params.trusted_source = true;

  auto trusted_popup = CreateWindowlessBrowser(trusted_popup_create_params);
  // Trusted popup windows should follow the saved show state and ignore the
  // last show state.
  EXPECT_EQ(ui::SHOW_STATE_DEFAULT,
            GetBrowserWindowShowState(
                ui::SHOW_STATE_DEFAULT, ui::SHOW_STATE_NORMAL, BOTH,
                trusted_popup.get(), p1280x1024, p1600x1200));
  // A popup that is sized to occupy the whole work area has default state.
  EXPECT_EQ(ui::SHOW_STATE_DEFAULT,
            GetBrowserWindowShowState(
                ui::SHOW_STATE_DEFAULT, ui::SHOW_STATE_NORMAL, BOTH,
                trusted_popup.get(), p1600x1200, p1600x1200));
}