// Copyright (C) 2003 Dolphin 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. // 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 SVN repository and contact information can be found at // http://code.google.com/p/dolphin-emu/ #include #include "GLInterface.h" #include "Android/TextureLoader.h" #include "Android/ButtonManager.h" extern void DrawButton(GLuint tex, float *coords); namespace ButtonManager { // Pair key is padID, BUTTONTYPE std::map, Button*> m_buttons; std::map, Axis*> m_axises; std::unordered_map m_controllers; const char *configStrings[] = { "InputA", "InputB", "InputStart", "InputX", "InputY", "InputZ", "DPadUp", "DPadDown", "DPadLeft", "DPadRight", "MainUp", "MainDown", "MainLeft", "MainRight", "CStickUp", "CStickDown", "CStickLeft", "CStickRight", "InputL", "InputR" }; const int configStringNum = 20; void AddBind(std::string dev, sBind *bind) { auto it = m_controllers.find(dev); if (it != m_controllers.end()) { it->second->AddBind(bind); return; } m_controllers[dev] = new InputDevice(dev); m_controllers[dev]->AddBind(bind); } void Init() { // Initialize our touchscreen buttons for (int a = 0; a < 4; ++a) { m_buttons[std::make_pair(a, BUTTON_A)] = new Button(); m_buttons[std::make_pair(a, BUTTON_B)] = new Button(); m_buttons[std::make_pair(a, BUTTON_START)] = new Button(); m_buttons[std::make_pair(a, BUTTON_X)] = new Button(); m_buttons[std::make_pair(a, BUTTON_Y)] = new Button(); m_buttons[std::make_pair(a, BUTTON_Z)] = new Button(); m_buttons[std::make_pair(a, BUTTON_UP)] = new Button(); m_buttons[std::make_pair(a, BUTTON_DOWN)] = new Button(); m_buttons[std::make_pair(a, BUTTON_LEFT)] = new Button(); m_buttons[std::make_pair(a, BUTTON_RIGHT)] = new Button(); m_axises[std::make_pair(a, STICK_MAIN_UP)] = new Axis(); m_axises[std::make_pair(a, STICK_MAIN_DOWN)] = new Axis(); m_axises[std::make_pair(a, STICK_MAIN_LEFT)] = new Axis(); m_axises[std::make_pair(a, STICK_MAIN_RIGHT)] = new Axis(); m_axises[std::make_pair(a, STICK_C_UP)] = new Axis(); m_axises[std::make_pair(a, STICK_C_DOWN)] = new Axis(); m_axises[std::make_pair(a, STICK_C_LEFT)] = new Axis(); m_axises[std::make_pair(a, STICK_C_RIGHT)] = new Axis(); m_axises[std::make_pair(a, TRIGGER_L)] = new Axis(); m_axises[std::make_pair(a, TRIGGER_R)] = new Axis(); } // Init our controller bindings IniFile ini; ini.Load(File::GetUserPath(D_CONFIG_IDX) + std::string("Dolphin.ini")); for (int a = 0; a < configStringNum; ++a) { BindType type; int bindnum; char dev[128]; bool hasbind = false; char modifier = 0; std::string value; ini.Get("Android", configStrings[a], &value, "None"); if (value == "None") continue; if (std::string::npos != value.find("Axis")) { hasbind = true; type = BIND_AXIS; sscanf(value.c_str(), "Device '%[^\']'-Axis %d%c", dev, &bindnum, &modifier); } else if (std::string::npos != value.find("Button")) { hasbind = true; type = BIND_BUTTON; sscanf(value.c_str(), "Device '%[^\']'-Button %d", dev, &bindnum); } if (hasbind) AddBind(std::string(dev), new sBind((ButtonType)a, type, bindnum, modifier == '-' ? -1.0f : 1.0f)); } } bool GetButtonPressed(int padID, ButtonType button) { bool pressed = false; pressed = m_buttons[std::make_pair(padID, button)]->Pressed(); for (auto it = m_controllers.begin(); it != m_controllers.end(); ++it) pressed |= it->second->ButtonValue(button); return pressed; } float GetAxisValue(int padID, ButtonType axis) { float value = 0.0f; value = m_axises[std::make_pair(padID, axis)]->AxisValue(); auto it = m_controllers.begin(); if (it == m_controllers.end()) return value; return it->second->AxisValue(axis); } void TouchEvent(int padID, int button, int action) { m_buttons[std::make_pair(padID, button)]->SetState(action ? BUTTON_PRESSED : BUTTON_RELEASED); } void TouchAxisEvent(int padID, int axis, float value) { m_axises[std::make_pair(padID, axis)]->SetValue(value); } void GamepadEvent(std::string dev, int button, int action) { auto it = m_controllers.find(dev); if (it != m_controllers.end()) { it->second->PressEvent(button, action); return; } m_controllers[dev] = new InputDevice(dev); m_controllers[dev]->PressEvent(button, action); } void GamepadAxisEvent(std::string dev, int axis, float value) { auto it = m_controllers.find(dev); if (it != m_controllers.end()) { it->second->AxisEvent(axis, value); return; } m_controllers[dev] = new InputDevice(dev); m_controllers[dev]->AxisEvent(axis, value); } void Shutdown() { for(auto it = m_buttons.begin(); it != m_buttons.end(); ++it) delete it->second; for (auto it = m_controllers.begin(); it != m_controllers.end(); ++it) delete it->second; m_controllers.clear(); m_buttons.clear(); } // InputDevice void InputDevice::PressEvent(int button, int action) { m_buttons[button] = action == 0 ? true : false; } void InputDevice::AxisEvent(int axis, float value) { m_axises[axis] = value; } bool InputDevice::ButtonValue(ButtonType button) { auto it = m_binds.find(button); if (it == m_binds.end()) return false; if (it->second->m_bindtype == BIND_BUTTON) return m_buttons[it->second->m_bind]; else return AxisValue(button); } float InputDevice::AxisValue(ButtonType axis) { auto it = m_binds.find(axis); if (it == m_binds.end()) return 0.0f; if (it->second->m_bindtype == BIND_BUTTON) return ButtonValue(axis); else return m_axises[it->second->m_bind] * it->second->m_neg; } }