Cxbx-Reloaded/Source/Win32/Cxbx/DlgControllerConfig.cpp

523 lines
19 KiB
C++
Raw Normal View History

2003-03-30 06:30:19 +00:00
// ******************************************************************
// *
// * .,-::::: .,:: .::::::::. .,:: .:
// * ,;;;'````' `;;;, .,;; ;;;'';;' `;;;, .,;;
// * [[[ '[[,,[[' [[[__[[\. '[[,,[['
// * $$$ Y$$$P $$""""Y$$ Y$$$P
// * `88bo,__,o, oP"``"Yo, _88o,,od8P oP"``"Yo,
// * "YUMMMMMP",m" "Mm,""YUMMMP" ,m" "Mm,
// *
// * Cxbx->Win32->Cxbx->DlgControllerConfig.cpp
// *
// * This file is part of the Cxbx project.
// *
// * Cxbx and Cxbe are free software; you can redistribute them
// * and/or modify them under the terms of the GNU General Public
// * License as published by the Free Software Foundation; either
// * version 2 of the license, or (at your option) any later version.
// *
// * 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 for more details.
// *
// * You should have recieved a copy of the GNU General Public License
// * along with this program; see the file COPYING.
// * If not, write to the Free Software Foundation, Inc.,
// * 59 Temple Place - Suite 330, Bostom, MA 02111-1307, USA.
// *
// * (c) 2002-2003 Aaron Robinson <caustik@caustik.com>
// *
// * All rights reserved
// *
// ******************************************************************
#include "DlgControllerConfig.h"
2003-04-02 02:23:30 +00:00
#include "ResCxbx.h"
#include "EmuShared.h"
2003-03-30 06:30:19 +00:00
#define DIRECTINPUT_VERSION 0x0800
#include <dinput.h>
2003-04-01 23:11:03 +00:00
#include <stdio.h>
2003-03-30 06:30:19 +00:00
// ******************************************************************
2003-04-02 22:57:26 +00:00
// * Exported Global(s)
2003-03-30 06:30:19 +00:00
// ******************************************************************
2003-04-01 23:11:03 +00:00
InputConfig g_InputConfig;
2003-03-30 06:30:19 +00:00
// ******************************************************************
2003-04-02 22:57:26 +00:00
// * Static Function(s)
2003-03-30 06:30:19 +00:00
// ******************************************************************
static INT_PTR CALLBACK DlgControllerConfigProc(HWND hWndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam);
static BOOL CALLBACK EnumGameCtrlCallback(LPCDIDEVICEINSTANCE lpddi, LPVOID pvRef);
static BOOL CALLBACK EnumObjectsCallback(LPCDIDEVICEOBJECTINSTANCE lpddoi, LPVOID pvRef);
static void ConfigureInput(HWND hWndDlg, HWND hWndButton, void (InputConfig::*MapFunc)(const char *, int, int));
2003-03-30 06:30:19 +00:00
2003-04-01 23:11:03 +00:00
// ******************************************************************
2003-04-02 22:57:26 +00:00
// * Static Variable(s)
2003-04-01 23:11:03 +00:00
// ******************************************************************
2003-04-02 22:57:26 +00:00
static LPDIRECTINPUT8 g_pDirectInput8 = NULL;
static LPDIRECTINPUTDEVICE8 g_pInputDev[MAX_INPUT_DEVICES] = {0};
static DWORD g_pInputDevFlags[MAX_INPUT_DEVICES] = {0};
static int g_pInputCur = 0;
static bool g_bConfigDone = true;
2003-04-01 23:11:03 +00:00
2003-03-30 06:30:19 +00:00
// ******************************************************************
2003-04-02 22:57:26 +00:00
// * Show Controller Configuration Dialog Window
2003-03-30 06:30:19 +00:00
// ******************************************************************
void ShowControllerConfig(HWND hwnd)
{
2003-04-02 22:57:26 +00:00
g_EmuShared->RetrieveInputConfiguration(&g_InputConfig);
2003-03-30 06:30:19 +00:00
DialogBox
(
GetModuleHandle(NULL),
MAKEINTRESOURCE(IDD_CONTROLLER_CFG),
hwnd,
DlgControllerConfigProc
);
}
// ******************************************************************
2003-04-02 22:57:26 +00:00
// * Controller Configuration Dialog Procedure
// ******************************************************************
INT_PTR CALLBACK DlgControllerConfigProc(HWND hWndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
2003-04-02 22:57:26 +00:00
{
switch(uMsg)
{
case WM_INITDIALOG:
SetClassLong(hWndDlg, GCL_HICON, (LONG)LoadIcon(GetModuleHandle(NULL), MAKEINTRESOURCE(IDI_CXBX)));
2003-04-02 20:01:37 +00:00
SetFocus(GetDlgItem(hWndDlg, IDC_SET_X));
break;
case WM_CLOSE:
EndDialog(hWndDlg, wParam);
break;
case WM_COMMAND:
{
HWND hWndButton = GetDlgItem(hWndDlg, LOWORD(wParam));
switch(LOWORD(wParam))
{
case IDC_INPUT_CONFIG_CANCEL:
EndDialog(hWndDlg, wParam);
break;
case IDC_INPUT_CONFIG_ACCEPT:
2003-04-01 08:16:06 +00:00
g_EmuShared->UpdateInputConfiguration(&g_InputConfig);
EndDialog(hWndDlg, wParam);
break;
case IDC_SET_LEFT_X:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapLThumbX);
break;
case IDC_SET_LEFT_Y:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapLThumbY);
break;
case IDC_SET_RIGHT_X:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapRThumbX);
break;
case IDC_SET_RIGHT_Y:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapRThumbY);
break;
case IDC_SET_X:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapX);
break;
case IDC_SET_Y:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapY);
break;
case IDC_SET_A:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapA);
break;
case IDC_SET_B:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapB);
break;
case IDC_SET_WHITE:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapWhite);
break;
case IDC_SET_BLACK:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapBlack);
break;
case IDC_SET_LTRIGGER:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapLTrigger);
break;
case IDC_SET_RTRIGGER:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapRTrigger);
break;
case IDC_SET_DPAD_UP:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapDPadUp);
break;
case IDC_SET_DPAD_DOWN:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapDPadDown);
break;
case IDC_SET_DPAD_LEFT:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapDPadLeft);
break;
case IDC_SET_DPAD_RIGHT:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapDPadRight);
break;
case IDC_SET_BACK:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapBack);
break;
case IDC_SET_START:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapStart);
break;
case IDC_SET_LTHUMB:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapLThumb);
break;
case IDC_SET_RTHUMB:
ConfigureInput(hWndDlg, hWndButton, InputConfig::MapRThumb);
break;
}
}
break;
}
return FALSE;
}
// ******************************************************************
2003-04-02 22:57:26 +00:00
// * Enumerate Game Controller(s)
// ******************************************************************
BOOL CALLBACK EnumGameCtrlCallback(LPCDIDEVICEINSTANCE lpddi, LPVOID pvRef)
{
HRESULT hRet = g_pDirectInput8->CreateDevice(lpddi->guidInstance, &g_pInputDev[g_pInputCur], NULL);
2003-04-02 22:57:26 +00:00
if(!FAILED(hRet))
{
g_pInputDevFlags[g_pInputCur] = INPUT_MAPPING_JOYSTICK;
g_pInputDev[g_pInputCur++]->SetDataFormat(&c_dfDIJoystick);
}
return DIENUM_CONTINUE;
}
// ******************************************************************
2003-04-02 22:57:26 +00:00
// * Enumerate Game Controller Object(s)
// ******************************************************************
BOOL CALLBACK EnumObjectsCallback(LPCDIDEVICEOBJECTINSTANCE lpddoi, LPVOID pvRef)
{
if(lpddoi->dwType & DIDFT_AXIS)
{
DIPROPRANGE diprg;
diprg.diph.dwSize = sizeof(DIPROPRANGE);
diprg.diph.dwHeaderSize = sizeof(DIPROPHEADER);
diprg.diph.dwHow = DIPH_BYID;
diprg.diph.dwObj = lpddoi->dwType;
2003-04-02 22:57:26 +00:00
diprg.lMin = 0 - 32768;
diprg.lMax = 0 + 32767;
// set axis range
HRESULT hRet = g_pInputDev[(int)pvRef]->SetProperty(DIPROP_RANGE, &diprg.diph);
if(FAILED(hRet))
return DIENUM_STOP;
}
return DIENUM_CONTINUE;
}
2003-03-30 06:30:19 +00:00
// ******************************************************************
// * ConfigureInput
// ******************************************************************
void ConfigureInput(HWND hWndDlg, HWND hWndButton, void (InputConfig::*MapFunc)(const char *, int, int))
2003-03-30 06:30:19 +00:00
{
if(!g_bConfigDone)
return;
2003-04-02 22:57:26 +00:00
2003-03-30 06:30:19 +00:00
g_bConfigDone = false;
char szOrgText[32];
char szNewText[255] = "Recieved no user input, try again...";
SetWindowText(GetDlgItem(hWndDlg, IDC_CONFIG_STATUS), "Waiting for your input...");
GetWindowText(hWndButton, szOrgText, 32);
// ******************************************************************
// * Create DirectInput object
// ******************************************************************
{
DirectInput8Create
(
GetModuleHandle(NULL),
DIRECTINPUT_VERSION,
IID_IDirectInput8,
(void**)&g_pDirectInput8,
NULL
);
}
2003-04-02 22:57:26 +00:00
// ******************************************************************
// * Create all the devices available (well...most of them)
// ******************************************************************
2003-03-30 06:30:19 +00:00
if(g_pDirectInput8 != 0)
2003-04-02 22:57:26 +00:00
{
HRESULT hRet;
hRet = g_pDirectInput8->EnumDevices
2003-03-30 06:30:19 +00:00
(
DI8DEVCLASS_GAMECTRL,
EnumGameCtrlCallback,
NULL,
DIEDFL_ATTACHEDONLY
);
2003-04-02 22:57:26 +00:00
hRet = g_pDirectInput8->CreateDevice(GUID_SysKeyboard, &g_pInputDev[g_pInputCur], NULL);
if(!FAILED(hRet))
{
g_pInputDevFlags[g_pInputCur] = INPUT_MAPPING_KEYBOARD;
g_pInputDev[g_pInputCur++]->SetDataFormat(&c_dfDIKeyboard);
}
hRet = g_pDirectInput8->CreateDevice(GUID_SysMouse, &g_pInputDev[g_pInputCur], NULL);
if(!FAILED(hRet))
{
g_pInputDevFlags[g_pInputCur] = INPUT_MAPPING_MOUSE;
g_pInputDev[g_pInputCur++]->SetDataFormat(&c_dfDIMouse2);
}
2003-03-30 06:30:19 +00:00
}
2003-04-02 22:57:26 +00:00
2003-03-30 06:30:19 +00:00
// ******************************************************************
// * Set cooperative level and acquire
// ******************************************************************
{
for(int v=g_pInputCur-1;v>=0;v--)
{
2003-04-02 22:57:26 +00:00
g_pInputDev[v]->SetCooperativeLevel(hWndDlg, DISCL_NONEXCLUSIVE | DISCL_FOREGROUND);
2003-03-30 06:30:19 +00:00
g_pInputDev[v]->Acquire();
HRESULT hRet = g_pInputDev[v]->Poll();
if(FAILED(hRet))
{
hRet = g_pInputDev[v]->Acquire();
while(hRet == DIERR_INPUTLOST)
hRet = g_pInputDev[v]->Acquire();
if(hRet != DIERR_INPUTLOST)
break;
}
}
}
// ******************************************************************
// * Enumerate Controller objects
// ******************************************************************
{
for(int v=0;v<g_pInputCur;v++)
g_pInputDev[v]->EnumObjects(EnumObjectsCallback, (LPVOID)v, DIDFT_ALL);
}
// ******************************************************************
// * Wait for input, or 5 second timeout
// ******************************************************************
{
DIDEVICEINSTANCE DeviceInstance;
DIDEVICEOBJECTINSTANCE ObjectInstance;
DeviceInstance.dwSize = sizeof(DIDEVICEINSTANCE);
ObjectInstance.dwSize = sizeof(DIDEVICEOBJECTINSTANCE);
2003-04-02 22:57:26 +00:00
for(int v=100;v>0 && !g_bConfigDone;v--)
2003-03-30 06:30:19 +00:00
{
// ******************************************************************
// * Update the button text every second
// ******************************************************************
2003-04-02 22:57:26 +00:00
if(v%20 == 0)
2003-03-30 06:30:19 +00:00
{
char szBuffer[255];
2003-04-02 22:57:26 +00:00
sprintf(szBuffer, "%d", (v+19)/20);
2003-03-30 06:30:19 +00:00
SetWindowText(hWndButton, szBuffer);
}
// ******************************************************************
// * Monitor for significant device state changes
// ******************************************************************
for(int v=g_pInputCur-1;v>=0 && !g_bConfigDone;v--)
{
HRESULT hRet = g_pInputDev[v]->Poll();
if(FAILED(hRet))
{
hRet = g_pInputDev[v]->Acquire();
while(hRet == DIERR_INPUTLOST)
hRet = g_pInputDev[v]->Acquire();
return;
}
DWORD dwHow = -1;
2003-04-02 22:57:26 +00:00
DWORD dwFlags = g_pInputDevFlags[v];
2003-03-30 06:30:19 +00:00
2003-04-02 22:57:26 +00:00
// ******************************************************************
// * Detect Joystick Input
// ******************************************************************
if(g_pInputDevFlags[v] & INPUT_MAPPING_JOYSTICK)
2003-03-30 06:30:19 +00:00
{
2003-04-02 22:57:26 +00:00
DIJOYSTATE InputState = {0};
g_pInputDev[v]->GetDeviceState(sizeof(DIJOYSTATE), &InputState);
2003-03-30 06:30:19 +00:00
int v=0;
2003-04-02 22:57:26 +00:00
if(abs(InputState.lX) > JOYSTICK_DETECT_SENSITIVITY)
{
dwHow = FIELD_OFFSET(DIJOYSTATE, lX);
2003-04-02 22:57:26 +00:00
dwFlags |= (InputState.lX > 0) ? INPUT_MAPPING_AXIS_POSITIVE : INPUT_MAPPING_AXIS_NEGATIVE;
}
2003-04-02 22:57:26 +00:00
else if(abs(InputState.lY) > JOYSTICK_DETECT_SENSITIVITY)
{
2003-03-30 06:30:19 +00:00
dwHow = FIELD_OFFSET(DIJOYSTATE, lY);
2003-04-02 22:57:26 +00:00
dwFlags |= (InputState.lY > 0) ? INPUT_MAPPING_AXIS_POSITIVE : INPUT_MAPPING_AXIS_NEGATIVE;
}
2003-04-02 22:57:26 +00:00
else if(abs(InputState.lZ) > JOYSTICK_DETECT_SENSITIVITY)
{
2003-03-30 06:30:19 +00:00
dwHow = FIELD_OFFSET(DIJOYSTATE, lZ);
2003-04-02 22:57:26 +00:00
dwFlags |= (InputState.lZ > 0) ? INPUT_MAPPING_AXIS_POSITIVE : INPUT_MAPPING_AXIS_NEGATIVE;
}
2003-04-02 22:57:26 +00:00
else if(abs(InputState.lRx) > JOYSTICK_DETECT_SENSITIVITY)
{
2003-03-30 06:30:19 +00:00
dwHow = FIELD_OFFSET(DIJOYSTATE, lRx);
2003-04-02 22:57:26 +00:00
dwFlags |= (InputState.lRx > 0) ? INPUT_MAPPING_AXIS_POSITIVE : INPUT_MAPPING_AXIS_NEGATIVE;
}
2003-04-02 22:57:26 +00:00
else if(abs(InputState.lRy) > JOYSTICK_DETECT_SENSITIVITY)
{
2003-03-30 06:30:19 +00:00
dwHow = FIELD_OFFSET(DIJOYSTATE, lRy);
2003-04-02 22:57:26 +00:00
dwFlags |= (InputState.lRy > 0) ? INPUT_MAPPING_AXIS_POSITIVE : INPUT_MAPPING_AXIS_NEGATIVE;
}
2003-04-02 22:57:26 +00:00
else if(abs(InputState.lRz) > JOYSTICK_DETECT_SENSITIVITY)
{
2003-03-30 06:30:19 +00:00
dwHow = FIELD_OFFSET(DIJOYSTATE, lRz);
2003-04-02 22:57:26 +00:00
dwFlags |= (InputState.lRz > 0) ? INPUT_MAPPING_AXIS_POSITIVE : INPUT_MAPPING_AXIS_NEGATIVE;
}
else for(v=0;v<2;v++)
2003-04-02 22:57:26 +00:00
if(abs(InputState.rglSlider[v]) > JOYSTICK_DETECT_SENSITIVITY)
dwHow = FIELD_OFFSET(DIJOYSTATE, rglSlider[v]);
else for(v=0;v<4;v++)
2003-04-02 22:57:26 +00:00
if(abs(InputState.rgdwPOV[v]) > POV_DETECT_SENSITIVITY)
dwHow = FIELD_OFFSET(DIJOYSTATE, rgdwPOV[v]);
else for(v=0;v<32;v++)
2003-04-02 22:57:26 +00:00
if(InputState.rgbButtons[v] > BUTTON_DETECT_SENSITIVITY)
dwHow = FIELD_OFFSET(DIJOYSTATE, rgbButtons[v]);
2003-04-02 22:57:26 +00:00
// ******************************************************************
// * Retrieve Object Info
// ******************************************************************
if(dwHow != -1)
{
g_pInputDev[v]->GetDeviceInfo(&DeviceInstance);
g_pInputDev[v]->GetObjectInfo(&ObjectInstance, dwHow, DIPH_BYOFFSET);
(g_InputConfig.*MapFunc)(DeviceInstance.tszInstanceName, ObjectInstance.dwType, dwFlags);
printf("Cxbx: Detected %s on %s (dwType : %.08X)\n", ObjectInstance.tszName, DeviceInstance.tszInstanceName, ObjectInstance.dwType);
sprintf(szNewText, "%s Successfully Mapped To %s On %s!", szOrgText, ObjectInstance.tszName, DeviceInstance.tszInstanceName);
}
}
// ******************************************************************
// * Detect Keyboard Input
// ******************************************************************
else if(g_pInputDevFlags[v] & INPUT_MAPPING_KEYBOARD)
{
BYTE InputState[256];
g_pInputDev[v]->GetDeviceState(256, InputState);
for(int v=0;v<256;v++)
{
if(InputState[v] != 0)
{
dwHow = v;
break;
}
}
if(dwHow != -1)
{
(g_InputConfig.*MapFunc)("SysKeyboard", dwHow, dwFlags);
printf("Cxbx: Detected Key %d on SysKeyboard\n", dwHow);
sprintf(szNewText, "%s Successfully Mapped To Key %d On %s!", szOrgText, dwHow, "SysKeyboard");
}
}
// ******************************************************************
// * Detect Mouse Input
// ******************************************************************
else if(g_pInputDevFlags[v] & INPUT_MAPPING_MOUSE)
{
DIMOUSESTATE2 InputState = {0};
g_pInputDev[v]->GetDeviceState(sizeof(DIMOUSESTATE2), &InputState);
for(int v=0;v<8;v++)
{
if(InputState.rgbButtons[v] & 0x80)
{
dwHow = v;
dwFlags &= INPUT_MAPPING_MOUSE_CLICK;
break;
}
}
if(dwHow != -1)
{
(g_InputConfig.*MapFunc)("SysMouse", dwHow, dwFlags);
printf("Cxbx: Detected Button %d on SysMouse\n", dwHow);
sprintf(szNewText, "%s Successfully Mapped To Button %d On %s!", szOrgText, dwHow, "SysMouse");
}
}
if(dwHow != -1)
g_bConfigDone = true;
2003-03-30 06:30:19 +00:00
}
2003-04-02 22:57:26 +00:00
Sleep(50);
2003-03-30 06:30:19 +00:00
}
}
2003-04-02 22:57:26 +00:00
// ******************************************************************
// * Cleanup Devices
// ******************************************************************
2003-03-30 06:30:19 +00:00
{
for(int v=g_pInputCur-1;v>=0;v--)
{
g_pInputDev[v]->Unacquire();
g_pInputDev[v]->Release();
g_pInputDev[v] = 0;
}
g_pInputCur = 0;
if(g_pDirectInput8 != 0)
{
g_pDirectInput8->Release();
g_pDirectInput8 = 0;
}
}
2003-04-02 22:57:26 +00:00
// ******************************************************************
// * Update Window
// ******************************************************************
{
SetWindowText(hWndButton, szOrgText);
SetWindowText(GetDlgItem(hWndDlg, IDC_CONFIG_STATUS), szNewText);
MSG Msg;
while(PeekMessage(&Msg, hWndDlg, WM_MOUSEFIRST, WM_MOUSELAST, PM_REMOVE));
while(PeekMessage(&Msg, hWndDlg, WM_KEYFIRST, WM_KEYLAST, PM_REMOVE));
}
2003-03-30 06:30:19 +00:00
g_bConfigDone = true;
}