2 * Audio management UI code
4 * Copyright 2004 Chris Morgan
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #define WIN32_LEAN_AND_MEAN
23 #define NONAMELESSSTRUCT
24 #define NONAMELESSUNION
27 #include "wine/port.h"
35 #include <wine/debug.h>
48 WINE_DEFAULT_DEBUG_CHANNEL(winecfg
);
50 #define DRIVER_MASK 0x80000000
51 #define DEVICE_MASK 0x40000000
53 typedef DWORD (WINAPI
* MessagePtr
)(UINT
, UINT
, DWORD
, DWORD
, DWORD
);
55 static struct DSOUNDACCEL
58 const char* settingStr
;
59 } const DSound_HW_Accels
[] = {
60 {IDS_ACCEL_FULL
, "Full"},
61 {IDS_ACCEL_STANDARD
, "Standard"},
62 {IDS_ACCEL_BASIC
, "Basic"},
63 {IDS_ACCEL_EMULATION
, "Emulation"},
67 static const char* DSound_Rates
[] = {
77 static const char* DSound_Bits
[] = {
83 static const AUDIO_DRIVER sAudioDrivers
[] = {
84 {IDS_DRIVER_ALSA
, "alsa"},
85 #if 0 /* arts library is buggy */
86 {IDS_DRIVER_ARTS
, "arts"},
88 {IDS_DRIVER_ESOUND
, "esd"},
89 {IDS_DRIVER_OSS
, "oss"},
90 {IDS_DRIVER_JACK
, "jack"},
91 {IDS_DRIVER_NAS
, "nas"},
92 {IDS_DRIVER_AUDIOIO
, "audioio"},
93 {IDS_DRIVER_COREAUDIO
, "coreaudio"},
97 /* list of available drivers */
98 static AUDIO_DRIVER
* loadedAudioDrv
;
100 /* local copy of registry setting */
101 static char curAudioDriver
[1024];
103 /* driver index to configure */
104 static int toConfigure
;
106 /* display a driver specific configuration dialog */
107 static void configureAudioDriver(HWND hDlg
)
109 const AUDIO_DRIVER
*pAudioDrv
= &loadedAudioDrv
[toConfigure
];
111 if (strlen(pAudioDrv
->szDriver
) != 0)
114 char wine_driver
[MAX_NAME_LENGTH
+ 9];
115 sprintf(wine_driver
, "wine%s.drv", pAudioDrv
->szDriver
);
116 hdrvr
= OpenDriverA(wine_driver
, 0, 0);
119 if (SendDriverMessage(hdrvr
, DRV_QUERYCONFIGURE
, 0, 0) != 0)
123 dci
.dwDCISize
= sizeof (dci
);
124 dci
.lpszDCISectionName
= NULL
;
125 dci
.lpszDCIAliasName
= NULL
;
126 lRes
= SendDriverMessage(hdrvr
, DRV_CONFIGURE
, 0, (LONG_PTR
)&dci
);
128 CloseDriver(hdrvr
, 0, 0);
132 WCHAR wine_driverW
[MAX_NAME_LENGTH
+9];
133 WCHAR messageStr
[256];
136 MultiByteToWideChar (CP_ACP
, 0, wine_driver
, -1, wine_driverW
,
137 sizeof (wine_driverW
)/sizeof(wine_driverW
[0]));
139 LoadStringW (GetModuleHandle (NULL
), IDS_OPEN_DRIVER_ERROR
, messageStr
,
140 sizeof(messageStr
)/sizeof(messageStr
[0]));
141 wsprintfW (str
, messageStr
, wine_driverW
);
142 MessageBoxW (hDlg
, str
, NULL
, MB_OK
| MB_ICONERROR
);
147 /* is driver in local copy of driver registry string */
148 static BOOL
isDriverSet(const char * driver
)
150 WINE_TRACE("driver = %s, curAudioDriver = %s\n", driver
, curAudioDriver
);
152 if (strstr(curAudioDriver
, driver
))
158 /* add driver to local copy of driver registry string */
159 static void addDriver(const char * driver
)
161 if (!isDriverSet(driver
))
163 if (strlen(curAudioDriver
))
164 strcat(curAudioDriver
, ",");
165 strcat(curAudioDriver
, driver
);
169 /* remove driver from local copy of driver registry string */
170 static void removeDriver(const char * driver
)
172 char before
[32], after
[32], * start
;
175 strcat(before
, driver
);
176 strcpy(after
, driver
);
179 if ((start
= strstr(curAudioDriver
, after
)))
181 int len
= strlen(after
);
182 char * end
= curAudioDriver
+ strlen(curAudioDriver
);
183 int i
, count
= end
- start
+ len
;
184 for (i
= 0; i
< count
; i
++)
186 if (start
+ len
>= end
)
193 else if ((start
= strstr(curAudioDriver
, before
)))
195 int len
= strlen(before
);
196 char * end
= curAudioDriver
+ strlen(curAudioDriver
);
197 int i
, count
= end
- start
+ len
;
198 for (i
= 0; i
< count
; i
++)
200 if (start
+ len
>= end
)
207 else if (strcmp(curAudioDriver
, driver
) == 0)
209 strcpy(curAudioDriver
, "");
213 static void initAudioDeviceTree(HWND hDlg
)
215 const AUDIO_DRIVER
*pAudioDrv
= NULL
;
217 TVINSERTSTRUCTW insert
;
218 HTREEITEM root
, driver
[10];
220 HIMAGELIST hImageList
;
223 WCHAR driver_type
[64], dev_type
[64];
225 tree
= GetDlgItem(hDlg
, IDC_AUDIO_TREE
);
230 /* set tree view style */
231 SetWindowLong(tree
, GWL_STYLE
, GetWindowLong(tree
, GWL_STYLE
) | TVS_HASBUTTONS
| TVS_HASLINES
| TVS_LINESATROOT
);
234 hImageList
= ImageList_Create(16, 16, FALSE
, 3, 0);
235 hBitMap
= LoadBitmap(GetModuleHandle(NULL
), MAKEINTRESOURCE(IDB_CHECKBOX
));
236 ImageList_Add(hImageList
, hBitMap
, NULL
);
237 DeleteObject(hBitMap
);
238 SendMessageW( tree
, TVM_SETIMAGELIST
, TVSIL_STATE
, (LPARAM
)hImageList
);
241 LoadStringW (GetModuleHandle (NULL
), IDS_SOUNDDRIVERS
, driver_type
,
242 sizeof(driver_type
)/sizeof(driver_type
[0]));
243 insert
.hParent
= TVI_ROOT
;
244 insert
.hInsertAfter
= TVI_LAST
;
245 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_CHILDREN
;
246 insert
.u
.item
.pszText
= driver_type
;
247 insert
.u
.item
.cChildren
= 1;
248 root
= (HTREEITEM
)SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
250 /* change to the wait cursor because this can take a while if there is a
251 * misbehaving driver that takes a long time to open
253 old_cursor
= SetCursor(LoadCursor(0, IDC_WAIT
));
255 /* iterate over list of loaded drivers */
256 for (pAudioDrv
= loadedAudioDrv
, i
= 0; pAudioDrv
->nameID
; i
++, pAudioDrv
++) {
259 WCHAR text
[MAX_PATH
];
261 sprintf(name
, "wine%s.drv", pAudioDrv
->szDriver
);
262 LoadStringW (GetModuleHandle (NULL
), pAudioDrv
->nameID
, text
,
263 sizeof(text
)/sizeof(text
[0]));
265 if ((hdrv
= OpenDriverA(name
, 0, 0)))
268 if ((lib
= GetDriverModuleHandle(hdrv
)))
270 int num_wod
= 0, num_wid
= 0, num_mod
= 0, num_mid
= 0, num_aux
= 0, num_mxd
= 0;
271 MessagePtr wodMessagePtr
= (MessagePtr
)GetProcAddress(lib
, "wodMessage");
272 MessagePtr widMessagePtr
= (MessagePtr
)GetProcAddress(lib
, "widMessage");
273 MessagePtr modMessagePtr
= (MessagePtr
)GetProcAddress(lib
, "modMessage");
274 MessagePtr midMessagePtr
= (MessagePtr
)GetProcAddress(lib
, "midMessage");
275 MessagePtr auxMessagePtr
= (MessagePtr
)GetProcAddress(lib
, "auxMessage");
276 MessagePtr mxdMessagePtr
= (MessagePtr
)GetProcAddress(lib
, "mxdMessage");
279 num_wod
= wodMessagePtr(0, WODM_GETNUMDEVS
, 0, 0, 0);
282 num_wid
= widMessagePtr(0, WIDM_GETNUMDEVS
, 0, 0, 0);
285 num_mod
= modMessagePtr(0, MODM_GETNUMDEVS
, 0, 0, 0);
288 num_mid
= midMessagePtr(0, MIDM_GETNUMDEVS
, 0, 0, 0);
291 num_aux
= auxMessagePtr(0, AUXDM_GETNUMDEVS
, 0, 0, 0);
294 num_mxd
= mxdMessagePtr(0, MXDM_GETNUMDEVS
, 0, 0, 0);
296 if (num_wod
== 0 && num_wid
== 0 && num_mod
== 0 && num_mid
== 0 && num_aux
== 0 && num_mxd
== 0)
298 insert
.hParent
= root
;
299 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_STATE
| TVIF_PARAM
;
300 insert
.u
.item
.pszText
= text
;
301 insert
.u
.item
.stateMask
= TVIS_STATEIMAGEMASK
;
302 insert
.u
.item
.lParam
= i
+ DRIVER_MASK
;
304 driver
[i
] = (HTREEITEM
)SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
310 insert
.hParent
= root
;
311 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_CHILDREN
| TVIF_STATE
| TVIF_PARAM
;
312 insert
.u
.item
.pszText
= text
;
313 insert
.u
.item
.cChildren
= 1;
314 insert
.u
.item
.stateMask
= TVIS_STATEIMAGEMASK
;
315 insert
.u
.item
.lParam
= i
+ DRIVER_MASK
;
317 if (isDriverSet(pAudioDrv
->szDriver
))
318 insert
.u
.item
.state
= INDEXTOSTATEIMAGEMASK(2);
320 insert
.u
.item
.state
= INDEXTOSTATEIMAGEMASK(1);
322 driver
[i
] = (HTREEITEM
)SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
326 LoadStringW (GetModuleHandle (NULL
), IDS_DEVICES_WAVEOUT
, dev_type
,
327 sizeof(dev_type
)/sizeof(dev_type
[0]));
329 insert
.hParent
= driver
[i
];
330 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_CHILDREN
;
331 insert
.u
.item
.pszText
= dev_type
;
332 insert
.u
.item
.cChildren
= 1;
334 type
= (HTREEITEM
)SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
336 for (j
= 0; j
< num_wod
; j
++)
340 wodMessagePtr(j
, WODM_GETDEVCAPS
, 0, (DWORD_PTR
)&caps
, sizeof(caps
));
342 insert
.hParent
= type
;
343 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_PARAM
;
344 insert
.u
.item
.pszText
= caps
.szPname
;
345 insert
.u
.item
.lParam
= j
+ DEVICE_MASK
;
347 SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
353 LoadStringW (GetModuleHandle (NULL
), IDS_DEVICES_WAVEIN
, dev_type
,
354 sizeof(dev_type
)/sizeof(dev_type
[0]));
356 insert
.hParent
= driver
[i
];
357 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_CHILDREN
;
358 insert
.u
.item
.pszText
= dev_type
;
359 insert
.u
.item
.cChildren
= 1;
361 type
= (HTREEITEM
)SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
363 for (j
= 0; j
< num_wid
; j
++)
367 widMessagePtr(j
, WIDM_GETDEVCAPS
, 0, (DWORD_PTR
)&caps
, sizeof(caps
));
369 insert
.hParent
= type
;
370 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_PARAM
;
371 insert
.u
.item
.pszText
= caps
.szPname
;
372 insert
.u
.item
.lParam
= j
+ DEVICE_MASK
;
374 SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
380 LoadStringW (GetModuleHandle (NULL
), IDS_DEVICES_MIDIOUT
, dev_type
,
381 sizeof(dev_type
)/sizeof(dev_type
[0]));
383 insert
.hParent
= driver
[i
];
384 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_CHILDREN
;
385 insert
.u
.item
.pszText
= dev_type
;
386 insert
.u
.item
.cChildren
= 1;
388 type
= (HTREEITEM
)SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
390 for (j
= 0; j
< num_mod
; j
++)
394 modMessagePtr(j
, MODM_GETDEVCAPS
, 0, (DWORD_PTR
)&caps
, sizeof(caps
));
396 insert
.hParent
= type
;
397 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_PARAM
;
398 insert
.u
.item
.pszText
= caps
.szPname
;
399 insert
.u
.item
.lParam
= j
+ DEVICE_MASK
;
401 SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
407 LoadStringW (GetModuleHandle (NULL
), IDS_DEVICES_MIDIIN
, dev_type
,
408 sizeof(dev_type
)/sizeof(dev_type
[0]));
410 insert
.hParent
= driver
[i
];
411 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_CHILDREN
;
412 insert
.u
.item
.pszText
= dev_type
;
413 insert
.u
.item
.cChildren
= 1;
415 type
= (HTREEITEM
)SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
417 for (j
= 0; j
< num_mid
; j
++)
421 midMessagePtr(j
, MIDM_GETDEVCAPS
, 0, (DWORD_PTR
)&caps
, sizeof(caps
));
423 insert
.hParent
= type
;
424 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_PARAM
;
425 insert
.u
.item
.pszText
= caps
.szPname
;
426 insert
.u
.item
.lParam
= j
+ DEVICE_MASK
;
428 SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
434 LoadStringW (GetModuleHandle (NULL
), IDS_DEVICES_AUX
, dev_type
,
435 sizeof(dev_type
)/sizeof(dev_type
[0]));
437 insert
.hParent
= driver
[i
];
438 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_CHILDREN
;
439 insert
.u
.item
.pszText
= dev_type
;
440 insert
.u
.item
.cChildren
= 1;
442 type
= (HTREEITEM
)SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
444 for (j
= 0; j
< num_aux
; j
++)
448 auxMessagePtr(j
, AUXDM_GETDEVCAPS
, 0, (DWORD_PTR
)&caps
, sizeof(caps
));
450 insert
.hParent
= type
;
451 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_PARAM
;
452 insert
.u
.item
.pszText
= caps
.szPname
;
453 insert
.u
.item
.lParam
= j
+ DEVICE_MASK
;
455 SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
461 LoadStringW (GetModuleHandle (NULL
), IDS_DEVICES_MIXER
, dev_type
,
462 sizeof(dev_type
)/sizeof(dev_type
[0]));
464 insert
.hParent
= driver
[i
];
465 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_CHILDREN
;
466 insert
.u
.item
.pszText
= dev_type
;
467 insert
.u
.item
.cChildren
= 1;
469 type
= (HTREEITEM
)SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
471 for (j
= 0; j
< num_mxd
; j
++)
475 mxdMessagePtr(j
, MXDM_GETDEVCAPS
, 0, (DWORD_PTR
)&caps
, sizeof(caps
));
477 insert
.hParent
= type
;
478 insert
.u
.item
.mask
= TVIF_TEXT
| TVIF_PARAM
;
479 insert
.u
.item
.pszText
= caps
.szPname
;
480 insert
.u
.item
.lParam
= j
+ DEVICE_MASK
;
482 SendDlgItemMessageW (hDlg
, IDC_AUDIO_TREE
, TVM_INSERTITEMW
, 0, (LPARAM
)&insert
);
490 /* restore the original cursor */
491 SetCursor(old_cursor
);
493 SendDlgItemMessage(hDlg
, IDC_AUDIO_TREE
, TVM_SELECTITEM
, 0, 0);
494 SendDlgItemMessage(hDlg
, IDC_AUDIO_TREE
, TVM_EXPAND
, TVE_EXPAND
, (LPARAM
)root
);
495 for (j
= 0; j
< i
; j
++)
496 SendDlgItemMessage(hDlg
, IDC_AUDIO_TREE
, TVM_EXPAND
, TVE_EXPAND
, (LPARAM
)driver
[j
]);
499 /* find all drivers that can be loaded */
500 static void findAudioDrivers(void)
503 const AUDIO_DRIVER
*pAudioDrv
= NULL
;
506 /* delete an existing list */
507 HeapFree(GetProcessHeap(), 0, loadedAudioDrv
);
510 /* change to the wait cursor because this can take a while if there is a
511 * misbehaving driver that takes a long time to open
513 old_cursor
= SetCursor(LoadCursor(0, IDC_WAIT
));
515 for (pAudioDrv
= sAudioDrivers
; pAudioDrv
->nameID
; pAudioDrv
++)
517 if (strlen(pAudioDrv
->szDriver
))
520 char driver
[MAX_PATH
];
522 sprintf(driver
, "wine%s.drv", pAudioDrv
->szDriver
);
524 if ((hdrv
= OpenDriverA(driver
, 0, 0)))
526 CloseDriver(hdrv
, 0, 0);
529 loadedAudioDrv
= HeapReAlloc(GetProcessHeap(), 0, loadedAudioDrv
, (numFound
+ 1) * sizeof(AUDIO_DRIVER
));
531 loadedAudioDrv
= HeapAlloc(GetProcessHeap(), 0, sizeof(AUDIO_DRIVER
));
533 CopyMemory(&loadedAudioDrv
[numFound
], pAudioDrv
, sizeof(AUDIO_DRIVER
));
539 /* restore the original cursor */
540 SetCursor(old_cursor
);
542 /* terminate list with empty driver */
544 loadedAudioDrv
= HeapReAlloc(GetProcessHeap(), 0, loadedAudioDrv
, (numFound
+ 1) * sizeof(AUDIO_DRIVER
));
545 CopyMemory(&loadedAudioDrv
[numFound
], pAudioDrv
, sizeof(AUDIO_DRIVER
));
547 loadedAudioDrv
= HeapAlloc(GetProcessHeap(), 0, sizeof(AUDIO_DRIVER
));
548 ZeroMemory(&loadedAudioDrv
[0], sizeof(AUDIO_DRIVER
));
552 /* check local copy of registry string for unloadable drivers */
553 static void checkRegistrySetting(HWND hDlg
)
555 const AUDIO_DRIVER
*pAudioDrv
;
556 char * token
, * tokens
= strdup(curAudioDriver
);
559 token
= strtok(tokens
, ",");
560 while (token
!= NULL
)
563 for (pAudioDrv
= loadedAudioDrv
; pAudioDrv
->nameID
; pAudioDrv
++)
565 if (strcmp(token
, pAudioDrv
->szDriver
) == 0)
573 WCHAR tokenW
[MAX_NAME_LENGTH
+1];
574 WCHAR messageStr
[256];
578 MultiByteToWideChar (CP_ACP
, 0, token
, -1, tokenW
, sizeof(tokenW
)/sizeof(tokenW
[0]));
580 LoadStringW (GetModuleHandle (NULL
), IDS_UNAVAILABLE_DRIVER
, messageStr
,
581 sizeof(messageStr
)/sizeof(messageStr
[0]));
582 wsprintfW (str
, messageStr
, tokenW
);
583 LoadStringW (GetModuleHandle (NULL
), IDS_WARNING
, caption
,
584 sizeof(caption
)/sizeof(caption
[0]));
585 if (MessageBoxW (hDlg
, str
, caption
, MB_ICONWARNING
| MB_YESNOCANCEL
) == IDYES
)
588 strcpy(tokens
, curAudioDriver
);
592 token
= strtok(NULL
, ",");
597 static void selectDriver(HWND hDlg
, const char * driver
)
602 strcpy(curAudioDriver
, driver
);
603 set_reg_key(config_key
, "Drivers", "Audio", curAudioDriver
);
605 if (LoadStringW(GetModuleHandle(NULL
), IDS_AUDIO_MISSING
, text
, sizeof(text
)/sizeof(text
[0])))
607 if (LoadStringW(GetModuleHandle(NULL
), IDS_WINECFG_TITLE
, caption
, sizeof(caption
)/sizeof(caption
[0])))
608 MessageBoxW(hDlg
, text
, caption
, MB_OK
| MB_ICONINFORMATION
);
611 SendMessage(GetParent(hDlg
), PSM_CHANGED
, (WPARAM
) hDlg
, 0); /* enable apply button */
614 static void initAudioDlg (HWND hDlg
)
621 /* make a list of all drivers that can be loaded */
624 /* get current registry setting if available */
625 buf
= get_reg_key(config_key
, "Drivers", "Audio", NULL
);
627 /* check for first time install and set a default driver
628 * select in this order: oss, alsa, first available driver, none
632 const AUDIO_DRIVER
*pAudioDrv
= NULL
;
634 /* select oss if available */
635 for (pAudioDrv
= loadedAudioDrv
; pAudioDrv
->nameID
; pAudioDrv
++)
637 if (strcmp(pAudioDrv
->szDriver
, "oss") == 0)
639 selectDriver(hDlg
, "oss");
644 if (strlen(curAudioDriver
) == 0)
646 /* select alsa if available */
647 for (pAudioDrv
= loadedAudioDrv
; pAudioDrv
->nameID
; pAudioDrv
++)
649 if (strcmp(pAudioDrv
->szDriver
, "alsa") == 0)
651 selectDriver(hDlg
, "alsa");
657 if (strlen(curAudioDriver
) == 0)
659 /* select first available driver */
660 if (*loadedAudioDrv
->szDriver
)
661 selectDriver(hDlg
, loadedAudioDrv
->szDriver
);
664 else /* make a local copy of the current registry setting */
665 strcpy(curAudioDriver
, buf
);
667 WINE_TRACE("curAudioDriver = %s\n", curAudioDriver
);
669 /* check for drivers that can't be loaded */
670 checkRegistrySetting(hDlg
);
672 initAudioDeviceTree(hDlg
);
674 SendDlgItemMessage(hDlg
, IDC_DSOUND_HW_ACCEL
, CB_RESETCONTENT
, 0, 0);
675 for (i
= 0; 0 != DSound_HW_Accels
[i
].displayID
; ++i
) {
677 LoadStringW (GetModuleHandle (NULL
), DSound_HW_Accels
[i
].displayID
, accelStr
,
678 sizeof(accelStr
)/sizeof(accelStr
[0]));
679 SendDlgItemMessageW (hDlg
, IDC_DSOUND_HW_ACCEL
, CB_ADDSTRING
, 0, (LPARAM
)accelStr
);
681 buf
= get_reg_key(config_key
, keypath("DirectSound"), "HardwareAcceleration", "Full");
682 for (i
= 0; NULL
!= DSound_HW_Accels
[i
].settingStr
; ++i
) {
683 if (strcmp(buf
, DSound_HW_Accels
[i
].settingStr
) == 0) {
684 SendDlgItemMessage(hDlg
, IDC_DSOUND_HW_ACCEL
, CB_SETCURSEL
, i
, 0);
688 if (NULL
== DSound_HW_Accels
[i
].settingStr
) {
689 WINE_ERR("Invalid Direct Sound HW Accel read from registry (%s)\n", buf
);
691 HeapFree(GetProcessHeap(), 0, buf
);
693 SendDlgItemMessage(hDlg
, IDC_DSOUND_RATES
, CB_RESETCONTENT
, 0, 0);
694 for (i
= 0; NULL
!= DSound_Rates
[i
]; ++i
) {
695 SendDlgItemMessage(hDlg
, IDC_DSOUND_RATES
, CB_ADDSTRING
, 0, (LPARAM
) DSound_Rates
[i
]);
697 buf
= get_reg_key(config_key
, keypath("DirectSound"), "DefaultSampleRate", "22050");
698 for (i
= 0; NULL
!= DSound_Rates
[i
]; ++i
) {
699 if (strcmp(buf
, DSound_Rates
[i
]) == 0) {
700 SendDlgItemMessage(hDlg
, IDC_DSOUND_RATES
, CB_SETCURSEL
, i
, 0);
705 SendDlgItemMessage(hDlg
, IDC_DSOUND_BITS
, CB_RESETCONTENT
, 0, 0);
706 for (i
= 0; NULL
!= DSound_Bits
[i
]; ++i
) {
707 SendDlgItemMessage(hDlg
, IDC_DSOUND_BITS
, CB_ADDSTRING
, 0, (LPARAM
) DSound_Bits
[i
]);
709 buf
= get_reg_key(config_key
, keypath("DirectSound"), "DefaultBitsPerSample", "8");
710 for (i
= 0; NULL
!= DSound_Bits
[i
]; ++i
) {
711 if (strcmp(buf
, DSound_Bits
[i
]) == 0) {
712 SendDlgItemMessage(hDlg
, IDC_DSOUND_BITS
, CB_SETCURSEL
, i
, 0);
717 buf
= get_reg_key(config_key
, keypath("DirectSound"), "EmulDriver", "N");
718 if (IS_OPTION_TRUE(*buf
))
719 CheckDlgButton(hDlg
, IDC_DSOUND_DRV_EMUL
, BST_CHECKED
);
721 CheckDlgButton(hDlg
, IDC_DSOUND_DRV_EMUL
, BST_UNCHECKED
);
722 HeapFree(GetProcessHeap(), 0, buf
);
726 AudioDlgProc (HWND hDlg
, UINT uMsg
, WPARAM wParam
, LPARAM lParam
)
730 switch (LOWORD(wParam
)) {
731 case IDC_AUDIO_CONFIGURE
:
732 configureAudioDriver(hDlg
);
734 case IDC_AUDIO_CONTROL_PANEL
:
735 MessageBox(NULL
, "Launching audio control panel not implemented yet!", "Fixme", MB_OK
| MB_ICONERROR
);
737 case IDC_DSOUND_HW_ACCEL
:
738 if (HIWORD(wParam
) == CBN_SELCHANGE
) {
739 int selected_dsound_accel
;
740 SendMessage(GetParent(hDlg
), PSM_CHANGED
, 0, 0);
741 selected_dsound_accel
= SendDlgItemMessage(hDlg
, IDC_DSOUND_HW_ACCEL
, CB_GETCURSEL
, 0, 0);
742 set_reg_key(config_key
, keypath("DirectSound"), "HardwareAcceleration",
743 DSound_HW_Accels
[selected_dsound_accel
].settingStr
);
746 case IDC_DSOUND_RATES
:
747 if (HIWORD(wParam
) == CBN_SELCHANGE
) {
748 int selected_dsound_rate
;
749 SendMessage(GetParent(hDlg
), PSM_CHANGED
, 0, 0);
750 selected_dsound_rate
= SendDlgItemMessage(hDlg
, IDC_DSOUND_RATES
, CB_GETCURSEL
, 0, 0);
751 set_reg_key(config_key
, keypath("DirectSound"), "DefaultSampleRate", DSound_Rates
[selected_dsound_rate
]);
754 case IDC_DSOUND_BITS
:
755 if (HIWORD(wParam
) == CBN_SELCHANGE
) {
756 int selected_dsound_bits
;
757 SendMessage(GetParent(hDlg
), PSM_CHANGED
, 0, 0);
758 selected_dsound_bits
= SendDlgItemMessage(hDlg
, IDC_DSOUND_BITS
, CB_GETCURSEL
, 0, 0);
759 set_reg_key(config_key
, keypath("DirectSound"), "DefaultBitsPerSample", DSound_Bits
[selected_dsound_bits
]);
762 case IDC_DSOUND_DRV_EMUL
:
763 if (HIWORD(wParam
) == BN_CLICKED
) {
764 SendMessage(GetParent(hDlg
), PSM_CHANGED
, 0, 0);
765 if (IsDlgButtonChecked(hDlg
, IDC_DSOUND_DRV_EMUL
) == BST_CHECKED
)
766 set_reg_key(config_key
, keypath("DirectSound"), "EmulDriver", "Y");
768 set_reg_key(config_key
, keypath("DirectSound"), "EmulDriver", "N");
775 set_window_title(hDlg
);
779 switch(((LPNMHDR
)lParam
)->code
) {
781 SetWindowLongPtr(hDlg
, DWLP_MSGRESULT
, FALSE
);
784 set_reg_key(config_key
, "Drivers", "Audio", curAudioDriver
);
786 SetWindowLongPtr(hDlg
, DWLP_MSGRESULT
, PSNRET_NOERROR
);
791 if (((LPNMHDR
)lParam
)->idFrom
== IDC_AUDIO_TREE
)
794 DWORD dwPos
= GetMessagePos();
795 HWND tree
= ((LPNMHDR
)lParam
)->hwndFrom
;
796 ZeroMemory(&ht
, sizeof(ht
));
797 ht
.pt
.x
= (short)LOWORD(dwPos
);
798 ht
.pt
.y
= (short)HIWORD(dwPos
);
799 MapWindowPoints(HWND_DESKTOP
, tree
, &ht
.pt
, 1);
800 SendMessageW( tree
, TVM_HITTEST
, 0, (LPARAM
)&ht
);
801 if (TVHT_ONITEMSTATEICON
& ht
.flags
)
805 ZeroMemory(&tvItem
, sizeof(tvItem
));
806 tvItem
.hItem
= ht
.hItem
;
807 SendMessageW( tree
, TVM_GETITEMW
, 0, (LPARAM
) &tvItem
);
809 index
= TreeView_GetItemState(tree
, ht
.hItem
, TVIS_STATEIMAGEMASK
);
810 if (index
== INDEXTOSTATEIMAGEMASK(1))
812 TreeView_SetItemState(tree
, ht
.hItem
, INDEXTOSTATEIMAGEMASK(2), TVIS_STATEIMAGEMASK
);
813 addDriver(loadedAudioDrv
[tvItem
.lParam
& 0xff].szDriver
);
814 SendMessage(GetParent(hDlg
), PSM_CHANGED
, (WPARAM
) hDlg
, 0); /* enable apply button */
816 else if (index
== INDEXTOSTATEIMAGEMASK(2))
818 TreeView_SetItemState(tree
, ht
.hItem
, INDEXTOSTATEIMAGEMASK(1), TVIS_STATEIMAGEMASK
);
819 removeDriver(loadedAudioDrv
[tvItem
.lParam
& 0xff].szDriver
);
820 SendMessage(GetParent(hDlg
), PSM_CHANGED
, (WPARAM
) hDlg
, 0); /* enable apply button */
826 if (((LPNMHDR
)lParam
)->idFrom
== IDC_AUDIO_TREE
)
829 DWORD dwPos
= GetMessagePos();
830 HWND tree
= ((LPNMHDR
)lParam
)->hwndFrom
;
832 ZeroMemory(&ht
, sizeof(ht
));
833 pt
.x
= (short)LOWORD(dwPos
);
834 pt
.y
= (short)HIWORD(dwPos
);
836 MapWindowPoints(HWND_DESKTOP
, tree
, &ht
.pt
, 1);
837 SendMessageW( tree
, TVM_HITTEST
, 0, (LPARAM
)&ht
);
838 if (TVHT_ONITEMLABEL
& ht
.flags
)
841 ZeroMemory(&tvItem
, sizeof(tvItem
));
842 tvItem
.hItem
= ht
.hItem
;
843 tvItem
.mask
= TVIF_PARAM
;
845 if (TreeView_GetItem(tree
, &tvItem
))
847 if (tvItem
.lParam
& DRIVER_MASK
)
851 TrackPopupMenu(GetSubMenu(hPopupMenus
, 0), TPM_RIGHTBUTTON
, pt
.x
, pt
.y
, 0, tree
, NULL
);
852 toConfigure
= tvItem
.lParam
& ~DRIVER_MASK
;
855 else if (tvItem
.lParam
& DEVICE_MASK
)