2016-03-30 23:07:55 +03:00
|
|
|
#import <OpenGL/gl.h>
|
2016-06-09 00:37:00 +03:00
|
|
|
#import <Carbon/Carbon.h>
|
2016-03-30 23:07:55 +03:00
|
|
|
#import "GBView.h"
|
2016-04-13 22:43:16 +03:00
|
|
|
#import "GBButtons.h"
|
|
|
|
#import "NSString+StringForKey.h"
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
@implementation GBView
|
|
|
|
{
|
|
|
|
uint32_t *image_buffers[3];
|
|
|
|
unsigned char current_buffer;
|
2016-07-05 23:34:33 +03:00
|
|
|
BOOL mouse_hidden;
|
|
|
|
NSTrackingArea *tracking_area;
|
|
|
|
BOOL _mouseHidingEnabled;
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
|
2016-06-14 14:11:37 +03:00
|
|
|
- (void) awakeFromNib
|
|
|
|
{
|
|
|
|
NSOpenGLPixelFormatAttribute attrs[] =
|
|
|
|
{
|
|
|
|
NSOpenGLPFAOpenGLProfile,
|
|
|
|
NSOpenGLProfileVersion3_2Core,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
NSOpenGLPixelFormat *pf = [[NSOpenGLPixelFormat alloc] initWithAttributes:attrs] ;
|
|
|
|
|
|
|
|
if (!pf)
|
|
|
|
{
|
|
|
|
NSLog(@"No OpenGL pixel format");
|
|
|
|
}
|
|
|
|
|
|
|
|
NSOpenGLContext* context = [[NSOpenGLContext alloc] initWithFormat:pf shareContext:nil] ;
|
|
|
|
|
|
|
|
[self setPixelFormat:pf];
|
|
|
|
|
|
|
|
[self setOpenGLContext:context];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
- (void) _init
|
|
|
|
{
|
|
|
|
image_buffers[0] = malloc(160 * 144 * 4);
|
|
|
|
image_buffers[1] = malloc(160 * 144 * 4);
|
|
|
|
image_buffers[2] = malloc(160 * 144 * 4);
|
|
|
|
_shouldBlendFrameWithPrevious = 1;
|
2016-04-28 23:07:05 +03:00
|
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(filterChanged) name:@"GBFilterChanged" object:nil];
|
2016-07-05 21:23:55 +03:00
|
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(ratioKeepingChanged) name:@"GBAspectChanged" object:nil];
|
2016-07-05 23:34:33 +03:00
|
|
|
tracking_area = [ [NSTrackingArea alloc] initWithRect:(NSRect){}
|
|
|
|
options:NSTrackingMouseEnteredAndExited | NSTrackingActiveAlways | NSTrackingInVisibleRect
|
|
|
|
owner:self
|
|
|
|
userInfo:nil];
|
|
|
|
[self addTrackingArea:tracking_area];
|
2016-04-28 23:07:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void) filterChanged
|
|
|
|
{
|
2016-06-09 00:06:55 +03:00
|
|
|
[self setNeedsDisplay:YES];
|
2016-04-28 23:07:05 +03:00
|
|
|
self.shader = nil;
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
|
2016-07-05 21:23:55 +03:00
|
|
|
- (void) ratioKeepingChanged
|
|
|
|
{
|
|
|
|
[self setFrame:self.superview.frame];
|
|
|
|
}
|
|
|
|
|
2016-06-09 00:06:55 +03:00
|
|
|
- (void) setShouldBlendFrameWithPrevious:(BOOL)shouldBlendFrameWithPrevious
|
|
|
|
{
|
|
|
|
_shouldBlendFrameWithPrevious = shouldBlendFrameWithPrevious;
|
|
|
|
[self setNeedsDisplay:YES];
|
|
|
|
}
|
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
- (unsigned char) numberOfBuffers
|
|
|
|
{
|
|
|
|
return _shouldBlendFrameWithPrevious? 3 : 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)dealloc
|
|
|
|
{
|
|
|
|
free(image_buffers[0]);
|
|
|
|
free(image_buffers[1]);
|
|
|
|
free(image_buffers[2]);
|
2016-07-05 23:34:33 +03:00
|
|
|
if (mouse_hidden) {
|
|
|
|
mouse_hidden = false;
|
|
|
|
[NSCursor unhide];
|
|
|
|
}
|
2016-04-28 23:07:05 +03:00
|
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
- (instancetype)initWithCoder:(NSCoder *)coder
|
|
|
|
{
|
|
|
|
if (!(self = [super initWithCoder:coder]))
|
|
|
|
{
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
[self _init];
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (instancetype)initWithFrame:(NSRect)frameRect
|
|
|
|
{
|
|
|
|
if (!(self = [super initWithFrame:frameRect]))
|
|
|
|
{
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
[self _init];
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
2016-07-03 20:58:25 +03:00
|
|
|
- (void)setFrame:(NSRect)frame
|
|
|
|
{
|
|
|
|
frame = self.superview.frame;
|
2016-07-05 21:23:55 +03:00
|
|
|
if (![[NSUserDefaults standardUserDefaults] boolForKey:@"GBAspectRatioUnkept"]) {
|
|
|
|
double ratio = frame.size.width / frame.size.height;
|
|
|
|
if (ratio >= 160.0/144.0) {
|
|
|
|
double new_width = round(frame.size.height / 144.0 * 160.0);
|
|
|
|
frame.origin.x = floor((frame.size.width - new_width) / 2);
|
|
|
|
frame.size.width = new_width;
|
|
|
|
frame.origin.y = 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
double new_height = round(frame.size.width / 160.0 * 144.0);
|
|
|
|
frame.origin.y = floor((frame.size.height - new_height) / 2);
|
|
|
|
frame.size.height = new_height;
|
|
|
|
frame.origin.x = 0;
|
|
|
|
}
|
2016-07-03 20:58:25 +03:00
|
|
|
}
|
2016-07-05 23:34:33 +03:00
|
|
|
|
2016-07-03 20:58:25 +03:00
|
|
|
[super setFrame:frame];
|
|
|
|
}
|
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
- (void)drawRect:(NSRect)dirtyRect {
|
2016-04-28 23:07:05 +03:00
|
|
|
if (!self.shader) {
|
|
|
|
self.shader = [[GBShader alloc] initWithName:[[NSUserDefaults standardUserDefaults] objectForKey:@"GBFilter"]];
|
|
|
|
}
|
2016-06-18 17:48:02 +03:00
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
double scale = self.window.backingScaleFactor;
|
2016-06-18 17:48:02 +03:00
|
|
|
glViewport(0, 0, self.bounds.size.width * scale, self.bounds.size.height * scale);
|
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
if (_shouldBlendFrameWithPrevious) {
|
2016-04-28 23:07:05 +03:00
|
|
|
[self.shader renderBitmap:image_buffers[current_buffer]
|
|
|
|
previous:image_buffers[(current_buffer + 2) % self.numberOfBuffers]
|
|
|
|
inSize:self.bounds.size
|
|
|
|
scale:scale];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
[self.shader renderBitmap:image_buffers[current_buffer]
|
|
|
|
previous:NULL
|
|
|
|
inSize:self.bounds.size
|
|
|
|
scale:scale];
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
glFlush();
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void) flip
|
|
|
|
{
|
|
|
|
current_buffer = (current_buffer + 1) % self.numberOfBuffers;
|
|
|
|
[self setNeedsDisplay:YES];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (uint32_t *) pixels
|
|
|
|
{
|
|
|
|
return image_buffers[(current_buffer + 1) % self.numberOfBuffers];
|
|
|
|
}
|
|
|
|
|
|
|
|
-(void)keyDown:(NSEvent *)theEvent
|
|
|
|
{
|
2017-01-24 21:00:56 +02:00
|
|
|
unsigned short keyCode = theEvent.keyCode;
|
2016-04-13 22:43:16 +03:00
|
|
|
bool handled = false;
|
|
|
|
|
|
|
|
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
|
|
|
|
for (GBButton i = 0; i < GBButtonCount; i++) {
|
2017-01-24 21:00:56 +02:00
|
|
|
if ([defaults integerForKey:button_to_preference_name(i)] == keyCode) {
|
2016-04-13 22:43:16 +03:00
|
|
|
handled = true;
|
|
|
|
switch (i) {
|
|
|
|
case GBTurbo:
|
|
|
|
_gb->turbo = true;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
_gb->keys[i] = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!handled) {
|
|
|
|
[super keyDown:theEvent];
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
-(void)keyUp:(NSEvent *)theEvent
|
|
|
|
{
|
2017-01-24 21:00:56 +02:00
|
|
|
unsigned short keyCode = theEvent.keyCode;
|
2016-04-13 22:43:16 +03:00
|
|
|
bool handled = false;
|
|
|
|
|
|
|
|
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
|
|
|
|
for (GBButton i = 0; i < GBButtonCount; i++) {
|
2017-01-24 21:00:56 +02:00
|
|
|
if ([defaults integerForKey:button_to_preference_name(i)] == keyCode) {
|
2016-04-13 22:43:16 +03:00
|
|
|
handled = true;
|
|
|
|
switch (i) {
|
|
|
|
case GBTurbo:
|
|
|
|
_gb->turbo = false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
_gb->keys[i] = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!handled) {
|
|
|
|
[super keyUp:theEvent];
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)acceptsFirstResponder
|
|
|
|
{
|
|
|
|
return YES;
|
|
|
|
}
|
2016-07-05 23:34:33 +03:00
|
|
|
|
|
|
|
- (void)mouseEntered:(NSEvent *)theEvent
|
|
|
|
{
|
|
|
|
if (!mouse_hidden) {
|
|
|
|
mouse_hidden = true;
|
|
|
|
if (_mouseHidingEnabled) {
|
|
|
|
[NSCursor hide];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
[super mouseEntered:theEvent];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)mouseExited:(NSEvent *)theEvent
|
|
|
|
{
|
|
|
|
if (mouse_hidden) {
|
|
|
|
mouse_hidden = false;
|
|
|
|
if (_mouseHidingEnabled) {
|
|
|
|
[NSCursor unhide];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
[super mouseExited:theEvent];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)setMouseHidingEnabled:(BOOL)mouseHidingEnabled
|
|
|
|
{
|
|
|
|
if (mouseHidingEnabled == _mouseHidingEnabled) return;
|
|
|
|
|
|
|
|
_mouseHidingEnabled = mouseHidingEnabled;
|
|
|
|
|
|
|
|
if (mouse_hidden && _mouseHidingEnabled) {
|
|
|
|
[NSCursor hide];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mouse_hidden && !_mouseHidingEnabled) {
|
|
|
|
[NSCursor unhide];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isMouseHidingEnabled
|
|
|
|
{
|
|
|
|
return _mouseHidingEnabled;
|
|
|
|
}
|
2016-03-30 23:07:55 +03:00
|
|
|
@end
|