mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2024-11-15 13:57:57 -07:00
2db2e88620
Add relative input for the Wiimote IR
1198 lines
37 KiB
C++
1198 lines
37 KiB
C++
// Copyright 2010 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <algorithm>
|
|
#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/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/slider.h>
|
|
#include <wx/spinctrl.h>
|
|
#include <wx/statbmp.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 "Core/Core.h"
|
|
#include "Core/HW/GCKeyboard.h"
|
|
#include "Core/HW/GCPad.h"
|
|
#include "Core/HW/Wiimote.h"
|
|
#include "Core/HotkeyManager.h"
|
|
#include "DolphinWX/InputConfigDiag.h"
|
|
#include "DolphinWX/WxUtils.h"
|
|
#include "InputCommon/ControllerEmu.h"
|
|
#include "InputCommon/ControllerInterface/ControllerInterface.h"
|
|
#include "InputCommon/ControllerInterface/Device.h"
|
|
#include "InputCommon/ControllerInterface/ExpressionParser.h"
|
|
#include "InputCommon/InputConfig.h"
|
|
|
|
using namespace ciface::ExpressionParser;
|
|
|
|
void GamepadPage::ConfigExtension(wxCommandEvent& event)
|
|
{
|
|
ControllerEmu::Extension* const ex = ((ExtensionButton*)event.GetEventObject())->extension;
|
|
|
|
// show config diag, if "none" isn't selected
|
|
if (ex->switch_extension)
|
|
{
|
|
wxDialog dlg(this, wxID_ANY,
|
|
wxGetTranslation(StrToWxStr(ex->attachments[ex->switch_extension]->GetName())));
|
|
|
|
wxBoxSizer* const main_szr = new wxBoxSizer(wxVERTICAL);
|
|
const std::size_t orig_size = control_groups.size();
|
|
|
|
ControlGroupsSizer* const szr = new ControlGroupsSizer(
|
|
ex->attachments[ex->switch_extension].get(), &dlg, this, &control_groups);
|
|
main_szr->Add(szr, 0, wxLEFT, 5);
|
|
main_szr->Add(dlg.CreateButtonSizer(wxOK), 0, wxEXPAND | wxLEFT | wxRIGHT | wxBOTTOM, 5);
|
|
dlg.SetSizerAndFit(main_szr);
|
|
dlg.Center();
|
|
|
|
dlg.ShowModal();
|
|
|
|
// remove the new groups that were just added, now that the window closed
|
|
control_groups.resize(orig_size);
|
|
}
|
|
}
|
|
|
|
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::ControlGroup::BooleanSetting* const _setting)
|
|
: PadSetting(new wxCheckBox(parent, wxID_ANY, wxGetTranslation(StrToWxStr(_setting->m_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());
|
|
}
|
|
|
|
void PadSettingSpin::UpdateGUI()
|
|
{
|
|
((wxSpinCtrl*)wxcontrol)->SetValue((int)(setting->GetValue() * 100));
|
|
}
|
|
|
|
void PadSettingSpin::UpdateValue()
|
|
{
|
|
setting->SetValue(ControlState(((wxSpinCtrl*)wxcontrol)->GetValue()) / 100);
|
|
}
|
|
|
|
ControlDialog::ControlDialog(GamepadPage* const parent, InputConfig& config,
|
|
ControllerInterface::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->controller->default_device;
|
|
|
|
// GetStrings() sounds slow :/
|
|
// device_cbox = new wxComboBox(this, wxID_ANY, StrToWxStr(ref->device_qualifier.ToString()),
|
|
// wxDefaultPosition, wxSize(256,-1), parent->device_cbox->GetStrings(), wxTE_PROCESS_ENTER);
|
|
device_cbox =
|
|
new wxComboBox(this, wxID_ANY, StrToWxStr(m_devq.ToString()), wxDefaultPosition,
|
|
wxSize(256, -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->Add(device_cbox, 0, wxEXPAND | wxALL, 5);
|
|
|
|
wxBoxSizer* const szr = new wxBoxSizer(wxVERTICAL);
|
|
szr->Add(d_szr, 0, wxEXPAND | wxLEFT | wxRIGHT | wxTOP, 5);
|
|
szr->Add(control_chooser, 1, wxEXPAND | wxALL, 5);
|
|
|
|
SetSizerAndFit(szr); // needed
|
|
|
|
UpdateGUI();
|
|
SetFocus();
|
|
}
|
|
|
|
ControlButton::ControlButton(wxWindow* const parent,
|
|
ControllerInterface::ControlReference* const _ref,
|
|
const std::string& name, const unsigned int width,
|
|
const std::string& label)
|
|
: wxButton(parent, wxID_ANY, "", wxDefaultPosition, wxSize(width, 20)), control_reference(_ref),
|
|
m_name(name)
|
|
{
|
|
if (label.empty())
|
|
SetLabel(StrToWxStr(_ref->expression));
|
|
else
|
|
SetLabel(StrToWxStr(label));
|
|
}
|
|
|
|
void InputConfigDialog::UpdateProfileComboBox()
|
|
{
|
|
std::string pname(File::GetUserPath(D_CONFIG_IDX));
|
|
pname += PROFILES_PATH;
|
|
pname += m_config.GetProfileName();
|
|
|
|
std::vector<std::string> sv = DoFileSearch({".ini"}, {pname});
|
|
|
|
wxArrayString strs;
|
|
for (const std::string& filename : sv)
|
|
{
|
|
std::string base;
|
|
SplitPath(filename, nullptr, &base, nullptr);
|
|
strs.push_back(StrToWxStr(base));
|
|
}
|
|
|
|
for (GamepadPage* page : m_padpages)
|
|
{
|
|
page->profile_cbox->Clear();
|
|
page->profile_cbox->Append(strs);
|
|
}
|
|
}
|
|
|
|
void InputConfigDialog::UpdateControlReferences()
|
|
{
|
|
for (GamepadPage* page : m_padpages)
|
|
{
|
|
page->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 range_slider->GetValue();
|
|
}
|
|
|
|
void ControlDialog::UpdateListContents()
|
|
{
|
|
control_lbox->Clear();
|
|
|
|
const auto dev = g_controller_interface.FindDevice(m_devq);
|
|
if (dev != nullptr)
|
|
{
|
|
if (control_reference->is_input)
|
|
{
|
|
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->parse_error)
|
|
{
|
|
case EXPRESSION_PARSE_SYNTAX_ERROR:
|
|
m_error_label->SetLabel(_("Syntax error"));
|
|
break;
|
|
case EXPRESSION_PARSE_NO_DEVICE:
|
|
m_error_label->SetLabel(_("Device not found"));
|
|
break;
|
|
default:
|
|
m_error_label->SetLabel("");
|
|
}
|
|
};
|
|
|
|
void GamepadPage::UpdateGUI()
|
|
{
|
|
device_cbox->SetValue(StrToWxStr(controller->default_device.ToString()));
|
|
|
|
for (ControlGroupBox* cgBox : control_groups)
|
|
{
|
|
for (ControlButton* button : cgBox->control_buttons)
|
|
{
|
|
wxString expr = StrToWxStr(button->control_reference->expression);
|
|
expr.Replace("&", "&&");
|
|
button->SetLabel(expr);
|
|
}
|
|
|
|
for (PadSetting* padSetting : cgBox->options)
|
|
{
|
|
padSetting->UpdateGUI();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GamepadPage::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 GamepadPage::LoadDefaults(wxCommandEvent&)
|
|
{
|
|
controller->LoadDefaults(g_controller_interface);
|
|
|
|
controller->UpdateReferences(g_controller_interface);
|
|
|
|
UpdateGUI();
|
|
}
|
|
|
|
bool ControlDialog::Validate()
|
|
{
|
|
control_reference->expression = WxStrToStr(textctrl->GetValue());
|
|
|
|
auto lock = ControllerEmu::GetStateLock();
|
|
g_controller_interface.UpdateReference(control_reference, m_parent->controller->default_device);
|
|
|
|
UpdateGUI();
|
|
|
|
return (control_reference->parse_error == EXPRESSION_PARSE_SUCCESS ||
|
|
control_reference->parse_error == EXPRESSION_PARSE_NO_DEVICE);
|
|
}
|
|
|
|
void GamepadPage::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();
|
|
|
|
auto lock = ControllerEmu::GetStateLock();
|
|
g_controller_interface.UpdateReference(control_reference, m_parent->controller->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->controller->default_device);
|
|
|
|
return true;
|
|
}
|
|
|
|
void ControlDialog::SetSelectedControl(wxCommandEvent&)
|
|
{
|
|
wxString expr;
|
|
|
|
if (!GetExpressionForSelectedControl(expr))
|
|
return;
|
|
|
|
textctrl->WriteText(expr);
|
|
control_reference->expression = textctrl->GetValue();
|
|
|
|
auto lock = ControllerEmu::GetStateLock();
|
|
g_controller_interface.UpdateReference(control_reference, m_parent->controller->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();
|
|
|
|
auto lock = ControllerEmu::GetStateLock();
|
|
g_controller_interface.UpdateReference(control_reference, m_parent->controller->default_device);
|
|
|
|
UpdateGUI();
|
|
}
|
|
|
|
void GamepadPage::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 GamepadPage::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 GamepadPage::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 GamepadPage::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 GamepadPage::AdjustControlOption(wxCommandEvent&)
|
|
{
|
|
m_control_dialog->control_reference->range =
|
|
(ControlState)(m_control_dialog->GetRangeSliderValue()) / SLIDER_TICK_COUNT;
|
|
}
|
|
|
|
void GamepadPage::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 GamepadPage::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 GamepadPage::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 GamepadPage::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;
|
|
auto lock = ControllerEmu::GetStateLock();
|
|
g_controller_interface.UpdateReference(button->control_reference, 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(GamepadPage* const parent)
|
|
{
|
|
wxStaticBoxSizer* const main_szr = new wxStaticBoxSizer(
|
|
wxVERTICAL, this, control_reference->is_input ? _("Input") : _("Output"));
|
|
|
|
textctrl = new wxTextCtrl(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(-1, 48),
|
|
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->is_input ? _("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, wxSize(-1, 64));
|
|
|
|
wxBoxSizer* const button_sizer = new wxBoxSizer(wxVERTICAL);
|
|
button_sizer->Add(detect_button, 1, 0, 5);
|
|
button_sizer->Add(select_button, 1, 0, 5);
|
|
button_sizer->Add(or_button, 1, 0, 5);
|
|
|
|
if (control_reference->is_input)
|
|
{
|
|
// 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, 0, 5);
|
|
button_sizer->Add(not_button, 1, 0, 5);
|
|
button_sizer->Add(add_button, 1, 0, 5);
|
|
}
|
|
|
|
range_slider =
|
|
new wxSlider(this, wxID_ANY, SLIDER_TICK_COUNT, -SLIDER_TICK_COUNT * 5, SLIDER_TICK_COUNT * 5,
|
|
wxDefaultPosition, wxDefaultSize, wxSL_TOP | wxSL_LABELS /*| wxSL_AUTOTICKS*/);
|
|
|
|
range_slider->SetValue((int)(control_reference->range * SLIDER_TICK_COUNT));
|
|
|
|
detect_button->Bind(wxEVT_BUTTON, &ControlDialog::DetectControl, this);
|
|
clear_button->Bind(wxEVT_BUTTON, &ControlDialog::ClearControl, this);
|
|
|
|
range_slider->Bind(wxEVT_SCROLL_CHANGED, &GamepadPage::AdjustControlOption, parent);
|
|
wxStaticText* const range_label = new wxStaticText(this, wxID_ANY, _("Range"));
|
|
|
|
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(range_label, 0, wxCENTER | wxLEFT, 5);
|
|
range_sizer->Add(range_slider, 1, wxEXPAND | wxLEFT, 5);
|
|
|
|
wxBoxSizer* const ctrls_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
ctrls_sizer->Add(control_lbox, 1, wxEXPAND, 0);
|
|
ctrls_sizer->Add(button_sizer, 0, wxEXPAND, 0);
|
|
|
|
wxSizer* const bottom_btns_sizer = CreateButtonSizer(wxOK | wxAPPLY);
|
|
bottom_btns_sizer->Prepend(clear_button, 0, wxLEFT, 5);
|
|
|
|
main_szr->Add(range_sizer, 0, wxEXPAND | wxLEFT | wxRIGHT, 5);
|
|
main_szr->Add(ctrls_sizer, 0, wxEXPAND | wxLEFT | wxRIGHT | wxBOTTOM, 5);
|
|
main_szr->Add(textctrl, 1, wxEXPAND | wxLEFT | wxRIGHT | wxBOTTOM, 5);
|
|
main_szr->Add(bottom_btns_sizer, 0, wxEXPAND | wxBOTTOM | wxRIGHT, 5);
|
|
main_szr->Add(m_bound_label, 0, wxCENTER, 0);
|
|
main_szr->Add(m_error_label, 0, wxCENTER, 0);
|
|
|
|
UpdateListContents();
|
|
|
|
return main_szr;
|
|
}
|
|
|
|
void GamepadPage::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";
|
|
}
|
|
}
|
|
|
|
void GamepadPage::LoadProfile(wxCommandEvent&)
|
|
{
|
|
std::string fname;
|
|
GamepadPage::GetProfilePath(fname);
|
|
|
|
if (!File::Exists(fname))
|
|
return;
|
|
|
|
IniFile inifile;
|
|
inifile.Load(fname);
|
|
|
|
controller->LoadConfig(inifile.GetOrCreateSection("Profile"));
|
|
controller->UpdateReferences(g_controller_interface);
|
|
|
|
UpdateGUI();
|
|
}
|
|
|
|
void GamepadPage::SaveProfile(wxCommandEvent&)
|
|
{
|
|
std::string fname;
|
|
GamepadPage::GetProfilePath(fname);
|
|
File::CreateFullPath(fname);
|
|
|
|
if (!fname.empty())
|
|
{
|
|
IniFile inifile;
|
|
controller->SaveConfig(inifile.GetOrCreateSection("Profile"));
|
|
inifile.Save(fname);
|
|
|
|
m_config_dialog->UpdateProfileComboBox();
|
|
}
|
|
else
|
|
{
|
|
WxUtils::ShowErrorDialog(_("You must enter a valid profile name."));
|
|
}
|
|
}
|
|
|
|
void GamepadPage::DeleteProfile(wxCommandEvent&)
|
|
{
|
|
std::string fname;
|
|
GamepadPage::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);
|
|
|
|
m_config_dialog->UpdateProfileComboBox();
|
|
}
|
|
}
|
|
|
|
void InputConfigDialog::UpdateDeviceComboBox()
|
|
{
|
|
for (GamepadPage* page : m_padpages)
|
|
{
|
|
page->device_cbox->Clear();
|
|
|
|
for (const std::string& device_string : g_controller_interface.GetAllDeviceStrings())
|
|
page->device_cbox->Append(StrToWxStr(device_string));
|
|
|
|
page->device_cbox->SetValue(StrToWxStr(page->controller->default_device.ToString()));
|
|
}
|
|
}
|
|
|
|
void GamepadPage::RefreshDevices(wxCommandEvent&)
|
|
{
|
|
bool was_unpaused = Core::PauseAndLock(true);
|
|
|
|
// refresh devices
|
|
g_controller_interface.Reinitialize();
|
|
|
|
// update all control references
|
|
m_config_dialog->UpdateControlReferences();
|
|
|
|
// update device cbox
|
|
m_config_dialog->UpdateDeviceComboBox();
|
|
|
|
Wiimote::LoadConfig();
|
|
Keyboard::LoadConfig();
|
|
Pad::LoadConfig();
|
|
HotkeyManagerEmu::LoadConfig();
|
|
|
|
UpdateGUI();
|
|
|
|
Core::PauseAndLock(false, was_unpaused);
|
|
}
|
|
|
|
ControlGroupBox::~ControlGroupBox()
|
|
{
|
|
for (PadSetting* padSetting : options)
|
|
delete padSetting;
|
|
}
|
|
|
|
ControlGroupBox::ControlGroupBox(ControllerEmu::ControlGroup* const group, wxWindow* const parent,
|
|
GamepadPage* const eventsink)
|
|
: wxBoxSizer(wxVERTICAL), control_group(group)
|
|
{
|
|
static_bitmap = nullptr;
|
|
const std::vector<std::string> exclude_buttons = {"Mic", "Modifier"};
|
|
const std::vector<std::string> exclude_groups = {"IR", "Swing", "Tilt", "Shake",
|
|
"UDP Wiimote", "Extension", "Rumble"};
|
|
|
|
wxFont m_SmallFont(7, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL, wxFONTWEIGHT_NORMAL);
|
|
for (auto& control : group->controls)
|
|
{
|
|
wxStaticText* const label =
|
|
new wxStaticText(parent, wxID_ANY, wxGetTranslation(StrToWxStr(control->name)));
|
|
|
|
ControlButton* const control_button =
|
|
new ControlButton(parent, control->control_ref.get(), control->name, 80);
|
|
control_button->SetFont(m_SmallFont);
|
|
|
|
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->name) ==
|
|
exclude_buttons.end())
|
|
eventsink->control_buttons.push_back(control_button);
|
|
|
|
if (control->control_ref->is_input)
|
|
{
|
|
control_button->SetToolTip(
|
|
_("Left-click to detect input.\nMiddle-click to clear.\nRight-click for more options."));
|
|
control_button->Bind(wxEVT_BUTTON, &GamepadPage::DetectControl, eventsink);
|
|
}
|
|
else
|
|
{
|
|
control_button->SetToolTip(_("Left/Right-click for more options.\nMiddle-click to clear."));
|
|
control_button->Bind(wxEVT_BUTTON, &GamepadPage::ConfigControl, eventsink);
|
|
}
|
|
|
|
control_button->Bind(wxEVT_MIDDLE_DOWN, &GamepadPage::ClearControl, eventsink);
|
|
control_button->Bind(wxEVT_RIGHT_UP, &GamepadPage::ConfigControl, eventsink);
|
|
|
|
wxBoxSizer* const control_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
control_sizer->AddStretchSpacer(1);
|
|
control_sizer->Add(label, 0, wxCENTER | wxRIGHT, 3);
|
|
control_sizer->Add(control_button, 0, 0, 0);
|
|
|
|
Add(control_sizer, 0, wxEXPAND | wxLEFT | wxRIGHT, 3);
|
|
}
|
|
|
|
wxMemoryDC dc;
|
|
|
|
switch (group->type)
|
|
{
|
|
case GROUP_TYPE_STICK:
|
|
case GROUP_TYPE_TILT:
|
|
case GROUP_TYPE_CURSOR:
|
|
case GROUP_TYPE_FORCE:
|
|
{
|
|
wxBitmap bitmap(64, 64);
|
|
dc.SelectObject(bitmap);
|
|
dc.Clear();
|
|
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, &GamepadPage::AdjustSetting, eventsink);
|
|
options.push_back(setting);
|
|
szr->Add(
|
|
new wxStaticText(parent, wxID_ANY, wxGetTranslation(StrToWxStr(groupSetting->m_name))));
|
|
szr->Add(setting->wxcontrol, 0, wxLEFT, 0);
|
|
}
|
|
for (auto& groupSetting : group->boolean_settings)
|
|
{
|
|
auto* checkbox = new PadSettingCheckBox(parent, groupSetting.get());
|
|
checkbox->wxcontrol->Bind(wxEVT_CHECKBOX, &GamepadPage::AdjustBooleanSetting, eventsink);
|
|
options.push_back(checkbox);
|
|
Add(checkbox->wxcontrol, 0, wxALL | wxLEFT, 5);
|
|
}
|
|
|
|
wxBoxSizer* const h_szr = new wxBoxSizer(wxHORIZONTAL);
|
|
h_szr->Add(szr, 1, 0, 5);
|
|
h_szr->Add(static_bitmap, 0, wxALL | wxCENTER, 3);
|
|
|
|
Add(h_szr, 0, wxEXPAND | wxLEFT | wxCENTER | wxTOP, 3);
|
|
}
|
|
break;
|
|
case GROUP_TYPE_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);
|
|
wxBitmap bitmap(int(12 * button_cols + 1), (11 * button_rows) + 1);
|
|
|
|
dc.SelectObject(bitmap);
|
|
dc.Clear();
|
|
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, &GamepadPage::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, wxCENTER | wxRIGHT, 3);
|
|
szr->Add(threshold_cbox->wxcontrol, 0, wxRIGHT, 3);
|
|
|
|
Add(szr, 0, wxALL | wxCENTER, 3);
|
|
Add(static_bitmap, 0, wxALL | wxCENTER, 3);
|
|
}
|
|
break;
|
|
case GROUP_TYPE_MIXED_TRIGGERS:
|
|
case GROUP_TYPE_TRIGGERS:
|
|
case GROUP_TYPE_SLIDER:
|
|
{
|
|
int height = (int)(12 * group->controls.size());
|
|
int width = 64;
|
|
|
|
if (GROUP_TYPE_MIXED_TRIGGERS == group->type)
|
|
width = 64 + 12 + 1;
|
|
|
|
if (GROUP_TYPE_TRIGGERS != group->type)
|
|
height /= 2;
|
|
|
|
wxBitmap bitmap(width, height + 1);
|
|
dc.SelectObject(bitmap);
|
|
dc.Clear();
|
|
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, &GamepadPage::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,
|
|
wxCENTER | wxRIGHT, 3);
|
|
szr->Add(setting->wxcontrol, 0, wxRIGHT, 3);
|
|
Add(szr, 0, wxALL | wxCENTER, 3);
|
|
}
|
|
|
|
Add(static_bitmap, 0, wxALL | wxCENTER, 3);
|
|
}
|
|
break;
|
|
case GROUP_TYPE_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, &GamepadPage::AdjustSetting, eventsink);
|
|
configure_btn->Bind(wxEVT_BUTTON, &GamepadPage::ConfigExtension, eventsink);
|
|
|
|
Add(attachments->wxcontrol, 0, wxTOP | wxLEFT | wxRIGHT | wxEXPAND, 3);
|
|
Add(configure_btn, 0, wxALL | wxEXPAND, 3);
|
|
}
|
|
break;
|
|
default:
|
|
{
|
|
// options
|
|
for (auto& groupSetting : group->boolean_settings)
|
|
{
|
|
PadSettingCheckBox* setting_cbox = new PadSettingCheckBox(parent, groupSetting.get());
|
|
setting_cbox->wxcontrol->Bind(wxEVT_CHECKBOX, &GamepadPage::AdjustBooleanSetting, eventsink);
|
|
if (groupSetting->m_name == "Iterative Input")
|
|
groupSetting->SetValue(false);
|
|
options.push_back(setting_cbox);
|
|
Add(setting_cbox->wxcontrol, 0, wxALL | wxLEFT, 5);
|
|
}
|
|
for (auto& groupSetting : group->numeric_settings)
|
|
{
|
|
PadSettingSpin* setting = new PadSettingSpin(parent, groupSetting.get());
|
|
setting->wxcontrol->Bind(wxEVT_SPINCTRL, &GamepadPage::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,
|
|
wxCENTER | wxRIGHT, 3);
|
|
szr->Add(setting->wxcontrol, 0, wxRIGHT, 3);
|
|
Add(szr, 0, wxALL | wxCENTER, 3);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
dc.SelectObject(wxNullBitmap);
|
|
|
|
// AddStretchSpacer(0);
|
|
}
|
|
|
|
ControlGroupsSizer::ControlGroupsSizer(ControllerEmu* const controller, wxWindow* const parent,
|
|
GamepadPage* const eventsink,
|
|
std::vector<ControlGroupBox*>* groups)
|
|
: wxBoxSizer(wxHORIZONTAL)
|
|
{
|
|
size_t col_size = 0;
|
|
|
|
wxBoxSizer* stacked_groups = nullptr;
|
|
for (auto& group : controller->groups)
|
|
{
|
|
ControlGroupBox* control_group_box = new ControlGroupBox(group.get(), parent, eventsink);
|
|
wxStaticBoxSizer* control_group =
|
|
new wxStaticBoxSizer(wxVERTICAL, parent, wxGetTranslation(StrToWxStr(group->ui_name)));
|
|
control_group->Add(control_group_box);
|
|
|
|
const size_t grp_size =
|
|
group->controls.size() + group->numeric_settings.size() + group->boolean_settings.size();
|
|
col_size += grp_size;
|
|
if (col_size > 8 || nullptr == stacked_groups)
|
|
{
|
|
if (stacked_groups)
|
|
Add(stacked_groups, 0, /*wxEXPAND|*/ wxBOTTOM | wxRIGHT, 5);
|
|
|
|
stacked_groups = new wxBoxSizer(wxVERTICAL);
|
|
stacked_groups->Add(control_group, 0, wxEXPAND);
|
|
|
|
col_size = grp_size;
|
|
}
|
|
else
|
|
{
|
|
stacked_groups->Add(control_group, 0, wxEXPAND);
|
|
}
|
|
|
|
if (groups)
|
|
groups->push_back(control_group_box);
|
|
}
|
|
|
|
if (stacked_groups)
|
|
Add(stacked_groups, 0, /*wxEXPAND|*/ wxBOTTOM | wxRIGHT, 5);
|
|
}
|
|
|
|
GamepadPage::GamepadPage(wxWindow* parent, InputConfig& config, const int pad_num,
|
|
InputConfigDialog* const config_dialog)
|
|
: wxPanel(parent, wxID_ANY), controller(config.GetController(pad_num)),
|
|
m_config_dialog(config_dialog), m_config(config)
|
|
{
|
|
wxBoxSizer* control_group_sizer = new ControlGroupsSizer(controller, this, this, &control_groups);
|
|
|
|
wxStaticBoxSizer* profile_sbox = new wxStaticBoxSizer(wxHORIZONTAL, this, _("Profile"));
|
|
|
|
// device chooser
|
|
|
|
wxStaticBoxSizer* const device_sbox = new wxStaticBoxSizer(wxHORIZONTAL, this, _("Device"));
|
|
|
|
device_cbox = new wxComboBox(this, wxID_ANY, "", wxDefaultPosition, wxSize(64, -1));
|
|
device_cbox->ToggleWindowStyle(wxTE_PROCESS_ENTER);
|
|
|
|
wxButton* refresh_button =
|
|
new wxButton(this, wxID_ANY, _("Refresh"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT);
|
|
|
|
device_cbox->Bind(wxEVT_COMBOBOX, &GamepadPage::SetDevice, this);
|
|
device_cbox->Bind(wxEVT_TEXT_ENTER, &GamepadPage::SetDevice, this);
|
|
refresh_button->Bind(wxEVT_BUTTON, &GamepadPage::RefreshDevices, this);
|
|
|
|
device_sbox->Add(device_cbox, 1, wxLEFT | wxRIGHT, 3);
|
|
device_sbox->Add(refresh_button, 0, wxRIGHT | wxBOTTOM, 3);
|
|
|
|
wxButton* const default_button =
|
|
new wxButton(this, wxID_ANY, _("Default"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT);
|
|
wxButton* const clearall_button =
|
|
new wxButton(this, wxID_ANY, _("Clear"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT);
|
|
|
|
wxStaticBoxSizer* const clear_sbox = new wxStaticBoxSizer(wxHORIZONTAL, this, _("Reset"));
|
|
clear_sbox->Add(default_button, 1, wxLEFT, 3);
|
|
clear_sbox->Add(clearall_button, 1, wxRIGHT, 3);
|
|
|
|
clearall_button->Bind(wxEVT_BUTTON, &GamepadPage::ClearAll, this);
|
|
default_button->Bind(wxEVT_BUTTON, &GamepadPage::LoadDefaults, this);
|
|
|
|
profile_cbox = new wxComboBox(this, wxID_ANY, "", wxDefaultPosition, wxSize(64, -1));
|
|
|
|
wxButton* const pload_btn =
|
|
new wxButton(this, wxID_ANY, _("Load"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT);
|
|
wxButton* const psave_btn =
|
|
new wxButton(this, wxID_ANY, _("Save"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT);
|
|
wxButton* const pdelete_btn =
|
|
new wxButton(this, wxID_ANY, _("Delete"), wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT);
|
|
|
|
pload_btn->Bind(wxEVT_BUTTON, &GamepadPage::LoadProfile, this);
|
|
psave_btn->Bind(wxEVT_BUTTON, &GamepadPage::SaveProfile, this);
|
|
pdelete_btn->Bind(wxEVT_BUTTON, &GamepadPage::DeleteProfile, this);
|
|
|
|
profile_sbox->Add(profile_cbox, 1, wxLEFT, 3);
|
|
profile_sbox->Add(pload_btn, 0, wxLEFT, 3);
|
|
profile_sbox->Add(psave_btn, 0, 0, 3);
|
|
profile_sbox->Add(pdelete_btn, 0, wxRIGHT | wxBOTTOM, 3);
|
|
|
|
wxBoxSizer* const dio = new wxBoxSizer(wxHORIZONTAL);
|
|
dio->Add(device_sbox, 1, wxEXPAND | wxRIGHT, 5);
|
|
dio->Add(clear_sbox, 0, wxEXPAND | wxRIGHT, 5);
|
|
dio->Add(profile_sbox, 1, wxEXPAND | wxRIGHT, 5);
|
|
|
|
wxBoxSizer* const mapping = new wxBoxSizer(wxVERTICAL);
|
|
|
|
mapping->Add(dio, 1, wxEXPAND | wxLEFT | wxTOP | wxBOTTOM, 5);
|
|
mapping->Add(control_group_sizer, 0, wxLEFT | wxEXPAND, 5);
|
|
|
|
UpdateGUI();
|
|
|
|
SetSizerAndFit(mapping); // needed
|
|
Layout();
|
|
};
|
|
|
|
InputConfigDialog::InputConfigDialog(wxWindow* const parent, InputConfig& config,
|
|
const wxString& name, const int tab_num)
|
|
: wxDialog(parent, wxID_ANY, name, wxPoint(128, -1)), m_config(config)
|
|
{
|
|
m_pad_notebook = new wxNotebook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxNB_DEFAULT);
|
|
GamepadPage* gp = new GamepadPage(m_pad_notebook, m_config, tab_num, this);
|
|
m_padpages.push_back(gp);
|
|
m_pad_notebook->AddPage(gp, wxString::Format("%s [%u]",
|
|
wxGetTranslation(StrToWxStr(m_config.GetGUIName())),
|
|
1 + tab_num));
|
|
|
|
m_pad_notebook->SetSelection(0);
|
|
|
|
UpdateDeviceComboBox();
|
|
UpdateProfileComboBox();
|
|
|
|
Bind(wxEVT_CLOSE_WINDOW, &InputConfigDialog::OnClose, this);
|
|
Bind(wxEVT_BUTTON, &InputConfigDialog::OnCloseButton, this, wxID_CLOSE);
|
|
|
|
wxBoxSizer* const szr = new wxBoxSizer(wxVERTICAL);
|
|
szr->Add(m_pad_notebook, 0, wxEXPAND | wxTOP | wxLEFT | wxRIGHT, 5);
|
|
szr->Add(CreateButtonSizer(wxCLOSE), 0, wxEXPAND | wxALL, 5);
|
|
|
|
SetLayoutAdaptationMode(wxDIALOG_ADAPTATION_MODE_ENABLED);
|
|
SetSizerAndFit(szr);
|
|
Center();
|
|
|
|
// live preview update timer
|
|
m_update_timer.SetOwner(this);
|
|
Bind(wxEVT_TIMER, &InputConfigDialog::UpdateBitmaps, this);
|
|
m_update_timer.Start(PREVIEW_UPDATE_TIME, wxTIMER_CONTINUOUS);
|
|
}
|
|
|
|
int InputEventFilter::FilterEvent(wxEvent& event)
|
|
{
|
|
if (m_block && ShouldCatchEventType(event.GetEventType()))
|
|
{
|
|
event.StopPropagation();
|
|
return Event_Processed;
|
|
}
|
|
|
|
return Event_Skip;
|
|
}
|