2016-06-09 00:37:00 +03:00
|
|
|
#import <Carbon/Carbon.h>
|
2016-03-30 23:07:55 +03:00
|
|
|
#import "GBView.h"
|
2018-06-11 20:23:51 +03:00
|
|
|
#import "GBViewGL.h"
|
2018-06-15 12:58:33 +03:00
|
|
|
#import "GBViewMetal.h"
|
2016-04-13 22:43:16 +03:00
|
|
|
#import "GBButtons.h"
|
|
|
|
#import "NSString+StringForKey.h"
|
2016-03-30 23:07:55 +03:00
|
|
|
|
2018-06-26 19:36:14 +03:00
|
|
|
#define JOYSTICK_HIGH 0x4000
|
|
|
|
#define JOYSTICK_LOW 0x3800
|
|
|
|
|
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;
|
2018-06-26 19:36:14 +03:00
|
|
|
bool axisActive[2];
|
2018-02-10 23:30:30 +02:00
|
|
|
bool underclockKeyDown;
|
|
|
|
double clockMultiplier;
|
|
|
|
NSEventModifierFlags previousModifiers;
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
|
2018-06-11 20:23:51 +03:00
|
|
|
+ (instancetype)alloc
|
2016-06-14 14:11:37 +03:00
|
|
|
{
|
2018-06-15 12:58:33 +03:00
|
|
|
return [self allocWithZone:NULL];
|
2018-06-11 20:23:51 +03:00
|
|
|
}
|
2016-06-14 14:11:37 +03:00
|
|
|
|
2018-06-11 20:23:51 +03:00
|
|
|
+ (instancetype)allocWithZone:(struct _NSZone *)zone
|
|
|
|
{
|
|
|
|
if (self == [GBView class]) {
|
2018-06-15 20:03:59 +03:00
|
|
|
if ([GBViewMetal isSupported]) {
|
|
|
|
return [GBViewMetal allocWithZone: zone];
|
|
|
|
}
|
|
|
|
return [GBViewGL allocWithZone: zone];
|
2018-06-11 20:23:51 +03:00
|
|
|
}
|
|
|
|
return [super allocWithZone:zone];
|
2016-06-14 14:11:37 +03:00
|
|
|
}
|
|
|
|
|
2018-06-11 20:23:51 +03:00
|
|
|
- (void) createInternalView
|
|
|
|
{
|
|
|
|
assert(false && "createInternalView must not be inherited");
|
|
|
|
}
|
2016-06-14 14:11:37 +03:00
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
- (void) _init
|
2018-11-23 19:59:15 +02:00
|
|
|
{
|
2016-03-30 23:07:55 +03:00
|
|
|
_shouldBlendFrameWithPrevious = 1;
|
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];
|
2018-02-10 23:30:30 +02:00
|
|
|
clockMultiplier = 1.0;
|
2018-06-11 20:23:51 +03:00
|
|
|
[self createInternalView];
|
|
|
|
[self addSubview:self.internalView];
|
|
|
|
self.internalView.autoresizingMask = NSViewWidthSizable | NSViewHeightSizable;
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
|
2018-11-15 00:21:21 +02:00
|
|
|
- (void)screenSizeChanged
|
|
|
|
{
|
|
|
|
if (image_buffers[0]) free(image_buffers[0]);
|
|
|
|
if (image_buffers[1]) free(image_buffers[1]);
|
|
|
|
if (image_buffers[2]) free(image_buffers[2]);
|
|
|
|
|
|
|
|
size_t buffer_size = sizeof(image_buffers[0][0]) * GB_get_screen_width(_gb) * GB_get_screen_height(_gb);
|
|
|
|
|
|
|
|
image_buffers[0] = malloc(buffer_size);
|
|
|
|
image_buffers[1] = malloc(buffer_size);
|
|
|
|
image_buffers[2] = malloc(buffer_size);
|
|
|
|
|
|
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
|
|
[self setFrame:self.superview.frame];
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
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;
|
2018-11-23 19:59:15 +02:00
|
|
|
if (_gb && ![[NSUserDefaults standardUserDefaults] boolForKey:@"GBAspectRatioUnkept"]) {
|
2016-07-05 21:23:55 +03:00
|
|
|
double ratio = frame.size.width / frame.size.height;
|
2018-11-15 00:21:21 +02:00
|
|
|
double width = GB_get_screen_width(_gb);
|
|
|
|
double height = GB_get_screen_height(_gb);
|
|
|
|
if (ratio >= width / height) {
|
|
|
|
double new_width = round(frame.size.height / height * width);
|
2016-07-05 21:23:55 +03:00
|
|
|
frame.origin.x = floor((frame.size.width - new_width) / 2);
|
|
|
|
frame.size.width = new_width;
|
|
|
|
frame.origin.y = 0;
|
|
|
|
}
|
|
|
|
else {
|
2018-11-15 00:21:21 +02:00
|
|
|
double new_height = round(frame.size.width / width * height);
|
2016-07-05 21:23:55 +03:00
|
|
|
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) flip
|
|
|
|
{
|
2018-02-10 23:30:30 +02:00
|
|
|
if (underclockKeyDown && clockMultiplier > 0.5) {
|
|
|
|
clockMultiplier -= 0.1;
|
|
|
|
GB_set_clock_multiplier(_gb, clockMultiplier);
|
|
|
|
}
|
|
|
|
if (!underclockKeyDown && clockMultiplier < 1.0) {
|
|
|
|
clockMultiplier += 0.1;
|
|
|
|
GB_set_clock_multiplier(_gb, clockMultiplier);
|
|
|
|
}
|
2016-03-30 23:07:55 +03:00
|
|
|
current_buffer = (current_buffer + 1) % self.numberOfBuffers;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (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];
|
2018-12-15 18:55:41 +02:00
|
|
|
unsigned player_count = GB_get_player_count(_gb);
|
2018-12-05 00:00:16 +02:00
|
|
|
for (unsigned player = 0; player < player_count; player++) {
|
|
|
|
for (GBButton button = 0; button < GBButtonCount; button++) {
|
|
|
|
NSNumber *key = [defaults valueForKey:button_to_preference_name(button, player)];
|
|
|
|
if (!key) continue;
|
|
|
|
|
|
|
|
if (key.unsignedShortValue == keyCode) {
|
|
|
|
handled = true;
|
|
|
|
switch (button) {
|
|
|
|
case GBTurbo:
|
|
|
|
GB_set_turbo_mode(_gb, true, self.isRewinding);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GBRewind:
|
|
|
|
self.isRewinding = true;
|
|
|
|
GB_set_turbo_mode(_gb, false, false);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GBUnderclock:
|
|
|
|
underclockKeyDown = true;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
GB_set_key_state_for_player(_gb, (GB_key_t)button, player, true);
|
|
|
|
break;
|
|
|
|
}
|
2016-04-13 22:43:16 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-02 19:37:40 +02:00
|
|
|
if (!handled && [theEvent type] != NSEventTypeFlagsChanged) {
|
2016-04-13 22:43:16 +03:00
|
|
|
[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];
|
2018-12-15 18:55:41 +02:00
|
|
|
unsigned player_count = GB_get_player_count(_gb);
|
2018-12-05 00:00:16 +02:00
|
|
|
for (unsigned player = 0; player < player_count; player++) {
|
|
|
|
for (GBButton button = 0; button < GBButtonCount; button++) {
|
|
|
|
NSNumber *key = [defaults valueForKey:button_to_preference_name(button, player)];
|
|
|
|
if (!key) continue;
|
|
|
|
|
|
|
|
if (key.unsignedShortValue == keyCode) {
|
|
|
|
handled = true;
|
|
|
|
switch (button) {
|
|
|
|
case GBTurbo:
|
|
|
|
GB_set_turbo_mode(_gb, false, false);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GBRewind:
|
|
|
|
self.isRewinding = false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GBUnderclock:
|
|
|
|
underclockKeyDown = false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
GB_set_key_state_for_player(_gb, (GB_key_t)button, player, false);
|
|
|
|
break;
|
|
|
|
}
|
2016-04-13 22:43:16 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-03-02 19:37:40 +02:00
|
|
|
if (!handled && [theEvent type] != NSEventTypeFlagsChanged) {
|
2016-04-13 22:43:16 +03:00
|
|
|
[super keyUp:theEvent];
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-30 16:23:17 +02:00
|
|
|
- (void) joystick:(NSString *)joystick_name button: (unsigned)button changedState: (bool) state
|
|
|
|
{
|
2018-12-15 18:55:41 +02:00
|
|
|
unsigned player_count = GB_get_player_count(_gb);
|
|
|
|
|
2018-02-02 19:22:08 +02:00
|
|
|
UpdateSystemActivity(UsrActivity);
|
2018-12-15 18:55:41 +02:00
|
|
|
for (unsigned player = 0; player < player_count; player++) {
|
|
|
|
NSString *preferred_joypad = [[[NSUserDefaults standardUserDefaults] dictionaryForKey:@"GBDefaultJoypads"]
|
|
|
|
objectForKey:[NSString stringWithFormat:@"%u", player]];
|
|
|
|
if (player_count != 1 && // Single player, accpet inputs from all joypads
|
|
|
|
!(player == 0 && !preferred_joypad) && // Multiplayer, but player 1 has no joypad configured, so it takes inputs from all joypads
|
|
|
|
![preferred_joypad isEqualToString:joystick_name]) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
NSDictionary *mapping = [[NSUserDefaults standardUserDefaults] dictionaryForKey:@"GBJoypadMappings"][joystick_name];
|
|
|
|
|
|
|
|
for (GBButton i = 0; i < GBButtonCount; i++) {
|
|
|
|
NSNumber *mapped_button = [mapping objectForKey:GBButtonNames[i]];
|
|
|
|
if (mapped_button && [mapped_button integerValue] == button) {
|
|
|
|
switch (i) {
|
|
|
|
case GBTurbo:
|
|
|
|
GB_set_turbo_mode(_gb, state, state && self.isRewinding);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GBRewind:
|
|
|
|
self.isRewinding = state;
|
|
|
|
if (state) {
|
|
|
|
GB_set_turbo_mode(_gb, false, false);
|
|
|
|
}
|
|
|
|
break;
|
2017-12-30 16:23:17 +02:00
|
|
|
|
2018-12-15 18:55:41 +02:00
|
|
|
case GBUnderclock:
|
|
|
|
underclockKeyDown = state;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
GB_set_key_state_for_player(_gb, (GB_key_t)i, player, state);
|
|
|
|
break;
|
|
|
|
}
|
2017-12-30 16:23:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void) joystick:(NSString *)joystick_name axis: (unsigned)axis movedTo: (signed) value
|
|
|
|
{
|
2018-12-15 18:55:41 +02:00
|
|
|
unsigned player_count = GB_get_player_count(_gb);
|
|
|
|
|
2018-02-02 19:22:08 +02:00
|
|
|
UpdateSystemActivity(UsrActivity);
|
2018-12-15 18:55:41 +02:00
|
|
|
for (unsigned player = 0; player < player_count; player++) {
|
|
|
|
NSString *preferred_joypad = [[[NSUserDefaults standardUserDefaults] dictionaryForKey:@"GBDefaultJoypads"]
|
|
|
|
objectForKey:[NSString stringWithFormat:@"%u", player]];
|
|
|
|
if (player_count != 1 && // Single player, accpet inputs from all joypads
|
|
|
|
!(player == 0 && !preferred_joypad) && // Multiplayer, but player 1 has no joypad configured, so it takes inputs from all joypads
|
|
|
|
![preferred_joypad isEqualToString:joystick_name]) {
|
|
|
|
continue;
|
2018-06-26 19:36:14 +03:00
|
|
|
}
|
2018-12-15 18:55:41 +02:00
|
|
|
|
|
|
|
NSDictionary *mapping = [[NSUserDefaults standardUserDefaults] dictionaryForKey:@"GBJoypadMappings"][joystick_name];
|
|
|
|
NSNumber *x_axis = [mapping objectForKey:@"XAxis"];
|
|
|
|
NSNumber *y_axis = [mapping objectForKey:@"YAxis"];
|
|
|
|
|
|
|
|
if (axis == [x_axis integerValue]) {
|
|
|
|
if (value > JOYSTICK_HIGH) {
|
|
|
|
axisActive[0] = true;
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_RIGHT, player, true);
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_LEFT, player, false);
|
|
|
|
}
|
|
|
|
else if (value < -JOYSTICK_HIGH) {
|
|
|
|
axisActive[0] = true;
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_RIGHT, player, false);
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_LEFT, player, true);
|
|
|
|
}
|
|
|
|
else if (axisActive[0] && value < JOYSTICK_LOW && value > -JOYSTICK_LOW) {
|
|
|
|
axisActive[0] = false;
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_RIGHT, player, false);
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_LEFT, player, false);
|
|
|
|
}
|
2018-06-26 19:36:14 +03:00
|
|
|
}
|
2018-12-15 18:55:41 +02:00
|
|
|
else if (axis == [y_axis integerValue]) {
|
|
|
|
if (value > JOYSTICK_HIGH) {
|
|
|
|
axisActive[1] = true;
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_DOWN, player, true);
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_UP, player, false);
|
|
|
|
}
|
|
|
|
else if (value < -JOYSTICK_HIGH) {
|
|
|
|
axisActive[1] = true;
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_DOWN, player, false);
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_UP, player, true);
|
|
|
|
}
|
|
|
|
else if (axisActive[1] && value < JOYSTICK_LOW && value > -JOYSTICK_LOW) {
|
|
|
|
axisActive[1] = false;
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_DOWN, player, false);
|
|
|
|
GB_set_key_state_for_player(_gb, GB_KEY_UP, player, false);
|
|
|
|
}
|
2018-06-26 19:36:14 +03:00
|
|
|
}
|
2017-12-30 16:23:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2017-12-30 16:23:17 +02:00
|
|
|
|
2018-02-10 23:30:30 +02:00
|
|
|
- (void) flagsChanged:(NSEvent *)event
|
|
|
|
{
|
|
|
|
if (event.modifierFlags > previousModifiers) {
|
|
|
|
[self keyDown:event];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
[self keyUp:event];
|
|
|
|
}
|
|
|
|
|
|
|
|
previousModifiers = event.modifierFlags;
|
|
|
|
}
|
|
|
|
|
2018-06-11 20:23:51 +03:00
|
|
|
- (uint32_t *)currentBuffer
|
|
|
|
{
|
|
|
|
return image_buffers[current_buffer];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (uint32_t *)previousBuffer
|
|
|
|
{
|
|
|
|
return image_buffers[(current_buffer + 2) % self.numberOfBuffers];
|
|
|
|
}
|
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
@end
|