mirror of
				https://github.com/TomHarte/CLK.git
				synced 2025-10-25 09:27:01 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			447 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Objective-C
		
	
	
	
	
	
			
		
		
	
	
			447 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Objective-C
		
	
	
	
	
	
| //
 | |
| //  CSScanTargetView
 | |
| //  CLK
 | |
| //
 | |
| //  Created by Thomas Harte on 16/07/2015.
 | |
| //  Copyright 2015 Thomas Harte. All rights reserved.
 | |
| //
 | |
| 
 | |
| #import "CSScanTargetView.h"
 | |
| #import "CSApplication.h"
 | |
| #import "CSScanTarget.h"
 | |
| @import CoreVideo;
 | |
| @import GLKit;
 | |
| 
 | |
| #include <stdatomic.h>
 | |
| 
 | |
| static const NSTimeInterval standardMouseHideInterval = 3.0;
 | |
| static const NSTimeInterval quickMouseHideInterval = 0.1;
 | |
| 
 | |
| @interface CSScanTargetView () <NSDraggingDestination, CSApplicationEventDelegate>
 | |
| @end
 | |
| 
 | |
| @implementation CSScanTargetView {
 | |
| 	CVDisplayLinkRef _displayLink;
 | |
| 	NSNumber *_currentScreenNumber;
 | |
| 
 | |
| 	NSTrackingArea *_windowTrackingArea;
 | |
| 	NSTrackingArea *_subviewTrackingArea;
 | |
| 	NSTimer *_mouseHideTimer;
 | |
| 	BOOL _mouseIsCaptured;
 | |
| 
 | |
| 	atomic_int _isDrawingFlag;
 | |
| 	BOOL _isInvalid;
 | |
| 
 | |
| 	CSScanTarget *_scanTarget;
 | |
| }
 | |
| 
 | |
| - (void)setupDisplayLink {
 | |
| 	// Kill the existing link if there is one.
 | |
| 	if(_displayLink) {
 | |
| 		[self stopDisplayLink];
 | |
| 		CVDisplayLinkRelease(_displayLink);
 | |
| 	}
 | |
| 
 | |
| 	// Create a display link for the display the window is currently on.
 | |
| 	NSNumber *const screenNumber = self.window.screen.deviceDescription[@"NSScreenNumber"];
 | |
| 	_currentScreenNumber = screenNumber;
 | |
| 	CVDisplayLinkCreateWithCGDisplay(screenNumber.unsignedIntValue, &_displayLink);
 | |
| 
 | |
| 	// Set the renderer output callback function.
 | |
| 	CVDisplayLinkSetOutputCallback(_displayLink, DisplayLinkCallback, (__bridge void * __nullable)(self));
 | |
| 
 | |
| 	// Activate the display link.
 | |
| 	CVDisplayLinkStart(_displayLink);
 | |
| }
 | |
| 
 | |
| static CVReturn DisplayLinkCallback(__unused CVDisplayLinkRef displayLink, const CVTimeStamp *now, const CVTimeStamp *outputTime, __unused CVOptionFlags flagsIn, __unused CVOptionFlags *flagsOut, void *displayLinkContext) {
 | |
| 	CSScanTargetView *const view = (__bridge CSScanTargetView *)displayLinkContext;
 | |
| 
 | |
| 	// Schedule an opportunity to check that the display link is still linked to the correct display.
 | |
| 	dispatch_async(dispatch_get_main_queue(), ^{
 | |
| 		[view checkDisplayLink];
 | |
| 	});
 | |
| 
 | |
| 	// Ensure _isDrawingFlag has value 1 when drawing, 0 otherwise.
 | |
| 	atomic_store(&view->_isDrawingFlag, 1);
 | |
| 
 | |
| 	[view.displayLinkDelegate scanTargetViewDisplayLinkDidFire:view now:now outputTime:outputTime];
 | |
| 	/*
 | |
| 		Do not touch the display link from after this call; there's a bit of a race condition with setupDisplayLink.
 | |
| 		Specifically: Apple provides CVDisplayLinkStop but a call to that merely prevents future calls to the callback,
 | |
| 		it doesn't wait for completion of any current calls. So I've set up a usleep for one callback's duration,
 | |
| 		so code in here gets one callback's duration to access the display link.
 | |
| 
 | |
| 		In practice, it should do so only upon entry, and before calling into the view. The view promises not to
 | |
| 		access the display link itself as part of -drawAtTime:frequency:.
 | |
| 	*/
 | |
| 
 | |
| 	atomic_store(&view->_isDrawingFlag, 0);
 | |
| 
 | |
| 	return kCVReturnSuccess;
 | |
| }
 | |
