// Copyright 2010 Dolphin Emulator Project // Licensed under GPLv2+ // Refer to the license.txt file included. #include "DolphinWX/Input/InputConfigDiag.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "Common/FileSearch.h" #include "Common/FileUtil.h" #include "Common/IniFile.h" #include "Common/MsgHandler.h" #include "Common/StringUtil.h" #include "Core/Core.h" #include "Core/HW/GCKeyboard.h" #include "Core/HW/GCPad.h" #include "Core/HW/Wiimote.h" #include "Core/HW/WiimoteEmu/WiimoteEmu.h" #include "Core/HotkeyManager.h" #include "DolphinWX/DolphinSlider.h" #include "DolphinWX/Input/ClassicInputConfigDiag.h" #include "DolphinWX/Input/DrumsInputConfigDiag.h" #include "DolphinWX/Input/GuitarInputConfigDiag.h" #include "DolphinWX/Input/NunchukInputConfigDiag.h" #include "DolphinWX/Input/TurntableInputConfigDiag.h" #include "DolphinWX/UINeedsControllerState.h" #include "DolphinWX/WxUtils.h" #include "InputCommon/ControlReference/ControlReference.h" #include "InputCommon/ControlReference/ExpressionParser.h" #include "InputCommon/ControllerEmu/Control/Control.h" #include "InputCommon/ControllerEmu/ControlGroup/ControlGroup.h" #include "InputCommon/ControllerEmu/ControlGroup/Extension.h" #include "InputCommon/ControllerEmu/ControllerEmu.h" #include "InputCommon/ControllerEmu/Setting/BooleanSetting.h" #include "InputCommon/ControllerEmu/Setting/NumericSetting.h" #include "InputCommon/ControllerInterface/ControllerInterface.h" #include "InputCommon/ControllerInterface/Device.h" #include "InputCommon/InputConfig.h" using ciface::ExpressionParser::ParseStatus; void InputConfigDialog::ConfigExtension(wxCommandEvent& event) { ControllerEmu::Extension* const ex = ((ExtensionButton*)event.GetEventObject())->extension; int extension_type = ex->switch_extension; // show config diag, if "none" isn't selected switch (extension_type) { case WiimoteEmu::EXT_NUNCHUK: { NunchukInputConfigDialog dlg(this, m_config, _("Nunchuk Configuration"), device_cbox, m_port_num); dlg.ShowModal(); } break; case WiimoteEmu::EXT_CLASSIC: { ClassicInputConfigDialog dlg(this, m_config, _("Classic Controller Configuration"), device_cbox, m_port_num); dlg.ShowModal(); } break; case WiimoteEmu::EXT_GUITAR: { GuitarInputConfigDialog dlg(this, m_config, _("Guitar Configuration"), device_cbox, m_port_num); dlg.ShowModal(); } break; case WiimoteEmu::EXT_DRUMS: { DrumsInputConfigDialog dlg(this, m_config, _("Drums Configuration"), device_cbox, m_port_num); dlg.ShowModal(); } break; case WiimoteEmu::EXT_TURNTABLE: { TurntableInputConfigDialog dlg(this, m_config, _("Turntable Configuration"), device_cbox, m_port_num); dlg.ShowModal(); } break; default: break; } } PadSetting::PadSetting(wxControl* const _control) : wxcontrol(_control) { wxcontrol->SetClientData(this); } PadSettingExtension::PadSettingExtension(wxWindow* const parent, ControllerEmu::Extension* const ext) : PadSetting(new wxChoice(parent, wxID_ANY)), extension(ext) { for (auto& attachment : extension->attachments) { ((wxChoice*)wxcontrol)->Append(wxGetTranslation(StrToWxStr(attachment->GetName()))); } UpdateGUI(); } void PadSettingExtension::UpdateGUI() { ((wxChoice*)wxcontrol)->Select(extension->switch_extension); } void PadSettingExtension::UpdateValue() { extension->switch_extension = ((wxChoice*)wxcontrol)->GetSelection(); } PadSettingCheckBox::PadSettingCheckBox(wxWindow* const parent, ControllerEmu::BooleanSetting* const _setting) : PadSetting( new wxCheckBox(parent, wxID_ANY, wxGetTranslation(StrToWxStr(_setting->m_ui_name)))), setting(_setting) { UpdateGUI(); } void PadSettingCheckBox::UpdateGUI() { ((wxCheckBox*)wxcontrol)->SetValue(setting->GetValue()); // Force WX to trigger an event after updating the value wxCommandEvent event(wxEVT_CHECKBOX); event.SetEventObject(wxcontrol); wxPostEvent(wxcontrol, event); } void PadSettingCheckBox::UpdateValue() { setting->SetValue(((wxCheckBox*)wxcontrol)->GetValue()); } PadSettingSpin::PadSettingSpin(wxWindow* const parent, ControllerEmu::NumericSetting* const settings) : PadSetting(new wxSpinCtrl(parent, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxSP_ARROW_KEYS, settings->m_low, settings->m_high, (int)(settings->m_value * 100))), setting(settings) { // Compute how wide the control needs to be to fit the maximum value in it. // This accounts for borders, margins and the spinner buttons. wxcontrol->SetMinSize(WxUtils::GetTextWidgetMinSize(static_cast(wxcontrol))); wxcontrol->SetLabelText(setting->m_name); } void PadSettingSpin::UpdateGUI() { ((wxSpinCtrl*)wxcontrol)->SetValue((int)(setting->GetValue() * 100)); } void PadSettingSpin::UpdateValue() { setting->SetValue(ControlState(((wxSpinCtrl*)wxcontrol)->GetValue()) / 100); } ControlDialog::ControlDialog(InputConfigDialog* const parent, InputConfig& config, ControlReference* const ref) : wxDialog(parent, wxID_ANY, _("Configure Control"), wxDefaultPosition, wxDefaultSize, wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER), control_reference(ref), m_config(config), m_parent(parent) { m_devq = m_parent->GetController()->GetDefaultDevice(); const int space5 = FromDIP(5); // GetStrings() sounds slow :/ device_cbox = new wxComboBox(this, wxID_ANY, StrToWxStr(m_devq.ToString()), wxDefaultPosition, wxDLG_UNIT(this, wxSize(180, -1)), parent->device_cbox->GetStrings(), wxTE_PROCESS_ENTER); device_cbox->Bind(wxEVT_COMBOBOX, &ControlDialog::SetDevice, this); device_cbox->Bind(wxEVT_TEXT_ENTER, &ControlDialog::SetDevice, this); wxStaticBoxSizer* const control_chooser = CreateControlChooser(parent); wxStaticBoxSizer* const d_szr = new wxStaticBoxSizer(wxVERTICAL, this, _("Device")); d_szr->AddSpacer(space5); d_szr->Add(device_cbox, 0, wxEXPAND | wxLEFT | wxRIGHT, space5); d_szr->AddSpacer(space5); wxBoxSizer* const szr = new wxBoxSizer(wxVERTICAL); szr->AddSpacer(space5); szr->Add(d_szr, 0, wxEXPAND | wxLEFT | wxRIGHT, space5); szr->AddSpacer(space5); szr->Add(control_chooser, 1, wxEXPAND | wxLEFT | wxRIGHT, space5); szr->AddSpacer(space5); SetLayoutAdaptationMode(wxDIALOG_ADAPTATION_MODE_ENABLED); SetLayoutAdaptationLevel(wxDIALOG_ADAPTATION_STANDARD_SIZER); SetSizerAndFit(szr); // needed UpdateGUI(); SetFocus(); } ExtensionButton::ExtensionButton(wxWindow* const parent, ControllerEmu::Extension* const ext) : wxButton(parent, wxID_ANY, _("Configure"), wxDefaultPosition), extension(ext) { } ControlButton::ControlButton(wxWindow* const parent, ControlReference* const _ref, const std::string& name, const unsigned int width, const std::string& label) : wxButton(parent, wxID_ANY), control_reference(_ref), m_name(name), m_configured_width(FromDIP(width)) { if (label.empty()) SetLabelText(StrToWxStr(_ref->GetExpression())); else SetLabel(StrToWxStr(label)); } wxSize ControlButton::DoGetBestSize() const { if (m_configured_width == wxDefaultCoord) return wxButton::DoGetBestSize(); static constexpr int PADDING_HEIGHT = 4; wxClientDC dc(const_cast(this)); wxFontMetrics metrics = dc.GetFontMetrics(); return {m_configured_width, metrics.height + FromDIP(PADDING_HEIGHT * 2)}; } void InputConfigDialog::UpdateProfileComboBox() { std::string pname(File::GetUserPath(D_CONFIG_IDX)); pname += PROFILES_PATH; pname += m_config.GetProfileName(); std::vector sv = Common::DoFileSearch({pname}, {".ini"}); wxArrayString strs; for (const std::string& filename : sv) { std::string base; SplitPath(filename, nullptr, &base, nullptr); strs.push_back(StrToWxStr(base)); } profile_cbox->Clear(); profile_cbox->Append(strs); } void InputConfigDialog::UpdateControlReferences() { controller->UpdateReferences(g_controller_interface); } void InputConfigDialog::OnClose(wxCloseEvent& event) { m_config.SaveConfig(); EndModal(wxID_OK); } void InputConfigDialog::OnCloseButton(wxCommandEvent& event) { Close(); } int ControlDialog::GetRangeSliderValue() const { return m_range_slider->GetValue(); } void ControlDialog::UpdateListContents() { control_lbox->Clear(); const auto dev = g_controller_interface.FindDevice(m_devq); if (dev != nullptr) { if (control_reference->IsInput()) { for (ciface::Core::Device::Input* input : dev->Inputs()) { control_lbox->Append(StrToWxStr(input->GetName())); } } else // It's an output { for (ciface::Core::Device::Output* output : dev->Outputs()) { control_lbox->Append(StrToWxStr(output->GetName())); } } } } void ControlDialog::SelectControl(const std::string& name) { // UpdateGUI(); const int f = control_lbox->FindString(StrToWxStr(name)); if (f >= 0) control_lbox->Select(f); } void ControlDialog::UpdateGUI() { // update textbox textctrl->SetValue(StrToWxStr(control_reference->GetExpression())); // updates the "bound controls:" label m_bound_label->SetLabel( wxString::Format(_("Bound Controls: %lu"), (unsigned long)control_reference->BoundCount())); switch (control_reference->GetParseStatus()) { case ParseStatus::SyntaxError: m_error_label->SetLabel(_("Syntax error")); break; case ParseStatus::Successful: m_error_label->SetLabel(control_reference->BoundCount() > 0 ? wxString{} : _("Device not found")); break; case ParseStatus::EmptyExpression: m_error_label->SetLabel(""); break; } } void InputConfigDialog::UpdateGUI() { if (device_cbox != nullptr) device_cbox->SetValue(StrToWxStr(controller->GetDefaultDevice().ToString())); for (ControlGroupBox* cgBox : control_groups) { for (ControlButton* button : cgBox->control_buttons) { button->SetLabelText(StrToWxStr(button->control_reference->GetExpression())); } for (PadSetting* padSetting : cgBox->options) { padSetting->UpdateGUI(); } } } void InputConfigDialog::ClearAll(wxCommandEvent&) { // just load an empty ini section to clear everything :P IniFile::Section section; controller->LoadConfig(§ion); // no point in using the real ControllerInterface i guess ControllerInterface face; controller->UpdateReferences(face); UpdateGUI(); } void InputConfigDialog::LoadDefaults(wxCommandEvent&) { controller->LoadDefaults(g_controller_interface); controller->UpdateReferences(g_controller_interface); UpdateGUI(); } bool ControlDialog::Validate() { control_reference->SetExpression(WxStrToStr(textctrl->GetValue())); const auto lock = ControllerEmu::EmulatedController::GetStateLock(); control_reference->UpdateReference(g_controller_interface, m_parent->GetController()->GetDefaultDevice()); UpdateGUI(); const auto parse_status = control_reference->GetParseStatus(); return parse_status == ParseStatus::Successful || parse_status == ParseStatus::EmptyExpression; } void InputConfigDialog::SetDevice(wxCommandEvent&) { controller->SetDefaultDevice(WxStrToStr(device_cbox->GetValue())); // show user what it was validated as device_cbox->SetValue(StrToWxStr(controller->GetDefaultDevice().ToString())); // update references controller->UpdateReferences(g_controller_interface); } void ControlDialog::SetDevice(wxCommandEvent&) { m_devq.FromString(WxStrToStr(device_cbox->GetValue())); // show user what it was validated as device_cbox->SetValue(StrToWxStr(m_devq.ToString())); // update gui UpdateListContents(); } void ControlDialog::ClearControl(wxCommandEvent&) { control_reference->SetExpression(""); const auto lock = ControllerEmu::EmulatedController::GetStateLock(); control_reference->UpdateReference(g_controller_interface, m_parent->GetController()->GetDefaultDevice()); UpdateGUI(); } inline bool IsAlphabetic(wxString& str) { for (wxUniChar c : str) if (!isalpha(c)) return false; return true; } inline void GetExpressionForControl(wxString& expr, wxString& control_name, const ciface::Core::DeviceQualifier* control_device = nullptr, const ciface::Core::DeviceQualifier* default_device = nullptr) { expr = ""; // non-default device if (control_device && default_device && !(*control_device == *default_device)) { expr += control_device->ToString(); expr += ":"; } // append the control name expr += control_name; if (!IsAlphabetic(expr)) expr = wxString::Format("`%s`", expr); } bool ControlDialog::GetExpressionForSelectedControl(wxString& expr) { const int num = control_lbox->GetSelection(); if (num < 0) return false; wxString control_name = control_lbox->GetString(num); GetExpressionForControl(expr, control_name, &m_devq, &m_parent->GetController()->GetDefaultDevice()); return true; } void ControlDialog::SetSelectedControl(wxCommandEvent&) { wxString expr; if (!GetExpressionForSelectedControl(expr)) return; textctrl->WriteText(expr); control_reference->SetExpression(WxStrToStr(textctrl->GetValue())); const auto lock = ControllerEmu::EmulatedController::GetStateLock(); control_reference->UpdateReference(g_controller_interface, m_parent->GetController()->GetDefaultDevice()); UpdateGUI(); } void ControlDialog::AppendControl(wxCommandEvent& event) { wxString device_expr, expr; const wxString lbl = ((wxButton*)event.GetEventObject())->GetLabel(); char op = lbl[0]; if (!GetExpressionForSelectedControl(device_expr)) return; // Unary ops (that is, '!') are a special case. When there's a selection, // put parens around it and prepend it with a '!', but when there's nothing, // just add a '!device'. if (op == '!') { wxString selection = textctrl->GetStringSelection(); if (selection == "") expr = wxString::Format("%c%s", op, device_expr); else expr = wxString::Format("%c(%s)", op, selection); } else { expr = wxString::Format(" %c %s", op, device_expr); } textctrl->WriteText(expr); control_reference->SetExpression(WxStrToStr(textctrl->GetValue())); const auto lock = ControllerEmu::EmulatedController::GetStateLock(); control_reference->UpdateReference(g_controller_interface, m_parent->GetController()->GetDefaultDevice()); UpdateGUI(); } void InputConfigDialog::EnablePadSetting(const std::string& group_name, const std::string& name, const bool enabled) { const auto box_iterator = std::find_if(control_groups.begin(), control_groups.end(), [&group_name](const auto& box) { return group_name == box->control_group->name; }); if (box_iterator == control_groups.end()) return; const auto* box = *box_iterator; const auto it = std::find_if(box->options.begin(), box->options.end(), [&name](const auto& pad_setting) { return pad_setting->wxcontrol->GetLabelText() == name; }); if (it == box->options.end()) return; (*it)->wxcontrol->Enable(enabled); } void InputConfigDialog::EnableControlButton(const std::string& group_name, const std::string& name, const bool enabled) { const auto box_iterator = std::find_if(control_groups.begin(), control_groups.end(), [&group_name](const auto& box) { return group_name == box->control_group->name; }); if (box_iterator == control_groups.end()) return; const auto* box = *box_iterator; const auto it = std::find_if(box->control_buttons.begin(), box->control_buttons.end(), [&name](const auto& control_button) { return control_button->m_name == name; }); if (it == box->control_buttons.end()) return; (*it)->Enable(enabled); } void ControlDialog::OnRangeSlide(wxScrollEvent& event) { m_range_spinner->SetValue(event.GetPosition()); control_reference->range = static_cast(event.GetPosition()) / SLIDER_TICK_COUNT; } void ControlDialog::OnRangeSpin(wxSpinEvent& event) { m_range_slider->SetValue(event.GetValue()); control_reference->range = static_cast(event.GetValue()) / SLIDER_TICK_COUNT; } void ControlDialog::OnRangeThumbtrack(wxScrollEvent& event) { m_range_spinner->SetValue(event.GetPosition()); } void InputConfigDialog::AdjustSetting(wxCommandEvent& event) { const auto* const control = static_cast(event.GetEventObject()); auto* const pad_setting = static_cast(control->GetClientData()); pad_setting->UpdateValue(); } void InputConfigDialog::AdjustBooleanSetting(wxCommandEvent& event) { const auto* const control = static_cast(event.GetEventObject()); auto* const pad_setting = static_cast(control->GetClientData()); pad_setting->UpdateValue(); // TODO: find a cleaner way to have actions depending on the setting if (control->GetLabelText() == "Iterative Input") { m_iterate = pad_setting->setting->GetValue(); } else if (control->GetLabelText() == "Relative Input") { EnablePadSetting("IR", "Dead Zone", pad_setting->setting->GetValue()); EnableControlButton("IR", "Recenter", pad_setting->setting->GetValue()); } } void InputConfigDialog::ConfigControl(wxEvent& event) { m_control_dialog = new ControlDialog(this, m_config, ((ControlButton*)event.GetEventObject())->control_reference); m_control_dialog->ShowModal(); m_control_dialog->Destroy(); // update changes that were made in the dialog UpdateGUI(); } void InputConfigDialog::ClearControl(wxEvent& event) { ControlButton* const btn = (ControlButton*)event.GetEventObject(); btn->control_reference->SetExpression(""); btn->control_reference->range = 1.0; controller->UpdateReferences(g_controller_interface); // update changes UpdateGUI(); } void ControlDialog::DetectControl(wxCommandEvent& event) { wxButton* const btn = (wxButton*)event.GetEventObject(); const wxString lbl = btn->GetLabel(); const auto dev = g_controller_interface.FindDevice(m_devq); if (dev != nullptr) { m_event_filter.BlockEvents(true); btn->SetLabel(_("[ waiting ]")); // This makes the "waiting" text work on Linux. true (only if needed) prevents crash on Windows wxTheApp->Yield(true); ciface::Core::Device::Control* const ctrl = control_reference->Detect(DETECT_WAIT_TIME, dev.get()); // if we got input, select it in the list if (ctrl) SelectControl(ctrl->GetName()); btn->SetLabel(lbl); // This lets the input events be sent to the filter and discarded before unblocking wxTheApp->Yield(true); m_event_filter.BlockEvents(false); } } void InputConfigDialog::DetectControl(wxCommandEvent& event) { auto* btn = static_cast(event.GetEventObject()); if (DetectButton(btn) && m_iterate) { auto it = std::find(control_buttons.begin(), control_buttons.end(), btn); // it can and will be control_buttons.end() for any control that is in the exclude list. if (it == control_buttons.end()) return; ++it; for (; it != control_buttons.end(); ++it) { if (!DetectButton(*it)) break; } } } bool InputConfigDialog::DetectButton(ControlButton* button) { bool success = false; // find device :/ const auto dev = g_controller_interface.FindDevice(controller->GetDefaultDevice()); if (dev != nullptr) { m_event_filter.BlockEvents(true); button->SetLabel(_("[ waiting ]")); // This makes the "waiting" text work on Linux. true (only if needed) prevents crash on Windows wxTheApp->Yield(true); ciface::Core::Device::Control* const ctrl = button->control_reference->Detect(DETECT_WAIT_TIME, dev.get()); // if we got input, update expression and reference if (ctrl) { wxString control_name = ctrl->GetName(); wxString expr; GetExpressionForControl(expr, control_name); button->control_reference->SetExpression(WxStrToStr(expr)); const auto lock = ControllerEmu::EmulatedController::GetStateLock(); button->control_reference->UpdateReference(g_controller_interface, controller->GetDefaultDevice()); success = true; } // This lets the input events be sent to the filter and discarded before unblocking wxTheApp->Yield(true); m_event_filter.BlockEvents(false); } UpdateGUI(); return success; } wxStaticBoxSizer* ControlDialog::CreateControlChooser(InputConfigDialog* const parent) { wxStaticBoxSizer* const main_szr = new wxStaticBoxSizer( wxVERTICAL, this, control_reference->IsInput() ? _("Input") : _("Output")); const int space5 = FromDIP(5); textctrl = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDLG_UNIT(this, wxSize(-1, 32)), wxTE_MULTILINE | wxTE_RICH2); wxFont font = textctrl->GetFont(); font.SetFamily(wxFONTFAMILY_MODERN); textctrl->SetFont(font); wxButton* const detect_button = new wxButton(this, wxID_ANY, control_reference->IsInput() ? _("Detect") : _("Test")); wxButton* const clear_button = new wxButton(this, wxID_ANY, _("Clear")); wxButton* const select_button = new wxButton(this, wxID_ANY, _("Select")); select_button->Bind(wxEVT_BUTTON, &ControlDialog::SetSelectedControl, this); wxButton* const or_button = new wxButton(this, wxID_ANY, _("| OR")); or_button->Bind(wxEVT_BUTTON, &ControlDialog::AppendControl, this); control_lbox = new wxListBox(this, wxID_ANY, wxDefaultPosition, wxDLG_UNIT(this, wxSize(-1, 48))); wxBoxSizer* const button_sizer = new wxBoxSizer(wxVERTICAL); button_sizer->Add(detect_button, 1); button_sizer->Add(select_button, 1); button_sizer->Add(or_button, 1); if (control_reference->IsInput()) { // TODO: check if && is good on other OS wxButton* const and_button = new wxButton(this, wxID_ANY, _("&& AND")); wxButton* const not_button = new wxButton(this, wxID_ANY, _("! NOT")); wxButton* const add_button = new wxButton(this, wxID_ANY, _("+ ADD")); and_button->Bind(wxEVT_BUTTON, &ControlDialog::AppendControl, this); not_button->Bind(wxEVT_BUTTON, &ControlDialog::AppendControl, this); add_button->Bind(wxEVT_BUTTON, &ControlDialog::AppendControl, this); button_sizer->Add(and_button, 1); button_sizer->Add(not_button, 1); button_sizer->Add(add_button, 1); } m_range_slider = new DolphinSlider( this, wxID_ANY, static_cast(control_reference->range * SLIDER_TICK_COUNT), -SLIDER_TICK_COUNT * 5, SLIDER_TICK_COUNT * 5, wxDefaultPosition, wxDefaultSize, wxSL_TOP); m_range_spinner = new wxSpinCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDLG_UNIT(this, wxSize(32, -1)), wxSP_ARROW_KEYS | wxALIGN_RIGHT, m_range_slider->GetMin(), m_range_slider->GetMax(), m_range_slider->GetValue()); detect_button->Bind(wxEVT_BUTTON, &ControlDialog::DetectControl, this); clear_button->Bind(wxEVT_BUTTON, &ControlDialog::ClearControl, this); m_range_slider->Bind(wxEVT_SCROLL_CHANGED, &ControlDialog::OnRangeSlide, this); m_range_slider->Bind(wxEVT_SCROLL_THUMBTRACK, &ControlDialog::OnRangeThumbtrack, this); m_range_spinner->Bind(wxEVT_SPINCTRL, &ControlDialog::OnRangeSpin, this); m_bound_label = new wxStaticText(this, wxID_ANY, ""); m_error_label = new wxStaticText(this, wxID_ANY, ""); wxBoxSizer* const range_sizer = new wxBoxSizer(wxHORIZONTAL); range_sizer->Add(new wxStaticText(this, wxID_ANY, _("Range")), 0, wxALIGN_CENTER_VERTICAL); range_sizer->Add( new wxStaticText(this, wxID_ANY, wxString::Format("%d", m_range_slider->GetMin())), 0, wxALIGN_CENTER_VERTICAL | wxLEFT, space5); range_sizer->Add(m_range_slider, 1, wxEXPAND); range_sizer->Add( new wxStaticText(this, wxID_ANY, wxString::Format("%d", m_range_slider->GetMax())), 0, wxALIGN_CENTER_VERTICAL); range_sizer->Add(m_range_spinner, 0, wxALIGN_CENTER_VERTICAL | wxLEFT, space5); wxBoxSizer* const ctrls_sizer = new wxBoxSizer(wxHORIZONTAL); ctrls_sizer->Add(control_lbox, 1, wxEXPAND); ctrls_sizer->Add(button_sizer, 0, wxEXPAND); wxSizer* const bottom_btns_sizer = CreateButtonSizer(wxOK | wxAPPLY); bottom_btns_sizer->Prepend(clear_button, 0, wxLEFT, space5); main_szr->Add(range_sizer, 0, wxEXPAND | wxLEFT | wxRIGHT, space5); main_szr->AddSpacer(space5); main_szr->Add(ctrls_sizer, 0, wxEXPAND | wxLEFT | wxRIGHT, space5); main_szr->AddSpacer(space5); main_szr->Add(textctrl, 1, wxEXPAND | wxLEFT | wxRIGHT, space5); main_szr->AddSpacer(space5); main_szr->Add(bottom_btns_sizer, 0, wxEXPAND | wxRIGHT, space5); main_szr->AddSpacer(space5); main_szr->Add(m_bound_label, 0, wxALIGN_CENTER_HORIZONTAL); main_szr->Add(m_error_label, 0, wxALIGN_CENTER_HORIZONTAL); UpdateListContents(); return main_szr; } void InputConfigDialog::GetProfilePath(std::string& path) { const wxString& name = profile_cbox->GetValue(); if (!name.empty()) { // TODO: check for dumb characters maybe path = File::GetUserPath(D_CONFIG_IDX); path += PROFILES_PATH; path += m_config.GetProfileName(); path += '/'; path += WxStrToStr(profile_cbox->GetValue()); path += ".ini"; } } ControllerEmu::EmulatedController* InputConfigDialog::GetController() const { return controller; } void InputConfigDialog::LoadProfile(wxCommandEvent&) { std::string fname; InputConfigDialog::GetProfilePath(fname); IniFile inifile; if (!inifile.Load(fname)) return; controller->LoadConfig(inifile.GetOrCreateSection("Profile")); controller->UpdateReferences(g_controller_interface); UpdateGUI(); } void InputConfigDialog::SaveProfile(wxCommandEvent&) { std::string fname; InputConfigDialog::GetProfilePath(fname); File::CreateFullPath(fname); if (!fname.empty()) { IniFile inifile; controller->SaveConfig(inifile.GetOrCreateSection("Profile")); inifile.Save(fname); UpdateProfileComboBox(); } else { WxUtils::ShowErrorDialog(_("You must enter a valid profile name.")); } } void InputConfigDialog::DeleteProfile(wxCommandEvent&) { std::string fname; InputConfigDialog::GetProfilePath(fname); const char* const fnamecstr = fname.c_str(); if (File::Exists(fnamecstr) && AskYesNoT("Are you sure you want to delete \"%s\"?", WxStrToStr(profile_cbox->GetValue()).c_str())) { File::Delete(fnamecstr); UpdateProfileComboBox(); } } void InputConfigDialog::UpdateDeviceComboBox() { device_cbox->Clear(); for (const std::string& device_string : g_controller_interface.GetAllDeviceStrings()) device_cbox->Append(StrToWxStr(device_string)); device_cbox->SetValue(StrToWxStr(controller->GetDefaultDevice().ToString())); } void InputConfigDialog::RefreshDevices(wxCommandEvent&) { Core::RunAsCPUThread([&] { // refresh devices g_controller_interface.RefreshDevices(); // update all control references UpdateControlReferences(); // update device cbox UpdateDeviceComboBox(); Wiimote::LoadConfig(); Keyboard::LoadConfig(); Pad::LoadConfig(); HotkeyManagerEmu::LoadConfig(); UpdateGUI(); }); } ControlGroupBox::~ControlGroupBox() { for (PadSetting* padSetting : options) delete padSetting; } bool ControlGroupBox::HasBitmapHeading() const { return control_group->type == ControllerEmu::GroupType::Stick || control_group->type == ControllerEmu::GroupType::Tilt || control_group->type == ControllerEmu::GroupType::Cursor || control_group->type == ControllerEmu::GroupType::Force; } ControlGroupBox::ControlGroupBox(ControllerEmu::ControlGroup* const group, wxWindow* const parent, InputConfigDialog* eventsink) : wxStaticBoxSizer(wxVERTICAL, parent, wxGetTranslation(StrToWxStr(group->ui_name))), control_group(group), static_bitmap(nullptr), m_scale(1) { static constexpr std::array exclude_buttons{{"Mic", "Modifier"}}; static constexpr std::array exclude_groups{ {"IR", "Swing", "Tilt", "Shake", "UDP Wiimote", "Extension", "Rumble"}}; wxFont small_font(7, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL, wxFONTWEIGHT_NORMAL); const int space3 = parent->FromDIP(3); wxFlexGridSizer* control_grid = new wxFlexGridSizer(2, 0, space3); control_grid->AddGrowableCol(0); for (const auto& control : group->controls) { wxStaticText* const label = new wxStaticText(parent, wxID_ANY, wxGetTranslation(StrToWxStr(control->ui_name))); ControlButton* const control_button = new ControlButton(parent, control->control_ref.get(), control->ui_name, 80); control_button->SetFont(small_font); control_buttons.push_back(control_button); if (std::find(exclude_groups.begin(), exclude_groups.end(), control_group->name) == exclude_groups.end() && std::find(exclude_buttons.begin(), exclude_buttons.end(), control->ui_name) == exclude_buttons.end()) eventsink->control_buttons.push_back(control_button); if (control->control_ref->IsInput()) { control_button->SetToolTip( _("Left-click to detect input.\nMiddle-click to clear.\nRight-click for more options.")); control_button->Bind(wxEVT_BUTTON, &InputConfigDialog::DetectControl, eventsink); } else { control_button->SetToolTip(_("Left/Right-click for more options.\nMiddle-click to clear.")); control_button->Bind(wxEVT_BUTTON, &InputConfigDialog::ConfigControl, eventsink); } control_button->Bind(wxEVT_MIDDLE_DOWN, &InputConfigDialog::ClearControl, eventsink); control_button->Bind(wxEVT_RIGHT_UP, &InputConfigDialog::ConfigControl, eventsink); control_grid->Add(label, 0, wxALIGN_CENTER_VERTICAL | wxALIGN_RIGHT); control_grid->Add(control_button, 0, wxALIGN_CENTER_VERTICAL); } Add(control_grid, 0, wxEXPAND | wxLEFT | wxRIGHT, space3); switch (group->type) { case ControllerEmu::GroupType::Stick: case ControllerEmu::GroupType::Tilt: case ControllerEmu::GroupType::Cursor: case ControllerEmu::GroupType::Force: { wxSize bitmap_size = parent->FromDIP(wxSize(64, 64)); m_scale = bitmap_size.GetWidth() / 64.0; wxBitmap bitmap; bitmap.CreateScaled(bitmap_size.GetWidth(), bitmap_size.GetHeight(), wxBITMAP_SCREEN_DEPTH, parent->GetContentScaleFactor()); wxMemoryDC dc(bitmap); dc.Clear(); dc.SelectObject(wxNullBitmap); static_bitmap = new wxStaticBitmap(parent, wxID_ANY, bitmap, wxDefaultPosition, wxDefaultSize, wxBITMAP_TYPE_BMP); wxBoxSizer* const szr = new wxBoxSizer(wxVERTICAL); for (auto& groupSetting : group->numeric_settings) { PadSettingSpin* setting = new PadSettingSpin(parent, groupSetting.get()); setting->wxcontrol->Bind(wxEVT_SPINCTRL, &InputConfigDialog::AdjustSetting, eventsink); options.push_back(setting); szr->Add( new wxStaticText(parent, wxID_ANY, wxGetTranslation(StrToWxStr(groupSetting->m_name)))); szr->Add(setting->wxcontrol); } for (auto& groupSetting : group->boolean_settings) { auto* checkbox = new PadSettingCheckBox(parent, groupSetting.get()); checkbox->wxcontrol->Bind(wxEVT_CHECKBOX, &InputConfigDialog::AdjustBooleanSetting, eventsink); options.push_back(checkbox); Add(checkbox->wxcontrol, 0, wxALL | wxLEFT, space3); } wxBoxSizer* const h_szr = new wxBoxSizer(wxHORIZONTAL); h_szr->Add(szr, 1, wxEXPAND | wxTOP | wxBOTTOM, space3); h_szr->AddSpacer(space3); h_szr->Add(static_bitmap, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM, space3); AddSpacer(space3); Add(h_szr, 0, wxEXPAND | wxLEFT | wxRIGHT, space3); } break; case ControllerEmu::GroupType::Buttons: { // Draw buttons in rows of 8 unsigned int button_cols = group->controls.size() > 8 ? 8 : group->controls.size(); unsigned int button_rows = ceil((float)group->controls.size() / 8.0f); wxSize bitmap_size(12 * button_cols + 1, 11 * button_rows + 1); wxSize bitmap_scaled_size = parent->FromDIP(bitmap_size); m_scale = static_cast(bitmap_scaled_size.GetWidth()) / bitmap_size.GetWidth(); wxBitmap bitmap; bitmap.CreateScaled(bitmap_scaled_size.GetWidth(), bitmap_scaled_size.GetHeight(), wxBITMAP_SCREEN_DEPTH, parent->GetContentScaleFactor()); wxMemoryDC dc(bitmap); dc.Clear(); dc.SelectObject(wxNullBitmap); static_bitmap = new wxStaticBitmap(parent, wxID_ANY, bitmap, wxDefaultPosition, wxDefaultSize, wxBITMAP_TYPE_BMP); auto* const threshold_cbox = new PadSettingSpin(parent, group->numeric_settings[0].get()); threshold_cbox->wxcontrol->Bind(wxEVT_SPINCTRL, &InputConfigDialog::AdjustSetting, eventsink); threshold_cbox->wxcontrol->SetToolTip( _("Adjust the analog control pressure required to activate buttons.")); options.push_back(threshold_cbox); wxBoxSizer* const szr = new wxBoxSizer(wxHORIZONTAL); szr->Add(new wxStaticText(parent, wxID_ANY, wxGetTranslation(StrToWxStr(group->numeric_settings[0]->m_name))), 0, wxALIGN_CENTER_VERTICAL | wxRIGHT, space3); szr->Add(threshold_cbox->wxcontrol, 0, wxRIGHT, space3); AddSpacer(space3); Add(szr, 0, wxALIGN_CENTER_HORIZONTAL | wxLEFT | wxRIGHT, space3); AddSpacer(space3); Add(static_bitmap, 0, wxALIGN_CENTER_HORIZONTAL | wxLEFT | wxRIGHT, space3); } break; case ControllerEmu::GroupType::MixedTriggers: case ControllerEmu::GroupType::Triggers: case ControllerEmu::GroupType::Slider: { int height = (int)(12 * group->controls.size()); int width = 64; if (group->type == ControllerEmu::GroupType::MixedTriggers) width = 64 + 12 + 1; if (group->type != ControllerEmu::GroupType::Triggers) height /= 2; height += 1; wxSize bitmap_size = parent->FromDIP(wxSize(width, height)); m_scale = static_cast(bitmap_size.GetWidth()) / width; wxBitmap bitmap; bitmap.CreateScaled(bitmap_size.GetWidth(), bitmap_size.GetHeight(), wxBITMAP_SCREEN_DEPTH, parent->GetContentScaleFactor()); wxMemoryDC dc(bitmap); dc.Clear(); dc.SelectObject(wxNullBitmap); static_bitmap = new wxStaticBitmap(parent, wxID_ANY, bitmap, wxDefaultPosition, wxDefaultSize, wxBITMAP_TYPE_BMP); for (auto& groupSetting : group->numeric_settings) { PadSettingSpin* setting = new PadSettingSpin(parent, groupSetting.get()); setting->wxcontrol->Bind(wxEVT_SPINCTRL, &InputConfigDialog::AdjustSetting, eventsink); options.push_back(setting); wxBoxSizer* const szr = new wxBoxSizer(wxHORIZONTAL); szr->Add( new wxStaticText(parent, wxID_ANY, wxGetTranslation(StrToWxStr(groupSetting->m_name))), 0, wxALIGN_CENTER_VERTICAL); szr->Add(setting->wxcontrol, 0, wxLEFT, space3); AddSpacer(space3); Add(szr, 0, wxALIGN_CENTER_HORIZONTAL | wxLEFT | wxRIGHT, space3); } AddSpacer(space3); Add(static_bitmap, 0, wxALIGN_CENTER_HORIZONTAL | wxLEFT | wxRIGHT, space3); } break; case ControllerEmu::GroupType::Extension: { PadSettingExtension* const attachments = new PadSettingExtension(parent, (ControllerEmu::Extension*)group); wxButton* const configure_btn = new ExtensionButton(parent, (ControllerEmu::Extension*)group); options.push_back(attachments); attachments->wxcontrol->Bind(wxEVT_CHOICE, &InputConfigDialog::AdjustSetting, eventsink); configure_btn->Bind(wxEVT_BUTTON, &InputConfigDialog::ConfigExtension, eventsink); AddSpacer(space3); Add(attachments->wxcontrol, 0, wxEXPAND | wxLEFT | wxRIGHT, space3); AddSpacer(space3); Add(configure_btn, 0, wxEXPAND | wxLEFT | wxRIGHT, space3); } break; default: { // options for (auto& groupSetting : group->boolean_settings) { PadSettingCheckBox* setting_cbox = new PadSettingCheckBox(parent, groupSetting.get()); setting_cbox->wxcontrol->Bind(wxEVT_CHECKBOX, &InputConfigDialog::AdjustBooleanSetting, eventsink); if (groupSetting->m_name == "Iterative Input") groupSetting->SetValue(false); options.push_back(setting_cbox); AddSpacer(space3); Add(setting_cbox->wxcontrol, 0, wxLEFT | wxRIGHT, space3); } for (auto& groupSetting : group->numeric_settings) { PadSettingSpin* setting = new PadSettingSpin(parent, groupSetting.get()); setting->wxcontrol->Bind(wxEVT_SPINCTRL, &InputConfigDialog::AdjustSetting, eventsink); options.push_back(setting); wxBoxSizer* const szr = new wxBoxSizer(wxHORIZONTAL); szr->Add( new wxStaticText(parent, wxID_ANY, wxGetTranslation(StrToWxStr(groupSetting->m_name))), 0, wxALIGN_CENTER_VERTICAL, space3); szr->Add(setting->wxcontrol, 0, wxLEFT, space3); AddSpacer(space3); Add(szr, 0, wxLEFT | wxRIGHT | wxALIGN_RIGHT, space3); } break; } } AddSpacer(space3); eventsink->control_groups.push_back(this); } wxBoxSizer* InputConfigDialog::CreateDeviceChooserGroupBox() { const int space3 = FromDIP(3); wxStaticBoxSizer* const device_sbox = new wxStaticBoxSizer(wxVERTICAL, this, _("Device")); device_cbox = new wxComboBox(device_sbox->GetStaticBox(), wxID_ANY, ""); device_cbox->ToggleWindowStyle(wxTE_PROCESS_ENTER); wxButton* refresh_button = new wxButton(device_sbox->GetStaticBox(), wxID_ANY, _("Refresh"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT); device_cbox->Bind(wxEVT_COMBOBOX, &InputConfigDialog::SetDevice, this); device_cbox->Bind(wxEVT_TEXT_ENTER, &InputConfigDialog::SetDevice, this); refresh_button->Bind(wxEVT_BUTTON, &InputConfigDialog::RefreshDevices, this); wxBoxSizer* const device_sbox_in = new wxBoxSizer(wxHORIZONTAL); device_sbox_in->Add(WxUtils::GiveMinSizeDIP(device_cbox, wxSize(64, -1)), 1, wxALIGN_CENTER_VERTICAL); device_sbox_in->Add(refresh_button, 0, wxALIGN_CENTER_VERTICAL | wxLEFT, space3); device_sbox->Add(device_sbox_in, 1, wxEXPAND | wxLEFT | wxRIGHT, space3); device_sbox->AddSpacer(space3); return device_sbox; } wxBoxSizer* InputConfigDialog::CreaterResetGroupBox(wxOrientation orientation) { const int space3 = FromDIP(3); wxStaticBoxSizer* const clear_sbox = new wxStaticBoxSizer(orientation, this, _("Reset")); wxButton* const default_button = new wxButton(clear_sbox->GetStaticBox(), wxID_ANY, _("Default"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT); wxButton* const clearall_button = new wxButton(clear_sbox->GetStaticBox(), wxID_ANY, _("Clear"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT); wxBoxSizer* clear_sbox_in = new wxBoxSizer(orientation); clear_sbox_in->Add(default_button, 1, wxEXPAND); clear_sbox_in->Add(clearall_button, 1, wxEXPAND); clear_sbox->Add(clear_sbox_in, 1, wxEXPAND | wxBOTTOM | wxLEFT | wxRIGHT, space3); clearall_button->Bind(wxEVT_BUTTON, &InputConfigDialog::ClearAll, this); default_button->Bind(wxEVT_BUTTON, &InputConfigDialog::LoadDefaults, this); return clear_sbox; } wxBoxSizer* InputConfigDialog::CreateProfileChooserGroupBox() { const int space3 = FromDIP(3); wxStaticBoxSizer* profile_sbox = new wxStaticBoxSizer(wxVERTICAL, this, _("Profile")); profile_cbox = new wxComboBox(profile_sbox->GetStaticBox(), wxID_ANY, ""); wxButton* const pload_btn = new wxButton(profile_sbox->GetStaticBox(), wxID_ANY, _("Load"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT); wxButton* const psave_btn = new wxButton(profile_sbox->GetStaticBox(), wxID_ANY, _("Save"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT); wxButton* const pdelete_btn = new wxButton(profile_sbox->GetStaticBox(), wxID_ANY, _("Delete"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT); pload_btn->Bind(wxEVT_BUTTON, &InputConfigDialog::LoadProfile, this); psave_btn->Bind(wxEVT_BUTTON, &InputConfigDialog::SaveProfile, this); pdelete_btn->Bind(wxEVT_BUTTON, &InputConfigDialog::DeleteProfile, this); wxBoxSizer* profile_sbox_in = new wxBoxSizer(wxHORIZONTAL); profile_sbox_in->Add(WxUtils::GiveMinSizeDIP(profile_cbox, wxSize(64, -1)), 1, wxALIGN_CENTER_VERTICAL); profile_sbox_in->AddSpacer(space3); profile_sbox_in->Add(pload_btn, 0, wxALIGN_CENTER_VERTICAL); profile_sbox_in->Add(psave_btn, 0, wxALIGN_CENTER_VERTICAL); profile_sbox_in->Add(pdelete_btn, 0, wxALIGN_CENTER_VERTICAL); profile_sbox->Add(profile_sbox_in, 1, wxEXPAND | wxLEFT | wxRIGHT, space3); profile_sbox->AddSpacer(space3); return profile_sbox; } InputConfigDialog::InputConfigDialog(wxWindow* const parent, InputConfig& config, const wxString& name, const int port_num) : wxDialog(parent, wxID_ANY, name), controller(config.GetController(port_num)), m_config(config), m_port_num(port_num) { Bind(wxEVT_CLOSE_WINDOW, &InputConfigDialog::OnClose, this); Bind(wxEVT_BUTTON, &InputConfigDialog::OnCloseButton, this, wxID_CLOSE); Bind(wxEVT_ACTIVATE, &InputConfigDialog::OnActivate, this); SetLayoutAdaptationMode(wxDIALOG_ADAPTATION_MODE_ENABLED); SetLayoutAdaptationLevel(wxDIALOG_ADAPTATION_STANDARD_SIZER); // live preview update timer m_update_timer.SetOwner(this); Bind(wxEVT_TIMER, &InputConfigDialog::UpdateBitmaps, this); m_update_timer.Start(PREVIEW_UPDATE_TIME, wxTIMER_CONTINUOUS); } void InputConfigDialog::OnActivate(wxActivateEvent& event) { // Needed for input bitmaps SetUINeedsControllerState(event.GetActive()); } InputEventFilter::InputEventFilter() { wxEvtHandler::AddFilter(this); } InputEventFilter::~InputEventFilter() { wxEvtHandler::RemoveFilter(this); } int InputEventFilter::FilterEvent(wxEvent& event) { if (m_block && ShouldCatchEventType(event.GetEventType())) { event.StopPropagation(); return Event_Processed; } return Event_Skip; } void InputEventFilter::BlockEvents(bool block) { m_block = block; } bool InputEventFilter::ShouldCatchEventType(wxEventType type) { return type == wxEVT_KEY_DOWN || type == wxEVT_KEY_UP || type == wxEVT_CHAR || type == wxEVT_CHAR_HOOK || type == wxEVT_LEFT_DOWN || type == wxEVT_LEFT_UP || type == wxEVT_MIDDLE_DOWN || type == wxEVT_MIDDLE_UP || type == wxEVT_RIGHT_DOWN || type == wxEVT_RIGHT_UP; }