macosx_event.mm 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571
  1. /*
  2. ===========================================================================
  3. Doom 3 GPL Source Code
  4. Copyright (C) 1999-2011 id Software LLC, a ZeniMax Media company.
  5. This file is part of the Doom 3 GPL Source Code (?Doom 3 Source Code?).
  6. Doom 3 Source Code is free software: you can redistribute it and/or modify
  7. it under the terms of the GNU General Public License as published by
  8. the Free Software Foundation, either version 3 of the License, or
  9. (at your option) any later version.
  10. Doom 3 Source Code is distributed in the hope that it will be useful,
  11. but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. GNU General Public License for more details.
  14. You should have received a copy of the GNU General Public License
  15. along with Doom 3 Source Code. If not, see <http://www.gnu.org/licenses/>.
  16. In addition, the Doom 3 Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 Source Code. If not, please request a copy in writing from id Software at the address below.
  17. If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
  18. ===========================================================================
  19. */
  20. #ifndef USE_SDL
  21. // -*- mode: objc -*-
  22. #import "../../idlib/precompiled.h"
  23. #ifdef USE_SDL
  24. #include <SDL.h>
  25. #include "../../renderer/tr_local.h"
  26. #endif
  27. #import "macosx_local.h"
  28. #import "macosx_sys.h"
  29. #import <AppKit/NSCursor.h>
  30. #import <AppKit/NSWindow.h>
  31. #import <AppKit/NSScreen.h>
  32. #import <AppKit/NSApplication.h>
  33. #import <AppKit/NSGraphicsContext.h>
  34. #import <AppKit/NSEvent.h>
  35. #import <Foundation/NSArray.h>
  36. #import <Foundation/NSString.h>
  37. #import <Foundation/NSRunLoop.h>
  38. #import <Carbon/Carbon.h>
  39. #import <ApplicationServices/ApplicationServices.h>
  40. #import <sys/types.h>
  41. #import <sys/time.h>
  42. #import <unistd.h>
  43. #include <pthread.h>
  44. static NSDate *distantPast = NULL;
  45. static bool inputActive = false;
  46. static bool mouseActive = false;
  47. static bool inputRectValid = NO;
  48. static CGRect inputRect;
  49. static const void *sKLuchrData = NULL;
  50. static const void *sKLKCHRData = NULL;
  51. int vkeyToDoom3Key[256] = {
  52. /*0x00*/ 'a', 's', 'd', 'f', 'h', 'g', 'z', 'x',
  53. /*0x08*/ 'c', 'v', '?', 'b', 'q', 'w', 'e', 'r',
  54. /*0x10*/ 'y', 't', '1', '2', '3', '4', '6', '5',
  55. /*0x18*/ '=', '9', '7', '-', '8', '0', ']', 'o',
  56. /*0x20*/ 'u', '[', 'i', 'p', K_ENTER, 'l', 'j', '\'',
  57. /*0x28*/ 'k', ';', '\\', ',', '/', 'n', 'm', '.',
  58. /*0x30*/ K_TAB, K_SPACE, '`', K_BACKSPACE, '?', K_ESCAPE, '?', K_COMMAND,
  59. /*0x38*/ K_SHIFT, K_CAPSLOCK, K_ALT, K_CTRL, '?', '?', '?', '?',
  60. /*0x40*/ '?', K_KP_DEL, '?', K_KP_STAR, '?', K_KP_PLUS, '?', K_KP_NUMLOCK,
  61. /*0x48*/ '?', '?', '?', K_KP_SLASH, K_KP_ENTER, '?', K_KP_MINUS, '?',
  62. /*0x50*/ '?', K_KP_EQUALS, K_KP_INS, K_KP_END, K_KP_DOWNARROW, K_KP_PGDN, K_KP_LEFTARROW, K_KP_5,
  63. /*0x58*/ K_KP_RIGHTARROW, K_KP_HOME, '?', K_KP_UPARROW, K_KP_PGUP, '?', '?', '?',
  64. /*0x60*/ K_F5, K_F6, K_F7, K_F3, K_F8, K_F9, '?', K_F11,
  65. /*0x68*/ '?', K_PRINT_SCR, '?', K_F14, '?', K_F10, '?', K_F12,
  66. /*0x70*/ '?', K_F15, K_INS, K_HOME, K_PGUP, K_DEL, K_F4, K_END,
  67. /*0x78*/ K_F2, K_PGDN, K_F1, K_LEFTARROW, K_RIGHTARROW, K_DOWNARROW, K_UPARROW, K_POWER
  68. };
  69. int vkeyToDoom3Key_French[256] = {
  70. /*0x00*/ 'q', 's', 'd', 'f', 'h', 'g', 'w', 'x',
  71. /*0x08*/ 'c', 'v', '?', 'b', 'a', 'z', 'e', 'r',
  72. /*0x10*/ 'y', 't', '1', '2', '3', '4', '6', '5',
  73. /*0x18*/ '-', '9', '7', ')', '8', '0', '$', 'o',
  74. /*0x20*/ 'u', '^', 'i', 'p', K_ENTER, 'l', 'j', 'ù',
  75. /*0x28*/ 'k', 'm', 0x60, ';', '=', 'n', ',', ':',
  76. /*0x30*/ K_TAB, K_SPACE, '<', K_BACKSPACE, '?', K_ESCAPE, '?', K_COMMAND,
  77. /*0x38*/ K_SHIFT, K_CAPSLOCK, K_ALT, K_CTRL, '?', '?', '?', '?',
  78. /*0x40*/ '?', K_KP_DEL, '?', K_KP_STAR, '?', K_KP_PLUS, '?', K_KP_NUMLOCK,
  79. /*0x48*/ '?', '?', '?', K_KP_SLASH, K_KP_ENTER, '?', K_KP_MINUS, '?',
  80. /*0x50*/ '?', K_KP_EQUALS, K_KP_INS, K_KP_END, K_KP_DOWNARROW, K_KP_PGDN, K_KP_LEFTARROW, K_KP_5,
  81. /*0x58*/ K_KP_RIGHTARROW, K_KP_HOME, '?', K_KP_UPARROW, K_KP_PGUP, '?', '?', '?',
  82. /*0x60*/ K_F5, K_F6, K_F7, K_F3, K_F8, K_F9, '?', K_F11,
  83. /*0x68*/ '?', K_PRINT_SCR, '?', K_F14, '?', K_F10, '?', K_F12,
  84. /*0x70*/ '?', K_F15, K_INS, K_HOME, K_PGUP, K_DEL, K_F4, K_END,
  85. /*0x78*/ K_F2, K_PGDN, K_F1, K_LEFTARROW, K_RIGHTARROW, K_DOWNARROW, K_UPARROW, K_POWER
  86. };
  87. int vkeyToDoom3Key_German[256] = {
  88. /*0x00*/ 'a', 's', 'd', 'f', 'h', 'g', 'y', 'x',
  89. /*0x08*/ 'c', 'v', '?', 'b', 'q', 'w', 'e', 'r',
  90. /*0x10*/ 'z', 't', '1', '2', '3', '4', '6', '5',
  91. /*0x18*/ '«', '9', '7', '-', '8', '0', '+', 'o',
  92. /*0x20*/ 'u', '[', 'i', 'p', K_ENTER, 'l', 'j', '\'',
  93. /*0x28*/ 'k', ';', '#', ',', '-', 'n', 'm', '.',
  94. /*0x30*/ K_TAB, K_SPACE, '`', K_BACKSPACE, '?', K_ESCAPE, '?', K_COMMAND,
  95. /*0x38*/ K_SHIFT, K_CAPSLOCK, K_ALT, K_CTRL, '?', '?', '?', '?',
  96. /*0x40*/ '?', K_KP_DEL, '?', K_KP_STAR, '?', K_KP_PLUS, '?', K_KP_NUMLOCK,
  97. /*0x48*/ '?', '?', '?', K_KP_SLASH, K_KP_ENTER, '?', K_KP_MINUS, '?',
  98. /*0x50*/ '?', K_KP_EQUALS, K_KP_INS, K_KP_END, K_KP_DOWNARROW, K_KP_PGDN, K_KP_LEFTARROW, K_KP_5,
  99. /*0x58*/ K_KP_RIGHTARROW, K_KP_HOME, '?', K_KP_UPARROW, K_KP_PGUP, '?', '?', '?',
  100. /*0x60*/ K_F5, K_F6, K_F7, K_F3, K_F8, K_F9, '?', K_F11,
  101. /*0x68*/ '?', K_PRINT_SCR, '?', K_F14, '?', K_F10, '?', K_F12,
  102. /*0x70*/ '?', K_F15, K_INS, K_HOME, K_PGUP, K_DEL, K_F4, K_END,
  103. /*0x78*/ K_F2, K_PGDN, K_F1, K_LEFTARROW, K_RIGHTARROW, K_DOWNARROW, K_UPARROW, K_POWER
  104. };
  105. static const int *vkeyTable = vkeyToDoom3Key;
  106. /*
  107. ===========
  108. Sys_InitScanTable
  109. ===========
  110. */
  111. void Sys_InitScanTable( void ) {
  112. KeyboardLayoutRef kbLayout;
  113. idStr lang = cvarSystem->GetCVarString( "sys_lang" );
  114. if ( lang.Length() == 0 ) {
  115. lang = "english";
  116. }
  117. if ( lang.Icmp( "english" ) == 0 ) {
  118. vkeyTable = vkeyToDoom3Key;
  119. } else if ( lang.Icmp( "french" ) == 0 ) {
  120. vkeyTable = vkeyToDoom3Key_French;
  121. } else if ( lang.Icmp( "german" ) == 0 ) {
  122. vkeyTable = vkeyToDoom3Key_German;
  123. }
  124. if ( KLGetCurrentKeyboardLayout( &kbLayout ) == 0 ) {
  125. if ( KLGetKeyboardLayoutProperty( kbLayout, kKLuchrData, &sKLuchrData ) ) {
  126. common->Warning("KLGetKeyboardLayoutProperty failed");
  127. }
  128. if ( !sKLuchrData ) {
  129. if ( KLGetKeyboardLayoutProperty( kbLayout, kKLKCHRData, &sKLKCHRData ) ) {
  130. common->Warning("KLGetKeyboardLayoutProperty failed");
  131. }
  132. }
  133. }
  134. if ( !sKLuchrData && !sKLKCHRData ) {
  135. common->Warning("Keyboard input initialziation failed");
  136. }
  137. }
  138. void Sys_InitInput( void ) {
  139. common->Printf( "------- Input Initialization -------\n" );
  140. if ( !distantPast ) {
  141. distantPast = [ [ NSDate distantPast ] retain ];
  142. }
  143. IN_ActivateMouse();
  144. inputActive = true;
  145. }
  146. void Sys_ShutdownInput( void ) {
  147. common->Printf( "------- Input Shutdown -------\n" );
  148. if ( !inputActive ) {
  149. return;
  150. }
  151. inputActive = false;
  152. if ( mouseActive ) {
  153. IN_DeactivateMouse();
  154. }
  155. common->Printf( "------------------------------\n" );
  156. }
  157. void processMouseMovedEvent( NSEvent *mouseMovedEvent ) {
  158. CGMouseDelta dx, dy;
  159. if ( !mouseActive ) {
  160. return;
  161. }
  162. #if 0
  163. #define ACT_LIKE_WINDOWS
  164. #ifdef ACT_LIKE_WINDOWS
  165. cvar_t *in_mouseLowEndSlope = Cvar_Get("in_mouseLowEndSlope", "3.5", CVAR_ARCHIVE);
  166. if (in_mouseLowEndSlope->value < 1) {
  167. Cvar_Set("in_mouseLowEndSlope", "1");
  168. }
  169. #else
  170. cvar_t *in_mouseLowEndSlope = Cvar_Get("in_mouseLowEndSlope", "1", CVAR_ARCHIVE);
  171. if (in_mouseLowEndSlope->value < 1) {
  172. Cvar_Set("in_mouseLowEndSlope", "1");
  173. }
  174. #endif
  175. cvar_t *in_mouseHighEndCutoff = Cvar_Get("in_mouseHighEndCutoff", "20", CVAR_ARCHIVE);
  176. if (in_mouseLowEndSlope->value < 1) {
  177. Cvar_Set("in_mouseHighEndCutoff", "1");
  178. }
  179. #endif
  180. CGGetLastMouseDelta(&dx, &dy);
  181. if ( dx || dy ) {
  182. #if 0 // this is be handled by the mouse driver clean me out later
  183. CGMouseDelta distSqr;
  184. float m0, N;
  185. distSqr = dx * dx + dy * dy;
  186. //Com_Printf("distSqr = %d\n", distSqr);
  187. /* This code is here to help people that like the feel of the Logitech USB Gaming Mouse with the Win98 drivers. By empirical testing, the Windows drivers seem to be more heavily accelerated at the low end of the curve. */
  188. //N = in_mouseHighEndCutoff->value;
  189. N = 1;
  190. if (distSqr < N*N) {
  191. float dist, accel, scale;
  192. //m0 = in_mouseLowEndSlope->value;
  193. m0 = 1;
  194. dist = sqrt(distSqr);
  195. accel = (((m0 - 1.0)/(N*N) * dist + (2.0 - 2.0*m0)/N) * dist + m0) * dist;
  196. scale = accel / dist;
  197. //Com_Printf("dx = %d, dy = %d, dist = %f, accel = %f, scale = %f\n", dx, dy, dist, accel, scale);
  198. dx *= scale;
  199. dy *= scale;
  200. }
  201. #endif
  202. Posix_QueEvent( SE_MOUSE, dx, dy, 0, NULL );
  203. Posix_AddMousePollEvent( M_DELTAX, dx );
  204. Posix_AddMousePollEvent( M_DELTAY, dy );
  205. }
  206. }
  207. inline bool OSX_LookupCharacter(unsigned short vkey, unsigned int modifiers, bool keyDownFlag, unsigned char *outChar)
  208. {
  209. UInt32 translated;
  210. UInt32 deadKeyState = 0;
  211. UniChar unicodeString[16];
  212. UniCharCount actualStringLength = 0;
  213. static UInt32 keyTranslateState = 0;
  214. // Only want character if Translate() returns a single character
  215. if ( sKLuchrData ) {
  216. UCKeyTranslate( (UCKeyboardLayout*)sKLuchrData, vkey, keyDownFlag ? kUCKeyActionDown : kUCKeyActionUp, modifiers,
  217. LMGetKbdType(), 0, &deadKeyState, 16, &actualStringLength, unicodeString );
  218. if ( actualStringLength == 1 ) {
  219. *outChar = (unsigned char)unicodeString[0];
  220. return true;
  221. }
  222. }
  223. else if ( sKLKCHRData ) {
  224. translated = KeyTranslate( sKLKCHRData, vkey, &keyTranslateState );
  225. if ( ( translated & 0x00ff0000 ) == 0 ) {
  226. *outChar = translated & 0xff;
  227. return true;
  228. }
  229. }
  230. return false;
  231. }
  232. void OSX_ProcessKeyEvent( NSEvent *keyEvent, bool keyDownFlag ) {
  233. unsigned char character;
  234. unsigned int modifiers = 0;
  235. unsigned short vkey = [ keyEvent keyCode ];
  236. if ( [ keyEvent modifierFlags ] & NSAlphaShiftKeyMask )
  237. modifiers |= alphaLock;
  238. if ( [ keyEvent modifierFlags ] & NSShiftKeyMask )
  239. modifiers |= shiftKey;
  240. if ( [ keyEvent modifierFlags ] & NSControlKeyMask )
  241. modifiers |= controlKey;
  242. if ( [ keyEvent modifierFlags ] & NSAlternateKeyMask )
  243. modifiers |= optionKey;
  244. if ( [ keyEvent modifierFlags ] & NSCommandKeyMask )
  245. modifiers |= cmdKey;
  246. modifiers >>= 8;
  247. int doomKey = (unsigned char)vkeyTable[vkey];
  248. Posix_QueEvent( SE_KEY, doomKey, keyDownFlag, 0, NULL );
  249. if ( keyDownFlag ) {
  250. if ( OSX_LookupCharacter(vkey, modifiers, keyDownFlag, &character ) &&
  251. character != Sys_GetConsoleKey( false ) && character != Sys_GetConsoleKey( true ) ) {
  252. Posix_QueEvent( SE_CHAR, character, 0, 0, NULL);
  253. }
  254. }
  255. Posix_AddKeyboardPollEvent( doomKey, keyDownFlag );
  256. return;
  257. }
  258. void sendEventForMaskChangeInFlags( int quakeKey, unsigned int modifierMask, unsigned int oldModifierFlags, unsigned int newModifierFlags ) {
  259. bool oldHadModifier, newHasModifier;
  260. oldHadModifier = (oldModifierFlags & modifierMask) != 0;
  261. newHasModifier = (newModifierFlags & modifierMask) != 0;
  262. if (oldHadModifier != newHasModifier) {
  263. //NSLog(@"Key %d posted for modifier mask modifierMask", quakeKey);
  264. Posix_QueEvent( SE_KEY, quakeKey, newHasModifier, 0, NULL);
  265. Posix_AddKeyboardPollEvent( quakeKey, newHasModifier );
  266. }
  267. }
  268. void processFlagsChangedEvent( NSEvent *flagsChangedEvent ) {
  269. static int oldModifierFlags;
  270. int newModifierFlags;
  271. newModifierFlags = [flagsChangedEvent modifierFlags];
  272. sendEventForMaskChangeInFlags( K_ALT, NSAlternateKeyMask, oldModifierFlags, newModifierFlags );
  273. sendEventForMaskChangeInFlags( K_CTRL, NSControlKeyMask, oldModifierFlags, newModifierFlags );
  274. sendEventForMaskChangeInFlags( K_SHIFT, NSShiftKeyMask, oldModifierFlags, newModifierFlags );
  275. oldModifierFlags = newModifierFlags;
  276. }
  277. void processSystemDefinedEvent( NSEvent *systemDefinedEvent ) {
  278. static int oldButtons = 0;
  279. int buttonsDelta;
  280. int buttons;
  281. int isDown;
  282. if ( [systemDefinedEvent subtype] == 7 ) {
  283. if ( !mouseActive ) {
  284. return;
  285. }
  286. buttons = [systemDefinedEvent data2];
  287. buttonsDelta = oldButtons ^ buttons;
  288. //common->Printf( "uberbuttons: %08lx %08lx\n", buttonsDelta, buttons );
  289. if (buttonsDelta & 1) {
  290. isDown = buttons & 1;
  291. Posix_QueEvent( SE_KEY, K_MOUSE1, isDown, 0, NULL);
  292. Posix_AddMousePollEvent( M_ACTION1, isDown );
  293. }
  294. if (buttonsDelta & 2) {
  295. isDown = buttons & 2;
  296. Posix_QueEvent( SE_KEY, K_MOUSE2, isDown, 0, NULL);
  297. Posix_AddMousePollEvent( M_ACTION2, isDown );
  298. }
  299. if (buttonsDelta & 4) {
  300. isDown = buttons & 4;
  301. Posix_QueEvent( SE_KEY, K_MOUSE3, isDown, 0, NULL);
  302. Posix_AddMousePollEvent( M_ACTION3, isDown );
  303. }
  304. if (buttonsDelta & 8) {
  305. isDown = buttons & 8;
  306. Posix_QueEvent( SE_KEY, K_MOUSE4, isDown, 0, NULL);
  307. Posix_AddMousePollEvent( M_ACTION4, isDown );
  308. }
  309. if (buttonsDelta & 16) {
  310. isDown = buttons & 16;
  311. Posix_QueEvent( SE_KEY, K_MOUSE5, isDown, 0, NULL);
  312. Posix_AddMousePollEvent( M_ACTION5, isDown );
  313. }
  314. oldButtons = buttons;
  315. }
  316. }
  317. void processEvent( NSEvent *event ) {
  318. NSEventType eventType;
  319. if ( !inputActive ) {
  320. return;
  321. }
  322. eventType = [ event type ];
  323. switch ( eventType ) {
  324. // These four event types are ignored since we do all of our mouse down/up process via the uber-mouse system defined event. We have to accept these events however since they get enqueued and the queue will fill up if we don't.
  325. case NSLeftMouseDown:
  326. case NSLeftMouseUp:
  327. case NSRightMouseDown:
  328. case NSRightMouseUp:
  329. //NSLog( @"ignore simple mouse event %@", event );
  330. return;
  331. case NSMouseMoved:
  332. case NSLeftMouseDragged:
  333. case NSRightMouseDragged:
  334. processMouseMovedEvent( event );
  335. return;
  336. case NSKeyDown:
  337. // Send ALL command key-ups to Quake, but not command key-downs, otherwise if the user hits a key, presses command, and lets up on the key, the key-up won't register.
  338. if ( [ event modifierFlags ] & NSCommandKeyMask ) {
  339. NSLog( @"command key up ignored: %@", event );
  340. break;
  341. }
  342. case NSKeyUp:
  343. OSX_ProcessKeyEvent( event, eventType == NSKeyDown );
  344. return;
  345. case NSFlagsChanged:
  346. processFlagsChangedEvent( event );
  347. return;
  348. case NSSystemDefined:
  349. processSystemDefinedEvent( event );
  350. return;
  351. case NSScrollWheel:
  352. if ([event deltaY] < 0.0) {
  353. Posix_QueEvent( SE_KEY, K_MWHEELDOWN, true, 0, NULL );
  354. Posix_QueEvent( SE_KEY, K_MWHEELDOWN, false, 0, NULL );
  355. Posix_AddMousePollEvent( M_DELTAZ, -1 );
  356. } else {
  357. Posix_QueEvent( SE_KEY, K_MWHEELUP, true, 0, NULL );
  358. Posix_QueEvent( SE_KEY, K_MWHEELUP, false, 0, NULL );
  359. Posix_AddMousePollEvent( M_DELTAZ, 1 );
  360. }
  361. return;
  362. default:
  363. //NSLog( @"handle event %@", event );
  364. break;
  365. }
  366. [NSApp sendEvent:event];
  367. }
  368. void Posix_PollInput( void ) {
  369. NSEvent *event;
  370. unsigned int eventMask;
  371. eventMask = NSAnyEventMask;
  372. while ( ( event = [ NSApp nextEventMatchingMask: eventMask
  373. untilDate: distantPast
  374. inMode: NSDefaultRunLoopMode
  375. dequeue:YES ] ) ) {
  376. processEvent( event );
  377. }
  378. }
  379. void Sys_PreventMouseMovement( CGPoint point ) {
  380. CGEventErr err;
  381. //common->Printf( "**** Calling CGAssociateMouseAndMouseCursorPosition(false)\n" );
  382. err = CGAssociateMouseAndMouseCursorPosition( false );
  383. if ( err != CGEventNoErr ) {
  384. common->Error( "Could not disable mouse movement, CGAssociateMouseAndMouseCursorPosition returned %d\n", err );
  385. }
  386. // Put the mouse in the position we want to leave it at
  387. err = CGWarpMouseCursorPosition( point );
  388. if ( err != CGEventNoErr ) {
  389. common->Error( "Could not disable mouse movement, CGWarpMouseCursorPosition returned %d\n", err );
  390. }
  391. }
  392. void Sys_ReenableMouseMovement() {
  393. CGEventErr err;
  394. //common->Printf( "**** Calling CGAssociateMouseAndMouseCursorPosition(true)\n" );
  395. err = CGAssociateMouseAndMouseCursorPosition( true );
  396. if ( err != CGEventNoErr ) {
  397. common->Error( "Could not reenable mouse movement, CGAssociateMouseAndMouseCursorPosition returned %d\n", err );
  398. }
  399. // Leave the mouse where it was -- don't warp here.
  400. }
  401. void Sys_LockMouseInInputRect(CGRect rect) {
  402. CGPoint center;
  403. center.x = rect.origin.x + rect.size.width / 2.0;
  404. center.y = rect.origin.y + rect.size.height / 2.0;
  405. // Now, put the mouse in the middle of the input rect (anywhere over it would do)
  406. // and don't allow it to move. This means that the user won't be able to accidentally
  407. // select another application.
  408. Sys_PreventMouseMovement(center);
  409. }
  410. void Sys_SetMouseInputRect(CGRect newRect) {
  411. inputRectValid = YES;
  412. inputRect = newRect;
  413. if ( mouseActive ) {
  414. Sys_LockMouseInInputRect( inputRect );
  415. }
  416. }
  417. void IN_ActivateMouse( void ) {
  418. if ( mouseActive ) {
  419. return;
  420. }
  421. #ifdef USE_SDL
  422. // unset modifier, in case alt-tab was used to leave window and ALT is still set
  423. // as that can cause fullscreen-toggling when pressing enter...
  424. SDL_Keymod currentmod = SDL_GetModState();
  425. int newmod = KMOD_NONE;
  426. if (currentmod & KMOD_CAPS) // preserve capslock
  427. newmod |= KMOD_CAPS;
  428. SDL_SetModState((SDL_Keymod)newmod);
  429. GLimp_GrabInput(GRAB_ENABLE | GRAB_REENABLE | GRAB_HIDECURSOR);
  430. #else
  431. if ( inputRectValid ) {
  432. // Make sure that if window moved we don't hose the user...
  433. Sys_UpdateWindowMouseInputRect();
  434. }
  435. Sys_LockMouseInInputRect( inputRect );
  436. CGDisplayHideCursor( Sys_DisplayToUse() );
  437. #endif
  438. mouseActive = true;
  439. }
  440. void IN_DeactivateMouse( void ) {
  441. if ( !mouseActive ) {
  442. return;
  443. }
  444. #ifdef USE_SDL
  445. GLimp_GrabInput(0);
  446. #else
  447. Sys_ReenableMouseMovement();
  448. CGDisplayShowCursor( Sys_DisplayToUse() );
  449. #endif
  450. mouseActive = false;
  451. }
  452. /*
  453. ===============
  454. Sys_MapCharForKey
  455. ===============
  456. */
  457. unsigned char Sys_MapCharForKey( int key ) {
  458. return (unsigned char)key;
  459. }
  460. /*
  461. ===============
  462. Sys_GetConsoleKey
  463. ===============
  464. */
  465. unsigned char Sys_GetConsoleKey( bool shifted ) {
  466. if ( vkeyTable == vkeyToDoom3Key_French ) {
  467. return shifted ? '>' : '<';
  468. }
  469. else {
  470. return shifted ? '~' : '`';
  471. }
  472. }
  473. #endif