/* * control.c * MACT library controller handling * * Derived from MACT386.LIB disassembly by Jonathon Fowler * */ #include "_control.h" #include "baselayer.h" #include "compat.h" #include "control.h" #include "joystick.h" #include "keyboard.h" #include "mouse.h" #include "osd.h" #include "pragmas.h" bool CONTROL_Started = false; bool CONTROL_MouseEnabled = false; bool CONTROL_MousePresent = false; bool CONTROL_JoyPresent = false; bool CONTROL_JoystickEnabled = false; LastSeenInput CONTROL_LastSeenInput; float CONTROL_MouseSensitivity = DEFAULTMOUSESENSITIVITY; static int32_t CONTROL_NumMouseButtons = 0; static int32_t CONTROL_NumJoyButtons = 0; static int32_t CONTROL_NumJoyAxes = 0; // static controlkeymaptype CONTROL_KeyMapping[CONTROL_NUM_FLAGS]; static int32_t CONTROL_MouseAxesScale[2]; static controlaxismaptype CONTROL_JoyAxesMap[MAXJOYAXES]; static controlaxistype CONTROL_JoyAxes[MAXJOYAXES]; static controlaxistype CONTROL_LastJoyAxes[MAXJOYAXES]; static int32_t CONTROL_JoyAxesScale[MAXJOYAXES]; static int8_t CONTROL_JoyAxesInvert[MAXJOYAXES]; static controlbuttontype CONTROL_MouseButtonMapping[MAXMOUSEBUTTONS]; static int32_t CONTROL_MouseButtonClicked[MAXMOUSEBUTTONS]; static int32_t CONTROL_MouseButtonClickedState[MAXMOUSEBUTTONS]; static int32_t CONTROL_MouseButtonClickedTime[MAXMOUSEBUTTONS]; static int32_t CONTROL_MouseButtonState[MAXMOUSEBUTTONS]; static uint8_t CONTROL_MouseButtonClickedCount[MAXMOUSEBUTTONS]; static controlbuttontype CONTROL_JoyButtonMapping[MAXJOYBUTTONS]; static int32_t CONTROL_JoyButtonClicked[MAXJOYBUTTONS]; static int32_t CONTROL_JoyButtonClickedState[MAXJOYBUTTONS]; static int32_t CONTROL_JoyButtonClickedTime[MAXJOYBUTTONS]; static int32_t CONTROL_JoyButtonState[MAXJOYBUTTONS]; static uint8_t CONTROL_JoyButtonClickedCount[MAXJOYBUTTONS]; static int32_t(*ExtGetTime)(void); //static int32_t ticrate; static uint8_t CONTROL_DoubleClickSpeed; int32_t CONTROL_ButtonFlags[CONTROL_NUM_FLAGS]; consolekeybind_t CONTROL_KeyBinds[NUMKEYS + MAXMOUSEBUTTONS]; bool CONTROL_BindsEnabled = 0; bool CONTROL_SmoothMouse = 0; #define CONTROL_CheckRange(which) ((unsigned)which >= (unsigned)CONTROL_NUM_FLAGS) #define BIND(x, s, r, k) do { Xfree(x.cmdstr); x.cmdstr = s; x.repeat = r; x.key = k; } while (0) void CONTROL_ClearAllBinds(void) { for (int i=0; imousex = mulscale16(Blrintf(finput.x * 4.f * CONTROL_MouseSensitivity), CONTROL_MouseAxesScale[0]); info->mousey = mulscale16(Blrintf(finput.y * 4.f * CONTROL_MouseSensitivity), CONTROL_MouseAxesScale[1]); } static int32_t CONTROL_GetTime(void) { static int32_t t = 0; t += 5; return t; } void CONTROL_MapButton(int whichfunction, int whichbutton, int doubleclicked, controldevice device) { controlbuttontype *set; if (CONTROL_CheckRange(whichfunction)) whichfunction = BUTTONUNDEFINED; switch (device) { case controldevice_mouse: if ((unsigned)whichbutton >= (unsigned)MAXMOUSEBUTTONS) { //Error("CONTROL_MapButton: button %d out of valid range for %d mouse buttons.", // whichbutton, CONTROL_NumMouseButtons); return; } set = CONTROL_MouseButtonMapping; break; case controldevice_joystick: if ((unsigned)whichbutton >= (unsigned)MAXJOYBUTTONS) { //Error("CONTROL_MapButton: button %d out of valid range for %d joystick buttons.", // whichbutton, CONTROL_NumJoyButtons); return; } set = CONTROL_JoyButtonMapping; break; default: //Error("CONTROL_MapButton: invalid controller device type"); return; } if (doubleclicked) set[whichbutton].doubleclicked = whichfunction; else set[whichbutton].singleclicked = whichfunction; } void CONTROL_MapAnalogAxis(int whichaxis, int whichanalog, controldevice device) { controlaxismaptype *set; if ((unsigned)whichanalog >= (unsigned)analog_maxtype && whichanalog != -1) { //Error("CONTROL_MapAnalogAxis: analog function %d out of valid range for %d analog functions.", // whichanalog, analog_maxtype); return; } switch (device) { case controldevice_joystick: if ((unsigned)whichaxis >= (unsigned)MAXJOYAXES) { //Error("CONTROL_MapAnalogAxis: axis %d out of valid range for %d joystick axes.", // whichaxis, MAXJOYAXES); return; } set = CONTROL_JoyAxesMap; break; default: //Error("CONTROL_MapAnalogAxis: invalid controller device type"); return; } set[whichaxis].analogmap = whichanalog; } void CONTROL_SetAnalogAxisScale(int32_t whichaxis, int32_t axisscale, controldevice device) { int32_t *set; switch (device) { case controldevice_mouse: if ((unsigned) whichaxis >= ARRAY_SIZE(CONTROL_MouseAxesScale)) { //Error("CONTROL_SetAnalogAxisScale: axis %d out of valid range for %d mouse axes.", // whichaxis, MAXMOUSEAXES); return; } set = CONTROL_MouseAxesScale; break; case controldevice_joystick: if ((unsigned) whichaxis >= (unsigned) MAXJOYAXES) { //Error("CONTROL_SetAnalogAxisScale: axis %d out of valid range for %d joystick axes.", // whichaxis, MAXJOYAXES); return; } set = CONTROL_JoyAxesScale; break; default: //Error("CONTROL_SetAnalogAxisScale: invalid controller device type"); return; } set[whichaxis] = axisscale; } void CONTROL_SetAnalogAxisInvert(int32_t whichaxis, int32_t invert, controldevice device) { int8_t * set; switch (device) { case controldevice_joystick: if ((unsigned) whichaxis >= (unsigned) MAXJOYAXES) { //Error("CONTROL_SetAnalogAxisInvert: axis %d out of valid range for %d joystick axes.", // whichaxis, MAXJOYAXES); return; } set = CONTROL_JoyAxesInvert; break; default: //Error("CONTROL_SetAnalogAxisInvert: invalid controller device type"); return; } set[whichaxis] = invert; } void CONTROL_MapDigitalAxis(int32_t whichaxis, int32_t whichfunction, int32_t direction, controldevice device) { controlaxismaptype *set; if (CONTROL_CheckRange(whichfunction)) whichfunction = AXISUNDEFINED; switch (device) { case controldevice_joystick: if ((unsigned) whichaxis >= (unsigned) MAXJOYAXES) { //Error("CONTROL_MapDigitalAxis: axis %d out of valid range for %d joystick axes.", // whichaxis, MAXJOYAXES); return; } set = CONTROL_JoyAxesMap; break; default: //Error("CONTROL_MapDigitalAxis: invalid controller device type"); return; } switch (direction) // JBF: this is all very much a guess. The ASM puzzles me. { case axis_up: case axis_left: set[whichaxis].minmap = whichfunction; break; case axis_down: case axis_right: set[whichaxis].maxmap = whichfunction; break; default: break; } } void CONTROL_ClearAssignments(void) { memset(CONTROL_JoyAxes, 0, sizeof(CONTROL_JoyAxes)); memset(CONTROL_JoyAxesInvert, 0, sizeof(CONTROL_JoyAxesInvert)); memset(CONTROL_JoyAxesMap, AXISUNDEFINED, sizeof(CONTROL_JoyAxesMap)); memset(CONTROL_JoyButtonMapping, BUTTONUNDEFINED, sizeof(CONTROL_JoyButtonMapping)); // memset(CONTROL_KeyMapping, KEYUNDEFINED, sizeof(CONTROL_KeyMapping)); memset(CONTROL_LastJoyAxes, 0, sizeof(CONTROL_LastJoyAxes)); memset(CONTROL_MouseButtonMapping, BUTTONUNDEFINED, sizeof(CONTROL_MouseButtonMapping)); for (int & i : CONTROL_MouseAxesScale) i = NORMALAXISSCALE; for (int & i : CONTROL_JoyAxesScale) i = NORMALAXISSCALE; } static int DoGetDeviceButtons( int32_t buttons, int32_t tm, int32_t NumButtons, int32_t *DeviceButtonState, int32_t *ButtonClickedTime, int32_t *ButtonClickedState, int32_t *ButtonClicked, uint8_t *ButtonClickedCount ) { int32_t i=NumButtons-1; int retval = 0; for (; i>=0; i--) { int const bs = (buttons >> i) & 1; DeviceButtonState[i] = bs; ButtonClickedState[i] = FALSE; if (bs) { retval = 1; if (ButtonClicked[i] == FALSE) { ButtonClicked[i] = TRUE; if (ButtonClickedCount[i] == 0 || tm > ButtonClickedTime[i]) { ButtonClickedTime[i] = tm + CONTROL_DoubleClickSpeed; ButtonClickedCount[i] = 1; } else if (tm < ButtonClickedTime[i]) { ButtonClickedState[i] = TRUE; ButtonClickedTime[i] = 0; ButtonClickedCount[i] = 2; } } else if (ButtonClickedCount[i] == 2) { ButtonClickedState[i] = TRUE; } continue; } if (ButtonClickedCount[i] == 2) ButtonClickedCount[i] = 0; ButtonClicked[i] = FALSE; } return retval; } static void CONTROL_GetDeviceButtons(void) { int32_t const t = ExtGetTime(); if (CONTROL_MouseEnabled) { DoGetDeviceButtons( MOUSE_GetButtons(), t, CONTROL_NumMouseButtons, CONTROL_MouseButtonState, CONTROL_MouseButtonClickedTime, CONTROL_MouseButtonClickedState, CONTROL_MouseButtonClicked, CONTROL_MouseButtonClickedCount ); } if (CONTROL_JoystickEnabled) { int retval = DoGetDeviceButtons( JOYSTICK_GetButtons(), t, CONTROL_NumJoyButtons, CONTROL_JoyButtonState, CONTROL_JoyButtonClickedTime, CONTROL_JoyButtonClickedState, CONTROL_JoyButtonClicked, CONTROL_JoyButtonClickedCount ); if (retval) CONTROL_LastSeenInput = LastSeenInput::Joystick; } } static int CONTROL_DigitizeAxis(int axis, controldevice device) { controlaxistype *set, *lastset; switch (device) { case controldevice_joystick: set = CONTROL_JoyAxes; lastset = CONTROL_LastJoyAxes; break; default: return 0; } set[axis].digitalClearedN = lastset[axis].digitalClearedN; set[axis].digitalClearedP = lastset[axis].digitalClearedP; if (set[axis].analog > 0) { set[axis].digitalClearedN = 0; if (set[axis].analog > THRESHOLD || (set[axis].analog > MINTHRESHOLD && lastset[axis].digital == 1)) set[axis].digital = 1; else set[axis].digitalClearedP = 0; return 1; } else if (set[axis].analog < 0) { set[axis].digitalClearedP = 0; if (set[axis].analog < -THRESHOLD || (set[axis].analog < -MINTHRESHOLD && lastset[axis].digital == -1)) set[axis].digital = -1; else set[axis].digitalClearedN = 0; return 1; } else { set[axis].digitalClearedN = 0; set[axis].digitalClearedP = 0; } return 0; } static void CONTROL_ScaleAxis(int axis, controldevice device) { controlaxistype *set; int32_t *scale; int8_t * invert; switch (device) { case controldevice_joystick: set = CONTROL_JoyAxes; scale = CONTROL_JoyAxesScale; invert = CONTROL_JoyAxesInvert; break; default: return; } int const invertResult = !!invert[axis]; set[axis].analog = (mulscale16(set[axis].analog, scale[axis]) ^ -invertResult) + invertResult; } static void CONTROL_ApplyAxis(int axis, ControlInfo *info, controldevice device) { controlaxistype *set; controlaxismaptype *map; switch (device) { case controldevice_joystick: set = CONTROL_JoyAxes; map = CONTROL_JoyAxesMap; break; default: return; } switch (map[axis].analogmap) { case analog_turning: info->dyaw += set[axis].analog; break; case analog_strafing: info->dx += set[axis].analog; break; case analog_lookingupanddown: info->dpitch += set[axis].analog; break; case analog_elevation: info->dy += set[axis].analog; break; case analog_rolling: info->droll += set[axis].analog; break; case analog_moving: info->dz += set[axis].analog; break; default: break; } } static void CONTROL_PollDevices(ControlInfo *info) { memset(info, 0, sizeof(ControlInfo)); if (CONTROL_MouseEnabled) CONTROL_GetMouseDelta(info); if (CONTROL_JoystickEnabled) { Bmemcpy(CONTROL_LastJoyAxes, CONTROL_JoyAxes, sizeof(CONTROL_JoyAxes)); memset(CONTROL_JoyAxes, 0, sizeof(CONTROL_JoyAxes)); for (int i=joystick.numAxes-1; i>=0; i--) { CONTROL_JoyAxes[i].analog = joystick.pAxis[i]; if (CONTROL_DigitizeAxis(i, controldevice_joystick)) CONTROL_LastSeenInput = LastSeenInput::Joystick; CONTROL_ScaleAxis(i, controldevice_joystick); LIMITCONTROL(&CONTROL_JoyAxes[i].analog); CONTROL_ApplyAxis(i, info, controldevice_joystick); } } CONTROL_GetDeviceButtons(); } static int CONTROL_HandleAxisFunction(int32_t *p1, controlaxistype *axes, controlaxismaptype *axismap, int numAxes) { int axis = numAxes - 1; int retval = 0; do { if (!axes[axis].digital) continue; int const j = (axes[axis].digital < 0) ? axismap[axis].minmap : axismap[axis].maxmap; if (j != AXISUNDEFINED) { p1[j] = 1; retval = 1; } } while (axis--); return retval; } static void CONTROL_AxisFunctionState(int32_t *p1) { if (CONTROL_NumJoyAxes) { if (CONTROL_HandleAxisFunction(p1, CONTROL_JoyAxes, CONTROL_JoyAxesMap, CONTROL_NumJoyAxes)) CONTROL_LastSeenInput = LastSeenInput::Joystick; } } static void CONTROL_ButtonFunctionState(int32_t *p1) { if (CONTROL_NumMouseButtons) { int i = CONTROL_NumMouseButtons-1, j; do { if (!CONTROL_KeyBinds[NUMKEYS + i].cmdstr) { j = CONTROL_MouseButtonMapping[i].doubleclicked; if (j != KEYUNDEFINED) p1[j] |= CONTROL_MouseButtonClickedState[i]; j = CONTROL_MouseButtonMapping[i].singleclicked; if (j != KEYUNDEFINED) p1[j] |= CONTROL_MouseButtonState[i]; } if (!CONTROL_BindsEnabled) continue; if (CONTROL_KeyBinds[NUMKEYS + i].cmdstr && CONTROL_MouseButtonState[i]) { if (CONTROL_KeyBinds[NUMKEYS + i].repeat || (CONTROL_KeyBinds[NUMKEYS + i].laststate == 0)) OSD_Dispatch(CONTROL_KeyBinds[NUMKEYS + i].cmdstr); } CONTROL_KeyBinds[NUMKEYS + i].laststate = CONTROL_MouseButtonState[i]; } while (i--); } if (CONTROL_NumJoyButtons) { int i=CONTROL_NumJoyButtons-1, j; int retval = 0; do { j = CONTROL_JoyButtonMapping[i].doubleclicked; if (j != KEYUNDEFINED) { auto const state = CONTROL_JoyButtonClickedState[i]; p1[j] |= state; retval |= state; } j = CONTROL_JoyButtonMapping[i].singleclicked; if (j != KEYUNDEFINED) { auto const state = CONTROL_JoyButtonState[i]; p1[j] |= state; retval |= state; } } while (i--); if (retval) CONTROL_LastSeenInput = LastSeenInput::Joystick; } } void CONTROL_ClearButton(int whichbutton) { if (CONTROL_CheckRange(whichbutton)) return; #ifdef __ANDROID__ CONTROL_Android_ClearButton(whichbutton); #endif inputState.ClearButton(whichbutton); CONTROL_Flags[whichbutton].cleared = TRUE; } int32_t CONTROL_GetGameControllerDigitalAxisPos(int32_t axis) { if (!joystick.isGameController) return 0; return CONTROL_JoyAxes[axis].digital > 0 && !CONTROL_JoyAxes[axis].digitalClearedP; } int32_t CONTROL_GetGameControllerDigitalAxisNeg(int32_t axis) { if (!joystick.isGameController) return 0; return CONTROL_JoyAxes[axis].digital < 0 && !CONTROL_JoyAxes[axis].digitalClearedN; } void CONTROL_ClearGameControllerDigitalAxisPos(int32_t axis) { if (!joystick.isGameController) return; CONTROL_JoyAxes[axis].digitalClearedP = 1; } void CONTROL_ClearGameControllerDigitalAxisNeg(int32_t axis) { if (!joystick.isGameController) return; CONTROL_JoyAxes[axis].digitalClearedN = 1; } void CONTROL_ProcessBinds(void) { if (!CONTROL_BindsEnabled) return; int i = NUMKEYS-1; do { if (CONTROL_KeyBinds[i].cmdstr) { auto const keyPressed = KB_KeyPressed(i); if (keyPressed && (CONTROL_KeyBinds[i].repeat || (CONTROL_KeyBinds[i].laststate == 0))) { CONTROL_LastSeenInput = LastSeenInput::Keyboard; OSD_Dispatch(CONTROL_KeyBinds[i].cmdstr); } CONTROL_KeyBinds[i].laststate = keyPressed; } } while (i--); } static void CONTROL_GetFunctionInput(void) { CONTROL_ButtonFunctionState(CONTROL_ButtonFlags); CONTROL_AxisFunctionState(CONTROL_ButtonFlags); for (int i = 0; i < NUMGAMEFUNCTIONS; i++ ) { inputState.UpdateButton(i, CONTROL_ButtonFlags[i]); } memset(CONTROL_ButtonFlags, 0, sizeof(CONTROL_ButtonFlags)); } void CONTROL_GetInput(ControlInfo *info) { CONTROL_PollDevices(info); CONTROL_GetFunctionInput(); inputchecked = 1; } static void CONTROL_ResetJoystickValues() { CONTROL_NumJoyAxes = min(MAXJOYAXES, joystick.numAxes); CONTROL_NumJoyButtons = min(MAXJOYBUTTONS, joystick.numButtons + 4 * (joystick.numHats > 0)); CONTROL_JoystickEnabled = CONTROL_JoyPresent = !!((inputdevices & 4) >> 2); } void CONTROL_ScanForControllers() { joyScanDevices(); CONTROL_ResetJoystickValues(); } bool CONTROL_Startup(controltype which, int32_t(*TimeFunction)(void), int32_t ticspersecond) { UNREFERENCED_PARAMETER(which); if (CONTROL_Started) return false; ExtGetTime = TimeFunction ? TimeFunction : CONTROL_GetTime; // what the fuck??? CONTROL_DoubleClickSpeed = (ticspersecond * 57) / 100; if (CONTROL_DoubleClickSpeed <= 0) CONTROL_DoubleClickSpeed = 1; if (initinput()) return true; KB_Startup(); CONTROL_NumMouseButtons = MAXMOUSEBUTTONS; CONTROL_MousePresent = Mouse_Init(); CONTROL_MouseEnabled = CONTROL_MousePresent; CONTROL_ResetJoystickValues(); CONTROL_Started = TRUE; return false; } void CONTROL_Shutdown(void) { if (!CONTROL_Started) return; CONTROL_ClearAllBinds(); MOUSE_Shutdown(); uninitinput(); CONTROL_Started = FALSE; }