2008-07-30 13:47:54 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
2014-02-18 01:34:21 +00:00
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
2021-12-26 17:47:58 +00:00
|
|
|
* 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, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
2014-02-18 01:34:21 +00:00
|
|
|
*
|
|
|
|
* 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 for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
2021-12-26 17:47:58 +00:00
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2014-02-18 01:34:21 +00:00
|
|
|
*
|
|
|
|
*/
|
2008-07-30 13:47:54 +00:00
|
|
|
|
2008-08-07 16:38:39 +00:00
|
|
|
#include "backends/keymapper/keymap.h"
|
2008-09-30 13:51:01 +00:00
|
|
|
|
2012-02-15 05:02:23 +00:00
|
|
|
#include "common/system.h"
|
2017-08-14 11:58:43 +00:00
|
|
|
#include "common/tokenizer.h"
|
2012-02-15 05:02:23 +00:00
|
|
|
|
2017-08-11 11:57:28 +00:00
|
|
|
#include "backends/keymapper/action.h"
|
2012-02-24 19:23:55 +00:00
|
|
|
#include "backends/keymapper/hardware-input.h"
|
2020-01-22 10:18:55 +00:00
|
|
|
#include "backends/keymapper/keymapper-defaults.h"
|
2008-07-19 00:57:37 +00:00
|
|
|
|
2008-08-18 10:07:11 +00:00
|
|
|
#define KEYMAP_KEY_PREFIX "keymap_"
|
|
|
|
|
2008-07-19 00:57:37 +00:00
|
|
|
namespace Common {
|
|
|
|
|
2020-06-13 16:42:25 +00:00
|
|
|
Keymap::Keymap(KeymapType type, const String &id, const U32String &description) :
|
2017-08-13 14:35:58 +00:00
|
|
|
_type(type),
|
2020-01-28 17:44:08 +00:00
|
|
|
_id(id),
|
|
|
|
_description(description),
|
2017-08-14 15:11:33 +00:00
|
|
|
_enabled(true),
|
2020-01-28 17:44:08 +00:00
|
|
|
_configDomain(nullptr),
|
2020-01-22 10:18:55 +00:00
|
|
|
_hardwareInputSet(nullptr),
|
|
|
|
_backendDefaultBindings(nullptr) {
|
2017-08-13 14:35:58 +00:00
|
|
|
|
2008-07-19 00:57:37 +00:00
|
|
|
}
|
|
|
|
|
2020-08-21 08:20:43 +00:00
|
|
|
Keymap::Keymap(KeymapType type, const String &id, const String &description) :
|
|
|
|
_type(type),
|
|
|
|
_id(id),
|
|
|
|
_description(U32String(description)),
|
|
|
|
_enabled(true),
|
|
|
|
_configDomain(nullptr),
|
|
|
|
_hardwareInputSet(nullptr),
|
|
|
|
_backendDefaultBindings(nullptr) {
|
|
|
|
|
2020-07-03 21:43:05 +00:00
|
|
|
}
|
|
|
|
|
2008-08-14 19:20:25 +00:00
|
|
|
Keymap::~Keymap() {
|
2017-08-14 10:59:37 +00:00
|
|
|
for (ActionArray::iterator it = _actions.begin(); it != _actions.end(); ++it)
|
2008-08-14 19:20:25 +00:00
|
|
|
delete *it;
|
|
|
|
}
|
|
|
|
|
2008-08-01 16:44:49 +00:00
|
|
|
void Keymap::addAction(Action *action) {
|
|
|
|
if (findAction(action->id))
|
2009-05-31 10:02:16 +00:00
|
|
|
error("Action with id %s already in KeyMap", action->id);
|
2009-05-10 17:18:59 +00:00
|
|
|
|
2008-07-19 00:57:37 +00:00
|
|
|
_actions.push_back(action);
|
|
|
|
}
|
|
|
|
|
2020-01-26 11:18:52 +00:00
|
|
|
void Keymap::registerMapping(Action *action, const HardwareInput &hwInput) {
|
2020-10-19 23:28:00 +00:00
|
|
|
ActionArray &actionArray = _hwActionMap.getOrCreateVal(hwInput);
|
2017-08-11 11:57:28 +00:00
|
|
|
|
2017-08-14 11:58:43 +00:00
|
|
|
// Don't allow an input to map to the same action multiple times
|
|
|
|
ActionArray::const_iterator found = find(actionArray.begin(), actionArray.end(), action);
|
|
|
|
if (found == actionArray.end()) {
|
|
|
|
actionArray.push_back(action);
|
|
|
|
}
|
2008-07-19 00:57:37 +00:00
|
|
|
}
|
|
|
|
|
2008-08-01 16:44:49 +00:00
|
|
|
void Keymap::unregisterMapping(Action *action) {
|
2017-08-14 11:58:43 +00:00
|
|
|
// Remove the action from all the input mappings
|
|
|
|
for (HardwareActionMap::iterator itInput = _hwActionMap.begin(); itInput != _hwActionMap.end(); itInput++) {
|
|
|
|
for (ActionArray::iterator itAction = itInput->_value.begin(); itAction != itInput->_value.end(); itAction++) {
|
|
|
|
if (*itAction == action) {
|
|
|
|
itInput->_value.erase(itAction);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (itInput->_value.empty()) {
|
|
|
|
_hwActionMap.erase(itInput);
|
2017-08-11 11:57:28 +00:00
|
|
|
}
|
2008-08-08 14:23:59 +00:00
|
|
|
}
|
2008-07-24 10:00:56 +00:00
|
|
|
}
|
|
|
|
|
2017-08-14 15:11:33 +00:00
|
|
|
void Keymap::resetMapping(Action *action) {
|
|
|
|
unregisterMapping(action);
|
|
|
|
|
2020-01-28 17:44:09 +00:00
|
|
|
StringArray hwInputIds = getActionDefaultMappings(action);
|
2017-08-14 15:11:33 +00:00
|
|
|
registerMappings(action, hwInputIds);
|
|
|
|
}
|
|
|
|
|
2020-01-28 17:44:08 +00:00
|
|
|
struct HardwareInputTypeIdComparator {
|
|
|
|
bool operator()(const HardwareInput &x, const HardwareInput &y) const {
|
|
|
|
if (x.type != y.type) {
|
|
|
|
return x.type < y.type;
|
|
|
|
}
|
|
|
|
return x.id.compareTo(y.id);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-12-03 09:16:14 +00:00
|
|
|
Array<HardwareInput> Keymap::getActionMapping(const Action *action) const {
|
2020-01-26 11:18:52 +00:00
|
|
|
Array<HardwareInput> inputs;
|
2017-08-14 11:58:43 +00:00
|
|
|
|
|
|
|
for (HardwareActionMap::iterator itInput = _hwActionMap.begin(); itInput != _hwActionMap.end(); itInput++) {
|
|
|
|
for (ActionArray::iterator itAction = itInput->_value.begin(); itAction != itInput->_value.end(); itAction++) {
|
|
|
|
if (*itAction == action) {
|
|
|
|
inputs.push_back(itInput->_key);
|
|
|
|
break;
|
|
|
|
}
|
2017-08-11 11:57:28 +00:00
|
|
|
}
|
2008-07-19 19:12:49 +00:00
|
|
|
}
|
2017-08-11 11:57:28 +00:00
|
|
|
|
2020-01-28 17:44:08 +00:00
|
|
|
// Sort the inputs by type and then id for the remap dialog
|
|
|
|
Common::sort(inputs.begin(), inputs.end(), HardwareInputTypeIdComparator());
|
|
|
|
|
2017-08-14 11:58:43 +00:00
|
|
|
return inputs;
|
2008-07-19 19:12:49 +00:00
|
|
|
}
|
|
|
|
|
2008-08-18 10:07:11 +00:00
|
|
|
const Action *Keymap::findAction(const char *id) const {
|
2017-08-14 10:59:37 +00:00
|
|
|
for (ActionArray::const_iterator it = _actions.begin(); it != _actions.end(); ++it) {
|
2017-08-11 12:19:41 +00:00
|
|
|
if (strcmp((*it)->id, id) == 0)
|
2008-08-01 16:44:49 +00:00
|
|
|
return *it;
|
2008-07-19 19:12:49 +00:00
|
|
|
}
|
2009-05-10 17:18:59 +00:00
|
|
|
|
2017-08-11 11:57:28 +00:00
|
|
|
return nullptr;
|
2008-07-19 19:12:49 +00:00
|
|
|
}
|
|
|
|
|
2020-05-27 03:00:49 +00:00
|
|
|
Keymap::KeymapMatch Keymap::getMappedActions(const Event &event, ActionArray &actions) const {
|
2020-01-26 11:18:52 +00:00
|
|
|
switch (event.type) {
|
|
|
|
case EVENT_KEYDOWN:
|
|
|
|
case EVENT_KEYUP: {
|
2020-02-13 06:25:01 +00:00
|
|
|
KeyState normalizedKeystate = KeyboardHardwareInputSet::normalizeKeyState(event.kbd);
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createKeyboard("", normalizedKeystate, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
if (!actions.empty()) {
|
|
|
|
return kKeymapMatchExact;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (normalizedKeystate.flags & KBD_NON_STICKY) {
|
|
|
|
// If no matching actions and non-sticky keyboard modifiers are down,
|
|
|
|
// check again for matches without the exact keyboard modifiers
|
|
|
|
for (HardwareActionMap::const_iterator itInput = _hwActionMap.begin(); itInput != _hwActionMap.end(); ++itInput) {
|
|
|
|
if (itInput->_key.type == kHardwareInputTypeKeyboard && itInput->_key.key.keycode == normalizedKeystate.keycode) {
|
|
|
|
int flags = itInput->_key.key.flags;
|
|
|
|
if (flags & KBD_NON_STICKY && (flags & normalizedKeystate.flags) == flags) {
|
|
|
|
actions.push_back(itInput->_value);
|
|
|
|
return kKeymapMatchPartial;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lastly check again for matches no non-sticky keyboard modifiers
|
|
|
|
normalizedKeystate.flags &= ~KBD_NON_STICKY;
|
2020-09-08 20:25:02 +00:00
|
|
|
hardwareInput = HardwareInput::createKeyboard("", normalizedKeystate, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
return actions.empty() ? kKeymapMatchNone : kKeymapMatchPartial;
|
|
|
|
}
|
|
|
|
break;
|
2020-01-26 15:33:25 +00:00
|
|
|
}
|
2020-01-28 17:44:08 +00:00
|
|
|
case EVENT_LBUTTONDOWN:
|
|
|
|
case EVENT_LBUTTONUP: {
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createMouse("", MOUSE_BUTTON_LEFT, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-01-28 17:44:08 +00:00
|
|
|
}
|
|
|
|
case EVENT_RBUTTONDOWN:
|
|
|
|
case EVENT_RBUTTONUP: {
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createMouse("", MOUSE_BUTTON_RIGHT, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-01-28 17:44:08 +00:00
|
|
|
}
|
|
|
|
case EVENT_MBUTTONDOWN:
|
|
|
|
case EVENT_MBUTTONUP: {
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createMouse("", MOUSE_BUTTON_MIDDLE, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-01-28 17:44:08 +00:00
|
|
|
}
|
2020-02-09 13:45:35 +00:00
|
|
|
case Common::EVENT_WHEELUP: {
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createMouse("", MOUSE_WHEEL_UP, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-02-09 13:45:35 +00:00
|
|
|
}
|
|
|
|
case Common::EVENT_WHEELDOWN: {
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createMouse("", MOUSE_WHEEL_DOWN, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-02-09 13:45:35 +00:00
|
|
|
}
|
2020-02-11 21:09:56 +00:00
|
|
|
case EVENT_X1BUTTONDOWN:
|
|
|
|
case EVENT_X1BUTTONUP: {
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createMouse("", MOUSE_BUTTON_X1, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-02-11 21:09:56 +00:00
|
|
|
}
|
|
|
|
case EVENT_X2BUTTONDOWN:
|
|
|
|
case EVENT_X2BUTTONUP: {
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createMouse("", MOUSE_BUTTON_X2, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-02-11 21:09:56 +00:00
|
|
|
}
|
2020-01-26 15:33:25 +00:00
|
|
|
case EVENT_JOYBUTTON_DOWN:
|
|
|
|
case EVENT_JOYBUTTON_UP: {
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createJoystickButton("", event.joystick.button, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-02-08 10:19:16 +00:00
|
|
|
}
|
|
|
|
case EVENT_JOYAXIS_MOTION: {
|
2020-03-02 19:24:43 +00:00
|
|
|
if (event.joystick.position != 0) {
|
|
|
|
bool positiveHalf = event.joystick.position >= 0;
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createJoystickHalfAxis("", event.joystick.axis, positiveHalf, U32String());
|
2021-02-17 00:03:52 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-03-02 19:24:43 +00:00
|
|
|
} else {
|
|
|
|
// Axis position zero is part of both half axes, and triggers actions bound to both
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInputPos = HardwareInput::createJoystickHalfAxis("", event.joystick.axis, true, U32String());
|
|
|
|
HardwareInput hardwareInputNeg = HardwareInput::createJoystickHalfAxis("", event.joystick.axis, false, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInputPos));
|
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInputNeg));
|
2020-03-02 19:24:43 +00:00
|
|
|
}
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-01-26 11:18:52 +00:00
|
|
|
}
|
|
|
|
case EVENT_CUSTOM_BACKEND_HARDWARE: {
|
2020-09-08 20:25:02 +00:00
|
|
|
HardwareInput hardwareInput = HardwareInput::createCustom("", event.customType, U32String());
|
2020-10-19 23:28:00 +00:00
|
|
|
actions.push_back(_hwActionMap.getValOrDefault(hardwareInput));
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-01-26 11:18:52 +00:00
|
|
|
}
|
|
|
|
default:
|
2020-05-27 03:00:49 +00:00
|
|
|
break;
|
2020-01-26 11:18:52 +00:00
|
|
|
}
|
2020-05-27 03:00:49 +00:00
|
|
|
|
|
|
|
return actions.empty() ? kKeymapMatchNone : kKeymapMatchExact;
|
2012-02-23 00:30:47 +00:00
|
|
|
}
|
|
|
|
|
2017-08-14 15:11:33 +00:00
|
|
|
void Keymap::setConfigDomain(ConfigManager::Domain *configDomain) {
|
|
|
|
_configDomain = configDomain;
|
2008-08-14 01:42:02 +00:00
|
|
|
}
|
|
|
|
|
2017-08-14 15:11:33 +00:00
|
|
|
void Keymap::setHardwareInputs(HardwareInputSet *hardwareInputSet) {
|
|
|
|
_hardwareInputSet = hardwareInputSet;
|
|
|
|
}
|
|
|
|
|
2020-01-28 17:44:09 +00:00
|
|
|
void Keymap::setBackendDefaultBindings(const KeymapperDefaultBindings *backendDefaultBindings) {
|
2020-01-22 10:18:55 +00:00
|
|
|
_backendDefaultBindings = backendDefaultBindings;
|
|
|
|
}
|
|
|
|
|
2020-01-28 17:44:09 +00:00
|
|
|
StringArray Keymap::getActionDefaultMappings(Action *action) {
|
|
|
|
// Backend default mappings overrides keymap default mappings, so backends can resolve mapping conflicts.
|
|
|
|
// Empty mappings are valid and mean the action should not be mapped by default.
|
|
|
|
if (_backendDefaultBindings) {
|
|
|
|
KeymapperDefaultBindings::const_iterator it = _backendDefaultBindings->findDefaultBinding(_id, action->id);
|
|
|
|
if (it != _backendDefaultBindings->end()) {
|
2023-03-24 10:28:24 +00:00
|
|
|
return it->_value;
|
2020-01-25 12:41:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If no keymap-specific default mapping was found, look for a standard action binding
|
2020-01-28 17:44:09 +00:00
|
|
|
it = _backendDefaultBindings->findDefaultBinding(kStandardActionsKeymapName, action->id);
|
|
|
|
if (it != _backendDefaultBindings->end()) {
|
2023-03-24 10:28:24 +00:00
|
|
|
return it->_value;
|
2020-01-22 10:18:55 +00:00
|
|
|
}
|
|
|
|
}
|
2020-01-28 17:44:09 +00:00
|
|
|
|
|
|
|
return action->getDefaultInputMapping();
|
2020-01-22 10:18:55 +00:00
|
|
|
}
|
|
|
|
|
2017-08-14 15:11:33 +00:00
|
|
|
void Keymap::loadMappings() {
|
2017-08-14 11:58:43 +00:00
|
|
|
assert(_configDomain);
|
2017-08-14 15:11:33 +00:00
|
|
|
assert(_hardwareInputSet);
|
2009-05-10 17:18:59 +00:00
|
|
|
|
2017-08-14 11:58:43 +00:00
|
|
|
if (_actions.empty()) {
|
2012-02-15 05:02:23 +00:00
|
|
|
return;
|
2017-08-14 11:58:43 +00:00
|
|
|
}
|
2008-08-06 19:21:45 +00:00
|
|
|
|
2020-01-28 17:44:08 +00:00
|
|
|
String prefix = KEYMAP_KEY_PREFIX + _id + "_";
|
2009-05-10 17:18:59 +00:00
|
|
|
|
2017-08-14 11:58:43 +00:00
|
|
|
_hwActionMap.clear();
|
2017-08-14 10:59:37 +00:00
|
|
|
for (ActionArray::const_iterator it = _actions.begin(); it != _actions.end(); ++it) {
|
2017-08-14 15:11:33 +00:00
|
|
|
Action *action = *it;
|
|
|
|
String confKey = prefix + action->id;
|
2012-02-15 05:02:23 +00:00
|
|
|
|
2020-01-28 17:44:09 +00:00
|
|
|
StringArray hwInputIds;
|
2017-08-14 15:11:33 +00:00
|
|
|
if (_configDomain->contains(confKey)) {
|
|
|
|
// The configuration value is a list of space separated hardware input ids
|
|
|
|
StringTokenizer hwInputTokenizer = _configDomain->getVal(confKey);
|
|
|
|
|
2020-01-28 17:44:09 +00:00
|
|
|
while (!hwInputTokenizer.empty()) {
|
|
|
|
hwInputIds.push_back(hwInputTokenizer.nextToken());
|
2017-08-14 15:11:33 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If the configuration key was not found, use the default mapping
|
2020-01-28 17:44:09 +00:00
|
|
|
hwInputIds = getActionDefaultMappings(action);
|
2017-08-14 15:11:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
registerMappings(action, hwInputIds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-28 17:44:09 +00:00
|
|
|
void Keymap::registerMappings(Action *action, const StringArray &hwInputIds) {
|
2017-08-14 15:11:33 +00:00
|
|
|
assert(_hardwareInputSet);
|
2012-02-15 05:02:23 +00:00
|
|
|
|
2017-08-14 15:11:33 +00:00
|
|
|
for (uint i = 0; i < hwInputIds.size(); i++) {
|
2020-01-28 17:44:09 +00:00
|
|
|
HardwareInput hwInput = _hardwareInputSet->findHardwareInput(hwInputIds[i]);
|
2008-08-06 19:21:45 +00:00
|
|
|
|
2020-01-26 11:18:52 +00:00
|
|
|
if (hwInput.type == kHardwareInputTypeInvalid) {
|
2017-08-14 15:11:33 +00:00
|
|
|
// Silently ignore unknown hardware ids because the current device may not have inputs matching the defaults
|
|
|
|
debug(1, "HardwareInput with ID '%s' not known", hwInputIds[i].c_str());
|
2017-08-14 11:58:43 +00:00
|
|
|
continue;
|
|
|
|
}
|
2009-05-10 17:18:59 +00:00
|
|
|
|
2017-08-14 11:58:43 +00:00
|
|
|
// map the key
|
2017-08-14 15:11:33 +00:00
|
|
|
registerMapping(action, hwInput);
|
2008-08-06 19:21:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-08-14 01:42:02 +00:00
|
|
|
void Keymap::saveMappings() {
|
2009-01-21 02:02:55 +00:00
|
|
|
if (!_configDomain)
|
|
|
|
return;
|
2009-05-10 17:18:59 +00:00
|
|
|
|
2020-01-28 17:44:08 +00:00
|
|
|
String prefix = KEYMAP_KEY_PREFIX + _id + "_";
|
2009-05-10 17:18:59 +00:00
|
|
|
|
2017-08-14 11:58:43 +00:00
|
|
|
for (ActionArray::const_iterator it = _actions.begin(); it != _actions.end(); it++) {
|
|
|
|
Action *action = *it;
|
2020-01-26 11:18:52 +00:00
|
|
|
Array<HardwareInput> mappedInputs = getActionMapping(action);
|
2017-08-14 11:58:43 +00:00
|
|
|
|
2020-01-28 17:44:09 +00:00
|
|
|
if (areMappingsIdentical(mappedInputs, getActionDefaultMappings(action))) {
|
2017-08-14 15:11:33 +00:00
|
|
|
// If the current mapping is the default, don't write anything to the config manager
|
|
|
|
_configDomain->erase(prefix + action->id);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-08-14 11:58:43 +00:00
|
|
|
// The configuration value is a list of space separated hardware input ids
|
|
|
|
String confValue;
|
|
|
|
for (uint j = 0; j < mappedInputs.size(); j++) {
|
|
|
|
if (!confValue.empty()) {
|
|
|
|
confValue += " ";
|
|
|
|
}
|
|
|
|
|
2020-01-26 11:18:52 +00:00
|
|
|
confValue += mappedInputs[j].id;
|
2017-08-14 11:58:43 +00:00
|
|
|
}
|
2009-05-10 17:18:59 +00:00
|
|
|
|
2017-08-14 11:58:43 +00:00
|
|
|
_configDomain->setVal(prefix + action->id, confValue);
|
2008-08-07 14:16:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-28 17:44:09 +00:00
|
|
|
bool Keymap::areMappingsIdentical(const Array<HardwareInput> &mappingsA, const StringArray &mappingsB) {
|
2017-08-14 15:11:33 +00:00
|
|
|
// Assumes array values are not duplicated, but registerMapping and addDefaultInputMapping ensure that
|
|
|
|
|
|
|
|
uint foundCount = 0;
|
2020-09-15 20:42:34 +00:00
|
|
|
uint validDefaultMappings = 0;
|
2020-01-28 17:44:09 +00:00
|
|
|
for (uint i = 0; i < mappingsB.size(); i++) {
|
|
|
|
// We resolve the hardware input to make sure it is not a default for some hardware we don't have currently
|
|
|
|
HardwareInput mappingB = _hardwareInputSet->findHardwareInput(mappingsB[i]);
|
|
|
|
if (mappingB.type == kHardwareInputTypeInvalid) continue;
|
2020-09-15 20:42:34 +00:00
|
|
|
validDefaultMappings++;
|
2020-01-28 17:44:09 +00:00
|
|
|
|
|
|
|
for (uint j = 0; j < mappingsA.size(); j++) {
|
|
|
|
if (mappingsA[j].id == mappingB.id) {
|
2017-08-14 15:11:33 +00:00
|
|
|
foundCount++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-15 20:42:34 +00:00
|
|
|
return foundCount == mappingsA.size() && foundCount == validDefaultMappings;
|
2017-08-14 15:11:33 +00:00
|
|
|
}
|
|
|
|
|
2009-10-04 21:26:33 +00:00
|
|
|
} // End of namespace Common
|