2 * PROJECT: ReactOS Multimedia Control Panel
3 * FILE: dll/cpl/mmsys/volume.c
4 * PURPOSE: ReactOS Multimedia Control Panel
5 * PROGRAMMER: Thomas Weidenmueller <w3seek@reactos.com>
6 * Johannes Anderwald <janderwald@reactos.com>
7 * Dmitry Chapyshev <dmitry@reactos.org>
14 #define VOLUME_DIVIDER 0xFFF
16 typedef struct _IMGINFO
24 typedef struct _GLOBAL_DATA
34 DWORD volumeControlID
;
39 } GLOBAL_DATA
, *PGLOBAL_DATA
;
43 InitImageInfo(PIMGINFO ImgInfo
)
47 ZeroMemory(ImgInfo
, sizeof(*ImgInfo
));
49 ImgInfo
->hBitmap
= LoadImage(hApplet
,
50 MAKEINTRESOURCE(IDB_SPEAKIMG
),
56 if (ImgInfo
->hBitmap
!= NULL
)
58 GetObject(ImgInfo
->hBitmap
, sizeof(BITMAP
), &bitmap
);
60 ImgInfo
->cxSource
= bitmap
.bmWidth
;
61 ImgInfo
->cySource
= bitmap
.bmHeight
;
67 GetMuteControl(PGLOBAL_DATA pGlobalData
)
71 MIXERLINECONTROLS mxlctrl
;
73 if (pGlobalData
->hMixer
== NULL
)
76 mxln
.cbStruct
= sizeof(MIXERLINE
);
77 mxln
.dwComponentType
= MIXERLINE_COMPONENTTYPE_DST_SPEAKERS
;
79 if (mixerGetLineInfo((HMIXEROBJ
)pGlobalData
->hMixer
, &mxln
, MIXER_OBJECTF_HMIXER
| MIXER_GETLINEINFOF_COMPONENTTYPE
)
80 != MMSYSERR_NOERROR
) return;
82 mxlctrl
.cbStruct
= sizeof(MIXERLINECONTROLS
);
83 mxlctrl
.dwLineID
= mxln
.dwLineID
;
84 mxlctrl
.dwControlType
= MIXERCONTROL_CONTROLTYPE_MUTE
;
85 mxlctrl
.cControls
= 1;
86 mxlctrl
.cbmxctrl
= sizeof(MIXERCONTROL
);
87 mxlctrl
.pamxctrl
= &mxc
;
89 if (mixerGetLineControls((HMIXEROBJ
)pGlobalData
->hMixer
, &mxlctrl
, MIXER_OBJECTF_HMIXER
| MIXER_GETLINECONTROLSF_ONEBYTYPE
)
90 != MMSYSERR_NOERROR
) return;
92 pGlobalData
->muteControlID
= mxc
.dwControlID
;
97 GetMuteState(PGLOBAL_DATA pGlobalData
)
99 MIXERCONTROLDETAILS_BOOLEAN mxcdMute
;
100 MIXERCONTROLDETAILS mxcd
;
102 if (pGlobalData
->hMixer
== NULL
)
105 mxcd
.cbStruct
= sizeof(MIXERCONTROLDETAILS
);
106 mxcd
.dwControlID
= pGlobalData
->muteControlID
;
108 mxcd
.cMultipleItems
= 0;
109 mxcd
.cbDetails
= sizeof(MIXERCONTROLDETAILS_BOOLEAN
);
110 mxcd
.paDetails
= &mxcdMute
;
112 if (mixerGetControlDetails((HMIXEROBJ
)pGlobalData
->hMixer
, &mxcd
, MIXER_OBJECTF_HMIXER
| MIXER_GETCONTROLDETAILSF_VALUE
)
116 pGlobalData
->muteVal
= mxcdMute
.fValue
;
121 SwitchMuteState(PGLOBAL_DATA pGlobalData
)
123 MIXERCONTROLDETAILS_BOOLEAN mxcdMute
;
124 MIXERCONTROLDETAILS mxcd
;
126 mxcd
.cbStruct
= sizeof(MIXERCONTROLDETAILS
);
127 mxcd
.dwControlID
= pGlobalData
->muteControlID
;
129 mxcd
.cMultipleItems
= 0;
130 mxcd
.cbDetails
= sizeof(MIXERCONTROLDETAILS_BOOLEAN
);
131 mxcd
.paDetails
= &mxcdMute
;
133 mxcdMute
.fValue
= !pGlobalData
->muteVal
;
134 if (mixerSetControlDetails((HMIXEROBJ
)pGlobalData
->hMixer
, &mxcd
, MIXER_OBJECTF_HMIXER
| MIXER_SETCONTROLDETAILSF_VALUE
)
138 pGlobalData
->muteVal
= mxcdMute
.fValue
;
143 GetVolumeControl(PGLOBAL_DATA pGlobalData
)
147 MIXERLINECONTROLS mxlc
;
149 if (pGlobalData
->hMixer
== NULL
)
152 mxln
.cbStruct
= sizeof(MIXERLINE
);
153 mxln
.dwComponentType
= MIXERLINE_COMPONENTTYPE_DST_SPEAKERS
;
154 if (mixerGetLineInfo((HMIXEROBJ
)pGlobalData
->hMixer
, &mxln
, MIXER_OBJECTF_HMIXER
| MIXER_GETLINEINFOF_COMPONENTTYPE
)
158 mxlc
.cbStruct
= sizeof(MIXERLINECONTROLS
);
159 mxlc
.dwLineID
= mxln
.dwLineID
;
160 mxlc
.dwControlType
= MIXERCONTROL_CONTROLTYPE_VOLUME
;
162 mxlc
.cbmxctrl
= sizeof(MIXERCONTROL
);
163 mxlc
.pamxctrl
= &mxc
;
164 if (mixerGetLineControls((HMIXEROBJ
)pGlobalData
->hMixer
, &mxlc
, MIXER_OBJECTF_HMIXER
| MIXER_GETLINECONTROLSF_ONEBYTYPE
)
168 pGlobalData
->volumeMinimum
= mxc
.Bounds
.dwMinimum
;
169 pGlobalData
->volumeMaximum
= mxc
.Bounds
.dwMaximum
;
170 pGlobalData
->volumeControlID
= mxc
.dwControlID
;
175 GetVolumeValue(PGLOBAL_DATA pGlobalData
)
177 MIXERCONTROLDETAILS_UNSIGNED mxcdVolume
;
178 MIXERCONTROLDETAILS mxcd
;
180 if (pGlobalData
->hMixer
== NULL
)
183 mxcd
.cbStruct
= sizeof(MIXERCONTROLDETAILS
);
184 mxcd
.dwControlID
= pGlobalData
->volumeControlID
;
186 mxcd
.cMultipleItems
= 0;
187 mxcd
.cbDetails
= sizeof(MIXERCONTROLDETAILS_UNSIGNED
);
188 mxcd
.paDetails
= &mxcdVolume
;
190 if (mixerGetControlDetails((HMIXEROBJ
)pGlobalData
->hMixer
, &mxcd
, MIXER_OBJECTF_HMIXER
| MIXER_GETCONTROLDETAILSF_VALUE
)
194 pGlobalData
->volumeValue
= mxcdVolume
.dwValue
;
199 SetVolumeValue(PGLOBAL_DATA pGlobalData
){
200 MIXERCONTROLDETAILS_UNSIGNED mxcdVolume
;
201 MIXERCONTROLDETAILS mxcd
;
203 if (pGlobalData
->hMixer
== NULL
)
206 mxcdVolume
.dwValue
= pGlobalData
->volumeValue
;
207 mxcd
.cbStruct
= sizeof(MIXERCONTROLDETAILS
);
208 mxcd
.dwControlID
= pGlobalData
->volumeControlID
;
210 mxcd
.cMultipleItems
= 0;
211 mxcd
.cbDetails
= sizeof(MIXERCONTROLDETAILS_UNSIGNED
);
212 mxcd
.paDetails
= &mxcdVolume
;
214 if (mixerSetControlDetails((HMIXEROBJ
)pGlobalData
->hMixer
, &mxcd
, MIXER_OBJECTF_HMIXER
| MIXER_SETCONTROLDETAILSF_VALUE
)
218 pGlobalData
->volumeValue
= mxcdVolume
.dwValue
;
223 InitVolumeControls(HWND hwndDlg
, PGLOBAL_DATA pGlobalData
)
227 TCHAR szNoDevices
[256];
229 LoadString(hApplet
, IDS_NO_DEVICES
, szNoDevices
, _countof(szNoDevices
));
231 NumMixers
= mixerGetNumDevs();
234 EnableWindow(GetDlgItem(hwndDlg
, IDC_VOLUME_TRACKBAR
), FALSE
);
235 EnableWindow(GetDlgItem(hwndDlg
, IDC_MUTE_CHECKBOX
), FALSE
);
236 EnableWindow(GetDlgItem(hwndDlg
, IDC_ICON_IN_TASKBAR
), FALSE
);
237 EnableWindow(GetDlgItem(hwndDlg
, IDC_ADVANCED_BTN
), FALSE
);
238 EnableWindow(GetDlgItem(hwndDlg
, IDC_SPEAKER_VOL_BTN
), FALSE
);
239 EnableWindow(GetDlgItem(hwndDlg
, IDC_ADVANCED2_BTN
), FALSE
);
240 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconNoHW
);
241 SetDlgItemText(hwndDlg
, IDC_DEVICE_NAME
, szNoDevices
);
245 if (mixerOpen(&pGlobalData
->hMixer
, 0, PtrToUlong(hwndDlg
), 0, MIXER_OBJECTF_MIXER
| CALLBACK_WINDOW
) != MMSYSERR_NOERROR
)
247 MessageBox(hwndDlg
, _T("Cannot open mixer"), NULL
, MB_OK
);
251 ZeroMemory(&mxc
, sizeof(MIXERCAPS
));
252 if (mixerGetDevCaps(PtrToUint(pGlobalData
->hMixer
), &mxc
, sizeof(MIXERCAPS
)) != MMSYSERR_NOERROR
)
254 MessageBox(hwndDlg
, _T("mixerGetDevCaps failed"), NULL
, MB_OK
);
258 GetMuteControl(pGlobalData
);
259 GetMuteState(pGlobalData
);
260 if (pGlobalData
->muteVal
)
262 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_CHECKED
, (LPARAM
)0);
263 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconMuted
);
267 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_UNCHECKED
, (LPARAM
)0);
268 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconUnMuted
);
271 GetVolumeControl(pGlobalData
);
272 GetVolumeValue(pGlobalData
);
274 SendDlgItemMessage(hwndDlg
, IDC_DEVICE_NAME
, WM_SETTEXT
, 0, (LPARAM
)mxc
.szPname
);
275 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETRANGE
, (WPARAM
)TRUE
,
276 (LPARAM
)MAKELONG(pGlobalData
->volumeMinimum
, pGlobalData
->volumeMaximum
/VOLUME_DIVIDER
));
277 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETPAGESIZE
, (WPARAM
)FALSE
, (LPARAM
)1);
278 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETSEL
, (WPARAM
)FALSE
,
279 (LPARAM
)MAKELONG(pGlobalData
->volumeMinimum
, pGlobalData
->volumeValue
/VOLUME_DIVIDER
));
280 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETPOS
, (WPARAM
)TRUE
, (LPARAM
)pGlobalData
->volumeValue
/VOLUME_DIVIDER
);
285 LaunchSoundControl(HWND hwndDlg
)
287 if ((INT_PTR
)ShellExecuteW(NULL
, L
"open", L
"sndvol32.exe", NULL
, NULL
, SW_SHOWNORMAL
) > 32)
289 MessageBox(hwndDlg
, _T("Cannot run sndvol32.exe"), NULL
, MB_OK
);
292 /* Volume property page dialog callback */
293 //static INT_PTR CALLBACK
295 VolumeDlgProc(HWND hwndDlg
,
300 static IMGINFO ImgInfo
;
301 PGLOBAL_DATA pGlobalData
;
302 UNREFERENCED_PARAMETER(lParam
);
303 UNREFERENCED_PARAMETER(wParam
);
305 pGlobalData
= (PGLOBAL_DATA
)GetWindowLongPtr(hwndDlg
, DWLP_USER
);
311 case MM_MIXM_LINE_CHANGE
:
313 GetMuteState(pGlobalData
);
314 if (pGlobalData
->muteVal
)
316 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_CHECKED
, (LPARAM
)0);
317 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconMuted
);
321 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_UNCHECKED
, (LPARAM
)0);
322 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconUnMuted
);
326 case MM_MIXM_CONTROL_CHANGE
:
328 GetVolumeValue(pGlobalData
);
329 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETSEL
, (WPARAM
)FALSE
, (LPARAM
)MAKELONG(pGlobalData
->volumeMinimum
, pGlobalData
->volumeValue
/VOLUME_DIVIDER
));
330 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETPOS
, (WPARAM
)TRUE
, (LPARAM
)pGlobalData
->volumeValue
/VOLUME_DIVIDER
);
335 pGlobalData
= (GLOBAL_DATA
*) HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(GLOBAL_DATA
));
336 SetWindowLongPtr(hwndDlg
, DWLP_USER
, (LONG_PTR
)pGlobalData
);
338 pGlobalData
->hIconUnMuted
= LoadImage(hApplet
, MAKEINTRESOURCE(IDI_CPLICON
), IMAGE_ICON
, 32, 32, LR_DEFAULTCOLOR
);
339 pGlobalData
->hIconMuted
= LoadImage(hApplet
, MAKEINTRESOURCE(IDI_MUTED_ICON
), IMAGE_ICON
, 32, 32, LR_DEFAULTCOLOR
);
340 pGlobalData
->hIconNoHW
= LoadImage(hApplet
, MAKEINTRESOURCE(IDI_NO_HW
), IMAGE_ICON
, 32, 32, LR_DEFAULTCOLOR
);
342 InitImageInfo(&ImgInfo
);
343 InitVolumeControls(hwndDlg
, pGlobalData
);
349 LPDRAWITEMSTRUCT lpDrawItem
;
350 lpDrawItem
= (LPDRAWITEMSTRUCT
) lParam
;
351 if(lpDrawItem
->CtlID
== IDC_SPEAKIMG
)
356 /* Position image in centre of dialog */
357 left
= (lpDrawItem
->rcItem
.right
- ImgInfo
.cxSource
) / 2;
359 hdcMem
= CreateCompatibleDC(lpDrawItem
->hDC
);
362 SelectObject(hdcMem
, ImgInfo
.hBitmap
);
363 BitBlt(lpDrawItem
->hDC
,
365 lpDrawItem
->rcItem
.top
,
366 lpDrawItem
->rcItem
.right
- lpDrawItem
->rcItem
.left
,
367 lpDrawItem
->rcItem
.bottom
- lpDrawItem
->rcItem
.top
,
380 switch (LOWORD(wParam
))
382 case IDC_MUTE_CHECKBOX
:
383 SwitchMuteState(pGlobalData
);
384 if (pGlobalData
->muteVal
)
386 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_CHECKED
, (LPARAM
)0);
387 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconMuted
);
391 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_UNCHECKED
, (LPARAM
)0);
392 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconUnMuted
);
395 case IDC_ADVANCED_BTN
:
396 LaunchSoundControl(hwndDlg
);
404 HWND hVolumeTrackbar
= GetDlgItem(hwndDlg
, IDC_VOLUME_TRACKBAR
);
405 if (hVolumeTrackbar
== (HWND
)lParam
)
407 pGlobalData
->volumeValue
= (DWORD
)SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_GETPOS
, 0, 0)*VOLUME_DIVIDER
;
408 SetVolumeValue(pGlobalData
);
409 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETSEL
, (WPARAM
)TRUE
,
410 (LPARAM
)MAKELONG(pGlobalData
->volumeMinimum
, pGlobalData
->volumeValue
/VOLUME_DIVIDER
));
416 mixerClose(pGlobalData
->hMixer
);
417 DestroyIcon(pGlobalData
->hIconMuted
);
418 DestroyIcon(pGlobalData
->hIconUnMuted
);
419 DestroyIcon(pGlobalData
->hIconNoHW
);
420 HeapFree(GetProcessHeap(), 0, pGlobalData
);