// 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/shell.h" #include #include "ash/app_list/app_list.h" #include "ash/ash_switches.h" #include "ash/drag_drop/drag_drop_controller.h" #include "ash/focus_cycler.h" #include "ash/ime/input_method_event_filter.h" #include "ash/launcher/launcher.h" #include "ash/shell_delegate.h" #include "ash/shell_factory.h" #include "ash/shell_window_ids.h" #include "ash/system/audio/tray_volume.h" #include "ash/system/brightness/tray_brightness.h" #include "ash/system/settings/tray_settings.h" #include "ash/system/power/power_status_controller.h" #include "ash/system/power/tray_power_date.h" #include "ash/system/tray/system_tray_delegate.h" #include "ash/system/tray/system_tray.h" #include "ash/system/tray/tray_empty.h" #include "ash/system/user/tray_user.h" #include "ash/tooltips/tooltip_controller.h" #include "ash/wm/activation_controller.h" #include "ash/wm/base_layout_manager.h" #include "ash/wm/compact_layout_manager.h" #include "ash/wm/compact_status_area_layout_manager.h" #include "ash/wm/custom_frame_view_ash.h" #include "ash/wm/dialog_frame_view.h" #include "ash/wm/panel_window_event_filter.h" #include "ash/wm/panel_layout_manager.h" #include "ash/wm/partial_screenshot_event_filter.h" #include "ash/wm/power_button_controller.h" #include "ash/wm/root_window_event_filter.h" #include "ash/wm/root_window_layout_manager.h" #include "ash/wm/shadow_controller.h" #include "ash/wm/shelf_layout_manager.h" #include "ash/wm/stacking_controller.h" #include "ash/wm/status_area_layout_manager.h" #include "ash/wm/system_modal_container_layout_manager.h" #include "ash/wm/toplevel_layout_manager.h" #include "ash/wm/toplevel_window_event_filter.h" #include "ash/wm/video_detector.h" #include "ash/wm/visibility_controller.h" #include "ash/wm/window_cycle_controller.h" #include "ash/wm/window_modality_controller.h" #include "ash/wm/window_util.h" #include "ash/wm/workspace_controller.h" #include "ash/wm/workspace/workspace_event_filter.h" #include "ash/wm/workspace/workspace_layout_manager.h" #include "ash/wm/workspace/workspace_manager.h" #include "base/bind.h" #include "base/command_line.h" #include "third_party/skia/include/core/SkBitmap.h" #include "ui/aura/client/aura_constants.h" #include "ui/aura/layout_manager.h" #include "ui/aura/root_window.h" #include "ui/aura/window.h" #include "ui/gfx/compositor/layer.h" #include "ui/gfx/compositor/layer_animator.h" #include "ui/gfx/screen.h" #include "ui/gfx/size.h" #include "ui/views/widget/native_widget_aura.h" #include "ui/views/widget/widget.h" #if !defined(OS_MACOSX) #include "ash/accelerators/accelerator_controller.h" #include "ash/accelerators/accelerator_filter.h" #include "ash/accelerators/nested_dispatcher_controller.h" #endif namespace ash { namespace { using aura::Window; using views::Widget; // Creates a new window for use as a container. aura::Window* CreateContainer(int window_id, aura::Window* parent) { aura::Window* container = new aura::Window(NULL); container->set_id(window_id); container->Init(ui::Layer::LAYER_NOT_DRAWN); parent->AddChild(container); if (window_id != internal::kShellWindowId_UnparentedControlContainer) container->Show(); return container; } // Creates each of the special window containers that holds windows of various // types in the shell UI. void CreateSpecialContainers(aura::Window* root_window) { // These containers are just used by PowerButtonController to animate groups // of containers simultaneously without messing up the current transformations // on those containers. These are direct children of the root window; all of // the other containers are their children. aura::Window* non_lock_screen_containers = CreateContainer( internal::kShellWindowId_NonLockScreenContainersContainer, root_window); aura::Window* lock_screen_containers = CreateContainer( internal::kShellWindowId_LockScreenContainersContainer, root_window); aura::Window* lock_screen_related_containers = CreateContainer( internal::kShellWindowId_LockScreenRelatedContainersContainer, root_window); CreateContainer(internal::kShellWindowId_UnparentedControlContainer, non_lock_screen_containers); CreateContainer(internal::kShellWindowId_DesktopBackgroundContainer, non_lock_screen_containers); aura::Window* default_container = CreateContainer( internal::kShellWindowId_DefaultContainer, non_lock_screen_containers); default_container->SetEventFilter( new ToplevelWindowEventFilter(default_container)); SetChildWindowVisibilityChangesAnimated(default_container); aura::Window* always_on_top_container = CreateContainer( internal::kShellWindowId_AlwaysOnTopContainer, non_lock_screen_containers); always_on_top_container->SetEventFilter( new ToplevelWindowEventFilter(always_on_top_container)); SetChildWindowVisibilityChangesAnimated(always_on_top_container); aura::Window* panel_container = CreateContainer( internal::kShellWindowId_PanelContainer, non_lock_screen_containers); if (CommandLine::ForCurrentProcess()-> HasSwitch(switches::kAuraPanelManager)) { internal::PanelLayoutManager* layout_manager = new internal::PanelLayoutManager(panel_container); panel_container->SetEventFilter( new internal::PanelWindowEventFilter(panel_container, layout_manager)); panel_container->SetLayoutManager(layout_manager); } CreateContainer(internal::kShellWindowId_LauncherContainer, non_lock_screen_containers); aura::Window* modal_container = CreateContainer( internal::kShellWindowId_SystemModalContainer, non_lock_screen_containers); modal_container->SetEventFilter( new ToplevelWindowEventFilter(modal_container)); modal_container->SetLayoutManager( new internal::SystemModalContainerLayoutManager(modal_container)); SetChildWindowVisibilityChangesAnimated(modal_container); // TODO(beng): Figure out if we can make this use // SystemModalContainerEventFilter instead of stops_event_propagation. aura::Window* lock_container = CreateContainer( internal::kShellWindowId_LockScreenContainer, lock_screen_containers); lock_container->SetLayoutManager(new internal::BaseLayoutManager); lock_container->set_stops_event_propagation(true); aura::Window* lock_modal_container = CreateContainer( internal::kShellWindowId_LockSystemModalContainer, lock_screen_containers); lock_modal_container->SetEventFilter( new ToplevelWindowEventFilter(lock_modal_container)); lock_modal_container->SetLayoutManager( new internal::SystemModalContainerLayoutManager(lock_modal_container)); SetChildWindowVisibilityChangesAnimated(lock_modal_container); CreateContainer(internal::kShellWindowId_StatusContainer, lock_screen_related_containers); aura::Window* menu_container = CreateContainer( internal::kShellWindowId_MenuContainer, lock_screen_related_containers); SetChildWindowVisibilityChangesAnimated(menu_container); aura::Window* drag_drop_container = CreateContainer( internal::kShellWindowId_DragImageAndTooltipContainer, lock_screen_related_containers); SetChildWindowVisibilityChangesAnimated(drag_drop_container); CreateContainer(internal::kShellWindowId_SettingBubbleContainer, lock_screen_related_containers); CreateContainer(internal::kShellWindowId_OverlayContainer, lock_screen_related_containers); } class DummySystemTrayDelegate : public SystemTrayDelegate { public: DummySystemTrayDelegate() : muted_(false), volume_(0.5) { } virtual ~DummySystemTrayDelegate() {} private: // Overridden from SystemTrayDelegate: virtual const std::string GetUserDisplayName() const OVERRIDE { return "chronos"; } virtual const std::string GetUserEmail() const OVERRIDE { return "chr@nos"; } virtual const SkBitmap& GetUserImage() const OVERRIDE { return null_image_; } virtual user::LoginStatus GetUserLoginStatus() const OVERRIDE { return user::LOGGED_IN_USER; } virtual void ShowSettings() OVERRIDE { } virtual void ShowHelp() OVERRIDE { } virtual bool IsAudioMuted() const OVERRIDE { return muted_; } virtual void SetAudioMuted(bool muted) OVERRIDE { muted_ = muted; } virtual float GetVolumeLevel() const OVERRIDE { return volume_; } virtual void SetVolumeLevel(float volume) OVERRIDE { volume_ = volume; } virtual void ShutDown() OVERRIDE {} virtual void SignOut() OVERRIDE {} virtual void LockScreen() OVERRIDE {} bool muted_; float volume_; SkBitmap null_image_; DISALLOW_COPY_AND_ASSIGN(DummySystemTrayDelegate); }; } // namespace // static Shell* Shell::instance_ = NULL; // static bool Shell::initially_hide_cursor_ = false; //////////////////////////////////////////////////////////////////////////////// // Shell::TestApi Shell::TestApi::TestApi(Shell* shell) : shell_(shell) {} Shell::WindowMode Shell::TestApi::ComputeWindowMode(CommandLine* cmd) const { return shell_->ComputeWindowMode(cmd); } internal::RootWindowLayoutManager* Shell::TestApi::root_window_layout() { return shell_->root_window_layout_; } internal::InputMethodEventFilter* Shell::TestApi::input_method_event_filter() { return shell_->input_method_filter_.get(); } internal::WorkspaceController* Shell::TestApi::workspace_controller() { return shell_->workspace_controller_.get(); } //////////////////////////////////////////////////////////////////////////////// // Shell, public: Shell::Shell(ShellDelegate* delegate) : root_window_(new aura::RootWindow), delegate_(delegate), audio_controller_(NULL), brightness_controller_(NULL), shelf_(NULL), window_mode_(MODE_MANAGED), desktop_background_mode_(BACKGROUND_IMAGE), root_window_layout_(NULL), status_widget_(NULL) { } Shell::~Shell() { RemoveRootWindowEventFilter(partial_screenshot_filter_.get()); RemoveRootWindowEventFilter(input_method_filter_.get()); RemoveRootWindowEventFilter(window_modality_controller_.get()); #if !defined(OS_MACOSX) RemoveRootWindowEventFilter(accelerator_filter_.get()); #endif // Close background widget now so that the focus manager of the // widget gets deleted in the final message loop run. root_window_layout_->SetBackgroundWidget(NULL); // TooltipController is deleted with the Shell so removing its references. RemoveRootWindowEventFilter(tooltip_controller_.get()); aura::client::SetTooltipClient(GetRootWindow(), NULL); // Make sure we delete WorkspaceController before launcher is // deleted as it has a reference to launcher model. workspace_controller_.reset(); // The system tray needs to be reset before all the windows are destroyed. tray_.reset(); // Delete containers now so that child windows does not access // observers when they are destructed. aura::RootWindow* root_window = GetRootWindow(); while (!root_window->children().empty()) { aura::Window* child = root_window->children()[0]; delete child; } // These need a valid Shell instance to clean up properly, so explicitly // delete them before invalidating the instance. // Alphabetical. activation_controller_.reset(); drag_drop_controller_.reset(); shadow_controller_.reset(); window_cycle_controller_.reset(); // Launcher widget has a InputMethodBridge that references to // input_method_filter_'s input_method_. So explicitly release launcher_ // before input_method_filter_. And this needs to be after we delete all // containers in case there are still live browser windows which access // LauncherModel during close. launcher_.reset(); DCHECK(instance_ == this); instance_ = NULL; } // static Shell* Shell::CreateInstance(ShellDelegate* delegate) { CHECK(!instance_); instance_ = new Shell(delegate); instance_->Init(); return instance_; } // static Shell* Shell::GetInstance() { DCHECK(instance_); return instance_; } // static bool Shell::HasInstance() { return !!instance_; } // static void Shell::DeleteInstance() { delete instance_; instance_ = NULL; } // static aura::RootWindow* Shell::GetRootWindow() { return GetInstance()->root_window_.get(); } void Shell::Init() { root_filter_ = new internal::RootWindowEventFilter; #if !defined(OS_MACOSX) nested_dispatcher_controller_.reset(new NestedDispatcherController); accelerator_controller_.reset(new AcceleratorController); #endif // Pass ownership of the filter to the root window. GetRootWindow()->SetEventFilter(root_filter_); DCHECK(!GetRootWindowEventFilterCount()); // PartialScreenshotEventFilter must be the first one to capture key // events when the taking partial screenshot UI is there. partial_screenshot_filter_.reset(new internal::PartialScreenshotEventFilter); AddRootWindowEventFilter(partial_screenshot_filter_.get()); // Then AcceleratorFilter and InputMethodEventFilter must be added (in this // order) since they have the second highest priority. DCHECK_EQ(1U, GetRootWindowEventFilterCount()); #if !defined(OS_MACOSX) accelerator_filter_.reset(new internal::AcceleratorFilter); AddRootWindowEventFilter(accelerator_filter_.get()); DCHECK_EQ(2U, GetRootWindowEventFilterCount()); #endif input_method_filter_.reset(new internal::InputMethodEventFilter); AddRootWindowEventFilter(input_method_filter_.get()); // Window mode must be set before computing containers or layout managers. CommandLine* command_line = CommandLine::ForCurrentProcess(); if (!delegate_.get() || !delegate_->GetOverrideWindowMode(&window_mode_)) window_mode_ = ComputeWindowMode(command_line); aura::RootWindow* root_window = GetRootWindow(); root_window->SetCursor(aura::kCursorPointer); if (initially_hide_cursor_) root_window->ShowCursor(false); activation_controller_.reset(new internal::ActivationController); CreateSpecialContainers(root_window); stacking_controller_.reset(new internal::StackingController); root_window_layout_ = new internal::RootWindowLayoutManager(root_window); root_window->SetLayoutManager(root_window_layout_); if (delegate_.get()) status_widget_ = delegate_->CreateStatusArea(); if (command_line->HasSwitch(switches::kAshUberTray)) { // TODO(sad): This is rather ugly at the moment. This is because we are // supporting both the old and the new status bar at the same time. This // will soon get better once the new one is ready and the old one goes out // the door. tray_.reset(new SystemTray()); if (status_widget_) { status_widget_->GetContentsView()->RemoveAllChildViews(false); status_widget_->GetContentsView()->AddChildView(tray_.get()); } if (delegate_.get()) tray_delegate_.reset(delegate_->CreateSystemTrayDelegate(tray_.get())); if (!tray_delegate_.get()) tray_delegate_.reset(new DummySystemTrayDelegate()); internal::TrayVolume* tray_volume = new internal::TrayVolume(); internal::TrayBrightness* tray_brightness = new internal::TrayBrightness(); internal::TrayPowerDate* tray_power_date = new internal::TrayPowerDate(); audio_controller_ = tray_volume; brightness_controller_ = tray_brightness; power_status_controller_ = tray_power_date; tray_->AddTrayItem(new internal::TrayUser()); tray_->AddTrayItem(new internal::TrayEmpty()); tray_->AddTrayItem(tray_power_date); tray_->AddTrayItem(tray_volume); tray_->AddTrayItem(tray_brightness); tray_->AddTrayItem(new internal::TraySettings()); } if (!status_widget_) status_widget_ = internal::CreateStatusArea(tray_.get()); aura::Window* default_container = GetContainer(internal::kShellWindowId_DefaultContainer); launcher_.reset(new Launcher(default_container)); if (window_mode_ == MODE_COMPACT) SetupCompactWindowMode(); else SetupManagedWindowMode(); if (!command_line->HasSwitch(switches::kAuraNoShadows)) shadow_controller_.reset(new internal::ShadowController()); focus_cycler_.reset(new internal::FocusCycler()); focus_cycler_->AddWidget(status_widget_); focus_cycler_->AddWidget(launcher_->widget()); launcher_->SetFocusCycler(focus_cycler_.get()); // Force a layout. root_window->layout_manager()->OnWindowResized(); window_modality_controller_.reset(new internal::WindowModalityController); AddRootWindowEventFilter(window_modality_controller_.get()); visibility_controller_.reset(new internal::VisibilityController); tooltip_controller_.reset(new internal::TooltipController); AddRootWindowEventFilter(tooltip_controller_.get()); drag_drop_controller_.reset(new internal::DragDropController); power_button_controller_.reset(new PowerButtonController); video_detector_.reset(new VideoDetector); window_cycle_controller_.reset(new WindowCycleController); } Shell::WindowMode Shell::ComputeWindowMode(CommandLine* command_line) const { // Some devices don't perform well with overlapping windows. if (command_line->HasSwitch(switches::kAuraForceCompactWindowMode)) return MODE_COMPACT; // If user set the flag, use their desired behavior. if (command_line->HasSwitch(switches::kAuraWindowMode)) { std::string mode = command_line->GetSwitchValueASCII(switches::kAuraWindowMode); if (mode == switches::kAuraWindowModeCompact) return MODE_COMPACT; if (mode == switches::kAuraWindowModeManaged) return MODE_MANAGED; } // Managed is the default. return Shell::MODE_MANAGED; } aura::Window* Shell::GetContainer(int container_id) { return const_cast( const_cast(this)->GetContainer(container_id)); } const aura::Window* Shell::GetContainer(int container_id) const { return GetRootWindow()->GetChildById(container_id); } void Shell::AddRootWindowEventFilter(aura::EventFilter* filter) { static_cast( GetRootWindow()->event_filter())->AddFilter(filter); } void Shell::RemoveRootWindowEventFilter(aura::EventFilter* filter) { static_cast( GetRootWindow()->event_filter())->RemoveFilter(filter); } size_t Shell::GetRootWindowEventFilterCount() const { return static_cast( GetRootWindow()->event_filter())->GetFilterCount(); } void Shell::ShowBackgroundMenu(views::Widget* widget, const gfx::Point& location) { if (workspace_controller_.get()) workspace_controller_->ShowMenu(widget, location); } void Shell::ToggleAppList() { if (!app_list_.get()) app_list_.reset(new internal::AppList); app_list_->SetVisible(!app_list_->IsVisible()); } void Shell::SetDesktopBackgroundMode(BackgroundMode mode) { if (mode == BACKGROUND_SOLID_COLOR) { // Set a solid black background. // TODO(derat): Remove this in favor of having the compositor only clear the // viewport when there are regions not covered by a layer: // http://crbug.com/113445 ui::Layer* background_layer = new ui::Layer(ui::Layer::LAYER_SOLID_COLOR); background_layer->SetColor(SK_ColorBLACK); GetContainer(internal::kShellWindowId_DesktopBackgroundContainer)-> layer()->Add(background_layer); root_window_layout_->SetBackgroundLayer(background_layer); root_window_layout_->SetBackgroundWidget(NULL); } else { // Create the desktop background image. root_window_layout_->SetBackgroundLayer(NULL); root_window_layout_->SetBackgroundWidget( internal::CreateDesktopBackground()); } desktop_background_mode_ = mode; } bool Shell::IsScreenLocked() const { const aura::Window* lock_screen_container = GetContainer( internal::kShellWindowId_LockScreenContainer); return lock_screen_container->StopsEventPropagation(); } bool Shell::IsModalWindowOpen() const { const aura::Window* modal_container = GetContainer( internal::kShellWindowId_SystemModalContainer); return !modal_container->children().empty(); } void Shell::SetCompactStatusAreaOffset(gfx::Size& offset) { compact_status_area_offset_ = offset; // Trigger a relayout. if (IsWindowModeCompact()) { aura::LayoutManager* layout_manager = GetContainer( internal::kShellWindowId_StatusContainer)->layout_manager(); if (layout_manager) layout_manager->OnWindowResized(); } } views::NonClientFrameView* Shell::CreateDefaultNonClientFrameView( views::Widget* widget) { if (CommandLine::ForCurrentProcess()->HasSwitch( switches::kAuraGoogleDialogFrames)) { return new internal::DialogFrameView; } if (CommandLine::ForCurrentProcess()->HasSwitch( switches::kAuraTranslucentFrames)) { internal::CustomFrameViewAsh* frame_view = new internal::CustomFrameViewAsh; frame_view->Init(widget); return frame_view; } return NULL; } void Shell::RotateFocus(Direction direction) { focus_cycler_->RotateFocus( direction == FORWARD ? internal::FocusCycler::FORWARD : internal::FocusCycler::BACKWARD); } //////////////////////////////////////////////////////////////////////////////// // Shell, private: // Compact mode has a simplified layout manager and doesn't use the shelf, // desktop background, etc. The launcher still exists so we can use its // data model and list of open windows, but we hide the UI to save space. void Shell::SetupCompactWindowMode() { DCHECK(root_window_layout_); DCHECK(status_widget_); // Don't use an event filter for the default container, as we don't support // window dragging, double-click to maximize, etc. aura::Window* default_container = GetContainer(internal::kShellWindowId_DefaultContainer); default_container->SetEventFilter(NULL); // Set up our new layout managers. internal::CompactLayoutManager* compact_layout_manager = new internal::CompactLayoutManager(); compact_layout_manager->set_status_area_widget(status_widget_); internal::CompactStatusAreaLayoutManager* status_area_layout_manager = new internal::CompactStatusAreaLayoutManager(status_widget_); GetContainer(internal::kShellWindowId_StatusContainer)-> SetLayoutManager(status_area_layout_manager); default_container->SetLayoutManager(compact_layout_manager); // Keep the launcher for its data model, but hide it. Do this before // maximizing the windows so the work area is the right size. launcher_->widget()->Hide(); // Set a solid black background. SetDesktopBackgroundMode(BACKGROUND_SOLID_COLOR); } void Shell::SetupManagedWindowMode() { DCHECK(root_window_layout_); DCHECK(status_widget_); internal::ShelfLayoutManager* shelf_layout_manager = new internal::ShelfLayoutManager(launcher_->widget(), status_widget_); GetContainer(internal::kShellWindowId_LauncherContainer)-> SetLayoutManager(shelf_layout_manager); shelf_ = shelf_layout_manager; internal::StatusAreaLayoutManager* status_area_layout_manager = new internal::StatusAreaLayoutManager(shelf_layout_manager); GetContainer(internal::kShellWindowId_StatusContainer)-> SetLayoutManager(status_area_layout_manager); aura::Window* default_container = GetContainer(internal::kShellWindowId_DefaultContainer); // Workspace manager has its own layout managers. workspace_controller_.reset( new internal::WorkspaceController(default_container)); workspace_controller_->workspace_manager()->set_shelf(shelf_layout_manager); // Ensure launcher is visible. launcher_->widget()->Show(); // Create the desktop background image. SetDesktopBackgroundMode(BACKGROUND_IMAGE); } void Shell::ResetLayoutManager(int container_id) { GetContainer(container_id)->SetLayoutManager(NULL); } void Shell::DisableWorkspaceGridLayout() { if (workspace_controller_.get()) workspace_controller_->workspace_manager()->set_grid_size(0); } } // namespace ash