| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
 100
 101
 102
 103
 104
 105
 106
 107
 108
 109
 110
 111
 112
 113
 114
 115
 116
 117
 118
 119
 120
 121
 122
 123
 124
 125
 126
 127
 128
 129
 130
 131
 132
 133
 134
 135
 136
 137
 138
 139
 140
 141
 142
 143
 144
 145
 146
 147
 148
 149
 150
 151
 152
 153
 154
 155
 156
 157
 158
 159
 160
 161
 162
 163
 164
 165
 166
 167
 168
 169
 170
 171
 172
 173
 174
 175
 176
 
 | // 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.
#ifndef CHROME_BROWSER_UI_WINDOW_SIZER_WINDOW_SIZER_H_
#define CHROME_BROWSER_UI_WINDOW_SIZER_WINDOW_SIZER_H_
#include <memory>
#include "base/memory/raw_ptr.h"
#include "ui/base/mojom/window_show_state.mojom-forward.h"
#include "ui/base/ui_base_types.h"
#include "ui/gfx/geometry/rect.h"
class Browser;
namespace display {
class Display;
class Screen;
}  // namespace display
///////////////////////////////////////////////////////////////////////////////
// WindowSizer
//
//  A class that determines the best new size and position for a window to be
//  shown at based several factors, including the position and size of the last
//  window of the same type, the last saved bounds of the window from the
//  previous session, and default system metrics if neither of the above two
//  conditions exist. The system has built-in providers for monitor metrics
//  and persistent storage (using preferences) but can be overrided with mocks
//  for testing.
//
// TODO(crbug.com/846736): Extract the platform-specific code out of this class.
class WindowSizer {
 public:
  // An interface implemented by an object that can retrieve state from either a
  // persistent store or an existing window.
  class StateProvider {
   public:
    virtual ~StateProvider() = default;
    // Retrieve the persisted bounds of the window. Returns true if there were
    // persisted bounds and false otherwise. If this method returns false, none
    // of the out parameters are touched. If it returns true, |bounds| was
    // overwritten, and |work_area| may have been overwritten if there was also
    // a saved work area.  The |show_state| variable will only be touched if
    // there was persisted data and the |show_state| variable is
    // WindowShowState::kDefault.
    virtual bool GetPersistentState(
        gfx::Rect* bounds,
        gfx::Rect* work_area,
        ui::mojom::WindowShowState* show_state) const = 0;
    // Retrieve the bounds of the most recent window of the matching type.
    // Returns true if there was a last active window to retrieve state
    // information from, false otherwise.
    // The |show_state| variable will only be touched if we have found a
    // suitable window and the |show_state| variable is
    // WindowShowState::kDefault.
    virtual bool GetLastActiveWindowState(
        gfx::Rect* bounds,
        ui::mojom::WindowShowState* show_state) const = 0;
  };
  WindowSizer(const WindowSizer&) = delete;
  WindowSizer& operator=(const WindowSizer&) = delete;
  // Determines the position and size for a window as it is created as well
  // as the initial state. This function uses several strategies to figure out
  // optimal size and placement, first looking for an existing active window,
  // then falling back to persisted data from a previous session, finally
  // utilizing a default algorithm. If |specified_bounds| are non-empty, this
  // value is returned instead. To explicitly specify a particular window to
  // base the bounds on, pass in a non-null value for |browser|.
  //
  // |show_state| will be overwritten and return the initial visual state of
  // the window to use.
  static void GetBrowserWindowBoundsAndShowState(
      const gfx::Rect& specified_bounds,
      const Browser* browser,
      gfx::Rect* window_bounds,
      ui::mojom::WindowShowState* show_state);
  // As above, but takes a state provider for testing.
  static void GetBrowserWindowBoundsAndShowState(
      std::unique_ptr<StateProvider> state_provider,
      const gfx::Rect& specified_bounds,
      const Browser* browser,
      gfx::Rect* window_bounds,
      ui::mojom::WindowShowState* show_state);
  // Returns the default origin for popups of the given size.
  static gfx::Point GetDefaultPopupOrigin(const gfx::Size& size);
  // How much horizontal and vertical offset there is between newly
  // opened windows.  This value may be different on each platform.
  static const int kWindowTilePixels;
  // The maximum default window width. This value may differ between platforms.
  static const int kWindowMaxDefaultWidth;
 protected:
  const StateProvider* state_provider() const { return state_provider_.get(); }
  const Browser* browser() const { return browser_; }
  // WindowSizer will use the platform's display::Screen.
  WindowSizer(std::unique_ptr<StateProvider> state_provider,
              const Browser* browser);
  virtual ~WindowSizer();
  // See GetBrowserWindowBoundsAndShowState() above.
  virtual void DetermineWindowBoundsAndShowState(
      const gfx::Rect& specified_bounds,
      gfx::Rect* bounds,
      ui::mojom::WindowShowState* show_state);
  // Adjusts the work area the platform-specific way.
  virtual void AdjustWorkAreaForPlatform(gfx::Rect& work_area);
  // Gets the size and placement of the last active window. Returns true if this
  // data is valid, false if there is no last window and the application should
  // restore saved state from preferences using RestoreWindowPosition.
  // |show_state| will only be changed if it was set to
  // WindowShowState::kDefault.
  bool GetLastActiveWindowBounds(gfx::Rect* bounds,
                                 ui::mojom::WindowShowState* show_state) const;
  // Gets the size and placement of the last window in the last session, saved
  // in local state preferences. Returns true if local state exists containing
  // this information, false if this information does not exist and a default
  // size should be used.
  // |show_state| will only be changed if it was set to
  // WindowShowState::kDefault.
  bool GetSavedWindowBounds(gfx::Rect* bounds,
                            ui::mojom::WindowShowState* show_state) const;
  // Gets the default window position and size to be shown on
  // |display| if there is no last window and no saved window
  // placement in prefs. This function determines the default size
  // based on monitor size, etc.
  virtual gfx::Rect GetDefaultWindowBounds(
      const display::Display& display) const;
  // Adjusts |bounds| to be visible on-screen, biased toward the work area of
  // the |display|.  Despite the name, this doesn't
  // guarantee the bounds are fully contained within this display's work rect;
  // it just tried to ensure the edges are visible on _some_ work rect.
  // If |saved_work_area| is non-empty, it is used to determine whether the
  // monitor configuration has changed. If it has, bounds are repositioned and
  // resized if necessary to make them completely contained in the current work
  // area.
  void AdjustBoundsToBeVisibleOnDisplay(const display::Display& display,
                                        const gfx::Rect& saved_work_area,
                                        gfx::Rect* bounds) const;
  // Determine the default show state for the window - not looking at other
  // windows or at persistent information.
  static ui::mojom::WindowShowState GetWindowDefaultShowState(
      const Browser* browser);
  // Returns the target display for a new window with |bounds| in screen
  // coordinates.
  static display::Display GetDisplayForNewWindow(
      const gfx::Rect& bounds = gfx::Rect());
 private:
  friend class WindowSizerTestUtil;
  // Providers for persistent storage and monitor metrics.
  std::unique_ptr<StateProvider> state_provider_;
  // Note that this browser handle might be NULL.
  const raw_ptr<const Browser> browser_;
};
#endif  // CHROME_BROWSER_UI_WINDOW_SIZER_WINDOW_SIZER_H_
 |