// Copyright 2021 Dolphin Emulator Project // SPDX-License-Identifier: GPL-2.0-or-later #include "DolphinQt/Config/WiimoteControllersWidget.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #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& 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 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()) { ERROR_LOG_FMT(COMMON, "Invalid Bluetooth device info selected in WiimoteControllersWidget"); return; } device_info = item_data.value(); } 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(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(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