// Copyright (C) 2003 Dolphin Project. // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, version 2.0. // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License 2.0 for more details. // A copy of the GPL 2.0 should have been included with the program. // If not, see http://www.gnu.org/licenses/ // Official SVN repository and contact information can be found at // http://code.google.com/p/dolphin-emu/ #include "wiimote_hid.h" #include "main.h" #include "ConfigPadDlg.h" #include "Config.h" #include "EmuMain.h" // for WiiMoteEmu class #include "EmuDefinitions.h" #if defined(HAVE_X11) && HAVE_X11 #include "X11InputBase.h" #endif // Replace the harder to understand -1 with "" for the sake of user friendliness void WiimotePadConfigDialog::ToBlank(bool ToBlank, int Id) { if (!ControlsCreated) return; if(ToBlank) { if (GetButtonText(Id) == wxString(wxT("-1"))) SetButtonText(Id, wxString()); } else { if (GetButtonText(Id).IsEmpty()) SetButtonText(Id, wxString(wxT("-1"))); } } void WiimotePadConfigDialog::DoChangeDeadZone() { float Rad; Rad = (float)WiiMoteEmu::WiiMapping[m_Page].DeadZoneL * ((float)BoxW / 100.0) * 0.5; m_bmpDeadZoneLeftIn[m_Page]->SetBitmap(CreateBitmapClear()); m_bmpDeadZoneLeftIn[m_Page]->SetSize(0, 0); m_bmpDeadZoneLeftIn[m_Page]->SetBitmap(CreateBitmapDeadZone((int)Rad)); m_bmpDeadZoneLeftIn[m_Page]->SetPosition(wxPoint(BoxW / 2 - (int)Rad, BoxH / 2 - (int)Rad)); m_bmpDeadZoneLeftIn[m_Page]->Refresh(); Rad = (float)WiiMoteEmu::WiiMapping[m_Page].DeadZoneR * ((float)BoxW / 100.0) * 0.5; m_bmpDeadZoneRightIn[m_Page]->SetBitmap(CreateBitmapClear()); m_bmpDeadZoneRightIn[m_Page]->SetSize(0, 0); m_bmpDeadZoneRightIn[m_Page]->SetBitmap(CreateBitmapDeadZone((int)Rad)); m_bmpDeadZoneRightIn[m_Page]->SetPosition(wxPoint(BoxW / 2 - (int)Rad, BoxH / 2 - (int)Rad)); m_bmpDeadZoneRightIn[m_Page]->Refresh(); } // Set dialog items from saved values // Update the textbox for the buttons void WiimotePadConfigDialog::SetButtonText(int id, const wxString &str) { if (IDB_ANALOG_LEFT_X <= id && id <= IDB_TRIGGER_R) m_Button_Analog[id - IDB_ANALOG_LEFT_X][m_Page]->SetLabel(str); else if (IDB_WM_A <= id && id <= IDB_WM_SHAKE) m_Button_Wiimote[id - IDB_WM_A][m_Page]->SetLabel(str); else if (IDB_NC_Z <= id && id <= IDB_NC_SHAKE) m_Button_NunChuck[id - IDB_NC_Z][m_Page]->SetLabel(str); else if (IDB_CC_A <= id && id <= IDB_CC_RD) m_Button_Classic[id - IDB_CC_A][m_Page]->SetLabel(str); else if (IDB_GH3_GREEN <= id && id <= IDB_GH3_STRUM_DOWN) m_Button_GH3[id - IDB_GH3_GREEN][m_Page]->SetLabel(str); } // Get the text in the textbox for the buttons wxString WiimotePadConfigDialog::GetButtonText(int id) { if (IDB_ANALOG_LEFT_X <= id && id <= IDB_TRIGGER_R) return m_Button_Analog[id - IDB_ANALOG_LEFT_X][m_Page]->GetLabel(); else if (IDB_WM_A <= id && id <= IDB_WM_SHAKE) return m_Button_Wiimote[id - IDB_WM_A][m_Page]->GetLabel(); else if (IDB_NC_Z <= id && id <= IDB_NC_SHAKE) return m_Button_NunChuck[id - IDB_NC_Z][m_Page]->GetLabel(); else if (IDB_CC_A <= id && id <= IDB_CC_RD) return m_Button_Classic[id - IDB_CC_A][m_Page]->GetLabel(); else if (IDB_GH3_GREEN <= id && id <= IDB_GH3_STRUM_DOWN) return m_Button_GH3[id - IDB_GH3_GREEN][m_Page]->GetLabel(); return wxString(); } // Configure button mapping // Wait for button press /* Loop or timer: There are basically two ways to do this. With a while() or for() loop, or with a timer. The downside with the while() or for() loop is that there is no way to stop it if the user should select to configure another button while we are still in an old loop. What will happen then is that we start another parallel loop (at least in Windows) that blocks the old loop. And our only option to wait for the old loop to finish is with a new loop, and that will block the old loop for as long as it's going on. Therefore a timer is easier to control. */ void WiimotePadConfigDialog::GetButtons(wxCommandEvent& event) { if (event.GetEventType() != wxEVT_COMMAND_BUTTON_CLICKED) return; if (m_ButtonMappingTimer->IsRunning()) return; wxButton* pButton = (wxButton *)event.GetEventObject(); OldLabel = pButton->GetLabel(); pButton->SetLabel(wxT("")); DoGetButtons(pButton->GetId()); } void WiimotePadConfigDialog::DoGetButtons(int _GetId) { // Collect the starting values // Get the current controller int PadID = WiiMoteEmu::WiiMapping[m_Page].ID; // Get the controller and trigger type int TriggerType = WiiMoteEmu::WiiMapping[m_Page].TriggerType; // Collect the accepted buttons for this slot bool LeftRight = (_GetId == IDB_TRIGGER_L || _GetId == IDB_TRIGGER_R); bool Axis = (_GetId >= IDB_ANALOG_LEFT_X && _GetId <= IDB_TRIGGER_R) // Don't allow SDL axis input for the shoulder buttons if XInput is selected && !(TriggerType == InputCommon::CTL_TRIGGER_XINPUT && (_GetId == IDB_TRIGGER_L || _GetId == IDB_TRIGGER_R) ); bool XInput = (TriggerType == InputCommon::CTL_TRIGGER_XINPUT); bool Button = (_GetId >= IDB_WM_A && _GetId <= IDB_GH3_STRUM_DOWN); bool Hat = (_GetId >= IDB_WM_A && _GetId <= IDB_GH3_STRUM_DOWN); bool NoTriggerFilter = false; // Values used in this function int Seconds = 4; // Seconds to wait for int TimesPerSecond = 40; // How often to run the check // Values returned from InputCommon::GetButton() int value; // Axis value or Hat value int type; // Button type int KeyPressed = 0; int pressed = 0; bool Succeed = false; bool Stop = false; // Stop the timer //DEBUG_LOG(WIIMOTE, "Before (%i) Id:%i %i IsRunning:%i", // GetButtonWaitingTimer, GetButtonWaitingID, _GetId, m_ButtonMappingTimer->IsRunning()); // If the Id has changed or the timer is not running we should start one if( GetButtonWaitingID != _GetId || !m_ButtonMappingTimer->IsRunning() ) { if(m_ButtonMappingTimer->IsRunning()) m_ButtonMappingTimer->Stop(); // Save the button Id GetButtonWaitingID = _GetId; GetButtonWaitingTimer = 0; // Start the timer #if wxUSE_TIMER m_ButtonMappingTimer->Start(1000 / TimesPerSecond); #endif DEBUG_LOG(WIIMOTE, "Timer Started: Pad:%i _GetId:%i " "Allowed input is Axis:%i LeftRight:%i XInput:%i Button:%i Hat:%i", WiiMoteEmu::WiiMapping[m_Page].ID, _GetId, Axis, LeftRight, XInput, Button, Hat); } // Check for buttons // If there is a timer we should not create a new one else if (WiiMoteEmu::NumGoodPads > 0) { InputCommon::GetButton( WiiMoteEmu::joyinfo[PadID].joy, PadID, WiiMoteEmu::joyinfo[PadID].NumButtons, WiiMoteEmu::joyinfo[PadID].NumAxes, WiiMoteEmu::joyinfo[PadID].NumHats, KeyPressed, value, type, pressed, Succeed, Stop, LeftRight, Axis, XInput, Button, Hat, NoTriggerFilter); } // Process results // Count each time GetButtonWaitingTimer++; // This is run every second if (GetButtonWaitingTimer % TimesPerSecond == 0) { // Current time int TmpTime = Seconds - (GetButtonWaitingTimer / TimesPerSecond); // Update text SetButtonText(_GetId, wxString::Format(wxT("[ %d ]"), TmpTime)); } // Time's up if (GetButtonWaitingTimer / TimesPerSecond >= Seconds) { Stop = true; // Revert back to old label SetButtonText(_GetId, OldLabel); } // If we got a button if(Succeed) { Stop = true; // We need to assign hat special code if (type == InputCommon::CTL_HAT) { // Index of pressed starts from 0 if (value & SDL_HAT_UP) pressed = 0x0100 + 0x0010 * pressed + SDL_HAT_UP; else if (value & SDL_HAT_DOWN) pressed = 0x0100 + 0x0010 * pressed + SDL_HAT_DOWN; else if (value & SDL_HAT_LEFT) pressed = 0x0100 + 0x0010 * pressed + SDL_HAT_LEFT; else if (value & SDL_HAT_RIGHT) pressed = 0x0100 + 0x0010 * pressed + SDL_HAT_RIGHT; else pressed = -1; } if (IDB_WM_A <= _GetId && _GetId <= IDB_GH3_STRUM_DOWN) { // Better make the pad button code far from virtual key code SaveButtonMapping(_GetId, 0x1000 + pressed); SetButtonText(_GetId, wxString::Format(wxT("PAD: %d"), pressed)); } else if (IDB_ANALOG_LEFT_X <= _GetId && _GetId <= IDB_TRIGGER_R) { SaveButtonMapping(_GetId, pressed); SetButtonText(_GetId, wxString::Format(wxT("%d"), pressed)); } } // Stop the timer if(Stop) { DEBUG_LOG(WIIMOTE, "Timer Stopped for Pad:%i _GetId:%i", WiiMoteEmu::WiiMapping[m_Page].ID, _GetId); m_ButtonMappingTimer->Stop(); GetButtonWaitingTimer = 0; GetButtonWaitingID = 0; ClickedButton = NULL; } // If we got a bad button if(KeyPressed == -1) { // Update text SetButtonText(_GetId, wxString(wxT("PAD: -1"))); // Notify the user wxMessageBox(wxString::Format( wxT("You selected a key with a to low key code (%i), please") wxT(" select another key with a higher key code."), pressed) , wxT("Notice"), wxICON_INFORMATION); } } // Convert the 0x8000 range values to BoxW and BoxH for the plot void WiimotePadConfigDialog::Convert2Box(int &x) { // Border adjustment int BoxW_ = BoxW - 2; int BoxH_ = BoxH - 2; // Convert values x = (BoxW_ / 2) + (x * BoxW_ / (32767 * 2)); } // Update the input status boxes void WiimotePadConfigDialog::UpdatePadInfo(wxTimerEvent& WXUNUSED(event)) { //DEBUG_LOG(WIIMOTE, "SDL_WasInit: %i", SDL_WasInit(0)); /* Return if it's not enabled or not detected. The ID should never be less than zero here, it can only be that because of a manual ini file change, but we make that check anway. */ if (WiiMoteEmu::WiiMapping[m_Page].ID < 0 || WiiMoteEmu::WiiMapping[m_Page].ID >= WiiMoteEmu::NumGoodPads) { m_tStatusLeftIn[m_Page]->SetLabel(wxT("Not connected")); m_tStatusLeftOut[m_Page]->SetLabel(wxT("Not connected")); m_tStatusRightIn[m_Page]->SetLabel(wxT("Not connected")); m_tStatusRightOut[m_Page]->SetLabel(wxT("Not connected")); m_TriggerStatusL[m_Page]->SetLabel(wxT("000")); m_TriggerStatusR[m_Page]->SetLabel(wxT("000")); return; } // Check that Dolphin is in focus, otherwise don't update the pad status //if (IsFocus()) WiiMoteEmu::GetAxisState(WiiMoteEmu::WiiMapping[m_Page]); // Analog stick // Get original values int main_x = WiiMoteEmu::WiiMapping[m_Page].AxisState.Lx; int main_y = WiiMoteEmu::WiiMapping[m_Page].AxisState.Ly; int right_x = WiiMoteEmu::WiiMapping[m_Page].AxisState.Rx; int right_y = WiiMoteEmu::WiiMapping[m_Page].AxisState.Ry; // Get adjusted values int main_x_after = main_x, main_y_after = main_y; int right_x_after = right_x, right_y_after = right_y; // Produce square if(WiiMoteEmu::WiiMapping[m_Page].bCircle2Square) InputCommon::Square2Circle(main_x_after, main_y_after, 0, WiiMoteEmu::WiiMapping[m_Page].Diagonal, true); // Check dead zone float DeadZoneLeft = (float)WiiMoteEmu::WiiMapping[m_Page].DeadZoneL / 100.0; float DeadZoneRight = (float)WiiMoteEmu::WiiMapping[m_Page].DeadZoneR / 100.0; if (InputCommon::IsDeadZone(DeadZoneLeft, main_x_after, main_y_after)) { main_x_after = 0; main_y_after = 0; } if (InputCommon::IsDeadZone(DeadZoneRight, right_x_after, right_y_after)) { right_x_after = 0; right_y_after = 0; } // Show the adjusted angles in the status box // Change 0x8000 to 180 /* float x8000 = 0x8000; main_x_after = main_x_after * (180 / x8000); main_y_after = main_y_after * (180 / x8000); float f_main_x_after = (float)main_x_after; float f_main_y_after = (float)main_y_after; WiiMoteEmu::AdjustAngles(f_main_x_after, f_main_y_after); //WiiMoteEmu::AdjustAngles(f_main_x_after, f_main_y_after, true); main_x_after = (int)f_main_x_after; main_y_after = (int)f_main_y_after; // Change back 180 to 0x8000 main_x_after = main_x_after * (x8000 / 180); main_y_after = main_y_after * (x8000 / 180); */ // Convert the values to fractions float f_x = main_x / 32767.0; float f_y = main_y / 32767.0; float f_x_aft = main_x_after / 32767.0; float f_y_aft = main_y_after / 32767.0; float f_Rx = right_x / 32767.0; float f_Ry = right_y / 32767.0; float f_Rx_aft = right_x_after / 32767.0; float f_Ry_aft = right_y_after / 32767.0; m_tStatusLeftIn[m_Page]->SetLabel(wxString::Format(wxT("x:%1.2f y:%1.2f"), f_x, f_y )); m_tStatusLeftOut[m_Page]->SetLabel(wxString::Format(wxT("x:%1.2f y:%1.2f"), f_x_aft, f_y_aft)); m_tStatusRightIn[m_Page]->SetLabel(wxString::Format(wxT("x:%1.2f y:%1.2f"), f_Rx, f_Ry)); m_tStatusRightOut[m_Page]->SetLabel(wxString::Format(wxT("x:%1.2f y:%1.2f"), f_Rx_aft, f_Ry_aft)); // Adjust the values for the plot Convert2Box(main_x); Convert2Box(main_y); Convert2Box(right_x); Convert2Box(right_y); Convert2Box(main_x_after); Convert2Box(main_y_after); Convert2Box(right_x_after); Convert2Box(right_y_after); // Adjust the dot m_bmpDotLeftIn[m_Page]->SetPosition(wxPoint(main_x, main_y)); m_bmpDotLeftOut[m_Page]->SetPosition(wxPoint(main_x_after, main_y_after)); m_bmpDotRightIn[m_Page]->SetPosition(wxPoint(right_x, right_y)); m_bmpDotRightOut[m_Page]->SetPosition(wxPoint(right_x_after, right_y_after)); // Get the trigger values int TriggerLeft = WiiMoteEmu::WiiMapping[m_Page].AxisState.Tl; int TriggerRight = WiiMoteEmu::WiiMapping[m_Page].AxisState.Tr; // Convert the triggers values if (WiiMoteEmu::WiiMapping[m_Page].TriggerType == InputCommon::CTL_TRIGGER_SDL) { TriggerLeft = InputCommon::Pad_Convert(TriggerLeft); TriggerRight = InputCommon::Pad_Convert(TriggerRight); } m_TriggerStatusL[m_Page]->SetLabel(wxString::Format(wxT("%03i"), TriggerLeft)); m_TriggerStatusR[m_Page]->SetLabel(wxString::Format(wxT("%03i"), TriggerRight)); }