| 
 | |
| - (void)checkDisplayLink {
 | |
| 	// Don't do anything if this view has already been invalidated.
 | |
| 	if(_isInvalid) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	// Test now whether the screen this view is on has changed since last time it was checked.
 | |
| 	// There's likely a callback available for this, on NSWindow if nowhere else, or an NSNotification,
 | |
| 	// but since this method is going to be called repeatedly anyway, and the test is cheap, polling
 | |
| 	// feels fine.
 | |
| 	NSNumber *const screenNumber = self.window.screen.deviceDescription[@"NSScreenNumber"];
 | |
| 	if(![_currentScreenNumber isEqual:screenNumber]) {
 | |
| 		// Also switch display links, to make sure synchronisation is with the display
 | |
| 		// the window is actually on, and at its rate.
 | |
| 		[self setupDisplayLink];
 | |
| 	}
 | |
| }
 | |
| 
 | |
| - (void)invalidate {
 | |
| 	_isInvalid = YES;
 | |
| 	[self stopDisplayLink];
 | |
| }
 | |
| 
 | |
| - (double)refreshPeriod {
 | |
| 	return CVDisplayLinkGetActualOutputVideoRefreshPeriod(_displayLink);
 | |
| }
 | |
| 
 | |
| - (void)stopDisplayLink {
 | |
| 	const double duration = [self refreshPeriod];
 | |
| 	CVDisplayLinkStop(_displayLink);
 | |
| 
 | |
| 	// This is a workaround; CVDisplayLinkStop does not wait for any existing call to the
 | |
| 	// display-link callback to stop. Furthermore there's a race condition between a callback
 | |
| 	// and any ability by me to set state.
 | |
| 	//
 | |
| 	// So: wait for a whole display link tick to avoid the second race condition. Then spin
 | |
| 	// on an atomic flag.
 | |
| 	usleep((useconds_t)ceil(duration * 1000000.0));
 | |
| 
 | |
| 	// Spin until _isDrawingFlag is 0 (and leave it as 0).
 | |
| 	int expected_value = 0;
 | |
| 	while(!atomic_compare_exchange_weak(&_isDrawingFlag, &expected_value, 0)) {
 | |
| 		expected_value = 0;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| - (void)dealloc {
 | |
| 	// Stop and release the display link
 | |
| 	CVDisplayLinkStop(_displayLink);
 | |
| 	CVDisplayLinkRelease(_displayLink);
 | |
| }
 | |
| 
 | |
| - (CSScanTarget *)scanTarget {
 | |
| 	return _scanTarget;
 | |
| }
 | |
| 
 | |
| - (void)willChangeScanTargetOwner {
 | |
| 	[_scanTarget willChangeOwner];
 | |
| }
 | |
| 
 | |
| - (void)updateBacking {
 | |
| 	[_scanTarget updateFrameBuffer];
 | |
| }
 | |
| 
 | |
| - (void)awakeFromNib {
 | |
| 	self.device = MTLCreateSystemDefaultDevice();
 | |
| 
 | |
| 	// Configure for explicit drawing.
 | |
| 	self.paused = YES;
 | |
| 	self.enableSetNeedsDisplay = NO;
 | |
| 
 | |
| 	// Create the scan target.
 | |
| 	_scanTarget = [[CSScanTarget alloc] initWithView:self];
 | |
| 	self.delegate = _scanTarget;
 | |
| 
 | |
| 	// Register to receive dragged and dropped file URLs.
 | |
| 	[self registerForDraggedTypes:@[(__bridge NSString *)kUTTypeFileURL]];
 | |
| 
 | |
| 	// Setup the [initial] display link.
 | |
| 	[self setupDisplayLink];
 | |
| }
 | |
| 
 | |
| #pragma mark - NSResponder
 | |
| 
 | |
| - (BOOL)acceptsFirstResponder {
 | |
| 	return YES;
 | |
| }
 | |
| 
 | |
| - (void)keyDown:(NSEvent *)event {
 | |
| 	[self.responderDelegate keyDown:event];
 | |
| }
 | |
| 
 | |
| - (void)keyUp:(NSEvent *)event {
 | |
| 	[self.responderDelegate keyUp:event];
 | |
| }
 | |
| 
 | |
| - (void)flagsChanged:(NSEvent *)event {
 | |
| 	// Release the mouse upon a control + command.
 | |
| 	if(_mouseIsCaptured &&
 | |
| 		event.modifierFlags & NSEventModifierFlagControl &&
 | |
| 		event.modifierFlags & NSEventModifierFlagCommand) {
 | |
| 		[self releaseMouse];
 | |
| 	}
 | |
| 
 | |
| 	[self.responderDelegate flagsChanged:event];
 | |
| }
 | |
| 
 | |
| - (BOOL)application:(nonnull CSApplication *)application shouldSendEvent:(nonnull NSEvent *)event {
 | |
| 	switch(event.type) {
 | |
| 		default: return YES;
 | |
| 		case NSEventTypeKeyUp:			[self keyUp:event];		return NO;
 | |
| 		case NSEventTypeKeyDown:		[self keyDown:event];	return NO;
 | |
| 		case NSEventTypeFlagsChanged:	[self flagsChanged:event];	return NO;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| - (void)paste:(id)sender {
 | |
| 	[self.responderDelegate paste:sender];
 | |
| }
 | |
| 
 | |
| - (NSBitmapImageRep *)imageRepresentation {
 | |
| 	return self.scanTarget.imageRepresentation;
 | |
| }
 | |
| 
 | |
| #pragma mark - NSDraggingDestination
 | |
| 
 | |
| - (BOOL)performDragOperation:(id <NSDraggingInfo>)sender {
 | |
| 	for(NSPasteboardItem *item in [[sender draggingPasteboard] pasteboardItems]) {
 | |
| 		NSURL *URL = [NSURL URLWithString:[item stringForType:(__bridge NSString *)kUTTypeFileURL]];
 | |
| 		[self.responderDelegate scanTargetView:self didReceiveFileAtURL:URL];
 | |
| 	}
 | |
| 	return YES;
 | |
| }
 | |
| 
 | |
| - (NSDragOperation)draggingEntered:(id < NSDraggingInfo >)sender {
 | |
| 	return NSDragOperationLink;
 | |
| }
 | |
| 
 | |
| #pragma mark - Mouse hiding
 | |
| 
 | |
| - (void)setShouldCaptureMouse:(BOOL)shouldCaptureMouse {
 | |
| 	_shouldCaptureMouse = shouldCaptureMouse;
 | |
| 
 | |
| 	if(_windowTrackingArea) {
 | |
| 		[self removeTrackingArea:_windowTrackingArea];
 | |
| 	}
 | |
| 	_windowTrackingArea =
 | |
| 		[[NSTrackingArea alloc]
 | |
| 		 initWithRect:self.bounds
 | |
| 			options:NSTrackingMouseEnteredAndExited | NSTrackingMouseMoved | NSTrackingActiveWhenFirstResponder
 | |
| 			owner:self
 | |
| 			userInfo:nil];
 | |
| 	[self addTrackingArea:_windowTrackingArea];
 | |
| }
 | |
| 
 | |
| - (void)recalculateSubviewTrackingAreas {
 | |
| 	if(_subviewTrackingArea) {
 | |
| 		[self removeTrackingArea:_subviewTrackingArea];
 | |
| 	}
 | |
| 
 | |
| 	// Use the union of rects of interesting subviews as the tracking area.
 | |
| 	const NSRect emptySentinel = NSMakeRect(-1, -1, -1, -1);
 | |
| 	NSRect trackingRect = emptySentinel;
 | |
| 	for(NSView *const subview in self.subviews) {
 | |
| 		if(
 | |
| 			[self.responderDelegate respondsToSelector:@selector(scanTargetView:shouldTrackMousovers:)] &&
 | |
| 			![self.responderDelegate scanTargetView:self shouldTrackMousovers:subview]
 | |
| 		) {
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		if(NSEqualRects(trackingRect, emptySentinel)) {
 | |
| 			trackingRect = subview.frame;
 | |
| 		} else {
 | |
| 			trackingRect = NSUnionRect(subview.frame, trackingRect);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if(NSEqualRects(trackingRect, emptySentinel)) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	_subviewTrackingArea =
 | |
| 		[[NSTrackingArea alloc]
 | |
| 			initWithRect:trackingRect
 | |
| 			options:NSTrackingMouseEnteredAndExited | NSTrackingMouseMoved | NSTrackingActiveWhenFirstResponder
 | |
| 			owner:self
 | |
| 			userInfo:nil];
 | |
| 	[self addTrackingArea:_subviewTrackingArea];
 | |
| }
 | |
| 
 | |
| - (void)updateTrackingAreas {
 | |
| 	[super updateTrackingAreas];
 | |
| 	[self recalculateSubviewTrackingAreas];
 | |
| }
 | |
| 
 | |
| - (void)didAddSubview:(NSView *)subview {
 | |
| 	[self recalculateSubviewTrackingAreas];
 | |
| }
 | |
| 
 | |
| - (void)layout {
 | |
| 	[super layout];
 | |
| 	[self recalculateSubviewTrackingAreas];
 | |
| }
 | |
| 
 | |
| - (void)cancelMouseHide {
 | |
| 	[_mouseHideTimer invalidate];
 | |
| }
 | |
| 
 | |
| - (void)scheduleMouseHideAfter:(NSTimeInterval)interval {
 | |
| 	[self cancelMouseHide];
 | |
| 
 | |
| 	_mouseHideTimer = [NSTimer scheduledTimerWithTimeInterval:interval repeats:NO block:^(__unused NSTimer * _Nonnull timer) {
 | |
| 		// Don't actually hide the mouse if this is a mouse-capture machine; that makes
 | |
| 		// it fairly confusing as to current application state.
 | |
| 		if(!self.shouldCaptureMouse) {
 | |
| 			[NSCursor setHiddenUntilMouseMoves:YES];
 | |
| 		}
 | |
| 		[self.responderDelegate scanTargetViewWouldHideOSMouseCursor:self];
 | |
| 	}];
 | |
| }
 | |
| 
 | |
| - (void)mouseEntered:(NSEvent *)event {
 | |
| 	[super mouseEntered:event];
 | |
| 
 | |
| 	if(event.trackingArea == _windowTrackingArea) {
 | |
| 		[self scheduleMouseHideAfter:standardMouseHideInterval];
 | |
| 	}
 | |
| 	if(event.trackingArea == _subviewTrackingArea && !_mouseIsCaptured) {
 | |
| 		[self cancelMouseHide];
 | |
| 		[self.responderDelegate scanTargetViewDidMouseoverSubviews:self];
 | |
| 	}
 | |
| }
 | |
| 
 | |
| - (void)mouseExited:(NSEvent *)event {
 | |
| 	[super mouseExited:event];
 | |
| 
 | |
| 	// Schedule a really short mouse-hiding interval.
 | |
| 	[self scheduleMouseHideAfter:quickMouseHideInterval];
 | |
| }
 | |
| 
 | |
| - (void)releaseMouse {
 | |
| 	if(_mouseIsCaptured) {
 | |
| 		_mouseIsCaptured = NO;
 | |
| 		CGAssociateMouseAndMouseCursorPosition(true);
 | |
| 		[NSCursor unhide];
 | |
| 		[self.responderDelegate scanTargetViewDidReleaseMouse:self];
 | |
| 		((CSApplication *)[NSApplication sharedApplication]).eventDelegate = nil;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| #pragma mark - Mouse motion
 | |
| 
 | |
| - (void)applyMouseMotion:(NSEvent *)event {
 | |
| 	if(!_mouseIsCaptured) {
 | |
| 		// Mouse capture is off, so don't play games with the cursor, just schedule it to
 | |
| 		// hide in the near future unless over an interesting subview.
 | |
| 		if(
 | |
| 			_subviewTrackingArea &&
 | |
| 			NSPointInRect([self convertPoint:event.locationInWindow fromView:nil], _subviewTrackingArea.rect)
 | |
| 		) {
 | |
| 			[self.responderDelegate scanTargetViewDidMouseoverSubviews:self];
 | |
| 		} else {
 | |
| 			[self scheduleMouseHideAfter:standardMouseHideInterval];
 | |
| 		}
 | |
| 	} else {
 | |
| 		// Mouse capture is on, so move the cursor back to the middle of the window, and
 | |
| 		// forward the deltas to the listener.
 | |
| 		//
 | |
| 		// TODO: should I really need to invert the y coordinate myself? It suggests I
 | |
| 		// might have an error in mapping here.
 | |
| 		const NSPoint windowCentre = [self convertPoint:CGPointMake(self.bounds.size.width * 0.5, self.bounds.size.height * 0.5) toView:nil];
 | |
| 		const NSPoint screenCentre = [self.window convertPointToScreen:windowCentre];
 | |
| 		const CGRect screenFrame = self.window.screen.frame;
 | |
| 		CGWarpMouseCursorPosition(NSMakePoint(
 | |
| 			screenFrame.origin.x + screenCentre.x,
 | |
| 			screenFrame.origin.y + screenFrame.size.height - screenCentre.y
 | |
| 		));
 | |
| 
 | |
| 		[self.responderDelegate mouseMoved:event];
 | |
| 	}
 | |
| }
 | |
| 
 | |
| - (void)mouseDragged:(NSEvent *)event {
 | |
| 	[self applyMouseMotion:event];
 | |
| 	[super mouseDragged:event];
 | |
| }
 | |
| 
 | |
| - (void)rightMouseDragged:(NSEvent *)event {
 | |
| 	[self applyMouseMotion:event];
 | |
| 	[super rightMouseDragged:event];
 | |
| }
 | |
| 
 | |
| - (void)otherMouseDragged:(NSEvent *)event {
 | |
| 	[self applyMouseMotion:event];
 | |
| 	[super otherMouseDragged:event];
 | |
| }
 | |
| 
 | |
| - (void)mouseMoved:(NSEvent *)event {
 | |
| 	[self applyMouseMotion:event];
 | |
| 	[super mouseMoved:event];
 | |
| }
 | |
| 
 | |
| #pragma mark - Mouse buttons
 | |
| 
 | |
| - (void)applyButtonDown:(NSEvent *)event {
 | |
| 	if(self.shouldCaptureMouse) {
 | |
| 		if(!_mouseIsCaptured) {
 | |
| 			_mouseIsCaptured = YES;
 | |
| 			[NSCursor hide];
 | |
| 			CGAssociateMouseAndMouseCursorPosition(false);
 | |
| 			[self.responderDelegate scanTargetViewWouldHideOSMouseCursor:self];
 | |
| 			[self.responderDelegate scanTargetViewDidCaptureMouse:self];
 | |
| 			if(self.shouldUsurpCommand) {
 | |
| 				((CSApplication *)[NSApplication sharedApplication]).eventDelegate = self;
 | |
| 			}
 | |
| 
 | |
| 			// Don't report the first click to the delegate; treat that as merely
 | |
| 			// an invitation to capture the cursor.
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		[self.responderDelegate mouseDown:event];
 | |
| 	}
 | |
| }
 | |
| 
 | |
| - (void)applyButtonUp:(NSEvent *)event {
 | |
| 	if(self.shouldCaptureMouse) {
 | |
| 		[self.responderDelegate mouseUp:event];
 | |
| 	}
 | |
| }
 | |
| 
 | |
| - (void)mouseDown:(NSEvent *)event {
 | |
| 	[self applyButtonDown:event];
 | |
| 	[super mouseDown:event];
 | |
| }
 | |
| 
 | |
| - (void)rightMouseDown:(NSEvent *)event {
 | |
| 	[self applyButtonDown:event];
 | |
| 	[super rightMouseDown:event];
 | |
| }
 | |
| 
 | |
| - (void)otherMouseDown:(NSEvent *)event {
 | |
| 	[self applyButtonDown:event];
 | |
| 	[super otherMouseDown:event];
 | |
| }
 | |
| 
 | |
| - (void)mouseUp:(NSEvent *)event {
 | |
| 	[self applyButtonUp:event];
 | |
| 	[super mouseUp:event];
 | |
| }
 | |
| 
 | |
| - (void)rightMouseUp:(NSEvent *)event {
 | |
| 	[self applyButtonUp:event];
 | |
| 	[super rightMouseUp:event];
 | |
| }
 | |
| 
 | |
| - (void)otherMouseUp:(NSEvent *)event {
 | |
| 	[self applyButtonUp:event];
 | |
| 	[super otherMouseUp:event];
 | |
| }
 | |
| 
 | |
| @end
 |