1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152
|
// 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/display/unified_mouse_warp_controller.h"
#include <cmath>
#include "ash/display/display_util.h"
#include "ash/display/mirror_window_controller.h"
#include "ash/display/window_tree_host_manager.h"
#include "ash/host/ash_window_tree_host.h"
#include "ash/shell.h"
#include "ui/aura/client/cursor_client.h"
#include "ui/aura/window.h"
#include "ui/aura/window_tree_host.h"
#include "ui/display/display_finder.h"
#include "ui/display/display_layout.h"
#include "ui/display/manager/display_manager.h"
#include "ui/display/manager/util/display_manager_util.h"
#include "ui/display/screen.h"
#include "ui/events/event_utils.h"
#include "ui/gfx/geometry/transform.h"
#include "ui/wm/core/coordinate_conversion.h"
namespace ash {
namespace {
AshWindowTreeHost* GetMirroringAshWindowTreeHostForDisplayId(
int64_t display_id) {
return Shell::Get()
->window_tree_host_manager()
->mirror_window_controller()
->GetAshWindowTreeHostForDisplayId(display_id);
}
const aura::WindowTreeHost* GetMirroringSourceHostForCurrentEvent() {
return Shell::Get()
->window_tree_host_manager()
->mirror_window_controller()
->current_event_targeter_src_host();
}
} // namespace
UnifiedMouseWarpController::UnifiedMouseWarpController()
: update_location_for_test_(false), display_boundaries_computed_(false) {}
UnifiedMouseWarpController::~UnifiedMouseWarpController() = default;
bool UnifiedMouseWarpController::WarpMouseCursor(ui::MouseEvent* event) {
// Mirroring windows are created asynchronously, so compute the edge
// beounds when we received an event instead of in constructor.
if (!display_boundaries_computed_)
ComputeBounds();
aura::Window* target = static_cast<aura::Window*>(event->target());
gfx::Point point_in_unified_host = event->location();
::wm::ConvertPointToScreen(target, &point_in_unified_host);
// The display bounds of the mirroring windows isn't scaled, so
// transform back to the host coordinates.
point_in_unified_host =
target->GetHost()->GetRootTransform().MapPoint(point_in_unified_host);
// A native event may not exist in unit test.
if (!event->HasNativeEvent())
return false;
// TODO(dnicoara): crbug.com/415680 Move cursor warping into Ozone once Ozone
// has access to the logical display layout.
// Native events in Ozone are in the native window coordinate system. We need
// to translate them to get the global position.
const auto* host = GetMirroringSourceHostForCurrentEvent();
if (!host)
return false;
gfx::Point point_in_native =
ui::EventSystemLocationFromNative(event->native_event());
point_in_native.Offset(host->GetBoundsInPixels().x(),
host->GetBoundsInPixels().y());
// TODO(afakhry): Remove implicit grab. crbug.com/773348.
const display::Display display =
display::Screen::GetScreen()->GetDisplayNearestWindow(
const_cast<aura::Window*>(host->window()));
return WarpMouseCursorInNativeCoords(display.id(), point_in_native,
point_in_unified_host,
update_location_for_test_);
}
void UnifiedMouseWarpController::SetEnabled(bool enabled) {
// Mouse warp should be always on in Unified mode.
}
void UnifiedMouseWarpController::ComputeBounds() {
display::Displays display_list =
Shell::Get()->display_manager()->software_mirroring_display_list();
if (display_list.size() < 2) {
LOG(ERROR) << "Mirroring Display lost during re-configuration";
return;
}
for (size_t i = 0; i < display_list.size() - 1; ++i) {
const display::Display& first = display_list[i];
for (size_t j = i + 1; j < display_list.size(); ++j) {
const display::Display& second = display_list[j];
gfx::Rect first_edge;
gfx::Rect second_edge;
if (display::ComputeBoundary(first, second, &first_edge, &second_edge)) {
first_edge = GetNativeEdgeBounds(
GetMirroringAshWindowTreeHostForDisplayId(first.id()), first_edge);
second_edge = GetNativeEdgeBounds(
GetMirroringAshWindowTreeHostForDisplayId(second.id()),
second_edge);
displays_edges_map_[first.id()].emplace_back(first.id(), second.id(),
first_edge);
displays_edges_map_[second.id()].emplace_back(second.id(), first.id(),
second_edge);
}
}
}
display_boundaries_computed_ = true;
}
bool UnifiedMouseWarpController::WarpMouseCursorInNativeCoords(
int64_t source_display,
const gfx::Point& point_in_native,
const gfx::Point& point_in_unified_host,
bool update_mouse_location_now) {
const auto edges_iter = displays_edges_map_.find(source_display);
if (edges_iter == displays_edges_map_.end())
return false;
const std::vector<DisplayEdge>& potential_edges = edges_iter->second;
for (const auto& edge : potential_edges) {
if (edge.edge_native_bounds_in_source_display.Contains(point_in_native)) {
AshWindowTreeHost* target_ash_host =
GetMirroringAshWindowTreeHostForDisplayId(edge.target_display_id);
MoveCursorTo(target_ash_host, point_in_unified_host,
update_mouse_location_now);
return true;
}
}
return false;
}
} // namespace ash
|