/* RetroArch - A frontend for libretro. * Copyright (C) 2011-2015 - 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 . */ #include #include #include "input_common.h" #include "input_keymaps.h" #include "drivers/apple_input.h" #include "../general.h" #include "../driver.h" #include "drivers/apple_keycode.h" #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 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 } }; driver_t *driver = driver_get_ptr(); apple_input_data_t *apple = (apple_input_data_t*)driver->input_data; unsigned translated_code = 0; if (!map_initialized) { int i; for (i = 0; mapping_def[i].orig; i ++) mapping[mapping_def[i].orig] = mapping_def[i].mod; map_initialized = true; } if (*code == KEY_RightShift) { apple->small_keyboard_active = down; *code = 0; return true; } translated_code = (*code < 128) ? mapping[*code] : 0; /* Allow old keys to be released. */ if (!down && apple->key_state[*code]) return false; if ((!down && apple->key_state[translated_code]) || apple->small_keyboard_active) { *code = translated_code; return true; } return false; } 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 }; driver_t *driver = driver_get_ptr(); apple_input_data_t *apple = (apple_input_data_t*)driver->input_data; if (apple->icade_enabled && (keycode < 0x20) && (icade_map[keycode].button >= 0)) { const int button = icade_map[keycode].button; if (icade_map[keycode].up) BIT32_CLEAR(apple->icade_buttons, button); else BIT32_SET(apple->icade_buttons, button); } } void apple_input_keyboard_event(bool down, unsigned code, uint32_t character, uint32_t mod, unsigned device) { driver_t *driver = driver_get_ptr(); apple_input_data_t *apple = (apple_input_data_t*)driver->input_data; if (!apple) return; code = HIDKEY(code); if (apple->icade_enabled) { handle_icade_event(code); return; } if (apple->small_keyboard_enabled && handle_small_keyboard(&code, down)) character = 0; if (code == 0 || code >= MAX_KEYS) return; apple->key_state[code] = down; input_keyboard_event(down, input_keymaps_translate_keysym_to_rk(code), character, (enum retro_mod)mod, device); }