Qt: Ability to prevent opposing directional input

This commit is contained in:
Jeffrey Pfau 2015-03-23 00:11:19 -07:00
parent 302b041507
commit 8fad1d136c
8 changed files with 83 additions and 38 deletions

View File

@ -31,6 +31,7 @@ Features:
- Ability to lock aspect ratio - Ability to lock aspect ratio
- Local link cable support - Local link cable support
- Ability to switch which game controller is in use per instance - Ability to switch which game controller is in use per instance
- Ability to prevent opposing directional input
Bugfixes: Bugfixes:
- ARM7: Extend prefetch by one stage - ARM7: Extend prefetch by one stage
- GBA Audio: Support 16-bit writes to FIFO audio - GBA Audio: Support 16-bit writes to FIFO audio

View File

@ -33,6 +33,7 @@ GameController::GameController(QObject* parent)
, m_drawContext(new uint32_t[256 * 256]) , m_drawContext(new uint32_t[256 * 256])
, m_threadContext() , m_threadContext()
, m_activeKeys(0) , m_activeKeys(0)
, m_inactiveKeys(0)
, m_logLevels(0) , m_logLevels(0)
, m_gameOpen(false) , m_gameOpen(false)
, m_audioThread(new QThread(this)) , m_audioThread(new QThread(this))
@ -390,17 +391,38 @@ void GameController::rewind(int states) {
void GameController::keyPressed(int key) { void GameController::keyPressed(int key) {
int mappedKey = 1 << key; int mappedKey = 1 << key;
m_activeKeys |= mappedKey; m_activeKeys |= mappedKey;
if (!m_inputController->allowOpposing()) {
if ((m_activeKeys & 0x30) == 0x30) {
m_inactiveKeys |= mappedKey ^ 0x30;
m_activeKeys ^= mappedKey ^ 0x30;
}
if ((m_activeKeys & 0xC0) == 0xC0) {
m_inactiveKeys |= mappedKey ^ 0xC0;
m_activeKeys ^= mappedKey ^ 0xC0;
}
}
updateKeys(); updateKeys();
} }
void GameController::keyReleased(int key) { void GameController::keyReleased(int key) {
int mappedKey = 1 << key; int mappedKey = 1 << key;
m_activeKeys &= ~mappedKey; m_activeKeys &= ~mappedKey;
if (!m_inputController->allowOpposing()) {
if (mappedKey & 0x30) {
m_activeKeys |= m_inactiveKeys & (0x30 ^ mappedKey);
m_inactiveKeys &= ~0x30;
}
if (mappedKey & 0xC0) {
m_activeKeys |= m_inactiveKeys & (0xC0 ^ mappedKey);
m_inactiveKeys &= ~0xC0;
}
}
updateKeys(); updateKeys();
} }
void GameController::clearKeys() { void GameController::clearKeys() {
m_activeKeys = 0; m_activeKeys = 0;
m_inactiveKeys = 0;
updateKeys(); updateKeys();
} }
@ -549,6 +571,7 @@ void GameController::updateKeys() {
#ifdef BUILD_SDL #ifdef BUILD_SDL
activeKeys |= m_activeButtons; activeKeys |= m_activeButtons;
#endif #endif
activeKeys &= ~m_inactiveKeys;
m_threadContext.activeKeys = activeKeys; m_threadContext.activeKeys = activeKeys;
} }

View File

@ -152,6 +152,7 @@ private:
GBAVideoSoftwareRenderer* m_renderer; GBAVideoSoftwareRenderer* m_renderer;
GBACheatDevice m_cheatDevice; GBACheatDevice m_cheatDevice;
int m_activeKeys; int m_activeKeys;
int m_inactiveKeys;
int m_logLevels; int m_logLevels;
bool m_gameOpen; bool m_gameOpen;

View File

