#include "inputbindingwidgets.h" #include "core/settings.h" #include "frontend-common/controller_interface.h" #include "qthostinterface.h" #include "qtutils.h" #include #include #include #include InputBindingWidget::InputBindingWidget(QtHostInterface* host_interface, QString setting_name, QWidget* parent) : QPushButton(parent), m_host_interface(host_interface), m_setting_name(std::move(setting_name)) { m_current_binding_value = m_host_interface->getSettingValue(m_setting_name).toString(); setText(m_current_binding_value); connect(this, &QPushButton::pressed, this, &InputBindingWidget::onPressed); } InputBindingWidget::~InputBindingWidget() { Q_ASSERT(!isListeningForInput()); } void InputBindingWidget::beginRebindAll() { m_is_binding_all = true; if (isListeningForInput()) stopListeningForInput(); startListeningForInput(TIMEOUT_FOR_ALL_BINDING); } bool InputBindingWidget::eventFilter(QObject* watched, QEvent* event) { const QEvent::Type event_type = event->type(); if (event_type == QEvent::MouseButtonPress || event_type == QEvent::MouseButtonRelease || event_type == QEvent::MouseButtonDblClick) { return true; } return false; } void InputBindingWidget::mouseReleaseEvent(QMouseEvent* e) { if (e->button() == Qt::RightButton) { clearBinding(); return; } QPushButton::mouseReleaseEvent(e); } void InputBindingWidget::setNewBinding() { if (m_new_binding_value.isEmpty()) return; m_host_interface->putSettingValue(m_setting_name, m_new_binding_value); m_host_interface->updateInputMap(); m_current_binding_value = std::move(m_new_binding_value); m_new_binding_value.clear(); } void InputBindingWidget::clearBinding() { m_current_binding_value.clear(); m_host_interface->removeSettingValue(m_setting_name); m_host_interface->updateInputMap(); setText(m_current_binding_value); } void InputBindingWidget::onPressed() { if (isListeningForInput()) stopListeningForInput(); startListeningForInput(TIMEOUT_FOR_SINGLE_BINDING); } void InputBindingWidget::onInputListenTimerTimeout() { m_input_listen_remaining_seconds--; if (m_input_listen_remaining_seconds == 0) { stopListeningForInput(); return; } setText(tr("Push Button/Axis... [%1]").arg(m_input_listen_remaining_seconds)); } void InputBindingWidget::startListeningForInput(u32 timeout_in_seconds) { m_input_listen_timer = new QTimer(this); m_input_listen_timer->setSingleShot(false); m_input_listen_timer->start(1000); m_input_listen_timer->connect(m_input_listen_timer, &QTimer::timeout, this, &InputBindingWidget::onInputListenTimerTimeout); m_input_listen_remaining_seconds = timeout_in_seconds; setText(tr("Push Button/Axis... [%1]").arg(m_input_listen_remaining_seconds)); installEventFilter(this); grabKeyboard(); grabMouse(); } void InputBindingWidget::stopListeningForInput() { setText(m_current_binding_value); delete m_input_listen_timer; m_input_listen_timer = nullptr; releaseMouse(); releaseKeyboard(); removeEventFilter(this); if (m_is_binding_all && m_next_widget) m_next_widget->beginRebindAll(); m_is_binding_all = false; } InputButtonBindingWidget::InputButtonBindingWidget(QtHostInterface* host_interface, QString setting_name, QWidget* parent) : InputBindingWidget(host_interface, setting_name, parent) { } InputButtonBindingWidget::~InputButtonBindingWidget() { if (isListeningForInput()) InputButtonBindingWidget::stopListeningForInput(); } bool InputButtonBindingWidget::eventFilter(QObject* watched, QEvent* event) { const QEvent::Type event_type = event->type(); // if the key is being released, set the input if (event_type == QEvent::KeyRelease) { setNewBinding(); stopListeningForInput(); return true; } else if (event_type == QEvent::KeyPress) { QString binding = QtUtils::KeyEventToString(static_cast(event)); if (!binding.isEmpty()) m_new_binding_value = QStringLiteral("Keyboard/%1").arg(binding); return true; } return InputBindingWidget::eventFilter(watched, event); } void InputButtonBindingWidget::hookControllerInput() { ControllerInterface* controller_interface = m_host_interface->getControllerInterface(); if (!controller_interface) return; m_host_interface->enableBackgroundControllerPolling(); controller_interface->SetHook([this](const ControllerInterface::Hook& ei) { if (ei.type == ControllerInterface::Hook::Type::Axis) { // wait until it's at least half pushed so we don't get confused between axises with small movement if (std::abs(ei.value) < 0.5f) return ControllerInterface::Hook::CallbackResult::ContinueMonitoring; // TODO: this probably should consider the "last value" QMetaObject::invokeMethod(this, "bindToControllerAxis", Q_ARG(int, ei.controller_index), Q_ARG(int, ei.button_or_axis_number), Q_ARG(bool, ei.value > 0)); return ControllerInterface::Hook::CallbackResult::StopMonitoring; } else if (ei.type == ControllerInterface::Hook::Type::Button && ei.value > 0.0f) { QMetaObject::invokeMethod(this, "bindToControllerButton", Q_ARG(int, ei.controller_index), Q_ARG(int, ei.button_or_axis_number)); return ControllerInterface::Hook::CallbackResult::StopMonitoring; } return ControllerInterface::Hook::CallbackResult::ContinueMonitoring; }); } void InputButtonBindingWidget::unhookControllerInput() { ControllerInterface* controller_interface = m_host_interface->getControllerInterface(); if (!controller_interface) return; controller_interface->ClearHook(); m_host_interface->disableBackgroundControllerPolling(); } void InputButtonBindingWidget::bindToControllerAxis(int controller_index, int axis_index, bool positive) { m_new_binding_value = QStringLiteral("Controller%1/%2Axis%3").arg(controller_index).arg(positive ? '+' : '-').arg(axis_index); setNewBinding(); stopListeningForInput(); } void InputButtonBindingWidget::bindToControllerButton(int controller_index, int button_index) { m_new_binding_value = QStringLiteral("Controller%1/Button%2").arg(controller_index).arg(button_index); setNewBinding(); stopListeningForInput(); } void InputButtonBindingWidget::startListeningForInput(u32 timeout_in_seconds) { InputBindingWidget::startListeningForInput(timeout_in_seconds); hookControllerInput(); } void InputButtonBindingWidget::stopListeningForInput() { unhookControllerInput(); InputBindingWidget::stopListeningForInput(); } InputAxisBindingWidget::InputAxisBindingWidget(QtHostInterface* host_interface, QString setting_name, QWidget* parent) : InputBindingWidget(host_interface, setting_name, parent) { } InputAxisBindingWidget::~InputAxisBindingWidget() { if (isListeningForInput()) InputAxisBindingWidget::stopListeningForInput(); } void InputAxisBindingWidget::hookControllerInput() { ControllerInterface* controller_interface = m_host_interface->getControllerInterface(); if (!controller_interface) return; m_host_interface->enableBackgroundControllerPolling(); controller_interface->SetHook([this](const ControllerInterface::Hook& ei) { if (ei.type == ControllerInterface::Hook::Type::Axis) { // wait until it's at least half pushed so we don't get confused between axises with small movement if (std::abs(ei.value) < 0.5f) return ControllerInterface::Hook::CallbackResult::ContinueMonitoring; QMetaObject::invokeMethod(this, "bindToControllerAxis", Q_ARG(int, ei.controller_index), Q_ARG(int, ei.button_or_axis_number)); return ControllerInterface::Hook::CallbackResult::StopMonitoring; } return ControllerInterface::Hook::CallbackResult::ContinueMonitoring; }); } void InputAxisBindingWidget::unhookControllerInput() { ControllerInterface* controller_interface = m_host_interface->getControllerInterface(); if (!controller_interface) return; controller_interface->ClearHook(); m_host_interface->disableBackgroundControllerPolling(); } void InputAxisBindingWidget::bindToControllerAxis(int controller_index, int axis_index) { m_new_binding_value = QStringLiteral("Controller%1/Axis%2").arg(controller_index).arg(axis_index); setNewBinding(); stopListeningForInput(); } void InputAxisBindingWidget::startListeningForInput(u32 timeout_in_seconds) { InputBindingWidget::startListeningForInput(timeout_in_seconds); hookControllerInput(); } void InputAxisBindingWidget::stopListeningForInput() { unhookControllerInput(); InputBindingWidget::stopListeningForInput(); }