mirror of
https://github.com/reactos/reactos.git
synced 2024-11-20 06:15:26 +00:00
544 lines
18 KiB
C
544 lines
18 KiB
C
/*
|
|
* Copyright (c) 2011 Lucas Fialho Zawacki
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#define NONAMELESSUNION
|
|
|
|
|
|
#include "wine/unicode.h"
|
|
#include "objbase.h"
|
|
#include "dinput_private.h"
|
|
#include "device_private.h"
|
|
#include "resource.h"
|
|
|
|
#include "wine/heap.h"
|
|
|
|
typedef struct {
|
|
int nobjects;
|
|
IDirectInputDevice8W *lpdid;
|
|
DIDEVICEINSTANCEW ddi;
|
|
DIDEVICEOBJECTINSTANCEW ddo[256];
|
|
/* ActionFormat for every user.
|
|
* In same order as ConfigureDevicesData usernames */
|
|
DIACTIONFORMATW *user_afs;
|
|
} DeviceData;
|
|
|
|
typedef struct {
|
|
int ndevices;
|
|
DeviceData *devices;
|
|
} DIDevicesData;
|
|
|
|
typedef struct {
|
|
IDirectInput8W *lpDI;
|
|
LPDIACTIONFORMATW original_lpdiaf;
|
|
DIDevicesData devices_data;
|
|
int display_only;
|
|
int nusernames;
|
|
WCHAR **usernames;
|
|
} ConfigureDevicesData;
|
|
|
|
/*
|
|
* Enumeration callback functions
|
|
*/
|
|
static BOOL CALLBACK collect_objects(LPCDIDEVICEOBJECTINSTANCEW lpddo, LPVOID pvRef)
|
|
{
|
|
DeviceData *data = (DeviceData*) pvRef;
|
|
|
|
data->ddo[data->nobjects] = *lpddo;
|
|
|
|
data->nobjects++;
|
|
return DIENUM_CONTINUE;
|
|
}
|
|
|
|
static BOOL CALLBACK collect_devices(LPCDIDEVICEINSTANCEW lpddi, IDirectInputDevice8W *lpdid, DWORD dwFlags, DWORD dwRemaining, LPVOID pvRef)
|
|
{
|
|
ConfigureDevicesData *data = (ConfigureDevicesData*) pvRef;
|
|
DeviceData *device;
|
|
int i, j;
|
|
|
|
IDirectInputDevice_AddRef(lpdid);
|
|
|
|
/* alloc array for devices if this is our first device */
|
|
if (!data->devices_data.ndevices)
|
|
data->devices_data.devices = HeapAlloc(GetProcessHeap(), 0, sizeof(DeviceData) * (dwRemaining + 1));
|
|
device = &data->devices_data.devices[data->devices_data.ndevices];
|
|
device->lpdid = lpdid;
|
|
device->ddi = *lpddi;
|
|
|
|
device->nobjects = 0;
|
|
IDirectInputDevice_EnumObjects(lpdid, collect_objects, (LPVOID) device, DIDFT_ALL);
|
|
|
|
device->user_afs = heap_alloc(sizeof(*device->user_afs) * data->nusernames);
|
|
memset(device->user_afs, 0, sizeof(*device->user_afs) * data->nusernames);
|
|
for (i = 0; i < data->nusernames; i++)
|
|
{
|
|
DIACTIONFORMATW *user_af = &device->user_afs[i];
|
|
user_af->dwNumActions = data->original_lpdiaf->dwNumActions;
|
|
user_af->guidActionMap = data->original_lpdiaf->guidActionMap;
|
|
user_af->rgoAction = heap_alloc(sizeof(DIACTIONW) * data->original_lpdiaf->dwNumActions);
|
|
memset(user_af->rgoAction, 0, sizeof(DIACTIONW) * data->original_lpdiaf->dwNumActions);
|
|
for (j = 0; j < user_af->dwNumActions; j++)
|
|
{
|
|
user_af->rgoAction[j].dwSemantic = data->original_lpdiaf->rgoAction[j].dwSemantic;
|
|
user_af->rgoAction[j].dwFlags = data->original_lpdiaf->rgoAction[j].dwFlags;
|
|
user_af->rgoAction[j].u.lptszActionName = data->original_lpdiaf->rgoAction[j].u.lptszActionName;
|
|
}
|
|
IDirectInputDevice8_BuildActionMap(lpdid, user_af, data->usernames[i], 0);
|
|
}
|
|
|
|
data->devices_data.ndevices++;
|
|
return DIENUM_CONTINUE;
|
|
}
|
|
|
|
/*
|
|
* Listview utility functions
|
|
*/
|
|
static void init_listview_columns(HWND dialog)
|
|
{
|
|
LVCOLUMNW listColumn;
|
|
RECT viewRect;
|
|
int width;
|
|
WCHAR column[MAX_PATH];
|
|
|
|
GetClientRect(GetDlgItem(dialog, IDC_DEVICEOBJECTSLIST), &viewRect);
|
|
width = (viewRect.right - viewRect.left)/2;
|
|
|
|
LoadStringW(DINPUT_instance, IDS_OBJECTCOLUMN, column, ARRAY_SIZE(column));
|
|
listColumn.mask = LVCF_TEXT | LVCF_WIDTH | LVCF_SUBITEM;
|
|
listColumn.pszText = column;
|
|
listColumn.cchTextMax = lstrlenW(listColumn.pszText);
|
|
listColumn.cx = width;
|
|
|
|
SendDlgItemMessageW (dialog, IDC_DEVICEOBJECTSLIST, LVM_INSERTCOLUMNW, 0, (LPARAM) &listColumn);
|
|
|
|
LoadStringW(DINPUT_instance, IDS_ACTIONCOLUMN, column, ARRAY_SIZE(column));
|
|
listColumn.cx = width;
|
|
listColumn.pszText = column;
|
|
listColumn.cchTextMax = lstrlenW(listColumn.pszText);
|
|
|
|
SendDlgItemMessageW(dialog, IDC_DEVICEOBJECTSLIST, LVM_INSERTCOLUMNW, 1, (LPARAM) &listColumn);
|
|
}
|
|
|
|
static int lv_get_cur_item(HWND dialog)
|
|
{
|
|
return SendDlgItemMessageW(dialog, IDC_DEVICEOBJECTSLIST, LVM_GETNEXTITEM, -1, LVNI_SELECTED);
|
|
}
|
|
|
|
static int lv_get_item_data(HWND dialog, int index)
|
|
{
|
|
LVITEMW item;
|
|
|
|
if (index < 0) return -1;
|
|
|
|
item.mask = LVIF_PARAM;
|
|
item.iItem = index;
|
|
item.iSubItem = 0;
|
|
|
|
SendDlgItemMessageW(dialog, IDC_DEVICEOBJECTSLIST, LVM_GETITEMW , 0, (LPARAM)&item);
|
|
|
|
return item.lParam;
|
|
}
|
|
|
|
static void lv_set_action(HWND dialog, int item, int action, LPDIACTIONFORMATW lpdiaf)
|
|
{
|
|
static const WCHAR no_action[] = {'-','\0'};
|
|
const WCHAR *action_text = no_action;
|
|
LVITEMW lvItem;
|
|
|
|
if (item < 0) return;
|
|
|
|
if (action != -1)
|
|
action_text = lpdiaf->rgoAction[action].u.lptszActionName;
|
|
|
|
/* Keep the action and text in the listview item */
|
|
lvItem.iItem = item;
|
|
|
|
lvItem.mask = LVIF_PARAM;
|
|
lvItem.iSubItem = 0;
|
|
lvItem.lParam = (LPARAM) action;
|
|
|
|
/* Action index */
|
|
SendDlgItemMessageW(dialog, IDC_DEVICEOBJECTSLIST, LVM_SETITEMW, 0, (LPARAM) &lvItem);
|
|
|
|
lvItem.mask = LVIF_TEXT;
|
|
lvItem.iSubItem = 1;
|
|
lvItem.pszText = (WCHAR *)action_text;
|
|
lvItem.cchTextMax = lstrlenW(lvItem.pszText);
|
|
|
|
/* Text */
|
|
SendDlgItemMessageW(dialog, IDC_DEVICEOBJECTSLIST, LVM_SETITEMW, 0, (LPARAM) &lvItem);
|
|
}
|
|
|
|
/*
|
|
* Utility functions
|
|
*/
|
|
static DeviceData* get_cur_device(HWND dialog)
|
|
{
|
|
ConfigureDevicesData *data = (ConfigureDevicesData*) GetWindowLongPtrW(dialog, DWLP_USER);
|
|
int sel = SendDlgItemMessageW(dialog, IDC_CONTROLLERCOMBO, CB_GETCURSEL, 0, 0);
|
|
return &data->devices_data.devices[sel];
|
|
}
|
|
|
|
static DIACTIONFORMATW *get_cur_lpdiaf(HWND dialog)
|
|
{
|
|
ConfigureDevicesData *data = (ConfigureDevicesData*) GetWindowLongPtrW(dialog, DWLP_USER);
|
|
int controller_sel = SendDlgItemMessageW(dialog, IDC_CONTROLLERCOMBO, CB_GETCURSEL, 0, 0);
|
|
int player_sel = SendDlgItemMessageW(dialog, IDC_PLAYERCOMBO, CB_GETCURSEL, 0, 0);
|
|
return &data->devices_data.devices[controller_sel].user_afs[player_sel];
|
|
}
|
|
|
|
static DIACTIONFORMATW *get_original_lpdiaf(HWND dialog)
|
|
{
|
|
ConfigureDevicesData *data = (ConfigureDevicesData*) GetWindowLongPtrW(dialog, DWLP_USER);
|
|
return data->original_lpdiaf;
|
|
}
|
|
|
|
static int dialog_display_only(HWND dialog)
|
|
{
|
|
ConfigureDevicesData *data = (ConfigureDevicesData*) GetWindowLongPtrW(dialog, DWLP_USER);
|
|
return data->display_only;
|
|
}
|
|
|
|
static void init_devices(HWND dialog, ConfigureDevicesData *data)
|
|
{
|
|
int i;
|
|
|
|
/* Collect and insert */
|
|
data->devices_data.ndevices = 0;
|
|
IDirectInput8_EnumDevicesBySemantics(data->lpDI, NULL, data->original_lpdiaf, collect_devices, (LPVOID) data, 0);
|
|
|
|
for (i = 0; i < data->devices_data.ndevices; i++)
|
|
SendDlgItemMessageW(dialog, IDC_CONTROLLERCOMBO, CB_ADDSTRING, 0, (LPARAM) data->devices_data.devices[i].ddi.tszProductName );
|
|
for (i = 0; i < data->nusernames; i++)
|
|
SendDlgItemMessageW(dialog, IDC_PLAYERCOMBO, CB_ADDSTRING, 0, (LPARAM) data->usernames[i]);
|
|
}
|
|
|
|
static void destroy_data(HWND dialog)
|
|
{
|
|
int i, j;
|
|
ConfigureDevicesData *data = (ConfigureDevicesData*) GetWindowLongPtrW(dialog, DWLP_USER);
|
|
DIDevicesData *devices_data = &data->devices_data;
|
|
|
|
/* Free the devices */
|
|
for (i=0; i < devices_data->ndevices; i++)
|
|
{
|
|
IDirectInputDevice8_Release(devices_data->devices[i].lpdid);
|
|
for (j=0; j < data->nusernames; j++)
|
|
heap_free(devices_data->devices[i].user_afs[j].rgoAction);
|
|
heap_free(devices_data->devices[i].user_afs);
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, devices_data->devices);
|
|
}
|
|
|
|
static void fill_device_object_list(HWND dialog)
|
|
{
|
|
DeviceData *device = get_cur_device(dialog);
|
|
LPDIACTIONFORMATW lpdiaf = get_cur_lpdiaf(dialog);
|
|
LVITEMW item;
|
|
int i, j;
|
|
|
|
/* Clean the listview */
|
|
SendDlgItemMessageW(dialog, IDC_DEVICEOBJECTSLIST, LVM_DELETEALLITEMS, 0, 0);
|
|
|
|
/* Add each object */
|
|
for (i=0; i < device->nobjects; i++)
|
|
{
|
|
DWORD ddo_inst, ddo_type;
|
|
int action = -1;
|
|
|
|
item.mask = LVIF_TEXT | LVIF_PARAM;
|
|
item.iItem = i;
|
|
item.iSubItem = 0;
|
|
item.pszText = device->ddo[i].tszName;
|
|
item.cchTextMax = lstrlenW(item.pszText);
|
|
|
|
/* Add the item */
|
|
SendDlgItemMessageW(dialog, IDC_DEVICEOBJECTSLIST, LVM_INSERTITEMW, 0, (LPARAM) &item);
|
|
ddo_inst = DIDFT_GETINSTANCE(device->ddo[i].dwType);
|
|
ddo_type = DIDFT_GETTYPE(device->ddo[i].dwType);
|
|
|
|
/* Search for an assigned action for this device */
|
|
for (j=0; j < lpdiaf->dwNumActions; j++)
|
|
{
|
|
DWORD af_inst = DIDFT_GETINSTANCE(lpdiaf->rgoAction[j].dwObjID);
|
|
DWORD af_type = DIDFT_GETTYPE(lpdiaf->rgoAction[j].dwObjID);
|
|
if (af_type == DIDFT_PSHBUTTON) af_type = DIDFT_BUTTON;
|
|
if (af_type == DIDFT_RELAXIS) af_type = DIDFT_AXIS;
|
|
/* NOTE previously compared dwType == dwObjId but default buildActionMap actions
|
|
* were PSHBUTTON and RELAXS and didnt show up on config */
|
|
if (IsEqualGUID(&lpdiaf->rgoAction[j].guidInstance, &device->ddi.guidInstance) &&
|
|
ddo_inst == af_inst && ddo_type & af_type)
|
|
{
|
|
action = j;
|
|
break;
|
|
}
|
|
}
|
|
|
|
lv_set_action(dialog, i, action, lpdiaf);
|
|
}
|
|
}
|
|
|
|
static void show_suitable_actions(HWND dialog)
|
|
{
|
|
DeviceData *device = get_cur_device(dialog);
|
|
LPDIACTIONFORMATW lpdiaf = get_original_lpdiaf(dialog);
|
|
int i, added = 0;
|
|
int obj = lv_get_cur_item(dialog);
|
|
|
|
if (obj < 0) return;
|
|
|
|
SendDlgItemMessageW(dialog, IDC_ACTIONLIST, LB_RESETCONTENT, 0, 0);
|
|
|
|
for (i=0; i < lpdiaf->dwNumActions; i++)
|
|
{
|
|
/* Skip keyboard actions for non keyboards */
|
|
if (GET_DIDEVICE_TYPE(device->ddi.dwDevType) != DI8DEVTYPE_KEYBOARD &&
|
|
(lpdiaf->rgoAction[i].dwSemantic & DIKEYBOARD_MASK) == DIKEYBOARD_MASK) continue;
|
|
|
|
/* Skip mouse actions for non mouses */
|
|
if (GET_DIDEVICE_TYPE(device->ddi.dwDevType) != DI8DEVTYPE_MOUSE &&
|
|
(lpdiaf->rgoAction[i].dwSemantic & DIMOUSE_MASK) == DIMOUSE_MASK) continue;
|
|
|
|
/* Add action string and index in the action format to the list entry */
|
|
if (DIDFT_GETINSTANCE(lpdiaf->rgoAction[i].dwSemantic) & DIDFT_GETTYPE(device->ddo[obj].dwType))
|
|
{
|
|
SendDlgItemMessageW(dialog, IDC_ACTIONLIST, LB_ADDSTRING, 0, (LPARAM)lpdiaf->rgoAction[i].u.lptszActionName);
|
|
SendDlgItemMessageW(dialog, IDC_ACTIONLIST, LB_SETITEMDATA, added, (LPARAM) i);
|
|
added++;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void assign_action(HWND dialog)
|
|
{
|
|
DeviceData *device = get_cur_device(dialog);
|
|
LPDIACTIONFORMATW lpdiaf = get_cur_lpdiaf(dialog);
|
|
LVFINDINFOW lvFind;
|
|
int sel = SendDlgItemMessageW(dialog, IDC_ACTIONLIST, LB_GETCURSEL, 0, 0);
|
|
int action = SendDlgItemMessageW(dialog, IDC_ACTIONLIST, LB_GETITEMDATA, sel, 0);
|
|
int obj = lv_get_cur_item(dialog);
|
|
int old_action = lv_get_item_data(dialog, obj);
|
|
int used_obj;
|
|
DWORD type;
|
|
|
|
if (old_action == action) return;
|
|
if (obj < 0) return;
|
|
if (lpdiaf->rgoAction[old_action].dwFlags & DIA_APPFIXED) return;
|
|
|
|
type = device->ddo[obj].dwType;
|
|
|
|
/* Clear old action */
|
|
if (old_action != -1)
|
|
{
|
|
lpdiaf->rgoAction[old_action].dwObjID = 0;
|
|
lpdiaf->rgoAction[old_action].guidInstance = GUID_NULL;
|
|
lpdiaf->rgoAction[old_action].dwHow = DIAH_UNMAPPED;
|
|
}
|
|
|
|
/* Find if action text is already set for other object and unset it */
|
|
lvFind.flags = LVFI_PARAM;
|
|
lvFind.lParam = action;
|
|
|
|
used_obj = SendDlgItemMessageW(dialog, IDC_DEVICEOBJECTSLIST, LVM_FINDITEMW, -1, (LPARAM) &lvFind);
|
|
|
|
lv_set_action(dialog, used_obj, -1, lpdiaf);
|
|
|
|
/* Set new action */
|
|
lpdiaf->rgoAction[action].dwObjID = type;
|
|
lpdiaf->rgoAction[action].guidInstance = device->ddi.guidInstance;
|
|
lpdiaf->rgoAction[action].dwHow = DIAH_USERCONFIG;
|
|
|
|
/* Set new action in the list */
|
|
lv_set_action(dialog, obj, action, lpdiaf);
|
|
}
|
|
|
|
static void reset_actions(HWND dialog)
|
|
{
|
|
ConfigureDevicesData *data = (ConfigureDevicesData*) GetWindowLongPtrW(dialog, DWLP_USER);
|
|
DIDevicesData *ddata = (DIDevicesData*) &data->devices_data;
|
|
unsigned i, j;
|
|
|
|
for (i = 0; i < data->devices_data.ndevices; i++)
|
|
{
|
|
DeviceData *device = &ddata->devices[i];
|
|
for (j = 0; j < data->nusernames; j++)
|
|
IDirectInputDevice8_BuildActionMap(device->lpdid, &device->user_afs[j], data->usernames[j], DIDBAM_HWDEFAULTS);
|
|
}
|
|
}
|
|
|
|
static void save_actions(HWND dialog) {
|
|
ConfigureDevicesData *data = (ConfigureDevicesData*) GetWindowLongPtrW(dialog, DWLP_USER);
|
|
DIDevicesData *ddata = (DIDevicesData*) &data->devices_data;
|
|
unsigned i, j;
|
|
if (!data->display_only) {
|
|
for (i = 0; i < ddata->ndevices; i++)
|
|
{
|
|
DeviceData *device = &ddata->devices[i];
|
|
for (j = 0; j < data->nusernames; j++)
|
|
{
|
|
if (save_mapping_settings(device->lpdid, &device->user_afs[j], data->usernames[j]) != DI_OK)
|
|
MessageBoxA(dialog, "Could not save settings", 0, MB_ICONERROR);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static INT_PTR CALLBACK ConfigureDevicesDlgProc(HWND dialog, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
switch(uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
ConfigureDevicesData *data = (ConfigureDevicesData*) lParam;
|
|
|
|
/* Initialize action format and enumerate devices */
|
|
init_devices(dialog, data);
|
|
|
|
/* Store information in the window */
|
|
SetWindowLongPtrW(dialog, DWLP_USER, (LONG_PTR) data);
|
|
|
|
init_listview_columns(dialog);
|
|
|
|
/* Select the first device and show its actions */
|
|
SendDlgItemMessageW(dialog, IDC_CONTROLLERCOMBO, CB_SETCURSEL, 0, 0);
|
|
SendDlgItemMessageW(dialog, IDC_PLAYERCOMBO, CB_SETCURSEL, 0, 0);
|
|
fill_device_object_list(dialog);
|
|
|
|
ShowCursor(TRUE);
|
|
|
|
break;
|
|
}
|
|
|
|
case WM_DESTROY:
|
|
ShowCursor(FALSE);
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
|
|
switch (((LPNMHDR)lParam)->code)
|
|
{
|
|
case LVN_ITEMCHANGED:
|
|
show_suitable_actions(dialog);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
|
|
case WM_COMMAND:
|
|
|
|
switch(LOWORD(wParam))
|
|
{
|
|
|
|
case IDC_ACTIONLIST:
|
|
|
|
switch (HIWORD(wParam))
|
|
{
|
|
case LBN_DBLCLK:
|
|
/* Ignore this if app did not ask for editing */
|
|
if (dialog_display_only(dialog)) break;
|
|
|
|
assign_action(dialog);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case IDC_CONTROLLERCOMBO:
|
|
case IDC_PLAYERCOMBO:
|
|
|
|
switch (HIWORD(wParam))
|
|
{
|
|
case CBN_SELCHANGE:
|
|
fill_device_object_list(dialog);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case IDOK:
|
|
save_actions(dialog);
|
|
EndDialog(dialog, 0);
|
|
destroy_data(dialog);
|
|
break;
|
|
|
|
case IDCANCEL:
|
|
EndDialog(dialog, 0);
|
|
destroy_data(dialog);
|
|
break;
|
|
|
|
case IDC_RESET:
|
|
reset_actions(dialog);
|
|
fill_device_object_list(dialog);
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
HRESULT _configure_devices(IDirectInput8W *iface,
|
|
LPDICONFIGUREDEVICESCALLBACK lpdiCallback,
|
|
LPDICONFIGUREDEVICESPARAMSW lpdiCDParams,
|
|
DWORD dwFlags,
|
|
LPVOID pvRefData
|
|
)
|
|
{
|
|
int i;
|
|
DWORD size;
|
|
WCHAR *username = NULL;
|
|
ConfigureDevicesData data;
|
|
data.lpDI = iface;
|
|
data.original_lpdiaf = lpdiCDParams->lprgFormats;
|
|
data.display_only = !(dwFlags & DICD_EDIT);
|
|
data.nusernames = lpdiCDParams->dwcUsers;
|
|
if (lpdiCDParams->lptszUserNames == NULL)
|
|
{
|
|
/* Get default user name */
|
|
GetUserNameW(NULL, &size);
|
|
username = heap_alloc(size * sizeof(WCHAR) );
|
|
GetUserNameW(username, &size);
|
|
data.nusernames = 1;
|
|
data.usernames = heap_alloc(sizeof(WCHAR *));
|
|
data.usernames[0] = username;
|
|
}
|
|
else
|
|
{
|
|
WCHAR *p = lpdiCDParams->lptszUserNames;
|
|
data.usernames = heap_alloc(sizeof(WCHAR *) * data.nusernames);
|
|
for (i = 0; i < data.nusernames; i++)
|
|
{
|
|
if (*p)
|
|
{
|
|
data.usernames[i] = p;
|
|
while (*(p++));
|
|
}
|
|
else
|
|
/* Return if there is an empty string */
|
|
return DIERR_INVALIDPARAM;
|
|
}
|
|
}
|
|
|
|
InitCommonControls();
|
|
|
|
DialogBoxParamW(DINPUT_instance, (const WCHAR *)MAKEINTRESOURCE(IDD_CONFIGUREDEVICES),
|
|
lpdiCDParams->hwnd, ConfigureDevicesDlgProc, (LPARAM)&data);
|
|
|
|
heap_free(username);
|
|
heap_free(data.usernames);
|
|
|
|
return DI_OK;
|
|
}
|