|
|
/*
File: add.cpp Project: Universal Joystick Control Panel OLE Client Author: Brycej Date: 14/Feb/97 Comments: window proc for add dialog
Copyright (c) 1995, Microsoft Corporation */
#include <afxcmn.h>
#include <windowsx.h>
#include <regstr.h> // for REGSTR_VAL_JOYOEMNAME reference!
#include "cpanel.h"
#include "joyarray.h"
#include <initguid.h>
#pragma warning(disable:4200)
#include <gameport.h>
#pragma warning(default:4200)
// table of default joystick configs
extern WCHAR *pwszTypeArray[MAX_DEVICES]; // List of enumerated gameport devices
extern WCHAR *pwszGameportBus[MAX_BUSSES]; // List of enumerated gameport buses
extern PJOY pAssigned[MAX_ASSIGNED]; // List of assigned devices
extern BYTE nAssigned; // Number of elements in pAssigned array
extern BYTE nTargetAssigned; // Number of elements expected when pending adds complete
extern BYTE nReEnum; // Counter used to decide when to reenumerate
extern GUID guidOccupied[MAX_BUSSES]; //Whether the gameport bus has been occupied.
extern BYTE nGameportBus, nGamingDevices;
extern IDirectInputJoyConfig *pDIJoyConfig; extern short iItem; extern short nFlags; // Flags set in CPANEL.CPP!
// extern functions defined in CPANEL.CPP
extern void OnContextMenu(WPARAM wParam, LPARAM lParam); extern void OnHelp (LPARAM);
// local message handlers
static BOOL OnInitDialog(HWND, HWND, LPARAM); static void OnClose(HWND); static void OnCommand(HWND, int, HWND, UINT);
static char AddSelectedItem( HWND hDlg ); static BOOL UpdateListCtrl ( HWND hCtrl ); static char GetNextAvailableID( void ); static BOOL IsTypeActive( short *nArrayIndex ); static BOOL GetNextAvailableVIDPID(LPWSTR lpwszType ); //static BOOL IsCustomType(BYTE nIndex);
static void PostHScrollBar(HWND hCtrl, LPCTSTR lpStr, BYTE nStrLen);
extern const DWORD g_aHelpIDs[];
extern HINSTANCE ghInstance;
LPTSTR lpszAutoDetect;
const int NO_ITEM = -1; #define MAX_ANALOG_BUTTONS 4
#define MAX_ANALOG_AXIS 4
// This will become a problem when we support > 500 ports!
#define AUTODETECT_PORT 100
// Consider achieving this from calculation of width of listview
static short iAddItem = NO_ITEM;
INT_PTR CALLBACK AddDialogProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch( uMsg ) { case WM_DESTROY: DestroyIcon((HICON)SendMessage(hDlg, WM_GETICON, (WPARAM)ICON_SMALL, 0));
if( nGameportBus > 1 ) if( lpszAutoDetect ) delete[] (lpszAutoDetect); break;
case WM_LBUTTONDOWN: // Click Drag service for PropSheets!
PostMessage(hDlg, WM_NCLBUTTONDOWN, (WPARAM)HTCAPTION, lParam); break;
case WM_INITDIALOG: return(BOOL)HANDLE_WM_INITDIALOG(hDlg, wParam, lParam, OnInitDialog);
case WM_COMMAND: HANDLE_WM_COMMAND(hDlg, wParam, lParam, OnCommand); return(1);
case WM_VKEYTOITEM: if( LOWORD(wParam) == VK_DELETE ) { HWND hWnd = GetDlgItem(hDlg, IDC_DEVICE_LIST);
// determine if it's a custom type... if so... delete it
// Get the next object with the SELECTED attribute
iAddItem = (short)::SendMessage(hWnd, LB_GETCURSEL, 0, 0);
short nArrayIndex = (short)::SendMessage(hWnd, LB_GETITEMDATA, (WPARAM)iAddItem, 0);
// test to make sure they aren't trying to delete a standard type
if( *pwszTypeArray[nArrayIndex] == L'#' ) break;
// test that the device is a custom type (user or IHV defined)
// ISSUE-2001/03/29-timgill the number of times we retrieve the same data is ridiculous
LPDIJOYTYPEINFO lpdiJoyInfo = new (DIJOYTYPEINFO); ASSERT (lpdiJoyInfo);
if( lpdiJoyInfo ) { /*
* warning, abuse of S_OK == 0 == strcmp( identical strings ) ahead */ HRESULT CmpRes;
ZeroMemory(lpdiJoyInfo, sizeof(*lpdiJoyInfo));
lpdiJoyInfo->dwSize = sizeof(*lpdiJoyInfo);
/*
* Test the hardware ID */ CmpRes = pDIJoyConfig->GetTypeInfo(pwszTypeArray[nArrayIndex], (LPDIJOYTYPEINFO)lpdiJoyInfo, DITC_HARDWAREID); if( SUCCEEDED( CmpRes ) ) { #ifndef WINNT
if( lpdiJoyInfo->wszHardwareId[0] == L'\0' ) { /*
* No hardware ID, so look for a callout VxD * If there is none, we have a custom type. */ CmpRes = pDIJoyConfig->GetTypeInfo(pwszTypeArray[nArrayIndex], (LPDIJOYTYPEINFO)lpdiJoyInfo, DITC_CALLOUT); CmpRes = ( SUCCEEDED( CmpRes ) && ( lpdiJoyInfo->wszCallout[0] == L'\0' ) ) ? S_OK : S_FALSE; } else #endif
{ const WCHAR wszAnalogRoot[] = L"gameport\\vid_045e&pid_01";
CmpRes = (HRESULT)_wcsnicmp( lpdiJoyInfo->wszHardwareId, wszAnalogRoot, (sizeof(wszAnalogRoot)/sizeof(wszAnalogRoot[0]) ) - 1 ); } } else { CmpRes = S_FALSE; }
// clean-up!
delete (lpdiJoyInfo);
if( CmpRes != S_OK ) { // This is not an analog type so leave alone
break; } } else { // If we can't test the type, do nothing
break; }
// test to make sure you are not deleting objects
if( IsTypeActive(&nArrayIndex) ) { Error((short)IDS_GEN_AREYOUSURE_TITLE, (short)IDS_NO_REMOVE); break; }
// This buffer has to be big enough for the name and the "are you sure..." message!
LPTSTR pszMsg = new TCHAR[MAX_STR_LEN+STR_LEN_64]; ASSERT (pszMsg);
LPTSTR pszTitle = new TCHAR[STR_LEN_128]; ASSERT (pszTitle);
// Query user if they are sure!
VERIFY(LoadString(ghInstance, IDS_GEN_AREYOUSURE, pszTitle, MAX_STR_LEN));
LPTSTR pszTmp = new TCHAR[(short)SendMessage(hWnd, LB_GETTEXTLEN, (WPARAM)iAddItem, 0)+1]; ASSERT (pszTmp);
SendMessage(hWnd, LB_GETTEXT, (WPARAM)iAddItem, (LPARAM)(LPCTSTR)pszTmp);
wsprintf( pszMsg, pszTitle, pszTmp);
if( pszTmp ) delete[] (pszTmp);
VERIFY(LoadString(ghInstance, IDS_GEN_AREYOUSURE_TITLE, pszTitle, STR_LEN_128));
HRESULT hr = MessageBox(hWnd, pszMsg, pszTitle, MB_ICONQUESTION | MB_YESNO | MB_APPLMODAL);
if( pszMsg ) delete[] (pszMsg); if( pszTitle ) delete[] (pszTitle);
if( IDYES == hr ) {
if( SUCCEEDED(hr = pDIJoyConfig->Acquire()) ) { // traverse the list and delete any configuration that may have this type assigned to it!
//DeleteAssignedType( pwszTypeArray[nArrayIndex] );
// returns E_ACCESSDENIED if it's a default item.
hr = pDIJoyConfig->DeleteType(pwszTypeArray[nArrayIndex]); ASSERT(SUCCEEDED(hr));
pDIJoyConfig->Unacquire();
// the dec is for the zero based device list!
// decrement the indexes
nGamingDevices--;
// don't move if you're on the last entry!
if( nArrayIndex != nGamingDevices ) { // To avoid deleting and recreating...
// Let's see if we have room to put this string!
BYTE nLen = (BYTE)wcslen(pwszTypeArray[nArrayIndex])+1;
// Make sure pwszTypeArray[nArrayIndex] is larger than pwszTypeArray[nCount]
if( nLen < (BYTE)wcslen(pwszTypeArray[nGamingDevices])+1 ) { if( pwszTypeArray[nArrayIndex] ) free(pwszTypeArray[nArrayIndex]);
pwszTypeArray[nArrayIndex] = _wcsdup(pwszTypeArray[nGamingDevices]); ASSERT (pwszTypeArray[nArrayIndex]); } // move the end element to the place of the deleted one
else wcsncpy(pwszTypeArray[nArrayIndex], pwszTypeArray[nGamingDevices], wcslen(pwszTypeArray[nGamingDevices])+1);
// update the extra memory to reflect the new index!
// First, find the item in the listbox and get the index...
LPDIJOYTYPEINFO_DX5 lpdiJoyInfo = new (DIJOYTYPEINFO_DX5); ASSERT (lpdiJoyInfo);
ZeroMemory(lpdiJoyInfo, sizeof(DIJOYTYPEINFO_DX5));
lpdiJoyInfo->dwSize = sizeof(DIJOYTYPEINFO_DX5);
if( SUCCEEDED(pDIJoyConfig->GetTypeInfo(pwszTypeArray[nArrayIndex], (LPDIJOYTYPEINFO)lpdiJoyInfo, DITC_DISPLAYNAME)) ) { char n = (char)SendMessage(hWnd, LB_FINDSTRINGEXACT, (WPARAM)-1, (LPARAM)lpdiJoyInfo->wszDisplayName); SendMessage(hWnd, LB_SETITEMDATA, (WPARAM)n, (LPARAM)nArrayIndex); }
// clean-up!
if( lpdiJoyInfo ) delete (lpdiJoyInfo); }
// delete the end element from the array
if( pwszTypeArray[nGamingDevices] ) { free(pwszTypeArray[nGamingDevices]); pwszTypeArray[nGamingDevices] = 0; }
// Remove the entry from the list control
SendMessage(hWnd, LB_DELETESTRING, (WPARAM)iAddItem, 0);
// Enable the buttons!
if( nGamingDevices < MAX_DEVICES-1 ) { HWND hParent = GetParent(hWnd); ASSERT (hParent);
PostDlgItemEnableWindow(hParent, IDC_CUSTOM, TRUE); PostDlgItemEnableWindow(hParent, IDC_ADD_NEW, TRUE); }
if( iAddItem != nGamingDevices-1 ) iAddItem--;
// Set the focus to the next available item
PostMessage(hWnd, LB_SETCURSEL, (WPARAM)(iAddItem == NO_ITEM) ? 0 : iAddItem, 0); } } break; } else return(-1);
// case WM_NOTIFY:
/// return HANDLE_WM_NOTIFY(hDlg, wParam, lParam, OnNotify);
case WM_HELP: OnHelp(lParam); return(1);
case WM_CONTEXTMENU: OnContextMenu(wParam, lParam); return(1); } return(0); }
BOOL OnInitDialog(HWND hDlg, HWND hWnd, LPARAM lParam) { HICON hIcon = (HICON)LoadImage(ghInstance, MAKEINTRESOURCE(IDI_CPANEL), IMAGE_ICON, 0, 0, LR_DEFAULTSIZE); ASSERT (hIcon);
if( hIcon ) ::PostMessage(hDlg, WM_SETICON, (WPARAM)ICON_SMALL, (LPARAM)hIcon);
if( nFlags & ON_NT ) DestroyWindow(GetDlgItem(hDlg, IDC_WDM)); else { HKEY hKey;
// remove the WDM flag unless otherwise specified!
if( RegOpenKey(HKEY_LOCAL_MACHINE, TEXT("SYSTEM\\CurrentControlSet\\Control\\MediaResources\\joystick\\<FixedKey>"), &hKey) == ERROR_SUCCESS ) { DWORD n; DWORD dwSize = sizeof (DWORD);
if( RegQueryValueEx(hKey, TEXT("UseWDM"), 0, 0, (LPBYTE)&n, &dwSize) == ERROR_SUCCESS ) { if( n ) DestroyWindow(GetDlgItem(hDlg, IDC_WDM)); } }
RegCloseKey(hKey); }
// a simple debug test to verify that the template hasn't gotten DIALOGEX style again!
HWND hCtrl = GetDlgItem(hDlg, IDC_DEVICE_LIST); ASSERT (hCtrl);
// Fill up the Device List
if( !UpdateListCtrl(hCtrl) ) { TRACE(TEXT("JOY.CPL: ADD.CPP: Failed UpdateListCtrl!\n")); return(FALSE); }
// If there is only one, don't confuse the user with the combo box
if( nGameportBus > 1 ) { // Allocate for AutoDetect Gameport!
lpszAutoDetect = new (TCHAR[STR_LEN_32]); ASSERT (lpszAutoDetect);
VERIFY(LoadString(ghInstance, IDS_AUTO_DETECT, lpszAutoDetect, STR_LEN_32));
hCtrl = GetDlgItem(hDlg, IDC_GAMEPORTLIST); ASSERT (hCtrl);
if( !PopulatePortList(hCtrl) ) { TRACE(TEXT("JOY.CPL: ADD.CPP: Failed PopulatePortList!\n")); return(FALSE); }
LPDIJOYTYPEINFO lpdiJoyInfo = new DIJOYTYPEINFO; ASSERT (lpdiJoyInfo);
ZeroMemory(lpdiJoyInfo, sizeof(DIJOYTYPEINFO));
lpdiJoyInfo->dwSize = sizeof(DIJOYTYPEINFO);
VERIFY(SUCCEEDED(pDIJoyConfig->GetTypeInfo(pwszTypeArray[0], (LPDIJOYTYPEINFO)lpdiJoyInfo, DITC_FLAGS1)));
// Search for AutoDetect in the list!
char nIndex = (char)::SendMessage(hCtrl, CB_FINDSTRINGEXACT, (WPARAM)-1, (LPARAM)(LPCTSTR)lpszAutoDetect);
// If it's got the JOYTYPE_NOAUTODETECTGAMEPORT flag, remove AutoDetect from the ListCtrl
if( lpdiJoyInfo->dwFlags1 & JOYTYPE_NOAUTODETECTGAMEPORT ) { // it could fail because the entry is not available...
if( nIndex != CB_ERR ) ::SendMessage(hCtrl, CB_DELETESTRING, (WPARAM)nIndex, 0);
} // Otherwise, verify that AutoDetect is present... if not, add it!
else { // it could fail because the entry is not available...
if( nIndex == CB_ERR ) ::SendMessage(hCtrl, CB_SETITEMDATA, ::SendMessage(hCtrl, CB_ADDSTRING, 0, (LPARAM)(LPCTSTR)lpszAutoDetect), AUTODETECT_PORT); }
if( lpdiJoyInfo ) delete (lpdiJoyInfo);
SetWindowPos( hCtrl, NULL, NULL, NULL, NULL, NULL, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_SHOWWINDOW);
SetWindowPos( GetDlgItem(hDlg, IDC_GAMEPORT), NULL, NULL, NULL, NULL, NULL, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_SHOWWINDOW); }
// blj: Warning Message that you can't add any more devices!
if( nGamingDevices == MAX_DEVICES-1 ) { // Disable the Custom Button!
PostDlgItemEnableWindow(hDlg, IDC_CUSTOM, FALSE); PostDlgItemEnableWindow(hDlg, IDC_ADD_NEW, FALSE);
// Give the user a error message!
Error((short)IDS_MAX_DEVICES_TITLE, (short)IDS_MAX_DEVICES_MSG); }
return(0); }
///////////////////////////////////////////////////////////////////////////////
// FUNCTION: OnCommand(HWND hDlg, int id, HWND hWndCtl, UINT code)
//
// PARAMETERS: hDlg -
// id -
// hWndCtl -
// code -
//
// PURPOSE: WM_COMMAND message handler
///////////////////////////////////////////////////////////////////////////////
void OnCommand(HWND hDlg, int id, HWND hWndCtl, UINT code) { switch( id ) { case IDC_ADD_NEW: if( SUCCEEDED(pDIJoyConfig->AddNewHardware(hDlg, GUID_MediaClass)) ) { ClearArrays();
if( FAILED(pDIJoyConfig->EnumTypes((LPDIJOYTYPECALLBACK)DIEnumJoyTypeProc, NULL)) ) { TRACE(TEXT("JOY.CPL: ADD.CPP: Failed BuildEnumList!\n")); return; }
// Warning Message that you can't add any more devices!
if( nGamingDevices == MAX_DEVICES-1 ) { // Disable the Custom Button!
PostDlgItemEnableWindow(hDlg, IDC_CUSTOM, FALSE); PostDlgItemEnableWindow(hDlg, IDC_ADD_NEW, FALSE);
// Give the user a error message!
Error((short)IDS_MAX_DEVICES_TITLE, (short)IDS_MAX_DEVICES_MSG); }
if( !UpdateListCtrl(GetDlgItem(hDlg, IDC_DEVICE_LIST)) ) { TRACE(TEXT("JOY.CPL: ADD.CPP: Failed to update the list control on the add page!\n")); return; } } break;
case IDC_CUSTOM: // DialogBox returns either IDCANCEL or the index into the list of
// defined types!
if( IDOK == DialogBox(ghInstance, (PTSTR)IDD_CUSTOM, hDlg, CustomDialogProc) ) { HWND hCtrl = GetDlgItem(hDlg, IDC_DEVICE_LIST);
// blj: Warning Message that you can't add any more devices!
if( nGamingDevices == MAX_DEVICES-1 ) { // Disable the Custom Button!
PostDlgItemEnableWindow(hDlg, IDC_CUSTOM, FALSE);
// Give the user a error message!
Error((short)IDS_MAX_DEVICES_TITLE, (short)IDS_MAX_DEVICES_MSG); }
// Now, put the focus on the newly created item!
LPDIJOYTYPEINFO_DX5 lpdiJoyInfo = new (DIJOYTYPEINFO_DX5); ASSERT (lpdiJoyInfo);
ZeroMemory(lpdiJoyInfo, sizeof(DIJOYTYPEINFO_DX5));
lpdiJoyInfo->dwSize = sizeof(DIJOYTYPEINFO_DX5);
// Get nGamingDevices from pwszTypeArray
// Subtract 1 from nGamingDevices because the list is 0 based!
if( SUCCEEDED(pDIJoyConfig->GetTypeInfo(pwszTypeArray[nGamingDevices-1], (LPDIJOYTYPEINFO)lpdiJoyInfo, DITC_DISPLAYNAME | DITC_REGHWSETTINGS)) ) { #ifdef _UNICODE
iAddItem = (short)SendMessage(hCtrl, LB_ADDSTRING, 0, (LPARAM)(LPCTSTR)lpdiJoyInfo->wszDisplayName); SendMessage(hCtrl, LB_SETITEMDATA, (WPARAM)iAddItem, (LPARAM)nGamingDevices-1); PostHScrollBar(hCtrl, lpdiJoyInfo->wszDisplayName, (BYTE)wcslen(lpdiJoyInfo->wszDisplayName)); #else
USES_CONVERSION; iAddItem = (short)SendMessage(hCtrl, LB_ADDSTRING, 0, (LPARAM)(LPCTSTR)W2A(lpdiJoyInfo->wszDisplayName)); SendMessage(hCtrl, LB_SETITEMDATA, (WPARAM)iAddItem, (LPARAM)nGamingDevices-1); PostHScrollBar(hCtrl, W2A(lpdiJoyInfo->wszDisplayName), (BYTE)wcslen(lpdiJoyInfo->wszDisplayName)); #endif
// SetFocus to that item!
SendMessage(hCtrl, LB_SETCURSEL, (WPARAM)iAddItem, 0);
OnCommand(hDlg, IDC_DEVICE_LIST, 0, LBN_SELCHANGE); }
// clean-up!
if( lpdiJoyInfo ) delete (lpdiJoyInfo); } break;
case IDC_DEVICE_LIST: if( code == LBN_SELCHANGE ) { iAddItem = (short)SendMessage(hWndCtl, LB_GETCURSEL, 0, 0);
BYTE nArrayID = (BYTE)SendMessage(hWndCtl, LB_GETITEMDATA, (WPARAM)iAddItem, 0);
LPDIJOYTYPEINFO lpdiJoyInfo = new DIJOYTYPEINFO; ASSERT (lpdiJoyInfo);
ZeroMemory(lpdiJoyInfo, sizeof(DIJOYTYPEINFO));
lpdiJoyInfo->dwSize = sizeof(DIJOYTYPEINFO);
DWORD dwFlags = DITC_REGHWSETTINGS;
if( nGameportBus > 1 ) dwFlags |= DITC_FLAGS1;
VERIFY(SUCCEEDED(pDIJoyConfig->GetTypeInfo(pwszTypeArray[nArrayID], (LPDIJOYTYPEINFO)lpdiJoyInfo, dwFlags))); /*
if (nGameportBus > 1) { HWND hCtrl = GetDlgItem(hDlg, IDC_GAMEPORTLIST);
// Search for AutoDetect in the list!
char nIndex = (char)::SendMessage(hCtrl, CB_FINDSTRINGEXACT, (WPARAM)-1, (LPARAM)(LPCTSTR)lpszAutoDetect);
// If it's got the JOYTYPE_NOAUTODETECTGAMEPORT flag, remove AutoDetect from the ListCtrl
if (lpdiJoyInfo->dwFlags1 & JOYTYPE_NOAUTODETECTGAMEPORT) { // it could fail because the entry is not available...
if (nIndex != CB_ERR) ::SendMessage(hCtrl, CB_DELETESTRING, (WPARAM)nIndex, 0); } // Otherwise, verify that AutoDetect is present... if not, add it!
else { // it could fail because the entry is not available...
if (nIndex == CB_ERR) ::SendMessage(hCtrl, CB_SETITEMDATA, ::SendMessage(hCtrl, CB_ADDSTRING, 0, (LPARAM)(LPCTSTR)lpszAutoDetect), AUTODETECT_PORT); } ::PostMessage(hCtrl, CB_SETCURSEL, (WPARAM)0, (LPARAM)0); } */
PostDlgItemEnableWindow(hDlg, IDC_JOY1HASRUDDER, (lpdiJoyInfo->hws.dwFlags & JOY_HWS_HASR) ? FALSE : TRUE);
if( lpdiJoyInfo ) delete (lpdiJoyInfo);
break; }
if( code != LBN_DBLCLK ) break;
case IDOK: iAddItem = (short)SendDlgItemMessage(hDlg, IDC_DEVICE_LIST, LB_GETCURSEL, 0, 0);
if( iAddItem == NO_ITEM ) break;
// Check to see if they have a GamePortList
{ HWND hCtrl = GetDlgItem(hDlg, IDC_GAMEPORTLIST); ASSERT (hCtrl);
if( IsWindowVisible(hCtrl) ) { // Check to see if the user has a port selected!
if( SendMessage(hCtrl, CB_GETCURSEL, 0, 0) == CB_ERR ) { // blj: TODO: Clear this message with UE
Error((short)IDS_NO_GAMEPORT_TITLE, (short)IDS_NO_GAMEPORT); break; } } }
iItem = 0; AddSelectedItem(hDlg);
case IDCANCEL: EndDialog(hDlg, id); break; } }
/// Custom Dialog procedure
INT_PTR CALLBACK CustomDialogProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { static LPWSTR lpwszVIDPID;
switch( uMsg ) { case WM_DESTROY: { HICON hIcon = (HICON)SendMessage(hDlg, WM_GETICON, (WPARAM)ICON_SMALL, 0); DestroyIcon(hIcon); } break;
case WM_LBUTTONDOWN: // Click Drag service for PropSheets!
PostMessage(hDlg, WM_NCLBUTTONDOWN, (WPARAM)HTCAPTION, lParam); break;
case WM_HELP: OnHelp(lParam); return(1);
case WM_CONTEXTMENU: OnContextMenu(wParam, lParam); return(TRUE);
case WM_INITDIALOG: // Set up the Buttons and Axis combo boxs!
{ HICON hIcon = (HICON)LoadImage(ghInstance, MAKEINTRESOURCE(IDI_CPANEL), IMAGE_ICON, 0, 0, LR_DEFAULTSIZE); ASSERT (hIcon);
if( hIcon ) ::PostMessage(hDlg, WM_SETICON, (WPARAM)ICON_SMALL, (LPARAM)hIcon);
// Return if no available VID/PIDs found!
lpwszVIDPID = new (WCHAR[STR_LEN_32]); ASSERT (lpwszVIDPID);
pDIJoyConfig->Acquire();
if( !GetNextAvailableVIDPID(lpwszVIDPID) ) { if( lpwszVIDPID ) delete[] (lpwszVIDPID);
EndDialog(hDlg, IDCANCEL);
// Let the user know that they need to remove some "Custom" devices!
Error((short)IDS_NO_NAME_TITLE, (short)IDS_NOAVAILABLEVIDPID );
return(FALSE); }
// init id list
BYTE nButtons = MAX_ANALOG_BUTTONS; TCHAR szTmp[3];
HWND hCtrl = GetDlgItem(hDlg, IDC_COMBO_BUTTONS);
// Fill the Buttons combo...
do { itoa(nButtons, (LPTSTR)&szTmp); SendMessage(hCtrl, CB_ADDSTRING, (WPARAM)0, (LPARAM)(LPCTSTR)szTmp); } while( nButtons-- );
// Set Default to four buttons
SendMessage(hCtrl, CB_SETCURSEL, MAX_ANALOG_BUTTONS, 0);
// Use nButtons for Axis...
nButtons = MAX_ANALOG_AXIS;
hCtrl = GetDlgItem(hDlg, IDC_COMBO_AXIS);
// Fill the Axis combo...
do { itoa(nButtons--, (LPTSTR)&szTmp); SendMessage(hCtrl, CB_ADDSTRING, (WPARAM)0, (LPARAM)(LPCTSTR)szTmp); } while( nButtons > 1 );
// Set Default to two Axis
SendMessage(hCtrl, CB_SETCURSEL, 0, 0); }
::PostMessage(GetDlgItem(hDlg, IDC_SPECIAL_JOYSTICK), BM_SETCHECK, BST_CHECKED, 0);
SendDlgItemMessage(hDlg, IDC_EDIT_NAME, EM_LIMITTEXT, (WPARAM)MAX_STR_LEN, 0); return(1);
case WM_COMMAND: switch( LOWORD(wParam) ) { case IDOK: { // defines for error states!
#define DUPLICATE_NAME 0x01
#define NO_NAME 0x02
#define INVALID_NAME 0x04
BYTE nLen = (BYTE)SendDlgItemMessage(hDlg, IDC_EDIT_NAME, EM_LINELENGTH, 0, 0);
LPTSTR pszTypeName = NULL;
if( nLen ) { pszTypeName = new (TCHAR[nLen+1]); ASSERT (pszTypeName);
// Get the Type/Display Name
GetDlgItemText(hDlg, IDC_EDIT_NAME, (LPTSTR)pszTypeName, nLen+1);
if( _tcschr(pszTypeName, TEXT('\\')) ) nLen = INVALID_NAME; else { // Make sure it's not a duplicate!
// start of fix #9269
LPDIJOYTYPEINFO_DX5 lpdiGetJoyInfo = new DIJOYTYPEINFO_DX5; ASSERT (lpdiGetJoyInfo);
ZeroMemory(lpdiGetJoyInfo, sizeof(DIJOYTYPEINFO_DX5));
lpdiGetJoyInfo->dwSize = sizeof(DIJOYTYPEINFO_DX5);
// Clean-up nLen!
BYTE n = nLen = 0;
// Search for a Duplicate Display Name!!!
while( pwszTypeArray[n] ) { if( FAILED(pDIJoyConfig->GetTypeInfo(pwszTypeArray[n++], (LPDIJOYTYPEINFO)lpdiGetJoyInfo, DITC_DISPLAYNAME)) ) { TRACE(TEXT("JOY.CPL: ADD.CPP: GetTypeInfo Failed!\n")); continue; } #ifndef _UNICODE
USES_CONVERSION; #endif
if( _tcsncmp(pszTypeName, #ifdef _UNICODE
lpdiGetJoyInfo->wszDisplayName, #else
W2A(lpdiGetJoyInfo->wszDisplayName), #endif
wcslen(lpdiGetJoyInfo->wszDisplayName)+1) == 0 ) { nLen = DUPLICATE_NAME; break; } }
if( lpdiGetJoyInfo ) { delete (lpdiGetJoyInfo); } // end of fix #9269
} } else nLen = NO_NAME;
// Check for an error!
if( nLen ) { if( pszTypeName ) delete[] (pszTypeName);
// Give the user the appropriate error!
switch( nLen ) { case DUPLICATE_NAME: Error((SHORT)IDS_NO_NAME_TITLE, (SHORT)IDS_DUPLICATE_TYPE); break;
case NO_NAME: Error((short)IDS_NO_NAME_TITLE, (short)IDS_NO_NAME); break;
case INVALID_NAME: Error((short)IDS_NO_NAME_TITLE, (short)IDS_INVALID_NAME); break; }
// Set Focus to the Dialog
SetFocus(hDlg);
HWND hCtrl = GetDlgItem(hDlg, IDC_EDIT_NAME); ASSERT (hCtrl);
// Set Focus to the Control
SetFocus(hCtrl);
// hilite the error
PostMessage(hCtrl, EM_SETSEL, (WPARAM)0, (LPARAM)-1);
return(FALSE); }
// set the type information
LPDIJOYTYPEINFO lpdiJoyInfo = new (DIJOYTYPEINFO); ASSERT (lpdiJoyInfo);
ZeroMemory(lpdiJoyInfo, sizeof(DIJOYTYPEINFO));
lpdiJoyInfo->dwSize = sizeof(DIJOYTYPEINFO);
#ifndef _UNICODE
USES_CONVERSION; #endif
// Set the Display Name
wcsncpy(lpdiJoyInfo->wszDisplayName, #ifdef _UNICODE
pszTypeName, #else
A2W(pszTypeName), #endif // _UNICODE
lstrlen(pszTypeName)+1);
if( pszTypeName ) delete[] (pszTypeName);
// Set the GUID - default to LegacyServer
// Per Marcus, we don't want to do this anymore!
//lpdiJoyInfo->clsidConfig = CLSID_LegacyServer;
// Set the Hardware Settings
lpdiJoyInfo->hws.dwNumButtons = (DWORD)SendDlgItemMessage(hDlg, IDC_COMBO_BUTTONS, CB_GETCURSEL, 0, 0);
switch( SendDlgItemMessage(hDlg, IDC_COMBO_AXIS, CB_GETCURSEL, 0, 0) ) { // R Axis
case 1: // Check to see which button got checked...
lpdiJoyInfo->hws.dwFlags |= ::SendMessage(GetDlgItem(hDlg, IDC_HASRUDDER), BM_GETCHECK, 0, 0) ? JOY_HWS_HASR : JOY_HWS_HASZ; break;
// Z Axis
case 2: lpdiJoyInfo->hws.dwFlags |= JOY_HWS_HASR; lpdiJoyInfo->hws.dwFlags |= JOY_HWS_HASZ; break;
// X/Y are default!
default: lpdiJoyInfo->hws.dwFlags = 0; break; }
lpdiJoyInfo->hws.dwFlags |= ::SendMessage(GetDlgItem(hDlg, IDS_CUSTOM_HASPOV), BM_GETCHECK, 0, 0) ? JOY_HWS_HASPOV : 0;
// Get Special char's status
lpdiJoyInfo->hws.dwFlags |= ::SendMessage(GetDlgItem(hDlg, IDC_SPECIAL_JOYSTICK), BM_GETCHECK, 0, 0) ? 0 : ::SendMessage(GetDlgItem(hDlg, IDC_SPECIAL_PAD), BM_GETCHECK, 0, 0) ? JOY_HWS_ISGAMEPAD : ::SendMessage(GetDlgItem(hDlg, IDC_SPECIAL_AUTO), BM_GETCHECK, 0, 0) ? JOY_HWS_ISCARCTRL : JOY_HWS_ISYOKE; // default to Yoke!
// Set up wszHardwareId
wcscpy(lpdiJoyInfo->wszHardwareId, L"GamePort\\");
// This blocks the Huge DisplayName bug!
// DINPUT change requires VID/PID at the end of the hardware ID... it was
//StrNCatW(lpdiJoyInfo->wszHardwareId, lpwszVIDPID, 245);
// Win95 does not like StrNCatW, we will use wcsncat
wcsncat(lpdiJoyInfo->wszHardwareId, lpwszVIDPID, 245);
if( SUCCEEDED(pDIJoyConfig->Acquire()) ) { #ifdef _UNICODE
if( FAILED(pDIJoyConfig->SetTypeInfo(lpwszVIDPID, lpdiJoyInfo, DITC_DISPLAYNAME | DITC_CLSIDCONFIG | DITC_REGHWSETTINGS | DITC_HARDWAREID)) ) #else
if( FAILED(pDIJoyConfig->SetTypeInfo(lpwszVIDPID, lpdiJoyInfo, DITC_DISPLAYNAME | DITC_CLSIDCONFIG | DITC_REGHWSETTINGS ) ) ) #endif
{ #ifdef _DEBUG
OutputDebugString(TEXT("JOY.CPL: ADD.CPP: CustomDlgProc: SetTypeInfo Failed!\n")); #endif
}
// Create the memory for the Custom device and stick it into the array!
pwszTypeArray[nGamingDevices++] = _wcsdup(lpwszVIDPID);
pDIJoyConfig->Unacquire(); }
if( lpdiJoyInfo ) delete (lpdiJoyInfo); }
case IDCANCEL: if( lpwszVIDPID ) delete[] (lpwszVIDPID);
EndDialog(hDlg, LOWORD(wParam)); break;
case IDC_SPECIAL_JOYSTICK: case IDC_SPECIAL_YOKE: case IDC_SPECIAL_PAD: case IDC_SPECIAL_AUTO: CheckRadioButton(hDlg, IDC_SPECIAL_JOYSTICK, IDC_SPECIAL_AUTO, IDC_SPECIAL_JOYSTICK); break;
case IDC_COMBO_AXIS: // Show/Hide IDC_HASRUDDER based on Selection of 3 Axis
if( HIWORD(wParam) == CBN_SELCHANGE ) { const USHORT nCtrlArray[] = {IDC_HASRUDDER, IDC_HASZAXIS}; BYTE nCtrls = sizeof(nCtrlArray)/sizeof(short);
// the '1' in the comparison is because the CB is Zero based!
BOOL bShow = (BOOL)(SendDlgItemMessage(hDlg, IDC_COMBO_AXIS, CB_GETCURSEL, 0, 0) == 1);
do { SetWindowPos( GetDlgItem(hDlg, nCtrlArray[--nCtrls]), NULL, NULL, NULL, NULL, NULL, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | ((bShow) ? SWP_SHOWWINDOW : SWP_HIDEWINDOW)); } while( nCtrls );
if( bShow ) ::PostMessage(GetDlgItem(hDlg, IDC_HASZAXIS), BM_SETCHECK, BST_CHECKED, 0); } break; } return(1);
case WM_CLOSE: EndDialog(hDlg, 0); return(1); } return(0); }
char GetNextAvailableID( void ) { LPDIJOYCONFIG_DX5 pJoyConfig = new (DIJOYCONFIG_DX5); ASSERT (pJoyConfig);
ZeroMemory(pJoyConfig, sizeof(DIJOYCONFIG_DX5));
pJoyConfig->dwSize = sizeof (DIJOYCONFIG_DX5);
char i = 0;
do { switch( pDIJoyConfig->GetConfig(i, (LPDIJOYCONFIG)pJoyConfig, DIJC_REGHWCONFIGTYPE) ) { case S_FALSE: case DIERR_NOMOREITEMS: case DIERR_NOTFOUND: case E_FAIL: goto EXIT;
default: i++; break; } } while( i < NUMJOYDEVS );
i = -1;
// And it's Error time!
Error((short)IDS_NO_IDS_TITLE, (short)IDS_NO_IDS);
EXIT: if( pJoyConfig ) delete (pJoyConfig);
return(i); }
///////////////////////////////////////////////////////////////////////////////
// FUNCTION: AddSelectedItem(HWND hDlg)
//
// PARAMETERS: hDlg - Handle to page
//
// PURPOSE: Adds selected item from List box.
///////////////////////////////////////////////////////////////////////////////
char AddSelectedItem( HWND hDlg ) { static BYTE n; DWORD dwFlags; HRESULT hr; int nID; #ifdef SUPPORT_TWO_2A2B
BOOL f2_2A2B = FALSE; #endif
nID = GetNextAvailableID();
// GetNextAvailableID returns -1 if it fails!
if( nID < 0 ) return((char)nID);
// Type info
LPDIJOYTYPEINFO_DX5 lpdiJoyInfo = new (DIJOYTYPEINFO_DX5); ASSERT (lpdiJoyInfo);
ZeroMemory(lpdiJoyInfo, sizeof(DIJOYTYPEINFO_DX5));
lpdiJoyInfo->dwSize = sizeof(DIJOYTYPEINFO_DX5);
BYTE nArrayID = (BYTE)SendDlgItemMessage(hDlg, IDC_DEVICE_LIST, LB_GETITEMDATA, (WPARAM)iAddItem, 0);
VERIFY(SUCCEEDED(pDIJoyConfig->GetTypeInfo(pwszTypeArray[nArrayID], (LPDIJOYTYPEINFO)lpdiJoyInfo, DITC_REGHWSETTINGS | DITC_CALLOUT | DITC_DISPLAYNAME)));
#ifdef SUPPORT_TWO_2A2B
#ifndef _UNICODE
if( wcscmp( pwszTypeArray[nArrayID], L"#<" ) == 0 ) { f2_2A2B = TRUE; lpdiJoyInfo->hws.dwFlags = 0; pwszTypeArray[nArrayID][1] = L'2'; } else { f2_2A2B = FALSE; } #endif
#endif
LPDIJOYCONFIG pTempJoyConfig = new DIJOYCONFIG; ASSERT (pTempJoyConfig);
ZeroMemory(pTempJoyConfig, sizeof(DIJOYCONFIG));
pTempJoyConfig->dwSize = sizeof (DIJOYCONFIG);
pTempJoyConfig->hwc.hws = lpdiJoyInfo->hws; pTempJoyConfig->hwc.hws.dwFlags |= JOY_HWS_ISANALOGPORTDRIVER;
// Do the Rudder Flags!
if( ::SendMessage(GetDlgItem(hDlg, IDC_JOY1HASRUDDER), BM_GETCHECK, 0, 0) ) { pTempJoyConfig->hwc.hws.dwFlags |= JOY_HWS_HASR; pTempJoyConfig->hwc.dwUsageSettings |= JOY_US_HASRUDDER; }
// set default to present
pTempJoyConfig->hwc.dwUsageSettings |= JOY_US_PRESENT;
pTempJoyConfig->hwc.dwType = nArrayID;
wcsncpy(pTempJoyConfig->wszCallout, lpdiJoyInfo->wszCallout, wcslen(lpdiJoyInfo->wszCallout)+1);
wcsncpy(pTempJoyConfig->wszType, pwszTypeArray[nArrayID], wcslen(pwszTypeArray[nArrayID])+1);
LPWSTR lpszPortName = NULL;
if( SUCCEEDED(pDIJoyConfig->Acquire()) ) { // This stops Memphis and gameport-less systems!!!
if( nGameportBus ) {
// no point asking the Combo box if there's really no choice!
if( nGameportBus > 1 ) { n = (BYTE)SendDlgItemMessage(hDlg, IDC_GAMEPORTLIST, CB_GETITEMDATA, (WPARAM)SendDlgItemMessage(hDlg, IDC_GAMEPORTLIST, CB_GETCURSEL, 0, 0), 0); } else { n = 0; }
if( n == AUTODETECT_PORT ) { #ifdef _DEBUG
OutputDebugString(TEXT("JOY.CPL: ADD.CPP: Selected Port is AutoDetect!\n")); #endif
pTempJoyConfig->guidGameport = GUID_GAMEENUM_BUS_ENUMERATOR; } else { // Zero the memory because the buffer still contains the old data!!!
ZeroMemory(lpdiJoyInfo, sizeof(DIJOYTYPEINFO_DX5)); lpdiJoyInfo->dwSize = sizeof(DIJOYTYPEINFO_DX5);
#ifdef _DEBUG
TRACE(TEXT("JOY.CPL: ADD.CPP: Port List index is %d or %s!\n"), n, pwszGameportBus[n]); #endif
if( SUCCEEDED(pDIJoyConfig->GetTypeInfo(pwszGameportBus[n], (LPDIJOYTYPEINFO)lpdiJoyInfo, DITC_CLSIDCONFIG | DITC_DISPLAYNAME)) ) { pTempJoyConfig->guidGameport = lpdiJoyInfo->clsidConfig; lpszPortName = _wcsdup(lpdiJoyInfo->wszDisplayName); } } }
// This is for Memphis and odd case NT systems!
if( pTempJoyConfig->guidGameport == GUID_NULL ) { pTempJoyConfig->guidGameport = GUID_GAMEENUM_BUS_ENUMERATOR; #ifdef _DEBUG
OutputDebugString(TEXT("JOY.CPL: ADD.CPP: Selected Port did not return a clsidConfig so AutoDetect is being used!\n")); #endif
}
// Set the hour glass
SetCursor(LoadCursor(NULL, IDC_WAIT));
// ISSUE-2001/03/29-timgill (MarcAnd) Why would we want to block updates when we're adding a device?
#if 1
// Unblock the WM_DEVICECHANGE message handler!
nFlags &= ~BLOCK_UPDATE; nFlags |= ON_PAGE;
/*
* Set the nReEnum counter going so that for the next n WM_TIMER * messages (or until the device arrives) we can consider * doing a refresh. * The value is somewhat arbitrary. */ nReEnum = 43; /*
* Set a target for the number of devices so that the extra * reenumerations can be avoided if this target is reached. */ nTargetAssigned = nAssigned+1; #else
nFlags |= BLOCK_UPDATE; #endif
dwFlags = DIJC_REGHWCONFIGTYPE | DIJC_CALLOUT; dwFlags |= ::SendDlgItemMessage(hDlg, IDC_WDM, BM_GETCHECK, 0, 0) ? DIJC_WDMGAMEPORT : 0;
if( FAILED(hr = pDIJoyConfig->SetConfig(nID, pTempJoyConfig, dwFlags)) ) { // Let the user know what happend!
if( hr == E_ACCESSDENIED ) { // Let the use know that the port is already occupied and that they need to remove that device or
// re-assign the device to an unoccupied port.
LPDIJOYCONFIG lpJoyCfg = new (DIJOYCONFIG); ASSERT (lpJoyCfg);
ZeroMemory(lpJoyCfg, sizeof(DIJOYCONFIG));
lpJoyCfg->dwSize = sizeof(DIJOYCONFIG);
// Cycle threw pAssigned and find the device with the same port name!
BYTE nIndex = nAssigned;
do { if( SUCCEEDED(pDIJoyConfig->GetConfig(pAssigned[--nIndex]->ID, lpJoyCfg, DIJC_WDMGAMEPORT)) ) { if( lpJoyCfg->guidGameport == pTempJoyConfig->guidGameport ) break; } } while( nIndex );
if( lpJoyCfg ) delete (lpJoyCfg);
DIPROPSTRING *pDIPropStr = new (DIPROPSTRING); ASSERT (pDIPropStr);
ZeroMemory(pDIPropStr, sizeof(DIPROPSTRING));
pDIPropStr->diph.dwSize = sizeof(DIPROPSTRING); pDIPropStr->diph.dwHeaderSize = sizeof(DIPROPHEADER); pDIPropStr->diph.dwHow = DIPH_DEVICE;
// Ok now.. you found it... use the Device pointer to get it's Friendly name!
if( SUCCEEDED(pAssigned[nIndex]->fnDeviceInterface->GetProperty(DIPROP_INSTANCENAME, &pDIPropStr->diph)) ) { // Put ellipse in text to avoid buffer over-flow.
// Limit displayed name to 50 chars... not completely arbitrary,
// we need to leave room in the Message string...
// who knows how long the string will get when translated!
if( wcslen(pDIPropStr->wsz) > 50 ) { pDIPropStr->wsz[47] = pDIPropStr->wsz[48] = pDIPropStr->wsz[49] = L'.'; pDIPropStr->wsz[50] = L'\0'; }
LPTSTR lptszMsgFormat = new (TCHAR[MAX_STR_LEN]); ASSERT (lptszMsgFormat); VERIFY(LoadString(ghInstance, IDS_ADD_PORT_MSGFORMAT, lptszMsgFormat, MAX_STR_LEN));
LPTSTR lptszMsg = new (TCHAR[MAX_STR_LEN]); ASSERT (lptszMsg);
// Format the message
wsprintf(lptszMsg, lptszMsgFormat, pDIPropStr->wsz, lpszPortName, pAssigned[nIndex]->ID+1);
VERIFY(LoadString(ghInstance, IDS_ADD_PORT_OCCUPIED, lptszMsgFormat, MAX_STR_LEN));
MessageBox(hDlg, lptszMsg, lptszMsgFormat, MB_ICONHAND | MB_OK | MB_APPLMODAL);
if( lptszMsgFormat ) delete[] (lptszMsgFormat);
if( lptszMsg ) delete[] (lptszMsg); }
if( pDIPropStr ) delete (pDIPropStr); } else if( hr == DIERR_DEVICEFULL ) { Error((short)IDS_GAMEPORT_OCCUPIED_TITLE, (short)IDS_GAMEPORT_OCCUPIED); } else { // Something Ugly happened!
Error((short)IDS_NO_GAMENUM_TITLE, (short)IDS_NO_GAMENUM); } } else { #ifdef _UNICODE
// Fix #55524
if( SUCCEEDED(pDIJoyConfig->GetConfig(nID, pTempJoyConfig, DIJC_REGHWCONFIGTYPE)) ) { if( !(pTempJoyConfig->hwc.dwUsageSettings & JOY_US_PRESENT) ) { pTempJoyConfig->hwc.dwUsageSettings |= JOY_US_PRESENT; pTempJoyConfig->hwc.hwv.dwCalFlags |= 0x80000000; pTempJoyConfig->hwc.hws.dwFlags |= JOY_HWS_ISANALOGPORTDRIVER;
VERIFY(SUCCEEDED(pDIJoyConfig->SetConfig(nID, pTempJoyConfig, DIJC_REGHWCONFIGTYPE))); } } // end of Fix #55524
#endif
// make sure VJOYD is notified
if( !(nFlags & ON_NT) ) { pDIJoyConfig->SendNotify(); Sleep(10); pDIJoyConfig->SendNotify(); }
#ifndef _UNICODE
#ifdef SUPPORT_TWO_2A2B
/*
* Add the other part of Two_2Axis_2Button joystick. */ if( f2_2A2B ) { nID = GetNextAvailableID(); if( nID >= 0 ){ hr = pDIJoyConfig->SetConfig(nID, pTempJoyConfig, dwFlags);
if( SUCCEEDED(hr) ) { if( !(nFlags & ON_NT) ) pDIJoyConfig->SendNotify(); } } } #endif
#endif
}
if( lpszPortName ) free(lpszPortName);
// Set the standard pointer
SetCursor(LoadCursor(NULL, IDC_ARROW));
pDIJoyConfig->Unacquire(); }
if( lpdiJoyInfo ) delete (lpdiJoyInfo);
if( pTempJoyConfig ) delete (pTempJoyConfig);
return((char)nID); }
BOOL UpdateListCtrl( HWND hCtrl ) { // Turn Redraw off here else it will flicker!
::SendMessage(hCtrl, WM_SETREDRAW, (WPARAM)FALSE, 0);
// delete all existing entries
::SendMessage(hCtrl, LB_RESETCONTENT, 0, 0);
// Type info
LPDIJOYTYPEINFO_DX5 lpdiJoyInfo = new (DIJOYTYPEINFO_DX5); ASSERT (lpdiJoyInfo);
ZeroMemory(lpdiJoyInfo, sizeof(DIJOYTYPEINFO_DX5));
lpdiJoyInfo->dwSize = sizeof(DIJOYTYPEINFO_DX5);
#ifndef _UNICODE
USES_CONVERSION; #endif
BYTE nIndex = nGamingDevices-1;
::SendMessage(hCtrl, LB_SETCOUNT, (WPARAM)(int)nIndex, 0);
LPWSTR lpStr = new WCHAR[MAX_STR_LEN]; ASSERT (lpStr);
BYTE nLargestStringLen = 0;
do { if( SUCCEEDED(pDIJoyConfig->GetTypeInfo(pwszTypeArray[nIndex], (LPDIJOYTYPEINFO)lpdiJoyInfo, DITC_DISPLAYNAME)) ) { #ifdef _UNICODE
::SendMessage(hCtrl, LB_SETITEMDATA, (WPARAM)::SendMessage(hCtrl, LB_ADDSTRING, 0, (LPARAM)(LPCTSTR)lpdiJoyInfo->wszDisplayName), (LPARAM)nIndex); #else
::SendMessage(hCtrl, LB_SETITEMDATA, (WPARAM)::SendMessage(hCtrl, LB_ADDSTRING, 0, (LPARAM)(LPCTSTR)W2A(lpdiJoyInfo->wszDisplayName)), (LPARAM)nIndex); #endif
if( wcslen(lpdiJoyInfo->wszDisplayName) > nLargestStringLen ) { nLargestStringLen = (BYTE)wcslen(lpdiJoyInfo->wszDisplayName); wcscpy(lpStr, lpdiJoyInfo->wszDisplayName); } } } while( nIndex-- );
if( lpdiJoyInfo ) delete (lpdiJoyInfo);
#ifdef _UNICODE
PostHScrollBar(hCtrl, lpStr, nLargestStringLen); #else
PostHScrollBar(hCtrl, W2A(lpStr), nLargestStringLen); #endif
if( lpStr ) delete[] (lpStr);
// Select the default selection to the 0th device type
iAddItem = 0;
::PostMessage(hCtrl, LB_SETCURSEL, (WPARAM)iAddItem, 0);
// Turn the redraw flag back on!
::SendMessage (hCtrl, WM_SETREDRAW, (WPARAM)TRUE, 0); InvalidateRect(hCtrl, NULL, TRUE); return(TRUE); }
// Please check to see if nGameportBus is > 0 before a call to this function!
// It will work, but What a waste!
BOOL PopulatePortList( HWND hCtrl ) { if( !::IsWindow(hCtrl) ) { TRACE(TEXT("JOY.CPL: PopulatePortList: HWND passed to PopulatePortList is NOT a valid Window!\n")); return(FALSE); }
SendMessage(hCtrl, CB_SETEXTENDEDUI, TRUE, 0);
// temp so we don't damage the global!
BYTE n = nGameportBus;
LPDIJOYTYPEINFO lpDIJoyTypeInfo = new (DIJOYTYPEINFO); ASSERT(lpDIJoyTypeInfo);
#ifndef _UNICODE
USES_CONVERSION; #endif
lpDIJoyTypeInfo->dwSize = sizeof(DIJOYTYPEINFO);
// Populate the list as they were enumerated.
do { if( FAILED(pDIJoyConfig->GetTypeInfo(pwszGameportBus[--n], lpDIJoyTypeInfo, DITC_DISPLAYNAME)) ) { TRACE(TEXT("JOY.CPL: ADD.CPP: GetTypeInfo failed!\n")); continue; }
// put the name in the list and place the index of the port array
// in it's item data...
SendMessage(hCtrl, CB_SETITEMDATA, #ifdef _UNICODE
SendMessage(hCtrl, CB_ADDSTRING, 0, (LPARAM)(LPCTSTR)lpDIJoyTypeInfo->wszDisplayName), n); #else
SendMessage(hCtrl, CB_ADDSTRING, 0, (LPARAM)(LPCSTR)W2A(lpDIJoyTypeInfo->wszDisplayName)), n); #endif
// Just in case you're curious, we're going backwards so the default
// port is the lowest available one.
} while( n );
// Don't forget Auto-Detect!
// But only if there's more than one port!
::SendMessage(hCtrl, CB_SETITEMDATA, ::SendMessage(hCtrl, CB_ADDSTRING, 0, (LPARAM)(LPCTSTR)lpszAutoDetect), AUTODETECT_PORT);
// While you're at it, check to see if one is available...
// if so, put it as the default!
::PostMessage(hCtrl, CB_SETCURSEL, (WPARAM)0, (LPARAM)0);
if( lpDIJoyTypeInfo ) delete (lpDIJoyTypeInfo);
return(TRUE); }
static BOOL IsTypeActive( short *nArrayIndex ) { if( *nArrayIndex > nGamingDevices ) { #ifdef _DEBUG
OutputDebugString(TEXT("JOY.CPL: IsTypeActive: nArrayIndex > nGamingDevices!\n")); #endif
return(FALSE); }
BOOL bRet = FALSE;
BYTE nIndex = nAssigned; BYTE nStrLen = (BYTE)wcslen(pwszTypeArray[*nArrayIndex])+1;
LPDIJOYCONFIG_DX5 lpDIJoyConfig = new (DIJOYCONFIG_DX5); ASSERT (lpDIJoyConfig);
ZeroMemory(lpDIJoyConfig, sizeof(DIJOYCONFIG_DX5));
lpDIJoyConfig->dwSize = sizeof (DIJOYCONFIG_DX5);
// find the assigned ID's
while( nIndex-- ) { if( SUCCEEDED(pDIJoyConfig->GetConfig(pAssigned[nIndex]->ID, (LPDIJOYCONFIG)lpDIJoyConfig, DIJC_REGHWCONFIGTYPE)) ) { // if you find it... break out!
if( wcsncmp(lpDIJoyConfig->wszType, pwszTypeArray[*nArrayIndex], nStrLen) == 0 ) { bRet = TRUE; *nArrayIndex = nIndex; break; } } }
if( lpDIJoyConfig ) delete (lpDIJoyConfig);
return(bRet); }
BOOL GetNextAvailableVIDPID(LPWSTR lpwszType ) { // Make the VID/PID to compare from the following formula
// VID_045e&PID_100+JOY_HW_LASTENTRY to 100+JOY_HW_LASTENTRY+0xf
HKEY hKey; BYTE n = JOY_HW_LASTENTRY;
wcsncpy(lpwszType, L"VID_045E&PID_0100", 18);
const WCHAR wszLookup[] = L"0123456789ABCDEF";
do { if( n < 0x10 ) { lpwszType[16] = wszLookup[n]; } else { lpwszType[15] = wszLookup[1]; lpwszType[16] = wszLookup[n%0x10]; }
n++;
if( FAILED(pDIJoyConfig->OpenTypeKey(lpwszType, KEY_READ, &hKey)) ) break;
RegCloseKey(hKey);
} while( n < (JOY_HW_LASTENTRY+0x11) );
return(BOOL)(n < 0x1d); }
//
// PostDlgItemEnableWindow(HWND hDlg, USHORT nItem, BOOL bEnabled)
//
void PostDlgItemEnableWindow(HWND hDlg, USHORT nItem, BOOL bEnabled) { HWND hCtrl = GetDlgItem(hDlg, nItem);
if( hCtrl ) PostEnableWindow(hCtrl, bEnabled); }
//
// PostEnableWindow(HWND hCtrl, BOOL bEnabled)
//
void PostEnableWindow(HWND hCtrl, BOOL bEnabled) { DWORD dwStyle = GetWindowLong(hCtrl, GWL_STYLE);
// No point Redrawing the Window if there's no change!
if( bEnabled ) { if( dwStyle & WS_DISABLED ) dwStyle &= ~WS_DISABLED; else return; } else { if( !(dwStyle & WS_DISABLED) ) dwStyle |= WS_DISABLED; else return; }
SetWindowLong(hCtrl, GWL_STYLE, dwStyle);
RedrawWindow(hCtrl, NULL, NULL, RDW_INTERNALPAINT | RDW_INVALIDATE); }
/*
BOOL IsCustomType(BYTE nIndex) { BOOL bRet = FALSE;
// First verify VID is 045E
WCHAR *pwStr = StrStrW(pwszTypeArray[nIndex],L"VID_"); if (pwStr) { // Increment the pointer over the VID_ and test for 045E
pwStr = &pwStr[4];
if (_wcsnicmp(pwStr, L"045e", 4) == 0) { OutputDebugString(TEXT("Hit \n"));
// Now, increment the pointer over 045e and &PID_ and test for the range!
pwStr = &pwStr[9];
// Second, verify PID is between 0x100 + JOY_HW_LASTENTRY and
// 0x100 + JOY_HW_LASTENTRY + 0xf
bRet = TRUE; } }
return bRet; } */
void PostHScrollBar(HWND hCtrl, LPCTSTR lpStr, BYTE nStrLen) { SIZE sz; HDC hDC = GetWindowDC(hCtrl); if( hDC != NULL ) /* PREFIX MANBUGS: 29336*/ { GetTextExtentPoint32(hDC, lpStr, nStrLen, &sz); ReleaseDC(hCtrl, hDC); ::PostMessage(hCtrl, LB_SETHORIZONTALEXTENT, (WPARAM)sz.cx, 0); } }
|