RetroArch/input/apple_input.c
2014-10-04 16:34:52 +02:00

576 lines
15 KiB
C

/* RetroArch - A frontend for libretro.
* Copyright (C) 2011-2014 - Daniel De Matteis
* Copyright (C) 2013-2014 - Jason Fetters
*
* RetroArch 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 Found-
* ation, either version 3 of the License, or (at your option) any later version.
*
* RetroArch 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 for more details.
*
* You should have received a copy of the GNU General Public License along with RetroArch.
* If not, see <http://www.gnu.org/licenses/>.
*/
#include <stdint.h>
#include <unistd.h>
#include "input_common.h"
#include "apple_input.h"
#include "../general.h"
#include "../driver.h"
#include "apple_keycode.h"
const struct apple_key_name_map_entry apple_key_name_map[] =
{
{ "left", KEY_Left },
{ "right", KEY_Right },
{ "up", KEY_Up },
{ "down", KEY_Down },
{ "enter", KEY_Enter },
{ "kp_enter", KP_Enter },
{ "space", KEY_Space },
{ "tab", KEY_Tab },
{ "shift", KEY_LeftShift },
{ "rshift", KEY_RightShift },
{ "ctrl", KEY_LeftControl },
{ "alt", KEY_LeftAlt },
{ "escape", KEY_Escape },
{ "backspace", KEY_DeleteForward },
{ "backquote", KEY_Grave },
{ "pause", KEY_Pause },
{ "f1", KEY_F1 },
{ "f2", KEY_F2 },
{ "f3", KEY_F3 },
{ "f4", KEY_F4 },
{ "f5", KEY_F5 },
{ "f6", KEY_F6 },
{ "f7", KEY_F7 },
{ "f8", KEY_F8 },
{ "f9", KEY_F9 },
{ "f10", KEY_F10 },
{ "f11", KEY_F11 },
{ "f12", KEY_F12 },
{ "num0", KEY_0 },
{ "num1", KEY_1 },
{ "num2", KEY_2 },
{ "num3", KEY_3 },
{ "num4", KEY_4 },
{ "num5", KEY_5 },
{ "num6", KEY_6 },
{ "num7", KEY_7 },
{ "num8", KEY_8 },
{ "num9", KEY_9 },
{ "insert", KEY_Insert },
{ "del", KEY_DeleteForward },
{ "home", KEY_Home },
{ "end", KEY_End },
{ "pageup", KEY_PageUp },
{ "pagedown", KEY_PageDown },
{ "add", KP_Add },
{ "subtract", KP_Subtract },
{ "multiply", KP_Multiply },
{ "divide", KP_Divide },
{ "keypad0", KP_0 },
{ "keypad1", KP_1 },
{ "keypad2", KP_2 },
{ "keypad3", KP_3 },
{ "keypad4", KP_4 },
{ "keypad5", KP_5 },
{ "keypad6", KP_6 },
{ "keypad7", KP_7 },
{ "keypad8", KP_8 },
{ "keypad9", KP_9 },
{ "period", KEY_Period },
{ "capslock", KEY_CapsLock },
{ "numlock", KP_NumLock },
{ "print_screen", KEY_PrintScreen },
{ "scroll_lock", KEY_ScrollLock },
{ "a", KEY_A },
{ "b", KEY_B },
{ "c", KEY_C },
{ "d", KEY_D },
{ "e", KEY_E },
{ "f", KEY_F },
{ "g", KEY_G },
{ "h", KEY_H },
{ "i", KEY_I },
{ "j", KEY_J },
{ "k", KEY_K },
{ "l", KEY_L },
{ "m", KEY_M },
{ "n", KEY_N },
{ "o", KEY_O },
{ "p", KEY_P },
{ "q", KEY_Q },
{ "r", KEY_R },
{ "s", KEY_S },
{ "t", KEY_T },
{ "u", KEY_U },
{ "v", KEY_V },
{ "w", KEY_W },
{ "x", KEY_X },
{ "y", KEY_Y },
{ "z", KEY_Z },
{ "nul", 0x00},
};
#if defined(IOS)
#define HIDKEY(X) X
#elif defined(OSX)
/* Taken from https://github.com/depp/keycode,
* check keycode.h for license. */
static const unsigned char MAC_NATIVE_TO_HID[128] = {
4, 22, 7, 9, 11, 10, 29, 27, 6, 25,255, 5, 20, 26, 8, 21,
28, 23, 30, 31, 32, 33, 35, 34, 46, 38, 36, 45, 37, 39, 48, 18,
24, 47, 12, 19, 40, 15, 13, 52, 14, 51, 49, 54, 56, 17, 16, 55,
43, 44, 53, 42,255, 41,231,227,225, 57,226,224,229,230,228,255,
108, 99,255, 85,255, 87,255, 83,255,255,255, 84, 88,255, 86,109,
110,103, 98, 89, 90, 91, 92, 93, 94, 95,111, 96, 97,255,255,255,
62, 63, 64, 60, 65, 66,255, 68,255,104,107,105,255, 67,255, 69,
255,106,117, 74, 75, 76, 61, 77, 59, 78, 58, 80, 79, 81, 82,255
};
#define HIDKEY(X) (X < 128) ? MAC_NATIVE_TO_HID[X] : 0
#endif
static bool icade_enabled;
static bool small_keyboard_enabled;
static bool small_keyboard_active;
static uint32_t icade_buttons;
static bool handle_small_keyboard(unsigned* code, bool down)
{
static uint8_t mapping[128];
static bool map_initialized;
static const struct { uint8_t orig; uint8_t mod; } mapping_def[] =
{
{ KEY_Grave, KEY_Escape }, { KEY_1, KEY_F1 },
{ KEY_2, KEY_F2 }, { KEY_3, KEY_F3 },
{ KEY_4, KEY_F4 }, { KEY_5, KEY_F5 },
{ KEY_6, KEY_F6 }, { KEY_7, KEY_F7 },
{ KEY_8, KEY_F8 }, { KEY_9, KEY_F9 },
{ KEY_0, KEY_F10 }, { KEY_Minus, KEY_F11 },
{ KEY_Equals, KEY_F12 }, { KEY_Up, KEY_PageUp },
{ KEY_Down, KEY_PageDown }, { KEY_Left, KEY_Home },
{ KEY_Right, KEY_End }, { KEY_Q, KP_7 },
{ KEY_W, KP_8 }, { KEY_E, KP_9 },
{ KEY_A, KP_4 }, { KEY_S, KP_5 },
{ KEY_D, KP_6 }, { KEY_Z, KP_1 },
{ KEY_X, KP_2 }, { KEY_C, KP_3 },
{ 0 }
};
apple_input_data_t *apple = (apple_input_data_t*)driver.input_data;
unsigned translated_code = 0;
if (!map_initialized)
{
for (int i = 0; mapping_def[i].orig; i ++)
mapping[mapping_def[i].orig] = mapping_def[i].mod;
map_initialized = true;
}
if (*code == KEY_RightShift)
{
small_keyboard_active = down;
*code = 0;
return true;
}
translated_code = (*code < 128) ? mapping[*code] : 0;
/* Allow old keys to be released. */
if (!down && apple->keys[*code])
return false;
if ((!down && apple->keys[translated_code]) ||
small_keyboard_active)
{
*code = translated_code;
return true;
}
return false;
}
void apple_input_enable_small_keyboard(bool on)
{
small_keyboard_enabled = on;
}
static void handle_icade_event(unsigned keycode)
{
static const struct
{
bool up;
int button;
} icade_map[0x20] =
{
{ false, -1 }, { false, -1 }, { false, -1 }, { false, -1 }, // 0
{ false, 2 }, { false, -1 }, { true , 3 }, { false, 3 }, // 4
{ true , 0 }, { true, 5 }, { true , 7 }, { false, 8 }, // 8
{ false, 6 }, { false, 9 }, { false, 10 }, { false, 11 }, // C
{ true , 6 }, { true , 9 }, { false, 7 }, { true, 10 }, // 0
{ true , 2 }, { true , 8 }, { false, -1 }, { true , 4 }, // 4
{ false, 5 }, { true , 11 }, { false, 0 }, { false, 1 }, // 8
{ false, 4 }, { true , 1 }, { false, -1 }, { false, -1 } // C
};
if (icade_enabled && (keycode < 0x20)
&& (icade_map[keycode].button >= 0))
{
const int button = icade_map[keycode].button;
if (icade_map[keycode].up)
icade_buttons &= ~(1 << button);
else
icade_buttons |= (1 << button);
}
}
void apple_input_enable_icade(bool on)
{
icade_enabled = on;
icade_buttons = 0;
}
uint32_t apple_input_get_icade_buttons(void)
{
return icade_enabled ? icade_buttons : 0;
}
void apple_input_reset_icade_buttons(void)
{
icade_buttons = 0;
}
/* This is copied here as it isn't
* defined in any standard iOS header */
enum
{
NSAlphaShiftKeyMask = 1 << 16,
NSShiftKeyMask = 1 << 17,
NSControlKeyMask = 1 << 18,
NSAlternateKeyMask = 1 << 19,
NSCommandKeyMask = 1 << 20,
NSNumericPadKeyMask = 1 << 21,
NSHelpKeyMask = 1 << 22,
NSFunctionKeyMask = 1 << 23,
NSDeviceIndependentModifierFlagsMask = 0xffff0000U
};
void apple_input_keyboard_event(bool down,
unsigned code, uint32_t character, uint32_t mod)
{
apple_input_data_t *apple = (apple_input_data_t*)driver.input_data;
enum retro_mod mods = RETROKMOD_NONE;
code = HIDKEY(code);
if (icade_enabled)
{
handle_icade_event(code);
return;
}
if (small_keyboard_enabled && handle_small_keyboard(&code, down))
character = 0;
if (code == 0 || code >= MAX_KEYS)
return;
apple->keys[code] = down;
mods |= (mod & NSAlphaShiftKeyMask) ? RETROKMOD_CAPSLOCK : 0;
mods |= (mod & NSShiftKeyMask) ? RETROKMOD_SHIFT : 0;
mods |= (mod & NSControlKeyMask) ? RETROKMOD_CTRL : 0;
mods |= (mod & NSAlternateKeyMask) ? RETROKMOD_ALT : 0;
mods |= (mod & NSCommandKeyMask) ? RETROKMOD_META : 0;
mods |= (mod & NSNumericPadKeyMask) ? RETROKMOD_NUMLOCK : 0;
input_keyboard_event(down,
input_translate_keysym_to_rk(code), character, mods);
}
int32_t apple_input_find_any_key(void)
{
unsigned i;
apple_input_data_t *apple = (apple_input_data_t*)driver.input_data;
if (!apple)
return 0;
if (apple->joypad)
apple->joypad->poll();
input_init_keyboard_lut(rarch_key_map_apple_hid);
for (i = 0; apple_key_name_map[i].hid_id; i++)
if (apple->keys[apple_key_name_map[i].hid_id])
return apple_key_name_map[i].hid_id;
return 0;
}
int32_t apple_input_find_any_button(uint32_t port)
{
unsigned i, buttons = 0;
apple_input_data_t *apple = (apple_input_data_t*)driver.input_data;
if (!apple)
return -1;
if (apple->joypad)
apple->joypad->poll();
buttons = apple->buttons[port];
if (port == 0)
buttons |= apple_input_get_icade_buttons();
if (buttons)
for (i = 0; i != 32; i ++)
if (buttons & (1 << i))
return i;
return -1;
}
int32_t apple_input_find_any_axis(uint32_t port)
{
int i;
apple_input_data_t *apple = (apple_input_data_t*)driver.input_data;
if (apple && apple->joypad)
apple->joypad->poll();
for (i = 0; i < 4; i++)
{
int16_t value = apple->axes[port][i];
if (abs(value) > 0x4000)
return (value < 0) ? -(i + 1) : i + 1;
}
return 0;
}
static bool apple_key_pressed(apple_input_data_t *apple,
enum retro_key id)
{
if (id < RETROK_LAST)
return apple->keys[input_translate_rk_to_keysym((enum retro_key)id)];
return false;
}
static int16_t apple_input_is_pressed(apple_input_data_t *apple, unsigned port_num,
const struct retro_keybind *binds, unsigned id)
{
if (id < RARCH_BIND_LIST_END)
{
const struct retro_keybind *bind = &binds[id];
return bind->valid && apple_key_pressed(apple, bind->key);
}
return 0;
}
static void *apple_input_init(void)
{
apple_input_data_t *apple = NULL;
if (driver.input_data)
return driver.input_data;
apple = (apple_input_data_t*)calloc(1, sizeof(*apple));
if (!apple)
return NULL;
input_init_keyboard_lut(rarch_key_map_apple_hid);
apple->joypad = input_joypad_init_driver(g_settings.input.joypad_driver);
driver.input_data_own = true;
return apple;
}
static void apple_input_poll(void *data)
{
uint32_t i;
apple_input_data_t *apple = (apple_input_data_t*)data;
if (!apple)
return;
for (i = 0; i < apple->touch_count; i++)
{
input_translate_coord_viewport(apple->touches[i].screen_x,
apple->touches[i].screen_y,
&apple->touches[i].fixed_x,
&apple->touches[i].fixed_y,
&apple->touches[i].full_x,
&apple->touches[i].full_y);
}
if (apple->joypad)
apple->joypad->poll();
apple->buttons[0] |= apple_input_get_icade_buttons();
apple->mouse_delta[0] = 0;
apple->mouse_delta[1] = 0;
}
static int16_t apple_mouse_state(apple_input_data_t *apple,
unsigned id)
{
switch (id)
{
case RETRO_DEVICE_ID_MOUSE_X:
return apple->mouse_delta[0];
case RETRO_DEVICE_ID_MOUSE_Y:
return apple->mouse_delta[1];
case RETRO_DEVICE_ID_MOUSE_LEFT:
return apple->mouse_buttons & 1;
case RETRO_DEVICE_ID_MOUSE_RIGHT:
return apple->mouse_buttons & 2;
}
return 0;
}
static int16_t apple_pointer_state(apple_input_data_t *apple,
unsigned device, unsigned index, unsigned id)
{
const bool want_full = (device == RARCH_DEVICE_POINTER_SCREEN);
if (index < apple->touch_count && index < MAX_TOUCHES)
{
const apple_touch_data_t *touch = (const apple_touch_data_t *)
&apple->touches[index];
int16_t x = want_full ? touch->full_x : touch->fixed_x;
int16_t y = want_full ? touch->full_y : touch->fixed_y;
switch (id)
{
case RETRO_DEVICE_ID_POINTER_PRESSED:
return (x != -0x8000) && (y != -0x8000);
case RETRO_DEVICE_ID_POINTER_X:
return x;
case RETRO_DEVICE_ID_POINTER_Y:
return y;
}
}
return 0;
}
static int16_t apple_input_state(void *data,
const struct retro_keybind **binds, unsigned port,
unsigned device, unsigned index, unsigned id)
{
apple_input_data_t *apple = (apple_input_data_t*)data;
switch (device)
{
case RETRO_DEVICE_JOYPAD:
return apple_input_is_pressed(apple, port, binds[port], id) ||
input_joypad_pressed(apple->joypad, port, binds[port], id);
case RETRO_DEVICE_ANALOG:
return input_joypad_analog(apple->joypad, port,
index, id, binds[port]);
case RETRO_DEVICE_KEYBOARD:
return apple_key_pressed(apple, id);
case RETRO_DEVICE_MOUSE:
return apple_mouse_state(apple, id);
case RETRO_DEVICE_POINTER:
case RARCH_DEVICE_POINTER_SCREEN:
return apple_pointer_state(apple, device, index, id);
}
return 0;
}
static bool apple_input_bind_button_pressed(void *data, int key)
{
apple_input_data_t *apple = (apple_input_data_t*)data;
return apple_input_is_pressed(apple, 0, g_settings.input.binds[0], key) ||
input_joypad_pressed(apple->joypad, 0, g_settings.input.binds[0], key);
}
static void apple_input_free(void *data)
{
apple_input_data_t *apple = (apple_input_data_t*)data;
if (!apple || !data)
return;
if (apple->joypad)
apple->joypad->destroy();
free(apple);
}
static bool apple_input_set_rumble(void *data,
unsigned port, enum retro_rumble_effect effect, uint16_t strength)
{
apple_input_data_t *apple = (apple_input_data_t*)data;
if (apple && apple->joypad)
return input_joypad_set_rumble(apple->joypad,
port, effect, strength);
return false;
}
static uint64_t apple_input_get_capabilities(void *data)
{
(void)data;
return
(1 << RETRO_DEVICE_JOYPAD) |
(1 << RETRO_DEVICE_MOUSE) |
(1 << RETRO_DEVICE_KEYBOARD) |
(1 << RETRO_DEVICE_POINTER) |
(1 << RETRO_DEVICE_ANALOG);
}
static void apple_input_grab_mouse(void *data, bool state)
{
/* Dummy for now. Might be useful in the future. */
(void)data;
(void)state;
}
static const rarch_joypad_driver_t *apple_input_get_joypad_driver(void *data)
{
apple_input_data_t *apple = (apple_input_data_t*)data;
if (apple && apple->joypad)
return apple->joypad;
return NULL;
}
input_driver_t input_apple = {
apple_input_init,
apple_input_poll,
apple_input_state,
apple_input_bind_button_pressed,
apple_input_free,
NULL,
NULL,
apple_input_get_capabilities,
"apple_input",
apple_input_grab_mouse,
apple_input_set_rumble,
apple_input_get_joypad_driver
};