Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(372)

Unified Diff: ash/wm/tablet_mode/tablet_mode_window_manager.cc

Issue 2905383003: (fix landed) Reland of name MaximizeMode to TabletMode (Closed)
Patch Set: Created 3 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: ash/wm/tablet_mode/tablet_mode_window_manager.cc
diff --git a/ash/wm/tablet_mode/tablet_mode_window_manager.cc b/ash/wm/tablet_mode/tablet_mode_window_manager.cc
new file mode 100644
index 0000000000000000000000000000000000000000..ee032c03b36457706da50c61e4bf327b651b2e62
--- /dev/null
+++ b/ash/wm/tablet_mode/tablet_mode_window_manager.cc
@@ -0,0 +1,306 @@
+// Copyright 2014 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/wm/tablet_mode/tablet_mode_window_manager.h"
+
+#include "ash/ash_switches.h"
+#include "ash/public/cpp/shell_window_ids.h"
+#include "ash/root_window_controller.h"
+#include "ash/session/session_state_delegate.h"
+#include "ash/shell.h"
+#include "ash/shell_port.h"
+#include "ash/wm/mru_window_tracker.h"
+#include "ash/wm/overview/window_selector_controller.h"
+#include "ash/wm/tablet_mode/tablet_mode_backdrop_delegate_impl.h"
+#include "ash/wm/tablet_mode/tablet_mode_event_handler.h"
+#include "ash/wm/tablet_mode/tablet_mode_window_state.h"
+#include "ash/wm/window_state.h"
+#include "ash/wm/wm_event.h"
+#include "ash/wm/workspace_controller.h"
+#include "base/command_line.h"
+#include "base/memory/ptr_util.h"
+#include "base/stl_util.h"
+#include "ui/aura/client/aura_constants.h"
+#include "ui/display/screen.h"
+
+namespace ash {
+
+namespace {
+
+// Exits overview mode if it is currently active.
+void CancelOverview() {
+ WindowSelectorController* controller =
+ Shell::Get()->window_selector_controller();
+ if (controller->IsSelecting())
+ controller->OnSelectionEnded();
+}
+
+} // namespace
+
+TabletModeWindowManager::~TabletModeWindowManager() {
+ // Overview mode needs to be ended before exiting tablet mode to prevent
+ // transforming windows which are currently in
+ // overview: http://crbug.com/366605
+ CancelOverview();
+ for (aura::Window* window : added_windows_)
+ window->RemoveObserver(this);
+ added_windows_.clear();
+ Shell::Get()->RemoveShellObserver(this);
+ display::Screen::GetScreen()->RemoveObserver(this);
+ EnableBackdropBehindTopWindowOnEachDisplay(false);
+ RemoveWindowCreationObservers();
+ RestoreAllWindows();
+}
+
+int TabletModeWindowManager::GetNumberOfManagedWindows() {
+ return window_state_map_.size();
+}
+
+void TabletModeWindowManager::AddWindow(aura::Window* window) {
+ // Only add the window if it is a direct dependent of a container window
+ // and not yet tracked.
+ if (!ShouldHandleWindow(window) ||
+ base::ContainsKey(window_state_map_, window) ||
+ !IsContainerWindow(window->parent())) {
+ return;
+ }
+
+ MaximizeAndTrackWindow(window);
+}
+
+void TabletModeWindowManager::WindowStateDestroyed(aura::Window* window) {
+ // At this time ForgetWindow() should already have been called. If not,
+ // someone else must have replaced the "window manager's state object".
+ DCHECK(!window->HasObserver(this));
+
+ auto it = window_state_map_.find(window);
+ DCHECK(it != window_state_map_.end());
+ window_state_map_.erase(it);
+}
+
+void TabletModeWindowManager::OnOverviewModeStarting() {
+ SetDeferBoundsUpdates(true);
+}
+
+void TabletModeWindowManager::OnOverviewModeEnded() {
+ SetDeferBoundsUpdates(false);
+}
+
+void TabletModeWindowManager::OnWindowDestroying(aura::Window* window) {
+ if (IsContainerWindow(window)) {
+ // container window can be removed on display destruction.
+ window->RemoveObserver(this);
+ observed_container_windows_.erase(window);
+ } else if (base::ContainsValue(added_windows_, window)) {
+ // Added window was destroyed before being shown.
+ added_windows_.erase(window);
+ window->RemoveObserver(this);
+ } else {
+ // If a known window gets destroyed we need to remove all knowledge about
+ // it.
+ ForgetWindow(window);
+ }
+}
+
+void TabletModeWindowManager::OnWindowHierarchyChanged(
+ const HierarchyChangeParams& params) {
+ // A window can get removed and then re-added by a drag and drop operation.
+ if (params.new_parent && IsContainerWindow(params.new_parent) &&
+ !base::ContainsKey(window_state_map_, params.target)) {
+ // Don't register the window if the window is invisible. Instead,
+ // wait until it becomes visible because the client may update the
+ // flag to control if the window should be added.
+ if (!params.target->IsVisible()) {
+ if (!base::ContainsValue(added_windows_, params.target)) {
+ added_windows_.insert(params.target);
+ params.target->AddObserver(this);
+ }
+ return;
+ }
+ MaximizeAndTrackWindow(params.target);
+ // When the state got added, the "WM_EVENT_ADDED_TO_WORKSPACE" event got
+ // already sent and we have to notify our state again.
+ if (base::ContainsKey(window_state_map_, params.target)) {
+ wm::WMEvent event(wm::WM_EVENT_ADDED_TO_WORKSPACE);
+ wm::GetWindowState(params.target)->OnWMEvent(&event);
+ }
+ }
+}
+
+void TabletModeWindowManager::OnWindowPropertyChanged(aura::Window* window,
+ const void* key,
+ intptr_t old) {
+ // Stop managing |window| if the always-on-top property is added.
+ if (key == aura::client::kAlwaysOnTopKey &&
+ window->GetProperty(aura::client::kAlwaysOnTopKey)) {
+ ForgetWindow(window);
+ }
+}
+
+void TabletModeWindowManager::OnWindowBoundsChanged(
+ aura::Window* window,
+ const gfx::Rect& old_bounds,
+ const gfx::Rect& new_bounds) {
+ if (!IsContainerWindow(window))
+ return;
+ // Reposition all non maximizeable windows.
+ for (auto& pair : window_state_map_)
+ pair.second->UpdateWindowPosition(wm::GetWindowState(pair.first));
+}
+
+void TabletModeWindowManager::OnWindowVisibilityChanged(aura::Window* window,
+ bool visible) {
+ // Skip if it's already managed.
+ if (base::ContainsKey(window_state_map_, window))
+ return;
+
+ if (IsContainerWindow(window->parent()) &&
+ base::ContainsValue(added_windows_, window) && visible) {
+ added_windows_.erase(window);
+ window->RemoveObserver(this);
+ MaximizeAndTrackWindow(window);
+ // When the state got added, the "WM_EVENT_ADDED_TO_WORKSPACE" event got
+ // already sent and we have to notify our state again.
+ if (base::ContainsKey(window_state_map_, window)) {
+ wm::WMEvent event(wm::WM_EVENT_ADDED_TO_WORKSPACE);
+ wm::GetWindowState(window)->OnWMEvent(&event);
+ }
+ }
+}
+
+void TabletModeWindowManager::OnDisplayAdded(const display::Display& display) {
+ DisplayConfigurationChanged();
+}
+
+void TabletModeWindowManager::OnDisplayRemoved(
+ const display::Display& display) {
+ DisplayConfigurationChanged();
+}
+
+void TabletModeWindowManager::OnDisplayMetricsChanged(const display::Display&,
+ uint32_t) {
+ // Nothing to do here.
+}
+
+void TabletModeWindowManager::SetIgnoreWmEventsForExit() {
+ for (auto& pair : window_state_map_) {
+ pair.second->set_ignore_wm_events(true);
+ }
+}
+
+TabletModeWindowManager::TabletModeWindowManager() {
+ // The overview mode needs to be ended before the tablet mode is started. To
+ // guarantee the proper order, it will be turned off from here.
+ CancelOverview();
+
+ MaximizeAllWindows();
+ AddWindowCreationObservers();
+ EnableBackdropBehindTopWindowOnEachDisplay(true);
+ display::Screen::GetScreen()->AddObserver(this);
+ Shell::Get()->AddShellObserver(this);
+ event_handler_ = ShellPort::Get()->CreateTabletModeEventHandler();
+}
+
+void TabletModeWindowManager::MaximizeAllWindows() {
+ MruWindowTracker::WindowList windows =
+ Shell::Get()->mru_window_tracker()->BuildWindowListIgnoreModal();
+ // Add all existing MRU windows.
+ for (auto* window : windows)
+ MaximizeAndTrackWindow(window);
+}
+
+void TabletModeWindowManager::RestoreAllWindows() {
+ while (window_state_map_.size())
+ ForgetWindow(window_state_map_.begin()->first);
+}
+
+void TabletModeWindowManager::SetDeferBoundsUpdates(bool defer_bounds_updates) {
+ for (auto& pair : window_state_map_)
+ pair.second->SetDeferBoundsUpdates(defer_bounds_updates);
+}
+
+void TabletModeWindowManager::MaximizeAndTrackWindow(aura::Window* window) {
+ if (!ShouldHandleWindow(window))
+ return;
+
+ DCHECK(!base::ContainsKey(window_state_map_, window));
+ window->AddObserver(this);
+
+ // We create and remember a tablet mode state which will attach itself to
+ // the provided state object.
+ window_state_map_[window] = new TabletModeWindowState(window, this);
+}
+
+void TabletModeWindowManager::ForgetWindow(aura::Window* window) {
+ WindowToState::iterator it = window_state_map_.find(window);
+
+ // The following DCHECK could fail if our window state object was destroyed
+ // earlier by someone else. However - at this point there is no other client
+ // which replaces the state object and therefore this should not happen.
+ DCHECK(it != window_state_map_.end());
+ window->RemoveObserver(this);
+
+ // By telling the state object to revert, it will switch back the old
+ // State object and destroy itself, calling WindowStateDestroyed().
+ it->second->LeaveTabletMode(wm::GetWindowState(it->first));
+ DCHECK(!base::ContainsKey(window_state_map_, window));
+}
+
+bool TabletModeWindowManager::ShouldHandleWindow(aura::Window* window) {
+ DCHECK(window);
+
+ // Windows with the always-on-top property should be free-floating and thus
+ // not managed by us.
+ if (window->GetProperty(aura::client::kAlwaysOnTopKey))
+ return false;
+
+ // If the changing bounds in the maximized/fullscreen is allowed, then
+ // let the client manage it even in tablet mode.
+ if (wm::GetWindowState(window)->allow_set_bounds_direct())
+ return false;
+
+ return window->type() == aura::client::WINDOW_TYPE_NORMAL;
+}
+
+void TabletModeWindowManager::AddWindowCreationObservers() {
+ DCHECK(observed_container_windows_.empty());
+ // Observe window activations/creations in the default containers on all root
+ // windows.
+ for (aura::Window* root : Shell::GetAllRootWindows()) {
+ aura::Window* default_container =
+ root->GetChildById(kShellWindowId_DefaultContainer);
+ DCHECK(!base::ContainsKey(observed_container_windows_, default_container));
+ default_container->AddObserver(this);
+ observed_container_windows_.insert(default_container);
+ }
+}
+
+void TabletModeWindowManager::RemoveWindowCreationObservers() {
+ for (aura::Window* window : observed_container_windows_)
+ window->RemoveObserver(this);
+ observed_container_windows_.clear();
+}
+
+void TabletModeWindowManager::DisplayConfigurationChanged() {
+ EnableBackdropBehindTopWindowOnEachDisplay(false);
+ RemoveWindowCreationObservers();
+ AddWindowCreationObservers();
+ EnableBackdropBehindTopWindowOnEachDisplay(true);
+}
+
+bool TabletModeWindowManager::IsContainerWindow(aura::Window* window) {
+ return base::ContainsKey(observed_container_windows_, window);
+}
+
+void TabletModeWindowManager::EnableBackdropBehindTopWindowOnEachDisplay(
+ bool enable) {
+ // Inform the WorkspaceLayoutManager that we want to show a backdrop behind
+ // the topmost window of its container.
+ for (auto* controller : Shell::GetAllRootWindowControllers()) {
+ controller->workspace_controller()->SetBackdropDelegate(
+ enable ? base::MakeUnique<TabletModeBackdropDelegateImpl>() : nullptr);
+ }
+}
+
+} // namespace ash
« no previous file with comments | « ash/wm/tablet_mode/tablet_mode_window_manager.h ('k') | ash/wm/tablet_mode/tablet_mode_window_manager_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698