chromium/ui/views/cocoa/cocoa_mouse_capture_unittest.mm

// 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>

#import "components/remote_cocoa/app_shim/mouse_capture_delegate.h"
#import "ui/base/test/cocoa_helper.h"
#import "ui/events/test/cocoa_test_event_utils.h"

// Simple test view that counts calls to -[NSView mouseDown:].
@interface CocoaMouseCaptureTestView : NSView
@property(readonly, nonatomic) int mouseDownCount;
@end

@implementation CocoaMouseCaptureTestView {
  int _mouseDownCount;
}

@synthesize mouseDownCount = _mouseDownCount;

- (void)mouseDown:(NSEvent*)theEvent {
  ++_mouseDownCount;
}

@end

namespace remote_cocoa {
namespace {

// Simple capture delegate that just counts events forwarded.
class TestCaptureDelegate : public CocoaMouseCaptureDelegate {
 public:
  explicit TestCaptureDelegate(NSWindow* window) : window_(window) {}

  TestCaptureDelegate(const TestCaptureDelegate&) = delete;
  TestCaptureDelegate& operator=(const TestCaptureDelegate&) = delete;

  void Acquire() { mouse_capture_ = std::make_unique<CocoaMouseCapture>(this); }
  bool IsActive() { return mouse_capture_ && mouse_capture_->IsActive(); }
  void SimulateDestroy() { mouse_capture_.reset(); }
  void set_should_claim_event(bool should_claim_event) {
    should_claim_event_ = should_claim_event;
  }

  int event_count() { return event_count_; }
  int capture_lost_count() { return capture_lost_count_; }

  // CocoaMouseCaptureDelegate:
  bool PostCapturedEvent(NSEvent* event) override {
    ++event_count_;
    return should_claim_event_;
  }
  void OnMouseCaptureLost() override { ++capture_lost_count_; }
  NSWindow* GetWindow() const override { return window_; }

 private:
  std::unique_ptr<CocoaMouseCapture> mouse_capture_;
  bool should_claim_event_ = true;
  int event_count_ = 0;
  int capture_lost_count_ = 0;
  NSWindow* __strong window_;
};

}  // namespace

using CocoaMouseCaptureTest = ui::CocoaTest;

// Test that a new capture properly "steals" capture from an existing one.
TEST_F(CocoaMouseCaptureTest, OnCaptureLost) {
  TestCaptureDelegate capture(nil);

  capture.Acquire();
  EXPECT_TRUE(capture.IsActive());
  {
    TestCaptureDelegate capture2(nil);
    EXPECT_EQ(0, capture.capture_lost_count());

    // A second capture steals from the first.
    capture2.Acquire();
    EXPECT_TRUE(capture2.IsActive());
    EXPECT_FALSE(capture.IsActive());
    EXPECT_EQ(1, capture.capture_lost_count());
    EXPECT_EQ(0, capture2.capture_lost_count());

    // Simulate capture2 going out of scope. Inspect it.
    capture2.SimulateDestroy();
    EXPECT_FALSE(capture2.IsActive());
    EXPECT_EQ(1, capture2.capture_lost_count());
  }

  // Re-acquiring is fine (not stealing).
  EXPECT_FALSE(capture.IsActive());
  capture.Acquire();
  EXPECT_TRUE(capture.IsActive());

  // Having no CocoaMouseCapture instance is fine.
  capture.SimulateDestroy();
  EXPECT_FALSE(capture.IsActive());
  // Receives OnMouseCaptureLost again, since reacquired.
  EXPECT_EQ(2, capture.capture_lost_count());
}

// Test event capture.
TEST_F(CocoaMouseCaptureTest, CaptureEvents) {
  CocoaMouseCaptureTestView* view =
      [[CocoaMouseCaptureTestView alloc] initWithFrame:NSZeroRect];
  test_window().contentView = view;
  NSArray<NSEvent*>* click = cocoa_test_event_utils::MouseClickInView(view, 1);

  // First check that the view receives events normally.
  EXPECT_EQ(0, view.mouseDownCount);
  [NSApp sendEvent:click[0]];
  EXPECT_EQ(1, view.mouseDownCount);

  {
    TestCaptureDelegate capture(test_window());
    capture.Acquire();

    // Now check that the capture captures events.
    EXPECT_EQ(0, capture.event_count());
    [NSApp sendEvent:click[0]];
    EXPECT_EQ(1, view.mouseDownCount);
    EXPECT_EQ(1, capture.event_count());
  }

  // After the capture goes away, events should be received again.
  [NSApp sendEvent:click[0]];
  EXPECT_EQ(2, view.mouseDownCount);
}

// Test local events properly swallowed / propagated.
TEST_F(CocoaMouseCaptureTest, SwallowOrPropagateEvents) {
  CocoaMouseCaptureTestView* view =
      [[CocoaMouseCaptureTestView alloc] initWithFrame:NSZeroRect];
  test_window().contentView = view;
  NSArray<NSEvent*>* click = cocoa_test_event_utils::MouseClickInView(view, 1);

  // First check that the view receives events normally.
  EXPECT_EQ(0, view.mouseDownCount);
  [NSApp sendEvent:click[0]];
  EXPECT_EQ(1, view.mouseDownCount);

  {
    TestCaptureDelegate capture(test_window());
    capture.Acquire();

    // By default, the delegate should claim events.
    EXPECT_EQ(0, capture.event_count());
    [NSApp sendEvent:click[0]];
    EXPECT_EQ(1, view.mouseDownCount);
    EXPECT_EQ(1, capture.event_count());

    // Set the delegate not to claim events.
    capture.set_should_claim_event(false);
    [NSApp sendEvent:click[0]];
    EXPECT_EQ(2, view.mouseDownCount);
    EXPECT_EQ(2, capture.event_count());

    // Setting it back should restart the claiming of events.
    capture.set_should_claim_event(true);
    [NSApp sendEvent:click[0]];
    EXPECT_EQ(2, view.mouseDownCount);
    EXPECT_EQ(3, capture.event_count());
  }
}

}  // namespace remote_cocoa