@ -28,6 +28,7 @@ InputController::InputController(int playerId, QObject* parent)
, m_config(nullptr) , m_config(nullptr)
, m_gamepadTimer(nullptr) , m_gamepadTimer(nullptr)
, m_playerAttached(false) , m_playerAttached(false)
, m_allowOpposing(false)
{ {
GBAInputMapInit(&m_inputMap); GBAInputMapInit(&m_inputMap);
@ -70,6 +71,7 @@ InputController::~InputController() {
void InputController::setConfiguration(ConfigController* config) { void InputController::setConfiguration(ConfigController* config) {
m_config = config; m_config = config;
setAllowOpposing(config->getOption("allowOpposingDirections").toInt());
loadConfiguration(KEYBOARD); loadConfiguration(KEYBOARD);
#ifdef BUILD_SDL #ifdef BUILD_SDL
GBASDLEventsLoadConfig(&s_sdlEvents, config->input()); GBASDLEventsLoadConfig(&s_sdlEvents, config->input());

View File

@ -41,6 +41,9 @@ public:
void saveProfile(uint32_t type, const QString& profile); void saveProfile(uint32_t type, const QString& profile);
const char* profileForType(uint32_t type); const char* profileForType(uint32_t type);
bool allowOpposing() const { return m_allowOpposing; }
void setAllowOpposing(bool allowOpposing) { m_allowOpposing = allowOpposing; }
GBAKey mapKeyboard(int key) const; GBAKey mapKeyboard(int key) const;
void bindKey(uint32_t type, int key, GBAKey); void bindKey(uint32_t type, int key, GBAKey);
@ -73,6 +76,7 @@ private:
GBAInputMap m_inputMap; GBAInputMap m_inputMap;
ConfigController* m_config; ConfigController* m_config;
int m_playerId; int m_playerId;
bool m_allowOpposing;
#ifdef BUILD_SDL #ifdef BUILD_SDL
static int s_sdlInited; static int s_sdlInited;

View File

@ -31,6 +31,7 @@ SettingsView::SettingsView(ConfigController* controller, QWidget* parent)
loadSetting("rewindBufferInterval", m_ui.rewindInterval); loadSetting("rewindBufferInterval", m_ui.rewindInterval);
loadSetting("rewindBufferCapacity", m_ui.rewindCapacity); loadSetting("rewindBufferCapacity", m_ui.rewindCapacity);
loadSetting("resampleVideo", m_ui.resampleVideo); loadSetting("resampleVideo", m_ui.resampleVideo);
loadSetting("allowOpposingDirections", m_ui.allowOpposingDirections);
QString idleOptimization = loadSetting("idleOptimization"); QString idleOptimization = loadSetting("idleOptimization");
if (idleOptimization == "ignore") { if (idleOptimization == "ignore") {
@ -81,6 +82,7 @@ void SettingsView::updateConfig() {
saveSetting("rewindBufferInterval", m_ui.rewindInterval); saveSetting("rewindBufferInterval", m_ui.rewindInterval);
saveSetting("rewindBufferCapacity", m_ui.rewindCapacity); saveSetting("rewindBufferCapacity", m_ui.rewindCapacity);
saveSetting("resampleVideo", m_ui.resampleVideo); saveSetting("resampleVideo", m_ui.resampleVideo);
saveSetting("allowOpposingDirections", m_ui.allowOpposingDirections);
switch (m_ui.idleOptimization->currentIndex() + IDLE_LOOP_IGNORE) { switch (m_ui.idleOptimization->currentIndex() + IDLE_LOOP_IGNORE) {
case IDLE_LOOP_IGNORE: case IDLE_LOOP_IGNORE:
@ -130,7 +132,7 @@ void SettingsView::saveSetting(const char* key, const QString& field) {
void SettingsView::loadSetting(const char* key, QAbstractButton* field) { void SettingsView::loadSetting(const char* key, QAbstractButton* field) {
QString option = loadSetting(key); QString option = loadSetting(key);
field->setChecked(option != "0"); field->setChecked(!option.isNull() && option != "0");
} }
void SettingsView::loadSetting(const char* key, QComboBox* field) { void SettingsView::loadSetting(const char* key, QComboBox* field) {

View File

@ -6,8 +6,8 @@
<rect> <rect>
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>355</width> <width>360</width>
<height>501</height> <height>569</height>
</rect> </rect>
</property> </property>
<property name="sizePolicy"> <property name="sizePolicy">
@ -198,6 +198,34 @@
</item> </item>
</layout> </layout>
</item> </item>
<item row="8" column="0">
<widget class="QLabel" name="label_3">
<property name="text">
<string>FPS target:</string>
</property>
</widget>
</item>
<item row="8" column="1">
<layout class="QHBoxLayout" name="horizontalLayout_2">
<item>
<widget class="QSpinBox" name="fpsTarget">
<property name="maximum">
<number>240</number>
</property>
<property name="value">
<number>60</number>
</property>
</widget>
</item>
<item>
<widget class="QLabel" name="label_11">
<property name="text">
<string>frames per second</string>
</property>
</widget>
</item>
</layout>
</item>
<item row="9" column="0" colspan="2"> <item row="9" column="0" colspan="2">
<widget class="Line" name="line"> <widget class="Line" name="line">
<property name="orientation"> <property name="orientation">
@ -282,35 +310,21 @@
</item> </item>
</layout> </layout>
</item> </item>
<item row="8" column="0"> <item row="17" column="0" colspan="2">
<widget class="QLabel" name="label_3"> <widget class="Line" name="line_4">
<property name="text"> <property name="orientation">
<string>FPS target:</string> <enum>Qt::Horizontal</enum>
</property> </property>
</widget> </widget>
</item> </item>
<item row="8" column="1"> <item row="18" column="0">
<layout class="QHBoxLayout" name="horizontalLayout_2"> <widget class="QLabel" name="label_15">
<item> <property name="text">
<widget class="QSpinBox" name="fpsTarget"> <string>Idle loops</string>
<property name="maximum"> </property>
<number>240</number> </widget>
</property>
<property name="value">
<number>60</number>
</property>
</widget>
</item>
<item>
<widget class="QLabel" name="label_11">
<property name="text">
<string>frames per second</string>
</property>
</widget>
</item>
</layout>
</item> </item>
<item row="17" column="1"> <item row="18" column="1">
<widget class="QComboBox" name="idleOptimization"> <widget class="QComboBox" name="idleOptimization">
<item> <item>
<property name="text"> <property name="text">
@ -329,17 +343,10 @@
</item> </item>
</widget> </widget>
</item> </item>
<item row="16" column="0" colspan="2"> <item row="16" column="1">
<widget class="Line" name="line_4"> <widget class="QCheckBox" name="allowOpposingDirections">
<property name="orientation">
<enum>Qt::Horizontal</enum>
</property>
</widget>
</item>
<item row="17" column="0">
<widget class="QLabel" name="label_15">
<property name="text"> <property name="text">
<string>Idle loops</string> <string>Allow opposing input directions</string>
</property> </property>
</widget> </widget>
</item> </item>

View File

@ -789,6 +789,11 @@ void Window::setupMenu(QMenuBar* menubar) {
m_controller->setRewind(m_config->getOption("rewindEnable").toInt(), m_config->getOption("rewindBufferCapacity").toInt(), value.toInt()); m_controller->setRewind(m_config->getOption("rewindEnable").toInt(), m_config->getOption("rewindBufferCapacity").toInt(), value.toInt());
}, this); }, this);
ConfigOption* allowOpposingDirections = m_config->addOption("allowOpposingDirections");
allowOpposingDirections->connect([this](const QVariant& value) {
m_inputController.setAllowOpposing(value.toBool());
}, this);
QMenu* other = new QMenu(tr("Other"), this); QMenu* other = new QMenu(tr("Other"), this);
m_shortcutController->addMenu(other); m_shortcutController->addMenu(other);
m_shortcutController->addFunctions(other, [this]() { m_shortcutController->addFunctions(other, [this]() {