blob: 66169f258994d690f58a10ce0371a095ae03f033 [file] [log] [blame]
//
// Copyright (c) 2015 The ANGLE Project Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
//
// OSXWindow.mm: Implementation of OSWindow for OSX
#include "osx/OSXWindow.h"
#include <set>
// Include Carbon to use the keycode names in Carbon's Event.h
#include <Carbon/Carbon.h>
#include <QuartzCore/CAMetalLayer.h>
// On OSX 10.12 a number of AppKit interfaces have been renamed for consistency, and the previous
// symbols tagged as deprecated. However we can't simply use the new symbols as it would break
// compilation on our automated testing that doesn't use OSX 10.12 yet. So we just ignore the
// warnings.
#pragma GCC diagnostic ignored "-Wdeprecated-declarations"
// Some events such as "ShouldTerminate" are sent to the whole application so we keep a list of
// all the windows in order to forward the event to each of them. However this and calling pushEvent
// in ApplicationDelegate is inherently unsafe in a multithreaded environment.
static std::set<OSXWindow*> gAllWindows;
@interface Application : NSApplication
@end
@implementation Application
- (void) sendEvent: (NSEvent*) nsEvent
{
if ([nsEvent type] == NSApplicationDefined)
{
for (auto window : gAllWindows)
{
if ([window->getNSWindow() windowNumber] == [nsEvent windowNumber])
{
Event event;
event.type = Event::EVENT_TEST;
window->pushEvent(event);
}
}
}
[super sendEvent: nsEvent];
}
@end
// The Delegate receiving application-wide events.
@interface ApplicationDelegate : NSObject
@end
@implementation ApplicationDelegate
- (NSApplicationTerminateReply) applicationShouldTerminate: (NSApplication*) sender
{
Event event;
event.type = Event::EVENT_CLOSED;
for (auto window : gAllWindows)
{
window->pushEvent(event);
}
return NSTerminateCancel;
}
@end
static ApplicationDelegate *gApplicationDelegate = nil;
static bool InitializeAppKit()
{
if (NSApp != nil)
{
return true;
}
// Initialize the global variable "NSApp"
[Application sharedApplication];
// Make us appear in the dock
[NSApp setActivationPolicy:NSApplicationActivationPolicyRegular];
// Register our global event handler
gApplicationDelegate = [[ApplicationDelegate alloc] init];
if (gApplicationDelegate == nil)
{
return false;
}
[NSApp setDelegate: static_cast<id>(gApplicationDelegate)];
// Set our status to "started" so we are not bouncing in the doc and can activate
[NSApp finishLaunching];
return true;
}
// NS's and CG's coordinate systems start at the bottom left, while OSWindow's coordinate
// system starts at the top left. This function converts the Y coordinate accordingly.
static float YCoordToFromCG(float y)
{
float screenHeight = CGDisplayBounds(CGMainDisplayID()).size.height;
return screenHeight - y;
}
// Delegate for window-wide events, note that the protocol doesn't contain anything input related.
@implementation WindowDelegate
- (id) initWithWindow: (OSXWindow*) window
{
self = [super init];
if (self != nil)
{
mWindow = window;
}
return self;
}
- (void) onOSXWindowDeleted
{
mWindow = nil;
}
- (BOOL) windowShouldClose: (id) sender
{
Event event;
event.type = Event::EVENT_CLOSED;
mWindow->pushEvent(event);
return NO;
}
- (void) windowDidResize: (NSNotification*) notification
{
NSSize windowSize = [[mWindow->getNSWindow() contentView] frame].size;
Event event;
event.type = Event::EVENT_RESIZED;
event.size.width = windowSize.width;
event.size.height = windowSize.height;
mWindow->pushEvent(event);
}
- (void) windowDidMove: (NSNotification*) notification
{
NSRect screenspace = [mWindow->getNSWindow() frame];
Event event;
event.type = Event::EVENT_MOVED;
event.move.x = screenspace.origin.x;
event.move.y = YCoordToFromCG(screenspace.origin.y + screenspace.size.height);
mWindow->pushEvent(event);
}
- (void) windowDidBecomeKey: (NSNotification*) notification
{
Event event;
event.type = Event::EVENT_GAINED_FOCUS;
mWindow->pushEvent(event);
[self retain];
}
- (void) windowDidResignKey: (NSNotification*) notification
{
if (mWindow != nil)
{
Event event;
event.type = Event::EVENT_LOST_FOCUS;
mWindow->pushEvent(event);
}
[self release];
}
@end
static Key NSCodeToKey(int keyCode)
{
// Missing KEY_PAUSE
switch (keyCode)
{
case kVK_Shift: return KEY_LSHIFT;
case kVK_RightShift: return KEY_RSHIFT;
case kVK_Option: return KEY_LALT;
case kVK_RightOption: return KEY_RALT;
case kVK_Control: return KEY_LCONTROL;
case kVK_RightControl: return KEY_RCONTROL;
case kVK_Command: return KEY_LSYSTEM;
// Right System doesn't have a name, but shows up as 0x36.
case 0x36: return KEY_RSYSTEM;
case kVK_Function: return KEY_MENU;
case kVK_ANSI_Semicolon: return KEY_SEMICOLON;
case kVK_ANSI_Slash: return KEY_SLASH;
case kVK_ANSI_Equal: return KEY_EQUAL;
case kVK_ANSI_Minus: return KEY_DASH;
case kVK_ANSI_LeftBracket: return KEY_LBRACKET;
case kVK_ANSI_RightBracket: return KEY_RBRACKET;
case kVK_ANSI_Comma: return KEY_COMMA;
case kVK_ANSI_Period: return KEY_PERIOD;
case kVK_ANSI_Backslash: return KEY_BACKSLASH;
case kVK_ANSI_Grave: return KEY_TILDE;
case kVK_Escape: return KEY_ESCAPE;
case kVK_Space: return KEY_SPACE;
case kVK_Return: return KEY_RETURN;
case kVK_Delete: return KEY_BACK;
case kVK_Tab: return KEY_TAB;
case kVK_PageUp: return KEY_PAGEUP;
case kVK_PageDown: return KEY_PAGEDOWN;
case kVK_End: return KEY_END;
case kVK_Home: return KEY_HOME;
case kVK_Help: return KEY_INSERT;
case kVK_ForwardDelete: return KEY_DELETE;
case kVK_ANSI_KeypadPlus: return KEY_ADD;
case kVK_ANSI_KeypadMinus: return KEY_SUBTRACT;
case kVK_ANSI_KeypadMultiply: return KEY_MULTIPLY;
case kVK_ANSI_KeypadDivide: return KEY_DIVIDE;
case kVK_F1: return KEY_F1;
case kVK_F2: return KEY_F2;
case kVK_F3: return KEY_F3;
case kVK_F4: return KEY_F4;
case kVK_F5: return KEY_F5;
case kVK_F6: return KEY_F6;
case kVK_F7: return KEY_F7;
case kVK_F8: return KEY_F8;
case kVK_F9: return KEY_F9;
case kVK_F10: return KEY_F10;
case kVK_F11: return KEY_F11;
case kVK_F12: return KEY_F12;
case kVK_F13: return KEY_F13;
case kVK_F14: return KEY_F14;
case kVK_F15: return KEY_F15;
case kVK_LeftArrow: return KEY_LEFT;
case kVK_RightArrow: return KEY_RIGHT;
case kVK_DownArrow: return KEY_DOWN;
case kVK_UpArrow: return KEY_UP;
case kVK_ANSI_Keypad0: return KEY_NUMPAD0;
case kVK_ANSI_Keypad1: return KEY_NUMPAD1;
case kVK_ANSI_Keypad2: return KEY_NUMPAD2;
case kVK_ANSI_Keypad3: return KEY_NUMPAD3;
case kVK_ANSI_Keypad4: return KEY_NUMPAD4;
case kVK_ANSI_Keypad5: return KEY_NUMPAD5;
case kVK_ANSI_Keypad6: return KEY_NUMPAD6;
case kVK_ANSI_Keypad7: return KEY_NUMPAD7;
case kVK_ANSI_Keypad8: return KEY_NUMPAD8;
case kVK_ANSI_Keypad9: return KEY_NUMPAD9;
case kVK_ANSI_A: return KEY_A;
case kVK_ANSI_B: return KEY_B;
case kVK_ANSI_C: return KEY_C;
case kVK_ANSI_D: return KEY_D;
case kVK_ANSI_E: return KEY_E;
case kVK_ANSI_F: return KEY_F;
case kVK_ANSI_G: return KEY_G;
case kVK_ANSI_H: return KEY_H;
case kVK_ANSI_I: return KEY_I;
case kVK_ANSI_J: return KEY_J;
case kVK_ANSI_K: return KEY_K;
case kVK_ANSI_L: return KEY_L;
case kVK_ANSI_M: return KEY_M;
case kVK_ANSI_N: return KEY_N;
case kVK_ANSI_O: return KEY_O;
case kVK_ANSI_P: return KEY_P;
case kVK_ANSI_Q: return KEY_Q;
case kVK_ANSI_R: return KEY_R;
case kVK_ANSI_S: return KEY_S;
case kVK_ANSI_T: return KEY_T;
case kVK_ANSI_U: return KEY_U;
case kVK_ANSI_V: return KEY_V;
case kVK_ANSI_W: return KEY_W;
case kVK_ANSI_X: return KEY_X;
case kVK_ANSI_Y: return KEY_Y;
case kVK_ANSI_Z: return KEY_Z;
case kVK_ANSI_1: return KEY_NUM1;
case kVK_ANSI_2: return KEY_NUM2;
case kVK_ANSI_3: return KEY_NUM3;
case kVK_ANSI_4: return KEY_NUM4;
case kVK_ANSI_5: return KEY_NUM5;
case kVK_ANSI_6: return KEY_NUM6;
case kVK_ANSI_7: return KEY_NUM7;
case kVK_ANSI_8: return KEY_NUM8;
case kVK_ANSI_9: return KEY_NUM9;
case kVK_ANSI_0: return KEY_NUM0;
}
return Key(0);
}
static void AddNSKeyStateToEvent(Event *event, int state)
{
event->key.shift = state & NSShiftKeyMask;
event->key.control = state & NSControlKeyMask;
event->key.alt = state & NSAlternateKeyMask;
event->key.system = state & NSCommandKeyMask;
}
static MouseButton TranslateMouseButton(int button)
{
switch (button)
{
case 2:
return MOUSEBUTTON_MIDDLE;
case 3:
return MOUSEBUTTON_BUTTON4;
case 4:
return MOUSEBUTTON_BUTTON5;
default:
return MOUSEBUTTON_UNKNOWN;
}
}
// Delegate for NSView events, mostly the input events
@implementation ContentView
- (id) initWithWindow: (OSXWindow*) window
{
self = [super init];
if (self != nil)
{
mWindow = window;
mTrackingArea = nil;
mCurrentModifier = 0;
[self updateTrackingAreas];
}
return self;
}
- (void) dealloc
{
[mTrackingArea release];
[super dealloc];
}
- (void) updateTrackingAreas
{
if (mTrackingArea != nil)
{
[self removeTrackingArea: mTrackingArea];
[mTrackingArea release];
mTrackingArea = nil;
}
NSRect bounds = [self bounds];
NSTrackingAreaOptions flags = NSTrackingMouseEnteredAndExited |
NSTrackingActiveInKeyWindow |
NSTrackingCursorUpdate |
NSTrackingInVisibleRect |
NSTrackingAssumeInside;
mTrackingArea = [[NSTrackingArea alloc] initWithRect: bounds
options: flags
owner: self
userInfo: nil];
[self addTrackingArea: mTrackingArea];
[super updateTrackingAreas];
}
// Helps with performance
- (BOOL) isOpaque
{
return YES;
}
- (BOOL) canBecomeKeyView
{
return YES;
}
- (BOOL) acceptsFirstResponder
{
return YES;
}
// Handle mouse events from the NSResponder protocol
- (float) translateMouseY: (float) y
{
return [self frame].size.height - y;
}
- (void) addButtonEvent: (NSEvent*) nsEvent type:(Event::EventType) eventType button:(MouseButton) button
{
Event event;
event.type = eventType;
event.mouseButton.button = button;
event.mouseButton.x = [nsEvent locationInWindow].x;
event.mouseButton.y = [self translateMouseY: [nsEvent locationInWindow].y];
mWindow->pushEvent(event);
}
- (void) mouseDown: (NSEvent*) event
{
[self addButtonEvent: event
type: Event::EVENT_MOUSE_BUTTON_PRESSED
button: MOUSEBUTTON_LEFT];
}
- (void) mouseDragged: (NSEvent*) event
{
[self mouseMoved: event];
}
- (void) mouseUp: (NSEvent*) event
{
[self addButtonEvent: event
type: Event::EVENT_MOUSE_BUTTON_RELEASED
button: MOUSEBUTTON_LEFT];
}
- (void) mouseMoved: (NSEvent*) nsEvent
{
Event event;
event.type = Event::EVENT_MOUSE_MOVED;
event.mouseMove.x = [nsEvent locationInWindow].x;
event.mouseMove.y = [self translateMouseY: [nsEvent locationInWindow].y];
mWindow->pushEvent(event);
}
- (void) mouseEntered: (NSEvent*) nsEvent
{
Event event;
event.type = Event::EVENT_MOUSE_ENTERED;
mWindow->pushEvent(event);
}
- (void) mouseExited: (NSEvent*) nsEvent
{
Event event;
event.type = Event::EVENT_MOUSE_LEFT;
mWindow->pushEvent(event);
}
- (void)rightMouseDown:(NSEvent *)event
{
[self addButtonEvent: event
type: Event::EVENT_MOUSE_BUTTON_PRESSED
button: MOUSEBUTTON_RIGHT];
}
- (void) rightMouseDragged: (NSEvent*) event
{
[self mouseMoved: event];
}
- (void) rightMouseUp: (NSEvent*)event
{
[self addButtonEvent: event
type: Event::EVENT_MOUSE_BUTTON_RELEASED
button: MOUSEBUTTON_RIGHT];
}
- (void) otherMouseDown: (NSEvent*) event
{
[self addButtonEvent: event
type: Event::EVENT_MOUSE_BUTTON_PRESSED
button: TranslateMouseButton([event buttonNumber])];
}
- (void) otherMouseDragged: (NSEvent*) event
{
[self mouseMoved: event];
}
- (void) otherMouseUp: (NSEvent*) event
{
[self addButtonEvent: event
type: Event::EVENT_MOUSE_BUTTON_RELEASED
button: TranslateMouseButton([event buttonNumber])];
}
- (void) scrollWheel: (NSEvent*) nsEvent
{
if (static_cast<int>([nsEvent deltaY]) == 0)
{
return;
}
Event event;
event.type = Event::EVENT_MOUSE_WHEEL_MOVED;
event.mouseWheel.delta = [nsEvent deltaY];
mWindow->pushEvent(event);
}
// Handle key events from the NSResponder protocol
- (void) keyDown: (NSEvent*) nsEvent
{
// TODO(cwallez) also send text events
Event event;
event.type = Event::EVENT_KEY_PRESSED;
event.key.code = NSCodeToKey([nsEvent keyCode]);
AddNSKeyStateToEvent(&event, [nsEvent modifierFlags]);
mWindow->pushEvent(event);
}
- (void) keyUp: (NSEvent*) nsEvent
{
Event event;
event.type = Event::EVENT_KEY_RELEASED;
event.key.code = NSCodeToKey([nsEvent keyCode]);
AddNSKeyStateToEvent(&event, [nsEvent modifierFlags]);
mWindow->pushEvent(event);
}
// Modifier keys do not trigger keyUp/Down events but only flagsChanged events.
- (void) flagsChanged: (NSEvent*) nsEvent
{
Event event;
// Guess if the key has been pressed or released with the change of modifiers
// It currently doesn't work when modifiers are unchanged, such as when pressing
// both shift keys. GLFW has a solution for this but it requires tracking the
// state of the keys. Implementing this is still TODO(cwallez)
int modifier = [nsEvent modifierFlags] & NSDeviceIndependentModifierFlagsMask;
if (modifier < mCurrentModifier)
{
event.type = Event::EVENT_KEY_RELEASED;
}
else
{
event.type = Event::EVENT_KEY_PRESSED;
}
mCurrentModifier = modifier;
event.key.code = NSCodeToKey([nsEvent keyCode]);
AddNSKeyStateToEvent(&event, [nsEvent modifierFlags]);
mWindow->pushEvent(event);
}
@end
OSXWindow::OSXWindow()
: mWindow(nil),
mDelegate(nil),
mView(nil)
{
}
OSXWindow::~OSXWindow()
{
destroy();
}
bool OSXWindow::initialize(const std::string &name, size_t width, size_t height)
{
if (!InitializeAppKit())
{
return false;
}
unsigned int styleMask = NSTitledWindowMask | NSClosableWindowMask | NSResizableWindowMask |
NSMiniaturizableWindowMask;
mWindow = [[NSWindow alloc] initWithContentRect: NSMakeRect(0, 0, width, height)
styleMask: styleMask
backing: NSBackingStoreBuffered
defer: NO];
if (mWindow == nil)
{
return false;
}
mDelegate = [[WindowDelegate alloc] initWithWindow: this];
if (mDelegate == nil)
{
return false;
}
[mWindow setDelegate: static_cast<id>(mDelegate)];
mView = [[ContentView alloc] initWithWindow: this];
if (mView == nil)
{
return false;
}
mLayer = [CAMetalLayer layer];
[mView setLayer:mLayer];
[mView setWantsLayer:YES];
[mWindow setContentView: mView];
[mWindow setTitle: [NSString stringWithUTF8String: name.c_str()]];
[mWindow setAcceptsMouseMovedEvents: YES];
[mWindow center];
[NSApp activateIgnoringOtherApps: YES];
mX = 0;
mY = 0;
mWidth = width;
mHeight = height;
gAllWindows.insert(this);
return true;
}
void OSXWindow::destroy()
{
gAllWindows.erase(this);
[mView release];
mView = nil;
[mDelegate onOSXWindowDeleted];
[mDelegate release];
mDelegate = nil;
[mWindow release];
mWindow = nil;
}
EGLNativeWindowType OSXWindow::getNativeWindow() const
{
return [mView layer];
}
EGLNativeDisplayType OSXWindow::getNativeDisplay() const
{
// TODO(cwallez): implement it once we have defined what EGLNativeDisplayType is
return static_cast<EGLNativeDisplayType>(0);
}
void* OSXWindow::getFramebufferNativeWindow() const
{
return static_cast<void*>(mWindow);
}
float OSXWindow::getDevicePixelRatio() const
{
return [[NSScreen mainScreen] backingScaleFactor];
}
void OSXWindow::messageLoop()
{
@autoreleasepool
{
while (true)
{
NSEvent* event = [NSApp nextEventMatchingMask: NSAnyEventMask
untilDate: [NSDate distantPast]
inMode: NSDefaultRunLoopMode
dequeue: YES];
if (event == nil)
{
break;
}
if ([event type] == NSAppKitDefined)
{
continue;
}
[NSApp sendEvent: event];
}
}
}
void OSXWindow::setMousePosition(int x, int y)
{
y = [mWindow frame].size.height - y -1;
NSPoint screenspace;
#if MAC_OS_X_VERSION_MIN_REQUIRED < MAC_OS_X_VERSION_10_7
screenspace = [mWindow convertBaseToScreen: NSMakePoint(x, y)];
#else
screenspace = [mWindow convertRectToScreen: NSMakeRect(x, y, 0, 0)].origin;
#endif
CGWarpMouseCursorPosition(CGPointMake(screenspace.x, YCoordToFromCG(screenspace.y)));
}
bool OSXWindow::setPosition(int x, int y)
{
// Given CG and NS's coordinate system, the "Y" position of a window is the Y coordinate
// of the bottom of the window.
int newBottom = [mWindow frame].size.height + y;
NSRect emptyRect = NSMakeRect(x, YCoordToFromCG(newBottom), 0, 0);
[mWindow setFrameOrigin: [mWindow frameRectForContentRect: emptyRect].origin];
return true;
}
bool OSXWindow::resize(int width, int height)
{
[mWindow setContentSize: NSMakeSize(width, height)];
return true;
}
void OSXWindow::setVisible(bool isVisible)
{
if (isVisible)
{
[mWindow makeKeyAndOrderFront: nil];
}
else
{
[mWindow orderOut: nil];
}
}
void OSXWindow::signalTestEvent()
{
@autoreleasepool
{
NSEvent *event = [NSEvent otherEventWithType: NSApplicationDefined
location: NSMakePoint(0, 0)
modifierFlags: 0
timestamp: 0.0
windowNumber: [mWindow windowNumber]
context: nil
subtype: 0
data1: 0
data2: 0];
[NSApp postEvent: event atStart: YES];
}
}
NSWindow* OSXWindow::getNSWindow() const
{
return mWindow;
}
OSWindow *CreateOSWindow()
{
return new OSXWindow;
}