2015-07-17 00:40:46 +00:00
|
|
|
//
|
2016-03-05 19:45:09 +00:00
|
|
|
// CSOpenGLView
|
2015-07-26 19:25:11 +00:00
|
|
|
// CLK
|
2015-07-17 00:40:46 +00:00
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 16/07/2015.
|
|
|
|
// Copyright © 2015 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
2016-03-05 19:45:09 +00:00
|
|
|
#import "CSOpenGLView.h"
|
2015-07-17 00:40:46 +00:00
|
|
|
@import CoreVideo;
|
2015-09-25 01:23:16 +00:00
|
|
|
@import GLKit;
|
2015-07-17 00:40:46 +00:00
|
|
|
|
2016-03-04 03:12:31 +00:00
|
|
|
typedef NS_ENUM(NSInteger, CSOpenGLViewCondition) {
|
|
|
|
CSOpenGLViewConditionReadyForUpdate,
|
|
|
|
CSOpenGLViewConditionUpdating
|
|
|
|
};
|
2015-08-02 18:25:21 +00:00
|
|
|
|
2016-03-05 19:45:09 +00:00
|
|
|
@implementation CSOpenGLView {
|
2016-02-05 03:28:50 +00:00
|
|
|
CVDisplayLinkRef _displayLink;
|
2016-03-04 03:12:31 +00:00
|
|
|
NSConditionLock *_runningLock;
|
|
|
|
dispatch_queue_t _dispatchQueue;
|
2015-07-17 00:40:46 +00:00
|
|
|
}
|
|
|
|
|
2015-07-17 01:16:21 +00:00
|
|
|
- (void)prepareOpenGL
|
2015-07-17 00:40:46 +00:00
|
|
|
{
|
2015-07-17 01:16:21 +00:00
|
|
|
// Synchronize buffer swaps with vertical refresh rate
|
|
|
|
GLint swapInt = 1;
|
|
|
|
[[self openGLContext] setValues:&swapInt forParameter:NSOpenGLCPSwapInterval];
|
2015-07-17 00:40:46 +00:00
|
|
|
|
2015-07-17 01:16:21 +00:00
|
|
|
// Create a display link capable of being used with all active displays
|
2016-02-05 03:28:50 +00:00
|
|
|
CVDisplayLinkCreateWithActiveCGDisplays(&_displayLink);
|
2015-07-17 01:16:21 +00:00
|
|
|
|
|
|
|
// Set the renderer output callback function
|
2016-02-05 03:28:50 +00:00
|
|
|
CVDisplayLinkSetOutputCallback(_displayLink, DisplayLinkCallback, (__bridge void * __nullable)(self));
|
2016-03-04 03:12:31 +00:00
|
|
|
|
|
|
|
// Create a queue and a condition lock for dispatching to it
|
|
|
|
_runningLock = [[NSConditionLock alloc] initWithCondition:CSOpenGLViewConditionReadyForUpdate];
|
|
|
|
_dispatchQueue = dispatch_queue_create("com.thomasharte.clocksignal.GL", DISPATCH_QUEUE_SERIAL);
|
2015-07-17 01:16:21 +00:00
|
|
|
|
|
|
|
// Set the display link for the current renderer
|
|
|
|
CGLContextObj cglContext = [[self openGLContext] CGLContextObj];
|
|
|
|
CGLPixelFormatObj cglPixelFormat = [[self pixelFormat] CGLPixelFormatObj];
|
2016-02-05 03:28:50 +00:00
|
|
|
CVDisplayLinkSetCurrentCGDisplayFromOpenGLContext(_displayLink, cglContext, cglPixelFormat);
|
2015-07-26 19:13:46 +00:00
|
|
|
|
2016-03-04 03:12:31 +00:00
|
|
|
// set the clear colour
|
2015-07-26 19:25:11 +00:00
|
|
|
[self.openGLContext makeCurrentContext];
|
2015-07-26 19:55:19 +00:00
|
|
|
glClearColor(0.0, 0.0, 0.0, 1.0);
|
2015-07-26 19:13:46 +00:00
|
|
|
|
2015-07-17 01:16:21 +00:00
|
|
|
// Activate the display link
|
2016-02-05 03:28:50 +00:00
|
|
|
CVDisplayLinkStart(_displayLink);
|
2015-07-27 03:50:43 +00:00
|
|
|
|
|
|
|
glEnable(GL_BLEND);
|
2015-07-31 21:47:10 +00:00
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE);
|
2015-07-17 01:16:21 +00:00
|
|
|
}
|
2015-07-17 00:40:46 +00:00
|
|
|
|
2015-07-27 03:50:43 +00:00
|
|
|
static CVReturn DisplayLinkCallback(CVDisplayLinkRef displayLink, const CVTimeStamp *now, const CVTimeStamp *outputTime, CVOptionFlags flagsIn, CVOptionFlags *flagsOut, void *displayLinkContext)
|
2015-07-17 01:16:21 +00:00
|
|
|
{
|
2016-03-05 19:45:09 +00:00
|
|
|
CSOpenGLView *view = (__bridge CSOpenGLView *)displayLinkContext;
|
2016-03-04 03:12:31 +00:00
|
|
|
[view drawAtTime:now];
|
2015-07-17 01:16:21 +00:00
|
|
|
return kCVReturnSuccess;
|
|
|
|
}
|
2015-07-28 01:15:10 +00:00
|
|
|
|
2016-03-04 03:12:31 +00:00
|
|
|
- (void)drawAtTime:(const CVTimeStamp *)now
|
|
|
|
{
|
|
|
|
if([_runningLock tryLockWhenCondition:CSOpenGLViewConditionReadyForUpdate])
|
|
|
|
{
|
|
|
|
CVTimeStamp timeStamp = *now;
|
|
|
|
dispatch_async(_dispatchQueue, ^{
|
|
|
|
[_runningLock lockWhenCondition:CSOpenGLViewConditionUpdating];
|
|
|
|
[self.delegate openGLView:self didUpdateToTime:timeStamp];
|
|
|
|
[self drawViewOnlyIfDirty:YES];
|
|
|
|
[_runningLock unlockWithCondition:CSOpenGLViewConditionReadyForUpdate];
|
|
|
|
});
|
|
|
|
[_runningLock unlockWithCondition:CSOpenGLViewConditionUpdating];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-28 01:15:10 +00:00
|
|
|
- (void)invalidate
|
|
|
|
{
|
2016-02-05 03:28:50 +00:00
|
|
|
CVDisplayLinkStop(_displayLink);
|
2016-03-04 03:12:31 +00:00
|
|
|
[_runningLock lockWhenCondition:CSOpenGLViewConditionReadyForUpdate];
|
|
|
|
[_runningLock unlock];
|
2015-07-28 01:15:10 +00:00
|
|
|
}
|
|
|
|
|
2015-07-17 01:16:21 +00:00
|
|
|
- (void)dealloc
|
|
|
|
{
|
|
|
|
// Release the display link
|
2016-02-05 03:28:50 +00:00
|
|
|
CVDisplayLinkRelease(_displayLink);
|
2015-07-17 00:40:46 +00:00
|
|
|
}
|
|
|
|
|
2016-02-05 03:28:50 +00:00
|
|
|
- (CGSize)backingSize
|
2016-01-07 04:14:36 +00:00
|
|
|
{
|
2016-02-05 03:28:50 +00:00
|
|
|
return [self convertSizeToBacking:self.bounds.size];
|
2016-01-07 04:14:36 +00:00
|
|
|
}
|
|
|
|
|
2015-07-24 02:51:53 +00:00
|
|
|
- (void)reshape
|
|
|
|
{
|
|
|
|
[super reshape];
|
|
|
|
|
2015-07-25 03:42:19 +00:00
|
|
|
[self.openGLContext makeCurrentContext];
|
2015-08-13 21:01:25 +00:00
|
|
|
CGLLockContext([[self openGLContext] CGLContextObj]);
|
|
|
|
|
2016-02-05 03:28:50 +00:00
|
|
|
CGSize viewSize = [self backingSize];
|
|
|
|
glViewport(0, 0, (GLsizei)viewSize.width, (GLsizei)viewSize.height);
|
2015-08-13 21:01:25 +00:00
|
|
|
|
|
|
|
CGLUnlockContext([[self openGLContext] CGLContextObj]);
|
2015-07-24 02:51:53 +00:00
|
|
|
}
|
|
|
|
|
2015-08-13 22:22:51 +00:00
|
|
|
- (void)awakeFromNib
|
2015-07-24 02:51:53 +00:00
|
|
|
{
|
|
|
|
NSOpenGLPixelFormatAttribute attributes[] =
|
|
|
|
{
|
|
|
|
NSOpenGLPFADoubleBuffer,
|
|
|
|
NSOpenGLPFAOpenGLProfile, NSOpenGLProfileVersion3_2Core,
|
2016-02-28 03:46:31 +00:00
|
|
|
NSOpenGLPFAMultisample,
|
2015-09-11 01:30:39 +00:00
|
|
|
NSOpenGLPFASampleBuffers, 1,
|
2016-02-19 04:21:25 +00:00
|
|
|
NSOpenGLPFASamples, 2,
|
2015-07-24 02:51:53 +00:00
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
NSOpenGLPixelFormat *pixelFormat = [[NSOpenGLPixelFormat alloc] initWithAttributes:attributes];
|
|
|
|
NSOpenGLContext *context = [[NSOpenGLContext alloc] initWithFormat:pixelFormat shareContext:nil];
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
// When we're using a CoreProfile context, crash if we call a legacy OpenGL function
|
|
|
|
// This will make it much more obvious where and when such a function call is made so
|
|
|
|
// that we can remove such calls.
|
|
|
|
// Without this we'd simply get GL_INVALID_OPERATION error for calling legacy functions
|
|
|
|
// but it would be more difficult to see where that function was called.
|
|
|
|
CGLEnable([context CGLContextObj], kCGLCECrashOnRemovedFunctions);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
self.pixelFormat = pixelFormat;
|
|
|
|
self.openGLContext = context;
|
|
|
|
self.wantsBestResolutionOpenGLSurface = YES;
|
|
|
|
}
|
|
|
|
|
2015-07-26 19:13:46 +00:00
|
|
|
- (void)drawRect:(NSRect)dirtyRect
|
2015-08-13 21:01:25 +00:00
|
|
|
{
|
2016-03-05 19:35:47 +00:00
|
|
|
dispatch_sync(_dispatchQueue, ^{
|
|
|
|
[self drawViewOnlyIfDirty:NO];
|
|
|
|
});
|
2015-08-13 21:01:25 +00:00
|
|
|
}
|
|
|
|
|
2016-02-05 04:05:47 +00:00
|
|
|
- (void)drawViewOnlyIfDirty:(BOOL)onlyIfDirty
|
2015-07-26 19:13:46 +00:00
|
|
|
{
|
|
|
|
[self.openGLContext makeCurrentContext];
|
2015-08-13 21:01:25 +00:00
|
|
|
CGLLockContext([[self openGLContext] CGLContextObj]);
|
2015-07-26 19:13:46 +00:00
|
|
|
|
2016-02-05 04:05:47 +00:00
|
|
|
[self.delegate openGLView:self drawViewOnlyIfDirty:onlyIfDirty];
|
2015-07-26 19:13:46 +00:00
|
|
|
|
2015-08-13 21:01:25 +00:00
|
|
|
CGLFlushDrawable([[self openGLContext] CGLContextObj]);
|
|
|
|
CGLUnlockContext([[self openGLContext] CGLContextObj]);
|
2015-07-26 19:13:46 +00:00
|
|
|
}
|
|
|
|
|
2015-08-19 00:33:24 +00:00
|
|
|
#pragma mark - NSResponder
|
|
|
|
|
|
|
|
- (BOOL)acceptsFirstResponder
|
|
|
|
{
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)keyDown:(NSEvent *)theEvent
|
|
|
|
{
|
2016-03-05 19:35:47 +00:00
|
|
|
dispatch_async(_dispatchQueue, ^{
|
|
|
|
[self.responderDelegate keyDown:theEvent];
|
|
|
|
});
|
2015-08-19 00:33:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)keyUp:(NSEvent *)theEvent
|
|
|
|
{
|
2016-03-05 19:35:47 +00:00
|
|
|
dispatch_async(_dispatchQueue, ^{
|
|
|
|
[self.responderDelegate keyUp:theEvent];
|
|
|
|
});
|
2015-08-19 00:33:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)flagsChanged:(NSEvent *)theEvent
|
|
|
|
{
|
2016-03-05 19:35:47 +00:00
|
|
|
dispatch_async(_dispatchQueue, ^{
|
|
|
|
[self.responderDelegate flagsChanged:theEvent];
|
|
|
|
});
|
2015-08-19 00:33:24 +00:00
|
|
|
}
|
|
|
|
|
2015-07-17 00:40:46 +00:00
|
|
|
@end
|