2013-06-18 12:09:20 +00:00
|
|
|
// 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/
|
|
|
|
|
2013-10-26 10:36:20 +00:00
|
|
|
#include <unordered_map>
|
2013-06-18 12:09:20 +00:00
|
|
|
#include "GLInterface.h"
|
|
|
|
#include "Android/TextureLoader.h"
|
|
|
|
#include "Android/ButtonManager.h"
|
|
|
|
|
|
|
|
extern void DrawButton(GLuint tex, float *coords);
|
|
|
|
|
|
|
|
namespace ButtonManager
|
|
|
|
{
|
2013-10-26 10:36:20 +00:00
|
|
|
std::unordered_map<int, Button*> m_buttons;
|
2013-11-14 21:17:51 +00:00
|
|
|
std::unordered_map<int, Axis*> m_axises;
|
2013-10-26 10:36:20 +00:00
|
|
|
std::unordered_map<std::string, InputDevice*> m_controllers;
|
2013-06-18 12:09:20 +00:00
|
|
|
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;
|
2013-10-29 05:23:17 +00:00
|
|
|
|
2013-06-18 12:09:20 +00:00
|
|
|
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
|
2013-10-26 10:36:20 +00:00
|
|
|
m_buttons[BUTTON_A] = new Button();
|
|
|
|
m_buttons[BUTTON_B] = new Button();
|
|
|
|
m_buttons[BUTTON_START] = new Button();
|
|
|
|
m_buttons[BUTTON_X] = new Button();
|
|
|
|
m_buttons[BUTTON_Y] = new Button();
|
|
|
|
m_buttons[BUTTON_Z] = new Button();
|
|
|
|
m_buttons[BUTTON_UP] = new Button();
|
|
|
|
m_buttons[BUTTON_DOWN] = new Button();
|
|
|
|
m_buttons[BUTTON_LEFT] = new Button();
|
|
|
|
m_buttons[BUTTON_RIGHT] = new Button();
|
|
|
|
|
2013-11-14 21:17:51 +00:00
|
|
|
m_axises[STICK_MAIN_UP] = new Axis();
|
|
|
|
m_axises[STICK_MAIN_DOWN] = new Axis();
|
|
|
|
m_axises[STICK_MAIN_LEFT] = new Axis();
|
|
|
|
m_axises[STICK_MAIN_RIGHT] = new Axis();
|
|
|
|
m_axises[STICK_C_UP] = new Axis();
|
|
|
|
m_axises[STICK_C_DOWN] = new Axis();
|
|
|
|
m_axises[STICK_C_LEFT] = new Axis();
|
|
|
|
m_axises[STICK_C_RIGHT] = new Axis();
|
|
|
|
m_axises[TRIGGER_L] = new Axis();
|
|
|
|
m_axises[TRIGGER_R] = new Axis();
|
2013-06-18 12:09:20 +00:00
|
|
|
|
|
|
|
// 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;
|
2013-10-29 05:23:17 +00:00
|
|
|
sscanf(value.c_str(), "Device '%[^\']'-Axis %d%c", dev, &bindnum, &modifier);
|
2013-06-18 12:09:20 +00:00
|
|
|
}
|
2013-10-29 05:23:17 +00:00
|
|
|
else if (std::string::npos != value.find("Button"))
|
2013-06-18 12:09:20 +00:00
|
|
|
{
|
|
|
|
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(ButtonType button)
|
|
|
|
{
|
|
|
|
bool pressed = false;
|
2013-10-26 10:36:20 +00:00
|
|
|
pressed = m_buttons[button]->Pressed();
|
2013-06-18 12:09:20 +00:00
|
|
|
|
|
|
|
for (auto it = m_controllers.begin(); it != m_controllers.end(); ++it)
|
|
|
|
pressed |= it->second->ButtonValue(button);
|
|
|
|
|
|
|
|
return pressed;
|
|
|
|
}
|
|
|
|
float GetAxisValue(ButtonType axis)
|
|
|
|
{
|
2013-11-14 21:17:51 +00:00
|
|
|
float value = 0.0f;
|
|
|
|
value = m_axises[axis]->AxisValue();
|
|
|
|
|
2013-06-18 12:09:20 +00:00
|
|
|
auto it = m_controllers.begin();
|
|
|
|
if (it == m_controllers.end())
|
2013-11-14 21:17:51 +00:00
|
|
|
return value;
|
2013-10-29 05:23:17 +00:00
|
|
|
return it->second->AxisValue(axis);
|
2013-06-18 12:09:20 +00:00
|
|
|
}
|
2013-10-26 10:36:20 +00:00
|
|
|
void TouchEvent(int button, int action)
|
2013-06-18 12:09:20 +00:00
|
|
|
{
|
2013-11-13 22:36:30 +00:00
|
|
|
m_buttons[button]->SetState(action ? BUTTON_PRESSED : BUTTON_RELEASED);
|
2013-06-18 12:09:20 +00:00
|
|
|
}
|
2013-11-14 21:17:51 +00:00
|
|
|
void TouchAxisEvent(int axis, float value)
|
|
|
|
{
|
|
|
|
m_axises[axis]->SetValue(value);
|
|
|
|
}
|
2013-06-18 12:09:20 +00:00
|
|
|
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)
|
2013-10-26 10:36:20 +00:00
|
|
|
delete it->second;
|
2013-06-18 12:09:20 +00:00
|
|
|
for (auto it = m_controllers.begin(); it != m_controllers.end(); ++it)
|
|
|
|
delete it->second;
|
2013-09-23 06:43:18 +00:00
|
|
|
m_controllers.clear();
|
|
|
|
m_buttons.clear();
|
2013-06-18 12:09:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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);
|
2013-10-29 05:23:17 +00:00
|
|
|
else
|
2013-06-18 12:09:20 +00:00
|
|
|
return m_axises[it->second->m_bind] * it->second->m_neg;
|
|
|
|
}
|
2013-10-29 05:23:17 +00:00
|
|
|
|
2013-06-18 12:09:20 +00:00
|
|
|
}
|