// Copyright 2018 Dolphin Emulator Project // SPDX-License-Identifier: GPL-2.0-or-later #include "DolphinQt/TAS/TASInputWindow.h" #include #include #include #include #include #include #include #include #include #include #include "Common/CommonTypes.h" #include "DolphinQt/QtUtils/AspectRatioWidget.h" #include "DolphinQt/QtUtils/QueueOnObject.h" #include "DolphinQt/Resources.h" #include "DolphinQt/TAS/StickWidget.h" #include "DolphinQt/TAS/TASCheckBox.h" #include "DolphinQt/TAS/TASSlider.h" #include "InputCommon/ControllerEmu/ControllerEmu.h" #include "InputCommon/ControllerEmu/StickGate.h" void InputOverrider::AddFunction(std::string_view group_name, std::string_view control_name, OverrideFunction function) { m_functions.emplace(std::make_pair(group_name, control_name), std::move(function)); } ControllerEmu::InputOverrideFunction InputOverrider::GetInputOverrideFunction() const { return [this](std::string_view group_name, std::string_view control_name, ControlState controller_state) { const auto it = m_functions.find(std::make_pair(group_name, control_name)); return it != m_functions.end() ? it->second(controller_state) : std::nullopt; }; } TASInputWindow::TASInputWindow(QWidget* parent) : QDialog(parent) { setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint); setWindowIcon(Resources::GetAppIcon()); QGridLayout* settings_layout = new QGridLayout; m_use_controller = new QCheckBox(QStringLiteral("Enable Controller Inpu&t")); m_use_controller->setToolTip(tr("Warning: Analog inputs may reset to controller values at " "random. In some cases this can be fixed by adding a deadzone.")); settings_layout->addWidget(m_use_controller, 0, 0, 1, 2); QLabel* turbo_press_label = new QLabel(tr("Duration of Turbo Button Press (frames):")); m_turbo_press_frames = new QSpinBox(); m_turbo_press_frames->setMinimum(1); settings_layout->addWidget(turbo_press_label, 1, 0); settings_layout->addWidget(m_turbo_press_frames, 1, 1); QLabel* turbo_release_label = new QLabel(tr("Duration of Turbo Button Release (frames):")); m_turbo_release_frames = new QSpinBox(); m_turbo_release_frames->setMinimum(1); settings_layout->addWidget(turbo_release_label, 2, 0); settings_layout->addWidget(m_turbo_release_frames, 2, 1); m_settings_box = new QGroupBox(tr("Settings")); m_settings_box->setLayout(settings_layout); } int TASInputWindow::GetTurboPressFrames() const { return m_turbo_press_frames->value(); } int TASInputWindow::GetTurboReleaseFrames() const { return m_turbo_release_frames->value(); } TASCheckBox* TASInputWindow::CreateButton(const QString& text, std::string_view group_name, std::string_view control_name, InputOverrider* overrider) { TASCheckBox* checkbox = new TASCheckBox(text, this); overrider->AddFunction(group_name, control_name, [this, checkbox](ControlState controller_state) { return GetButton(checkbox, controller_state); }); return checkbox; } QGroupBox* TASInputWindow::CreateStickInputs(const QString& text, std::string_view group_name, InputOverrider* overrider, QSpinBox*& x_value, QSpinBox*& y_value, u16 min_x, u16 min_y, u16 max_x, u16 max_y, Qt::Key x_shortcut_key, Qt::Key y_shortcut_key) { const QKeySequence x_shortcut_key_sequence = QKeySequence(Qt::ALT | x_shortcut_key); const QKeySequence y_shortcut_key_sequence = QKeySequence(Qt::ALT | y_shortcut_key); auto* box = new QGroupBox(QStringLiteral("%1 (%2/%3)") .arg(text, x_shortcut_key_sequence.toString(QKeySequence::NativeText), y_shortcut_key_sequence.toString(QKeySequence::NativeText))); const int x_default = static_cast(std::round(max_x / 2.)); const int y_default = static_cast(std::round(max_y / 2.)); auto* x_layout = new QHBoxLayout; x_value = CreateSliderValuePair(x_layout, x_default, max_x, x_shortcut_key_sequence, Qt::Horizontal, box); auto* y_layout = new QVBoxLayout; y_value = CreateSliderValuePair(y_layout, y_default, max_y, y_shortcut_key_sequence, Qt::Vertical, box); y_value->setMaximumWidth(60); auto* visual = new StickWidget(this, max_x, max_y); visual->SetX(x_default); visual->SetY(y_default); connect(x_value, qOverload(&QSpinBox::valueChanged), visual, &StickWidget::SetX); connect(y_value, qOverload(&QSpinBox::valueChanged), visual, &StickWidget::SetY); connect(visual, &StickWidget::ChangedX, x_value, &QSpinBox::setValue); connect(visual, &StickWidget::ChangedY, y_value, &QSpinBox::setValue); auto* visual_ar = new AspectRatioWidget(visual, max_x, max_y); auto* visual_layout = new QHBoxLayout; visual_layout->addWidget(visual_ar); visual_layout->addLayout(y_layout); auto* layout = new QVBoxLayout; layout->addLayout(x_layout); layout->addLayout(visual_layout); box->setLayout(layout); overrider->AddFunction(group_name, ControllerEmu::ReshapableInput::X_INPUT_OVERRIDE, [this, x_value, x_default, min_x, max_x](ControlState controller_state) { return GetSpinBox(x_value, x_default, min_x, max_x, controller_state); }); overrider->AddFunction(group_name, ControllerEmu::ReshapableInput::Y_INPUT_OVERRIDE, [this, y_value, y_default, min_y, max_y](ControlState controller_state) { return GetSpinBox(y_value, y_default, min_y, max_y, controller_state); }); return box; } QBoxLayout* TASInputWindow::CreateSliderValuePairLayout( const QString& text, std::string_view group_name, std::string_view control_name, InputOverrider* overrider, QSpinBox*& value, u16 zero, int default_, u16 min, u16 max, Qt::Key shortcut_key, QWidget* shortcut_widget, std::optional scale) { const QKeySequence shortcut_key_sequence = QKeySequence(Qt::ALT | shortcut_key); auto* label = new QLabel(QStringLiteral("%1 (%2)").arg( text, shortcut_key_sequence.toString(QKeySequence::NativeText))); QBoxLayout* layout = new QHBoxLayout; layout->addWidget(label); value = CreateSliderValuePair(group_name, control_name, overrider, layout, zero, default_, min, max, shortcut_key_sequence, Qt::Horizontal, shortcut_widget, scale); return layout; } QSpinBox* TASInputWindow::CreateSliderValuePair( std::string_view group_name, std::string_view control_name, InputOverrider* overrider, QBoxLayout* layout, u16 zero, int default_, u16 min, u16 max, QKeySequence shortcut_key_sequence, Qt::Orientation orientation, QWidget* shortcut_widget, std::optional scale) { QSpinBox* value = CreateSliderValuePair(layout, default_, max, shortcut_key_sequence, orientation, shortcut_widget); InputOverrider::OverrideFunction func; if (scale) { func = [this, value, zero, scale](ControlState controller_state) { return GetSpinBox(value, zero, controller_state, *scale); }; } else { func = [this, value, zero, min, max](ControlState controller_state) { return GetSpinBox(value, zero, min, max, controller_state); }; } overrider->AddFunction(group_name, control_name, std::move(func)); return value; } // The shortcut_widget argument needs to specify the container widget that will be hidden/shown. // This is done to avoid ambigous shortcuts QSpinBox* TASInputWindow::CreateSliderValuePair(QBoxLayout* layout, int default_, u16 max, QKeySequence shortcut_key_sequence, Qt::Orientation orientation, QWidget* shortcut_widget) { auto* value = new QSpinBox(); value->setRange(0, 99999); value->setValue(default_); connect(value, qOverload(&QSpinBox::valueChanged), [value, max](int i) { if (i > max) value->setValue(max); }); auto* slider = new TASSlider(default_, orientation); slider->setRange(0, max); slider->setValue(default_); slider->setFocusPolicy(Qt::ClickFocus); connect(slider, &QSlider::valueChanged, value, &QSpinBox::setValue); connect(value, qOverload(&QSpinBox::valueChanged), slider, &QSlider::setValue); auto* shortcut = new QShortcut(shortcut_key_sequence, shortcut_widget); connect(shortcut, &QShortcut::activated, [value] { value->setFocus(); value->selectAll(); }); layout->addWidget(slider); layout->addWidget(value); if (orientation == Qt::Vertical) layout->setAlignment(slider, Qt::AlignRight); return value; } std::optional TASInputWindow::GetButton(TASCheckBox* checkbox, ControlState controller_state) { const bool pressed = std::llround(controller_state) > 0; if (m_use_controller->isChecked()) checkbox->OnControllerValueChanged(pressed); return checkbox->GetValue() ? 1.0 : 0.0; } std::optional TASInputWindow::GetSpinBox(QSpinBox* spin, u16 zero, u16 min, u16 max, ControlState controller_state) { const u16 controller_value = ControllerEmu::EmulatedController::MapFloat(controller_state, zero, 0, max); if (m_use_controller->isChecked()) { if (!m_spinbox_most_recent_values.count(spin) || m_spinbox_most_recent_values[spin] != controller_value) { QueueOnObjectBlocking(spin, [spin, controller_value] { spin->setValue(controller_value); }); } m_spinbox_most_recent_values[spin] = controller_value; } else { m_spinbox_most_recent_values.clear(); } return ControllerEmu::EmulatedController::MapToFloat(spin->value(), zero, min, max); } std::optional TASInputWindow::GetSpinBox(QSpinBox* spin, u16 zero, ControlState controller_state, ControlState scale) { const u16 controller_value = static_cast(std::llround(controller_state * scale + zero)); if (m_use_controller->isChecked()) { if (!m_spinbox_most_recent_values.count(spin) || m_spinbox_most_recent_values[spin] != controller_value) { QueueOnObjectBlocking(spin, [spin, controller_value] { spin->setValue(controller_value); }); } m_spinbox_most_recent_values[spin] = controller_value; } else { m_spinbox_most_recent_values.clear(); } return (spin->value() - zero) / scale; }