ExpressionParser: Suppress inputs when hotkey modifiers are pressed.

This commit is contained in:
Jordan Woyak 2019-10-23 16:45:21 -05:00
parent 52547379c9
commit e6ba495486
1 changed files with 52 additions and 5 deletions

View File

@ -6,6 +6,7 @@
#include <cassert> #include <cassert>
#include <cmath> #include <cmath>
#include <iostream> #include <iostream>
#include <map>
#include <memory> #include <memory>
#include <regex> #include <regex>
#include <string> #include <string>
@ -13,6 +14,7 @@
#include <vector> #include <vector>
#include "Common/Common.h" #include "Common/Common.h"
#include "Common/ScopeGuard.h"
#include "Common/StringUtil.h" #include "Common/StringUtil.h"
#include "InputCommon/ControlReference/ExpressionParser.h" #include "InputCommon/ControlReference/ExpressionParser.h"
@ -22,6 +24,32 @@ namespace ciface::ExpressionParser
{ {
using namespace ciface::Core; using namespace ciface::Core;
class HotkeySuppressions
{
public:
bool IsSuppressed(Device::Input* input) const { return m_suppressions.count(input) != 0; }
using Suppressor = std::unique_ptr<Common::ScopeGuard>;
Suppressor MakeSuppressor(Device::Input* input)
{
++m_suppressions[input];
return std::make_unique<Common::ScopeGuard>([this, input]() { RemoveSuppression(input); });
}
private:
void RemoveSuppression(Device::Input* input)
{
auto it = m_suppressions.find(input);
if (--(it->second) == 0)
m_suppressions.erase(it);
}
std::map<Device::Input*, u16> m_suppressions;
};
static HotkeySuppressions s_hotkey_suppressions;
Token::Token(TokenType type_) : type(type_) Token::Token(TokenType type_) : type(type_)
{ {
} }
@ -201,7 +229,7 @@ public:
explicit ControlExpression(ControlQualifier qualifier_) : qualifier(qualifier_) {} explicit ControlExpression(ControlQualifier qualifier_) : qualifier(qualifier_) {}
ControlState GetValue() const override ControlState GetValue() const override
{ {
if (!input) if (!input || s_hotkey_suppressions.IsSuppressed(input))
return 0.0; return 0.0;
// Note: Inputs may return negative values in situations where opposing directions are // Note: Inputs may return negative values in situations where opposing directions are
@ -225,6 +253,8 @@ public:
output = env.FindOutput(qualifier); output = env.FindOutput(qualifier);
} }
Device::Input* GetInput() const { return input; };
private: private:
ControlQualifier qualifier; ControlQualifier qualifier;
Device::Input* input = nullptr; Device::Input* input = nullptr;
@ -398,17 +428,31 @@ public:
if (modifiers_pressed) if (modifiers_pressed)
{ {
auto& final_input = **m_inputs.rbegin();
// Remove supression before getting value.
m_suppressor = {};
const ControlState final_input_state = final_input.GetValue();
// TODO: kill magic number. // TODO: kill magic number.
const bool final_input_pressed = (**m_inputs.rbegin()).GetValue() > 0.5; if (final_input_state < 0.5)
m_is_ready = true;
if (m_is_ready) if (m_is_ready)
return final_input_pressed; {
// Only suppress input when we have at least one modifier.
if (m_inputs.size() > 1)
m_suppressor = s_hotkey_suppressions.MakeSuppressor(final_input.GetInput());
if (!final_input_pressed) return final_input_state;
m_is_ready = true; }
} }
else else
{
m_suppressor = {};
m_is_ready = false; m_is_ready = false;
}
return 0; return 0;
} }
@ -425,12 +469,15 @@ public:
void UpdateReferences(ControlEnvironment& env) override void UpdateReferences(ControlEnvironment& env) override
{ {
m_suppressor = {};
for (auto& input : m_inputs) for (auto& input : m_inputs)
input->UpdateReferences(env); input->UpdateReferences(env);
} }
private: private:
std::vector<std::unique_ptr<ControlExpression>> m_inputs; std::vector<std::unique_ptr<ControlExpression>> m_inputs;
mutable HotkeySuppressions::Suppressor m_suppressor;
mutable bool m_is_ready = false; mutable bool m_is_ready = false;
}; };