Don't show supervised user as "already on this device" while they're being imported.
[chromium-blink-merge.git] / extensions / shell / browser / shell_desktop_controller_aura.cc
blob5ab8a3c8d25ab72aa9d97ae58e63bb7ee8d7b032
1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "extensions/shell/browser/shell_desktop_controller_aura.h"
7 #include <algorithm>
8 #include <string>
9 #include <vector>
11 #include "base/command_line.h"
12 #include "extensions/browser/app_window/app_window.h"
13 #include "extensions/browser/app_window/native_app_window.h"
14 #include "extensions/shell/browser/shell_app_delegate.h"
15 #include "extensions/shell/browser/shell_app_window_client.h"
16 #include "extensions/shell/browser/shell_screen.h"
17 #include "extensions/shell/common/switches.h"
18 #include "ui/aura/client/cursor_client.h"
19 #include "ui/aura/client/default_capture_client.h"
20 #include "ui/aura/layout_manager.h"
21 #include "ui/aura/window.h"
22 #include "ui/aura/window_event_dispatcher.h"
23 #include "ui/aura/window_tree_host.h"
24 #include "ui/base/cursor/cursor.h"
25 #include "ui/base/cursor/image_cursors.h"
26 #include "ui/base/ime/input_method_initializer.h"
27 #include "ui/base/user_activity/user_activity_detector.h"
28 #include "ui/gfx/geometry/size.h"
29 #include "ui/gfx/native_widget_types.h"
30 #include "ui/gfx/screen.h"
31 #include "ui/wm/core/base_focus_rules.h"
32 #include "ui/wm/core/compound_event_filter.h"
33 #include "ui/wm/core/cursor_manager.h"
34 #include "ui/wm/core/focus_controller.h"
35 #include "ui/wm/core/input_method_event_filter.h"
36 #include "ui/wm/core/native_cursor_manager.h"
37 #include "ui/wm/core/native_cursor_manager_delegate.h"
39 #if defined(OS_CHROMEOS)
40 #include "chromeos/dbus/dbus_thread_manager.h"
41 #include "ui/chromeos/user_activity_power_manager_notifier.h"
42 #include "ui/display/types/display_mode.h"
43 #include "ui/display/types/display_snapshot.h"
44 #endif
46 namespace extensions {
47 namespace {
49 // A simple layout manager that makes each new window fill its parent.
50 class FillLayout : public aura::LayoutManager {
51 public:
52 FillLayout() {}
53 ~FillLayout() override {}
55 private:
56 // aura::LayoutManager:
57 void OnWindowResized() override {}
59 void OnWindowAddedToLayout(aura::Window* child) override {
60 if (!child->parent())
61 return;
63 // Create a rect at 0,0 with the size of the parent.
64 gfx::Size parent_size = child->parent()->bounds().size();
65 child->SetBounds(gfx::Rect(parent_size));
68 void OnWillRemoveWindowFromLayout(aura::Window* child) override {}
70 void OnWindowRemovedFromLayout(aura::Window* child) override {}
72 void OnChildWindowVisibilityChanged(aura::Window* child,
73 bool visible) override {}
75 void SetChildBounds(aura::Window* child,
76 const gfx::Rect& requested_bounds) override {
77 SetChildBoundsDirect(child, requested_bounds);
80 DISALLOW_COPY_AND_ASSIGN(FillLayout);
83 // A class that bridges the gap between CursorManager and Aura. It borrows
84 // heavily from AshNativeCursorManager.
85 class ShellNativeCursorManager : public wm::NativeCursorManager {
86 public:
87 explicit ShellNativeCursorManager(aura::WindowTreeHost* host)
88 : host_(host), image_cursors_(new ui::ImageCursors) {}
89 ~ShellNativeCursorManager() override {}
91 // wm::NativeCursorManager overrides.
92 void SetDisplay(const gfx::Display& display,
93 wm::NativeCursorManagerDelegate* delegate) override {
94 if (image_cursors_->SetDisplay(display, display.device_scale_factor()))
95 SetCursor(delegate->GetCursor(), delegate);
98 void SetCursor(gfx::NativeCursor cursor,
99 wm::NativeCursorManagerDelegate* delegate) override {
100 image_cursors_->SetPlatformCursor(&cursor);
101 cursor.set_device_scale_factor(image_cursors_->GetScale());
102 delegate->CommitCursor(cursor);
104 if (delegate->IsCursorVisible())
105 ApplyCursor(cursor);
108 void SetVisibility(bool visible,
109 wm::NativeCursorManagerDelegate* delegate) override {
110 delegate->CommitVisibility(visible);
112 if (visible) {
113 SetCursor(delegate->GetCursor(), delegate);
114 } else {
115 gfx::NativeCursor invisible_cursor(ui::kCursorNone);
116 image_cursors_->SetPlatformCursor(&invisible_cursor);
117 ApplyCursor(invisible_cursor);
121 void SetCursorSet(ui::CursorSetType cursor_set,
122 wm::NativeCursorManagerDelegate* delegate) override {
123 image_cursors_->SetCursorSet(cursor_set);
124 delegate->CommitCursorSet(cursor_set);
125 if (delegate->IsCursorVisible())
126 SetCursor(delegate->GetCursor(), delegate);
129 void SetMouseEventsEnabled(
130 bool enabled,
131 wm::NativeCursorManagerDelegate* delegate) override {
132 delegate->CommitMouseEventsEnabled(enabled);
133 SetVisibility(delegate->IsCursorVisible(), delegate);
136 private:
137 // Sets |cursor| as the active cursor within Aura.
138 void ApplyCursor(gfx::NativeCursor cursor) { host_->SetCursor(cursor); }
140 aura::WindowTreeHost* host_; // Not owned.
142 scoped_ptr<ui::ImageCursors> image_cursors_;
144 DISALLOW_COPY_AND_ASSIGN(ShellNativeCursorManager);
147 class AppsFocusRules : public wm::BaseFocusRules {
148 public:
149 AppsFocusRules() {}
150 ~AppsFocusRules() override {}
152 bool SupportsChildActivation(aura::Window* window) const override {
153 return true;
156 private:
157 DISALLOW_COPY_AND_ASSIGN(AppsFocusRules);
160 } // namespace
162 ShellDesktopControllerAura::ShellDesktopControllerAura()
163 : app_window_client_(new ShellAppWindowClient) {
164 extensions::AppWindowClient::Set(app_window_client_.get());
166 #if defined(OS_CHROMEOS)
167 chromeos::DBusThreadManager::Get()->GetPowerManagerClient()->AddObserver(
168 this);
169 display_configurator_.reset(new ui::DisplayConfigurator);
170 display_configurator_->Init(false);
171 display_configurator_->ForceInitialConfigure(0);
172 display_configurator_->AddObserver(this);
173 #endif
174 CreateRootWindow();
177 ShellDesktopControllerAura::~ShellDesktopControllerAura() {
178 CloseAppWindows();
179 DestroyRootWindow();
180 #if defined(OS_CHROMEOS)
181 chromeos::DBusThreadManager::Get()->GetPowerManagerClient()->RemoveObserver(
182 this);
183 #endif
184 extensions::AppWindowClient::Set(NULL);
187 gfx::Size ShellDesktopControllerAura::GetWindowSize() {
188 return host_->window()->bounds().size();
191 AppWindow* ShellDesktopControllerAura::CreateAppWindow(
192 content::BrowserContext* context,
193 const Extension* extension) {
194 app_windows_.push_back(
195 new AppWindow(context, new ShellAppDelegate, extension));
196 return app_windows_.back();
199 void ShellDesktopControllerAura::AddAppWindow(gfx::NativeWindow window) {
200 aura::Window* root_window = host_->window();
201 root_window->AddChild(window);
204 void ShellDesktopControllerAura::RemoveAppWindow(AppWindow* window) {
205 auto iter = std::find(app_windows_.begin(), app_windows_.end(), window);
206 DCHECK(iter != app_windows_.end());
207 app_windows_.erase(iter);
210 void ShellDesktopControllerAura::CloseAppWindows() {
211 // Create a copy of the window vector, because closing the windows will
212 // trigger RemoveAppWindow, which will invalidate the iterator.
213 // This vector should be small enough that this should not be an issue.
214 std::vector<AppWindow*> app_windows(app_windows_);
215 for (AppWindow* app_window : app_windows)
216 app_window->GetBaseWindow()->Close(); // Close() deletes |app_window|.
217 app_windows_.clear();
220 aura::Window* ShellDesktopControllerAura::GetDefaultParent(
221 aura::Window* context,
222 aura::Window* window,
223 const gfx::Rect& bounds) {
224 return host_->window();
227 #if defined(OS_CHROMEOS)
228 void ShellDesktopControllerAura::PowerButtonEventReceived(
229 bool down,
230 const base::TimeTicks& timestamp) {
231 if (down) {
232 chromeos::DBusThreadManager::Get()
233 ->GetPowerManagerClient()
234 ->RequestShutdown();
238 void ShellDesktopControllerAura::OnDisplayModeChanged(
239 const ui::DisplayConfigurator::DisplayStateList& displays) {
240 gfx::Size size = GetPrimaryDisplaySize();
241 if (!size.IsEmpty())
242 host_->UpdateRootWindowSize(size);
244 #endif
246 void ShellDesktopControllerAura::OnHostCloseRequested(
247 const aura::WindowTreeHost* host) {
248 DCHECK_EQ(host_.get(), host);
249 CloseAppWindows();
250 base::MessageLoop::current()->PostTask(FROM_HERE,
251 base::MessageLoop::QuitClosure());
254 void ShellDesktopControllerAura::InitWindowManager() {
255 wm::FocusController* focus_controller =
256 new wm::FocusController(new AppsFocusRules());
257 aura::client::SetFocusClient(host_->window(), focus_controller);
258 host_->window()->AddPreTargetHandler(focus_controller);
259 aura::client::SetActivationClient(host_->window(), focus_controller);
260 focus_client_.reset(focus_controller);
262 input_method_filter_.reset(
263 new wm::InputMethodEventFilter(host_->GetAcceleratedWidget()));
264 input_method_filter_->SetInputMethodPropertyInRootWindow(host_->window());
265 root_window_event_filter_->AddHandler(input_method_filter_.get());
267 capture_client_.reset(
268 new aura::client::DefaultCaptureClient(host_->window()));
270 // Ensure new windows fill the display.
271 host_->window()->SetLayoutManager(new FillLayout);
273 cursor_manager_.reset(
274 new wm::CursorManager(scoped_ptr<wm::NativeCursorManager>(
275 new ShellNativeCursorManager(host_.get()))));
276 cursor_manager_->SetDisplay(
277 gfx::Screen::GetNativeScreen()->GetPrimaryDisplay());
278 cursor_manager_->SetCursor(ui::kCursorPointer);
279 aura::client::SetCursorClient(host_->window(), cursor_manager_.get());
281 user_activity_detector_.reset(new ui::UserActivityDetector);
282 #if defined(OS_CHROMEOS)
283 user_activity_notifier_.reset(
284 new ui::UserActivityPowerManagerNotifier(user_activity_detector_.get()));
285 #endif
288 void ShellDesktopControllerAura::CreateRootWindow() {
289 // Set up basic pieces of ui::wm.
290 gfx::Size size;
291 base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
292 if (command_line->HasSwitch(switches::kAppShellHostWindowSize)) {
293 const std::string size_str =
294 command_line->GetSwitchValueASCII(switches::kAppShellHostWindowSize);
295 int width, height;
296 CHECK_EQ(2, sscanf(size_str.c_str(), "%dx%d", &width, &height));
297 size = gfx::Size(width, height);
298 } else {
299 size = GetPrimaryDisplaySize();
301 if (size.IsEmpty())
302 size = gfx::Size(1920, 1080);
304 screen_.reset(new ShellScreen(size));
305 gfx::Screen::SetScreenInstance(gfx::SCREEN_TYPE_NATIVE, screen_.get());
306 // TODO(mukai): Set up input method.
308 host_.reset(screen_->CreateHostForPrimaryDisplay());
309 aura::client::SetWindowTreeClient(host_->window(), this);
310 root_window_event_filter_.reset(new wm::CompoundEventFilter);
311 host_->window()->AddPreTargetHandler(root_window_event_filter_.get());
312 InitWindowManager();
314 host_->AddObserver(this);
316 // Ensure the X window gets mapped.
317 host_->Show();
320 void ShellDesktopControllerAura::DestroyRootWindow() {
321 host_->RemoveObserver(this);
322 if (input_method_filter_)
323 root_window_event_filter_->RemoveHandler(input_method_filter_.get());
324 wm::FocusController* focus_controller =
325 static_cast<wm::FocusController*>(focus_client_.get());
326 if (focus_controller) {
327 host_->window()->RemovePreTargetHandler(focus_controller);
328 aura::client::SetActivationClient(host_->window(), NULL);
330 root_window_event_filter_.reset();
331 capture_client_.reset();
332 input_method_filter_.reset();
333 focus_client_.reset();
334 cursor_manager_.reset();
335 #if defined(OS_CHROMEOS)
336 user_activity_notifier_.reset();
337 #endif
338 user_activity_detector_.reset();
339 host_.reset();
340 screen_.reset();
343 gfx::Size ShellDesktopControllerAura::GetPrimaryDisplaySize() {
344 #if defined(OS_CHROMEOS)
345 const ui::DisplayConfigurator::DisplayStateList& displays =
346 display_configurator_->cached_displays();
347 if (displays.empty())
348 return gfx::Size();
349 const ui::DisplayMode* mode = displays[0]->current_mode();
350 return mode ? mode->size() : gfx::Size();
351 #else
352 return gfx::Size();
353 #endif
356 } // namespace extensions