mirror of
https://github.com/hrydgard/ppsspp.git
synced 2024-11-23 13:30:02 +00:00
848 lines
24 KiB
C++
848 lines
24 KiB
C++
// Copyright (c) 2013- PPSSPP Project.
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
// Official git repository and contact information can be found at
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
#include <algorithm>
|
|
#include <set>
|
|
#include <unordered_map>
|
|
|
|
#include "ppsspp_config.h"
|
|
|
|
#include "Common/System/NativeApp.h"
|
|
#include "Common/System/System.h"
|
|
#include "Common/Data/Format/IniFile.h"
|
|
#include "Common/Input/InputState.h"
|
|
#include "Common/VR/PPSSPPVR.h"
|
|
#include "Common/Log.h"
|
|
#include "Common/StringUtils.h"
|
|
#include "Core/HLE/sceUtility.h"
|
|
#include "Core/Config.h"
|
|
#include "Core/KeyMap.h"
|
|
#include "Core/KeyMapDefaults.h"
|
|
|
|
namespace KeyMap {
|
|
|
|
KeyDef AxisDef(int deviceId, int axisId, int direction);
|
|
|
|
KeyMapping g_controllerMap;
|
|
// Incremented on modification, so we know when to update menus.
|
|
int g_controllerMapGeneration = 0;
|
|
std::set<std::string> g_seenPads;
|
|
std::map<int, std::string> g_padNames;
|
|
std::set<int> g_seenDeviceIds;
|
|
|
|
bool g_swapped_keys = false;
|
|
|
|
void KeyCodesFromPspButton(int btn, std::vector<keycode_t> *keycodes) {
|
|
for (auto i = g_controllerMap[btn].begin(), end = g_controllerMap[btn].end(); i != end; ++i) {
|
|
keycodes->push_back((keycode_t)i->keyCode);
|
|
}
|
|
}
|
|
|
|
// TODO: This is such a mess...
|
|
void UpdateNativeMenuKeys() {
|
|
std::vector<KeyDef> confirmKeys, cancelKeys;
|
|
std::vector<KeyDef> tabLeft, tabRight;
|
|
std::vector<KeyDef> upKeys, downKeys, leftKeys, rightKeys;
|
|
|
|
int confirmKey = g_Config.iButtonPreference == PSP_SYSTEMPARAM_BUTTON_CROSS ? CTRL_CROSS : CTRL_CIRCLE;
|
|
int cancelKey = g_Config.iButtonPreference == PSP_SYSTEMPARAM_BUTTON_CROSS ? CTRL_CIRCLE : CTRL_CROSS;
|
|
|
|
// Mouse mapping might be problematic in UI, so let's ignore mouse for UI
|
|
KeyFromPspButton(confirmKey, &confirmKeys, true);
|
|
KeyFromPspButton(cancelKey, &cancelKeys, true);
|
|
KeyFromPspButton(CTRL_LTRIGGER, &tabLeft, true);
|
|
KeyFromPspButton(CTRL_RTRIGGER, &tabRight, true);
|
|
KeyFromPspButton(CTRL_UP, &upKeys, true);
|
|
KeyFromPspButton(CTRL_DOWN, &downKeys, true);
|
|
KeyFromPspButton(CTRL_LEFT, &leftKeys, true);
|
|
KeyFromPspButton(CTRL_RIGHT, &rightKeys, true);
|
|
|
|
#ifdef __ANDROID__
|
|
// Hardcode DPAD on Android
|
|
upKeys.push_back(KeyDef(DEVICE_ID_ANY, NKCODE_DPAD_UP));
|
|
downKeys.push_back(KeyDef(DEVICE_ID_ANY, NKCODE_DPAD_DOWN));
|
|
leftKeys.push_back(KeyDef(DEVICE_ID_ANY, NKCODE_DPAD_LEFT));
|
|
rightKeys.push_back(KeyDef(DEVICE_ID_ANY, NKCODE_DPAD_RIGHT));
|
|
#endif
|
|
|
|
// Push several hard-coded keys before submitting to native.
|
|
const KeyDef hardcodedConfirmKeys[] = {
|
|
KeyDef(DEVICE_ID_KEYBOARD, NKCODE_SPACE),
|
|
KeyDef(DEVICE_ID_KEYBOARD, NKCODE_ENTER),
|
|
KeyDef(DEVICE_ID_KEYBOARD, NKCODE_NUMPAD_ENTER),
|
|
KeyDef(DEVICE_ID_ANY, NKCODE_BUTTON_A),
|
|
KeyDef(DEVICE_ID_PAD_0, NKCODE_DPAD_CENTER), // A number of Android devices.
|
|
};
|
|
|
|
// If they're not already bound, add them in.
|
|
for (size_t i = 0; i < ARRAY_SIZE(hardcodedConfirmKeys); i++) {
|
|
if (std::find(confirmKeys.begin(), confirmKeys.end(), hardcodedConfirmKeys[i]) == confirmKeys.end())
|
|
confirmKeys.push_back(hardcodedConfirmKeys[i]);
|
|
}
|
|
|
|
const KeyDef hardcodedCancelKeys[] = {
|
|
KeyDef(DEVICE_ID_KEYBOARD, NKCODE_ESCAPE),
|
|
KeyDef(DEVICE_ID_ANY, NKCODE_BACK),
|
|
KeyDef(DEVICE_ID_ANY, NKCODE_BUTTON_B),
|
|
KeyDef(DEVICE_ID_MOUSE, NKCODE_EXT_MOUSEBUTTON_4),
|
|
};
|
|
|
|
for (size_t i = 0; i < ARRAY_SIZE(hardcodedCancelKeys); i++) {
|
|
if (std::find(cancelKeys.begin(), cancelKeys.end(), hardcodedCancelKeys[i]) == cancelKeys.end())
|
|
cancelKeys.push_back(hardcodedCancelKeys[i]);
|
|
}
|
|
|
|
SetDPadKeys(upKeys, downKeys, leftKeys, rightKeys);
|
|
SetConfirmCancelKeys(confirmKeys, cancelKeys);
|
|
SetTabLeftRightKeys(tabLeft, tabRight);
|
|
|
|
std::unordered_map<int, int> flipYByDeviceId;
|
|
for (int deviceId : g_seenDeviceIds) {
|
|
auto analogs = MappedAxesForDevice(deviceId);
|
|
flipYByDeviceId[deviceId] = analogs.leftY.direction;
|
|
}
|
|
SetAnalogFlipY(flipYByDeviceId);
|
|
}
|
|
|
|
static const KeyMap_IntStrPair key_names[] = {
|
|
{NKCODE_A, "A"},
|
|
{NKCODE_B, "B"},
|
|
{NKCODE_C, "C"},
|
|
{NKCODE_D, "D"},
|
|
{NKCODE_E, "E"},
|
|
{NKCODE_F, "F"},
|
|
{NKCODE_G, "G"},
|
|
{NKCODE_H, "H"},
|
|
{NKCODE_I, "I"},
|
|
{NKCODE_J, "J"},
|
|
{NKCODE_K, "K"},
|
|
{NKCODE_L, "L"},
|
|
{NKCODE_M, "M"},
|
|
{NKCODE_N, "N"},
|
|
{NKCODE_O, "O"},
|
|
{NKCODE_P, "P"},
|
|
{NKCODE_Q, "Q"},
|
|
{NKCODE_R, "R"},
|
|
{NKCODE_S, "S"},
|
|
{NKCODE_T, "T"},
|
|
{NKCODE_U, "U"},
|
|
{NKCODE_V, "V"},
|
|
{NKCODE_W, "W"},
|
|
{NKCODE_X, "X"},
|
|
{NKCODE_Y, "Y"},
|
|
{NKCODE_Z, "Z"},
|
|
|
|
{NKCODE_0, "0"},
|
|
{NKCODE_1, "1"},
|
|
{NKCODE_2, "2"},
|
|
{NKCODE_3, "3"},
|
|
{NKCODE_4, "4"},
|
|
{NKCODE_5, "5"},
|
|
{NKCODE_6, "6"},
|
|
{NKCODE_7, "7"},
|
|
{NKCODE_8, "8"},
|
|
{NKCODE_9, "9"},
|
|
|
|
{NKCODE_F1, "F1"},
|
|
{NKCODE_F2, "F2"},
|
|
{NKCODE_F3, "F3"},
|
|
{NKCODE_F4, "F4"},
|
|
{NKCODE_F5, "F5"},
|
|
{NKCODE_F6, "F6"},
|
|
{NKCODE_F7, "F7"},
|
|
{NKCODE_F8, "F8"},
|
|
{NKCODE_F9, "F9"},
|
|
{NKCODE_F10, "F10"},
|
|
{NKCODE_F11, "F11"},
|
|
{NKCODE_F12, "F12"},
|
|
|
|
{NKCODE_GRAVE, "`"},
|
|
{NKCODE_SLASH, "/"},
|
|
{NKCODE_BACKSLASH, "\\"},
|
|
{NKCODE_SEMICOLON, ";"},
|
|
{NKCODE_COMMA, ","},
|
|
{NKCODE_PERIOD, "."},
|
|
{NKCODE_LEFT_BRACKET, "["},
|
|
{NKCODE_RIGHT_BRACKET, "]"},
|
|
{NKCODE_APOSTROPHE, "'"},
|
|
{NKCODE_MINUS, "-"},
|
|
{NKCODE_PLUS, "+"},
|
|
{NKCODE_SYSRQ, "Print"},
|
|
{NKCODE_SCROLL_LOCK, "ScrLock"},
|
|
{NKCODE_BREAK, "Pause"},
|
|
|
|
{NKCODE_BACK, "Back"},
|
|
{NKCODE_TAB, "Tab"},
|
|
{NKCODE_ENTER, "Enter"},
|
|
{NKCODE_SHIFT_LEFT, "LShift"},
|
|
{NKCODE_SHIFT_RIGHT, "RShift"},
|
|
{NKCODE_CTRL_LEFT, "LCtrl"},
|
|
{NKCODE_CTRL_RIGHT, "RCtrl"},
|
|
{NKCODE_ALT_LEFT, "LAlt"},
|
|
{NKCODE_ALT_RIGHT, "RAlt"},
|
|
{NKCODE_SPACE, "Space"},
|
|
{NKCODE_WINDOW, "Windows"},
|
|
{NKCODE_DEL, "Backspace"},
|
|
{NKCODE_FORWARD_DEL, "Delete"},
|
|
{NKCODE_MOVE_HOME, "Home"},
|
|
{NKCODE_MOVE_END, "End"},
|
|
{NKCODE_ESCAPE, "Esc"},
|
|
{NKCODE_CAPS_LOCK, "CapsLock"},
|
|
|
|
{NKCODE_VOLUME_UP, "Vol +"},
|
|
{NKCODE_VOLUME_DOWN, "Vol -"},
|
|
{NKCODE_HOME, "Home"},
|
|
{NKCODE_INSERT, "Ins"},
|
|
{NKCODE_PAGE_UP, "PgUp"},
|
|
{NKCODE_PAGE_DOWN, "PgDn"},
|
|
{NKCODE_CLEAR, "Clear"}, // 5 when numlock off
|
|
{NKCODE_CALL, "Call"},
|
|
{NKCODE_ENDCALL, "End Call"},
|
|
|
|
{NKCODE_DPAD_LEFT, "Left"},
|
|
{NKCODE_DPAD_UP, "Up"},
|
|
{NKCODE_DPAD_RIGHT, "Right"},
|
|
{NKCODE_DPAD_DOWN, "Down"},
|
|
|
|
{NKCODE_BUTTON_L1, "L1"},
|
|
{NKCODE_BUTTON_L2, "L2"},
|
|
{NKCODE_BUTTON_R1, "R1"},
|
|
{NKCODE_BUTTON_R2, "R2"},
|
|
|
|
{NKCODE_BUTTON_A, "[A]"},
|
|
{NKCODE_BUTTON_B, "[B]"},
|
|
{NKCODE_BUTTON_C, "[C]"},
|
|
{NKCODE_BUTTON_X, "[X]"},
|
|
{NKCODE_BUTTON_Y, "[Y]"},
|
|
{NKCODE_BUTTON_Z, "[Z]"},
|
|
{NKCODE_BUTTON_1, "b1"},
|
|
{NKCODE_BUTTON_2, "b2"},
|
|
{NKCODE_BUTTON_3, "b3"},
|
|
{NKCODE_BUTTON_4, "b4"},
|
|
{NKCODE_BUTTON_5, "b5"},
|
|
{NKCODE_BUTTON_6, "b6"},
|
|
{NKCODE_BUTTON_7, "b7"},
|
|
{NKCODE_BUTTON_8, "b8"},
|
|
{NKCODE_BUTTON_9, "b9"},
|
|
{NKCODE_BUTTON_10, "b10"},
|
|
{NKCODE_BUTTON_11, "b11"},
|
|
{NKCODE_BUTTON_12, "b12"},
|
|
{NKCODE_BUTTON_13, "b13"},
|
|
{NKCODE_BUTTON_14, "b14"},
|
|
{NKCODE_BUTTON_15, "b15"},
|
|
{NKCODE_BUTTON_16, "b16"},
|
|
{NKCODE_BUTTON_START, "Start"},
|
|
{NKCODE_BUTTON_SELECT, "Select"},
|
|
{NKCODE_BUTTON_CIRCLE, "Circle"},
|
|
{NKCODE_BUTTON_CIRCLE_PS3, "Circle3"},
|
|
{NKCODE_BUTTON_CROSS, "Cross"},
|
|
{NKCODE_BUTTON_CROSS_PS3, "Cross3"},
|
|
{NKCODE_BUTTON_TRIANGLE, "Triangle"},
|
|
{NKCODE_BUTTON_SQUARE, "Square"},
|
|
{NKCODE_BUTTON_THUMBL, "ThumbL"},
|
|
{NKCODE_BUTTON_THUMBR, "ThumbR"},
|
|
{NKCODE_BUTTON_MODE, "Mode"},
|
|
|
|
{NKCODE_EXT_PIPE, "|"},
|
|
{NKCODE_NUMPAD_DIVIDE, "Num/"},
|
|
{NKCODE_NUMPAD_MULTIPLY, "Num*"},
|
|
{NKCODE_NUMPAD_ADD, "Num+"},
|
|
{NKCODE_NUMPAD_SUBTRACT, "Num-"},
|
|
{NKCODE_NUMPAD_DOT, "Num."},
|
|
{NKCODE_NUMPAD_COMMA, "Num,"},
|
|
{NKCODE_NUMPAD_ENTER, "NumEnter"},
|
|
{NKCODE_NUMPAD_EQUALS, "Num="},
|
|
{NKCODE_NUMPAD_LEFT_PAREN, "Num("},
|
|
{NKCODE_NUMPAD_RIGHT_PAREN, "Num)"},
|
|
{NKCODE_NUMPAD_0, "Num0"},
|
|
{NKCODE_NUMPAD_1, "Num1"},
|
|
{NKCODE_NUMPAD_2, "Num2"},
|
|
{NKCODE_NUMPAD_3, "Num3"},
|
|
{NKCODE_NUMPAD_4, "Num4"},
|
|
{NKCODE_NUMPAD_5, "Num5"},
|
|
{NKCODE_NUMPAD_6, "Num6"},
|
|
{NKCODE_NUMPAD_7, "Num7"},
|
|
{NKCODE_NUMPAD_8, "Num8"},
|
|
{NKCODE_NUMPAD_9, "Num9"},
|
|
|
|
{NKCODE_LANGUAGE_SWITCH, "Language"},
|
|
{NKCODE_MANNER_MODE, "Manner"},
|
|
{NKCODE_3D_MODE, "3D Mode"},
|
|
{NKCODE_CONTACTS, "Contacts"},
|
|
{NKCODE_CALENDAR, "Calendar"},
|
|
{NKCODE_MUSIC, "Music"},
|
|
{NKCODE_CALCULATOR, "Calc"},
|
|
{NKCODE_ZENKAKU_HANKAKU, "Zenkaku"},
|
|
{NKCODE_EISU, "Eisu"},
|
|
{NKCODE_MUHENKAN, "Muhenkan"},
|
|
{NKCODE_HENKAN, "Henkan"},
|
|
{NKCODE_KATAKANA_HIRAGANA, "Katakana"},
|
|
{NKCODE_YEN, "Yen"},
|
|
{NKCODE_RO, "Ro"},
|
|
{NKCODE_KANA, "Kana"},
|
|
{NKCODE_ASSIST, "Assist"},
|
|
|
|
{NKCODE_EXT_MOUSEBUTTON_1, "MB1"},
|
|
{NKCODE_EXT_MOUSEBUTTON_2, "MB2"},
|
|
{NKCODE_EXT_MOUSEBUTTON_3, "MB3"},
|
|
{NKCODE_EXT_MOUSEBUTTON_4, "MB4"},
|
|
{NKCODE_EXT_MOUSEBUTTON_5, "MB5"},
|
|
{NKCODE_EXT_MOUSEWHEEL_UP, "MWheelU"},
|
|
{NKCODE_EXT_MOUSEWHEEL_DOWN, "MWheelD"},
|
|
|
|
|
|
{NKCODE_EXT_MOTION_UP, "MotionUp"},
|
|
{NKCODE_EXT_MOTION_DOWN, "MotionDown"},
|
|
{NKCODE_EXT_MOTION_LEFT, "MotionLeft"},
|
|
{NKCODE_EXT_MOTION_RIGHT, "MotionRight"},
|
|
{NKCODE_EXT_MOTION_FORWARD, "MotionFwd"},
|
|
{NKCODE_EXT_ROTATION_UP, "RotationUp"},
|
|
{NKCODE_EXT_ROTATION_DOWN, "RotationDown"},
|
|
{NKCODE_EXT_ROTATION_LEFT, "RotationLeft"},
|
|
{NKCODE_EXT_ROTATION_RIGHT, "RotationRight"},
|
|
|
|
{NKCODE_START_QUESTION, "¿"},
|
|
{NKCODE_LEFTBRACE, "{"},
|
|
{NKCODE_RIGHTBRACE, "}"},
|
|
};
|
|
|
|
static const KeyMap_IntStrPair axis_names[] = {
|
|
{JOYSTICK_AXIS_X, "X Axis"},
|
|
{JOYSTICK_AXIS_Y, "Y Axis"},
|
|
{JOYSTICK_AXIS_PRESSURE, "Pressure"},
|
|
{JOYSTICK_AXIS_SIZE, "Size"},
|
|
{JOYSTICK_AXIS_TOUCH_MAJOR, "Touch Major"},
|
|
{JOYSTICK_AXIS_TOUCH_MINOR, "Touch Minor"},
|
|
{JOYSTICK_AXIS_TOOL_MAJOR, "Tool Major"},
|
|
{JOYSTICK_AXIS_TOOL_MINOR, "Tool Minor"},
|
|
{JOYSTICK_AXIS_ORIENTATION, "Orient"},
|
|
{JOYSTICK_AXIS_VSCROLL, "Vert Scroll"},
|
|
{JOYSTICK_AXIS_HSCROLL, "Horiz Scroll"},
|
|
{JOYSTICK_AXIS_Z, "Z Axis"}, // Also used as second stick X on many controllers - rename?
|
|
{JOYSTICK_AXIS_RX, "X Rotation"},
|
|
{JOYSTICK_AXIS_RY, "Y Rotation"},
|
|
{JOYSTICK_AXIS_RZ, "Z Rotation"}, // Also used as second stick Y on many controllers - rename?
|
|
{JOYSTICK_AXIS_HAT_X, "X HAT"},
|
|
{JOYSTICK_AXIS_HAT_Y, "Y HAT"},
|
|
{JOYSTICK_AXIS_LTRIGGER, "TriggerL"},
|
|
{JOYSTICK_AXIS_RTRIGGER, "TriggerR"},
|
|
{JOYSTICK_AXIS_THROTTLE, "Throttle"},
|
|
{JOYSTICK_AXIS_RUDDER, "Rudder"},
|
|
{JOYSTICK_AXIS_WHEEL, "Wheel"},
|
|
{JOYSTICK_AXIS_GAS, "Gas"},
|
|
{JOYSTICK_AXIS_BRAKE, "Brake"},
|
|
{JOYSTICK_AXIS_DISTANCE, "Distance"},
|
|
{JOYSTICK_AXIS_TILT, "Tilt"},
|
|
{JOYSTICK_AXIS_MOUSE_REL_X, "MouseDX"},
|
|
{JOYSTICK_AXIS_MOUSE_REL_Y, "MouseDY"},
|
|
{JOYSTICK_AXIS_ACCELEROMETER_X, "AccelX"},
|
|
{JOYSTICK_AXIS_ACCELEROMETER_Y, "AccelY"},
|
|
{JOYSTICK_AXIS_ACCELEROMETER_Z, "AccelZ"},
|
|
};
|
|
|
|
static std::string unknown_key_name = "??";
|
|
|
|
const KeyMap_IntStrPair psp_button_names[] = {
|
|
{CTRL_UP, "Up"},
|
|
{CTRL_DOWN, "Down"},
|
|
{CTRL_LEFT, "Left"},
|
|
{CTRL_RIGHT, "Right"},
|
|
{CTRL_CIRCLE, "Circle"},
|
|
{CTRL_CROSS, "Cross"},
|
|
{CTRL_SQUARE, "Square"},
|
|
{CTRL_TRIANGLE, "Triangle"},
|
|
{CTRL_START, "Start"},
|
|
{CTRL_SELECT, "Select"},
|
|
{CTRL_LTRIGGER, "L"},
|
|
{CTRL_RTRIGGER, "R"},
|
|
|
|
{VIRTKEY_AXIS_Y_MAX, "An.Up"},
|
|
{VIRTKEY_AXIS_Y_MIN, "An.Down"},
|
|
{VIRTKEY_AXIS_X_MIN, "An.Left"},
|
|
{VIRTKEY_AXIS_X_MAX, "An.Right"},
|
|
{VIRTKEY_ANALOG_LIGHTLY, "Analog limiter"},
|
|
|
|
{VIRTKEY_RAPID_FIRE, "RapidFire"},
|
|
{VIRTKEY_FASTFORWARD, "Fast-forward"},
|
|
{VIRTKEY_SPEED_TOGGLE, "SpeedToggle"},
|
|
{VIRTKEY_SPEED_CUSTOM1, "Alt speed 1"},
|
|
{VIRTKEY_SPEED_CUSTOM2, "Alt speed 2"},
|
|
{VIRTKEY_SPEED_ANALOG, "Analog speed"},
|
|
{VIRTKEY_PAUSE, "Pause"},
|
|
#ifndef MOBILE_DEVICE
|
|
{VIRTKEY_FRAME_ADVANCE, "Frame Advance"},
|
|
{VIRTKEY_RECORD, "Audio/Video Recording" },
|
|
#endif
|
|
{VIRTKEY_REWIND, "Rewind"},
|
|
{VIRTKEY_SAVE_STATE, "Save State"},
|
|
{VIRTKEY_LOAD_STATE, "Load State"},
|
|
{VIRTKEY_NEXT_SLOT, "Next Slot"},
|
|
#if !defined(MOBILE_DEVICE)
|
|
{VIRTKEY_TOGGLE_FULLSCREEN, "Toggle Fullscreen"},
|
|
#endif
|
|
|
|
{VIRTKEY_AXIS_RIGHT_Y_MAX, "RightAn.Up"},
|
|
{VIRTKEY_AXIS_RIGHT_Y_MIN, "RightAn.Down"},
|
|
{VIRTKEY_AXIS_RIGHT_X_MIN, "RightAn.Left"},
|
|
{VIRTKEY_AXIS_RIGHT_X_MAX, "RightAn.Right"},
|
|
{VIRTKEY_OPENCHAT, "OpenChat" },
|
|
|
|
{VIRTKEY_AXIS_SWAP, "AxisSwap"},
|
|
{VIRTKEY_DEVMENU, "DevMenu"},
|
|
{VIRTKEY_TEXTURE_DUMP, "Texture Dumping"},
|
|
{VIRTKEY_TEXTURE_REPLACE, "Texture Replacement"},
|
|
{VIRTKEY_SCREENSHOT, "Screenshot"},
|
|
{VIRTKEY_MUTE_TOGGLE, "Mute toggle"},
|
|
{VIRTKEY_ANALOG_ROTATE_CW, "Rotate Analog (CW)"},
|
|
{VIRTKEY_ANALOG_ROTATE_CCW, "Rotate Analog (CCW)"},
|
|
|
|
#ifdef OPENXR
|
|
{VIRTKEY_VR_CAMERA_ADJUST, "VR camera adjust"},
|
|
{VIRTKEY_VR_CAMERA_RESET, "VR camera reset"},
|
|
#else
|
|
{VIRTKEY_SCREEN_ROTATION_VERTICAL, "Display Portrait"},
|
|
{VIRTKEY_SCREEN_ROTATION_VERTICAL180, "Display Portrait Reversed"},
|
|
{VIRTKEY_SCREEN_ROTATION_HORIZONTAL, "Display Landscape"},
|
|
{VIRTKEY_SCREEN_ROTATION_HORIZONTAL180, "Display Landscape Reversed"},
|
|
#endif
|
|
|
|
{CTRL_HOME, "Home"},
|
|
{CTRL_HOLD, "Hold"},
|
|
{CTRL_WLAN, "Wlan"},
|
|
{CTRL_REMOTE_HOLD, "Remote hold"},
|
|
{CTRL_VOL_UP, "Vol +"},
|
|
{CTRL_VOL_DOWN, "Vol -"},
|
|
{CTRL_SCREEN, "Screen"},
|
|
{CTRL_NOTE, "Note"},
|
|
};
|
|
|
|
const int AXIS_BIND_NKCODE_START = 4000;
|
|
|
|
static std::string FindName(int key, const KeyMap_IntStrPair list[], size_t size) {
|
|
for (size_t i = 0; i < size; i++)
|
|
if (list[i].key == key)
|
|
return list[i].name;
|
|
return StringFromFormat("%02x?", key);
|
|
}
|
|
|
|
std::string GetKeyName(int keyCode) {
|
|
return FindName(keyCode, key_names, ARRAY_SIZE(key_names));
|
|
}
|
|
|
|
std::string GetKeyOrAxisName(int keyCode) {
|
|
if (keyCode >= AXIS_BIND_NKCODE_START) {
|
|
int direction;
|
|
int axis = TranslateKeyCodeToAxis(keyCode, direction);
|
|
std::string temp = GetAxisName(axis);
|
|
if (direction == 1)
|
|
temp += "+";
|
|
else if (direction == -1)
|
|
temp += "-";
|
|
return temp;
|
|
}
|
|
return FindName(keyCode, key_names, ARRAY_SIZE(key_names));
|
|
}
|
|
|
|
std::string GetAxisName(int axisId) {
|
|
return FindName(axisId, axis_names, ARRAY_SIZE(axis_names));
|
|
}
|
|
|
|
std::string GetPspButtonName(int btn) {
|
|
return FindName(btn, psp_button_names, ARRAY_SIZE(psp_button_names));
|
|
}
|
|
|
|
const char* GetPspButtonNameCharPointer(int btn) {
|
|
for (size_t i = 0; i < ARRAY_SIZE(psp_button_names); i++)
|
|
if (psp_button_names[i].key == btn)
|
|
return psp_button_names[i].name;
|
|
return nullptr;
|
|
}
|
|
|
|
std::vector<KeyMap_IntStrPair> GetMappableKeys() {
|
|
std::vector<KeyMap_IntStrPair> temp;
|
|
for (size_t i = 0; i < ARRAY_SIZE(psp_button_names); i++) {
|
|
temp.push_back(psp_button_names[i]);
|
|
}
|
|
return temp;
|
|
}
|
|
|
|
int TranslateKeyCodeToAxis(int keyCode, int &direction) {
|
|
if (keyCode < AXIS_BIND_NKCODE_START)
|
|
return 0;
|
|
|
|
int v = keyCode - AXIS_BIND_NKCODE_START;
|
|
// Even/odd for direction.
|
|
direction = v & 1 ? -1 : 1;
|
|
return v / 2;
|
|
}
|
|
|
|
int TranslateKeyCodeFromAxis(int axisId, int direction) {
|
|
direction = direction < 0 ? 1 : 0;
|
|
return AXIS_BIND_NKCODE_START + axisId * 2 + direction;
|
|
}
|
|
|
|
KeyDef AxisDef(int deviceId, int axisId, int direction) {
|
|
return KeyDef(deviceId, TranslateKeyCodeFromAxis(axisId, direction));
|
|
}
|
|
|
|
int CheckAxisSwap(int btn) {
|
|
if (g_swapped_keys) {
|
|
switch (btn) {
|
|
case CTRL_UP: btn = VIRTKEY_AXIS_Y_MAX;
|
|
break;
|
|
case VIRTKEY_AXIS_Y_MAX: btn = CTRL_UP;
|
|
break;
|
|
case CTRL_DOWN: btn = VIRTKEY_AXIS_Y_MIN;
|
|
break;
|
|
case VIRTKEY_AXIS_Y_MIN: btn = CTRL_DOWN;
|
|
break;
|
|
case CTRL_LEFT: btn = VIRTKEY_AXIS_X_MIN;
|
|
break;
|
|
case VIRTKEY_AXIS_X_MIN: btn = CTRL_LEFT;
|
|
break;
|
|
case CTRL_RIGHT: btn = VIRTKEY_AXIS_X_MAX;
|
|
break;
|
|
case VIRTKEY_AXIS_X_MAX: btn = CTRL_RIGHT;
|
|
break;
|
|
}
|
|
}
|
|
return btn;
|
|
}
|
|
|
|
static bool FindKeyMapping(int deviceId, int key, std::vector<int> *psp_button) {
|
|
// Brute force, let's optimize later
|
|
for (auto iter = g_controllerMap.begin(); iter != g_controllerMap.end(); ++iter) {
|
|
for (auto iter2 = iter->second.begin(); iter2 != iter->second.end(); ++iter2) {
|
|
if (*iter2 == KeyDef(deviceId, key)) {
|
|
psp_button->push_back(CheckAxisSwap(iter->first));
|
|
}
|
|
}
|
|
}
|
|
return psp_button->size() > 0;
|
|
}
|
|
|
|
bool KeyToPspButton(int deviceId, int key, std::vector<int> *pspKeys) {
|
|
return FindKeyMapping(deviceId, key, pspKeys);
|
|
}
|
|
|
|
// TODO: vector output
|
|
bool KeyFromPspButton(int btn, std::vector<KeyDef> *keys, bool ignoreMouse) {
|
|
for (auto iter = g_controllerMap.begin(); iter != g_controllerMap.end(); ++iter) {
|
|
if (iter->first == btn) {
|
|
for (auto iter2 = iter->second.begin(); iter2 != iter->second.end(); ++iter2) {
|
|
if (!ignoreMouse || iter2->deviceId != DEVICE_ID_MOUSE)
|
|
keys->push_back(*iter2);
|
|
}
|
|
}
|
|
}
|
|
return keys->size() > 0;
|
|
}
|
|
|
|
bool AxisToPspButton(int deviceId, int axisId, int direction, std::vector<int> *pspKeys) {
|
|
int key = TranslateKeyCodeFromAxis(axisId, direction);
|
|
return KeyToPspButton(deviceId, key, pspKeys);
|
|
}
|
|
|
|
bool AxisFromPspButton(int btn, int *deviceId, int *axisId, int *direction) {
|
|
for (auto iter = g_controllerMap.begin(); iter != g_controllerMap.end(); ++iter) {
|
|
for (auto iter2 = iter->second.begin(); iter2 != iter->second.end(); ++iter2) {
|
|
if (iter->first == btn && iter2->keyCode >= AXIS_BIND_NKCODE_START) {
|
|
if (deviceId)
|
|
*deviceId = iter2->deviceId;
|
|
if (axisId)
|
|
*axisId = TranslateKeyCodeToAxis(iter2->keyCode, *direction);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
MappedAnalogAxes MappedAxesForDevice(int deviceId) {
|
|
MappedAnalogAxes result{};
|
|
|
|
// Find the axisId mapped for a specific virtual button.
|
|
auto findAxisId = [&](int btn) -> MappedAnalogAxis {
|
|
MappedAnalogAxis info{ -1 };
|
|
for (const auto &key : g_controllerMap[btn]) {
|
|
if (key.deviceId == deviceId) {
|
|
info.axisId = TranslateKeyCodeToAxis(key.keyCode, info.direction);
|
|
return info;
|
|
}
|
|
}
|
|
return info;
|
|
};
|
|
|
|
// Find the axisId of a pair of opposing buttons.
|
|
auto findAxisIdPair = [&](int minBtn, int maxBtn) -> MappedAnalogAxis {
|
|
MappedAnalogAxis foundMin = findAxisId(minBtn);
|
|
MappedAnalogAxis foundMax = findAxisId(maxBtn);
|
|
if (foundMin.axisId == foundMax.axisId) {
|
|
return foundMax;
|
|
}
|
|
return MappedAnalogAxis{ -1 };
|
|
};
|
|
|
|
result.leftX = findAxisIdPair(VIRTKEY_AXIS_X_MIN, VIRTKEY_AXIS_X_MAX);
|
|
result.leftY = findAxisIdPair(VIRTKEY_AXIS_Y_MIN, VIRTKEY_AXIS_Y_MAX);
|
|
result.rightX = findAxisIdPair(VIRTKEY_AXIS_RIGHT_X_MIN, VIRTKEY_AXIS_RIGHT_X_MAX);
|
|
result.rightY = findAxisIdPair(VIRTKEY_AXIS_RIGHT_Y_MIN, VIRTKEY_AXIS_RIGHT_Y_MAX);
|
|
return result;
|
|
}
|
|
|
|
void RemoveButtonMapping(int btn) {
|
|
for (auto iter = g_controllerMap.begin(); iter != g_controllerMap.end(); ++iter) {
|
|
if (iter->first == btn) {
|
|
g_controllerMap.erase(iter);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool IsKeyMapped(int device, int key) {
|
|
for (auto &iter : g_controllerMap) {
|
|
for (auto &mappedKey : iter.second) {
|
|
if (mappedKey == KeyDef(device, key)) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool ReplaceSingleKeyMapping(int btn, int index, KeyDef key) {
|
|
// Check for duplicate
|
|
for (int i = 0; i < (int)g_controllerMap[btn].size(); ++i) {
|
|
if (i != index && g_controllerMap[btn][i] == key) {
|
|
g_controllerMap[btn].erase(g_controllerMap[btn].begin()+index);
|
|
g_controllerMapGeneration++;
|
|
|
|
UpdateNativeMenuKeys();
|
|
return false;
|
|
}
|
|
}
|
|
KeyMap::g_controllerMap[btn][index] = key;
|
|
g_controllerMapGeneration++;
|
|
|
|
g_seenDeviceIds.insert(key.deviceId);
|
|
UpdateNativeMenuKeys();
|
|
return true;
|
|
}
|
|
|
|
void SetKeyMapping(int btn, KeyDef key, bool replace) {
|
|
if (key.keyCode < 0)
|
|
return;
|
|
if (replace) {
|
|
RemoveButtonMapping(btn);
|
|
g_controllerMap[btn].clear();
|
|
g_controllerMap[btn].push_back(key);
|
|
} else {
|
|
for (auto iter = g_controllerMap[btn].begin(); iter != g_controllerMap[btn].end(); ++iter) {
|
|
if (*iter == key)
|
|
return;
|
|
}
|
|
g_controllerMap[btn].push_back(key);
|
|
}
|
|
g_controllerMapGeneration++;
|
|
|
|
g_seenDeviceIds.insert(key.deviceId);
|
|
UpdateNativeMenuKeys();
|
|
}
|
|
|
|
void SetAxisMapping(int btn, int deviceId, int axisId, int direction, bool replace) {
|
|
int key = TranslateKeyCodeFromAxis(axisId, direction);
|
|
SetKeyMapping(btn, KeyDef(deviceId, key), replace);
|
|
}
|
|
|
|
void RestoreDefault() {
|
|
g_controllerMap.clear();
|
|
g_controllerMapGeneration++;
|
|
|
|
if (IsVREnabled()) {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_VR_HEADSET, false);
|
|
return;
|
|
}
|
|
|
|
#if PPSSPP_PLATFORM(WINDOWS)
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_KEYBOARD, true);
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_XINPUT, false);
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_PAD, false);
|
|
#elif PPSSPP_PLATFORM(ANDROID)
|
|
// Autodetect a few common (and less common) devices
|
|
// Note that here we check the device name, not the controller name. We don't get
|
|
// the controller name until a button has been pressed so can't use it to set defaults.
|
|
std::string name = System_GetProperty(SYSPROP_NAME);
|
|
if (IsNvidiaShield(name)) {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_SHIELD, false);
|
|
} else if (IsOuya(name)) {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_OUYA, false);
|
|
} else if (IsXperiaPlay(name)) {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_XPERIA_PLAY, false);
|
|
} else if (IsMOQII7S(name)) {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_MOQI_I7S, false);
|
|
} else if (IsRetroid(name)) {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_RETRO_STATION_CONTROLLER, false);
|
|
} else {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_ANDROID_PAD, false);
|
|
}
|
|
#else
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_KEYBOARD, true);
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_PAD, false);
|
|
#endif
|
|
}
|
|
|
|
// TODO: Make the ini format nicer.
|
|
void LoadFromIni(IniFile &file) {
|
|
RestoreDefault();
|
|
if (!file.HasSection("ControlMapping")) {
|
|
return;
|
|
}
|
|
|
|
Section *controls = file.GetOrCreateSection("ControlMapping");
|
|
for (size_t i = 0; i < ARRAY_SIZE(psp_button_names); i++) {
|
|
std::string value;
|
|
controls->Get(psp_button_names[i].name, &value, "");
|
|
|
|
// Erase default mapping
|
|
g_controllerMap.erase(psp_button_names[i].key);
|
|
if (value.empty())
|
|
continue;
|
|
|
|
std::vector<std::string> mappings;
|
|
SplitString(value, ',', mappings);
|
|
|
|
for (size_t j = 0; j < mappings.size(); j++) {
|
|
std::vector<std::string> parts;
|
|
SplitString(mappings[j], '-', parts);
|
|
int deviceId = atoi(parts[0].c_str());
|
|
int keyCode = atoi(parts[1].c_str());
|
|
|
|
SetKeyMapping(psp_button_names[i].key, KeyDef(deviceId, keyCode), false);
|
|
g_seenDeviceIds.insert(deviceId);
|
|
}
|
|
}
|
|
|
|
UpdateNativeMenuKeys();
|
|
}
|
|
|
|
void SaveToIni(IniFile &file) {
|
|
Section *controls = file.GetOrCreateSection("ControlMapping");
|
|
|
|
for (size_t i = 0; i < ARRAY_SIZE(psp_button_names); i++) {
|
|
std::vector<KeyDef> keys;
|
|
KeyFromPspButton(psp_button_names[i].key, &keys, false);
|
|
|
|
std::string value;
|
|
for (size_t j = 0; j < keys.size(); j++) {
|
|
char temp[128];
|
|
sprintf(temp, "%i-%i", keys[j].deviceId, keys[j].keyCode);
|
|
value += temp;
|
|
if (j != keys.size() - 1)
|
|
value += ",";
|
|
}
|
|
|
|
controls->Set(psp_button_names[i].name, value, "");
|
|
}
|
|
}
|
|
|
|
bool IsOuya(const std::string &name) {
|
|
return name == "OUYA:OUYA Console";
|
|
}
|
|
|
|
bool IsNvidiaShield(const std::string &name) {
|
|
return name == "NVIDIA:SHIELD";
|
|
}
|
|
|
|
bool IsRetroid(const std::string &name) {
|
|
// TODO: Not sure if there are differences between different Retroid devices.
|
|
// The one I have is a "Retroid Pocket 2+".
|
|
return startsWith(name, "Retroid:");
|
|
}
|
|
|
|
bool IsNvidiaShieldTV(const std::string &name) {
|
|
return name == "NVIDIA:SHIELD Android TV";
|
|
}
|
|
|
|
bool IsXperiaPlay(const std::string &name) {
|
|
return name == "Sony Ericsson:R800a" || name == "Sony Ericsson:R800i" || name == "Sony Ericsson:R800x" || name == "Sony Ericsson:R800at" || name == "Sony Ericsson:SO-01D" || name == "Sony Ericsson:zeus";
|
|
}
|
|
|
|
bool IsMOQII7S(const std::string &name) {
|
|
return name == "MOQI:I7S";
|
|
}
|
|
|
|
bool HasBuiltinController(const std::string &name) {
|
|
return IsOuya(name) || IsXperiaPlay(name) || IsNvidiaShield(name) || IsMOQII7S(name) || IsRetroid(name);
|
|
}
|
|
|
|
void NotifyPadConnected(int deviceId, const std::string &name) {
|
|
g_seenPads.insert(name);
|
|
g_padNames[deviceId] = name;
|
|
}
|
|
|
|
void AutoConfForPad(const std::string &name) {
|
|
g_controllerMap.clear();
|
|
|
|
INFO_LOG(SYSTEM, "Autoconfiguring pad for '%s'", name.c_str());
|
|
|
|
#if PPSSPP_PLATFORM(ANDROID)
|
|
if (name.find("Xbox") != std::string::npos) {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_ANDROID_XBOX, false);
|
|
} else if (name == "Retro Station Controller") {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_RETRO_STATION_CONTROLLER, false);
|
|
}
|
|
#else
|
|
// TODO: Should actually check for XInput?
|
|
if (name.find("Xbox") != std::string::npos) {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_XINPUT, false);
|
|
} else {
|
|
SetDefaultKeyMap(DEFAULT_MAPPING_PAD, false);
|
|
}
|
|
#endif
|
|
|
|
// Add a couple of convenient keyboard mappings by default, too.
|
|
g_controllerMap[VIRTKEY_PAUSE].push_back(KeyDef(DEVICE_ID_KEYBOARD, NKCODE_ESCAPE));
|
|
g_controllerMap[VIRTKEY_FASTFORWARD].push_back(KeyDef(DEVICE_ID_KEYBOARD, NKCODE_TAB));
|
|
g_controllerMapGeneration++;
|
|
}
|
|
|
|
const std::set<std::string> &GetSeenPads() {
|
|
return g_seenPads;
|
|
}
|
|
|
|
std::string PadName(int deviceId) {
|
|
auto it = g_padNames.find(deviceId);
|
|
if (it != g_padNames.end())
|
|
return it->second;
|
|
return "";
|
|
}
|
|
|
|
// Swap direction buttons and left analog axis
|
|
void SwapAxis() {
|
|
g_swapped_keys = !g_swapped_keys;
|
|
}
|
|
|
|
bool HasChanged(int &prevGeneration) {
|
|
if (prevGeneration != g_controllerMapGeneration) {
|
|
prevGeneration = g_controllerMapGeneration;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
} // KeyMap
|