mirror of
https://github.com/ekeeke/Genesis-Plus-GX.git
synced 2025-01-26 09:55:27 +01:00
1994 lines
61 KiB
C
1994 lines
61 KiB
C
/****************************************************************************
|
|
* gx_input.c
|
|
*
|
|
* Genesis Plus GX input support
|
|
*
|
|
* Copyright Eke-Eke (2007-2022)
|
|
*
|
|
* Redistribution and use of this code or any derivative works are permitted
|
|
* provided that the following conditions are met:
|
|
*
|
|
* - Redistributions may not be sold, nor may they be used in a commercial
|
|
* product or activity.
|
|
*
|
|
* - Redistributions that are modified from the original source must include the
|
|
* complete source code, including the source code for all components used by a
|
|
* binary built from the modified sources. However, as a special exception, the
|
|
* source code distributed need not include anything that is normally distributed
|
|
* (in either source or binary form) with the major components (compiler, kernel,
|
|
* and so on) of the operating system on which the executable runs, unless that
|
|
* component itself accompanies the executable.
|
|
*
|
|
* - Redistributions must reproduce the above copyright notice, this list of
|
|
* conditions and the following disclaimer in the documentation and/or other
|
|
* materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
****************************************************************************************/
|
|
|
|
#include "shared.h"
|
|
#include "font.h"
|
|
#include "gui.h"
|
|
#include "menu.h"
|
|
#include "cheats.h"
|
|
|
|
#ifdef HW_RVL
|
|
#include <ogc/usbmouse.h>
|
|
#include "wiidrc.h"
|
|
#include "retrousb.h"
|
|
#endif
|
|
|
|
/* Analog sticks sensitivity */
|
|
#define ANALOG_SENSITIVITY 30
|
|
|
|
/* Delay before held keys triggering */
|
|
/* higher is the value, longer must the key remain held */
|
|
#define HELD_DELAY 30
|
|
|
|
/* Direction & selection update speed when a key is being held */
|
|
/* lower is the value, faster is the key update (min value = 1) */
|
|
#define HELD_SPEED 2
|
|
|
|
|
|
/* Configurable keys */
|
|
#define KEY_BUTTONA 0
|
|
#define KEY_BUTTONB 1
|
|
#define KEY_BUTTONC 2
|
|
#define KEY_START 3
|
|
#define KEY_BUTTONX 4
|
|
#define KEY_BUTTONY 5
|
|
#define KEY_BUTTONZ 6
|
|
#define KEY_MODE 7
|
|
#define KEY_MENU 8
|
|
|
|
#ifdef HW_RVL
|
|
|
|
#define PAD_UP 0
|
|
#define PAD_DOWN 1
|
|
#define PAD_LEFT 2
|
|
#define PAD_RIGHT 3
|
|
|
|
/* default directions mapping */
|
|
static u32 wpad_dirmap[5][4] =
|
|
{
|
|
{WPAD_BUTTON_RIGHT, WPAD_BUTTON_LEFT, WPAD_BUTTON_UP, WPAD_BUTTON_DOWN}, /* WIIMOTE */
|
|
{WPAD_BUTTON_UP, WPAD_BUTTON_DOWN, WPAD_BUTTON_LEFT, WPAD_BUTTON_RIGHT}, /* WIIMOTE + NUNCHUK */
|
|
{WPAD_CLASSIC_BUTTON_UP, WPAD_CLASSIC_BUTTON_DOWN, WPAD_CLASSIC_BUTTON_LEFT, WPAD_CLASSIC_BUTTON_RIGHT}, /* CLASSIC */
|
|
{WIIDRC_BUTTON_UP, WIIDRC_BUTTON_DOWN, WIIDRC_BUTTON_LEFT, WIIDRC_BUTTON_RIGHT}, /* WIIU GAMEPAD */
|
|
{PAD_BUTTON_UP, PAD_BUTTON_DOWN, PAD_BUTTON_LEFT, PAD_BUTTON_RIGHT}
|
|
};
|
|
|
|
#define WPAD_BUTTONS_HELD (WPAD_BUTTON_UP | WPAD_BUTTON_DOWN | WPAD_BUTTON_LEFT | WPAD_BUTTON_RIGHT | \
|
|
WPAD_BUTTON_MINUS | WPAD_BUTTON_PLUS | WPAD_BUTTON_A | WPAD_BUTTON_2 | \
|
|
WPAD_CLASSIC_BUTTON_UP | WPAD_CLASSIC_BUTTON_DOWN | WPAD_CLASSIC_BUTTON_LEFT | WPAD_CLASSIC_BUTTON_RIGHT | \
|
|
WPAD_CLASSIC_BUTTON_FULL_L | WPAD_CLASSIC_BUTTON_FULL_R | WPAD_CLASSIC_BUTTON_A)
|
|
|
|
#define WIIU_BUTTONS_HELD (WIIDRC_BUTTON_UP | WIIDRC_BUTTON_DOWN | WIIDRC_BUTTON_LEFT | WIIDRC_BUTTON_RIGHT | \
|
|
WIIDRC_BUTTON_MINUS | WIIDRC_BUTTON_PLUS | WIIDRC_BUTTON_A | \
|
|
WIIDRC_BUTTON_L | WIIDRC_BUTTON_ZL | WIIDRC_BUTTON_R | WIIDRC_BUTTON_ZR)
|
|
|
|
#endif
|
|
|
|
#define PAD_BUTTONS_HELD (PAD_BUTTON_UP | PAD_BUTTON_DOWN | PAD_BUTTON_LEFT | PAD_BUTTON_RIGHT | \
|
|
PAD_TRIGGER_L | PAD_TRIGGER_R | PAD_BUTTON_A)
|
|
|
|
static char keyname[MAX_KEYS][16];
|
|
|
|
static int held_cnt = 0;
|
|
static int inputs_disabled = 0;
|
|
|
|
/***************************************************************************************/
|
|
/* Gamecube PAD support */
|
|
/***************************************************************************************/
|
|
static void pad_config(int chan, int first_key, int last_key)
|
|
{
|
|
u16 p = 0;
|
|
char msg[64];
|
|
|
|
/* disable background PAD scanning */
|
|
inputs_disabled = 1;
|
|
|
|
/* Check if PAD is connected */
|
|
VIDEO_WaitVSync();
|
|
if (!(PAD_ScanPads() & (1<<chan)))
|
|
{
|
|
/* restore inputs update callback */
|
|
sprintf(msg, "PAD #%d is not connected !", chan+1);
|
|
GUI_WaitPrompt("Error",msg);
|
|
|
|
/* re-enable background PAD scanning and exit */
|
|
inputs_disabled = 0;
|
|
return;
|
|
}
|
|
|
|
/* Configure each keys */
|
|
do
|
|
{
|
|
/* ignore unused keys */
|
|
if (strcmp(keyname[first_key], "N.A"))
|
|
{
|
|
/* remove any pending keys */
|
|
while (PAD_ButtonsHeld(chan))
|
|
{
|
|
VIDEO_WaitVSync();
|
|
PAD_ScanPads();
|
|
}
|
|
|
|
/* configurable button */
|
|
sprintf(msg,"Press key for %s\n(D-PAD to return)",keyname[first_key]);
|
|
GUI_MsgBoxUpdate(0,msg);
|
|
|
|
/* wait for user input */
|
|
p = 0;
|
|
while (!p)
|
|
{
|
|
VIDEO_WaitVSync();
|
|
PAD_ScanPads();
|
|
p = PAD_ButtonsDown(chan);
|
|
}
|
|
|
|
/* find pressed key */
|
|
if (p & PAD_BUTTON_A) p = PAD_BUTTON_A;
|
|
else if (p & PAD_BUTTON_B) p = PAD_BUTTON_B;
|
|
else if (p & PAD_BUTTON_X) p = PAD_BUTTON_X;
|
|
else if (p & PAD_BUTTON_Y) p = PAD_BUTTON_Y;
|
|
else if (p & PAD_TRIGGER_R) p = PAD_TRIGGER_R;
|
|
else if (p & PAD_TRIGGER_L) p = PAD_TRIGGER_L;
|
|
else if (p & PAD_TRIGGER_Z) p = PAD_TRIGGER_Z;
|
|
else if (p & PAD_BUTTON_START) p = PAD_BUTTON_START;
|
|
else first_key = MAX_KEYS;
|
|
|
|
/* update key mapping */
|
|
if (first_key < MAX_KEYS)
|
|
{
|
|
config.pad_keymap[chan][first_key] = p;
|
|
}
|
|
}
|
|
}
|
|
while (first_key++ < last_key);
|
|
|
|
/* remove any pending keys */
|
|
while (PAD_ButtonsHeld(chan))
|
|
{
|
|
VIDEO_WaitVSync();
|
|
PAD_ScanPads();
|
|
}
|
|
|
|
/* Configurable menu key */
|
|
GUI_MsgBoxUpdate(0,"Press key(s) for MENU");
|
|
|
|
/* reset key combo */
|
|
config.pad_keymap[chan][KEY_MENU] = 0;
|
|
|
|
/* wait for user input */
|
|
p = 0;
|
|
while (!p)
|
|
{
|
|
/* update PAD status */
|
|
VIDEO_WaitVSync();
|
|
PAD_ScanPads();
|
|
p = PAD_ButtonsHeld(chan);
|
|
}
|
|
|
|
/* register keys until none are pressed anymore */
|
|
while (p)
|
|
{
|
|
/* update key combo */
|
|
config.pad_keymap[chan][KEY_MENU] |= p;
|
|
|
|
/* update PAD status */
|
|
VIDEO_WaitVSync();
|
|
PAD_ScanPads();
|
|
p = PAD_ButtonsHeld(chan);
|
|
}
|
|
|
|
/* re-enable background PAD scanning and exit */
|
|
inputs_disabled = 0;
|
|
}
|
|
|
|
static void pad_update(s8 chan, u8 i)
|
|
{
|
|
/* PAD status */
|
|
u16 p = PAD_ButtonsHeld(chan);
|
|
s8 x = PAD_StickX (chan);
|
|
s8 y = PAD_StickY (chan);
|
|
|
|
/* Retrieve current key mapping */
|
|
u16 *pad_keymap = config.pad_keymap[chan];
|
|
|
|
/* Default fast-forward key combo */
|
|
if ((p & PAD_TRIGGER_R) && (PAD_ButtonsDown(0) & PAD_BUTTON_START))
|
|
{
|
|
audioSync ^= AUDIO_WAIT;
|
|
videoSync = (audioSync && config.vsync && (gc_pal != vdp_pal)) ? VIDEO_WAIT : 0;
|
|
return;
|
|
}
|
|
|
|
/* User configurable menu combo */
|
|
if ((p & pad_keymap[KEY_MENU]) == pad_keymap[KEY_MENU])
|
|
{
|
|
ConfigRequested = 1;
|
|
return;
|
|
}
|
|
|
|
/* Default menu key (right analog stick if not needed by emulated device) */
|
|
if ((input.dev[i] < DEVICE_XE_1AP) && (PAD_SubStickX(chan) > ANALOG_SENSITIVITY))
|
|
{
|
|
ConfigRequested = 1;
|
|
return;
|
|
}
|
|
|
|
/* Emulated device */
|
|
switch (input.dev[i])
|
|
{
|
|
case DEVICE_PAD6B:
|
|
{
|
|
/* X,Y,Z,MODE buttons */
|
|
if (p & pad_keymap[KEY_BUTTONX]) input.pad[i] |= INPUT_X;
|
|
if (p & pad_keymap[KEY_BUTTONY]) input.pad[i] |= INPUT_Y;
|
|
if (p & pad_keymap[KEY_BUTTONZ]) input.pad[i] |= INPUT_Z;
|
|
if (p & pad_keymap[KEY_MODE]) input.pad[i] |= INPUT_MODE;
|
|
}
|
|
|
|
case DEVICE_PAD3B:
|
|
{
|
|
/* A button */
|
|
if (p & pad_keymap[KEY_BUTTONA]) input.pad[i] |= INPUT_A;
|
|
}
|
|
|
|
case DEVICE_PAD2B:
|
|
{
|
|
/* D-PAD */
|
|
if ((p & PAD_BUTTON_UP) || (y > ANALOG_SENSITIVITY)) input.pad[i] |= INPUT_UP;
|
|
else if ((p & PAD_BUTTON_DOWN) || (y < -ANALOG_SENSITIVITY)) input.pad[i] |= INPUT_DOWN;
|
|
if ((p & PAD_BUTTON_LEFT) || (x < -ANALOG_SENSITIVITY)) input.pad[i] |= INPUT_LEFT;
|
|
else if ((p & PAD_BUTTON_RIGHT) || (x > ANALOG_SENSITIVITY)) input.pad[i] |= INPUT_RIGHT;
|
|
|
|
/* default buttons */
|
|
if (p & pad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_B;
|
|
if (p & pad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_C;
|
|
if (p & pad_keymap[KEY_START]) input.pad[i] |= INPUT_START;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_XE_1AP:
|
|
{
|
|
/* Left Stick analog position [0-255] */
|
|
input.analog[i][0] = (x + 128);
|
|
input.analog[i][1] = y ? (127 - y) : (128 - y);
|
|
|
|
/* Right Stick analog position [0-255] */
|
|
x = PAD_SubStickX(chan);
|
|
y = PAD_SubStickY(chan);
|
|
|
|
/* Emulated stick is unidirectional but can be rotated */
|
|
if (abs(x) > abs(y))
|
|
{
|
|
input.analog[i+1][0] = (x + 128);
|
|
}
|
|
else
|
|
{
|
|
input.analog[i+1][0] = (y + 128);
|
|
}
|
|
|
|
/* Buttons */
|
|
if (p & pad_keymap[KEY_BUTTONA]) input.pad[i] |= INPUT_XE_A;
|
|
if (p & pad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_XE_B;
|
|
if (p & pad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_XE_C;
|
|
if (p & pad_keymap[KEY_START]) input.pad[i] |= INPUT_XE_START;
|
|
if (p & pad_keymap[KEY_BUTTONX]) input.pad[i] |= INPUT_XE_D;
|
|
if (p & pad_keymap[KEY_BUTTONY]) input.pad[i] |= INPUT_XE_E1;
|
|
if (p & pad_keymap[KEY_BUTTONZ]) input.pad[i] |= INPUT_XE_E2;
|
|
if (p & pad_keymap[KEY_MODE]) input.pad[i] |= INPUT_XE_SELECT;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_SPORTSPAD:
|
|
{
|
|
/* Y analog position [0-255] */
|
|
input.analog[i][1] = y ? (127 - y) : (128 - y);
|
|
}
|
|
|
|
case DEVICE_PADDLE:
|
|
{
|
|
/* X analog position [0-255] */
|
|
input.analog[i][0] = (x + 128);
|
|
|
|
/* Buttons */
|
|
if (p & pad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_BUTTON1;
|
|
if (p & pad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_BUTTON2;
|
|
if (p & pad_keymap[KEY_START]) input.pad[i] |= INPUT_START;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_LIGHTGUN:
|
|
{
|
|
/* Gun screen position (x,y) */
|
|
input.analog[i][0] += x / ANALOG_SENSITIVITY;
|
|
input.analog[i][1] -= y / ANALOG_SENSITIVITY;
|
|
|
|
/* Limits */
|
|
if (input.analog[i][0] < 0) input.analog[i][0] = 0;
|
|
else if (input.analog[i][0] > bitmap.viewport.w) input.analog[i][0] = bitmap.viewport.w;
|
|
if (input.analog[i][1] < 0) input.analog[i][1] = 0;
|
|
else if (input.analog[i][1] > bitmap.viewport.h) input.analog[i][1] = bitmap.viewport.h;
|
|
|
|
/* Buttons */
|
|
if (p & pad_keymap[KEY_BUTTONA]) input.pad[i] |= INPUT_A;
|
|
if (p & pad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_B;
|
|
if (p & pad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_C;
|
|
if (p & pad_keymap[KEY_START]) input.pad[i] |= INPUT_START;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_MOUSE:
|
|
{
|
|
/* Mouse relative movement (-255,255) */
|
|
input.analog[i][0] = (x / ANALOG_SENSITIVITY) * 2;
|
|
input.analog[i][1] = (y / ANALOG_SENSITIVITY) * 2;
|
|
|
|
/* Y-Axis inversion */
|
|
if (config.invert_mouse)
|
|
{
|
|
input.analog[i][1] = -input.analog[i][1];
|
|
}
|
|
|
|
/* Buttons */
|
|
if (p & pad_keymap[KEY_BUTTONA]) input.pad[i] |= INPUT_MOUSE_CENTER;
|
|
if (p & pad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_MOUSE_LEFT;
|
|
if (p & pad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_MOUSE_RIGHT;
|
|
if (p & pad_keymap[KEY_START]) input.pad[i] |= INPUT_START;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_PICO:
|
|
{
|
|
/* D-PAD */
|
|
if (p & PAD_BUTTON_UP) input.pad[0] |= INPUT_UP;
|
|
else if (p & PAD_BUTTON_DOWN) input.pad[0] |= INPUT_DOWN;
|
|
if (p & PAD_BUTTON_LEFT) input.pad[0] |= INPUT_LEFT;
|
|
else if (p & PAD_BUTTON_RIGHT) input.pad[0] |= INPUT_RIGHT;
|
|
|
|
/* PEN screen position (x,y) */
|
|
input.analog[0][0] += x / ANALOG_SENSITIVITY;
|
|
input.analog[0][1] -= y / ANALOG_SENSITIVITY;
|
|
|
|
/* Limits */
|
|
if (input.analog[0][0] > 0x17c) input.analog[0][0] = 0x17c;
|
|
else if (input.analog[0][0] < 0x3c) input.analog[0][0] = 0x3c;
|
|
if (input.analog[0][1] < 0x1fc) input.analog[0][1] = 0x1fc;
|
|
else if (input.analog[0][1] > 0x2f7) input.analog[0][1] = 0x2f7;
|
|
|
|
/* PEN & RED button */
|
|
if (p & pad_keymap[KEY_BUTTONA]) input.pad[0] |= INPUT_PICO_RED;
|
|
if (p & pad_keymap[KEY_BUTTONB]) input.pad[0] |= INPUT_PICO_PEN;
|
|
|
|
/* PAGE index increment */
|
|
if (p & pad_keymap[KEY_BUTTONC]) pico_current = (pico_current + 1) & 7;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_TEREBI:
|
|
{
|
|
/* PEN screen position (x,y) */
|
|
input.analog[0][0] += x / ANALOG_SENSITIVITY;
|
|
input.analog[0][1] -= y / ANALOG_SENSITIVITY;
|
|
|
|
/* Limits */
|
|
if (input.analog[0][0] < 0) input.analog[0][0] = 0;
|
|
else if (input.analog[0][0] > 250) input.analog[0][0] = 250;
|
|
if (input.analog[0][1] < 0) input.analog[0][1] = 0;
|
|
else if (input.analog[0][1] > 250) input.analog[0][1] = 250;
|
|
|
|
/* PEN button */
|
|
if (p & pad_keymap[KEY_BUTTONA]) input.pad[0] |= INPUT_BUTTON1;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_GRAPHIC_BOARD:
|
|
{
|
|
/* PEN screen position (x,y) */
|
|
input.analog[0][0] += x / ANALOG_SENSITIVITY;
|
|
input.analog[0][1] -= y / ANALOG_SENSITIVITY;
|
|
|
|
/* Limits */
|
|
if (input.analog[0][0] < 0) input.analog[0][0] = 0;
|
|
else if (input.analog[0][0] > 255) input.analog[0][0] = 255;
|
|
if (input.analog[0][1] < 0) input.analog[0][1] = 0;
|
|
else if (input.analog[0][1] > 255) input.analog[0][1] = 255;
|
|
|
|
/* MODE buttons */
|
|
if (p & pad_keymap[KEY_BUTTONA]) input.pad[0] |= INPUT_GRAPHIC_PEN;
|
|
if (p & pad_keymap[KEY_BUTTONB]) input.pad[0] |= INPUT_GRAPHIC_DO;
|
|
if (p & pad_keymap[KEY_BUTTONC]) input.pad[0] |= INPUT_GRAPHIC_MENU;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_ACTIVATOR:
|
|
{
|
|
/* Left & right analog stick angle [0-360] */
|
|
float ang;
|
|
|
|
/* Left stick values */
|
|
if ((abs(x) > ANALOG_SENSITIVITY) || (abs(y) > ANALOG_SENSITIVITY))
|
|
{
|
|
/* Calculate angle (in degree) */
|
|
ang = 90.0 - (atan((float)y / (float)x) * 180.0 / M_PI);
|
|
if (x < 0) ang += 180.0;
|
|
|
|
/* 8 bottom sensors = 8 areas */
|
|
if ((ang > 22.5) && (ang <= 67.5)) input.pad[i] |= INPUT_ACTIVATOR_2L;
|
|
else if ((ang > 67.5) && (ang <= 112.5)) input.pad[i] |= INPUT_ACTIVATOR_3L;
|
|
else if ((ang > 112.5) && (ang <= 157.5)) input.pad[i] |= INPUT_ACTIVATOR_4L;
|
|
else if ((ang > 157.5) && (ang <= 202.5)) input.pad[i] |= INPUT_ACTIVATOR_5L;
|
|
else if ((ang > 202.5) && (ang <= 247.5)) input.pad[i] |= INPUT_ACTIVATOR_6L;
|
|
else if ((ang > 247.5) && (ang <= 292.5)) input.pad[i] |= INPUT_ACTIVATOR_7L;
|
|
else if ((ang > 292.5) && (ang <= 337.5)) input.pad[i] |= INPUT_ACTIVATOR_8L;
|
|
else input.pad[i] |= INPUT_ACTIVATOR_1L;
|
|
}
|
|
|
|
/* Right stick values */
|
|
x = PAD_SubStickX(chan);
|
|
y = PAD_SubStickY(chan);
|
|
|
|
if ((abs(x) > ANALOG_SENSITIVITY) || (abs(y) > ANALOG_SENSITIVITY))
|
|
{
|
|
/* Calculate angle (in degree) */
|
|
ang = 90.0 - (atan((float)y / (float)x) * 180.0 / M_PI);
|
|
if (x < 0) ang += 180.0;
|
|
|
|
/* 8 top sensors = 8 areas */
|
|
if ((ang > 22.5) && (ang <= 67.5)) input.pad[i] |= INPUT_ACTIVATOR_2U;
|
|
else if ((ang > 67.5) && (ang <= 112.5)) input.pad[i] |= INPUT_ACTIVATOR_3U;
|
|
else if ((ang > 112.5) && (ang <= 157.5)) input.pad[i] |= INPUT_ACTIVATOR_4U;
|
|
else if ((ang > 157.5) && (ang <= 202.5)) input.pad[i] |= INPUT_ACTIVATOR_5U;
|
|
else if ((ang > 202.5) && (ang <= 247.5)) input.pad[i] |= INPUT_ACTIVATOR_6U;
|
|
else if ((ang > 247.5) && (ang <= 292.5)) input.pad[i] |= INPUT_ACTIVATOR_7U;
|
|
else if ((ang > 292.5) && (ang <= 337.5)) input.pad[i] |= INPUT_ACTIVATOR_8U;
|
|
else input.pad[i] |= INPUT_ACTIVATOR_1U;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/***************************************************************************************/
|
|
/* Wii WPAD support */
|
|
/***************************************************************************************/
|
|
#ifdef HW_RVL
|
|
|
|
static int wpad_StickX(WPADData *data, u8 right)
|
|
{
|
|
struct joystick_t* js = NULL;
|
|
|
|
switch (data->exp.type)
|
|
{
|
|
case WPAD_EXP_NUNCHUK:
|
|
js = right ? NULL : &data->exp.nunchuk.js;
|
|
break;
|
|
|
|
case WPAD_EXP_CLASSIC:
|
|
js = right ? &data->exp.classic.rjs : &data->exp.classic.ljs;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (js)
|
|
{
|
|
/* raw X position */
|
|
int pos = js->pos.x;
|
|
|
|
/* X range calibration */
|
|
int min = js->min.x;
|
|
int max = js->max.x;
|
|
int center = js->center.x;
|
|
|
|
/* some 3rd party controllers return invalid analog sticks calibration data */
|
|
if ((min >= center) || (max <= center))
|
|
{
|
|
/* force default calibration settings */
|
|
min = js->min.x = 0;
|
|
max = js->max.x = right ? 32 : 64;
|
|
center = js->center.x = right ? 16 : 32;
|
|
}
|
|
|
|
/* value returned could be above calibration limits */
|
|
if (pos > max) return 127;
|
|
if (pos < min) return -128;
|
|
|
|
/* adjust against center position */
|
|
pos -= center;
|
|
|
|
/* return interpolated range [-128;127] */
|
|
if (pos > 0)
|
|
{
|
|
return (int)(127.0 * ((float)pos / (float)(max - center)));
|
|
}
|
|
else
|
|
{
|
|
return (int)(128.0 * ((float)pos / (float)(center - min)));
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int wpad_StickY(WPADData *data, u8 right)
|
|
{
|
|
struct joystick_t* js = NULL;
|
|
|
|
switch (data->exp.type)
|
|
{
|
|
case WPAD_EXP_NUNCHUK:
|
|
js = right ? NULL : &data->exp.nunchuk.js;
|
|
break;
|
|
|
|
case WPAD_EXP_CLASSIC:
|
|
js = right ? &data->exp.classic.rjs : &data->exp.classic.ljs;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (js)
|
|
{
|
|
/* raw Y position */
|
|
int pos = js->pos.y;
|
|
|
|
/* Y range calibration */
|
|
int min = js->min.y;
|
|
int max = js->max.y;
|
|
int center = js->center.y;
|
|
|
|
/* some 3rd party controllers return invalid analog sticks calibration data */
|
|
if ((min >= center) || (max <= center))
|
|
{
|
|
/* force default calibration settings */
|
|
min = js->min.y = 0;
|
|
max = js->max.y = right ? 32 : 64;
|
|
center = js->center.y = right ? 16 : 32;
|
|
}
|
|
|
|
/* value returned could be above calibration limits */
|
|
if (pos > max) return 127;
|
|
if (pos < min) return -128;
|
|
|
|
/* adjust against center position */
|
|
pos -= center;
|
|
|
|
/* return interpolated range [-128;127] */
|
|
if (pos > 0)
|
|
{
|
|
return (int)(127.0 * ((float)pos / (float)(max - center)));
|
|
}
|
|
else
|
|
{
|
|
return (int)(128.0 * ((float)pos / (float)(center - min)));
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void wpad_config(u8 exp, int chan, int first_key, int last_key)
|
|
{
|
|
char msg[64];
|
|
u32 p = 255;
|
|
bool isRetroUSB = false;
|
|
|
|
/* Disable background PAD scanning */
|
|
inputs_disabled = 1;
|
|
|
|
/* Check if device is connected */
|
|
if (exp <= WPAD_EXP_CLASSIC)
|
|
{
|
|
WPAD_Probe(chan, &p);
|
|
}
|
|
else if (WiiDRC_Inited() && WiiDRC_Connected())
|
|
{
|
|
p = exp;
|
|
}
|
|
else if (RetroUSB_OK())
|
|
{
|
|
p = exp;
|
|
isRetroUSB = true;
|
|
}
|
|
|
|
/* Device not detected */
|
|
if (((exp > WPAD_EXP_NONE) && (p != exp)) || (p == 255))
|
|
{
|
|
if (exp == WPAD_EXP_NONE) sprintf(msg, "WIIMOTE #%d is not connected !", chan+1);
|
|
else if (exp == WPAD_EXP_NUNCHUK) sprintf(msg, "NUNCHUK #%d is not connected !", chan+1);
|
|
else if (exp == WPAD_EXP_CLASSIC) sprintf(msg, "CLASSIC #%d is not connected !", chan+1);
|
|
else sprintf(msg, "WIIU GAMEPAD or RETROUSB is not connected !");
|
|
GUI_WaitPrompt("Error",msg);
|
|
|
|
/* re-enable background PAD scanning and exit */
|
|
inputs_disabled = 0;
|
|
return;
|
|
}
|
|
|
|
/* Configure each keys */
|
|
do
|
|
{
|
|
/* ignore unused keys */
|
|
if (strcmp(keyname[first_key], "N.A"))
|
|
{
|
|
/* remove any pending buttons */
|
|
if (exp <= WPAD_EXP_CLASSIC)
|
|
{
|
|
while (WPAD_ButtonsHeld(chan))
|
|
{
|
|
VIDEO_WaitVSync();
|
|
WPAD_ScanPads();
|
|
}
|
|
}
|
|
else if (isRetroUSB)
|
|
{
|
|
while (RetroUSB_ButtonsHeld())
|
|
{
|
|
VIDEO_WaitVSync();
|
|
RetroUSB_ScanPads();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while (WiiDRC_ButtonsHeld())
|
|
{
|
|
VIDEO_WaitVSync();
|
|
WiiDRC_ScanPads();
|
|
}
|
|
}
|
|
|
|
/* configurable button */
|
|
sprintf(msg,"Press key for %s\n(HOME to return)",keyname[first_key]);
|
|
GUI_MsgBoxUpdate(0, msg);
|
|
|
|
/* wait for user input */
|
|
p = 0;
|
|
while (!p)
|
|
{
|
|
VIDEO_WaitVSync();
|
|
if (exp <= WPAD_EXP_CLASSIC)
|
|
{
|
|
WPAD_ScanPads();
|
|
p = WPAD_ButtonsDown(chan);
|
|
}
|
|
else if (isRetroUSB)
|
|
{
|
|
RetroUSB_ScanPads();
|
|
p = RetroUSB_ButtonsHeld();
|
|
}
|
|
else
|
|
{
|
|
WiiDRC_ScanPads();
|
|
p = WiiDRC_ButtonsDown();
|
|
}
|
|
}
|
|
|
|
/* detect pressed key */
|
|
switch (exp)
|
|
{
|
|
/* Wiimote (TODO: add motion sensing !) */
|
|
case WPAD_EXP_NONE:
|
|
{
|
|
if (p & WPAD_BUTTON_2) p = WPAD_BUTTON_2;
|
|
else if (p & WPAD_BUTTON_1) p = WPAD_BUTTON_1;
|
|
else if (p & WPAD_BUTTON_B) p = WPAD_BUTTON_B;
|
|
else if (p & WPAD_BUTTON_A) p = WPAD_BUTTON_A;
|
|
else if (p & WPAD_BUTTON_PLUS) p = WPAD_BUTTON_PLUS;
|
|
else if (p & WPAD_BUTTON_MINUS) p = WPAD_BUTTON_MINUS;
|
|
else first_key = MAX_KEYS;
|
|
break;
|
|
}
|
|
|
|
/* Wiimote + Nunchuk (TODO: add motion sensing !) */
|
|
case WPAD_EXP_NUNCHUK:
|
|
{
|
|
if (p & WPAD_BUTTON_2) p = WPAD_BUTTON_2;
|
|
else if (p & WPAD_BUTTON_1) p = WPAD_BUTTON_1;
|
|
else if (p & WPAD_BUTTON_B) p = WPAD_BUTTON_B;
|
|
else if (p & WPAD_BUTTON_A) p = WPAD_BUTTON_A;
|
|
else if (p & WPAD_BUTTON_PLUS) p = WPAD_BUTTON_PLUS;
|
|
else if (p & WPAD_BUTTON_MINUS) p= WPAD_BUTTON_MINUS;
|
|
else if (p & WPAD_NUNCHUK_BUTTON_Z) p = WPAD_NUNCHUK_BUTTON_Z;
|
|
else if (p & WPAD_NUNCHUK_BUTTON_C) p = WPAD_NUNCHUK_BUTTON_C;
|
|
else first_key = MAX_KEYS;
|
|
break;
|
|
}
|
|
|
|
/* Classic Controller */
|
|
case WPAD_EXP_CLASSIC:
|
|
{
|
|
if (p & WPAD_CLASSIC_BUTTON_X) p = WPAD_CLASSIC_BUTTON_X;
|
|
else if (p & WPAD_CLASSIC_BUTTON_A) p = WPAD_CLASSIC_BUTTON_A;
|
|
else if (p & WPAD_CLASSIC_BUTTON_Y) p = WPAD_CLASSIC_BUTTON_Y;
|
|
else if (p & WPAD_CLASSIC_BUTTON_B) p = WPAD_CLASSIC_BUTTON_B;
|
|
else if (p & WPAD_CLASSIC_BUTTON_ZL) p = WPAD_CLASSIC_BUTTON_ZL;
|
|
else if (p & WPAD_CLASSIC_BUTTON_ZR) p = WPAD_CLASSIC_BUTTON_ZR;
|
|
else if (p & WPAD_CLASSIC_BUTTON_PLUS) p = WPAD_CLASSIC_BUTTON_PLUS;
|
|
else if (p & WPAD_CLASSIC_BUTTON_MINUS) p = WPAD_CLASSIC_BUTTON_MINUS;
|
|
else if (p & WPAD_CLASSIC_BUTTON_FULL_L) p = WPAD_CLASSIC_BUTTON_FULL_L;
|
|
else if (p & WPAD_CLASSIC_BUTTON_FULL_R) p = WPAD_CLASSIC_BUTTON_FULL_R;
|
|
else first_key = MAX_KEYS;
|
|
break;
|
|
}
|
|
|
|
/* WiiU GamePad Controller */
|
|
default:
|
|
{
|
|
if (isRetroUSB) {
|
|
if (p & PAD_BUTTON_A) p = PAD_BUTTON_A;
|
|
else if (p & PAD_BUTTON_B) p = PAD_BUTTON_B;
|
|
else if (p & PAD_BUTTON_X) p = PAD_BUTTON_X;
|
|
else if (p & PAD_BUTTON_Y) p = PAD_BUTTON_Y;
|
|
else if (p & PAD_TRIGGER_Z) p = PAD_TRIGGER_Z;
|
|
else if (p & PAD_BUTTON_START) p = PAD_BUTTON_START;
|
|
else if (p & PAD_TRIGGER_L) p = PAD_TRIGGER_L;
|
|
else if (p & PAD_TRIGGER_R) p = PAD_TRIGGER_R;
|
|
else first_key = MAX_KEYS;
|
|
}
|
|
else {
|
|
if (p & WIIDRC_BUTTON_A) p = WIIDRC_BUTTON_A;
|
|
else if (p & WIIDRC_BUTTON_B) p = WIIDRC_BUTTON_B;
|
|
else if (p & WIIDRC_BUTTON_X) p = WIIDRC_BUTTON_X;
|
|
else if (p & WIIDRC_BUTTON_Y) p = WIIDRC_BUTTON_Y;
|
|
else if (p & WIIDRC_BUTTON_ZL) p = WIIDRC_BUTTON_ZL;
|
|
else if (p & WIIDRC_BUTTON_ZR) p = WIIDRC_BUTTON_ZR;
|
|
else if (p & WIIDRC_BUTTON_PLUS) p = WIIDRC_BUTTON_PLUS;
|
|
else if (p & WIIDRC_BUTTON_MINUS) p = WIIDRC_BUTTON_MINUS;
|
|
else if (p & WIIDRC_BUTTON_L) p = WIIDRC_BUTTON_L;
|
|
else if (p & WIIDRC_BUTTON_R) p = WIIDRC_BUTTON_R;
|
|
else first_key = MAX_KEYS;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* update key mapping */
|
|
if (first_key < MAX_KEYS)
|
|
{
|
|
config.wpad_keymap[4*exp + chan][first_key] = p;
|
|
}
|
|
|
|
if (isRetroUSB)
|
|
{
|
|
while (RetroUSB_ButtonsHeld())
|
|
{
|
|
VIDEO_WaitVSync();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
while (first_key++ < last_key);
|
|
|
|
/* remove any pending buttons */
|
|
if (exp <= WPAD_EXP_CLASSIC)
|
|
{
|
|
while (WPAD_ButtonsHeld(chan))
|
|
{
|
|
VIDEO_WaitVSync();
|
|
WPAD_ScanPads();
|
|
}
|
|
}
|
|
else if (isRetroUSB)
|
|
{
|
|
while (RetroUSB_ButtonsHeld())
|
|
{
|
|
VIDEO_WaitVSync();
|
|
RetroUSB_ScanPads();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while (WiiDRC_ButtonsHeld())
|
|
{
|
|
VIDEO_WaitVSync();
|
|
WiiDRC_ScanPads();
|
|
}
|
|
}
|
|
|
|
if (!isRetroUSB)
|
|
{
|
|
/* Configurable menu key */
|
|
GUI_MsgBoxUpdate(0, "Press key(s) for MENU");
|
|
|
|
/* reset key combo */
|
|
config.wpad_keymap[4 * exp + chan][KEY_MENU] = 0;
|
|
|
|
/* wait for user input */
|
|
p = 0;
|
|
while (!p)
|
|
{
|
|
VIDEO_WaitVSync();
|
|
if (exp <= WPAD_EXP_CLASSIC)
|
|
{
|
|
WPAD_ScanPads();
|
|
p = WPAD_ButtonsHeld(chan);
|
|
}
|
|
else
|
|
{
|
|
WiiDRC_ScanPads();
|
|
p = WiiDRC_ButtonsHeld();
|
|
}
|
|
}
|
|
|
|
/* register keys until none are pressed anymore */
|
|
while (p)
|
|
{
|
|
/* update key combo */
|
|
config.wpad_keymap[4 * exp + chan][KEY_MENU] |= p;
|
|
|
|
/* update WPAD status */
|
|
VIDEO_WaitVSync();
|
|
if (exp <= WPAD_EXP_CLASSIC)
|
|
{
|
|
WPAD_ScanPads();
|
|
p = WPAD_ButtonsHeld(chan);
|
|
}
|
|
else
|
|
{
|
|
WiiDRC_ScanPads();
|
|
p = WiiDRC_ButtonsHeld();
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
config.wpad_keymap[4 * exp + chan][KEY_MENU] = 0xFFFF;
|
|
}
|
|
/* re-enable background WPAD scanning and exit */
|
|
inputs_disabled = 0;
|
|
}
|
|
|
|
static void wpad_update(s8 chan, u8 i, u32 exp)
|
|
{
|
|
/* WPAD data */
|
|
WPADData *data = WPAD_Data(chan);
|
|
|
|
/* Button status */
|
|
u32 p = data->btns_h;
|
|
|
|
/* Analog sticks */
|
|
s16 x = 0;
|
|
s16 y = 0;
|
|
|
|
/* WiiU GamePad Controller support */
|
|
if (exp > WPAD_EXP_CLASSIC)
|
|
{
|
|
if (RetroUSB_OK())
|
|
{
|
|
p = RetroUSB_ButtonsHeld();
|
|
}
|
|
else
|
|
{
|
|
WiiDRC_ScanPads();
|
|
if (WiiDRC_ShutdownRequested())
|
|
{
|
|
Shutdown = ConfigRequested = 1;
|
|
reload = 0;
|
|
return;
|
|
}
|
|
|
|
p = WiiDRC_ButtonsHeld();
|
|
|
|
/* Default fast-forward key combo */
|
|
if (WiiDRC_ButtonsDown() & WIIDRC_BUTTON_HOME)
|
|
{
|
|
if (p & WIIDRC_BUTTON_MINUS)
|
|
{
|
|
audioSync ^= AUDIO_WAIT;
|
|
videoSync = (audioSync && config.vsync && (gc_pal != vdp_pal)) ? VIDEO_WAIT : 0;
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* Left Analog Stick */
|
|
x = (WiiDRC_lStickX() * 128) / 75;
|
|
y = (WiiDRC_lStickY() * 128) / 75;
|
|
if (x > 127) x = 127;
|
|
else if (x < -128) x = -128;
|
|
if (y > 127) y = 127;
|
|
else if (y < -128) y = -128;
|
|
}
|
|
}
|
|
else if (exp != WPAD_EXP_NONE)
|
|
{
|
|
/* Left Analog Stick */
|
|
x = wpad_StickX(data,0);
|
|
y = wpad_StickY(data,0);
|
|
}
|
|
|
|
/* Retrieve current key mapping */
|
|
u32 *wpad_keymap = config.wpad_keymap[4*exp + chan];
|
|
|
|
/* User configurable menu combo */
|
|
if ((p & wpad_keymap[KEY_MENU]) == wpad_keymap[KEY_MENU])
|
|
{
|
|
ConfigRequested = 1;
|
|
return;
|
|
}
|
|
|
|
/* Emulated device */
|
|
switch (input.dev[i])
|
|
{
|
|
case DEVICE_PAD6B:
|
|
{
|
|
/* X,Y,Z,MODE buttons */
|
|
if (p & wpad_keymap[KEY_BUTTONX]) input.pad[i] |= INPUT_X;
|
|
if (p & wpad_keymap[KEY_BUTTONY]) input.pad[i] |= INPUT_Y;
|
|
if (p & wpad_keymap[KEY_BUTTONZ]) input.pad[i] |= INPUT_Z;
|
|
if (p & wpad_keymap[KEY_MODE]) input.pad[i] |= INPUT_MODE;
|
|
}
|
|
|
|
case DEVICE_PAD3B:
|
|
{
|
|
/* A button */
|
|
if (p & wpad_keymap[KEY_BUTTONA]) input.pad[i] |= INPUT_A;
|
|
}
|
|
|
|
case DEVICE_PAD2B:
|
|
{
|
|
/* D-PAD */
|
|
if ((p & wpad_dirmap[exp][PAD_UP]) || (y > ANALOG_SENSITIVITY)) input.pad[i] |= INPUT_UP;
|
|
else if ((p & wpad_dirmap[exp][PAD_DOWN]) || (y < -ANALOG_SENSITIVITY)) input.pad[i] |= INPUT_DOWN;
|
|
if ((p & wpad_dirmap[exp][PAD_LEFT]) || (x < -ANALOG_SENSITIVITY)) input.pad[i] |= INPUT_LEFT;
|
|
else if ((p & wpad_dirmap[exp][PAD_RIGHT]) || (x > ANALOG_SENSITIVITY)) input.pad[i] |= INPUT_RIGHT;
|
|
|
|
/* default buttons */
|
|
if (p & wpad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_B;
|
|
if (p & wpad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_C;
|
|
if (p & wpad_keymap[KEY_START]) input.pad[i] |= INPUT_START;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_XE_1AP:
|
|
{
|
|
/* Left Stick analog position [0-255] */
|
|
input.analog[i][0] = (x + 128);
|
|
input.analog[i][1] = y ? (127 - y) : 128;
|
|
|
|
/* Right Stick analog position [0-255] */
|
|
if (exp >= WPAD_EXP_CLASSIC && !RetroUSB_OK())
|
|
{
|
|
if (exp > WPAD_EXP_CLASSIC)
|
|
{
|
|
/* WiiU GamePad Controller right stick */
|
|
x = (WiiDRC_rStickX() * 128) / 75;
|
|
y = (WiiDRC_rStickY() * 128) / 75;
|
|
if (x > 127) x = 127;
|
|
else if (x < -128) x = -128;
|
|
if (y > 127) y = 127;
|
|
else if (y < -128) y = -128;
|
|
}
|
|
else
|
|
{
|
|
/* Classic Controller right stick */
|
|
x = wpad_StickX(data,1);
|
|
y = wpad_StickY(data,1);
|
|
}
|
|
|
|
/* Emulated stick is unidirectional but can be rotated */
|
|
if (abs(x) > abs(y))
|
|
{
|
|
input.analog[i+1][0] = (x + 128);
|
|
}
|
|
else
|
|
{
|
|
input.analog[i+1][0] = (y + 128);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Wiimote D-PAD */
|
|
if ((p & wpad_dirmap[exp][PAD_DOWN]) || (p & wpad_dirmap[exp][PAD_LEFT])) input.analog[i+1][0]-=2;
|
|
else if ((p & wpad_dirmap[exp][PAD_UP]) || (p & wpad_dirmap[exp][PAD_RIGHT])) input.analog[i+1][0]+=2;
|
|
|
|
/* Limits */
|
|
if (input.analog[i+1][0] < 0) input.analog[i+1][0] = 0;
|
|
else if (input.analog[i+1][0] > 255) input.analog[i+1][0] = 255;
|
|
}
|
|
|
|
/* Buttons */
|
|
if (p & wpad_keymap[KEY_BUTTONA]) input.pad[i] |= INPUT_XE_A;
|
|
if (p & wpad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_XE_B;
|
|
if (p & wpad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_XE_C;
|
|
if (p & wpad_keymap[KEY_START]) input.pad[i] |= INPUT_XE_START;
|
|
if (p & wpad_keymap[KEY_BUTTONX]) input.pad[i] |= INPUT_XE_D;
|
|
if (p & wpad_keymap[KEY_BUTTONY]) input.pad[i] |= INPUT_XE_E1;
|
|
if (p & wpad_keymap[KEY_BUTTONZ]) input.pad[i] |= INPUT_XE_E2;
|
|
if (p & wpad_keymap[KEY_MODE]) input.pad[i] |= INPUT_XE_SELECT;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_SPORTSPAD:
|
|
{
|
|
/* X analog position [0-255] */
|
|
if (p & wpad_dirmap[exp][PAD_LEFT]) input.analog[i][0]-=2;
|
|
else if (p & wpad_dirmap[exp][PAD_RIGHT]) input.analog[i][0]+=2;
|
|
else input.analog[i][0] = (x + 128);
|
|
|
|
/* Y analog position [0-255] */
|
|
if (p & wpad_dirmap[exp][PAD_UP]) input.analog[i][1]-=2;
|
|
else if (p & wpad_dirmap[exp][PAD_DOWN]) input.analog[i][1]+=2;
|
|
else input.analog[i][1] = y ? (127 - y) : (128 - y);
|
|
|
|
/* Limits */
|
|
if (input.analog[i][0] < 0) input.analog[i][0] = 0;
|
|
else if (input.analog[i][0] > 255) input.analog[i][0] = 255;
|
|
if (input.analog[i][1] < 0) input.analog[i][1] = 0;
|
|
else if (input.analog[i][1] > 255) input.analog[i][1] = 255;
|
|
|
|
/* Buttons */
|
|
if (p & wpad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_BUTTON1;
|
|
if (p & wpad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_BUTTON2;
|
|
if (p & wpad_keymap[KEY_START]) input.pad[i] |= INPUT_START;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_PADDLE:
|
|
{
|
|
/* X analog position [0-255] */
|
|
if (exp == WPAD_EXP_NONE)
|
|
{
|
|
/* Wiimote D-PAD */
|
|
if (p & wpad_dirmap[exp][PAD_LEFT]) input.analog[i][0]-=2;
|
|
else if (p & wpad_dirmap[exp][PAD_RIGHT]) input.analog[i][0]+=2;
|
|
|
|
/* Limits */
|
|
if (input.analog[i][0] < 0) input.analog[i][0] = 0;
|
|
else if (input.analog[i][0] > 255) input.analog[i][0] = 255;
|
|
}
|
|
else
|
|
{
|
|
/* Left analog stick */
|
|
input.analog[i][0] = (x + 128);
|
|
}
|
|
|
|
/* Buttons */
|
|
if (p & wpad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_BUTTON1;
|
|
if (p & wpad_keymap[KEY_START]) input.pad[i] |= INPUT_START;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_LIGHTGUN:
|
|
{
|
|
/* Gun screen position (x,y) */
|
|
if (exp < WPAD_EXP_CLASSIC)
|
|
{
|
|
/* Wiimote IR */
|
|
struct ir_t ir;
|
|
WPAD_IR(chan, &ir);
|
|
|
|
if (ir.valid)
|
|
{
|
|
/* screen position */
|
|
input.analog[i][0] = ((ir.x + config.calx) * bitmap.viewport.w) / 640;
|
|
input.analog[i][1] = ((ir.y + config.caly) * bitmap.viewport.h) / 480;
|
|
}
|
|
else
|
|
{
|
|
/* lightgun should point outside screen area */
|
|
input.analog[i][0] = 512;
|
|
input.analog[i][1] = 512;
|
|
}
|
|
}
|
|
else if(!RetroUSB_OK())
|
|
{
|
|
/* Left analog stick */
|
|
input.analog[i][0] += x / ANALOG_SENSITIVITY;
|
|
input.analog[i][1] -= y / ANALOG_SENSITIVITY;
|
|
|
|
/* Limits */
|
|
if (input.analog[i][0] < 0) input.analog[i][0] = 0;
|
|
else if (input.analog[i][0] > bitmap.viewport.w) input.analog[i][0] = bitmap.viewport.w;
|
|
if (input.analog[i][1] < 0) input.analog[i][1] = 0;
|
|
else if (input.analog[i][1] > bitmap.viewport.h) input.analog[i][1] = bitmap.viewport.h;
|
|
}
|
|
|
|
/* Buttons */
|
|
if (p & wpad_keymap[KEY_BUTTONA]) input.pad[i] |= INPUT_A;
|
|
if (p & wpad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_B;
|
|
if (p & wpad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_C;
|
|
if (p & wpad_keymap[KEY_START]) input.pad[i] |= INPUT_START;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_MOUSE:
|
|
{
|
|
/* Mouse relative movement (-255,255) */
|
|
if (MOUSE_IsConnected())
|
|
{
|
|
/* USB mouse support */
|
|
mouse_event event;
|
|
MOUSE_GetEvent(&event);
|
|
MOUSE_FlushEvents();
|
|
|
|
/* USB mouse position (-127;+127) -> (-255;+255) */
|
|
input.analog[i][0] = event.rx * 2;
|
|
input.analog[i][1] = event.ry * 2;
|
|
|
|
/* USB mouse buttons */
|
|
if (event.button & 1) input.pad[i] |= INPUT_MOUSE_RIGHT;
|
|
if (event.button & 2) input.pad[i] |= INPUT_MOUSE_CENTER;
|
|
if (event.button & 4) input.pad[i] |= INPUT_MOUSE_LEFT;
|
|
}
|
|
else if (exp == WPAD_EXP_NONE)
|
|
{
|
|
/* Wiimote IR (buggy) */
|
|
struct ir_t ir;
|
|
WPAD_IR(chan, &ir);
|
|
|
|
/* Only if Wiimote is pointed to screen */
|
|
if(ir.smooth_valid)
|
|
{
|
|
input.analog[i][0] = (int)((ir.sx - 512) / 2 / ANALOG_SENSITIVITY);
|
|
input.analog[i][1] = (int)((ir.sy - 384) * 2 / 3 / ANALOG_SENSITIVITY);
|
|
}
|
|
}
|
|
else if(!RetroUSB_OK())
|
|
{
|
|
/* Left analog stick position (-127;+127) -> (-255;+255) */
|
|
input.analog[i][0] = (x / ANALOG_SENSITIVITY) * 2;
|
|
input.analog[i][1] = (y / ANALOG_SENSITIVITY) * 2;
|
|
}
|
|
|
|
/* Y-Axis inversion */
|
|
if (config.invert_mouse)
|
|
{
|
|
input.analog[i][1] = -input.analog[i][1];
|
|
}
|
|
|
|
/* Buttons */
|
|
if (p & wpad_keymap[KEY_BUTTONA]) input.pad[i] |= INPUT_MOUSE_CENTER;
|
|
if (p & wpad_keymap[KEY_BUTTONB]) input.pad[i] |= INPUT_MOUSE_LEFT;
|
|
if (p & wpad_keymap[KEY_BUTTONC]) input.pad[i] |= INPUT_MOUSE_RIGHT;
|
|
if (p & wpad_keymap[KEY_START]) input.pad[i] |= INPUT_START;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_PICO:
|
|
{
|
|
/* D-PAD */
|
|
if (p & PAD_BUTTON_UP) input.pad[i] |= INPUT_UP;
|
|
else if (p & PAD_BUTTON_DOWN) input.pad[i] |= INPUT_DOWN;
|
|
if (p & PAD_BUTTON_LEFT) input.pad[i] |= INPUT_LEFT;
|
|
else if (p & PAD_BUTTON_RIGHT) input.pad[i] |= INPUT_RIGHT;
|
|
|
|
/* PEN screen position (x,y) */
|
|
if (exp < WPAD_EXP_CLASSIC)
|
|
{
|
|
/* Wiimote IR */
|
|
struct ir_t ir;
|
|
WPAD_IR(chan, &ir);
|
|
if (ir.valid)
|
|
{
|
|
input.analog[0][0] = 0x3c + ((ir.x + config.calx) * (0x17c - 0x3c + 1)) / 640;
|
|
input.analog[0][1] = 0x1fc + ((ir.y + config.caly) * (0x2f7 - 0x1fc + 1)) / 480;
|
|
}
|
|
}
|
|
else if(!RetroUSB_OK())
|
|
{
|
|
/* Left analog stick */
|
|
input.analog[0][0] += x / ANALOG_SENSITIVITY;
|
|
input.analog[0][1] -= y / ANALOG_SENSITIVITY;
|
|
|
|
/* Limits */
|
|
if (input.analog[0][0] > 0x17c) input.analog[0][0] = 0x17c;
|
|
else if (input.analog[0][0] < 0x3c) input.analog[0][0] = 0x3c;
|
|
if (input.analog[0][1] < 0x1fc) input.analog[0][1] = 0x1fc;
|
|
else if (input.analog[0][1] > 0x2f7) input.analog[0][1] = 0x2f7;
|
|
}
|
|
|
|
/* PEN & RED buttons */
|
|
if (p & wpad_keymap[KEY_BUTTONA]) input.pad[0] |= INPUT_PICO_PEN;
|
|
if (p & wpad_keymap[KEY_BUTTONB]) input.pad[0] |= INPUT_PICO_RED;
|
|
|
|
/* PAGE index increment */
|
|
if (p & wpad_keymap[KEY_BUTTONC]) pico_current = (pico_current + 1) & 7;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_TEREBI:
|
|
{
|
|
/* PEN screen position (x,y) */
|
|
if (exp < WPAD_EXP_CLASSIC)
|
|
{
|
|
/* Wiimote IR */
|
|
struct ir_t ir;
|
|
WPAD_IR(chan, &ir);
|
|
if (ir.valid)
|
|
{
|
|
input.analog[0][0] = ((ir.x + config.calx) * 250) / 640;
|
|
input.analog[0][1] = ((ir.y + config.caly) * 250) / 480;
|
|
}
|
|
}
|
|
else if(!RetroUSB_OK())
|
|
{
|
|
/* Left analog stick */
|
|
input.analog[0][0] += x / ANALOG_SENSITIVITY;
|
|
input.analog[0][1] -= y / ANALOG_SENSITIVITY;
|
|
|
|
/* Limits */
|
|
if (input.analog[0][0] < 0)input.analog[0][0] = 0;
|
|
else if (input.analog[0][0] > 250) input.analog[0][0] = 250;
|
|
if (input.analog[0][1] < 0) input.analog[0][1] = 0;
|
|
else if (input.analog[0][1] > 250) input.analog[0][1] = 250;
|
|
}
|
|
|
|
/* PEN button */
|
|
if (p & wpad_keymap[KEY_BUTTONA]) input.pad[0] |= INPUT_BUTTON1;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_GRAPHIC_BOARD:
|
|
{
|
|
/* PEN screen position (x,y) */
|
|
if (exp < WPAD_EXP_CLASSIC)
|
|
{
|
|
/* Wiimote IR */
|
|
struct ir_t ir;
|
|
WPAD_IR(chan, &ir);
|
|
if (ir.valid)
|
|
{
|
|
input.analog[0][0] = ((ir.x + config.calx) * 255) / 640;
|
|
input.analog[0][1] = ((ir.y + config.caly) * 255) / 480;
|
|
}
|
|
}
|
|
else if(!RetroUSB_OK())
|
|
{
|
|
/* Left analog stick */
|
|
input.analog[0][0] += x / ANALOG_SENSITIVITY;
|
|
input.analog[0][1] -= y / ANALOG_SENSITIVITY;
|
|
|
|
/* Limits */
|
|
if (input.analog[0][0] < 0)input.analog[0][0] = 0;
|
|
else if (input.analog[0][0] > 255) input.analog[0][0] = 255;
|
|
if (input.analog[0][1] < 0) input.analog[0][1] = 0;
|
|
else if (input.analog[0][1] > 255) input.analog[0][1] = 255;
|
|
}
|
|
|
|
/* MODE Buttons */
|
|
if (p & wpad_keymap[KEY_BUTTONA]) input.pad[0] |= INPUT_GRAPHIC_PEN;
|
|
if (p & wpad_keymap[KEY_BUTTONB]) input.pad[0] |= INPUT_GRAPHIC_DO;
|
|
if (p & wpad_keymap[KEY_BUTTONC]) input.pad[0] |= INPUT_GRAPHIC_MENU;
|
|
|
|
break;
|
|
}
|
|
|
|
case DEVICE_ACTIVATOR:
|
|
{
|
|
/* Classic Controller only */
|
|
if (exp == WPAD_EXP_CLASSIC)
|
|
{
|
|
/* Left stick */
|
|
float mag = data->exp.classic.ljs.mag;
|
|
float ang = data->exp.classic.ljs.ang;
|
|
|
|
if (mag > 0.5)
|
|
{
|
|
/* 8 bottom sensors = 8 areas */
|
|
if ((ang > 22.5) && (ang <= 67.5)) input.pad[i] |= INPUT_ACTIVATOR_2L;
|
|
else if ((ang > 67.5) && (ang <= 112.5)) input.pad[i] |= INPUT_ACTIVATOR_3L;
|
|
else if ((ang > 112.5) && (ang <= 157.5)) input.pad[i] |= INPUT_ACTIVATOR_4L;
|
|
else if ((ang > 157.5) && (ang <= 202.5)) input.pad[i] |= INPUT_ACTIVATOR_5L;
|
|
else if ((ang > 202.5) && (ang <= 247.5)) input.pad[i] |= INPUT_ACTIVATOR_6L;
|
|
else if ((ang > 247.5) && (ang <= 292.5)) input.pad[i] |= INPUT_ACTIVATOR_7L;
|
|
else if ((ang > 292.5) && (ang <= 337.5)) input.pad[i] |= INPUT_ACTIVATOR_8L;
|
|
else input.pad[i] |= INPUT_ACTIVATOR_1L;
|
|
}
|
|
|
|
/* Right stick */
|
|
mag = data->exp.classic.rjs.mag;
|
|
ang = data->exp.classic.rjs.ang;
|
|
|
|
if (mag > 0.5)
|
|
{
|
|
/* 8 top sensors = 8 areas */
|
|
if ((ang > 22.5) && (ang <= 67.5)) input.pad[i] |= INPUT_ACTIVATOR_2U;
|
|
else if ((ang > 67.5) && (ang <= 112.5)) input.pad[i] |= INPUT_ACTIVATOR_3U;
|
|
else if ((ang > 112.5) && (ang <= 157.5)) input.pad[i] |= INPUT_ACTIVATOR_4U;
|
|
else if ((ang > 157.5) && (ang <= 202.5)) input.pad[i] |= INPUT_ACTIVATOR_5U;
|
|
else if ((ang > 202.5) && (ang <= 247.5)) input.pad[i] |= INPUT_ACTIVATOR_6U;
|
|
else if ((ang > 247.5) && (ang <= 292.5)) input.pad[i] |= INPUT_ACTIVATOR_7U;
|
|
else if ((ang > 292.5) && (ang <= 337.5)) input.pad[i] |= INPUT_ACTIVATOR_8U;
|
|
else input.pad[i] |= INPUT_ACTIVATOR_1U;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
/***************************************************************************************/
|
|
/* GX Input interface */
|
|
/***************************************************************************************/
|
|
void gx_input_Init(void)
|
|
{
|
|
PAD_Init();
|
|
#ifdef HW_RVL
|
|
WPAD_Init();
|
|
WPAD_SetDataFormat(WPAD_CHAN_ALL,WPAD_FMT_BTNS_ACC_IR);
|
|
WPAD_SetVRes(WPAD_CHAN_ALL,640,480);
|
|
WiiDRC_Init();
|
|
RetroUSB_OK();
|
|
#endif
|
|
}
|
|
|
|
int gx_input_FindDevices(void)
|
|
{
|
|
int i;
|
|
#ifdef HW_RVL
|
|
u32 wpad;
|
|
#endif
|
|
int found = 0;
|
|
int player = 0;
|
|
|
|
VIDEO_WaitVSync();
|
|
u32 pad = PAD_ScanPads();
|
|
|
|
for (i=0; i<MAX_DEVICES; i++)
|
|
{
|
|
/* check emulated peripheral */
|
|
if (input.dev[i] != NO_DEVICE)
|
|
{
|
|
/* test input device */
|
|
switch (config.input[player].device)
|
|
{
|
|
case 0: /* Gamecube Controller */
|
|
{
|
|
if (pad & (1 << config.input[player].port))
|
|
{
|
|
found++;
|
|
}
|
|
break;
|
|
}
|
|
|
|
#ifdef HW_RVL
|
|
case 1: /* Wiimote */
|
|
{
|
|
wpad = 255;
|
|
WPAD_Probe(config.input[player].port, &wpad);
|
|
|
|
/* make sure this is not a Wii U Pro Controller */
|
|
if (wpad == WPAD_EXP_CLASSIC)
|
|
{
|
|
WPADData *data = WPAD_Data(config.input[player].port);
|
|
if (data->exp.classic.type != 2)
|
|
{
|
|
found++;
|
|
}
|
|
}
|
|
else if (wpad != 255)
|
|
{
|
|
found++;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 2: /* Expansion Controller */
|
|
case 3:
|
|
{
|
|
wpad = 255;
|
|
WPAD_Probe(config.input[player].port, &wpad);
|
|
if (wpad == (config.input[player].device - 1))
|
|
{
|
|
found++;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 4: /* WiiU GamePad Controller */
|
|
{
|
|
if (WiiDRC_Inited() && WiiDRC_Connected())
|
|
{
|
|
found++;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 5: /* RetroUSB */
|
|
{
|
|
if (RetroUSB_OK())
|
|
{
|
|
found++;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
default:
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* next configured player */
|
|
player ++;
|
|
}
|
|
}
|
|
|
|
/* return number of connected devices */
|
|
return found;
|
|
}
|
|
|
|
void gx_input_SetDefault(void)
|
|
{
|
|
int i,j;
|
|
u32 exp;
|
|
|
|
/* set default key mapping for each type of devices */
|
|
for (i=0; i<4; i++)
|
|
{
|
|
config.pad_keymap[i][KEY_BUTTONA] = PAD_BUTTON_B;
|
|
config.pad_keymap[i][KEY_BUTTONB] = PAD_BUTTON_A;
|
|
config.pad_keymap[i][KEY_BUTTONC] = PAD_BUTTON_X;
|
|
config.pad_keymap[i][KEY_START] = PAD_BUTTON_START;
|
|
config.pad_keymap[i][KEY_BUTTONX] = PAD_TRIGGER_L;
|
|
config.pad_keymap[i][KEY_BUTTONY] = PAD_BUTTON_Y;
|
|
config.pad_keymap[i][KEY_BUTTONZ] = PAD_TRIGGER_R;
|
|
config.pad_keymap[i][KEY_MODE] = PAD_TRIGGER_Z;
|
|
config.pad_keymap[i][KEY_MENU] = PAD_TRIGGER_Z | PAD_BUTTON_RIGHT;
|
|
}
|
|
|
|
#ifdef HW_RVL
|
|
for (i=0; i<4; i++)
|
|
{
|
|
/* Wiimote (horizontal) */
|
|
config.wpad_keymap[4*WPAD_EXP_NONE + i][KEY_BUTTONA] = WPAD_BUTTON_A;
|
|
config.wpad_keymap[4*WPAD_EXP_NONE + i][KEY_BUTTONB] = WPAD_BUTTON_1;
|
|
config.wpad_keymap[4*WPAD_EXP_NONE + i][KEY_BUTTONC] = WPAD_BUTTON_2;
|
|
config.wpad_keymap[4*WPAD_EXP_NONE + i][KEY_START] = WPAD_BUTTON_PLUS;
|
|
config.wpad_keymap[4*WPAD_EXP_NONE + i][KEY_BUTTONX] = 0;
|
|
config.wpad_keymap[4*WPAD_EXP_NONE + i][KEY_BUTTONY] = 0;
|
|
config.wpad_keymap[4*WPAD_EXP_NONE + i][KEY_BUTTONZ] = 0;
|
|
config.wpad_keymap[4*WPAD_EXP_NONE + i][KEY_MODE] = 0;
|
|
config.wpad_keymap[4*WPAD_EXP_NONE + i][KEY_MENU] = WPAD_BUTTON_HOME;
|
|
|
|
/* Wiimote + Nunchuk */
|
|
config.wpad_keymap[4*WPAD_EXP_NUNCHUK + i][KEY_BUTTONA] = WPAD_NUNCHUK_BUTTON_Z;
|
|
config.wpad_keymap[4*WPAD_EXP_NUNCHUK + i][KEY_BUTTONB] = WPAD_BUTTON_B;
|
|
config.wpad_keymap[4*WPAD_EXP_NUNCHUK + i][KEY_BUTTONC] = WPAD_BUTTON_A;
|
|
config.wpad_keymap[4*WPAD_EXP_NUNCHUK + i][KEY_START] = WPAD_BUTTON_PLUS;
|
|
config.wpad_keymap[4*WPAD_EXP_NUNCHUK + i][KEY_BUTTONX] = WPAD_NUNCHUK_BUTTON_C;
|
|
config.wpad_keymap[4*WPAD_EXP_NUNCHUK + i][KEY_BUTTONY] = WPAD_BUTTON_1;
|
|
config.wpad_keymap[4*WPAD_EXP_NUNCHUK + i][KEY_BUTTONZ] = WPAD_BUTTON_2;
|
|
config.wpad_keymap[4*WPAD_EXP_NUNCHUK + i][KEY_MODE] = WPAD_BUTTON_MINUS;
|
|
config.wpad_keymap[4*WPAD_EXP_NUNCHUK + i][KEY_MENU] = WPAD_BUTTON_HOME;
|
|
|
|
/* Classic Controller */
|
|
config.wpad_keymap[4*WPAD_EXP_CLASSIC + i][KEY_BUTTONA] = WPAD_CLASSIC_BUTTON_Y;
|
|
config.wpad_keymap[4*WPAD_EXP_CLASSIC + i][KEY_BUTTONB] = WPAD_CLASSIC_BUTTON_B;
|
|
config.wpad_keymap[4*WPAD_EXP_CLASSIC + i][KEY_BUTTONC] = WPAD_CLASSIC_BUTTON_A;
|
|
config.wpad_keymap[4*WPAD_EXP_CLASSIC + i][KEY_START] = WPAD_CLASSIC_BUTTON_PLUS;
|
|
config.wpad_keymap[4*WPAD_EXP_CLASSIC + i][KEY_BUTTONX] = WPAD_CLASSIC_BUTTON_ZL;
|
|
config.wpad_keymap[4*WPAD_EXP_CLASSIC + i][KEY_BUTTONY] = WPAD_CLASSIC_BUTTON_ZR;
|
|
config.wpad_keymap[4*WPAD_EXP_CLASSIC + i][KEY_BUTTONZ] = WPAD_CLASSIC_BUTTON_X;
|
|
config.wpad_keymap[4*WPAD_EXP_CLASSIC + i][KEY_MODE] = WPAD_CLASSIC_BUTTON_MINUS;
|
|
config.wpad_keymap[4*WPAD_EXP_CLASSIC + i][KEY_MENU] = WPAD_CLASSIC_BUTTON_HOME;
|
|
|
|
/* WiiU GamePad Controller */
|
|
config.wpad_keymap[4*3][KEY_BUTTONA] = WIIDRC_BUTTON_Y;
|
|
config.wpad_keymap[4*3][KEY_BUTTONB] = WIIDRC_BUTTON_B;
|
|
config.wpad_keymap[4*3][KEY_BUTTONC] = WIIDRC_BUTTON_A;
|
|
config.wpad_keymap[4*3][KEY_START] = WIIDRC_BUTTON_PLUS;
|
|
config.wpad_keymap[4*3][KEY_BUTTONX] = WIIDRC_BUTTON_L;
|
|
config.wpad_keymap[4*3][KEY_BUTTONY] = WIIDRC_BUTTON_R;
|
|
config.wpad_keymap[4*3][KEY_BUTTONZ] = WIIDRC_BUTTON_X;
|
|
config.wpad_keymap[4*3][KEY_MODE] = WIIDRC_BUTTON_MINUS;
|
|
config.wpad_keymap[4*3][KEY_MENU] = WIIDRC_BUTTON_HOME;
|
|
|
|
/* RetroUSB */
|
|
config.wpad_keymap[4 * 3 + 1][KEY_BUTTONA] = PAD_BUTTON_Y;
|
|
config.wpad_keymap[4 * 3 + 1][KEY_BUTTONB] = PAD_BUTTON_B;
|
|
config.wpad_keymap[4 * 3 + 1][KEY_BUTTONC] = PAD_BUTTON_A;
|
|
config.wpad_keymap[4 * 3 + 1][KEY_START] = PAD_BUTTON_START;
|
|
config.wpad_keymap[4 * 3 + 1][KEY_BUTTONX] = PAD_TRIGGER_L;
|
|
config.wpad_keymap[4 * 3 + 1][KEY_BUTTONY] = PAD_TRIGGER_R;
|
|
config.wpad_keymap[4 * 3 + 1][KEY_BUTTONZ] = PAD_BUTTON_X;
|
|
config.wpad_keymap[4 * 3 + 1][KEY_MODE] = PAD_TRIGGER_Z;
|
|
config.wpad_keymap[4 * 3 + 1][KEY_MENU] = 0;
|
|
}
|
|
#endif
|
|
|
|
/* Default player inputs */
|
|
for (i=0; i<MAX_INPUTS; i++)
|
|
{
|
|
config.input[i].device = -1;
|
|
config.input[i].port = i%4;
|
|
config.input[i].padtype = DEVICE_PAD2B | DEVICE_PAD3B | DEVICE_PAD6B; /* autodetected */
|
|
}
|
|
|
|
#ifdef HW_RVL
|
|
|
|
i = 0;
|
|
/* autodetect connected WiiU Gamepad Controller */
|
|
if (WiiDRC_Inited() && WiiDRC_Connected())
|
|
{
|
|
config.input[i].device = 4;
|
|
config.input[i].port = 0;
|
|
i++;
|
|
}
|
|
/* autodetect RetroUSB */
|
|
if (RetroUSB_OK())
|
|
{
|
|
config.input[i].device = 5;
|
|
config.input[i].port = 0;
|
|
i++;
|
|
}
|
|
|
|
/* autodetect connected Wii Controllers */
|
|
do
|
|
{
|
|
exp = 255;
|
|
WPAD_Probe(i, &exp);
|
|
|
|
if (exp == WPAD_EXP_CLASSIC)
|
|
{
|
|
/* use Classic Controller */
|
|
config.input[i].device = 3;
|
|
config.input[i].port = i;
|
|
}
|
|
else if (exp == WPAD_EXP_NUNCHUK)
|
|
{
|
|
/* use Wiimote + Nunchuk */
|
|
config.input[i].device = 2;
|
|
config.input[i].port = i;
|
|
}
|
|
else if (exp != 255)
|
|
{
|
|
/* use Wiimote by default */
|
|
config.input[i].device = 1;
|
|
config.input[i].port = i;
|
|
}
|
|
else
|
|
{
|
|
/* look for unused Wiimotes */
|
|
for (j=0; j<i; j++)
|
|
{
|
|
/* Wiimote could still be used when Classic Controller has been assigned */
|
|
if (config.input[j].device == 3)
|
|
{
|
|
/* make sure this is not a Wii U Pro Controller */
|
|
WPADData *data = WPAD_Data(config.input[j].port);
|
|
if (data->exp.classic.type != 2)
|
|
{
|
|
/* Wiimote is available */
|
|
config.input[i].device = 1;
|
|
config.input[i].port = j;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
i++;
|
|
}
|
|
while (i < 4);
|
|
#endif
|
|
|
|
/* Autodetect Gamecube Controllers */
|
|
VIDEO_WaitVSync();
|
|
exp = PAD_ScanPads();
|
|
for (i=0; i<4; i++)
|
|
{
|
|
/* check if Gamecube Controller is connected */
|
|
if (exp & (1 << i))
|
|
{
|
|
for (j=0; j<MAX_INPUTS; j++)
|
|
{
|
|
/* look for the first unassigned player */
|
|
if (config.input[j].device == -1)
|
|
{
|
|
config.input[j].device = 0;
|
|
config.input[j].port = i;
|
|
j = MAX_INPUTS;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void gx_input_Config(u8 chan, u8 device, u8 type)
|
|
{
|
|
int first_key, last_key;
|
|
|
|
/* emulated device */
|
|
switch (type)
|
|
{
|
|
case DEVICE_PADDLE:
|
|
case DEVICE_PAD2B:
|
|
case DEVICE_SPORTSPAD:
|
|
{
|
|
first_key = KEY_BUTTONB;
|
|
last_key = KEY_START;
|
|
sprintf(keyname[KEY_BUTTONB],"Button 1");
|
|
sprintf(keyname[KEY_BUTTONC],"Button 2");
|
|
sprintf(keyname[KEY_START],"PAUSE Button");
|
|
break;
|
|
}
|
|
|
|
case DEVICE_XE_1AP:
|
|
{
|
|
first_key = KEY_BUTTONA;
|
|
last_key = KEY_MODE;
|
|
sprintf(keyname[KEY_BUTTONA],"Button A");
|
|
sprintf(keyname[KEY_BUTTONB],"Button B");
|
|
sprintf(keyname[KEY_BUTTONC],"Button C");
|
|
sprintf(keyname[KEY_START],"START Button");
|
|
sprintf(keyname[KEY_BUTTONX],"Button D");
|
|
sprintf(keyname[KEY_BUTTONY],"Button E1");
|
|
sprintf(keyname[KEY_BUTTONZ],"Button E2");
|
|
sprintf(keyname[KEY_MODE],"SELECT Button");
|
|
break;
|
|
}
|
|
|
|
case DEVICE_MOUSE:
|
|
{
|
|
first_key = KEY_BUTTONA;
|
|
last_key = KEY_START;
|
|
sprintf(keyname[KEY_BUTTONA],"Middle Button");
|
|
sprintf(keyname[KEY_BUTTONB],"Left Button");
|
|
sprintf(keyname[KEY_BUTTONC],"Right Button");
|
|
sprintf(keyname[KEY_START],"START Button");
|
|
break;
|
|
}
|
|
|
|
case DEVICE_PAD3B:
|
|
{
|
|
first_key = KEY_BUTTONA;
|
|
last_key = KEY_START;
|
|
sprintf(keyname[KEY_BUTTONA],"Button A");
|
|
sprintf(keyname[KEY_BUTTONB],"Button B");
|
|
sprintf(keyname[KEY_BUTTONC],"Button C");
|
|
sprintf(keyname[KEY_START],"START Button");
|
|
break;
|
|
}
|
|
|
|
case DEVICE_PAD6B:
|
|
{
|
|
first_key = KEY_BUTTONA;
|
|
last_key = KEY_MODE;
|
|
sprintf(keyname[KEY_BUTTONA],"Button A");
|
|
sprintf(keyname[KEY_BUTTONB],"Button B");
|
|
sprintf(keyname[KEY_BUTTONC],"Button C");
|
|
sprintf(keyname[KEY_START],"START Button");
|
|
sprintf(keyname[KEY_BUTTONX],"Button X");
|
|
sprintf(keyname[KEY_BUTTONY],"Button Y");
|
|
sprintf(keyname[KEY_BUTTONZ],"Button Z");
|
|
sprintf(keyname[KEY_MODE],"MODE Button");
|
|
break;
|
|
}
|
|
|
|
case DEVICE_LIGHTGUN:
|
|
{
|
|
first_key = KEY_BUTTONA;
|
|
last_key = KEY_START;
|
|
if (input.system[1] == SYSTEM_MENACER)
|
|
{
|
|
sprintf(keyname[KEY_BUTTONA],"TRIGGER Button");
|
|
sprintf(keyname[KEY_BUTTONB],"Button B");
|
|
sprintf(keyname[KEY_BUTTONC],"Button C");
|
|
sprintf(keyname[KEY_START],"START Button");
|
|
}
|
|
else if (input.system[1] == SYSTEM_JUSTIFIER)
|
|
{
|
|
sprintf(keyname[KEY_BUTTONA],"TRIGGER Button");
|
|
sprintf(keyname[KEY_BUTTONB],"N.A");
|
|
sprintf(keyname[KEY_BUTTONC],"N.A");
|
|
sprintf(keyname[KEY_START],"START Button");
|
|
}
|
|
else
|
|
{
|
|
sprintf(keyname[KEY_BUTTONA],"TRIGGER Button");
|
|
sprintf(keyname[KEY_BUTTONB],"N.A");
|
|
sprintf(keyname[KEY_BUTTONC],"N.A");
|
|
sprintf(keyname[KEY_START],"PAUSE Button");
|
|
}
|
|
break;
|
|
}
|
|
|
|
case DEVICE_PICO:
|
|
{
|
|
first_key = KEY_BUTTONA;
|
|
last_key = KEY_BUTTONB;
|
|
sprintf(keyname[KEY_BUTTONA],"PEN Button");
|
|
sprintf(keyname[KEY_BUTTONB],"RED Button");
|
|
break;
|
|
}
|
|
|
|
case DEVICE_TEREBI:
|
|
{
|
|
first_key = KEY_BUTTONA;
|
|
last_key = KEY_BUTTONA;
|
|
sprintf(keyname[KEY_BUTTONA],"PEN Button");
|
|
break;
|
|
}
|
|
|
|
case DEVICE_GRAPHIC_BOARD:
|
|
{
|
|
first_key = KEY_BUTTONA;
|
|
last_key = KEY_START;
|
|
sprintf(keyname[KEY_BUTTONA],"PEN Button");
|
|
sprintf(keyname[KEY_BUTTONB],"DO Button");
|
|
sprintf(keyname[KEY_BUTTONC],"MENU Button");
|
|
sprintf(keyname[KEY_START],"PAUSE Button");
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
first_key = KEY_BUTTONA;
|
|
last_key = KEY_BUTTONA;
|
|
sprintf(keyname[KEY_BUTTONA],"N.A");
|
|
GUI_WaitPrompt("Info","Activator is not configurable !");
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Input device */
|
|
switch (device)
|
|
{
|
|
case 0:
|
|
{
|
|
pad_config(chan, first_key, last_key);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
#ifdef HW_RVL
|
|
wpad_config(device - 1, chan, first_key, last_key);
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void gx_input_UpdateEmu(void)
|
|
{
|
|
/* Update GC controllers status */
|
|
PAD_ScanPads();
|
|
|
|
#ifdef HW_RVL
|
|
/* Update Wii controllers status */
|
|
WPAD_ScanPads();
|
|
|
|
/* Default Wii controller menu keys */
|
|
if (WPAD_ButtonsDown(0) & (WPAD_BUTTON_HOME | WPAD_CLASSIC_BUTTON_HOME))
|
|
{
|
|
/* Default fast-forward key combo */
|
|
if (WPAD_ButtonsHeld(0) & (WPAD_BUTTON_MINUS | WPAD_CLASSIC_BUTTON_MINUS))
|
|
{
|
|
audioSync ^= AUDIO_WAIT;
|
|
videoSync = (audioSync && config.vsync && (gc_pal != vdp_pal)) ? VIDEO_WAIT : 0;
|
|
return;
|
|
}
|
|
|
|
/* Return to main menu */
|
|
ConfigRequested = 1;
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
int i, player = 0;
|
|
for (i=0; i<MAX_DEVICES; i++)
|
|
{
|
|
/* update inputs */
|
|
if (input.dev[i] != NO_DEVICE)
|
|
{
|
|
/* clear key status */
|
|
input.pad[i] = 0;
|
|
|
|
if (config.input[player].device == 0)
|
|
{
|
|
pad_update(config.input[player].port, i);
|
|
}
|
|
#ifdef HW_RVL
|
|
else if (config.input[player].device > 0)
|
|
{
|
|
wpad_update(config.input[player].port, i, config.input[player].device - 1);
|
|
}
|
|
#endif
|
|
|
|
/* increment player index */
|
|
player ++;
|
|
}
|
|
}
|
|
|
|
/* Update RAM patches */
|
|
RAMCheatUpdate();
|
|
}
|
|
|
|
/* Menu inputs update function */
|
|
void gx_input_UpdateMenu(void)
|
|
{
|
|
/* Check if inputs update are disabled */
|
|
if (inputs_disabled) return;
|
|
|
|
/* PAD status update */
|
|
PAD_ScanPads();
|
|
|
|
/* PAD pressed keys */
|
|
s16 pp = PAD_ButtonsDown(0);
|
|
|
|
/* PAD held keys (direction/selection) */
|
|
s16 hp = PAD_ButtonsHeld(0) & PAD_BUTTONS_HELD;
|
|
|
|
/* PAD analog sticks (handled as PAD held direction keys) */
|
|
s8 x = PAD_StickX(0);
|
|
s8 y = PAD_StickY(0);
|
|
if (x > ANALOG_SENSITIVITY) hp |= PAD_BUTTON_RIGHT;
|
|
else if (x < -ANALOG_SENSITIVITY) hp |= PAD_BUTTON_LEFT;
|
|
else if (y > ANALOG_SENSITIVITY) hp |= PAD_BUTTON_UP;
|
|
else if (y < -ANALOG_SENSITIVITY) hp |= PAD_BUTTON_DOWN;
|
|
|
|
#ifdef HW_RVL
|
|
/* WPAD status update */
|
|
WPAD_ScanPads();
|
|
WPADData *data = WPAD_Data(0);
|
|
|
|
/* WPAD pressed keys */
|
|
u32 pw = data->btns_d;
|
|
|
|
/* WPAD held keys (direction/selection) */
|
|
u32 hw = data->btns_h & WPAD_BUTTONS_HELD;
|
|
|
|
/* WPAD analog sticks (handled as PAD held direction keys) */
|
|
x = wpad_StickX(data, 0);
|
|
y = wpad_StickY(data, 0);
|
|
if (x > ANALOG_SENSITIVITY) hp |= PAD_BUTTON_RIGHT;
|
|
else if (x < -ANALOG_SENSITIVITY) hp |= PAD_BUTTON_LEFT;
|
|
else if (y > ANALOG_SENSITIVITY) hp |= PAD_BUTTON_UP;
|
|
else if (y < -ANALOG_SENSITIVITY) hp |= PAD_BUTTON_DOWN;
|
|
|
|
/* WiiU GamePad status */
|
|
u32 pwu = 0;
|
|
u32 hwu = 0;
|
|
if (WiiDRC_Inited())
|
|
{
|
|
WiiDRC_ScanPads();
|
|
if (WiiDRC_ShutdownRequested())
|
|
{
|
|
Shutdown = ConfigRequested = 1;
|
|
reload = 0;
|
|
return;
|
|
}
|
|
pwu = WiiDRC_ButtonsDown();
|
|
hwu = WiiDRC_ButtonsHeld() & WIIU_BUTTONS_HELD;
|
|
x = WiiDRC_lStickX();
|
|
y = WiiDRC_lStickY();
|
|
if (x > ANALOG_SENSITIVITY) hp |= PAD_BUTTON_RIGHT;
|
|
else if (x < -ANALOG_SENSITIVITY) hp |= PAD_BUTTON_LEFT;
|
|
else if (y > ANALOG_SENSITIVITY) hp |= PAD_BUTTON_UP;
|
|
else if (y < -ANALOG_SENSITIVITY) hp |= PAD_BUTTON_DOWN;
|
|
}
|
|
|
|
#endif
|
|
|
|
/* check if any direction/selection key is being held or just being pressed/released */
|
|
#ifdef HW_RVL
|
|
if (pp||pw||pwu) held_cnt = 0;
|
|
else if (hp||hw||hwu) held_cnt++;
|
|
else held_cnt = 0;
|
|
#else
|
|
if (pp) held_cnt = 0;
|
|
else if (hp) held_cnt++;
|
|
else held_cnt = 0;
|
|
#endif
|
|
|
|
/* initial delay (prevents triggering to start immediately) */
|
|
if (held_cnt > HELD_DELAY)
|
|
{
|
|
/* key triggering */
|
|
pp |= hp;
|
|
#ifdef HW_RVL
|
|
pw |= hw;
|
|
pwu |= hwu;
|
|
#endif
|
|
|
|
/* delay until next triggering (adjusts direction/selection update speed) */
|
|
held_cnt -= HELD_SPEED;
|
|
}
|
|
|
|
#ifdef HW_RVL
|
|
/* Wiimote & Classic Controller direction keys */
|
|
WPAD_IR(0, &m_input.ir);
|
|
if (m_input.ir.valid)
|
|
{
|
|
/* Wiimote is handled vertically */
|
|
if (pw & (WPAD_BUTTON_UP|WPAD_CLASSIC_BUTTON_UP)) pp |= PAD_BUTTON_UP;
|
|
else if (pw & (WPAD_BUTTON_DOWN|WPAD_CLASSIC_BUTTON_DOWN)) pp |= PAD_BUTTON_DOWN;
|
|
else if (pw & (WPAD_BUTTON_LEFT|WPAD_CLASSIC_BUTTON_LEFT)) pp |= PAD_BUTTON_LEFT;
|
|
else if (pw & (WPAD_BUTTON_RIGHT|WPAD_CLASSIC_BUTTON_RIGHT)) pp |= PAD_BUTTON_RIGHT;
|
|
|
|
/* Wiimote pointer user calibration */
|
|
m_input.ir.x += config.calx;
|
|
m_input.ir.y += config.caly;
|
|
}
|
|
else
|
|
{
|
|
/* Wiimote is handled horizontally */
|
|
if (pw & (WPAD_BUTTON_UP|WPAD_CLASSIC_BUTTON_LEFT)) pp |= PAD_BUTTON_LEFT;
|
|
else if (pw & (WPAD_BUTTON_DOWN|WPAD_CLASSIC_BUTTON_RIGHT)) pp |= PAD_BUTTON_RIGHT;
|
|
else if (pw & (WPAD_BUTTON_LEFT|WPAD_CLASSIC_BUTTON_DOWN)) pp |= PAD_BUTTON_DOWN;
|
|
else if (pw & (WPAD_BUTTON_RIGHT|WPAD_CLASSIC_BUTTON_UP)) pp |= PAD_BUTTON_UP;
|
|
}
|
|
|
|
/* WPAD button keys */
|
|
if (pw & (WPAD_BUTTON_2|WPAD_BUTTON_A|WPAD_CLASSIC_BUTTON_A)) pp |= PAD_BUTTON_A;
|
|
if (pw & (WPAD_BUTTON_1|WPAD_BUTTON_B|WPAD_CLASSIC_BUTTON_B)) pp |= PAD_BUTTON_B;
|
|
if (pw & (WPAD_BUTTON_HOME|WPAD_CLASSIC_BUTTON_HOME)) pp |= PAD_TRIGGER_Z;
|
|
if (pw & (WPAD_BUTTON_PLUS|WPAD_CLASSIC_BUTTON_PLUS|WPAD_CLASSIC_BUTTON_FULL_L)) pp |= PAD_TRIGGER_L;
|
|
if (pw & (WPAD_BUTTON_MINUS|WPAD_CLASSIC_BUTTON_MINUS|WPAD_CLASSIC_BUTTON_FULL_R)) pp |= PAD_TRIGGER_R;
|
|
|
|
/* WiiU GamePad direction keys */
|
|
if (pwu & WIIDRC_BUTTON_UP) pp |= PAD_BUTTON_UP;
|
|
else if (pwu & WIIDRC_BUTTON_DOWN) pp |= PAD_BUTTON_DOWN;
|
|
else if (pwu & WIIDRC_BUTTON_LEFT) pp |= PAD_BUTTON_LEFT;
|
|
else if (pwu & WIIDRC_BUTTON_RIGHT) pp |= PAD_BUTTON_RIGHT;
|
|
|
|
/* WiiU GamePad button keys */
|
|
if (pwu & WIIDRC_BUTTON_A) pp |= PAD_BUTTON_A;
|
|
if (pwu & WIIDRC_BUTTON_B) pp |= PAD_BUTTON_B;
|
|
if (pwu & WIIDRC_BUTTON_HOME) pp |= PAD_TRIGGER_Z;
|
|
if (pwu & (WIIDRC_BUTTON_PLUS|WIIDRC_BUTTON_L|WIIDRC_BUTTON_ZL)) pp |= PAD_TRIGGER_L;
|
|
if (pwu & (WIIDRC_BUTTON_MINUS|WIIDRC_BUTTON_R|WIIDRC_BUTTON_ZR)) pp |= PAD_TRIGGER_R;
|
|
#endif
|
|
|
|
/* Update menu inputs */
|
|
m_input.keys = pp;
|
|
}
|