// Copyright 2015 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 "mash/wm/user_window_controller_impl.h" #include "components/mus/public/cpp/property_type_converters.h" #include "components/mus/public/cpp/window.h" #include "components/mus/public/cpp/window_property.h" #include "components/mus/public/cpp/window_tree_connection.h" #include "mash/wm/public/interfaces/container.mojom.h" #include "mash/wm/root_window_controller.h" #include "mojo/common/common_type_converters.h" namespace mash { namespace wm { namespace { // Get the title property from a mus::Window. mojo::String GetWindowTitle(mus::Window* window) { if (window->HasSharedProperty( mus::mojom::WindowManager::kWindowTitle_Property)) { return mojo::String::From(window->GetSharedProperty( mus::mojom::WindowManager::kWindowTitle_Property)); } return mojo::String(std::string()); } // Returns |window|, or an ancestor thereof, parented to |container|, or null. mus::Window* GetTopLevelWindow(mus::Window* window, mus::Window* container) { while (window && window->parent() != container) window = window->parent(); return window; } // Get a UserWindow struct from a mus::Window. mojom::UserWindowPtr GetUserWindow(mus::Window* window) { mojom::UserWindowPtr user_window(mojom::UserWindow::New()); user_window->window_id = window->id(); user_window->window_title = GetWindowTitle(window); mus::Window* focused = window->connection()->GetFocusedWindow(); focused = GetTopLevelWindow(focused, window->parent()); user_window->window_has_focus = focused == window; return user_window; } } // namespace // Observes title changes on user windows. UserWindowControllerImpl uses this // separate observer to avoid observing duplicate tree change notifications. class WindowTitleObserver : public mus::WindowObserver { public: explicit WindowTitleObserver(UserWindowControllerImpl* controller) : controller_(controller) {} ~WindowTitleObserver() override {} private: // mus::WindowObserver: void OnWindowSharedPropertyChanged( mus::Window* window, const std::string& name, const std::vector* old_data, const std::vector* new_data) override { if (controller_->user_window_observer() && name == mus::mojom::WindowManager::kWindowTitle_Property) { controller_->user_window_observer()->OnUserWindowTitleChanged( window->id(), GetWindowTitle(window)); } } UserWindowControllerImpl* controller_; DISALLOW_COPY_AND_ASSIGN(WindowTitleObserver); }; UserWindowControllerImpl::UserWindowControllerImpl() : root_controller_(nullptr) {} UserWindowControllerImpl::~UserWindowControllerImpl() { if (!root_controller_) return; // TODO(msw): should really listen for user window container being destroyed // and cleanup there. mus::Window* user_container = GetUserWindowContainer(); if (!user_container) return; user_container->RemoveObserver(this); for (auto iter : user_container->children()) iter->RemoveObserver(window_title_observer_.get()); } void UserWindowControllerImpl::Initialize( RootWindowController* root_controller) { DCHECK(root_controller); DCHECK(!root_controller_); root_controller_ = root_controller; GetUserWindowContainer()->AddObserver(this); GetUserWindowContainer()->connection()->AddObserver(this); window_title_observer_.reset(new WindowTitleObserver(this)); for (auto iter : GetUserWindowContainer()->children()) iter->AddObserver(window_title_observer_.get()); } mus::Window* UserWindowControllerImpl::GetUserWindowContainer() const { return root_controller_->GetWindowForContainer( mojom::Container::USER_WINDOWS); } void UserWindowControllerImpl::OnTreeChanging(const TreeChangeParams& params) { DCHECK(root_controller_); if (params.new_parent == GetUserWindowContainer()) { params.target->AddObserver(window_title_observer_.get()); if (user_window_observer_) user_window_observer_->OnUserWindowAdded(GetUserWindow(params.target)); } else if (params.old_parent == GetUserWindowContainer()) { params.target->RemoveObserver(window_title_observer_.get()); if (user_window_observer_) user_window_observer_->OnUserWindowRemoved(params.target->id()); } } void UserWindowControllerImpl::OnWindowTreeFocusChanged( mus::Window* gained_focus, mus::Window* lost_focus) { if (!user_window_observer_) return; // Treat focus in the user window hierarchy as focus of the top-level window. gained_focus = GetTopLevelWindow(gained_focus, GetUserWindowContainer()); lost_focus = GetTopLevelWindow(lost_focus, GetUserWindowContainer()); if (gained_focus == lost_focus) return; if (lost_focus) user_window_observer_->OnUserWindowFocusChanged(lost_focus->id(), false); if (gained_focus) user_window_observer_->OnUserWindowFocusChanged(gained_focus->id(), true); } void UserWindowControllerImpl::AddUserWindowObserver( mojom::UserWindowObserverPtr observer) { // TODO(msw): Support multiple observers. user_window_observer_ = std::move(observer); const mus::Window::Children& windows = GetUserWindowContainer()->children(); mojo::Array user_windows = mojo::Array::New(windows.size()); for (size_t i = 0; i < windows.size(); ++i) user_windows[i] = GetUserWindow(windows[i]); user_window_observer_->OnUserWindowObserverAdded(std::move(user_windows)); } void UserWindowControllerImpl::FocusUserWindow(uint32_t window_id) { mus::Window* window = GetUserWindowContainer()->GetChildById(window_id); if (window) window->SetFocus(); } } // namespace wm } // namespace mash