| // Copyright 2015 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "ash/host/ash_window_tree_host_unified.h" |
| |
| #include <memory> |
| #include <tuple> |
| #include <utility> |
| |
| #include "ash/host/ash_window_tree_host_delegate.h" |
| #include "ash/host/root_window_transformer.h" |
| #include "base/check.h" |
| #include "base/containers/contains.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/notreached.h" |
| #include "base/ranges/algorithm.h" |
| #include "base/types/cxx23_to_underlying.h" |
| #include "ui/aura/window.h" |
| #include "ui/aura/window_event_dispatcher.h" |
| #include "ui/aura/window_targeter.h" |
| #include "ui/compositor/compositor.h" |
| #include "ui/gfx/geometry/insets.h" |
| #include "ui/platform_window/stub/stub_window.h" |
| |
| namespace ash { |
| |
| class UnifiedEventTargeter : public aura::WindowTargeter { |
| public: |
| UnifiedEventTargeter(aura::Window* src_root, |
| aura::Window* dst_root, |
| AshWindowTreeHostDelegate* delegate) |
| : src_root_(src_root), dst_root_(dst_root), delegate_(delegate) { |
| DCHECK(delegate); |
| } |
| |
| UnifiedEventTargeter(const UnifiedEventTargeter&) = delete; |
| UnifiedEventTargeter& operator=(const UnifiedEventTargeter&) = delete; |
| ~UnifiedEventTargeter() override { delegate_ = nullptr; } |
| |
| // aura::WindowTargeter: |
| ui::EventTarget* FindTargetForEvent(ui::EventTarget* root, |
| ui::Event* event) override { |
| delegate_->SetCurrentEventTargeterSourceHost(nullptr); |
| if (root == src_root_ && !event->target()) { |
| return root; |
| } else { |
| NOTREACHED() << "event type:" << base::to_underlying(event->type()); |
| } |
| } |
| ui::EventSink* GetNewEventSinkForEvent(const ui::EventTarget* current_root, |
| ui::EventTarget* target, |
| ui::Event* in_out_event) override { |
| if (current_root == src_root_ && !in_out_event->target()) { |
| delegate_->SetCurrentEventTargeterSourceHost(src_root_->GetHost()); |
| if (in_out_event->IsLocatedEvent()) { |
| ui::LocatedEvent* located_event = in_out_event->AsLocatedEvent(); |
| located_event->ConvertLocationToTarget( |
| static_cast<aura::Window*>(nullptr), dst_root_.get()); |
| } |
| return dst_root_->GetHost()->GetEventSink(); |
| } |
| return nullptr; |
| } |
| |
| private: |
| raw_ptr<aura::Window> src_root_; |
| raw_ptr<aura::Window> dst_root_; |
| raw_ptr<AshWindowTreeHostDelegate> delegate_; // Not owned. |
| }; |
| |
| AshWindowTreeHostUnified::AshWindowTreeHostUnified( |
| const gfx::Rect& initial_bounds, |
| AshWindowTreeHostDelegate* delegate, |
| size_t compositor_memory_limit_mb) |
| : AshWindowTreeHostPlatform( |
| std::make_unique<ui::StubWindow>(initial_bounds), |
| delegate, |
| compositor_memory_limit_mb) { |
| ui::StubWindow* stub_window = static_cast<ui::StubWindow*>(platform_window()); |
| stub_window->InitDelegate(this, true); |
| // TODO(b/356098565): Remove the log once the issue is fixed. |
| LOG(ERROR) << "Creating Unified Desktop bounds=" << initial_bounds.ToString(); |
| } |
| |
| AshWindowTreeHostUnified::~AshWindowTreeHostUnified() { |
| for (ash::AshWindowTreeHost* ash_host : mirroring_hosts_) { |
| ash_host->AsWindowTreeHost()->window()->RemoveObserver(this); |
| } |
| } |
| |
| void AshWindowTreeHostUnified::PrepareForShutdown() { |
| AshWindowTreeHostPlatform::PrepareForShutdown(); |
| |
| for (ash::AshWindowTreeHost* host : mirroring_hosts_) { |
| host->PrepareForShutdown(); |
| } |
| } |
| |
| void AshWindowTreeHostUnified::RegisterMirroringHost( |
| AshWindowTreeHost* mirroring_ash_host) { |
| aura::Window* src_root = mirroring_ash_host->AsWindowTreeHost()->window(); |
| src_root->SetEventTargeter( |
| std::make_unique<UnifiedEventTargeter>(src_root, window(), delegate_)); |
| DCHECK(!base::Contains(mirroring_hosts_, mirroring_ash_host)); |
| mirroring_hosts_.push_back(mirroring_ash_host); |
| src_root->AddObserver(this); |
| mirroring_ash_host->UpdateCursorConfig(); |
| } |
| |
| // Do nothing, since mirroring hosts had their cursor config updated when they |
| // were registered. |
| void AshWindowTreeHostUnified::UpdateCursorConfig() {} |
| |
| void AshWindowTreeHostUnified::ClearCursorConfig() { |
| for (ash::AshWindowTreeHost* host : mirroring_hosts_) { |
| host->ClearCursorConfig(); |
| } |
| } |
| |
| void AshWindowTreeHostUnified::SetCursorNative(gfx::NativeCursor cursor) { |
| for (ash::AshWindowTreeHost* host : mirroring_hosts_) { |
| host->AsWindowTreeHost()->SetCursor(cursor); |
| } |
| } |
| |
| void AshWindowTreeHostUnified::OnCursorVisibilityChangedNative(bool show) { |
| for (ash::AshWindowTreeHost* host : mirroring_hosts_) { |
| host->AsWindowTreeHost()->OnCursorVisibilityChanged(show); |
| } |
| } |
| |
| void AshWindowTreeHostUnified::OnBoundsChanged(const BoundsChange& change) { |
| if (platform_window()) |
| OnHostResizedInPixels(platform_window()->GetBoundsInPixels().size()); |
| } |
| |
| void AshWindowTreeHostUnified::OnWindowDestroying(aura::Window* window) { |
| auto iter = base::ranges::find( |
| mirroring_hosts_, window, [](AshWindowTreeHost* ash_host) { |
| return ash_host->AsWindowTreeHost()->window(); |
| }); |
| DCHECK(iter != mirroring_hosts_.end()); |
| window->RemoveObserver(this); |
| mirroring_hosts_.erase(iter); |
| } |
| |
| } // namespace ash |