2015-06-21 17:33:46 +02:00
/ *
Simple DirectMedia Layer
Copyright ( C ) 1997 -2015 Sam Lantinga < slouken @ libsdl . org >
This software is provided ' as - is ' , without any express or implied
warranty . In no event will the authors be held liable for any damages
arising from the use of this software .
Permission is granted to anyone to use this software for any purpose ,
including commercial applications , and to alter it and redistribute it
freely , subject to the following restrictions :
1. The origin of this software must not be misrepresented ; you must not
claim that you wrote the original software . If you use this software
in a product , an acknowledgment in the product documentation would be
appreciated but is not required .
2. Altered source versions must be plainly marked as such , and must not be
misrepresented as being the original software .
3. This notice may not be removed or altered from any source distribution .
* /
# include "../../SDL_internal.h"
# if SDL_VIDEO _DRIVER _COCOA
# include "SDL_cocoavideo.h"
2015-11-14 09:24:39 +01:00
# include "../../events/SDL_events_c.h"
2015-06-21 17:33:46 +02:00
# include "../../events/SDL_keyboard_c.h"
# include "../../events/scancodes_darwin.h"
# include < Carbon / Carbon . h >
/ * # define DEBUG_IME NSLog * /
# define DEBUG_IME ( . . . )
2015-12-11 03:17:22 +01:00
@ interface SDLTranslatorResponder : NSView < NSTextInputClient > {
2015-06-21 17:33:46 +02:00
NSString * _markedText ;
NSRange _markedRange ;
NSRange _selectedRange ;
SDL_Rect _inputRect ;
}
2015-12-11 03:17:22 +01:00
- ( void ) doCommandBySelector : ( SEL ) myselector ;
- ( void ) setInputRect : ( SDL_Rect * ) rect ;
2015-06-21 17:33:46 +02:00
@ end
@ implementation SDLTranslatorResponder
2015-12-11 03:17:22 +01:00
- ( void ) setInputRect : ( SDL_Rect * ) rect
2015-06-21 17:33:46 +02:00
{
_inputRect = * rect ;
}
2015-12-11 03:17:22 +01:00
- ( void ) insertText : ( id ) aString replacementRange : ( NSRange ) replacementRange
2015-06-21 17:33:46 +02:00
{
2015-12-11 03:17:22 +01:00
/ * TODO : Make use of replacementRange ? * /
2015-06-21 17:33:46 +02:00
const char * str ;
DEBUG_IME ( @ "insertText: %@" , aString ) ;
/ * Could be NSString or NSAttributedString , so we have
* to test and convert it before return as SDL event * /
if ( [ aString isKindOfClass : [ NSAttributedString class ] ] ) {
str = [ [ aString string ] UTF8String ] ;
} else {
str = [ aString UTF8String ] ;
}
SDL_SendKeyboardText ( str ) ;
}
2015-12-11 03:17:22 +01:00
- ( void ) insertText : ( id ) insertString
{
/ * This method is part of NSTextInput and not NSTextInputClient , but
* apparently it still might be called in OS X 10.5 and can cause beeps if
* the implementation is missing : http : // crbug . com / 47890 * /
[ self insertText : insertString replacementRange : NSMakeRange ( 0 , 0 ) ] ;
}
- ( void ) doCommandBySelector : ( SEL ) myselector
2015-06-21 17:33:46 +02:00
{
/ * No need to do anything since we are not using Cocoa
selectors to handle special keys , instead we use SDL
key events to do the same job .
* /
}
2015-12-11 03:17:22 +01:00
- ( BOOL ) hasMarkedText
2015-06-21 17:33:46 +02:00
{
return _markedText ! = nil ;
}
2015-12-11 03:17:22 +01:00
- ( NSRange ) markedRange
2015-06-21 17:33:46 +02:00
{
return _markedRange ;
}
2015-12-11 03:17:22 +01:00
- ( NSRange ) selectedRange
2015-06-21 17:33:46 +02:00
{
return _selectedRange ;
}
2015-12-11 03:17:22 +01:00
- ( void ) setMarkedText : ( id ) aString selectedRange : ( NSRange ) selectedRange replacementRange : ( NSRange ) replacementRange ;
2015-06-21 17:33:46 +02:00
{
if ( [ aString isKindOfClass : [ NSAttributedString class ] ] ) {
aString = [ aString string ] ;
}
if ( [ aString length ] = = 0 ) {
[ self unmarkText ] ;
return ;
}
if ( _markedText ! = aString ) {
[ _markedText release ] ;
_markedText = [ aString retain ] ;
}
2015-12-11 03:17:22 +01:00
_selectedRange = selectedRange ;
2015-06-21 17:33:46 +02:00
_markedRange = NSMakeRange ( 0 , [ aString length ] ) ;
SDL_SendEditingText ( [ aString UTF8String ] ,
2015-12-11 03:17:22 +01:00
selectedRange . location , selectedRange . length ) ;
2015-06-21 17:33:46 +02:00
DEBUG_IME ( @ "setMarkedText: %@, (%d, %d)" , _markedText ,
selRange . location , selRange . length ) ;
}
2015-12-11 03:17:22 +01:00
- ( void ) unmarkText
2015-06-21 17:33:46 +02:00
{
[ _markedText release ] ;
_markedText = nil ;
SDL_SendEditingText ( "" , 0 , 0 ) ;
}
2015-12-11 03:17:22 +01:00
- ( NSRect ) firstRectForCharacterRange : ( NSRange ) aRange actualRange : ( NSRangePointer ) actualRange ;
2015-06-21 17:33:46 +02:00
{
NSWindow * window = [ self window ] ;
2015-12-11 03:17:22 +01:00
NSRect contentRect = [ window contentRectForFrameRect : [ window frame ] ] ;
2015-06-21 17:33:46 +02:00
float windowHeight = contentRect . size . height ;
NSRect rect = NSMakeRect ( _inputRect . x , windowHeight - _inputRect . y - _inputRect . h ,
_inputRect . w , _inputRect . h ) ;
2015-12-11 03:17:22 +01:00
if ( actualRange ) {
* actualRange = aRange ;
}
2015-06-21 17:33:46 +02:00
DEBUG_IME ( @ "firstRectForCharacterRange: (%d, %d): windowHeight = %g, rect = %@" ,
2015-12-11 03:17:22 +01:00
aRange . location , aRange . length , windowHeight ,
2015-06-21 17:33:46 +02:00
NSStringFromRect ( rect ) ) ;
2015-12-11 03:17:22 +01:00
if ( [ [ self window ] respondsToSelector : @ selector ( convertRectToScreen : ) ] ) {
rect = [ [ self window ] convertRectToScreen : rect ] ;
} else {
rect . origin = [ [ self window ] convertBaseToScreen : rect . origin ] ;
}
2015-06-21 17:33:46 +02:00
return rect ;
}
2015-12-11 03:17:22 +01:00
- ( NSAttributedString * ) attributedSubstringForProposedRange : ( NSRange ) aRange actualRange : ( NSRangePointer ) actualRange ;
2015-06-21 17:33:46 +02:00
{
2015-12-11 03:17:22 +01:00
DEBUG_IME ( @ "attributedSubstringFromRange: (%d, %d)" , aRange . location , aRange . length ) ;
2015-06-21 17:33:46 +02:00
return nil ;
}
2015-12-11 03:17:22 +01:00
- ( NSInteger ) conversationIdentifier
2015-06-21 17:33:46 +02:00
{
return ( NSInteger ) self ;
}
/ * This method returns the index for character that is
* nearest to thePoint . thPoint is in screen coordinate system .
* /
2015-12-11 03:17:22 +01:00
- ( NSUInteger ) characterIndexForPoint : ( NSPoint ) thePoint
2015-06-21 17:33:46 +02:00
{
DEBUG_IME ( @ "characterIndexForPoint: (%g, %g)" , thePoint . x , thePoint . y ) ;
return 0 ;
}
/ * This method is the key to attribute extension .
* We could add new attributes through this method .
* NSInputServer examines the return value of this
* method & constructs appropriate attributed string .
* /
2015-12-11 03:17:22 +01:00
- ( NSArray * ) validAttributesForMarkedText
2015-06-21 17:33:46 +02:00
{
return [ NSArray array ] ;
}
@ end
/ * This is a helper function for HandleModifierSide . This
* function reverts back to behavior before the distinction between
* sides was made .
* /
static void
HandleNonDeviceModifier ( unsigned int device_independent _mask ,
unsigned int oldMods ,
unsigned int newMods ,
SDL_Scancode scancode )
{
unsigned int oldMask , newMask ;
/ * Isolate just the bits we care about in the depedent bits so we can
* figure out what changed
* /
oldMask = oldMods & device_independent _mask ;
newMask = newMods & device_independent _mask ;
if ( oldMask && oldMask ! = newMask ) {
SDL_SendKeyboardKey ( SDL_RELEASED , scancode ) ;
} else if ( newMask && oldMask ! = newMask ) {
SDL_SendKeyboardKey ( SDL_PRESSED , scancode ) ;
}
}
/ * This is a helper function for HandleModifierSide .
* This function sets the actual SDL_PrivateKeyboard event .
* /
static void
HandleModifierOneSide ( unsigned int oldMods , unsigned int newMods ,
SDL_Scancode scancode ,
unsigned int sided_device _dependent _mask )
{
unsigned int old_dep _mask , new_dep _mask ;
/ * Isolate just the bits we care about in the depedent bits so we can
* figure out what changed
* /
old_dep _mask = oldMods & sided_device _dependent _mask ;
new_dep _mask = newMods & sided_device _dependent _mask ;
/ * We now know that this side bit flipped . But we don ' t know if
* it went pressed to released or released to pressed , so we must
* find out which it is .
* /
if ( new_dep _mask && old_dep _mask ! = new_dep _mask ) {
SDL_SendKeyboardKey ( SDL_PRESSED , scancode ) ;
} else {
SDL_SendKeyboardKey ( SDL_RELEASED , scancode ) ;
}
}
/ * This is a helper function for DoSidedModifiers .
* This function will figure out if the modifier key is the left or right side ,
* e . g . left - shift vs right - shift .
* /
static void
HandleModifierSide ( int device_independent _mask ,
unsigned int oldMods , unsigned int newMods ,
SDL_Scancode left_scancode ,
SDL_Scancode right_scancode ,
unsigned int left_device _dependent _mask ,
unsigned int right_device _dependent _mask )
{
unsigned int device_dependent _mask = ( left_device _dependent _mask |
right_device _dependent _mask ) ;
unsigned int diff_mod ;
/ * On the basis that the device independent mask is set , but there are
* no device dependent flags set , we ' ll assume that we can ' t detect this
* keyboard and revert to the unsided behavior .
* /
if ( ( device_dependent _mask & newMods ) = = 0 ) {
/ * Revert to the old behavior * /
HandleNonDeviceModifier ( device_independent _mask , oldMods , newMods , left_scancode ) ;
return ;
}
/ * XOR the previous state against the new state to see if there ' s a change * /
diff_mod = ( device_dependent _mask & oldMods ) ^
( device_dependent _mask & newMods ) ;
if ( diff_mod ) {
/ * A change in state was found . Isolate the left and right bits
* to handle them separately just in case the values can simulataneously
* change or if the bits don ' t both exist .
* /
if ( left_device _dependent _mask & diff_mod ) {
HandleModifierOneSide ( oldMods , newMods , left_scancode , left_device _dependent _mask ) ;
}
if ( right_device _dependent _mask & diff_mod ) {
HandleModifierOneSide ( oldMods , newMods , right_scancode , right_device _dependent _mask ) ;
}
}
}
/ * This is a helper function for DoSidedModifiers .
* This function will release a key press in the case that
* it is clear that the modifier has been released ( i . e . one side
* can ' t still be down ) .
* /
static void
ReleaseModifierSide ( unsigned int device_independent _mask ,
unsigned int oldMods , unsigned int newMods ,
SDL_Scancode left_scancode ,
SDL_Scancode right_scancode ,
unsigned int left_device _dependent _mask ,
unsigned int right_device _dependent _mask )
{
unsigned int device_dependent _mask = ( left_device _dependent _mask |
right_device _dependent _mask ) ;
/ * On the basis that the device independent mask is set , but there are
* no device dependent flags set , we ' ll assume that we can ' t detect this
* keyboard and revert to the unsided behavior .
* /
if ( ( device_dependent _mask & oldMods ) = = 0 ) {
/ * In this case , we can ' t detect the keyboard , so use the left side
* to represent both , and release it .
* /
SDL_SendKeyboardKey ( SDL_RELEASED , left_scancode ) ;
return ;
}
/ *
* This could have been done in an if - else case because at this point ,
* we know that all keys have been released when calling this function .
* But I ' m being paranoid so I want to handle each separately ,
* so I hope this doesn ' t cause other problems .
* /
if ( left_device _dependent _mask & oldMods ) {
SDL_SendKeyboardKey ( SDL_RELEASED , left_scancode ) ;
}
if ( right_device _dependent _mask & oldMods ) {
SDL_SendKeyboardKey ( SDL_RELEASED , right_scancode ) ;
}
}
/ * This is a helper function for DoSidedModifiers .
* This function handles the CapsLock case .
* /
static void
HandleCapsLock ( unsigned short scancode ,
unsigned int oldMods , unsigned int newMods )
{
unsigned int oldMask , newMask ;
oldMask = oldMods & NSAlphaShiftKeyMask ;
newMask = newMods & NSAlphaShiftKeyMask ;
if ( oldMask ! = newMask ) {
SDL_SendKeyboardKey ( SDL_PRESSED , SDL_SCANCODE _CAPSLOCK ) ;
SDL_SendKeyboardKey ( SDL_RELEASED , SDL_SCANCODE _CAPSLOCK ) ;
}
}
/ * This function will handle the modifier keys and also determine the
* correct side of the key .
* /
static void
DoSidedModifiers ( unsigned short scancode ,
unsigned int oldMods , unsigned int newMods )
{
/ * Set up arrays for the key syms for the left and right side . * /
const SDL_Scancode left_mapping [ ] = {
SDL_SCANCODE _LSHIFT ,
SDL_SCANCODE _LCTRL ,
SDL_SCANCODE _LALT ,
SDL_SCANCODE _LGUI
} ;
const SDL_Scancode right_mapping [ ] = {
SDL_SCANCODE _RSHIFT ,
SDL_SCANCODE _RCTRL ,
SDL_SCANCODE _RALT ,
SDL_SCANCODE _RGUI
} ;
/ * Set up arrays for the device dependent masks with indices that
* correspond to the _mapping arrays
* /
const unsigned int left_device _mapping [ ] = { NX_DEVICELSHIFTKEYMASK , NX_DEVICELCTLKEYMASK , NX_DEVICELALTKEYMASK , NX_DEVICELCMDKEYMASK } ;
const unsigned int right_device _mapping [ ] = { NX_DEVICERSHIFTKEYMASK , NX_DEVICERCTLKEYMASK , NX_DEVICERALTKEYMASK , NX_DEVICERCMDKEYMASK } ;
unsigned int i , bit ;
/ * Handle CAPSLOCK separately because it doesn ' t have a left / right side * /
HandleCapsLock ( scancode , oldMods , newMods ) ;
/ * Iterate through the bits , testing each against the old modifiers * /
for ( i = 0 , bit = NSShiftKeyMask ; bit <= NSCommandKeyMask ; bit < <= 1 , + + i ) {
unsigned int oldMask , newMask ;
oldMask = oldMods & bit ;
newMask = newMods & bit ;
/ * If the bit is set , we must always examine it because the left
* and right side keys may alternate or both may be pressed .
* /
if ( newMask ) {
HandleModifierSide ( bit , oldMods , newMods ,
left_mapping [ i ] , right_mapping [ i ] ,
left_device _mapping [ i ] , right_device _mapping [ i ] ) ;
}
/ * If the state changed from pressed to unpressed , we must examine
* the device dependent bits to release the correct keys .
* /
else if ( oldMask && oldMask ! = newMask ) {
ReleaseModifierSide ( bit , oldMods , newMods ,
left_mapping [ i ] , right_mapping [ i ] ,
left_device _mapping [ i ] , right_device _mapping [ i ] ) ;
}
}
}
static void
HandleModifiers ( _THIS , unsigned short scancode , unsigned int modifierFlags )
{
SDL_VideoData * data = ( SDL_VideoData * ) _this -> driverdata ;
if ( modifierFlags = = data -> modifierFlags ) {
return ;
}
DoSidedModifiers ( scancode , data -> modifierFlags , modifierFlags ) ;
data -> modifierFlags = modifierFlags ;
}
static void
2015-10-27 19:17:32 +01:00
UpdateKeymap ( SDL_VideoData * data , SDL_bool send_event )
2015-06-21 17:33:46 +02:00
{
TISInputSourceRef key_layout ;
const void * chr_data ;
int i ;
SDL_Scancode scancode ;
SDL_Keycode keymap [ SDL_NUM _SCANCODES ] ;
/ * See if the keymap needs to be updated * /
key_layout = TISCopyCurrentKeyboardLayoutInputSource ( ) ;
if ( key_layout = = data -> key_layout ) {
return ;
}
data -> key_layout = key_layout ;
SDL_GetDefaultKeymap ( keymap ) ;
/ * Try Unicode data first * /
CFDataRef uchrDataRef = TISGetInputSourceProperty ( key_layout , kTISPropertyUnicodeKeyLayoutData ) ;
if ( uchrDataRef ) {
chr_data = CFDataGetBytePtr ( uchrDataRef ) ;
} else {
goto cleanup ;
}
if ( chr_data ) {
UInt32 keyboard_type = LMGetKbdType ( ) ;
OSStatus err ;
for ( i = 0 ; i < SDL_arraysize ( darwin_scancode _table ) ; i + + ) {
UniChar s [ 8 ] ;
UniCharCount len ;
UInt32 dead_key _state ;
/ * Make sure this scancode is a valid character scancode * /
scancode = darwin_scancode _table [ i ] ;
if ( scancode = = SDL_SCANCODE _UNKNOWN ||
( keymap [ scancode ] & SDLK_SCANCODE _MASK ) ) {
continue ;
}
dead_key _state = 0 ;
err = UCKeyTranslate ( ( UCKeyboardLayout * ) chr_data ,
i , kUCKeyActionDown ,
0 , keyboard_type ,
kUCKeyTranslateNoDeadKeysMask ,
& dead_key _state , 8 , & len , s ) ;
if ( err ! = noErr ) {
continue ;
}
if ( len > 0 && s [ 0 ] ! = 0 x10 ) {
keymap [ scancode ] = s [ 0 ] ;
}
}
SDL_SetKeymap ( 0 , keymap , SDL_NUM _SCANCODES ) ;
2015-10-27 19:17:32 +01:00
if ( send_event ) {
SDL_SendKeymapChangedEvent ( ) ;
}
2015-06-21 17:33:46 +02:00
return ;
}
cleanup :
CFRelease ( key_layout ) ;
}
void
Cocoa_InitKeyboard ( _THIS )
{
SDL_VideoData * data = ( SDL_VideoData * ) _this -> driverdata ;
2015-10-27 19:17:32 +01:00
UpdateKeymap ( data , SDL_FALSE ) ;
2015-06-21 17:33:46 +02:00
/ * Set our own names for the platform - dependent but layout - independent keys * /
/ * This key is NumLock on the MacBook keyboard . : ) * /
/ * SDL_SetScancodeName ( SDL_SCANCODE _NUMLOCKCLEAR , "Clear" ) ; * /
SDL_SetScancodeName ( SDL_SCANCODE _LALT , "Left Option" ) ;
SDL_SetScancodeName ( SDL_SCANCODE _LGUI , "Left Command" ) ;
SDL_SetScancodeName ( SDL_SCANCODE _RALT , "Right Option" ) ;
SDL_SetScancodeName ( SDL_SCANCODE _RGUI , "Right Command" ) ;
}
void
Cocoa_StartTextInput ( _THIS )
{ @ autoreleasepool
{
SDL_VideoData * data = ( SDL_VideoData * ) _this -> driverdata ;
SDL_Window * window = SDL_GetKeyboardFocus ( ) ;
NSWindow * nswindow = nil ;
if ( window ) {
nswindow = ( ( SDL_WindowData * ) window -> driverdata ) -> nswindow ;
}
NSView * parentView = [ nswindow contentView ] ;
/ * We only keep one field editor per process , since only the front most
* window can receive text input events , so it make no sense to keep more
* than one copy . When we switched to another window and requesting for
* text input , simply remove the field editor from its superview then add
* it to the front most window ' s content view * /
if ( ! data -> fieldEdit ) {
data -> fieldEdit =
[ [ SDLTranslatorResponder alloc ] initWithFrame : NSMakeRect ( 0.0 , 0.0 , 0.0 , 0.0 ) ] ;
}
2015-12-11 03:17:22 +01:00
if ( ! [ [ data -> fieldEdit superview ] isEqual : parentView ] ) {
2015-06-21 17:33:46 +02:00
/ * DEBUG_IME ( @ "add fieldEdit to window contentView" ) ; * /
[ data -> fieldEdit removeFromSuperview ] ;
[ parentView addSubview : data -> fieldEdit ] ;
[ nswindow makeFirstResponder : data -> fieldEdit ] ;
}
} }
void
Cocoa_StopTextInput ( _THIS )
{ @ autoreleasepool
{
SDL_VideoData * data = ( SDL_VideoData * ) _this -> driverdata ;
if ( data && data -> fieldEdit ) {
[ data -> fieldEdit removeFromSuperview ] ;
[ data -> fieldEdit release ] ;
data -> fieldEdit = nil ;
}
} }
void
Cocoa_SetTextInputRect ( _THIS , SDL_Rect * rect )
{
SDL_VideoData * data = ( SDL_VideoData * ) _this -> driverdata ;
if ( ! rect ) {
SDL_InvalidParamError ( "rect" ) ;
return ;
}
2015-12-11 03:17:22 +01:00
[ data -> fieldEdit setInputRect : rect ] ;
2015-06-21 17:33:46 +02:00
}
void
Cocoa_HandleKeyEvent ( _THIS , NSEvent * event )
{
SDL_VideoData * data = ( SDL_VideoData * ) _this -> driverdata ;
if ( ! data ) {
return ; / * can happen when returning from fullscreen Space on shutdown * /
}
unsigned short scancode = [ event keyCode ] ;
SDL_Scancode code ;
# if 0
const char * text ;
# endif
if ( ( scancode = = 10 || scancode = = 50 ) && KBGetLayoutType ( LMGetKbdType ( ) ) = = kKeyboardISO ) {
/ * see comments in SDL_cocoakeys . h * /
scancode = 60 - scancode ;
}
if ( scancode < SDL_arraysize ( darwin_scancode _table ) ) {
code = darwin_scancode _table [ scancode ] ;
} else {
/ * Hmm , does this ever happen ? If so , need to extend the keymap . . . * /
code = SDL_SCANCODE _UNKNOWN ;
}
switch ( [ event type ] ) {
case NSKeyDown :
if ( ! [ event isARepeat ] ) {
/ * See if we need to rebuild the keyboard layout * /
2015-10-27 19:17:32 +01:00
UpdateKeymap ( data , SDL_TRUE ) ;
2015-06-21 17:33:46 +02:00
}
SDL_SendKeyboardKey ( SDL_PRESSED , code ) ;
# if 1
if ( code = = SDL_SCANCODE _UNKNOWN ) {
fprintf ( stderr , "The key you just pressed is not recognized by SDL. To help get this fixed, report this to the SDL mailing list <sdl@libsdl.org> or to Christian Walther <cwalther@gmx.ch>. Mac virtual key code is %d.\n" , scancode ) ;
}
# endif
if ( SDL_EventState ( SDL_TEXTINPUT , SDL_QUERY ) ) {
/ * FIXME CW 2007 -08 -16 : only send those events to the field editor for which we actually want text events , not e . g . esc or function keys . Arrow keys in particular seem to produce crashes sometimes . * /
[ data -> fieldEdit interpretKeyEvents : [ NSArray arrayWithObject : event ] ] ;
# if 0
text = [ [ event characters ] UTF8String ] ;
if ( text && * text ) {
SDL_SendKeyboardText ( text ) ;
[ data -> fieldEdit setString : @ "" ] ;
}
# endif
}
break ;
case NSKeyUp :
SDL_SendKeyboardKey ( SDL_RELEASED , code ) ;
break ;
case NSFlagsChanged :
/ * FIXME CW 2007 -08 -14 : check if this whole mess that takes up half of this file is really necessary * /
HandleModifiers ( _this , scancode , [ event modifierFlags ] ) ;
break ;
default : / * just to avoid compiler warnings * /
break ;
}
}
void
Cocoa_QuitKeyboard ( _THIS )
{
}
# endif / * SDL_VIDEO _DRIVER _COCOA * /
/ * vi : set ts = 4 sw = 4 expandtab : * /