| // Copyright 2014 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #import "components/remote_cocoa/app_shim/mouse_capture.h" |
| |
| #import <Cocoa/Cocoa.h> |
| |
| #include <memory> |
| |
| #include "base/check_op.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/memory/weak_ptr.h" |
| #import "components/remote_cocoa/app_shim/mouse_capture_delegate.h" |
| |
| namespace remote_cocoa { |
| |
| // The ActiveEventTap is a RAII handle on the resources being used to capture |
| // events. There is either 0 or 1 active instance of this class. If a second |
| // instance is created, it will destroy the other instance before returning from |
| // its constructor. |
| class CocoaMouseCapture::ActiveEventTap { |
| public: |
| explicit ActiveEventTap(CocoaMouseCapture* owner); |
| |
| ActiveEventTap(const ActiveEventTap&) = delete; |
| ActiveEventTap& operator=(const ActiveEventTap&) = delete; |
| |
| ~ActiveEventTap(); |
| |
| // Returns the NSWindow with capture or nil if no window has capture |
| // currently. |
| static NSWindow* GetGlobalCaptureWindow(); |
| |
| void Init(); |
| |
| private: |
| // Returns the associated NSWindow with capture. |
| NSWindow* GetCaptureWindow() const; |
| |
| // The currently active event tap, or null if there is none. |
| static ActiveEventTap* g_active_event_tap; |
| |
| raw_ptr<CocoaMouseCapture> owner_; // Weak. Owns this. |
| id local_monitor_ = nil; |
| id global_monitor_ = nil; |
| }; |
| |
| CocoaMouseCapture::ActiveEventTap* |
| CocoaMouseCapture::ActiveEventTap::g_active_event_tap = nullptr; |
| |
| CocoaMouseCapture::ActiveEventTap::ActiveEventTap(CocoaMouseCapture* owner) |
| : owner_(owner) { |
| if (g_active_event_tap) |
| g_active_event_tap->owner_->OnOtherClientGotCapture(); |
| DCHECK(!g_active_event_tap); |
| g_active_event_tap = this; |
| } |
| |
| CocoaMouseCapture::ActiveEventTap::~ActiveEventTap() { |
| DCHECK_EQ(g_active_event_tap, this); |
| [NSEvent removeMonitor:global_monitor_]; |
| [NSEvent removeMonitor:local_monitor_]; |
| g_active_event_tap = nullptr; |
| owner_->delegate_->OnMouseCaptureLost(); |
| } |
| |
| // static |
| NSWindow* CocoaMouseCapture::ActiveEventTap::GetGlobalCaptureWindow() { |
| return g_active_event_tap ? g_active_event_tap->GetCaptureWindow() : nil; |
| } |
| |
| void CocoaMouseCapture::ActiveEventTap::Init() { |
| // Consume most things, but not NSEventTypeMouseEntered/Exited: The Widget |
| // doing capture will still see its own Entered/Exit events, but not those for |
| // other NSViews, since consuming those would break their tracking area logic. |
| NSEventMask event_mask = NSEventMaskLeftMouseDown | NSEventMaskLeftMouseUp | |
| NSEventMaskRightMouseDown | NSEventMaskRightMouseUp | |
| NSEventMaskMouseMoved | NSEventMaskLeftMouseDragged | |
| NSEventMaskRightMouseDragged | |
| NSEventMaskScrollWheel | NSEventMaskOtherMouseDown | |
| NSEventMaskOtherMouseUp | |
| NSEventMaskOtherMouseDragged; |
| |
| // Capture a WeakPtr. This allows the block to detect another event monitor |
| // for the same event deleting |owner_|. |
| base::WeakPtr<CocoaMouseCapture> weak_ptr = owner_->factory_.GetWeakPtr(); |
| |
| auto local_block = ^NSEvent*(NSEvent* event) { |
| if (!weak_ptr) { |
| return event; |
| } |
| |
| bool handled = weak_ptr->delegate_->PostCapturedEvent(event); |
| return handled ? nil : event; |
| }; |
| auto global_block = ^void(NSEvent* event) { |
| CocoaMouseCapture* owner = weak_ptr.get(); |
| if (owner) { |
| owner->delegate_->PostCapturedEvent(event); |
| } |
| }; |
| local_monitor_ = [NSEvent addLocalMonitorForEventsMatchingMask:event_mask |
| handler:local_block]; |
| global_monitor_ = |
| [NSEvent addGlobalMonitorForEventsMatchingMask:event_mask |
| handler:global_block]; |
| } |
| |
| NSWindow* CocoaMouseCapture::ActiveEventTap::GetCaptureWindow() const { |
| return owner_->delegate_->GetWindow(); |
| } |
| |
| CocoaMouseCapture::CocoaMouseCapture(CocoaMouseCaptureDelegate* delegate) |
| : delegate_(delegate), |
| active_handle_(std::make_unique<ActiveEventTap>(this)) { |
| active_handle_->Init(); |
| } |
| |
| CocoaMouseCapture::~CocoaMouseCapture() = default; |
| |
| // static |
| NSWindow* CocoaMouseCapture::GetGlobalCaptureWindow() { |
| return ActiveEventTap::GetGlobalCaptureWindow(); |
| } |
| |
| void CocoaMouseCapture::OnOtherClientGotCapture() { |
| DCHECK(active_handle_); |
| active_handle_.reset(); |
| } |
| |
| } // namespace remote_cocoa |