mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-25 23:41:19 +01:00
f106a9637d
Core::PauseAndLock requires all calls to it to be balanced, like this: const bool was_unpaused = Core::PauseAndLock(true); // do stuff on the CPU thread Core::PauseAndLock(false, was_unpaused); Aside from being a bit cumbersome, it turns out all callers really don't need to know about was_unpaused at all. They just need to do something on the CPU thread safely, including locking/unlocking. So this commit replaces Core::PauseAndLock with a function that makes both the purpose and the scope of what is being run on the CPU thread visually clear. This makes it harder to accidentally run something on the wrong thread, or forget the second call to PauseAndLock to unpause, or forget that it needs to be passed was_unpaused at the end. We also don't need comments to indicate code X is being run on the CPU thread anymore, as the function name makes it obvious.
1311 lines
43 KiB
C++
1311 lines
43 KiB
C++
// Copyright 2010 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include "DolphinWX/Input/InputConfigDiag.h"
|
|
|
|
#include <algorithm>
|
|
#include <array>
|
|
#include <cctype>
|
|
#include <cstddef>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <string>
|
|
#include <vector>
|
|
#include <wx/app.h>
|
|
#include <wx/bitmap.h>
|
|
#include <wx/button.h>
|
|
#include <wx/checkbox.h>
|
|
#include <wx/choice.h>
|
|
#include <wx/combobox.h>
|
|
#include <wx/control.h>
|
|
#include <wx/dcclient.h>
|
|
#include <wx/dcmemory.h>
|
|
#include <wx/dialog.h>
|
|
#include <wx/event.h>
|
|
#include <wx/font.h>
|
|
#include <wx/listbox.h>
|
|
#include <wx/msgdlg.h>
|
|
#include <wx/notebook.h>
|
|
#include <wx/panel.h>
|
|
#include <wx/sizer.h>
|
|
#include <wx/spinctrl.h>
|
|
#include <wx/statbmp.h>
|
|
#include <wx/statbox.h>
|
|
#include <wx/stattext.h>
|
|
#include <wx/textctrl.h>
|
|
#include <wx/timer.h>
|
|
|
|
#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 namespace ciface::ExpressionParser;
|
|
|
|
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<wxSpinCtrl*>(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()->default_device;
|
|
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->expression));
|
|
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<ControlButton*>(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<std::string> 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->expression));
|
|
|
|
// 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::NoDevice:
|
|
m_error_label->SetLabel(_("Device not found"));
|
|
break;
|
|
default:
|
|
m_error_label->SetLabel("");
|
|
}
|
|
};
|
|
|
|
void InputConfigDialog::UpdateGUI()
|
|
{
|
|
if (device_cbox != nullptr)
|
|
device_cbox->SetValue(StrToWxStr(controller->default_device.ToString()));
|
|
|
|
for (ControlGroupBox* cgBox : control_groups)
|
|
{
|
|
for (ControlButton* button : cgBox->control_buttons)
|
|
{
|
|
button->SetLabelText(StrToWxStr(button->control_reference->expression));
|
|
}
|
|
|
|
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->expression = WxStrToStr(textctrl->GetValue());
|
|
|
|
const auto lock = ControllerEmu::EmulatedController::GetStateLock();
|
|
control_reference->UpdateReference(g_controller_interface,
|
|
m_parent->GetController()->default_device);
|
|
|
|
UpdateGUI();
|
|
|
|
const auto parse_status = control_reference->GetParseStatus();
|
|
return parse_status == ParseStatus::Successful || parse_status == ParseStatus::NoDevice;
|
|
}
|
|
|
|
void InputConfigDialog::SetDevice(wxCommandEvent&)
|
|
{
|
|
controller->default_device.FromString(WxStrToStr(device_cbox->GetValue()));
|
|
|
|
// show user what it was validated as
|
|
device_cbox->SetValue(StrToWxStr(controller->default_device.ToString()));
|
|
|
|
// this will set all the controls to this default device
|
|
controller->UpdateDefaultDevice();
|
|
|
|
// 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->expression.clear();
|
|
|
|
const auto lock = ControllerEmu::EmulatedController::GetStateLock();
|
|
control_reference->UpdateReference(g_controller_interface,
|
|
m_parent->GetController()->default_device);
|
|
|
|
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,
|
|
ciface::Core::DeviceQualifier* control_device = nullptr,
|
|
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()->default_device);
|
|
|
|
return true;
|
|
}
|
|
|
|
void ControlDialog::SetSelectedControl(wxCommandEvent&)
|
|
{
|
|
wxString expr;
|
|
|
|
if (!GetExpressionForSelectedControl(expr))
|
|
return;
|
|
|
|
textctrl->WriteText(expr);
|
|
control_reference->expression = textctrl->GetValue();
|
|
|
|
const auto lock = ControllerEmu::EmulatedController::GetStateLock();
|
|
control_reference->UpdateReference(g_controller_interface,
|
|
m_parent->GetController()->default_device);
|
|
|
|
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->expression = textctrl->GetValue();
|
|
|
|
const auto lock = ControllerEmu::EmulatedController::GetStateLock();
|
|
control_reference->UpdateReference(g_controller_interface,
|
|
m_parent->GetController()->default_device);
|
|
|
|
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<ControlState>(event.GetPosition()) / SLIDER_TICK_COUNT;
|
|
}
|
|
|
|
void ControlDialog::OnRangeSpin(wxSpinEvent& event)
|
|
{
|
|
m_range_slider->SetValue(event.GetValue());
|
|
control_reference->range = static_cast<ControlState>(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<wxControl*>(event.GetEventObject());
|
|
auto* const pad_setting = static_cast<PadSetting*>(control->GetClientData());
|
|
pad_setting->UpdateValue();
|
|
}
|
|
|
|
void InputConfigDialog::AdjustBooleanSetting(wxCommandEvent& event)
|
|
{
|
|
const auto* const control = static_cast<wxControl*>(event.GetEventObject());
|
|
auto* const pad_setting = static_cast<PadSettingCheckBox*>(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->expression.clear();
|
|
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<ControlButton*>(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->default_device);
|
|
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->expression = expr;
|
|
const auto lock = ControllerEmu::EmulatedController::GetStateLock();
|
|
button->control_reference->UpdateReference(g_controller_interface,
|
|
controller->default_device);
|
|
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<int>(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->default_device.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<const char* const, 2> exclude_buttons{{"Mic", "Modifier"}};
|
|
static constexpr std::array<const char* const, 7> 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<double>(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<double>(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;
|
|
}
|