mirror of
https://github.com/RetroDECK/ES-DE.git
synced 2024-11-27 08:35:39 +00:00
203 lines
4.4 KiB
C++
203 lines
4.4 KiB
C++
#include "InputConfig.h"
|
|
#include <string>
|
|
#include <algorithm>
|
|
#include <SDL.h>
|
|
#include <iostream>
|
|
#include "Log.h"
|
|
#include "InputManager.h"
|
|
|
|
//some util functions
|
|
std::string inputTypeToString(InputType type)
|
|
{
|
|
switch(type)
|
|
{
|
|
case TYPE_AXIS:
|
|
return "axis";
|
|
case TYPE_BUTTON:
|
|
return "button";
|
|
case TYPE_HAT:
|
|
return "hat";
|
|
case TYPE_KEY:
|
|
return "key";
|
|
default:
|
|
return "error";
|
|
}
|
|
}
|
|
|
|
InputType stringToInputType(const std::string& type)
|
|
{
|
|
if(type == "axis")
|
|
return TYPE_AXIS;
|
|
if(type == "button")
|
|
return TYPE_BUTTON;
|
|
if(type == "hat")
|
|
return TYPE_HAT;
|
|
if(type == "key")
|
|
return TYPE_KEY;
|
|
return TYPE_COUNT;
|
|
}
|
|
|
|
|
|
std::string toLower(std::string str)
|
|
{
|
|
for(unsigned int i = 0; i < str.length(); i++)
|
|
{
|
|
str[i] = tolower(str[i]);
|
|
}
|
|
|
|
return str;
|
|
}
|
|
//end util functions
|
|
|
|
InputConfig::InputConfig(int deviceId, const std::string& deviceName, const std::string& deviceGUID) : mDeviceId(deviceId), mDeviceName(deviceName), mDeviceGUID(deviceGUID)
|
|
{
|
|
}
|
|
|
|
void InputConfig::clear()
|
|
{
|
|
mNameMap.clear();
|
|
}
|
|
|
|
bool InputConfig::isConfigured()
|
|
{
|
|
return mNameMap.size() > 0;
|
|
}
|
|
|
|
void InputConfig::mapInput(const std::string& name, Input input)
|
|
{
|
|
mNameMap[toLower(name)] = input;
|
|
}
|
|
|
|
void InputConfig::unmapInput(const std::string& name)
|
|
{
|
|
auto it = mNameMap.find(toLower(name));
|
|
if(it != mNameMap.end())
|
|
mNameMap.erase(it);
|
|
}
|
|
|
|
bool InputConfig::getInputByName(const std::string& name, Input* result)
|
|
{
|
|
auto it = mNameMap.find(toLower(name));
|
|
if(it != mNameMap.end())
|
|
{
|
|
*result = it->second;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool InputConfig::isMappedTo(const std::string& name, Input input)
|
|
{
|
|
Input comp;
|
|
if(!getInputByName(name, &comp))
|
|
return false;
|
|
|
|
if(comp.configured && comp.type == input.type && comp.id == input.id)
|
|
{
|
|
if(comp.type == TYPE_HAT)
|
|
{
|
|
return (input.value == 0 || input.value & comp.value);
|
|
}
|
|
|
|
if(comp.type == TYPE_AXIS)
|
|
{
|
|
return input.value == 0 || comp.value == input.value;
|
|
}else{
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
std::vector<std::string> InputConfig::getMappedTo(Input input)
|
|
{
|
|
std::vector<std::string> maps;
|
|
|
|
typedef std::map<std::string, Input>::iterator it_type;
|
|
for(it_type iterator = mNameMap.begin(); iterator != mNameMap.end(); iterator++)
|
|
{
|
|
Input chk = iterator->second;
|
|
|
|
if(!chk.configured)
|
|
continue;
|
|
|
|
if(chk.device == input.device && chk.type == input.type && chk.id == input.id)
|
|
{
|
|
if(chk.type == TYPE_HAT)
|
|
{
|
|
if(input.value == 0 || input.value & chk.value)
|
|
{
|
|
maps.push_back(iterator->first);
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if(input.type == TYPE_AXIS)
|
|
{
|
|
if(input.value == 0 || chk.value == input.value)
|
|
maps.push_back(iterator->first);
|
|
}else{
|
|
maps.push_back(iterator->first);
|
|
}
|
|
}
|
|
}
|
|
|
|
return maps;
|
|
}
|
|
|
|
void InputConfig::loadFromXML(pugi::xml_node node)
|
|
{
|
|
clear();
|
|
|
|
for(pugi::xml_node input = node.child("input"); input; input = input.next_sibling("input"))
|
|
{
|
|
std::string name = input.attribute("name").as_string();
|
|
std::string type = input.attribute("type").as_string();
|
|
InputType typeEnum = stringToInputType(type);
|
|
|
|
if(typeEnum == TYPE_COUNT)
|
|
{
|
|
LOG(LogError) << "InputConfig load error - input of type \"" << type << "\" is invalid! Skipping input \"" << name << "\".\n";
|
|
continue;
|
|
}
|
|
|
|
int id = input.attribute("id").as_int();
|
|
int value = input.attribute("value").as_int();
|
|
|
|
if(value == 0)
|
|
LOG(LogWarning) << "WARNING: InputConfig value is 0 for " << type << " " << id << "!\n";
|
|
|
|
mNameMap[toLower(name)] = Input(mDeviceId, typeEnum, id, value, true);
|
|
}
|
|
}
|
|
|
|
void InputConfig::writeToXML(pugi::xml_node parent)
|
|
{
|
|
pugi::xml_node cfg = parent.append_child("inputConfig");
|
|
|
|
if(mDeviceId == DEVICE_KEYBOARD)
|
|
{
|
|
cfg.append_attribute("type") = "keyboard";
|
|
cfg.append_attribute("deviceName") = "Keyboard";
|
|
}else{
|
|
cfg.append_attribute("type") = "joystick";
|
|
cfg.append_attribute("deviceName") = mDeviceName.c_str();
|
|
}
|
|
|
|
cfg.append_attribute("deviceGUID") = mDeviceGUID.c_str();
|
|
|
|
typedef std::map<std::string, Input>::iterator it_type;
|
|
for(it_type iterator = mNameMap.begin(); iterator != mNameMap.end(); iterator++)
|
|
{
|
|
if(!iterator->second.configured)
|
|
continue;
|
|
|
|
pugi::xml_node input = cfg.append_child("input");
|
|
input.append_attribute("name") = iterator->first.c_str();
|
|
input.append_attribute("type") = inputTypeToString(iterator->second.type).c_str();
|
|
input.append_attribute("id").set_value(iterator->second.id);
|
|
input.append_attribute("value").set_value(iterator->second.value);
|
|
}
|
|
}
|