mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-19 04:21:19 +01:00
e149ad4f0a
SPDX standardizes how source code conveys its copyright and licensing information. See https://spdx.github.io/spdx-spec/1-rationale/ . SPDX tags are adopted in many large projects, including things like the Linux kernel.
198 lines
5.8 KiB
C++
198 lines
5.8 KiB
C++
// Copyright 2021 Dolphin Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include "DolphinQt/Config/GamecubeControllersWidget.h"
|
|
|
|
#include <QComboBox>
|
|
#include <QGridLayout>
|
|
#include <QGroupBox>
|
|
#include <QLabel>
|
|
#include <QPushButton>
|
|
#include <QVBoxLayout>
|
|
|
|
#include <map>
|
|
#include <optional>
|
|
|
|
#include "Core/ConfigManager.h"
|
|
#include "Core/Core.h"
|
|
#include "Core/HW/SI/SI.h"
|
|
#include "Core/HW/SI/SI_Device.h"
|
|
|
|
#include "DolphinQt/Config/Mapping/GCPadWiiUConfigDialog.h"
|
|
#include "DolphinQt/Config/Mapping/MappingWindow.h"
|
|
#include "DolphinQt/Settings.h"
|
|
|
|
#include "InputCommon/GCAdapter.h"
|
|
|
|
static const std::map<SerialInterface::SIDevices, int> s_gc_types = {
|
|
{SerialInterface::SIDEVICE_NONE, 0}, {SerialInterface::SIDEVICE_GC_CONTROLLER, 1},
|
|
{SerialInterface::SIDEVICE_WIIU_ADAPTER, 2}, {SerialInterface::SIDEVICE_GC_STEERING, 3},
|
|
{SerialInterface::SIDEVICE_DANCEMAT, 4}, {SerialInterface::SIDEVICE_GC_TARUKONGA, 5},
|
|
{SerialInterface::SIDEVICE_GC_GBA, 6}, {SerialInterface::SIDEVICE_GC_KEYBOARD, 7}};
|
|
|
|
static std::optional<int> ToGCMenuIndex(const SerialInterface::SIDevices sidevice)
|
|
{
|
|
auto it = s_gc_types.find(sidevice);
|
|
return it != s_gc_types.end() ? it->second : std::optional<int>();
|
|
}
|
|
|
|
static std::optional<SerialInterface::SIDevices> FromGCMenuIndex(const int menudevice)
|
|
{
|
|
auto it = std::find_if(s_gc_types.begin(), s_gc_types.end(),
|
|
[=](auto pair) { return pair.second == menudevice; });
|
|
return it != s_gc_types.end() ? it->first : std::optional<SerialInterface::SIDevices>();
|
|
}
|
|
|
|
GamecubeControllersWidget::GamecubeControllersWidget(QWidget* parent) : QWidget(parent)
|
|
{
|
|
CreateLayout();
|
|
LoadSettings();
|
|
ConnectWidgets();
|
|
}
|
|
|
|
void GamecubeControllersWidget::CreateLayout()
|
|
{
|
|
m_gc_box = new QGroupBox(tr("GameCube Controllers"));
|
|
m_gc_layout = new QGridLayout();
|
|
m_gc_layout->setVerticalSpacing(7);
|
|
m_gc_layout->setColumnStretch(1, 1);
|
|
|
|
for (size_t i = 0; i < m_gc_groups.size(); i++)
|
|
{
|
|
auto* gc_label = new QLabel(tr("Port %1").arg(i + 1));
|
|
auto* gc_box = m_gc_controller_boxes[i] = new QComboBox();
|
|
auto* gc_button = m_gc_buttons[i] = new QPushButton(tr("Configure"));
|
|
|
|
for (const auto& item :
|
|
{tr("None"), tr("Standard Controller"), tr("GameCube Adapter for Wii U"),
|
|
tr("Steering Wheel"), tr("Dance Mat"), tr("DK Bongos"), tr("GBA"), tr("Keyboard")})
|
|
{
|
|
gc_box->addItem(item);
|
|
}
|
|
|
|
int controller_row = m_gc_layout->rowCount();
|
|
m_gc_layout->addWidget(gc_label, controller_row, 0);
|
|
m_gc_layout->addWidget(gc_box, controller_row, 1);
|
|
m_gc_layout->addWidget(gc_button, controller_row, 2);
|
|
}
|
|
m_gc_box->setLayout(m_gc_layout);
|
|
|
|
auto* layout = new QVBoxLayout;
|
|
layout->setMargin(0);
|
|
layout->setAlignment(Qt::AlignTop);
|
|
layout->addWidget(m_gc_box);
|
|
setLayout(layout);
|
|
}
|
|
|
|
void GamecubeControllersWidget::ConnectWidgets()
|
|
{
|
|
for (size_t i = 0; i < m_gc_controller_boxes.size(); i++)
|
|
{
|
|
connect(m_gc_controller_boxes[i], qOverload<int>(&QComboBox::currentIndexChanged), this,
|
|
&GamecubeControllersWidget::SaveSettings);
|
|
connect(m_gc_controller_boxes[i], qOverload<int>(&QComboBox::currentIndexChanged), this,
|
|
&GamecubeControllersWidget::OnGCTypeChanged);
|
|
connect(m_gc_buttons[i], &QPushButton::clicked, this,
|
|
&GamecubeControllersWidget::OnGCPadConfigure);
|
|
}
|
|
}
|
|
|
|
void GamecubeControllersWidget::OnGCTypeChanged(int type)
|
|
{
|
|
const auto* box = static_cast<QComboBox*>(QObject::sender());
|
|
|
|
for (size_t i = 0; i < m_gc_groups.size(); i++)
|
|
{
|
|
if (m_gc_controller_boxes[i] == box)
|
|
{
|
|
const int index = box->currentIndex();
|
|
m_gc_buttons[i]->setEnabled(index != 0 && index != 6);
|
|
return;
|
|
}
|
|
}
|
|
|
|
SaveSettings();
|
|
}
|
|
|
|
void GamecubeControllersWidget::OnGCPadConfigure()
|
|
{
|
|
size_t index;
|
|
for (index = 0; index < m_gc_groups.size(); index++)
|
|
{
|
|
if (m_gc_buttons[index] == QObject::sender())
|
|
break;
|
|
}
|
|
|
|
MappingWindow::Type type;
|
|
|
|
switch (m_gc_controller_boxes[index]->currentIndex())
|
|
{
|
|
case 0: // None
|
|
case 6: // GBA
|
|
return;
|
|
case 1: // Standard Controller
|
|
type = MappingWindow::Type::MAPPING_GCPAD;
|
|
break;
|
|
case 2: // GameCube Adapter for Wii U
|
|
GCPadWiiUConfigDialog(static_cast<int>(index), this).exec();
|
|
return;
|
|
case 3: // Steering Wheel
|
|
type = MappingWindow::Type::MAPPING_GC_STEERINGWHEEL;
|
|
break;
|
|
case 4: // Dance Mat
|
|
type = MappingWindow::Type::MAPPING_GC_DANCEMAT;
|
|
break;
|
|
case 5: // DK Bongos
|
|
type = MappingWindow::Type::MAPPING_GC_BONGOS;
|
|
break;
|
|
case 7: // Keyboard
|
|
type = MappingWindow::Type::MAPPING_GC_KEYBOARD;
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
MappingWindow* window = new MappingWindow(this, type, static_cast<int>(index));
|
|
window->setAttribute(Qt::WA_DeleteOnClose, true);
|
|
window->setWindowModality(Qt::WindowModality::WindowModal);
|
|
window->show();
|
|
}
|
|
|
|
void GamecubeControllersWidget::LoadSettings()
|
|
{
|
|
for (size_t i = 0; i < m_gc_groups.size(); i++)
|
|
{
|
|
const std::optional<int> gc_index = ToGCMenuIndex(SConfig::GetInstance().m_SIDevice[i]);
|
|
if (gc_index)
|
|
{
|
|
m_gc_controller_boxes[i]->setCurrentIndex(*gc_index);
|
|
m_gc_buttons[i]->setEnabled(*gc_index != 0 && *gc_index != 6);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GamecubeControllersWidget::SaveSettings()
|
|
{
|
|
for (size_t i = 0; i < m_gc_groups.size(); i++)
|
|
{
|
|
const int index = m_gc_controller_boxes[i]->currentIndex();
|
|
const std::optional<SerialInterface::SIDevices> si_device = FromGCMenuIndex(index);
|
|
if (si_device)
|
|
{
|
|
SConfig::GetInstance().m_SIDevice[i] = *si_device;
|
|
|
|
if (Core::IsRunning())
|
|
SerialInterface::ChangeDevice(*si_device, static_cast<s32>(i));
|
|
}
|
|
|
|
m_gc_buttons[i]->setEnabled(index != 0 && index != 6);
|
|
}
|
|
|
|
if (GCAdapter::UseAdapter())
|
|
GCAdapter::StartScanThread();
|
|
else
|
|
GCAdapter::StopScanThread();
|
|
|
|
SConfig::GetInstance().SaveSettings();
|
|
}
|