mirror of
https://github.com/ptitSeb/Serious-Engine
synced 2025-01-04 19:50:22 +01:00
1012 lines
30 KiB
C++
Executable File
1012 lines
30 KiB
C++
Executable File
/* Copyright (c) 2002-2012 Croteam Ltd. All rights reserved. */
|
|
|
|
/* rcg10072001 Moved stuff into this file. */
|
|
|
|
#include "SDL.h"
|
|
|
|
#include <Engine/Base/Timer.h>
|
|
#include <Engine/Base/Input.h>
|
|
#include <Engine/Base/Translation.h>
|
|
#include <Engine/Base/KeyNames.h>
|
|
#include <Engine/Math/Functions.h>
|
|
#include <Engine/Graphics/ViewPort.h>
|
|
#include <Engine/Base/Console.h>
|
|
#include <Engine/Base/Synchronization.h>
|
|
#include <Engine/Base/SDL/SDLEvents.h>
|
|
#include <Engine/Base/Shell.h>
|
|
#include <Engine/Base/ErrorReporting.h>
|
|
|
|
extern INDEX inp_iKeyboardReadingMethod;
|
|
extern FLOAT inp_fMouseSensitivity;
|
|
extern INDEX inp_bAllowMouseAcceleration;
|
|
extern INDEX inp_bMousePrecision;
|
|
extern FLOAT inp_fMousePrecisionFactor;
|
|
extern FLOAT inp_fMousePrecisionThreshold;
|
|
extern FLOAT inp_fMousePrecisionTimeout;
|
|
extern FLOAT inp_bInvertMouse;
|
|
extern INDEX inp_bFilterMouse;
|
|
extern INDEX inp_bAllowPrescan;
|
|
|
|
extern INDEX inp_i2ndMousePort;
|
|
extern FLOAT inp_f2ndMouseSensitivity;
|
|
extern INDEX inp_b2ndMousePrecision;
|
|
extern FLOAT inp_f2ndMousePrecisionThreshold;
|
|
extern FLOAT inp_f2ndMousePrecisionTimeout;
|
|
extern FLOAT inp_f2ndMousePrecisionFactor;
|
|
extern INDEX inp_bFilter2ndMouse;
|
|
extern INDEX inp_bInvert2ndMouse;
|
|
|
|
static BOOL inp_bSDLPermitCtrlG = TRUE;
|
|
static BOOL inp_bSDLGrabInput = TRUE;
|
|
static Sint16 mouse_relative_x = 0;
|
|
static Sint16 mouse_relative_y = 0;
|
|
|
|
INDEX inp_iMButton4Dn = 0x20040;
|
|
INDEX inp_iMButton4Up = 0x20000;
|
|
INDEX inp_iMButton5Dn = 0x10020;
|
|
INDEX inp_iMButton5Up = 0x10000;
|
|
INDEX inp_bMsgDebugger = FALSE;
|
|
INDEX inp_bForceJoystickPolling = 0;
|
|
INDEX inp_bAutoDisableJoysticks = 0;
|
|
|
|
static CTCriticalSection sl_csInput;
|
|
|
|
extern INDEX inp_ctJoysticksAllowed;
|
|
|
|
/*
|
|
|
|
NOTE: Two different types of key codes are used here:
|
|
1) kid - engine internal type - defined in KeyNames.h
|
|
2) virtkey - virtual key codes used by SDL.
|
|
|
|
*/
|
|
|
|
// name that is not translated (international)
|
|
#define INTNAME(str) str, ""
|
|
// name that is translated
|
|
#define TRANAME(str) str, "ETRS" str
|
|
|
|
// basic key conversion table
|
|
struct KeyConversion {
|
|
INDEX kc_iKID;
|
|
SDL_Keycode kc_iVirtKey;
|
|
const char *kc_strName;
|
|
const char *kc_strNameTrans;
|
|
};
|
|
|
|
static const KeyConversion _akcKeys[] = {
|
|
// reserved for 'no-key-pressed'
|
|
{ KID_NONE, -1, TRANAME("None")},
|
|
|
|
// numbers row
|
|
{ KID_1 , SDLK_1, INTNAME("1")},
|
|
{ KID_2 , SDLK_2, INTNAME("2")},
|
|
{ KID_3 , SDLK_3, INTNAME("3")},
|
|
{ KID_4 , SDLK_4, INTNAME("4")},
|
|
{ KID_5 , SDLK_5, INTNAME("5")},
|
|
{ KID_6 , SDLK_6, INTNAME("6")},
|
|
{ KID_7 , SDLK_7, INTNAME("7")},
|
|
{ KID_8 , SDLK_8, INTNAME("8")},
|
|
{ KID_9 , SDLK_9, INTNAME("9")},
|
|
{ KID_0 , SDLK_0, INTNAME("0")},
|
|
{ KID_MINUS , SDLK_MINUS, INTNAME("-")},
|
|
{ KID_EQUALS , SDLK_EQUALS, INTNAME("=")},
|
|
|
|
// 1st alpha row
|
|
{ KID_Q , SDLK_q, INTNAME("Q")},
|
|
{ KID_W , SDLK_w, INTNAME("W")},
|
|
{ KID_E , SDLK_e, INTNAME("E")},
|
|
{ KID_R , SDLK_r, INTNAME("R")},
|
|
{ KID_T , SDLK_t, INTNAME("T")},
|
|
{ KID_Y , SDLK_y, INTNAME("Y")},
|
|
{ KID_U , SDLK_u, INTNAME("U")},
|
|
{ KID_I , SDLK_i, INTNAME("I")},
|
|
{ KID_O , SDLK_o, INTNAME("O")},
|
|
{ KID_P , SDLK_p, INTNAME("P")},
|
|
{ KID_LBRACKET , SDLK_RIGHTPAREN, INTNAME("[")},
|
|
{ KID_RBRACKET , SDLK_RIGHTPAREN, INTNAME("]")},
|
|
{ KID_BACKSLASH , SDLK_BACKSLASH, INTNAME("\\")},
|
|
|
|
// 2nd alpha row
|
|
{ KID_A , SDLK_a, INTNAME("A")},
|
|
{ KID_S , SDLK_s, INTNAME("S")},
|
|
{ KID_D , SDLK_d, INTNAME("D")},
|
|
{ KID_F , SDLK_f, INTNAME("F")},
|
|
{ KID_G , SDLK_g, INTNAME("G")},
|
|
{ KID_H , SDLK_h, INTNAME("H")},
|
|
{ KID_J , SDLK_j, INTNAME("J")},
|
|
{ KID_K , SDLK_k, INTNAME("K")},
|
|
{ KID_L , SDLK_l, INTNAME("L")},
|
|
{ KID_SEMICOLON , SDLK_SEMICOLON, INTNAME(";")},
|
|
{ KID_APOSTROPHE , SDLK_QUOTE, INTNAME("'")},
|
|
|
|
// 3rd alpha row
|
|
{ KID_Z , SDLK_z, INTNAME("Z")},
|
|
{ KID_X , SDLK_x, INTNAME("X")},
|
|
{ KID_C , SDLK_c, INTNAME("C")},
|
|
{ KID_V , SDLK_v, INTNAME("V")},
|
|
{ KID_B , SDLK_b, INTNAME("B")},
|
|
{ KID_N , SDLK_n, INTNAME("N")},
|
|
{ KID_M , SDLK_m, INTNAME("M")},
|
|
{ KID_COMMA , SDLK_COMMA, INTNAME(",")},
|
|
{ KID_PERIOD , SDLK_PERIOD, INTNAME(".")},
|
|
{ KID_SLASH , SDLK_SLASH, INTNAME("/")},
|
|
|
|
// row with F-keys
|
|
{ KID_F1 , SDLK_F1, INTNAME("F1")},
|
|
{ KID_F2 , SDLK_F2, INTNAME("F2")},
|
|
{ KID_F3 , SDLK_F3, INTNAME("F3")},
|
|
{ KID_F4 , SDLK_F4, INTNAME("F4")},
|
|
{ KID_F5 , SDLK_F5, INTNAME("F5")},
|
|
{ KID_F6 , SDLK_F6, INTNAME("F6")},
|
|
{ KID_F7 , SDLK_F7, INTNAME("F7")},
|
|
{ KID_F8 , SDLK_F8, INTNAME("F8")},
|
|
{ KID_F9 , SDLK_F9, INTNAME("F9")},
|
|
{ KID_F10 , SDLK_F10, INTNAME("F10")},
|
|
{ KID_F11 , SDLK_F11, INTNAME("F11")},
|
|
{ KID_F12 , SDLK_F12, INTNAME("F12")},
|
|
|
|
// extra keys
|
|
{ KID_ESCAPE , SDLK_ESCAPE, TRANAME("Escape")},
|
|
{ KID_TILDE , -1, TRANAME("Tilde")},
|
|
{ KID_BACKSPACE , SDLK_BACKSPACE, TRANAME("Backspace")},
|
|
{ KID_TAB , SDLK_TAB, TRANAME("Tab")},
|
|
{ KID_CAPSLOCK , SDLK_CAPSLOCK, TRANAME("Caps Lock")},
|
|
{ KID_ENTER , SDLK_RETURN, TRANAME("Enter")},
|
|
{ KID_SPACE , SDLK_SPACE, TRANAME("Space")},
|
|
|
|
// modifier keys
|
|
{ KID_LSHIFT , SDLK_LSHIFT , TRANAME("Left Shift")},
|
|
{ KID_RSHIFT , SDLK_RSHIFT , TRANAME("Right Shift")},
|
|
{ KID_LCONTROL , SDLK_LCTRL, TRANAME("Left Control")},
|
|
{ KID_RCONTROL , SDLK_RCTRL, TRANAME("Right Control")},
|
|
{ KID_LALT , SDLK_LALT , TRANAME("Left Alt")},
|
|
{ KID_RALT , SDLK_RALT , TRANAME("Right Alt")},
|
|
|
|
// navigation keys
|
|
{ KID_ARROWUP , SDLK_UP, TRANAME("Arrow Up")},
|
|
{ KID_ARROWDOWN , SDLK_DOWN, TRANAME("Arrow Down")},
|
|
{ KID_ARROWLEFT , SDLK_LEFT, TRANAME("Arrow Left")},
|
|
{ KID_ARROWRIGHT , SDLK_RIGHT, TRANAME("Arrow Right")},
|
|
{ KID_INSERT , SDLK_INSERT, TRANAME("Insert")},
|
|
{ KID_DELETE , SDLK_DELETE, TRANAME("Delete")},
|
|
{ KID_HOME , SDLK_HOME, TRANAME("Home")},
|
|
{ KID_END , SDLK_END, TRANAME("End")},
|
|
{ KID_PAGEUP , SDLK_PAGEUP, TRANAME("Page Up")},
|
|
{ KID_PAGEDOWN , SDLK_PAGEDOWN, TRANAME("Page Down")},
|
|
{ KID_PRINTSCR , SDLK_PRINTSCREEN, TRANAME("Print Screen")},
|
|
{ KID_SCROLLLOCK , SDLK_SCROLLLOCK, TRANAME("Scroll Lock")},
|
|
{ KID_PAUSE , SDLK_PAUSE, TRANAME("Pause")},
|
|
|
|
// numpad numbers
|
|
{ KID_NUM0 , SDLK_KP_0, INTNAME("Num 0")},
|
|
{ KID_NUM1 , SDLK_KP_1, INTNAME("Num 1")},
|
|
{ KID_NUM2 , SDLK_KP_2, INTNAME("Num 2")},
|
|
{ KID_NUM3 , SDLK_KP_3, INTNAME("Num 3")},
|
|
{ KID_NUM4 , SDLK_KP_4, INTNAME("Num 4")},
|
|
{ KID_NUM5 , SDLK_KP_5, INTNAME("Num 5")},
|
|
{ KID_NUM6 , SDLK_KP_6, INTNAME("Num 6")},
|
|
{ KID_NUM7 , SDLK_KP_7, INTNAME("Num 7")},
|
|
{ KID_NUM8 , SDLK_KP_8, INTNAME("Num 8")},
|
|
{ KID_NUM9 , SDLK_KP_9, INTNAME("Num 9")},
|
|
{ KID_NUMDECIMAL , SDLK_KP_PERIOD, INTNAME("Num .")},
|
|
|
|
// numpad gray keys
|
|
{ KID_NUMLOCK , SDLK_NUMLOCKCLEAR, INTNAME("Num Lock")},
|
|
{ KID_NUMSLASH , SDLK_KP_DIVIDE, INTNAME("Num /")},
|
|
{ KID_NUMMULTIPLY , SDLK_KP_MULTIPLY, INTNAME("Num *")},
|
|
{ KID_NUMMINUS , SDLK_KP_MINUS, INTNAME("Num -")},
|
|
{ KID_NUMPLUS , SDLK_KP_PLUS, INTNAME("Num +")},
|
|
{ KID_NUMENTER , SDLK_KP_ENTER, TRANAME("Num Enter")},
|
|
|
|
// mouse buttons
|
|
{ KID_MOUSE1 , -1, TRANAME("Mouse Button 1")},
|
|
{ KID_MOUSE2 , -1, TRANAME("Mouse Button 2")},
|
|
{ KID_MOUSE3 , -1, TRANAME("Mouse Button 3")},
|
|
{ KID_MOUSE4 , -1, TRANAME("Mouse Button 4")},
|
|
{ KID_MOUSE5 , -1, TRANAME("Mouse Button 5")},
|
|
{ KID_MOUSEWHEELUP , -1, TRANAME("Mouse Wheel Up")},
|
|
{ KID_MOUSEWHEELDOWN , -1, TRANAME("Mouse Wheel Down")},
|
|
|
|
// 2nd mouse buttons
|
|
{ KID_2MOUSE1 , -1, TRANAME("2nd Mouse Button 1")},
|
|
{ KID_2MOUSE2 , -1, TRANAME("2nd Mouse Button 2")},
|
|
{ KID_2MOUSE3 , -1, TRANAME("2nd Mouse Button 3")},
|
|
|
|
};
|
|
|
|
|
|
// autogenerated fast conversion tables
|
|
static INDEX _aiScancodeToKid[SDL_NUM_SCANCODES];
|
|
|
|
// make fast conversion tables from the general table
|
|
static void MakeConversionTables(void)
|
|
{
|
|
// clear conversion tables
|
|
for (int i = 0; i < ARRAYCOUNT(_aiScancodeToKid); i++) {
|
|
_aiScancodeToKid[i] = -1;
|
|
}
|
|
|
|
// for each Key
|
|
for (INDEX iKey=0; iKey<ARRAYCOUNT(_akcKeys); iKey++) {
|
|
const KeyConversion &kc = _akcKeys[iKey];
|
|
|
|
// get codes
|
|
const INDEX iKID = kc.kc_iKID;
|
|
//INDEX iScan = kc.kc_iScanCode;
|
|
const SDL_Keycode iVirt = kc.kc_iVirtKey;
|
|
|
|
if (iVirt>=0) {
|
|
_aiScancodeToKid[SDL_GetScancodeFromKey(iVirt)] = iKID;
|
|
}
|
|
}
|
|
|
|
_aiScancodeToKid[SDL_SCANCODE_UNKNOWN] = -1; // in case several items set this.
|
|
}
|
|
|
|
// variables for message interception
|
|
//static HHOOK _hGetMsgHook = NULL;
|
|
//static HHOOK _hSendMsgHook = NULL;
|
|
static int _iMouseZ = 0;
|
|
static BOOL _bWheelUp = FALSE;
|
|
static BOOL _bWheelDn = FALSE;
|
|
|
|
// emulate Win32: A single mouse wheel rotation
|
|
// is +120 (upwards) or -120 (downwards)
|
|
#define MOUSE_SCROLL_INTERVAL 120
|
|
|
|
CTCriticalSection csInput;
|
|
|
|
// which keys are pressed, as recorded by message interception (by KIDs)
|
|
static UBYTE _abKeysPressed[256];
|
|
|
|
// set a key according to a keydown/keyup message
|
|
static void SetKeyFromEvent(const SDL_Event *event, const BOOL bDown)
|
|
{
|
|
ASSERT((event->type == SDL_KEYUP) || (event->type == SDL_KEYDOWN));
|
|
|
|
if ( (event->key.keysym.mod & KMOD_CTRL) &&
|
|
(event->key.keysym.sym == SDLK_g) &&
|
|
(event->type == SDL_KEYDOWN) &&
|
|
(inp_bSDLPermitCtrlG) )
|
|
{
|
|
if (inp_bSDLGrabInput)
|
|
{
|
|
// turn off input grab.
|
|
SDL_SetRelativeMouseMode(SDL_FALSE);
|
|
inp_bSDLGrabInput = FALSE;
|
|
} // if
|
|
else
|
|
{
|
|
// turn on input grab.
|
|
SDL_SetRelativeMouseMode(SDL_TRUE);
|
|
inp_bSDLGrabInput = TRUE;
|
|
} // else
|
|
|
|
mouse_relative_x = mouse_relative_y = 0;
|
|
return;
|
|
} // if
|
|
|
|
#ifdef PLATFORM_PANDORA
|
|
if(event->key.keysym.sym==SDLK_RSHIFT) {
|
|
_abKeysPressed[KID_MOUSE1] = bDown;
|
|
return;
|
|
} else if(event->key.keysym.sym==SDLK_RCTRL) {
|
|
_abKeysPressed[KID_MOUSE2] = bDown;
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
// convert virtualkey to kid
|
|
const INDEX iKID = _aiScancodeToKid[event->key.keysym.scancode];
|
|
|
|
if (iKID>=0 && iKID<ARRAYCOUNT(_abKeysPressed)) {
|
|
//CPrintF("%s: %d\n", _pInput->inp_strButtonNames[iKID], bDown);
|
|
_abKeysPressed[iKID] = bDown;
|
|
}
|
|
}
|
|
|
|
|
|
static void sdl_event_handler(const SDL_Event *event)
|
|
{
|
|
switch (event->type)
|
|
{
|
|
case SDL_MOUSEMOTION:
|
|
mouse_relative_x += event->motion.xrel;
|
|
mouse_relative_y += event->motion.yrel;
|
|
break;
|
|
|
|
case SDL_MOUSEBUTTONDOWN:
|
|
case SDL_MOUSEBUTTONUP:
|
|
if (event->button.button <= 5) {
|
|
int button = KID_MOUSE1;
|
|
switch(event->button.button) {
|
|
case SDL_BUTTON_RIGHT: button = KID_MOUSE2; break;
|
|
case SDL_BUTTON_MIDDLE: button = KID_MOUSE3; break;
|
|
case 4: button = KID_MOUSE4; break;
|
|
case 5: button = KID_MOUSE5; break;
|
|
}
|
|
_abKeysPressed[button] = (event->button.state == SDL_PRESSED) ? TRUE : FALSE;
|
|
}
|
|
break;
|
|
|
|
case SDL_MOUSEWHEEL:
|
|
_iMouseZ += event->wheel.y * MOUSE_SCROLL_INTERVAL;
|
|
break;
|
|
|
|
case SDL_KEYDOWN:
|
|
SetKeyFromEvent(event, TRUE);
|
|
break;
|
|
|
|
case SDL_KEYUP:
|
|
SetKeyFromEvent(event, FALSE);
|
|
break;
|
|
|
|
default: break;
|
|
} // switch
|
|
} // sdl_event_handler
|
|
|
|
|
|
// This keeps the input subsystem in sync with everything else, by
|
|
// making sure all SDL events tunnel through one function.
|
|
// DO NOT DIRECTLY MESS WITH THE SDL EVENT QUEUE THROUGH ANY OTHER FUNCTION.
|
|
// Parameters/retval are same as SDL_PollEvent().
|
|
int SE_SDL_InputEventPoll(SDL_Event *sdlevent)
|
|
{
|
|
ASSERT(sdlevent != NULL);
|
|
CTSingleLock slInput(&sl_csInput, FALSE);
|
|
if(slInput.TryToLock()) {
|
|
const int retval = SDL_PollEvent(sdlevent);
|
|
if (retval)
|
|
sdl_event_handler(sdlevent);
|
|
return retval;
|
|
}
|
|
return 0;
|
|
} // SE_SDL_InputEventPoll
|
|
|
|
|
|
#if 0 // !!! FIXME: Can we support this?
|
|
// --------- 2ND MOUSE HANDLING
|
|
|
|
#define MOUSECOMBUFFERSIZE 256L
|
|
static HANDLE _h2ndMouse = NONE;
|
|
static BOOL _bIgnoreMouse2 = TRUE;
|
|
static INDEX _i2ndMouseX, _i2ndMouseY, _i2ndMouseButtons;
|
|
static INDEX _iByteNum = 0;
|
|
static UBYTE _aubComBytes[4] = {0,0,0,0};
|
|
static INDEX _iLastPort = -1;
|
|
|
|
|
|
|
|
static void Poll2ndMouse(void)
|
|
{
|
|
// reset (mouse reading is relative)
|
|
_i2ndMouseX = 0;
|
|
_i2ndMouseY = 0;
|
|
if( _h2ndMouse==NONE) return;
|
|
|
|
// check
|
|
COMSTAT csComStat;
|
|
DWORD dwErrorFlags;
|
|
ClearCommError( _h2ndMouse, &dwErrorFlags, &csComStat);
|
|
DWORD dwLength = Min( MOUSECOMBUFFERSIZE, (INDEX)csComStat.cbInQue);
|
|
if( dwLength<=0) return;
|
|
|
|
// readout
|
|
UBYTE aubMouseBuffer[MOUSECOMBUFFERSIZE];
|
|
INDEX iRetries = 999;
|
|
while( iRetries>0 && !ReadFile( _h2ndMouse, aubMouseBuffer, dwLength, &dwLength, NULL)) iRetries--;
|
|
if( iRetries<=0) return; // what, didn't make it?
|
|
|
|
// parse the mouse packets
|
|
for( INDEX i=0; i<dwLength; i++)
|
|
{
|
|
// prepare
|
|
if( aubMouseBuffer[i] & 64) _iByteNum = 0;
|
|
if( _iByteNum<4) _aubComBytes[_iByteNum] = aubMouseBuffer[i];
|
|
_iByteNum++;
|
|
|
|
// buttons ?
|
|
if( _iByteNum==1) {
|
|
_i2ndMouseButtons &= ~3;
|
|
_i2ndMouseButtons |= (_aubComBytes[0] & (32+16)) >>4;
|
|
}
|
|
// axes ?
|
|
else if( _iByteNum==3) {
|
|
char iDX = ((_aubComBytes[0] & 3) <<6) + _aubComBytes[1];
|
|
char iDY = ((_aubComBytes[0] & 12) <<4) + _aubComBytes[2];
|
|
_i2ndMouseX += iDX;
|
|
_i2ndMouseY += iDY;
|
|
}
|
|
// 3rd button?
|
|
else if( _iByteNum==4) {
|
|
_i2ndMouseButtons &= ~4;
|
|
if( aubMouseBuffer[i]&32) _i2ndMouseButtons |= 4;
|
|
}
|
|
}
|
|
|
|
// ignore pooling?
|
|
if( _bIgnoreMouse2) {
|
|
if( _i2ndMouseX!=0 || _i2ndMouseY!=0) _bIgnoreMouse2 = FALSE;
|
|
_i2ndMouseX = 0;
|
|
_i2ndMouseY = 0;
|
|
_i2ndMouseButtons = 0;
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
static void Startup2ndMouse(INDEX iPort)
|
|
{
|
|
// skip if disabled
|
|
ASSERT( iPort>=0 && iPort<=4);
|
|
if( iPort==0) return;
|
|
// determine port string
|
|
CTString str2ndMousePort( 0, "COM%d", iPort);
|
|
|
|
// create COM handle if needed
|
|
if( _h2ndMouse==NONE) {
|
|
_h2ndMouse = CreateFile( str2ndMousePort, GENERIC_READ|GENERIC_WRITE, 0, NULL,
|
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
|
|
if( _h2ndMouse==INVALID_HANDLE_VALUE) {
|
|
// failed! :(
|
|
INDEX iError = GetLastError();
|
|
if( iError==5) CPrintF( "Cannot open %s (access denied).\n"
|
|
"The port is probably already used by another device (mouse, modem...)\n",
|
|
str2ndMousePort);
|
|
else CPrintF( "Cannot open %s (error %d).\n", str2ndMousePort, iError);
|
|
_h2ndMouse = NONE;
|
|
return;
|
|
}
|
|
}
|
|
// setup and purge buffers
|
|
SetupComm( _h2ndMouse, MOUSECOMBUFFERSIZE, MOUSECOMBUFFERSIZE);
|
|
PurgeComm( _h2ndMouse, PURGE_TXABORT | PURGE_RXABORT | PURGE_TXCLEAR | PURGE_RXCLEAR);
|
|
|
|
// setup port to 1200 7N1
|
|
DCB dcb;
|
|
dcb.DCBlength = sizeof(DCB);
|
|
GetCommState( _h2ndMouse, &dcb);
|
|
dcb.BaudRate = CBR_1200;
|
|
dcb.ByteSize = 7;
|
|
dcb.Parity = NOPARITY;
|
|
dcb.StopBits = ONESTOPBIT;
|
|
dcb.fDtrControl = DTR_CONTROL_ENABLE;
|
|
dcb.fRtsControl = RTS_CONTROL_ENABLE;
|
|
dcb.fBinary = TRUE;
|
|
dcb.fParity = TRUE;
|
|
SetCommState( _h2ndMouse, &dcb);
|
|
|
|
// reset
|
|
_iByteNum = 0;
|
|
_aubComBytes[0] = _aubComBytes[1] = _aubComBytes[2] = _aubComBytes[3] = 0;
|
|
_bIgnoreMouse2 = TRUE; // ignore mouse polling until 1 after non-0 readout
|
|
_iLastPort = iPort;
|
|
//CPrintF( "STARTUP M2!\n");
|
|
}
|
|
|
|
|
|
static void Shutdown2ndMouse(void)
|
|
{
|
|
// skip if already disabled
|
|
if( _h2ndMouse==NONE) return;
|
|
|
|
// disable!
|
|
SetCommMask( _h2ndMouse, 0);
|
|
EscapeCommFunction( _h2ndMouse, CLRDTR);
|
|
EscapeCommFunction( _h2ndMouse, CLRRTS);
|
|
PurgeComm( _h2ndMouse, PURGE_TXABORT | PURGE_RXABORT | PURGE_TXCLEAR | PURGE_RXCLEAR);
|
|
// close port if changed
|
|
if( _iLastPort != inp_i2ndMousePort) {
|
|
CloseHandle( _h2ndMouse);
|
|
_h2ndMouse = NONE;
|
|
} // over and out
|
|
_bIgnoreMouse2 = TRUE;
|
|
}
|
|
#endif
|
|
|
|
static SDL_Joystick **sticks = NULL;
|
|
static int ctJoysticks = 0;
|
|
|
|
BOOL CInput::PlatformInit(void)
|
|
{
|
|
#if 0
|
|
_h2ndMouse = NONE;
|
|
#endif
|
|
|
|
ASSERT(sticks == NULL);
|
|
ASSERT(ctJoysticks == 0);
|
|
|
|
_pShell->DeclareSymbol("persistent user INDEX inp_bSDLPermitCtrlG;", (void*)&inp_bSDLPermitCtrlG);
|
|
_pShell->DeclareSymbol("persistent user INDEX inp_bSDLGrabInput;", (void*)&inp_bSDLGrabInput);
|
|
MakeConversionTables();
|
|
|
|
sl_csInput.cs_iIndex = 3000;
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
|
|
// destructor
|
|
CInput::~CInput()
|
|
{
|
|
if (sticks != NULL) {
|
|
int max = ctJoysticks;
|
|
for (int i = 0; i < max; i++) {
|
|
if (sticks[i] != NULL) {
|
|
SDL_JoystickClose(sticks[i]);
|
|
}
|
|
}
|
|
delete[] sticks;
|
|
sticks = NULL;
|
|
}
|
|
|
|
ctJoysticks = 0;
|
|
|
|
#if 0
|
|
if (_h2ndMouse != NONE)
|
|
CloseHandle(_h2ndMouse);
|
|
_h2ndMouse = NONE;
|
|
#endif
|
|
}
|
|
|
|
|
|
BOOL CInput::PlatformSetKeyNames(void)
|
|
{
|
|
// for each Key
|
|
for (INDEX iKey=0; iKey<ARRAYCOUNT(_akcKeys); iKey++) {
|
|
const KeyConversion &kc = _akcKeys[iKey];
|
|
// set the name
|
|
if (kc.kc_strName!=NULL) {
|
|
inp_strButtonNames[kc.kc_iKID] = kc.kc_strName;
|
|
if (strlen(kc.kc_strNameTrans)==0) {
|
|
inp_strButtonNamesTra[kc.kc_iKID] = kc.kc_strName;
|
|
} else {
|
|
inp_strButtonNamesTra[kc.kc_iKID] = TranslateConst(kc.kc_strNameTrans, 4);
|
|
}
|
|
}
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
|
|
LONG CInput::PlatformGetJoystickCount(void)
|
|
{
|
|
LONG retval = (LONG) SDL_NumJoysticks();
|
|
if (retval > 0) {
|
|
sticks = new SDL_Joystick *[retval];
|
|
ctJoysticks = (int) retval;
|
|
memset(sticks, '\0', sizeof (SDL_Joystick *) * retval);
|
|
}
|
|
|
|
return(retval);
|
|
}
|
|
|
|
|
|
// check if a joystick exists
|
|
BOOL CInput::CheckJoystick(INDEX iJoy)
|
|
{
|
|
CPrintF(TRANSV(" joy %d:"), iJoy + 1);
|
|
|
|
ASSERT(ctJoysticks > iJoy);
|
|
CPrintF(" '%s'\n", SDL_JoystickNameForIndex(iJoy));
|
|
|
|
SDL_Joystick *stick = SDL_JoystickOpen(iJoy);
|
|
if (stick == NULL) {
|
|
CPrintF(" ...can't open joystick.\n reason: %s\n", SDL_GetError());
|
|
return FALSE;
|
|
}
|
|
|
|
ASSERT(sticks != NULL);
|
|
ASSERT(sticks[iJoy] == NULL);
|
|
sticks[iJoy] = stick;
|
|
|
|
int ctAxes = SDL_JoystickNumAxes(stick);
|
|
CPrintF(TRANSV(" %d axes\n"), ctAxes);
|
|
CPrintF(TRANSV(" %d buttons\n"), SDL_JoystickNumButtons(stick));
|
|
if (SDL_JoystickNumHats(stick) > 0) {
|
|
CPrintF(TRANSV(" POV hat present\n"));
|
|
}
|
|
|
|
// for each axis
|
|
for(INDEX iAxis=0; iAxis<MAX_AXES_PER_JOYSTICK; iAxis++) {
|
|
ControlAxisInfo &cai= inp_caiAllAxisInfo[ FIRST_JOYAXIS+iJoy*MAX_AXES_PER_JOYSTICK+iAxis];
|
|
// remember min/max info
|
|
cai.cai_slMin = -32768; cai.cai_slMax = 32767;
|
|
cai.cai_bExisting = (iAxis < ctAxes);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
void CInput::EnableInput(HWND hwnd)
|
|
{
|
|
// skip if already enabled
|
|
if( inp_bInputEnabled) return;
|
|
|
|
SDL_JoystickEventState(SDL_ENABLE);
|
|
|
|
// determine screen center position
|
|
int winw, winh;
|
|
SDL_GetWindowSize((SDL_Window *) hwnd, &winw, &winh);
|
|
inp_slScreenCenterX = winw / 2;
|
|
inp_slScreenCenterY = winh / 2;
|
|
|
|
// remember mouse pos
|
|
int mousex, mousey;
|
|
SDL_GetMouseState(&mousex, &mousey); // !!! FIXME: this isn't necessarily (hwnd)
|
|
inp_ptOldMousePos.x = mousex;
|
|
inp_ptOldMousePos.y = mousey;
|
|
|
|
SDL_SetRelativeMouseMode(inp_bSDLGrabInput ? SDL_TRUE : SDL_FALSE);
|
|
|
|
// save system mouse settings
|
|
memset(&inp_mscMouseSettings, '\0', sizeof (MouseSpeedControl));
|
|
|
|
#if 0
|
|
// if required, try to enable 2nd mouse
|
|
Shutdown2ndMouse();
|
|
inp_i2ndMousePort = Clamp( inp_i2ndMousePort, 0L, 4L);
|
|
Startup2ndMouse(inp_i2ndMousePort);
|
|
#endif
|
|
|
|
// clear button's buffer
|
|
memset( _abKeysPressed, 0, sizeof( _abKeysPressed));
|
|
|
|
// remember current status
|
|
inp_bInputEnabled = TRUE;
|
|
inp_bPollJoysticks = FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
* Disable direct input
|
|
*/
|
|
void CInput::DisableInput( void)
|
|
{
|
|
// skip if allready disabled
|
|
if( !inp_bInputEnabled) return;
|
|
|
|
SDL_JoystickEventState(SDL_DISABLE);
|
|
|
|
// show mouse on screen
|
|
SDL_SetRelativeMouseMode(SDL_FALSE);
|
|
|
|
// eventually disable 2nd mouse
|
|
#if 0
|
|
Shutdown2ndMouse();
|
|
#endif
|
|
|
|
// remember current status
|
|
inp_bInputEnabled = FALSE;
|
|
inp_bPollJoysticks = FALSE;
|
|
}
|
|
|
|
#define USE_MOUSEWARP 1
|
|
// Define this to use GetMouse instead of using Message to read mouse coordinates
|
|
|
|
// blank any queued mousemove events...SDLInput.cpp needs this when
|
|
// returning from the menus/console to game or the viewport will jump...
|
|
void CInput::ClearRelativeMouseMotion(void)
|
|
{
|
|
#if USE_MOUSEWARP
|
|
SDL_GetRelativeMouseState(NULL, NULL);
|
|
#endif
|
|
mouse_relative_x = mouse_relative_y = 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Scan states of all available input sources
|
|
*/
|
|
void CInput::GetInput(BOOL bPreScan)
|
|
{
|
|
// CTSingleLock sl(&csInput, TRUE);
|
|
|
|
if (!inp_bInputEnabled) {
|
|
return;
|
|
}
|
|
|
|
if (bPreScan && !inp_bAllowPrescan) {
|
|
return;
|
|
}
|
|
|
|
#if 0
|
|
// should not be usefull
|
|
SDL_Event event;
|
|
while (SE_SDL_InputEventPoll(&event)) { /* do nothing... */ }
|
|
#endif
|
|
|
|
// if not pre-scanning
|
|
if (!bPreScan) {
|
|
// clear button's buffer
|
|
memset( inp_ubButtonsBuffer, 0, sizeof( inp_ubButtonsBuffer));
|
|
|
|
// for each Key
|
|
for (INDEX iKey=0; iKey<ARRAYCOUNT(_akcKeys); iKey++) {
|
|
const KeyConversion &kc = _akcKeys[iKey];
|
|
// get codes
|
|
INDEX iKID = kc.kc_iKID;
|
|
//INDEX iScan = kc.kc_iScanCode;
|
|
INDEX iVirt = kc.kc_iVirtKey;
|
|
// if reading async keystate
|
|
if (inp_iKeyboardReadingMethod==0) {
|
|
// if there is a valid virtkey
|
|
if (iVirt>=0) {
|
|
// is state is pressed
|
|
if (SDL_GetKeyboardState(NULL)[SDL_GetScancodeFromKey((SDL_Keycode)iVirt)]) {
|
|
// mark it as pressed
|
|
inp_ubButtonsBuffer[iKID] = 0xFF;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// if snooped that key is pressed
|
|
if (_abKeysPressed[iKID]) {
|
|
// mark it as pressed
|
|
inp_ubButtonsBuffer[iKID] = 0xFF;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// read mouse position
|
|
#ifdef USE_MOUSEWARP
|
|
int iMx, iMy;
|
|
SDL_GetRelativeMouseState(&iMx, &iMy);
|
|
mouse_relative_x = iMx;
|
|
mouse_relative_y = iMy;
|
|
#else
|
|
if ((mouse_relative_x != 0) || (mouse_relative_y != 0))
|
|
#endif
|
|
{
|
|
FLOAT fDX = FLOAT( mouse_relative_x );
|
|
FLOAT fDY = FLOAT( mouse_relative_y );
|
|
|
|
mouse_relative_x = mouse_relative_y = 0;
|
|
|
|
FLOAT fSensitivity = inp_fMouseSensitivity;
|
|
if( inp_bAllowMouseAcceleration) fSensitivity *= 0.25f;
|
|
|
|
FLOAT fD = Sqrt(fDX*fDX+fDY*fDY);
|
|
if (inp_bMousePrecision) {
|
|
static FLOAT _tmTime = 0.0f;
|
|
if( fD<inp_fMousePrecisionThreshold) _tmTime += 0.05f;
|
|
else _tmTime = 0.0f;
|
|
if( _tmTime>inp_fMousePrecisionTimeout) fSensitivity /= inp_fMousePrecisionFactor;
|
|
}
|
|
|
|
static FLOAT fDXOld;
|
|
static FLOAT fDYOld;
|
|
static TIME tmOldDelta;
|
|
static CTimerValue tvBefore;
|
|
CTimerValue tvNow = _pTimer->GetHighPrecisionTimer();
|
|
TIME tmNowDelta = (tvNow-tvBefore).GetSeconds();
|
|
if (tmNowDelta<0.001f) {
|
|
tmNowDelta = 0.001f;
|
|
}
|
|
tvBefore = tvNow;
|
|
|
|
FLOAT fDXSmooth = (fDXOld*tmOldDelta+fDX*tmNowDelta)/(tmOldDelta+tmNowDelta);
|
|
FLOAT fDYSmooth = (fDYOld*tmOldDelta+fDY*tmNowDelta)/(tmOldDelta+tmNowDelta);
|
|
fDXOld = fDX;
|
|
fDYOld = fDY;
|
|
tmOldDelta = tmNowDelta;
|
|
if (inp_bFilterMouse) {
|
|
fDX = fDXSmooth;
|
|
fDY = fDYSmooth;
|
|
}
|
|
|
|
// get final mouse values
|
|
FLOAT fMouseRelX = +fDX*fSensitivity;
|
|
FLOAT fMouseRelY = -fDY*fSensitivity;
|
|
if (inp_bInvertMouse) {
|
|
fMouseRelY = -fMouseRelY;
|
|
}
|
|
FLOAT fMouseRelZ = _iMouseZ;
|
|
|
|
// just interpret values as normal
|
|
inp_caiAllAxisInfo[1].cai_fReading = fMouseRelX;
|
|
inp_caiAllAxisInfo[2].cai_fReading = fMouseRelY;
|
|
inp_caiAllAxisInfo[3].cai_fReading = fMouseRelZ;
|
|
}
|
|
#ifndef USE_MOUSEWARP
|
|
else {
|
|
inp_caiAllAxisInfo[1].cai_fReading = 0.0;
|
|
inp_caiAllAxisInfo[2].cai_fReading = 0.0;
|
|
inp_caiAllAxisInfo[3].cai_fReading = 0.0;
|
|
}
|
|
#endif
|
|
|
|
// if not pre-scanning
|
|
if (!bPreScan) {
|
|
// detect wheel up/down movement
|
|
|
|
if (_iMouseZ>0) {
|
|
if (_bWheelUp) {
|
|
inp_ubButtonsBuffer[KID_MOUSEWHEELUP] = 0x00;
|
|
} else {
|
|
inp_ubButtonsBuffer[KID_MOUSEWHEELUP] = 0xFF;
|
|
_iMouseZ = ClampDn(_iMouseZ-MOUSE_SCROLL_INTERVAL, 0);
|
|
}
|
|
}
|
|
_bWheelUp = inp_ubButtonsBuffer[KID_MOUSEWHEELUP];
|
|
if (_iMouseZ<0) {
|
|
if (_bWheelDn) {
|
|
inp_ubButtonsBuffer[KID_MOUSEWHEELDOWN] = 0x00;
|
|
} else {
|
|
inp_ubButtonsBuffer[KID_MOUSEWHEELDOWN] = 0xFF;
|
|
_iMouseZ = ClampUp(_iMouseZ+MOUSE_SCROLL_INTERVAL, 0);
|
|
}
|
|
}
|
|
_bWheelDn = inp_ubButtonsBuffer[KID_MOUSEWHEELDOWN];
|
|
}
|
|
|
|
inp_bLastPrescan = bPreScan;
|
|
|
|
// !!! FIXME
|
|
#if 0
|
|
// readout 2nd mouse if enabled
|
|
if( _h2ndMouse!=NONE)
|
|
{
|
|
Poll2ndMouse();
|
|
//CPrintF( "m2X: %4d, m2Y: %4d, m2B: 0x%02X\n", _i2ndMouseX, _i2ndMouseY, _i2ndMouseButtons);
|
|
|
|
// handle 2nd mouse buttons
|
|
if( _i2ndMouseButtons & 2) inp_ubButtonsBuffer[KID_2MOUSE1] = 0xFF;
|
|
if( _i2ndMouseButtons & 1) inp_ubButtonsBuffer[KID_2MOUSE2] = 0xFF;
|
|
if( _i2ndMouseButtons & 4) inp_ubButtonsBuffer[KID_2MOUSE3] = 0xFF;
|
|
|
|
// handle 2nd mouse movement
|
|
FLOAT fDX = _i2ndMouseX;
|
|
FLOAT fDY = _i2ndMouseY;
|
|
FLOAT fSensitivity = inp_f2ndMouseSensitivity;
|
|
|
|
FLOAT fD = Sqrt(fDX*fDX+fDY*fDY);
|
|
if( inp_b2ndMousePrecision) {
|
|
static FLOAT _tm2Time = 0.0f;
|
|
if( fD<inp_f2ndMousePrecisionThreshold) _tm2Time += 0.05f;
|
|
else _tm2Time = 0.0f;
|
|
if( _tm2Time>inp_f2ndMousePrecisionTimeout) fSensitivity /= inp_f2ndMousePrecisionFactor;
|
|
}
|
|
|
|
static FLOAT f2DXOld;
|
|
static FLOAT f2DYOld;
|
|
static TIME tm2OldDelta;
|
|
static CTimerValue tv2Before;
|
|
CTimerValue tvNow = _pTimer->GetHighPrecisionTimer();
|
|
TIME tmNowDelta = (tvNow-tv2Before).GetSeconds();
|
|
if( tmNowDelta<0.001f) tmNowDelta = 0.001f;
|
|
tv2Before = tvNow;
|
|
|
|
FLOAT fDXSmooth = (f2DXOld*tm2OldDelta+fDX*tmNowDelta) / (tm2OldDelta+tmNowDelta);
|
|
FLOAT fDYSmooth = (f2DYOld*tm2OldDelta+fDY*tmNowDelta) / (tm2OldDelta+tmNowDelta);
|
|
f2DXOld = fDX;
|
|
f2DYOld = fDY;
|
|
tm2OldDelta = tmNowDelta;
|
|
if( inp_bFilter2ndMouse) {
|
|
fDX = fDXSmooth;
|
|
fDY = fDYSmooth;
|
|
}
|
|
|
|
// get final mouse values
|
|
FLOAT fMouseRelX = +fDX*fSensitivity;
|
|
FLOAT fMouseRelY = -fDY*fSensitivity;
|
|
if( inp_bInvert2ndMouse) fMouseRelY = -fMouseRelY;
|
|
|
|
// just interpret values as normal
|
|
inp_caiAllAxisInfo[4].cai_fReading = fMouseRelX;
|
|
inp_caiAllAxisInfo[5].cai_fReading = fMouseRelY;
|
|
}
|
|
#endif
|
|
|
|
// if joystick polling is enabled
|
|
if (inp_bPollJoysticks || inp_bForceJoystickPolling) {
|
|
// scan all available joysticks
|
|
for( INDEX iJoy=0; iJoy<MAX_JOYSTICKS; iJoy++) {
|
|
if (inp_abJoystickOn[iJoy] && iJoy<inp_ctJoysticksAllowed) {
|
|
// scan joy state
|
|
BOOL bSucceeded = ScanJoystick(iJoy, bPreScan);
|
|
// if joystick reading failed
|
|
if (!bSucceeded && inp_bAutoDisableJoysticks) {
|
|
// kill it, so it doesn't slow down CPU
|
|
CPrintF(TRANSV("Joystick %d failed, disabling it!\n"), iJoy+1);
|
|
inp_abJoystickOn[iJoy] = FALSE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Scans axis and buttons for given joystick
|
|
*/
|
|
BOOL CInput::ScanJoystick(INDEX iJoy, BOOL bPreScan)
|
|
{
|
|
ASSERT(ctJoysticks > iJoy);
|
|
ASSERT(sticks != NULL);
|
|
ASSERT(sticks[iJoy] != NULL);
|
|
|
|
SDL_Joystick *stick = sticks[iJoy];
|
|
|
|
// for each available axis
|
|
for( INDEX iAxis=0; iAxis<MAX_AXES_PER_JOYSTICK; iAxis++) {
|
|
ControlAxisInfo &cai = inp_caiAllAxisInfo[ FIRST_JOYAXIS+iJoy*MAX_AXES_PER_JOYSTICK+iAxis];
|
|
// if the axis is not present
|
|
if (!cai.cai_bExisting) {
|
|
// read as zero
|
|
cai.cai_fReading = 0.0f;
|
|
// skip to next axis
|
|
continue;
|
|
}
|
|
// read its state
|
|
SLONG slAxisReading = SDL_JoystickGetAxis(stick, iAxis);
|
|
|
|
// convert from min..max to -1..+1
|
|
FLOAT fAxisReading = FLOAT(slAxisReading-cai.cai_slMin)/(cai.cai_slMax-cai.cai_slMin)*2.0f-1.0f;
|
|
|
|
// set current axis value
|
|
cai.cai_fReading = fAxisReading;
|
|
}
|
|
|
|
// if not pre-scanning
|
|
if (!bPreScan) {
|
|
INDEX iButtonTotal = FIRST_JOYBUTTON+iJoy*MAX_BUTTONS_PER_JOYSTICK;
|
|
// for each available button
|
|
for( INDEX iButton=0; iButton<32; iButton++) {
|
|
// test if the button is pressed
|
|
if (SDL_JoystickGetButton(stick, iButton)) {
|
|
inp_ubButtonsBuffer[ iButtonTotal++] = 128;
|
|
} else {
|
|
inp_ubButtonsBuffer[ iButtonTotal++] = 0;
|
|
}
|
|
}
|
|
|
|
/*
|
|
!!! FIXME: Support hats as extra buttons...
|
|
// POV hat initially not pressed
|
|
// CPrintF("%d\n", ji.dwPOV);
|
|
INDEX iStartPOV = iButtonTotal;
|
|
inp_ubButtonsBuffer[ iStartPOV+0] = 0;
|
|
inp_ubButtonsBuffer[ iStartPOV+1] = 0;
|
|
inp_ubButtonsBuffer[ iStartPOV+2] = 0;
|
|
inp_ubButtonsBuffer[ iStartPOV+3] = 0;
|
|
// check the four pov directions
|
|
if (ji.dwPOV==JOY_POVFORWARD) {
|
|
inp_ubButtonsBuffer[ iStartPOV+0] = 128;
|
|
} else if (ji.dwPOV==JOY_POVRIGHT) {
|
|
inp_ubButtonsBuffer[ iStartPOV+1] = 128;
|
|
} else if (ji.dwPOV==JOY_POVBACKWARD) {
|
|
inp_ubButtonsBuffer[ iStartPOV+2] = 128;
|
|
} else if (ji.dwPOV==JOY_POVLEFT) {
|
|
inp_ubButtonsBuffer[ iStartPOV+3] = 128;
|
|
// and four mid-positions
|
|
} else if (ji.dwPOV==JOY_POVFORWARD+4500) {
|
|
inp_ubButtonsBuffer[ iStartPOV+0] = 128;
|
|
inp_ubButtonsBuffer[ iStartPOV+1] = 128;
|
|
} else if (ji.dwPOV==JOY_POVRIGHT+4500) {
|
|
inp_ubButtonsBuffer[ iStartPOV+1] = 128;
|
|
inp_ubButtonsBuffer[ iStartPOV+2] = 128;
|
|
} else if (ji.dwPOV==JOY_POVBACKWARD+4500) {
|
|
inp_ubButtonsBuffer[ iStartPOV+2] = 128;
|
|
inp_ubButtonsBuffer[ iStartPOV+3] = 128;
|
|
} else if (ji.dwPOV==JOY_POVLEFT+4500) {
|
|
inp_ubButtonsBuffer[ iStartPOV+3] = 128;
|
|
inp_ubButtonsBuffer[ iStartPOV+0] = 128;
|
|
}
|
|
*/
|
|
|
|
}
|
|
|
|
// successful
|
|
return TRUE;
|
|
}
|
|
|
|
// end of SDLInput.cpp ...
|
|
|
|
|