/* * 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/debug.h" #include "wine/unicode.h" #include "objbase.h" #include "dinput_private.h" #include "device_private.h" #include "resource.h" WINE_DEFAULT_DEBUG_CHANNEL(dinput); typedef struct { int nobjects; IDirectInputDevice8W *lpdid; DIDEVICEINSTANCEW ddi; DIDEVICEOBJECTINSTANCEW ddo[256]; } DeviceData; typedef struct { int ndevices; DeviceData *devices; } DIDevicesData; typedef struct { IDirectInput8W *lpDI; LPDIACTIONFORMATW lpdiaf; LPDIACTIONFORMATW original_lpdiaf; DIDevicesData devices_data; int display_only; } 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 count_devices(LPCDIDEVICEINSTANCEW lpddi, IDirectInputDevice8W *lpdid, DWORD dwFlags, DWORD dwRemaining, LPVOID pvRef) { DIDevicesData *data = (DIDevicesData*) pvRef; data->ndevices++; return DIENUM_CONTINUE; } static BOOL CALLBACK collect_devices(LPCDIDEVICEINSTANCEW lpddi, IDirectInputDevice8W *lpdid, DWORD dwFlags, DWORD dwRemaining, LPVOID pvRef) { DIDevicesData *data = (DIDevicesData*) pvRef; DeviceData *device = &data->devices[data->ndevices]; device->lpdid = lpdid; device->ddi = *lpddi; IDirectInputDevice_AddRef(lpdid); device->nobjects = 0; IDirectInputDevice_EnumObjects(lpdid, collect_objects, (LPVOID) device, DIDFT_ALL); data->ndevices++; return DIENUM_CONTINUE; } /* * Listview utility functions */ static void init_listview_columns(HWND dialog) { HINSTANCE hinstance = (HINSTANCE) GetWindowLongPtrW(dialog, GWLP_HINSTANCE); LVCOLUMNW listColumn; RECT viewRect; int width; WCHAR column[MAX_PATH]; GetClientRect(GetDlgItem(dialog, IDC_DEVICEOBJECTSLIST), &viewRect); width = (viewRect.right - viewRect.left)/2; LoadStringW(hinstance, IDS_OBJECTCOLUMN, column, sizeof(column)/sizeof(column[0])); 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(hinstance, IDS_ACTIONCOLUMN, column, sizeof(column)/sizeof(column[0])); 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 LPDIACTIONFORMATW get_cur_lpdiaf(HWND dialog) { ConfigureDevicesData *data = (ConfigureDevicesData*) GetWindowLongPtrW(dialog, DWLP_USER); return data->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, IDirectInput8W *lpDI, DIDevicesData *data, LPDIACTIONFORMATW lpdiaf) { int i; /* Count devices */ data->ndevices = 0; IDirectInput8_EnumDevicesBySemantics(lpDI, NULL, lpdiaf, count_devices, (LPVOID) data, 0); /* Allocate devices */ data->devices = HeapAlloc(GetProcessHeap(), 0, sizeof(DeviceData) * data->ndevices); /* Collect and insert */ data->ndevices = 0; IDirectInput8_EnumDevicesBySemantics(lpDI, NULL, lpdiaf, collect_devices, (LPVOID) data, 0); for (i=0; i < data->ndevices; i++) SendDlgItemMessageW(dialog, IDC_CONTROLLERCOMBO, CB_ADDSTRING, 0, (LPARAM) data->devices[i].ddi.tszProductName ); } static void destroy_data(HWND dialog) { int i; 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); HeapFree(GetProcessHeap(), 0, devices_data->devices); /* Free the backup LPDIACTIONFORMATW */ HeapFree(GetProcessHeap(), 0, data->original_lpdiaf->rgoAction); HeapFree(GetProcessHeap(), 0, data->original_lpdiaf); } 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++) { 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); /* Search for an assigned action for this device */ for (j=0; j < lpdiaf->dwNumActions; j++) { if (IsEqualGUID(&lpdiaf->rgoAction[j].guidInstance, &device->ddi.guidInstance) && lpdiaf->rgoAction[j].dwObjID == device->ddo[i].dwType) { 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_cur_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; DIDEVICEOBJECTINSTANCEW ddo = device->ddo[obj]; if (old_action == action) return; /* 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 = ddo.dwType; 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 copy_actions(LPDIACTIONFORMATW to, LPDIACTIONFORMATW from) { int i; for (i=0; i < from->dwNumActions; i++) { to->rgoAction[i].guidInstance = from->rgoAction[i].guidInstance; to->rgoAction[i].dwObjID = from->rgoAction[i].dwObjID; to->rgoAction[i].dwHow = from->rgoAction[i].dwHow; to->rgoAction[i].u.lptszActionName = from->rgoAction[i].u.lptszActionName; } } static void reset_actions(HWND dialog) { ConfigureDevicesData *data = (ConfigureDevicesData*) GetWindowLongPtrW(dialog, DWLP_USER); LPDIACTIONFORMATW to = data->lpdiaf, from = data->original_lpdiaf; copy_actions(to, from); } 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->lpDI, &data->devices_data, data->lpdiaf); /* Store information in the window */ SetWindowLongPtrW(dialog, DWLP_USER, (LONG_PTR) data); init_listview_columns(dialog); /* Create a backup action format for CANCEL and RESET operations */ data->original_lpdiaf = HeapAlloc(GetProcessHeap(), 0, sizeof(*data->original_lpdiaf)); data->original_lpdiaf->dwNumActions = data->lpdiaf->dwNumActions; data->original_lpdiaf->rgoAction = HeapAlloc(GetProcessHeap(), 0, sizeof(DIACTIONW)*data->lpdiaf->dwNumActions); copy_actions(data->original_lpdiaf, data->lpdiaf); /* Select the first device and show its actions */ SendDlgItemMessageW(dialog, IDC_CONTROLLERCOMBO, CB_SETCURSEL, 0, 0); fill_device_object_list(dialog); 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: switch (HIWORD(wParam)) { case CBN_SELCHANGE: fill_device_object_list(dialog); break; } break; case IDOK: EndDialog(dialog, 0); destroy_data(dialog); break; case IDCANCEL: reset_actions(dialog); 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 ) { ConfigureDevicesData data; data.lpDI = iface; data.lpdiaf = lpdiCDParams->lprgFormats; data.display_only = !(dwFlags & DICD_EDIT); InitCommonControls(); DialogBoxParamW(GetModuleHandleA("dinput.dll"), (LPCWSTR) MAKEINTRESOURCE(IDD_CONFIGUREDEVICES), lpdiCDParams->hwnd, ConfigureDevicesDlgProc, (LPARAM) &data); return DI_OK; }