350 lines
12 KiB
C++
350 lines
12 KiB
C++
// Copyright 2017 Dolphin Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include "DolphinQt/Config/Mapping/MappingWidget.h"
|
|
|
|
#include <fmt/core.h>
|
|
|
|
#include <QCheckBox>
|
|
#include <QDialogButtonBox>
|
|
#include <QFormLayout>
|
|
#include <QGroupBox>
|
|
#include <QHBoxLayout>
|
|
#include <QLabel>
|
|
#include <QPushButton>
|
|
|
|
#include "DolphinQt/Config/Mapping/IOWindow.h"
|
|
#include "DolphinQt/Config/Mapping/MappingButton.h"
|
|
#include "DolphinQt/Config/Mapping/MappingIndicator.h"
|
|
#include "DolphinQt/Config/Mapping/MappingNumeric.h"
|
|
#include "DolphinQt/Config/Mapping/MappingWindow.h"
|
|
#include "DolphinQt/QtUtils/SetWindowDecorations.h"
|
|
|
|
#include "InputCommon/ControlReference/ControlReference.h"
|
|
#include "InputCommon/ControllerEmu/Control/Control.h"
|
|
#include "InputCommon/ControllerEmu/ControlGroup/ControlGroup.h"
|
|
#include "InputCommon/ControllerEmu/ControlGroup/MixedTriggers.h"
|
|
#include "InputCommon/ControllerEmu/ControllerEmu.h"
|
|
#include "InputCommon/ControllerEmu/Setting/NumericSetting.h"
|
|
#include "InputCommon/ControllerEmu/StickGate.h"
|
|
#include "InputCommon/ControllerInterface/ControllerInterface.h"
|
|
|
|
MappingWidget::MappingWidget(MappingWindow* parent) : m_parent(parent)
|
|
{
|
|
connect(parent, &MappingWindow::Update, this, &MappingWidget::Update);
|
|
connect(parent, &MappingWindow::Save, this, &MappingWidget::SaveSettings);
|
|
connect(parent, &MappingWindow::ConfigChanged, this, &MappingWidget::ConfigChanged);
|
|
}
|
|
|
|
MappingWindow* MappingWidget::GetParent() const
|
|
{
|
|
return m_parent;
|
|
}
|
|
|
|
int MappingWidget::GetPort() const
|
|
{
|
|
return m_parent->GetPort();
|
|
}
|
|
|
|
QGroupBox* MappingWidget::CreateGroupBox(ControllerEmu::ControlGroup* group)
|
|
{
|
|
return CreateGroupBox(tr(group->ui_name.c_str()), group);
|
|
}
|
|
|
|
QGroupBox* MappingWidget::CreateGroupBox(const QString& name, ControllerEmu::ControlGroup* group)
|
|
{
|
|
QGroupBox* group_box = new QGroupBox(name);
|
|
QFormLayout* form_layout = new QFormLayout();
|
|
|
|
group_box->setLayout(form_layout);
|
|
|
|
MappingIndicator* indicator = nullptr;
|
|
|
|
switch (group->type)
|
|
{
|
|
case ControllerEmu::GroupType::Shake:
|
|
indicator = new ShakeMappingIndicator(*static_cast<ControllerEmu::Shake*>(group));
|
|
break;
|
|
|
|
case ControllerEmu::GroupType::MixedTriggers:
|
|
indicator = new MixedTriggersIndicator(*static_cast<ControllerEmu::MixedTriggers*>(group));
|
|
break;
|
|
|
|
case ControllerEmu::GroupType::Tilt:
|
|
indicator = new TiltIndicator(*static_cast<ControllerEmu::Tilt*>(group));
|
|
break;
|
|
|
|
case ControllerEmu::GroupType::Cursor:
|
|
indicator = new CursorIndicator(*static_cast<ControllerEmu::Cursor*>(group));
|
|
break;
|
|
|
|
case ControllerEmu::GroupType::Force:
|
|
indicator = new SwingIndicator(*static_cast<ControllerEmu::Force*>(group));
|
|
break;
|
|
|
|
case ControllerEmu::GroupType::IMUAccelerometer:
|
|
indicator =
|
|
new AccelerometerMappingIndicator(*static_cast<ControllerEmu::IMUAccelerometer*>(group));
|
|
break;
|
|
|
|
case ControllerEmu::GroupType::IMUGyroscope:
|
|
indicator = new GyroMappingIndicator(*static_cast<ControllerEmu::IMUGyroscope*>(group));
|
|
break;
|
|
|
|
case ControllerEmu::GroupType::Stick:
|
|
indicator = new AnalogStickIndicator(*static_cast<ControllerEmu::ReshapableInput*>(group));
|
|
break;
|
|
|
|
case ControllerEmu::GroupType::IRPassthrough:
|
|
indicator =
|
|
new IRPassthroughMappingIndicator(*static_cast<ControllerEmu::IRPassthrough*>(group));
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (indicator)
|
|
{
|
|
const auto indicator_layout = new QBoxLayout(QBoxLayout::Direction::Down);
|
|
indicator_layout->addWidget(indicator);
|
|
indicator_layout->setAlignment(Qt::AlignCenter);
|
|
form_layout->addRow(indicator_layout);
|
|
|
|
connect(this, &MappingWidget::Update, indicator, qOverload<>(&MappingIndicator::update));
|
|
|
|
const bool need_calibration = group->type == ControllerEmu::GroupType::Cursor ||
|
|
group->type == ControllerEmu::GroupType::Stick ||
|
|
group->type == ControllerEmu::GroupType::Tilt ||
|
|
group->type == ControllerEmu::GroupType::Force;
|
|
|
|
if (need_calibration)
|
|
{
|
|
const auto calibrate =
|
|
new CalibrationWidget(*static_cast<ControllerEmu::ReshapableInput*>(group),
|
|
*static_cast<ReshapableInputIndicator*>(indicator));
|
|
|
|
form_layout->addRow(calibrate);
|
|
}
|
|
}
|
|
|
|
for (auto& control : group->controls)
|
|
CreateControl(control.get(), form_layout, !indicator);
|
|
|
|
AddSettingWidgets(form_layout, group, ControllerEmu::SettingVisibility::Normal);
|
|
|
|
if (group->default_value != ControllerEmu::ControlGroup::DefaultValue::AlwaysEnabled)
|
|
{
|
|
QLabel* group_enable_label = new QLabel(tr("Enable"));
|
|
QCheckBox* group_enable_checkbox = new QCheckBox();
|
|
group_enable_checkbox->setChecked(group->enabled);
|
|
form_layout->insertRow(0, group_enable_label, group_enable_checkbox);
|
|
auto enable_group_by_checkbox = [group, form_layout, group_enable_label,
|
|
group_enable_checkbox] {
|
|
group->enabled = group_enable_checkbox->isChecked();
|
|
for (int i = 0; i < form_layout->count(); ++i)
|
|
{
|
|
QWidget* widget = form_layout->itemAt(i)->widget();
|
|
if (widget != nullptr && widget != group_enable_label && widget != group_enable_checkbox)
|
|
widget->setEnabled(group->enabled);
|
|
}
|
|
};
|
|
enable_group_by_checkbox();
|
|
connect(group_enable_checkbox, &QCheckBox::toggled, this, enable_group_by_checkbox);
|
|
connect(this, &MappingWidget::ConfigChanged, this,
|
|
[group_enable_checkbox, group] { group_enable_checkbox->setChecked(group->enabled); });
|
|
}
|
|
|
|
const auto advanced_setting_count = std::count_if(
|
|
group->numeric_settings.begin(), group->numeric_settings.end(), [](auto& setting) {
|
|
return setting->GetVisibility() == ControllerEmu::SettingVisibility::Advanced;
|
|
});
|
|
|
|
if (advanced_setting_count != 0)
|
|
{
|
|
const auto advanced_button = new QPushButton(tr("Advanced"));
|
|
form_layout->addRow(advanced_button);
|
|
connect(advanced_button, &QPushButton::clicked,
|
|
[this, group] { ShowAdvancedControlGroupDialog(group); });
|
|
}
|
|
|
|
if (group->type == ControllerEmu::GroupType::Cursor)
|
|
{
|
|
QPushButton* mouse_button = new QPushButton(tr("Use Mouse Controlled Pointing"));
|
|
form_layout->insertRow(2, mouse_button);
|
|
connect(mouse_button, &QCheckBox::clicked, [this, group] {
|
|
std::string default_device = g_controller_interface.GetDefaultDeviceString() + ":";
|
|
const std::string controller_device = GetController()->GetDefaultDevice().ToString() + ":";
|
|
if (default_device == controller_device)
|
|
{
|
|
default_device.clear();
|
|
}
|
|
group->SetControlExpression(0, fmt::format("`{}Cursor Y-`", default_device));
|
|
group->SetControlExpression(1, fmt::format("`{}Cursor Y+`", default_device));
|
|
group->SetControlExpression(2, fmt::format("`{}Cursor X-`", default_device));
|
|
group->SetControlExpression(3, fmt::format("`{}Cursor X+`", default_device));
|
|
emit ConfigChanged();
|
|
GetController()->UpdateReferences(g_controller_interface);
|
|
});
|
|
}
|
|
|
|
return group_box;
|
|
}
|
|
|
|
void MappingWidget::AddSettingWidgets(QFormLayout* layout, ControllerEmu::ControlGroup* group,
|
|
ControllerEmu::SettingVisibility visibility)
|
|
{
|
|
for (auto& setting : group->numeric_settings)
|
|
{
|
|
if (setting->GetVisibility() != visibility)
|
|
continue;
|
|
|
|
QWidget* setting_widget = nullptr;
|
|
|
|
switch (setting->GetType())
|
|
{
|
|
case ControllerEmu::SettingType::Double:
|
|
setting_widget = new MappingDouble(
|
|
this, static_cast<ControllerEmu::NumericSetting<double>*>(setting.get()));
|
|
break;
|
|
|
|
case ControllerEmu::SettingType::Bool:
|
|
setting_widget =
|
|
new MappingBool(this, static_cast<ControllerEmu::NumericSetting<bool>*>(setting.get()));
|
|
break;
|
|
|
|
default:
|
|
// FYI: Widgets for additional types can be implemented as needed.
|
|
break;
|
|
}
|
|
|
|
if (setting_widget)
|
|
{
|
|
const auto hbox = new QHBoxLayout;
|
|
|
|
hbox->addWidget(setting_widget);
|
|
hbox->addWidget(CreateSettingAdvancedMappingButton(*setting));
|
|
|
|
layout->addRow(tr(setting->GetUIName()), hbox);
|
|
}
|
|
}
|
|
}
|
|
|
|
void MappingWidget::ShowAdvancedControlGroupDialog(ControllerEmu::ControlGroup* group)
|
|
{
|
|
QDialog dialog{this};
|
|
dialog.setWindowTitle(tr(group->ui_name.c_str()));
|
|
|
|
const auto group_box = new QGroupBox(tr("Advanced Settings"));
|
|
|
|
QFormLayout* form_layout = new QFormLayout();
|
|
|
|
AddSettingWidgets(form_layout, group, ControllerEmu::SettingVisibility::Advanced);
|
|
|
|
const auto reset_button = new QPushButton(tr("Reset All"));
|
|
form_layout->addRow(reset_button);
|
|
|
|
connect(reset_button, &QPushButton::clicked, [this, group] {
|
|
for (auto& setting : group->numeric_settings)
|
|
{
|
|
if (setting->GetVisibility() != ControllerEmu::SettingVisibility::Advanced)
|
|
continue;
|
|
|
|
setting->SetToDefault();
|
|
}
|
|
|
|
emit ConfigChanged();
|
|
});
|
|
|
|
const auto main_layout = new QVBoxLayout();
|
|
const auto button_box = new QDialogButtonBox(QDialogButtonBox::Close);
|
|
|
|
group_box->setLayout(form_layout);
|
|
|
|
main_layout->addWidget(group_box);
|
|
main_layout->addWidget(button_box);
|
|
|
|
dialog.setLayout(main_layout);
|
|
|
|
// Focusing something else by default instead of the first spin box.
|
|
// Dynamically changing expression-backed settings pause when taking input.
|
|
// This just avoids that weird edge case behavior when the dialog is first open.
|
|
button_box->setFocus();
|
|
|
|
// Signal the newly created numeric setting widgets to display the current values.
|
|
emit ConfigChanged();
|
|
|
|
// Enable "Close" button functionality.
|
|
connect(button_box, &QDialogButtonBox::rejected, &dialog, &QDialog::reject);
|
|
|
|
SetQWidgetWindowDecorations(&dialog);
|
|
dialog.exec();
|
|
}
|
|
|
|
QGroupBox* MappingWidget::CreateControlsBox(const QString& name, ControllerEmu::ControlGroup* group,
|
|
int columns)
|
|
{
|
|
auto* group_box = new QGroupBox(name);
|
|
auto* hbox_layout = new QHBoxLayout();
|
|
|
|
group_box->setLayout(hbox_layout);
|
|
|
|
std::vector<QFormLayout*> form_layouts;
|
|
for (int i = 0; i < columns; ++i)
|
|
{
|
|
form_layouts.push_back(new QFormLayout());
|
|
hbox_layout->addLayout(form_layouts[i]);
|
|
}
|
|
|
|
for (size_t i = 0; i < group->controls.size(); ++i)
|
|
{
|
|
CreateControl(group->controls[i].get(), form_layouts[i % columns], true);
|
|
}
|
|
|
|
return group_box;
|
|
}
|
|
|
|
void MappingWidget::CreateControl(const ControllerEmu::Control* control, QFormLayout* layout,
|
|
bool indicator)
|
|
{
|
|
auto* button = new MappingButton(this, control->control_ref.get(), indicator);
|
|
|
|
button->setMinimumWidth(100);
|
|
button->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed);
|
|
const bool translate = control->translate == ControllerEmu::Translatability::Translate;
|
|
const QString translated_name =
|
|
translate ? tr(control->ui_name.c_str()) : QString::fromStdString(control->ui_name);
|
|
layout->addRow(translated_name, button);
|
|
}
|
|
|
|
ControllerEmu::EmulatedController* MappingWidget::GetController() const
|
|
{
|
|
return m_parent->GetController();
|
|
}
|
|
|
|
QPushButton*
|
|
MappingWidget::CreateSettingAdvancedMappingButton(ControllerEmu::NumericSettingBase& setting)
|
|
{
|
|
const auto button = new QPushButton(tr("..."));
|
|
button->setFixedWidth(QFontMetrics(font()).boundingRect(button->text()).width() * 2);
|
|
|
|
button->connect(button, &QPushButton::clicked, [this, &setting]() {
|
|
if (setting.IsSimpleValue())
|
|
setting.SetExpressionFromValue();
|
|
|
|
// Ensure the UI has the game-controller indicator while editing the expression.
|
|
ConfigChanged();
|
|
|
|
IOWindow io(this, GetController(), &setting.GetInputReference(), IOWindow::Type::Input);
|
|
SetQWidgetWindowDecorations(&io);
|
|
io.exec();
|
|
|
|
setting.SimplifyIfPossible();
|
|
|
|
ConfigChanged();
|
|
SaveSettings();
|
|
});
|
|
|
|
return button;
|
|
}
|