blob: a3878ef1193614da02672d4be574439c7b020c81 [file] [log] [blame]
// Copyright 2014 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 "remoting/host/chromeos/mouse_cursor_monitor_aura.h"
#include <utility>
#include "ash/shell.h"
#include "base/bind.h"
#include "base/callback.h"
#include "base/location.h"
#include "remoting/host/chromeos/skia_bitmap_desktop_frame.h"
#include "third_party/webrtc/modules/desktop_capture/mouse_cursor.h"
#include "ui/aura/env.h"
#include "ui/aura/window.h"
#include "ui/aura/window_tree_host.h"
#include "ui/base/cursor/cursor_lookup.h"
#include "ui/base/cursor/mojom/cursor_type.mojom-shared.h"
namespace {
// Creates an empty webrtc::MouseCursor. The caller is responsible for
// destroying the returned cursor.
webrtc::MouseCursor* CreateEmptyMouseCursor() {
return new webrtc::MouseCursor(
new webrtc::BasicDesktopFrame(webrtc::DesktopSize(0, 0)),
webrtc::DesktopVector(0, 0));
}
} // namespace
namespace remoting {
MouseCursorMonitorAura::MouseCursorMonitorAura()
: callback_(nullptr),
mode_(SHAPE_AND_POSITION) {
}
void MouseCursorMonitorAura::Init(Callback* callback, Mode mode) {
DCHECK(!callback_);
DCHECK(callback);
callback_ = callback;
mode_ = mode;
}
void MouseCursorMonitorAura::Capture() {
// Check if the cursor is different.
gfx::NativeCursor cursor =
ash::Shell::GetPrimaryRootWindow()->GetHost()->last_cursor();
if (cursor != last_cursor_) {
last_cursor_ = cursor;
NotifyCursorChanged(cursor);
}
// Check if we need to update the location.
if (mode_ == SHAPE_AND_POSITION) {
gfx::Point position = aura::Env::GetInstance()->last_mouse_location();
if (position != last_mouse_location_) {
last_mouse_location_ = position;
callback_->OnMouseCursorPosition(
webrtc::DesktopVector(position.x(), position.y()));
}
}
}
void MouseCursorMonitorAura::NotifyCursorChanged(const ui::Cursor& cursor) {
if (cursor.type() == ui::mojom::CursorType::kNone) {
callback_->OnMouseCursor(CreateEmptyMouseCursor());
return;
}
std::unique_ptr<SkBitmap> cursor_bitmap =
std::make_unique<SkBitmap>(GetCursorBitmap(cursor));
gfx::Point cursor_hotspot = GetCursorHotspot(cursor);
if (cursor_bitmap->isNull()) {
LOG(ERROR) << "Failed to load bitmap for cursor type:"
<< static_cast<int>(cursor.type());
callback_->OnMouseCursor(CreateEmptyMouseCursor());
return;
}
std::unique_ptr<webrtc::DesktopFrame> image(
SkiaBitmapDesktopFrame::Create(std::move(cursor_bitmap)));
std::unique_ptr<webrtc::MouseCursor> cursor_shape(new webrtc::MouseCursor(
image.release(),
webrtc::DesktopVector(cursor_hotspot.x(), cursor_hotspot.y())));
callback_->OnMouseCursor(cursor_shape.release());
}
} // namespace remoting