mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 09:29:43 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			576 lines
		
	
	
	
		
			21 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			576 lines
		
	
	
	
		
			21 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2021 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "DolphinQt/Config/WiimoteControllersWidget.h"
 | |
| 
 | |
| #include <QApplication>
 | |
| #include <QCheckBox>
 | |
| #include <QComboBox>
 | |
| #include <QGridLayout>
 | |
| #include <QGroupBox>
 | |
| #include <QLabel>
 | |
| #include <QPushButton>
 | |
| #include <QRadioButton>
 | |
| #include <QScreen>
 | |
| #include <QTimer>
 | |
| #include <QToolButton>
 | |
| #include <QVBoxLayout>
 | |
| #include <QVariant>
 | |
| 
 | |
| #include "Common/Config/Config.h"
 | |
| #include "Common/WorkQueueThread.h"
 | |
| 
 | |
| #include "Core/Config/MainSettings.h"
 | |
| #include "Core/Config/WiimoteSettings.h"
 | |
| #include "Core/ConfigManager.h"
 | |
| #include "Core/Core.h"
 | |
| #include "Core/HW/Wiimote.h"
 | |
| #include "Core/HW/WiimoteReal/WiimoteReal.h"
 | |
| #include "Core/IOS/IOS.h"
 | |
| #include "Core/IOS/USB/Bluetooth/LibUSBBluetoothAdapter.h"
 | |
| #include "Core/NetPlayProto.h"
 | |
| #include "Core/System.h"
 | |
| #include "Core/USBUtils.h"
 | |
| #include "Core/WiiUtils.h"
 | |
| 
 | |
| #include "DolphinQt/Config/Mapping/MappingWindow.h"
 | |
| #include "DolphinQt/QtUtils/ModalMessageBox.h"
 | |
| #include "DolphinQt/QtUtils/NonDefaultQPushButton.h"
 | |
| #include "DolphinQt/QtUtils/QueueOnObject.h"
 | |
| #include "DolphinQt/QtUtils/SignalBlocking.h"
 | |
| #include "DolphinQt/Settings.h"
 | |
| #include "DolphinQt/Settings/USBDevicePicker.h"
 | |
| 
 | |
| #if defined(_WIN32)
 | |
| #include "Core/HW//WiimoteReal/IOWin.h"
 | |
| #endif
 | |
| 
 | |
| WiimoteControllersWidget::WiimoteControllersWidget(QWidget* parent) : QWidget(parent)
 | |
| {
 | |
|   CreateLayout();
 | |
|   ConnectWidgets();
 | |
| 
 | |
|   connect(&Settings::Instance(), &Settings::ConfigChanged, this,
 | |
|           [this] { LoadSettings(Core::GetState(Core::System::GetInstance())); });
 | |
|   connect(&Settings::Instance(), &Settings::EmulationStateChanged, this,
 | |
|           [this](Core::State state) { LoadSettings(state); });
 | |
|   LoadSettings(Core::GetState(Core::System::GetInstance()));
 | |
| 
 | |
|   m_bluetooth_adapter_refresh_thread.Reset("Bluetooth Adapter Refresh Thread");
 | |
|   StartBluetoothAdapterRefresh();
 | |
| }
 | |
| 
 | |
| WiimoteControllersWidget::~WiimoteControllersWidget()
 | |
