This fixes a problem where opening the keymapper dialog would cause the current game keymap to be displayed as the active keymap but then changing the keymap selection back to it would cause the GUI keymap to be displayed as the active one. The GUI keymap was indeed at the top of the stack but that's not the desired effect. Also move the pushing and popping of the keymap to Dialog::Open/Close Also constantify the GUI keymap name
447 lines
12 KiB
C++
447 lines
12 KiB
C++
/* ScummVM - Graphic Adventure Engine
|
|
*
|
|
* 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.
|
|
*
|
|
* 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 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* 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
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "backends/keymapper/remap-dialog.h"
|
|
|
|
#ifdef ENABLE_KEYMAPPER
|
|
|
|
#include "common/system.h"
|
|
#include "gui/gui-manager.h"
|
|
#include "gui/widgets/popup.h"
|
|
#include "gui/widgets/scrollbar.h"
|
|
#include "gui/ThemeEval.h"
|
|
|
|
#include "common/translation.h"
|
|
|
|
namespace Common {
|
|
|
|
enum {
|
|
kRemapCmd = 'REMP',
|
|
kClearCmd = 'CLER',
|
|
kCloseCmd = 'CLOS'
|
|
};
|
|
|
|
RemapDialog::RemapDialog()
|
|
: Dialog("KeyMapper"), _keymapTable(0), _activeRemapAction(0), _topAction(0), _remapTimeout(0) {
|
|
|
|
_keymapper = g_system->getEventManager()->getKeymapper();
|
|
assert(_keymapper);
|
|
|
|
_kmPopUpDesc = new GUI::StaticTextWidget(this, "KeyMapper.PopupDesc", _("Keymap:"));
|
|
_kmPopUp = new GUI::PopUpWidget(this, "KeyMapper.Popup");
|
|
|
|
_scrollBar = new GUI::ScrollBarWidget(this, 0, 0, 0, 0);
|
|
|
|
new GUI::ButtonWidget(this, "KeyMapper.Close", _("Close"), 0, kCloseCmd);
|
|
}
|
|
|
|
RemapDialog::~RemapDialog() {
|
|
free(_keymapTable);
|
|
}
|
|
|
|
void RemapDialog::open() {
|
|
bool divider = false;
|
|
const Stack<Keymapper::MapRecord> &activeKeymaps = _keymapper->getActiveStack();
|
|
|
|
if (activeKeymaps.size() > 0) {
|
|
_kmPopUp->appendEntry(activeKeymaps.top().keymap->getName() + _(" (Active)"));
|
|
divider = true;
|
|
}
|
|
|
|
Keymapper::Domain *_globalKeymaps = &_keymapper->getGlobalDomain();
|
|
Keymapper::Domain *_gameKeymaps = 0;
|
|
|
|
int keymapCount = 0;
|
|
|
|
if (_globalKeymaps->empty())
|
|
_globalKeymaps = 0;
|
|
else
|
|
keymapCount += _globalKeymaps->size();
|
|
|
|
if (ConfMan.getActiveDomain() != 0) {
|
|
_gameKeymaps = &_keymapper->getGameDomain();
|
|
|
|
if (_gameKeymaps->empty())
|
|
_gameKeymaps = 0;
|
|
else
|
|
keymapCount += _gameKeymaps->size();
|
|
}
|
|
|
|
debug(3, "RemapDialog::open keymaps: %d", keymapCount);
|
|
|
|
_keymapTable = (Keymap **)malloc(sizeof(Keymap*) * keymapCount);
|
|
|
|
Keymapper::Domain::iterator it;
|
|
uint32 idx = 0;
|
|
|
|
if (_globalKeymaps) {
|
|
if (divider)
|
|
_kmPopUp->appendEntry("");
|
|
for (it = _globalKeymaps->begin(); it != _globalKeymaps->end(); ++it) {
|
|
_kmPopUp->appendEntry(it->_value->getName() + _(" (Global)"), idx);
|
|
_keymapTable[idx++] = it->_value;
|
|
}
|
|
divider = true;
|
|
}
|
|
|
|
if (_gameKeymaps) {
|
|
if (divider)
|
|
_kmPopUp->appendEntry("");
|
|
for (it = _gameKeymaps->begin(); it != _gameKeymaps->end(); ++it) {
|
|
_kmPopUp->appendEntry(it->_value->getName() + _(" (Game)"), idx);
|
|
_keymapTable[idx++] = it->_value;
|
|
}
|
|
}
|
|
|
|
_changes = false;
|
|
|
|
Dialog::open();
|
|
|
|
_kmPopUp->setSelected(0);
|
|
loadKeymap();
|
|
}
|
|
|
|
void RemapDialog::close() {
|
|
_kmPopUp->clearEntries();
|
|
|
|
free(_keymapTable);
|
|
_keymapTable = 0;
|
|
|
|
if (_changes)
|
|
ConfMan.flushToDisk();
|
|
|
|
Dialog::close();
|
|
}
|
|
|
|
void RemapDialog::reflowLayout() {
|
|
Dialog::reflowLayout();
|
|
|
|
int buttonHeight = g_gui.xmlEval()->getVar("Globals.Button.Height", 0);
|
|
int scrollbarWidth = g_gui.xmlEval()->getVar("Globals.Scrollbar.Width", 0);
|
|
|
|
int16 areaX, areaY;
|
|
uint16 areaW, areaH;
|
|
int spacing = g_gui.xmlEval()->getVar("Globals.KeyMapper.Spacing");
|
|
int labelWidth = g_gui.xmlEval()->getVar("Globals.KeyMapper.LabelWidth");
|
|
int keyButtonWidth = g_gui.xmlEval()->getVar("Globals.KeyMapper.ButtonWidth");
|
|
int clearButtonWidth = g_gui.xmlEval()->getVar("Globals.Line.Height");
|
|
int clearButtonHeight = g_gui.xmlEval()->getVar("Globals.Line.Height");
|
|
int colWidth = labelWidth + keyButtonWidth + clearButtonWidth + spacing;
|
|
|
|
g_gui.xmlEval()->getWidgetData((const String&)String("KeyMapper.KeymapArea"), areaX, areaY, areaW, areaH);
|
|
|
|
_colCount = (areaW - scrollbarWidth) / colWidth;
|
|
_rowCount = (areaH + spacing) / (buttonHeight + spacing);
|
|
if (_colCount <= 0 || _rowCount <= 0)
|
|
error("Remap dialog too small to display any keymaps");
|
|
|
|
_scrollBar->resize(areaX + areaW - scrollbarWidth, areaY, scrollbarWidth, areaH);
|
|
_scrollBar->_entriesPerPage = _rowCount;
|
|
_scrollBar->_numEntries = 1;
|
|
_scrollBar->recalc();
|
|
|
|
uint textYOff = (buttonHeight - kLineHeight) / 2;
|
|
uint clearButtonYOff = (buttonHeight - clearButtonHeight) / 2;
|
|
uint oldSize = _keymapWidgets.size();
|
|
uint newSize = _rowCount * _colCount;
|
|
|
|
_keymapWidgets.reserve(newSize);
|
|
|
|
for (uint i = 0; i < newSize; i++) {
|
|
ActionWidgets widg;
|
|
|
|
if (i >= _keymapWidgets.size()) {
|
|
widg.actionText =
|
|
new GUI::StaticTextWidget(this, 0, 0, 0, 0, "", Graphics::kTextAlignRight);
|
|
widg.keyButton =
|
|
new GUI::ButtonWidget(this, 0, 0, 0, 0, "", 0, kRemapCmd + i);
|
|
widg.clearButton =
|
|
new GUI::ButtonWidget(this, 0, 0, 0, 0, "C", 0, kClearCmd + i);
|
|
|
|
_keymapWidgets.push_back(widg);
|
|
} else {
|
|
widg = _keymapWidgets[i];
|
|
}
|
|
|
|
uint x = areaX + (i % _colCount) * colWidth;
|
|
uint y = areaY + (i / _colCount) * (buttonHeight + spacing);
|
|
|
|
widg.actionText->resize(x, y + textYOff, labelWidth, kLineHeight);
|
|
widg.keyButton->resize(x + labelWidth, y, keyButtonWidth, buttonHeight);
|
|
widg.clearButton->resize(x + labelWidth + keyButtonWidth + spacing, y + clearButtonYOff, clearButtonWidth, clearButtonHeight);
|
|
}
|
|
while (oldSize > newSize) {
|
|
ActionWidgets widg = _keymapWidgets.remove_at(--oldSize);
|
|
|
|
removeWidget(widg.actionText);
|
|
delete widg.actionText;
|
|
|
|
removeWidget(widg.keyButton);
|
|
delete widg.keyButton;
|
|
|
|
removeWidget(widg.clearButton);
|
|
delete widg.clearButton;
|
|
}
|
|
}
|
|
|
|
void RemapDialog::handleCommand(GUI::CommandSender *sender, uint32 cmd, uint32 data) {
|
|
debug(3, "RemapDialog::handleCommand %u %u", cmd, data);
|
|
|
|
if (cmd >= kRemapCmd && cmd < kRemapCmd + _keymapWidgets.size()) {
|
|
startRemapping(cmd - kRemapCmd);
|
|
} else if (cmd >= kClearCmd && cmd < kClearCmd + _keymapWidgets.size()) {
|
|
clearMapping(cmd - kClearCmd);
|
|
} else if (cmd == GUI::kPopUpItemSelectedCmd) {
|
|
loadKeymap();
|
|
} else if (cmd == GUI::kSetPositionCmd) {
|
|
refreshKeymap();
|
|
} else if (cmd == kCloseCmd) {
|
|
close();
|
|
} else {
|
|
GUI::Dialog::handleCommand(sender, cmd, data);
|
|
}
|
|
}
|
|
|
|
void RemapDialog::clearMapping(uint i) {
|
|
if (_topAction + i >= _currentActions.size())
|
|
return;
|
|
|
|
debug(3, "clear the mapping %u", i);
|
|
_activeRemapAction = _currentActions[_topAction + i].action;
|
|
_activeRemapAction->mapKey(0);
|
|
_activeRemapAction->getParent()->saveMappings();
|
|
_changes = true;
|
|
|
|
// force refresh
|
|
_topAction = -1;
|
|
refreshKeymap();
|
|
|
|
_activeRemapAction = 0;
|
|
}
|
|
|
|
void RemapDialog::startRemapping(uint i) {
|
|
if (_topAction + i >= _currentActions.size())
|
|
return;
|
|
|
|
_remapTimeout = g_system->getMillis() + kRemapTimeoutDelay;
|
|
_activeRemapAction = _currentActions[_topAction + i].action;
|
|
_keymapWidgets[i].keyButton->setLabel("...");
|
|
_keymapWidgets[i].keyButton->draw();
|
|
_keymapper->setEnabled(false);
|
|
|
|
}
|
|
|
|
void RemapDialog::stopRemapping() {
|
|
_topAction = -1;
|
|
|
|
refreshKeymap();
|
|
|
|
_activeRemapAction = 0;
|
|
|
|
_keymapper->setEnabled(true);
|
|
}
|
|
|
|
void RemapDialog::handleKeyDown(Common::KeyState state) {
|
|
if (_activeRemapAction)
|
|
return;
|
|
|
|
GUI::Dialog::handleKeyDown(state);
|
|
}
|
|
|
|
void RemapDialog::handleKeyUp(Common::KeyState state) {
|
|
if (_activeRemapAction) {
|
|
const HardwareKey *hwkey = _keymapper->findHardwareKey(state);
|
|
|
|
debug(4, "RemapDialog::handleKeyUp Key: %d, %d (%c), %x", state.keycode, state.ascii, (state.ascii ? state.ascii : ' '), state.flags);
|
|
|
|
if (hwkey) {
|
|
_activeRemapAction->mapKey(hwkey);
|
|
_activeRemapAction->getParent()->saveMappings();
|
|
_changes = true;
|
|
stopRemapping();
|
|
}
|
|
} else {
|
|
GUI::Dialog::handleKeyUp(state);
|
|
}
|
|
}
|
|
|
|
void RemapDialog::handleMouseDown(int x, int y, int button, int clickCount) {
|
|
if (_activeRemapAction)
|
|
stopRemapping();
|
|
else
|
|
Dialog::handleMouseDown(x, y, button, clickCount);
|
|
}
|
|
|
|
void RemapDialog::handleTickle() {
|
|
if (_activeRemapAction && g_system->getMillis() > _remapTimeout)
|
|
stopRemapping();
|
|
Dialog::handleTickle();
|
|
}
|
|
|
|
void RemapDialog::loadKeymap() {
|
|
_currentActions.clear();
|
|
const Stack<Keymapper::MapRecord> &activeKeymaps = _keymapper->getActiveStack();
|
|
|
|
debug(3, "RemapDialog::loadKeymap active keymaps: %u", activeKeymaps.size());
|
|
|
|
if (!activeKeymaps.empty() && _kmPopUp->getSelected() == 0) {
|
|
// load active keymaps
|
|
|
|
List<const HardwareKey*> freeKeys(_keymapper->getHardwareKeys());
|
|
|
|
int topIndex = activeKeymaps.size() - 1;
|
|
// skip all gui maps
|
|
// TODO: Don't use the keymap name as a way to discriminate GUI maps
|
|
while (topIndex > 0 && activeKeymaps[topIndex].keymap->getName().equals(kGuiKeymapName))
|
|
--topIndex;
|
|
|
|
// add most active keymap's keys
|
|
Keymapper::MapRecord top = activeKeymaps[topIndex];
|
|
List<Action*>::iterator actIt;
|
|
debug(3, "RemapDialog::loadKeymap top keymap: %s", top.keymap->getName().c_str());
|
|
for (actIt = top.keymap->getActions().begin(); actIt != top.keymap->getActions().end(); ++actIt) {
|
|
Action *act = *actIt;
|
|
ActionInfo info = {act, false, act->description};
|
|
|
|
_currentActions.push_back(info);
|
|
|
|
if (act->getMappedKey())
|
|
freeKeys.remove(act->getMappedKey());
|
|
}
|
|
|
|
// loop through remaining finding mappings for unmapped keys
|
|
if (top.inherit && topIndex >= 0) {
|
|
for (int i = topIndex - 1; i >= 0; --i) {
|
|
Keymapper::MapRecord mr = activeKeymaps[i];
|
|
debug(3, "RemapDialog::loadKeymap keymap: %s", mr.keymap->getName().c_str());
|
|
List<const HardwareKey*>::iterator keyIt = freeKeys.begin();
|
|
|
|
while (keyIt != freeKeys.end()) {
|
|
Action *act = mr.keymap->getMappedAction((*keyIt)->key);
|
|
|
|
if (act) {
|
|
ActionInfo info = {act, true, act->description + " (" + mr.keymap->getName() + ")"};
|
|
_currentActions.push_back(info);
|
|
freeKeys.erase(keyIt++);
|
|
} else {
|
|
++keyIt;
|
|
}
|
|
}
|
|
|
|
if (mr.inherit == false || freeKeys.empty())
|
|
break;
|
|
}
|
|
}
|
|
|
|
// get the mapping out of the global keymap
|
|
if (_keymapper->getGlobalKeymap()) {
|
|
List<const HardwareKey*>::iterator keyIt = freeKeys.begin();
|
|
Keymap *globalKeymap = _keymapper->getGlobalKeymap();
|
|
while (keyIt != freeKeys.end()) {
|
|
Action *act = globalKeymap->getMappedAction((*keyIt)->key);
|
|
|
|
if (act) {
|
|
ActionInfo info = {act, true, act->description + " (" + globalKeymap->getName() + ")"};
|
|
_currentActions.push_back(info);
|
|
freeKeys.erase(keyIt++);
|
|
} else {
|
|
++keyIt;
|
|
}
|
|
if (freeKeys.empty())
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
} else if (_kmPopUp->getSelected() != -1) {
|
|
Keymap *km = _keymapTable[_kmPopUp->getSelectedTag()];
|
|
|
|
List<Action*>::iterator it;
|
|
|
|
for (it = km->getActions().begin(); it != km->getActions().end(); ++it) {
|
|
ActionInfo info = {*it, false, (*it)->description};
|
|
|
|
_currentActions.push_back(info);
|
|
}
|
|
}
|
|
|
|
// refresh scroll bar
|
|
_scrollBar->_currentPos = 0;
|
|
_scrollBar->_numEntries = (_currentActions.size() + _colCount - 1) / _colCount;
|
|
_scrollBar->recalc();
|
|
|
|
// force refresh
|
|
_topAction = -1;
|
|
refreshKeymap();
|
|
}
|
|
|
|
void RemapDialog::refreshKeymap() {
|
|
int newTopAction = _scrollBar->_currentPos * _colCount;
|
|
|
|
if (newTopAction == _topAction)
|
|
return;
|
|
|
|
_topAction = newTopAction;
|
|
|
|
//_container->draw();
|
|
_scrollBar->draw();
|
|
|
|
uint actionI = _topAction;
|
|
|
|
for (uint widgetI = 0; widgetI < _keymapWidgets.size(); widgetI++) {
|
|
ActionWidgets& widg = _keymapWidgets[widgetI];
|
|
|
|
if (actionI < _currentActions.size()) {
|
|
debug(8, "RemapDialog::refreshKeymap actionI=%u", actionI);
|
|
ActionInfo& info = _currentActions[actionI];
|
|
|
|
widg.actionText->setLabel(info.description + ": ");
|
|
widg.actionText->setEnabled(!info.inherited);
|
|
|
|
const HardwareKey *mappedKey = info.action->getMappedKey();
|
|
|
|
if (mappedKey)
|
|
widg.keyButton->setLabel(mappedKey->description);
|
|
else
|
|
widg.keyButton->setLabel("-");
|
|
|
|
widg.actionText->setVisible(true);
|
|
widg.keyButton->setVisible(true);
|
|
widg.clearButton->setVisible(true);
|
|
|
|
actionI++;
|
|
} else {
|
|
widg.actionText->setVisible(false);
|
|
widg.keyButton->setVisible(false);
|
|
widg.clearButton->setVisible(false);
|
|
}
|
|
//widg.actionText->draw();
|
|
//widg.keyButton->draw();
|
|
}
|
|
// need to redraw entire Dialog so that invisible
|
|
// widgets disappear
|
|
draw();
|
|
}
|
|
|
|
|
|
} // End of namespace Common
|
|
|
|
#endif // #ifdef ENABLE_KEYMAPPER
|