/* * control.c * MACT library controller handling * * Derived from MACT386.LIB disassembly by Jonathon Fowler * */ #include "compat.h" #include "keyboard.h" #include "mouse.h" #include "joystick.h" #include "control.h" #include "_control.h" #include "baselayer.h" #include "osd.h" #include "pragmas.h" int32_t CONTROL_JoyPresent = FALSE; int32_t CONTROL_JoystickEnabled = FALSE; int32_t CONTROL_MousePresent = FALSE; int32_t CONTROL_MouseEnabled = FALSE; uint64_t CONTROL_ButtonState = 0; uint64_t CONTROL_ButtonHeldState = 0; // static int32_t CONTROL_UserInputDelay = -1; float CONTROL_MouseSensitivity = DEFAULTMOUSESENSITIVITY; static int32_t CONTROL_NumMouseButtons = 0; static int32_t CONTROL_NumMouseAxes = 0; static int32_t CONTROL_NumJoyButtons = 0; static int32_t CONTROL_NumJoyAxes = 0; static controlflags CONTROL_Flags[CONTROL_NUM_FLAGS]; static controlbuttontype CONTROL_MouseButtonMapping[MAXMOUSEBUTTONS], CONTROL_JoyButtonMapping[MAXJOYBUTTONS]; //static controlkeymaptype CONTROL_KeyMapping[CONTROL_NUM_FLAGS]; static controlaxismaptype CONTROL_MouseAxesMap[MAXMOUSEAXES], // maps physical axes onto virtual ones CONTROL_JoyAxesMap[MAXJOYAXES]; static controlaxistype CONTROL_MouseAxes[MAXMOUSEAXES], // physical axes CONTROL_JoyAxes[MAXJOYAXES]; static controlaxistype CONTROL_LastMouseAxes[MAXMOUSEAXES], CONTROL_LastJoyAxes[MAXJOYAXES]; static int32_t CONTROL_MouseAxesScale[MAXMOUSEAXES], CONTROL_JoyAxesScale[MAXJOYAXES]; static int32_t CONTROL_MouseButtonState[MAXMOUSEBUTTONS], CONTROL_JoyButtonState[MAXJOYBUTTONS]; static int32_t CONTROL_MouseButtonClickedTime[MAXMOUSEBUTTONS], CONTROL_JoyButtonClickedTime[MAXJOYBUTTONS]; static int32_t CONTROL_MouseButtonClickedState[MAXMOUSEBUTTONS], CONTROL_JoyButtonClickedState[MAXJOYBUTTONS]; static int32_t CONTROL_MouseButtonClicked[MAXMOUSEBUTTONS], CONTROL_JoyButtonClicked[MAXJOYBUTTONS]; static uint8_t CONTROL_MouseButtonClickedCount[MAXMOUSEBUTTONS], CONTROL_JoyButtonClickedCount[MAXJOYBUTTONS]; static int32_t CONTROL_UserInputCleared[3]; static int32_t(*GetTime)(void); int32_t CONTROL_Started = FALSE; //static int32_t ticrate; static int32_t CONTROL_DoubleClickSpeed; int32_t extinput[CONTROL_NUM_FLAGS]; keybind KeyBindings[MAXBOUNDKEYS], MouseBindings[MAXMOUSEBUTTONS]; int32_t bindsenabled = 0; int32_t control_smoothmouse = 0; static void CONTROL_GetMouseDelta(void) { int32_t x,y; MOUSE_GetDelta(&x, &y); if (control_smoothmouse) { static int32_t lastx = 0, lasty = 0; CONTROL_MouseAxes[0].analog = (int32_t)(((x + lastx) / 2.0f) * 4.0f * CONTROL_MouseSensitivity); CONTROL_MouseAxes[1].analog = (int32_t)((((y + lasty) / 2.0f) * 4.0f * CONTROL_MouseSensitivity) * 2.0f); lastx = x; lasty = y; return; } CONTROL_MouseAxes[0].analog = (int32_t)(x * 4.0f * CONTROL_MouseSensitivity); CONTROL_MouseAxes[1].analog = (int32_t)((y * 4.0f * CONTROL_MouseSensitivity) * 2.0f); } static void CONTROL_GetJoyDelta(void) { int32_t i; for (i=0; i> 5; } static int32_t CONTROL_StartJoy(int32_t joy) { UNREFERENCED_PARAMETER(joy); return (inputdevices & 4) == 4; } static int32_t CONTROL_GetTime(void) { static int32_t t = 0; t += 5; return t; } static inline int32_t CONTROL_CheckRange(int32_t which) { if ((uint32_t)which >= (uint32_t)CONTROL_NUM_FLAGS) return TRUE; //Error("CONTROL_CheckRange: Index %d out of valid range for %d control flags.", // which, CONTROL_NUM_FLAGS); return FALSE; } static void CONTROL_SetFlag(int32_t which, int32_t active) { if (CONTROL_CheckRange(which)) return; if (CONTROL_Flags[which].toggle == INSTANT_ONOFF) { CONTROL_Flags[which].active = active; return; } if (active) { CONTROL_Flags[which].buttonheld = FALSE; } else if (CONTROL_Flags[which].buttonheld == FALSE) { CONTROL_Flags[which].buttonheld = TRUE; CONTROL_Flags[which].active = (CONTROL_Flags[which].active ? FALSE : TRUE); } } #if 0 int32_t CONTROL_KeyboardFunctionPressed(int32_t which) { int32_t key1 = 0, key2 = 0; if (CONTROL_CheckRange(which)) return FALSE; if (!CONTROL_Flags[which].used) return FALSE; if (CONTROL_KeyMapping[which].key1 != KEYUNDEFINED && !KeyBindings[CONTROL_KeyMapping[which].key1].cmd[0]) key1 = KB_KeyDown[ CONTROL_KeyMapping[which].key1 ] ? TRUE : FALSE; if (CONTROL_KeyMapping[which].key2 != KEYUNDEFINED && !KeyBindings[CONTROL_KeyMapping[which].key2].cmd[0]) key2 = KB_KeyDown[ CONTROL_KeyMapping[which].key2 ] ? TRUE : FALSE; return (key1 | key2); } void CONTROL_ClearKeyboardFunction(int32_t which) { if (CONTROL_CheckRange(which)) return; if (!CONTROL_Flags[which].used) return; if (CONTROL_KeyMapping[which].key1 != KEYUNDEFINED) KB_KeyDown[ CONTROL_KeyMapping[which].key1 ] = 0; if (CONTROL_KeyMapping[which].key2 != KEYUNDEFINED) KB_KeyDown[ CONTROL_KeyMapping[which].key2 ] = 0; } #endif void CONTROL_DefineFlag(int32_t which, int32_t toggle) { if (CONTROL_CheckRange(which)) return; CONTROL_Flags[which].active = FALSE; CONTROL_Flags[which].used = TRUE; CONTROL_Flags[which].toggle = toggle; CONTROL_Flags[which].buttonheld = FALSE; CONTROL_Flags[which].cleared = 0; } int32_t CONTROL_FlagActive(int32_t which) { if (CONTROL_CheckRange(which)) return FALSE; return CONTROL_Flags[which].used; } #if 0 void CONTROL_MapKey(int32_t which, kb_scancode key1, kb_scancode key2) { if (CONTROL_CheckRange(which)) return; CONTROL_KeyMapping[which].key1 = key1 ? key1 : KEYUNDEFINED; CONTROL_KeyMapping[which].key2 = key2 ? key2 : KEYUNDEFINED; } void CONTROL_PrintKeyMap(void) { int32_t i; for (i=0; i= (uint32_t)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 ((uint32_t)whichbutton >= (uint32_t)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(int32_t whichaxis, int32_t whichanalog, controldevice device) { controlaxismaptype *set; if ((uint32_t)whichanalog >= (uint32_t)analog_maxtype) { //Error("CONTROL_MapAnalogAxis: analog function %d out of valid range for %d analog functions.", // whichanalog, analog_maxtype); return; } switch (device) { case controldevice_mouse: if ((uint32_t)whichaxis >= (uint32_t)MAXMOUSEAXES) { //Error("CONTROL_MapAnalogAxis: axis %d out of valid range for %d mouse axes.", // whichaxis, MAXMOUSEAXES); return; } set = CONTROL_MouseAxesMap; break; case controldevice_joystick: if ((uint32_t)whichaxis >= (uint32_t)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 ((uint32_t)whichaxis >= (uint32_t)MAXMOUSEAXES) { //Error("CONTROL_SetAnalogAxisScale: axis %d out of valid range for %d mouse axes.", // whichaxis, MAXMOUSEAXES); return; } set = CONTROL_MouseAxesScale; break; case controldevice_joystick: if ((uint32_t)whichaxis >= (uint32_t)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_MapDigitalAxis(int32_t whichaxis, int32_t whichfunction, int32_t direction, controldevice device) { controlaxismaptype *set; if (CONTROL_CheckRange(whichfunction)) whichfunction = AXISUNDEFINED; switch (device) { case controldevice_mouse: if ((uint32_t)whichaxis >= (uint32_t)MAXMOUSEAXES) { //Error("CONTROL_MapDigitalAxis: axis %d out of valid range for %d mouse axes.", // whichaxis, MAXMOUSEAXES); return; } set = CONTROL_MouseAxesMap; break; case controldevice_joystick: if ((uint32_t)whichaxis >= (uint32_t)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) { int32_t i; memset(CONTROL_MouseButtonMapping, BUTTONUNDEFINED, sizeof(CONTROL_MouseButtonMapping)); memset(CONTROL_JoyButtonMapping, BUTTONUNDEFINED, sizeof(CONTROL_JoyButtonMapping)); // memset(CONTROL_KeyMapping, KEYUNDEFINED, sizeof(CONTROL_KeyMapping)); memset(CONTROL_MouseAxesMap, AXISUNDEFINED, sizeof(CONTROL_MouseAxesMap)); memset(CONTROL_JoyAxesMap, AXISUNDEFINED, sizeof(CONTROL_JoyAxesMap)); memset(CONTROL_MouseAxes, 0, sizeof(CONTROL_MouseAxes)); memset(CONTROL_JoyAxes, 0, sizeof(CONTROL_JoyAxes)); memset(CONTROL_LastMouseAxes, 0, sizeof(CONTROL_LastMouseAxes)); memset(CONTROL_LastJoyAxes, 0, sizeof(CONTROL_LastJoyAxes)); for (i=0; i=0; i--) { int32_t bs = (buttons >> i) & 1; DeviceButtonState[i] = bs; ButtonClickedState[i] = FALSE; if (bs) { 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; } } static void CONTROL_GetDeviceButtons(void) { int32_t t = GetTime(); if (CONTROL_MouseEnabled) { DoGetDeviceButtons( MOUSE_GetButtons(), t, CONTROL_NumMouseButtons, CONTROL_MouseButtonState, CONTROL_MouseButtonClickedTime, CONTROL_MouseButtonClickedState, CONTROL_MouseButtonClicked, CONTROL_MouseButtonClickedCount ); } if (CONTROL_JoystickEnabled) { int32_t buttons = JOYSTICK_GetButtons(); if (joynumhats > 0) { int32_t hat = JOYSTICK_GetHat(0); if (hat != 0) buttons |= hat << min(MAXJOYBUTTONS,joynumbuttons); } DoGetDeviceButtons( buttons, t, CONTROL_NumJoyButtons, CONTROL_JoyButtonState, CONTROL_JoyButtonClickedTime, CONTROL_JoyButtonClickedState, CONTROL_JoyButtonClicked, CONTROL_JoyButtonClickedCount ); } } static void CONTROL_DigitizeAxis(int32_t axis, controldevice device) { controlaxistype *set, *lastset; switch (device) { case controldevice_mouse: set = CONTROL_MouseAxes; lastset = CONTROL_LastMouseAxes; break; case controldevice_joystick: set = CONTROL_JoyAxes; lastset = CONTROL_LastJoyAxes; break; default: return; } if (set[axis].analog > 0) { if (set[axis].analog > THRESHOLD || (set[axis].analog > MINTHRESHOLD && lastset[axis].digital == 1)) set[axis].digital = 1; } else { if (set[axis].analog < -THRESHOLD || (set[axis].analog < -MINTHRESHOLD && lastset[axis].digital == -1)) set[axis].digital = -1; } } static void CONTROL_ScaleAxis(int32_t axis, controldevice device) { controlaxistype *set; int32_t *scale; switch (device) { case controldevice_mouse: set = CONTROL_MouseAxes; scale = CONTROL_MouseAxesScale; break; case controldevice_joystick: set = CONTROL_JoyAxes; scale = CONTROL_JoyAxesScale; break; default: return; } set[axis].analog = mulscale16(set[axis].analog, scale[axis]); } static void CONTROL_ApplyAxis(int32_t axis, ControlInfo *info, controldevice device) { controlaxistype *set; controlaxismaptype *map; switch (device) { case controldevice_mouse: set = CONTROL_MouseAxes; map = CONTROL_MouseAxesMap; break; 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) { Bmemcpy(CONTROL_LastMouseAxes, CONTROL_MouseAxes, sizeof(CONTROL_MouseAxes)); Bmemcpy(CONTROL_LastJoyAxes, CONTROL_JoyAxes, sizeof(CONTROL_JoyAxes)); memset(CONTROL_MouseAxes, 0, sizeof(CONTROL_MouseAxes)); memset(CONTROL_JoyAxes, 0, sizeof(CONTROL_JoyAxes)); memset(info, 0, sizeof(ControlInfo)); if (CONTROL_MouseEnabled) { int32_t i = MAXMOUSEAXES-1; CONTROL_GetMouseDelta(); for (; i>=0; i--) { CONTROL_DigitizeAxis(i, controldevice_mouse); CONTROL_ScaleAxis(i, controldevice_mouse); LIMITCONTROL(&CONTROL_MouseAxes[i].analog); CONTROL_ApplyAxis(i, info, controldevice_mouse); } } if (CONTROL_JoystickEnabled) { int32_t i = MAXJOYAXES-1; CONTROL_GetJoyDelta(); for (; i>=0; i--) { CONTROL_DigitizeAxis(i, controldevice_joystick); CONTROL_ScaleAxis(i, controldevice_joystick); LIMITCONTROL(&CONTROL_JoyAxes[i].analog); CONTROL_ApplyAxis(i, info, controldevice_joystick); } } CONTROL_GetDeviceButtons(); } static void CONTROL_AxisFunctionState(int32_t *p1) { if (CONTROL_NumMouseAxes) { int32_t j, i = CONTROL_NumMouseAxes-1; do { if (!CONTROL_MouseAxes[i].digital) continue; if (CONTROL_MouseAxes[i].digital < 0) j = CONTROL_MouseAxesMap[i].minmap; else j = CONTROL_MouseAxesMap[i].maxmap; if (j != AXISUNDEFINED) p1[j] = 1; } while (i--); } if (CONTROL_NumJoyAxes) { int32_t j, i = CONTROL_NumJoyAxes-1; do { if (!CONTROL_JoyAxes[i].digital) continue; if (CONTROL_JoyAxes[i].digital < 0) j = CONTROL_JoyAxesMap[i].minmap; else j = CONTROL_JoyAxesMap[i].maxmap; if (j != AXISUNDEFINED) p1[j] = 1; } while (i--); } } static void CONTROL_ButtonFunctionState(int32_t *p1) { if (CONTROL_NumMouseButtons) { int32_t i = CONTROL_NumMouseButtons-1, j; do { if (!MouseBindings[i].cmd[0]) { 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 (!bindsenabled) continue; if (MouseBindings[i].cmd[0] && CONTROL_MouseButtonState[i]) { if (MouseBindings[i].repeat || (MouseBindings[i].laststate == 0)) OSD_Dispatch(MouseBindings[i].cmd); } MouseBindings[i].laststate = CONTROL_MouseButtonState[i]; } while (i--); } if (CONTROL_NumJoyButtons) { int32_t i=CONTROL_NumJoyButtons-1, j; do { j = CONTROL_JoyButtonMapping[i].doubleclicked; if (j != KEYUNDEFINED) p1[j] |= CONTROL_JoyButtonClickedState[i]; j = CONTROL_JoyButtonMapping[i].singleclicked; if (j != KEYUNDEFINED) p1[j] |= CONTROL_JoyButtonState[i]; } while (i--); } } void CONTROL_ClearButton(int32_t whichbutton) { if (CONTROL_CheckRange(whichbutton)) return; BUTTONCLEAR(whichbutton); CONTROL_Flags[whichbutton].cleared = TRUE; } void CONTROL_ProcessBinds(void) { int32_t i=MAXBOUNDKEYS-1; if (!bindsenabled) return; do { if (KeyBindings[i].cmd[0]) { if (KB_KeyPressed(i) && (KeyBindings[i].repeat || (KeyBindings[i].laststate == 0))) { OSD_Dispatch(KeyBindings[i].cmd); } KeyBindings[i].laststate = KB_KeyPressed(i); } } while (i--); } static void CONTROL_GetFunctionInput(void) { int32_t periphs[CONTROL_NUM_FLAGS]; int32_t i = CONTROL_NUM_FLAGS-1; memset(periphs, 0, sizeof(periphs)); CONTROL_ButtonFunctionState(periphs); CONTROL_AxisFunctionState(periphs); CONTROL_ButtonHeldState = CONTROL_ButtonState; CONTROL_ButtonState = 0; do { CONTROL_SetFlag(i, /*CONTROL_KeyboardFunctionPressed(i) | */periphs[i] | extinput[i]); if (CONTROL_Flags[i].cleared == FALSE) BUTTONSET(i, CONTROL_Flags[i].active); else if (CONTROL_Flags[i].active == FALSE) CONTROL_Flags[i].cleared = 0; } while (i--); memset(extinput, 0, sizeof(extinput)); } void CONTROL_GetInput(ControlInfo *info) { CONTROL_PollDevices(info); CONTROL_GetFunctionInput(); } int32_t CONTROL_Startup(controltype which, int32_t(*TimeFunction)(void), int32_t ticspersecond) { int32_t i; UNREFERENCED_PARAMETER(which); if (CONTROL_Started) return FALSE; if (TimeFunction) GetTime = TimeFunction; else GetTime = CONTROL_GetTime; // ticrate = ticspersecond; CONTROL_DoubleClickSpeed = (ticspersecond*57)/100; if (CONTROL_DoubleClickSpeed <= 0) CONTROL_DoubleClickSpeed = 1; if (initinput()) return TRUE; CONTROL_MousePresent = CONTROL_MouseEnabled = FALSE; CONTROL_JoyPresent = CONTROL_JoystickEnabled = FALSE; CONTROL_NumMouseButtons = CONTROL_NumJoyButtons = 0; CONTROL_NumMouseAxes = CONTROL_NumJoyAxes = 0; KB_Startup(); //switch (which) { // case controltype_keyboard: // break; // case controltype_keyboardandmouse: CONTROL_NumMouseAxes = MAXMOUSEAXES; CONTROL_NumMouseButtons = MAXMOUSEBUTTONS; CONTROL_MousePresent = Mouse_Init(); CONTROL_MouseEnabled = CONTROL_MousePresent; // break; // case controltype_keyboardandjoystick: CONTROL_NumJoyAxes = min(MAXJOYAXES,joynumaxes); CONTROL_NumJoyButtons = min(MAXJOYBUTTONS,joynumbuttons + 4*(joynumhats>0)); CONTROL_JoyPresent = CONTROL_StartJoy(0); CONTROL_JoystickEnabled = CONTROL_JoyPresent; // break; //} #ifdef GEKKO if (CONTROL_MousePresent) initprintf("CONTROL_Startup: Mouse Present\n"); if (CONTROL_JoyPresent) initprintf("CONTROL_Startup: Joystick Present\n"); #endif CONTROL_ButtonState = 0; CONTROL_ButtonHeldState = 0; memset(CONTROL_UserInputCleared, 0, sizeof(CONTROL_UserInputCleared)); for (i=0; i