| {
 | |
|   m_bluetooth_adapter_refresh_thread.WaitForCompletion();
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::UpdateBluetoothAvailableStatus()
 | |
| {
 | |
|   m_bluetooth_unavailable->setHidden(WiimoteReal::IsScannerReady());
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::StartBluetoothAdapterRefresh()
 | |
| {
 | |
| #ifdef __LIBUSB__
 | |
|   if (m_bluetooth_adapter_scan_in_progress)
 | |
|     return;
 | |
| 
 | |
|   m_bluetooth_adapters->clear();
 | |
|   m_bluetooth_adapters->setDisabled(true);
 | |
|   m_bluetooth_adapters->addItem(tr("Scanning for adapters..."));
 | |
| 
 | |
|   m_bluetooth_adapter_scan_in_progress = true;
 | |
| 
 | |
|   const auto scan_func = [this]() {
 | |
|     INFO_LOG_FMT(COMMON, "Refreshing Bluetooth adapter list...");
 | |
|     auto device_list = USBUtils::ListDevices(LibUSBBluetoothAdapter::IsBluetoothDevice);
 | |
|     INFO_LOG_FMT(COMMON, "{} Bluetooth adapters available.", device_list.size());
 | |
|     const auto refresh_complete_func = [this, devices = std::move(device_list)]() {
 | |
|       OnBluetoothAdapterRefreshComplete(devices);
 | |
|     };
 | |
|     QueueOnObject(this, refresh_complete_func);
 | |
|   };
 | |
| 
 | |
|   m_bluetooth_adapter_refresh_thread.Push(scan_func);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::OnBluetoothAdapterRefreshComplete(
 | |
|     const std::vector<USBUtils::DeviceInfo>& devices)
 | |
| {
 | |
|   const int configured_vid = Config::Get(Config::MAIN_BLUETOOTH_PASSTHROUGH_VID);
 | |
|   const int configured_pid = Config::Get(Config::MAIN_BLUETOOTH_PASSTHROUGH_PID);
 | |
|   bool found_configured_device = configured_vid == -1 || configured_pid == -1;
 | |
| 
 | |
|   m_bluetooth_adapters->clear();
 | |
|   m_bluetooth_adapter_scan_in_progress = false;
 | |
| 
 | |
|   const auto state = Core::GetState(Core::System::GetInstance());
 | |
|   UpdateBluetoothAdapterWidgetsEnabled(state);
 | |
| 
 | |
|   m_bluetooth_adapters->addItem(tr("Automatic"));
 | |
| 
 | |
|   for (auto& device : devices)
 | |
|   {
 | |
|     m_bluetooth_adapters->addItem(QString::fromStdString(device.ToDisplayString()),
 | |
|                                   QVariant::fromValue(device));
 | |
| 
 | |
|     if (!found_configured_device &&
 | |
|         LibUSBBluetoothAdapter::IsConfiguredBluetoothDevice(device.vid, device.pid))
 | |
|     {
 | |
|       found_configured_device = true;
 | |
|       m_bluetooth_adapters->setCurrentIndex(m_bluetooth_adapters->count() - 1);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!found_configured_device)
 | |
|   {
 | |
|     const QString name = QLatin1Char{'['} + tr("disconnected") + QLatin1Char(']');
 | |
|     const std::string name_str = name.toStdString();
 | |
| 
 | |
|     USBUtils::DeviceInfo disconnected_device;
 | |
|     disconnected_device.vid = configured_vid;
 | |
|     disconnected_device.pid = configured_pid;
 | |
| 
 | |
|     const QString device_info =
 | |
|         QString::fromStdString(disconnected_device.ToDisplayString(name_str));
 | |
| 
 | |
|     m_bluetooth_adapters->insertSeparator(m_bluetooth_adapters->count());
 | |
|     m_bluetooth_adapters->addItem(device_info, QVariant::fromValue(disconnected_device));
 | |
|     m_bluetooth_adapters->setCurrentIndex(m_bluetooth_adapters->count() - 1);
 | |
|   }
 | |
| 
 | |
|   m_bluetooth_adapters->insertSeparator(m_bluetooth_adapters->count());
 | |
|   m_bluetooth_adapters->addItem(tr("More Options..."));
 | |
| }
 | |
| 
 | |
| static int GetRadioButtonIndicatorWidth()
 | |
| {
 | |
|   const QStyle* style = QApplication::style();
 | |
|   QStyleOptionButton opt;
 | |
| 
 | |
|   // TODO: why does the macOS style act different? Is it because of the magic with
 | |
|   // Cocoa widgets it does behind the scenes?
 | |
|   if (style->objectName() == QStringLiteral("macintosh"))
 | |
|     return style->subElementRect(QStyle::SE_RadioButtonIndicator, &opt).width();
 | |
| 
 | |
|   return style->subElementRect(QStyle::SE_RadioButtonContents, &opt).left();
 | |
| }
 | |
| 
 | |
| static int GetLayoutHorizontalSpacing(const QGridLayout* layout)
 | |
| {
 | |
|   // TODO: shouldn't layout->horizontalSpacing() do all this? Why does it return -1?
 | |
|   int hspacing = layout->horizontalSpacing();
 | |
|   if (hspacing >= 0)
 | |
|     return hspacing;
 | |
| 
 | |
|   // According to docs, this is the fallback if horizontalSpacing() isn't set.
 | |
|   auto style = layout->parentWidget()->style();
 | |
|   hspacing = style->pixelMetric(QStyle::PM_LayoutHorizontalSpacing);
 | |
|   if (hspacing >= 0)
 | |
|     return hspacing;
 | |
| 
 | |
|   // Docs claim this is deprecated, but on macOS with Qt 5.8 this is the only one that actually
 | |
|   // works.
 | |
|   float pixel_ratio = QGuiApplication::primaryScreen()->devicePixelRatio();
 | |
| #ifdef __APPLE__
 | |
|   // TODO is this still required?
 | |
|   hspacing = pixel_ratio * style->pixelMetric(QStyle::PM_LayoutHorizontalSpacing);
 | |
|   if (hspacing >= 0)
 | |
|     return hspacing;
 | |
| #endif
 | |
| 
 | |
|   // Ripped from qtbase/src/widgets/styles/qcommonstyle.cpp
 | |
|   return pixel_ratio * 6;
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::CreateLayout()
 | |
| {
 | |
|   m_wiimote_layout = new QGridLayout();
 | |
|   m_wiimote_box = new QGroupBox(tr("Wii Remotes"));
 | |
|   m_wiimote_box->setLayout(m_wiimote_layout);
 | |
| 
 | |
|   m_wiimote_passthrough = new QRadioButton(tr("Passthrough a Bluetooth adapter"));
 | |
|   m_bluetooth_adapters_label = new QLabel(tr("Adapter"));
 | |
|   m_bluetooth_adapters = new QComboBox();
 | |
|   m_bluetooth_adapters_refresh = new NonDefaultQPushButton(tr("Refresh"));
 | |
|   m_wiimote_sync = new NonDefaultQPushButton(tr("Sync"));
 | |
|   m_wiimote_reset = new NonDefaultQPushButton(tr("Reset"));
 | |
| 
 | |
|   m_wiimote_refresh_indicator = new QLabel{};
 | |
|   m_wiimote_refresh_indicator->hide();
 | |
|   m_wiimote_refresh = new QToolButton();
 | |
|   auto* const wiimote_refresh_action = new QAction(tr("Refresh"), m_wiimote_refresh);
 | |
|   m_wiimote_refresh->setDefaultAction(wiimote_refresh_action);
 | |
|   connect(wiimote_refresh_action, &QAction::triggered, this,
 | |
|           &WiimoteControllersWidget::OnWiimoteRefreshPressed);
 | |
|   m_wiimote_refresh->setPopupMode(QToolButton::ToolButtonPopupMode::MenuButtonPopup);
 | |
| 
 | |
| #if defined(_WIN32)
 | |
|   m_wiimote_refresh_indicator->setPixmap(
 | |
|       style()->standardIcon(QStyle::SP_BrowserReload).pixmap(16, 16));
 | |
| 
 | |
|   auto* const wiimote_sync_action = new QAction(tr("Sync"), m_wiimote_refresh);
 | |
|   m_wiimote_refresh->addAction(wiimote_sync_action);
 | |
|   connect(wiimote_sync_action, &QAction::triggered, this,
 | |
|           &WiimoteControllersWidget::TriggerHostWiimoteSync);
 | |
| 
 | |
|   auto* const wiimote_reset_action = new QAction(tr("Reset"), m_wiimote_refresh);
 | |
|   m_wiimote_refresh->addAction(wiimote_reset_action);
 | |
|   connect(wiimote_reset_action, &QAction::triggered, this,
 | |
|           &WiimoteControllersWidget::TriggerHostWiimoteReset);
 | |
| #endif
 | |
| 
 | |
|   m_wiimote_pt_labels[0] = new QLabel(tr("Sync real Wii Remotes and pair them"));
 | |
|   m_wiimote_pt_labels[1] = new QLabel(tr("Reset all saved Wii Remote pairings"));
 | |
|   m_wiimote_emu = new QRadioButton(tr("Emulate the Wii's Bluetooth adapter"));
 | |
|   m_wiimote_continuous_scanning = new QCheckBox(tr("Continuous Scanning"));
 | |
|   m_wiimote_real_balance_board = new QCheckBox(tr("Real Balance Board"));
 | |
|   m_wiimote_speaker_data = new QCheckBox(tr("Enable Speaker Data"));
 | |
|   m_wiimote_ciface = new QCheckBox(tr("Connect Wii Remotes for Emulated Controllers"));
 | |
| 
 | |
|   m_wiimote_layout->setVerticalSpacing(7);
 | |
|   m_wiimote_layout->setColumnMinimumWidth(0, GetRadioButtonIndicatorWidth() -
 | |
|                                                  GetLayoutHorizontalSpacing(m_wiimote_layout));
 | |
|   m_wiimote_layout->setColumnStretch(2, 1);
 | |
| 
 | |
|   // Passthrough BT
 | |
|   m_wiimote_layout->addWidget(m_wiimote_passthrough, m_wiimote_layout->rowCount(), 0, 1, -1);
 | |
| 
 | |
|   int adapter_row = m_wiimote_layout->rowCount();
 | |
|   m_wiimote_layout->addWidget(m_bluetooth_adapters_label, adapter_row, 1, 1, 1);
 | |
|   m_wiimote_layout->addWidget(m_bluetooth_adapters, adapter_row, 2, 1, 1);
 | |
|   m_wiimote_layout->addWidget(m_bluetooth_adapters_refresh, adapter_row, 3, 1, 1);
 | |
| 
 | |
|   int sync_row = m_wiimote_layout->rowCount();
 | |
|   m_wiimote_layout->addWidget(m_wiimote_pt_labels[0], sync_row, 1, 1, 2);
 | |
|   m_wiimote_layout->addWidget(m_wiimote_sync, sync_row, 3);
 | |
| 
 | |
|   int reset_row = m_wiimote_layout->rowCount();
 | |
|   m_wiimote_layout->addWidget(m_wiimote_pt_labels[1], reset_row, 1, 1, 2);
 | |
|   m_wiimote_layout->addWidget(m_wiimote_reset, reset_row, 3);
 | |
| 
 | |
|   // Emulated BT
 | |
|   m_wiimote_layout->addWidget(m_wiimote_emu, m_wiimote_layout->rowCount(), 0, 1, -1);
 | |
| 
 | |
|   for (size_t i = 0; i < m_wiimote_groups.size(); i++)
 | |
|   {
 | |
|     auto* wm_label = m_wiimote_labels[i] = new QLabel(tr("Wii Remote %1").arg(i + 1));
 | |
|     auto* wm_box = m_wiimote_boxes[i] = new QComboBox();
 | |
|     auto* wm_button = m_wiimote_buttons[i] = new NonDefaultQPushButton(tr("Configure"));
 | |
| 
 | |
|     for (const auto& item : {tr("None"), tr("Emulated Wii Remote"), tr("Real Wii Remote")})
 | |
|       wm_box->addItem(item);
 | |
| 
 | |
|     int wm_row = m_wiimote_layout->rowCount();
 | |
|     m_wiimote_layout->addWidget(wm_label, wm_row, 1);
 | |
|     m_wiimote_layout->addWidget(wm_box, wm_row, 2);
 | |
|     m_wiimote_layout->addWidget(wm_button, wm_row, 3);
 | |
|   }
 | |
| 
 | |
|   m_wiimote_layout->addWidget(m_wiimote_real_balance_board, m_wiimote_layout->rowCount(), 1, 1, -1);
 | |
|   m_wiimote_layout->addWidget(m_wiimote_speaker_data, m_wiimote_layout->rowCount(), 1, 1, -1);
 | |
| 
 | |
|   m_wiimote_layout->addWidget(m_wiimote_ciface, m_wiimote_layout->rowCount(), 0, 1, -1);
 | |
| 
 | |
|   int continuous_scanning_row = m_wiimote_layout->rowCount();
 | |
| 
 | |
|   auto* const left_of_refresh_button_layout = new QHBoxLayout;
 | |
|   left_of_refresh_button_layout->addWidget(m_wiimote_continuous_scanning);
 | |
|   left_of_refresh_button_layout->addStretch(1);
 | |
|   left_of_refresh_button_layout->addWidget(m_wiimote_refresh_indicator);
 | |
| 
 | |
|   m_wiimote_layout->addLayout(left_of_refresh_button_layout, continuous_scanning_row, 0, 1, 3);
 | |
|   m_wiimote_layout->addWidget(m_wiimote_refresh, continuous_scanning_row, 3);
 | |
| 
 | |
|   m_bluetooth_unavailable = new QLabel(tr("A supported Bluetooth device could not be found.\n"
 | |
|                                           "You must manually connect your Wii Remote."));
 | |
|   m_wiimote_layout->addWidget(m_bluetooth_unavailable, m_wiimote_layout->rowCount(), 1, 1, -1);
 | |
| 
 | |
|   auto* layout = new QVBoxLayout;
 | |
|   layout->setContentsMargins(0, 0, 0, 0);
 | |
|   layout->setAlignment(Qt::AlignTop);
 | |
|   layout->addWidget(m_wiimote_box);
 | |
|   setLayout(layout);
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::ConnectWidgets()
 | |
| {
 | |
|   connect(m_wiimote_passthrough, &QRadioButton::toggled, this, [this] {
 | |
|     SaveSettings();
 | |
|     LoadSettings(Core::GetState(Core::System::GetInstance()));
 | |
|   });
 | |
|   connect(m_wiimote_ciface, &QCheckBox::toggled, this, [this] {
 | |
|     SaveSettings();
 | |
|     LoadSettings(Core::GetState(Core::System::GetInstance()));
 | |
|     WiimoteReal::HandleWiimotesInControllerInterfaceSettingChange();
 | |
|   });
 | |
|   connect(m_wiimote_continuous_scanning, &QCheckBox::toggled, this, [this] {
 | |
|     SaveSettings();
 | |
|     LoadSettings(Core::GetState(Core::System::GetInstance()));
 | |
|   });
 | |
| 
 | |
|   connect(m_wiimote_real_balance_board, &QCheckBox::toggled, this,
 | |
|           &WiimoteControllersWidget::SaveSettings);
 | |
|   connect(m_wiimote_speaker_data, &QCheckBox::toggled, this,
 | |
|           &WiimoteControllersWidget::SaveSettings);
 | |
|   connect(m_bluetooth_adapters, &QComboBox::activated, this,
 | |
|           &WiimoteControllersWidget::OnBluetoothPassthroughDeviceChanged);
 | |
|   connect(m_bluetooth_adapters_refresh, &QPushButton::clicked, this,
 | |
|           &WiimoteControllersWidget::StartBluetoothAdapterRefresh);
 | |
|   connect(m_wiimote_sync, &QPushButton::clicked, this,
 | |
|           &WiimoteControllersWidget::OnBluetoothPassthroughSyncPressed);
 | |
|   connect(m_wiimote_reset, &QPushButton::clicked, this,
 | |
|           &WiimoteControllersWidget::OnBluetoothPassthroughResetPressed);
 | |
| 
 | |
|   for (size_t i = 0; i < m_wiimote_groups.size(); i++)
 | |
|   {
 | |
|     connect(m_wiimote_boxes[i], &QComboBox::currentIndexChanged, this, [this] {
 | |
|       SaveSettings();
 | |
|       LoadSettings(Core::GetState(Core::System::GetInstance()));
 | |
|     });
 | |
|     connect(m_wiimote_buttons[i], &QPushButton::clicked, this,
 | |
|             [this, i] { OnWiimoteConfigure(i); });
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::OnBluetoothPassthroughDeviceChanged(int index)
 | |
| {
 | |
|   std::optional<USBUtils::DeviceInfo> device_info;
 | |
|   bool needs_refresh = false;
 | |
|   // "Automatic" selection
 | |
|   if (index == 0)
 | |
|   {
 | |
|     Config::DeleteKey(Config::GetActiveLayerForConfig(Config::MAIN_BLUETOOTH_PASSTHROUGH_PID),
 | |
|                       Config::MAIN_BLUETOOTH_PASSTHROUGH_PID);
 | |
|     Config::DeleteKey(Config::GetActiveLayerForConfig(Config::MAIN_BLUETOOTH_PASSTHROUGH_VID),
 | |
|                       Config::MAIN_BLUETOOTH_PASSTHROUGH_VID);
 | |
|     return;
 | |
|   }
 | |
|   // "More Options..." selection
 | |
|   else if (index == m_bluetooth_adapters->count() - 1)
 | |
|   {
 | |
|     device_info = USBDevicePicker::Run(this, tr("Select A Bluetooth Device"));
 | |
|     needs_refresh = true;
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     const QVariant item_data = m_bluetooth_adapters->itemData(index);
 | |
| 
 | |
|     if (!item_data.isValid() || !item_data.canConvert<USBUtils::DeviceInfo>())
 | |
|     {
 | |
|       ERROR_LOG_FMT(COMMON, "Invalid Bluetooth device info selected in WiimoteControllersWidget");
 | |
|       return;
 | |
|     }
 | |
|     device_info = item_data.value<USBUtils::DeviceInfo>();
 | |
|   }
 | |
| 
 | |
|   if (device_info.has_value())
 | |
|   {
 | |
|     Config::SetBaseOrCurrent(Config::MAIN_BLUETOOTH_PASSTHROUGH_PID, device_info->pid);
 | |
|     Config::SetBaseOrCurrent(Config::MAIN_BLUETOOTH_PASSTHROUGH_VID, device_info->vid);
 | |
|   }
 | |
|   if (needs_refresh)
 | |
|     StartBluetoothAdapterRefresh();
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::OnBluetoothPassthroughResetPressed()
 | |
| {
 | |
|   const auto ios = Core::System::GetInstance().GetIOS();
 | |
| 
 | |
|   if (!ios)
 | |
|   {
 | |
|     ModalMessageBox::warning(
 | |
|         this, tr("Warning"),
 | |
|         tr("Saved Wii Remote pairings can only be reset when a Wii game is running."));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   auto device = WiiUtils::GetBluetoothRealDevice();
 | |
|   if (device != nullptr)
 | |
|     device->TriggerSyncButtonHeldEvent();
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::OnBluetoothPassthroughSyncPressed()
 | |
| {
 | |
|   const auto ios = Core::System::GetInstance().GetIOS();
 | |
| 
 | |
|   if (!ios)
 | |
|   {
 | |
|     ModalMessageBox::warning(this, tr("Warning"),
 | |
|                              tr("A sync can only be triggered when a Wii game is running."));
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   auto device = WiiUtils::GetBluetoothRealDevice();
 | |
|   if (device != nullptr)
 | |
|     device->TriggerSyncButtonPressedEvent();
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::OnWiimoteRefreshPressed()
 | |
| {
 | |
|   WiimoteReal::Refresh();
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::OnWiimoteConfigure(size_t index)
 | |
| {
 | |
|   MappingWindow::Type type;
 | |
|   switch (m_wiimote_boxes[index]->currentIndex())
 | |
|   {
 | |
|   case 0:  // None
 | |
|   case 2:  // Real Wii Remote
 | |
|     return;
 | |
|   case 1:  // Emulated Wii Remote
 | |
|     type = MappingWindow::Type::MAPPING_WIIMOTE_EMU;
 | |
|     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 WiimoteControllersWidget::UpdateBluetoothAdapterWidgetsEnabled(const Core::State state)
 | |
| {
 | |
|   const bool running = state != Core::State::Uninitialized;
 | |
|   const bool running_wii = running && Core::System::GetInstance().IsWii();
 | |
|   const bool enable_adapter_refresh = m_wiimote_passthrough->isChecked() && !running_wii;
 | |
|   const bool enable_adapter_selection =
 | |
|       enable_adapter_refresh && !m_bluetooth_adapter_scan_in_progress;
 | |
| 
 | |
|   m_bluetooth_adapters_label->setEnabled(enable_adapter_selection);
 | |
|   m_bluetooth_adapters->setEnabled(enable_adapter_selection);
 | |
|   m_bluetooth_adapters_refresh->setEnabled(enable_adapter_refresh);
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::LoadSettings(Core::State state)
 | |
| {
 | |
|   for (size_t i = 0; i < m_wiimote_groups.size(); i++)
 | |
|   {
 | |
|     SignalBlocking(m_wiimote_boxes[i])
 | |
|         ->setCurrentIndex(int(Config::Get(Config::GetInfoForWiimoteSource(int(i)))));
 | |
|   }
 | |
|   SignalBlocking(m_wiimote_real_balance_board)
 | |
|       ->setChecked(Config::Get(Config::WIIMOTE_BB_SOURCE) == WiimoteSource::Real);
 | |
|   SignalBlocking(m_wiimote_speaker_data)
 | |
|       ->setChecked(Config::Get(Config::MAIN_WIIMOTE_ENABLE_SPEAKER));
 | |
|   SignalBlocking(m_wiimote_ciface)
 | |
|       ->setChecked(Config::Get(Config::MAIN_CONNECT_WIIMOTES_FOR_CONTROLLER_INTERFACE));
 | |
|   SignalBlocking(m_wiimote_continuous_scanning)
 | |
|       ->setChecked(Config::Get(Config::MAIN_WIIMOTE_CONTINUOUS_SCANNING));
 | |
| 
 | |
|   if (Config::Get(Config::MAIN_BLUETOOTH_PASSTHROUGH_ENABLED))
 | |
|     SignalBlocking(m_wiimote_passthrough)->setChecked(true);
 | |
|   else
 | |
|     SignalBlocking(m_wiimote_emu)->setChecked(true);
 | |
| 
 | |
|   // Make sure continuous scanning setting is applied.
 | |
|   WiimoteReal::Initialize(::Wiimote::InitializeMode::DO_NOT_WAIT_FOR_WIIMOTES);
 | |
| 
 | |
|   const bool running = state != Core::State::Uninitialized;
 | |
| 
 | |
|   m_wiimote_emu->setEnabled(!running);
 | |
|   m_wiimote_passthrough->setEnabled(!running);
 | |
| 
 | |
|   const bool running_gc = running && !Core::System::GetInstance().IsWii();
 | |
|   const bool enable_passthrough = m_wiimote_passthrough->isChecked() && !running_gc;
 | |
|   const bool enable_emu_bt = !m_wiimote_passthrough->isChecked() && !running_gc;
 | |
|   const bool is_netplay = NetPlay::IsNetPlayRunning();
 | |
|   const bool running_netplay = running && is_netplay;
 | |
| 
 | |
|   UpdateBluetoothAdapterWidgetsEnabled(state);
 | |
| 
 | |
|   m_wiimote_sync->setEnabled(enable_passthrough);
 | |
|   m_wiimote_reset->setEnabled(enable_passthrough);
 | |
| 
 | |
|   for (auto* pt_label : m_wiimote_pt_labels)
 | |
|     pt_label->setEnabled(enable_passthrough);
 | |
| 
 | |
|   const int num_local_wiimotes = is_netplay ? NetPlay::NumLocalWiimotes() : 4;
 | |
|   for (size_t i = 0; i < m_wiimote_groups.size(); i++)
 | |
|   {
 | |
|     m_wiimote_labels[i]->setEnabled(enable_emu_bt);
 | |
|     m_wiimote_boxes[i]->setEnabled(enable_emu_bt && !running_netplay);
 | |
| 
 | |
|     const bool is_emu_wiimote = m_wiimote_boxes[i]->currentIndex() == 1;
 | |
|     m_wiimote_buttons[i]->setEnabled(enable_emu_bt && is_emu_wiimote &&
 | |
|                                      static_cast<int>(i) < num_local_wiimotes);
 | |
|   }
 | |
| 
 | |
|   m_wiimote_real_balance_board->setEnabled(enable_emu_bt && !running_netplay);
 | |
|   m_wiimote_speaker_data->setEnabled(enable_emu_bt && !running_netplay);
 | |
| 
 | |
|   const bool ciface_wiimotes = m_wiimote_ciface->isChecked();
 | |
| 
 | |
|   m_wiimote_refresh->setEnabled((enable_emu_bt || ciface_wiimotes) &&
 | |
|                                 !m_wiimote_continuous_scanning->isChecked());
 | |
|   m_wiimote_continuous_scanning->setEnabled(enable_emu_bt || ciface_wiimotes);
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::SaveSettings()
 | |
| {
 | |
|   {
 | |
|     Config::ConfigChangeCallbackGuard config_guard;
 | |
|     Config::SetBaseOrCurrent(Config::MAIN_WIIMOTE_ENABLE_SPEAKER,
 | |
|                              m_wiimote_speaker_data->isChecked());
 | |
|     Config::SetBaseOrCurrent(Config::MAIN_CONNECT_WIIMOTES_FOR_CONTROLLER_INTERFACE,
 | |
|                              m_wiimote_ciface->isChecked());
 | |
|     Config::SetBaseOrCurrent(Config::MAIN_WIIMOTE_CONTINUOUS_SCANNING,
 | |
|                              m_wiimote_continuous_scanning->isChecked());
 | |
|     Config::SetBaseOrCurrent(Config::MAIN_BLUETOOTH_PASSTHROUGH_ENABLED,
 | |
|                              m_wiimote_passthrough->isChecked());
 | |
| 
 | |
|     const WiimoteSource bb_source =
 | |
|         m_wiimote_real_balance_board->isChecked() ? WiimoteSource::Real : WiimoteSource::None;
 | |
|     Config::SetBaseOrCurrent(Config::WIIMOTE_BB_SOURCE, bb_source);
 | |
| 
 | |
|     for (size_t i = 0; i < m_wiimote_groups.size(); i++)
 | |
|     {
 | |
|       const int index = m_wiimote_boxes[i]->currentIndex();
 | |
|       Config::SetBaseOrCurrent(Config::GetInfoForWiimoteSource(int(i)), WiimoteSource(index));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   SConfig::GetInstance().SaveSettings();
 | |
| }
 | |
| 
 | |
| #if defined(_WIN32)
 | |
| void WiimoteControllersWidget::AsyncRefreshActionHelper(std::invocable<> auto func)
 | |
| {
 | |
|   m_wiimote_refresh->setEnabled(false);
 | |
|   m_wiimote_refresh_indicator->show();
 | |
| 
 | |
|   auto result = std::async(std::launch::async, std::move(func));
 | |
| 
 | |
|   auto* const animation = new QTimer{this};
 | |
|   connect(animation, &QTimer::timeout, this, [this, animation, result = std::move(result)] {
 | |
|     // Spin the refresh indicator.
 | |
|     m_wiimote_refresh_indicator->setPixmap(
 | |
|         m_wiimote_refresh_indicator->pixmap().transformed(QTransform().rotate(90)));
 | |
| 
 | |
|     if (result.wait_for(std::chrono::seconds{}) != std::future_status::ready)
 | |
|       return;
 | |
| 
 | |
|     // When the async task is done, re-enable the button and hide the indicator.
 | |
|     animation->deleteLater();
 | |
|     m_wiimote_refresh_indicator->hide();
 | |
|     m_wiimote_refresh->setEnabled(true);
 | |
|   });
 | |
| 
 | |
|   animation->start(250);
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::TriggerHostWiimoteSync()
 | |
| {
 | |
|   AsyncRefreshActionHelper(WiimoteReal::WiimoteScannerWindows::FindAndAuthenticateWiimotes);
 | |
| }
 | |
| 
 | |
| void WiimoteControllersWidget::TriggerHostWiimoteReset()
 | |
| {
 | |
|   AsyncRefreshActionHelper(WiimoteReal::WiimoteScannerWindows::RemoveRememberedWiimotes);
 | |
| }
 | |
| #endif
 |