// 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 "remoting/host/local_input_monitor.h" #import #include #include #include "base/bind.h" #include "base/compiler_specific.h" #include "base/location.h" #include "base/logging.h" #include "base/mac/scoped_cftyperef.h" #include "base/macros.h" #include "base/memory/ref_counted.h" #include "base/single_thread_task_runner.h" #include "base/synchronization/lock.h" #include "base/threading/non_thread_safe.h" #include "remoting/host/client_session_control.h" #import "third_party/google_toolbox_for_mac/src/AppKit/GTMCarbonEvent.h" #include "third_party/webrtc/modules/desktop_capture/desktop_geometry.h" // Esc Key Code is 53. // http://boredzo.org/blog/wp-content/uploads/2007/05/IMTx-virtual-keycodes.pdf static const NSUInteger kEscKeyCode = 53; namespace remoting { namespace { class LocalInputMonitorMac : public base::NonThreadSafe, public LocalInputMonitor { public: // Invoked by LocalInputMonitorManager. class EventHandler { public: virtual ~EventHandler() {} virtual void OnLocalMouseMoved(const webrtc::DesktopVector& position) = 0; virtual void OnDisconnectShortcut() = 0; }; LocalInputMonitorMac( scoped_refptr caller_task_runner, scoped_refptr ui_task_runner, base::WeakPtr client_session_control); ~LocalInputMonitorMac() override; private: // The actual implementation resides in LocalInputMonitorMac::Core class. class Core; scoped_refptr core_; DISALLOW_COPY_AND_ASSIGN(LocalInputMonitorMac); }; } // namespace } // namespace remoting @interface LocalInputMonitorManager : NSObject { @private GTMCarbonHotKey* hotKey_; CFRunLoopSourceRef mouseRunLoopSource_; base::ScopedCFTypeRef mouseMachPort_; remoting::LocalInputMonitorMac::EventHandler* monitor_; } - (id)initWithMonitor:(remoting::LocalInputMonitorMac::EventHandler*)monitor; // Called when the hotKey is hit. - (void)hotKeyHit:(GTMCarbonHotKey*)hotKey; // Called when the local mouse moves - (void)localMouseMoved:(const webrtc::DesktopVector&)mousePos; // Must be called when the LocalInputMonitorManager is no longer to be used. // Similar to NSTimer in that more than a simple release is required. - (void)invalidate; @end static CGEventRef LocalMouseMoved(CGEventTapProxy proxy, CGEventType type, CGEventRef event, void* context) { int64_t pid = CGEventGetIntegerValueField(event, kCGEventSourceUnixProcessID); if (pid == 0) { CGPoint cgMousePos = CGEventGetLocation(event); webrtc::DesktopVector mousePos(cgMousePos.x, cgMousePos.y); [static_cast(context) localMouseMoved:mousePos]; } return nullptr; } @implementation LocalInputMonitorManager - (id)initWithMonitor:(remoting::LocalInputMonitorMac::EventHandler*)monitor { if ((self = [super init])) { monitor_ = monitor; GTMCarbonEventDispatcherHandler* handler = [GTMCarbonEventDispatcherHandler sharedEventDispatcherHandler]; hotKey_ = [handler registerHotKey:kEscKeyCode modifiers:(NSAlternateKeyMask | NSControlKeyMask) target:self action:@selector(hotKeyHit:) userInfo:nil whenPressed:YES]; if (!hotKey_) { LOG(ERROR) << "registerHotKey failed."; } mouseMachPort_.reset(CGEventTapCreate( kCGSessionEventTap, kCGHeadInsertEventTap, kCGEventTapOptionListenOnly, 1 << kCGEventMouseMoved, LocalMouseMoved, self)); if (mouseMachPort_) { mouseRunLoopSource_ = CFMachPortCreateRunLoopSource( nullptr, mouseMachPort_, 0); CFRunLoopAddSource( CFRunLoopGetMain(), mouseRunLoopSource_, kCFRunLoopCommonModes); } else { LOG(ERROR) << "CGEventTapCreate failed."; } if (!hotKey_ && !mouseMachPort_) { [self release]; return nil; } } return self; } - (void)hotKeyHit:(GTMCarbonHotKey*)hotKey { monitor_->OnDisconnectShortcut(); } - (void)localMouseMoved:(const webrtc::DesktopVector&)mousePos { monitor_->OnLocalMouseMoved(mousePos); } - (void)invalidate { if (hotKey_) { GTMCarbonEventDispatcherHandler* handler = [GTMCarbonEventDispatcherHandler sharedEventDispatcherHandler]; [handler unregisterHotKey:hotKey_]; hotKey_ = nullptr; } if (mouseRunLoopSource_) { CFMachPortInvalidate(mouseMachPort_); CFRunLoopRemoveSource( CFRunLoopGetMain(), mouseRunLoopSource_, kCFRunLoopCommonModes); CFRelease(mouseRunLoopSource_); mouseMachPort_.reset(0); mouseRunLoopSource_ = nullptr; } } @end namespace remoting { namespace { class LocalInputMonitorMac::Core : public base::RefCountedThreadSafe, public EventHandler { public: Core(scoped_refptr caller_task_runner, scoped_refptr ui_task_runner, base::WeakPtr client_session_control); void Start(); void Stop(); private: friend class base::RefCountedThreadSafe; ~Core() override; void StartOnUiThread(); void StopOnUiThread(); // EventHandler interface. void OnLocalMouseMoved(const webrtc::DesktopVector& position) override; void OnDisconnectShortcut() override; // Task runner on which public methods of this class must be called. scoped_refptr caller_task_runner_; // Task runner on which |window_| is created. scoped_refptr ui_task_runner_; LocalInputMonitorManager* manager_; // Invoked in the |caller_task_runner_| thread to report local mouse events // and session disconnect requests. base::WeakPtr client_session_control_; webrtc::DesktopVector mouse_position_; DISALLOW_COPY_AND_ASSIGN(Core); }; LocalInputMonitorMac::LocalInputMonitorMac( scoped_refptr caller_task_runner, scoped_refptr ui_task_runner, base::WeakPtr client_session_control) : core_(new Core(caller_task_runner, ui_task_runner, client_session_control)) { core_->Start(); } LocalInputMonitorMac::~LocalInputMonitorMac() { core_->Stop(); } LocalInputMonitorMac::Core::Core( scoped_refptr caller_task_runner, scoped_refptr ui_task_runner, base::WeakPtr client_session_control) : caller_task_runner_(caller_task_runner), ui_task_runner_(ui_task_runner), manager_(nil), client_session_control_(client_session_control) { DCHECK(client_session_control_); } void LocalInputMonitorMac::Core::Start() { DCHECK(caller_task_runner_->BelongsToCurrentThread()); ui_task_runner_->PostTask(FROM_HERE, base::Bind(&Core::StartOnUiThread, this)); } void LocalInputMonitorMac::Core::Stop() { DCHECK(caller_task_runner_->BelongsToCurrentThread()); ui_task_runner_->PostTask(FROM_HERE, base::Bind(&Core::StopOnUiThread, this)); } LocalInputMonitorMac::Core::~Core() { DCHECK(manager_ == nil); } void LocalInputMonitorMac::Core::StartOnUiThread() { DCHECK(ui_task_runner_->BelongsToCurrentThread()); manager_ = [[LocalInputMonitorManager alloc] initWithMonitor:this]; } void LocalInputMonitorMac::Core::StopOnUiThread() { DCHECK(ui_task_runner_->BelongsToCurrentThread()); [manager_ invalidate]; [manager_ release]; manager_ = nil; } void LocalInputMonitorMac::Core::OnLocalMouseMoved( const webrtc::DesktopVector& position) { // In some cases OS may emit bogus mouse-move events even when cursor is not // actually moving. To handle this case properly verify that mouse position // has changed. See crbug.com/360912 . if (position.equals(mouse_position_)) { return; } mouse_position_ = position; caller_task_runner_->PostTask( FROM_HERE, base::Bind(&ClientSessionControl::OnLocalMouseMoved, client_session_control_, position)); } void LocalInputMonitorMac::Core::OnDisconnectShortcut() { caller_task_runner_->PostTask( FROM_HERE, base::Bind(&ClientSessionControl::DisconnectSession, client_session_control_, protocol::OK)); } } // namespace scoped_ptr LocalInputMonitor::Create( scoped_refptr caller_task_runner, scoped_refptr input_task_runner, scoped_refptr ui_task_runner, base::WeakPtr client_session_control) { return make_scoped_ptr(new LocalInputMonitorMac( caller_task_runner, ui_task_runner, client_session_control)); } } // namespace remoting