2 * PROJECT: ReactOS Multimedia Control Panel
3 * FILE: dll/cpl/mmsys/mmsys.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>
12 #define VOLUME_DIVIDER 0xFFF
14 typedef struct _IMGINFO
22 typedef struct _GLOBAL_DATA
31 DWORD volumeControlID
;
36 } GLOBAL_DATA
, *PGLOBAL_DATA
;
40 InitImageInfo(PIMGINFO ImgInfo
)
44 ZeroMemory(ImgInfo
, sizeof(*ImgInfo
));
46 ImgInfo
->hBitmap
= LoadImage(hApplet
,
47 MAKEINTRESOURCE(IDB_SPEAKIMG
),
53 if (ImgInfo
->hBitmap
!= NULL
)
55 GetObject(ImgInfo
->hBitmap
, sizeof(BITMAP
), &bitmap
);
57 ImgInfo
->cxSource
= bitmap
.bmWidth
;
58 ImgInfo
->cySource
= bitmap
.bmHeight
;
64 GetMuteControl(PGLOBAL_DATA pGlobalData
)
68 MIXERLINECONTROLS mxlctrl
;
70 if (pGlobalData
->hMixer
== NULL
)
73 mxln
.cbStruct
= sizeof(MIXERLINE
);
74 mxln
.dwComponentType
= MIXERLINE_COMPONENTTYPE_DST_SPEAKERS
;
76 if (mixerGetLineInfo((HMIXEROBJ
)pGlobalData
->hMixer
, &mxln
, MIXER_OBJECTF_HMIXER
| MIXER_GETLINEINFOF_COMPONENTTYPE
)
77 != MMSYSERR_NOERROR
) return;
79 mxlctrl
.cbStruct
= sizeof(MIXERLINECONTROLS
);
80 mxlctrl
.dwLineID
= mxln
.dwLineID
;
81 mxlctrl
.dwControlType
= MIXERCONTROL_CONTROLTYPE_MUTE
;
82 mxlctrl
.cControls
= 1;
83 mxlctrl
.cbmxctrl
= sizeof(MIXERCONTROL
);
84 mxlctrl
.pamxctrl
= &mxc
;
86 if (mixerGetLineControls((HMIXEROBJ
)pGlobalData
->hMixer
, &mxlctrl
, MIXER_OBJECTF_HMIXER
| MIXER_GETLINECONTROLSF_ONEBYTYPE
)
87 != MMSYSERR_NOERROR
) return;
89 pGlobalData
->muteControlID
= mxc
.dwControlID
;
94 GetMuteState(PGLOBAL_DATA pGlobalData
)
96 MIXERCONTROLDETAILS_BOOLEAN mxcdMute
;
97 MIXERCONTROLDETAILS mxcd
;
99 if (pGlobalData
->hMixer
== NULL
)
102 mxcd
.cbStruct
= sizeof(MIXERCONTROLDETAILS
);
103 mxcd
.dwControlID
= pGlobalData
->muteControlID
;
105 mxcd
.cMultipleItems
= 0;
106 mxcd
.cbDetails
= sizeof(MIXERCONTROLDETAILS_BOOLEAN
);
107 mxcd
.paDetails
= &mxcdMute
;
109 if (mixerGetControlDetails((HMIXEROBJ
)pGlobalData
->hMixer
, &mxcd
, MIXER_OBJECTF_HMIXER
| MIXER_GETCONTROLDETAILSF_VALUE
)
113 pGlobalData
->muteVal
= mxcdMute
.fValue
;
118 SwitchMuteState(PGLOBAL_DATA pGlobalData
)
120 MIXERCONTROLDETAILS_BOOLEAN mxcdMute
;
121 MIXERCONTROLDETAILS mxcd
;
123 mxcd
.cbStruct
= sizeof(MIXERCONTROLDETAILS
);
124 mxcd
.dwControlID
= pGlobalData
->muteControlID
;
126 mxcd
.cMultipleItems
= 0;
127 mxcd
.cbDetails
= sizeof(MIXERCONTROLDETAILS_BOOLEAN
);
128 mxcd
.paDetails
= &mxcdMute
;
130 mxcdMute
.fValue
= !pGlobalData
->muteVal
;
131 if (mixerSetControlDetails((HMIXEROBJ
)pGlobalData
->hMixer
, &mxcd
, MIXER_OBJECTF_HMIXER
| MIXER_SETCONTROLDETAILSF_VALUE
)
135 pGlobalData
->muteVal
= mxcdMute
.fValue
;
140 GetVolumeControl(PGLOBAL_DATA pGlobalData
)
144 MIXERLINECONTROLS mxlc
;
146 if (pGlobalData
->hMixer
== NULL
)
149 mxln
.cbStruct
= sizeof(MIXERLINE
);
150 mxln
.dwComponentType
= MIXERLINE_COMPONENTTYPE_DST_SPEAKERS
;
151 if (mixerGetLineInfo((HMIXEROBJ
)pGlobalData
->hMixer
, &mxln
, MIXER_OBJECTF_HMIXER
| MIXER_GETLINEINFOF_COMPONENTTYPE
)
155 mxlc
.cbStruct
= sizeof(MIXERLINECONTROLS
);
156 mxlc
.dwLineID
= mxln
.dwLineID
;
157 mxlc
.dwControlType
= MIXERCONTROL_CONTROLTYPE_VOLUME
;
159 mxlc
.cbmxctrl
= sizeof(MIXERCONTROL
);
160 mxlc
.pamxctrl
= &mxc
;
161 if (mixerGetLineControls((HMIXEROBJ
)pGlobalData
->hMixer
, &mxlc
, MIXER_OBJECTF_HMIXER
| MIXER_GETLINECONTROLSF_ONEBYTYPE
)
165 pGlobalData
->volumeMinimum
= mxc
.Bounds
.dwMinimum
;
166 pGlobalData
->volumeMaximum
= mxc
.Bounds
.dwMaximum
;
167 pGlobalData
->volumeControlID
= mxc
.dwControlID
;
172 GetVolumeValue(PGLOBAL_DATA pGlobalData
)
174 MIXERCONTROLDETAILS_UNSIGNED mxcdVolume
;
175 MIXERCONTROLDETAILS mxcd
;
177 if (pGlobalData
->hMixer
== NULL
)
180 mxcd
.cbStruct
= sizeof(MIXERCONTROLDETAILS
);
181 mxcd
.dwControlID
= pGlobalData
->volumeControlID
;
183 mxcd
.cMultipleItems
= 0;
184 mxcd
.cbDetails
= sizeof(MIXERCONTROLDETAILS_UNSIGNED
);
185 mxcd
.paDetails
= &mxcdVolume
;
187 if (mixerGetControlDetails((HMIXEROBJ
)pGlobalData
->hMixer
, &mxcd
, MIXER_OBJECTF_HMIXER
| MIXER_GETCONTROLDETAILSF_VALUE
)
191 pGlobalData
->volumeValue
= mxcdVolume
.dwValue
;
196 SetVolumeValue(PGLOBAL_DATA pGlobalData
){
197 MIXERCONTROLDETAILS_UNSIGNED mxcdVolume
;
198 MIXERCONTROLDETAILS mxcd
;
200 if (pGlobalData
->hMixer
== NULL
)
203 mxcdVolume
.dwValue
= pGlobalData
->volumeValue
;
204 mxcd
.cbStruct
= sizeof(MIXERCONTROLDETAILS
);
205 mxcd
.dwControlID
= pGlobalData
->volumeControlID
;
207 mxcd
.cMultipleItems
= 0;
208 mxcd
.cbDetails
= sizeof(MIXERCONTROLDETAILS_UNSIGNED
);
209 mxcd
.paDetails
= &mxcdVolume
;
211 if (mixerSetControlDetails((HMIXEROBJ
)pGlobalData
->hMixer
, &mxcd
, MIXER_OBJECTF_HMIXER
| MIXER_SETCONTROLDETAILSF_VALUE
)
215 pGlobalData
->volumeValue
= mxcdVolume
.dwValue
;
220 InitVolumeControls(HWND hwndDlg
, PGLOBAL_DATA pGlobalData
)
225 NumMixers
= mixerGetNumDevs();
228 EnableWindow(GetDlgItem(hwndDlg
, IDC_VOLUME_TRACKBAR
), FALSE
);
229 EnableWindow(GetDlgItem(hwndDlg
, IDC_MUTE_CHECKBOX
), FALSE
);
230 EnableWindow(GetDlgItem(hwndDlg
, IDC_ICON_IN_TASKBAR
), FALSE
);
231 EnableWindow(GetDlgItem(hwndDlg
, IDC_ADVANCED_BTN
), FALSE
);
232 EnableWindow(GetDlgItem(hwndDlg
, IDC_SPEAKER_SET_BTN
), FALSE
);
233 EnableWindow(GetDlgItem(hwndDlg
, IDC_SPEAKER_VOL_BTN
), FALSE
);
234 EnableWindow(GetDlgItem(hwndDlg
, IDC_ADVANCED2_BTN
), FALSE
);
235 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconMuted
);
239 if (mixerOpen(&pGlobalData
->hMixer
, 0, PtrToUlong(hwndDlg
), 0, MIXER_OBJECTF_MIXER
| CALLBACK_WINDOW
) != MMSYSERR_NOERROR
)
241 MessageBox(hwndDlg
, _T("Cannot open mixer"), NULL
, MB_OK
);
245 ZeroMemory(&mxc
, sizeof(MIXERCAPS
));
246 if (mixerGetDevCaps(PtrToUint(pGlobalData
->hMixer
), &mxc
, sizeof(MIXERCAPS
)) != MMSYSERR_NOERROR
)
248 MessageBox(hwndDlg
, _T("mixerGetDevCaps failed"), NULL
, MB_OK
);
252 GetMuteControl(pGlobalData
);
253 GetMuteState(pGlobalData
);
254 if (pGlobalData
->muteVal
)
256 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_CHECKED
, (LPARAM
)0);
257 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconMuted
);
261 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_UNCHECKED
, (LPARAM
)0);
262 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconUnMuted
);
265 GetVolumeControl(pGlobalData
);
266 GetVolumeValue(pGlobalData
);
268 SendDlgItemMessage(hwndDlg
, IDC_DEVICE_NAME
, WM_SETTEXT
, 0, (LPARAM
)mxc
.szPname
);
269 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETRANGE
, (WPARAM
)TRUE
,
270 (LPARAM
)MAKELONG(pGlobalData
->volumeMinimum
, pGlobalData
->volumeMaximum
/VOLUME_DIVIDER
));
271 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETPAGESIZE
, (WPARAM
)FALSE
, (LPARAM
)1);
272 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETSEL
, (WPARAM
)FALSE
,
273 (LPARAM
)MAKELONG(pGlobalData
->volumeMinimum
, pGlobalData
->volumeValue
/VOLUME_DIVIDER
));
274 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETPOS
, (WPARAM
)TRUE
, (LPARAM
)pGlobalData
->volumeValue
/VOLUME_DIVIDER
);
279 LaunchSoundControl(HWND hwndDlg
)
281 if ((INT_PTR
)ShellExecuteW(NULL
, L
"open", L
"sndvol32.exe", NULL
, NULL
, SW_SHOWNORMAL
) > 32)
283 MessageBox(hwndDlg
, _T("Cannot run sndvol32.exe"), NULL
, MB_OK
);
286 /* Volume property page dialog callback */
287 //static INT_PTR CALLBACK
289 VolumeDlgProc(HWND hwndDlg
,
294 static IMGINFO ImgInfo
;
295 PGLOBAL_DATA pGlobalData
;
296 UNREFERENCED_PARAMETER(lParam
);
297 UNREFERENCED_PARAMETER(wParam
);
299 pGlobalData
= (PGLOBAL_DATA
)GetWindowLongPtr(hwndDlg
, DWLP_USER
);
305 case MM_MIXM_LINE_CHANGE
:
307 GetMuteState(pGlobalData
);
308 if (pGlobalData
->muteVal
)
310 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_CHECKED
, (LPARAM
)0);
311 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconMuted
);
315 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_UNCHECKED
, (LPARAM
)0);
316 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconUnMuted
);
320 case MM_MIXM_CONTROL_CHANGE
:
322 GetVolumeValue(pGlobalData
);
323 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETSEL
, (WPARAM
)FALSE
, (LPARAM
)MAKELONG(pGlobalData
->volumeMinimum
, pGlobalData
->volumeValue
/VOLUME_DIVIDER
));
324 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETPOS
, (WPARAM
)TRUE
, (LPARAM
)pGlobalData
->volumeValue
/VOLUME_DIVIDER
);
329 pGlobalData
= (GLOBAL_DATA
*) HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(GLOBAL_DATA
));
330 SetWindowLongPtr(hwndDlg
, DWLP_USER
, (LONG_PTR
)pGlobalData
);
332 pGlobalData
->hIconUnMuted
= LoadImage(hApplet
, MAKEINTRESOURCE(IDI_CPLICON
), IMAGE_ICON
, 32, 32, LR_DEFAULTCOLOR
);
333 pGlobalData
->hIconMuted
= LoadImage(hApplet
, MAKEINTRESOURCE(IDI_MUTED_ICON
), IMAGE_ICON
, 32, 32, LR_DEFAULTCOLOR
);
335 InitImageInfo(&ImgInfo
);
336 InitVolumeControls(hwndDlg
, pGlobalData
);
342 LPDRAWITEMSTRUCT lpDrawItem
;
343 lpDrawItem
= (LPDRAWITEMSTRUCT
) lParam
;
344 if(lpDrawItem
->CtlID
== IDC_SPEAKIMG
)
349 /* Position image in centre of dialog */
350 left
= (lpDrawItem
->rcItem
.right
- ImgInfo
.cxSource
) / 2;
352 hdcMem
= CreateCompatibleDC(lpDrawItem
->hDC
);
355 SelectObject(hdcMem
, ImgInfo
.hBitmap
);
356 BitBlt(lpDrawItem
->hDC
,
358 lpDrawItem
->rcItem
.top
,
359 lpDrawItem
->rcItem
.right
- lpDrawItem
->rcItem
.left
,
360 lpDrawItem
->rcItem
.bottom
- lpDrawItem
->rcItem
.top
,
373 switch (LOWORD(wParam
))
375 case IDC_MUTE_CHECKBOX
:
376 SwitchMuteState(pGlobalData
);
377 if (pGlobalData
->muteVal
)
379 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_CHECKED
, (LPARAM
)0);
380 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconMuted
);
384 SendDlgItemMessage(hwndDlg
, IDC_MUTE_CHECKBOX
, BM_SETCHECK
, (WPARAM
)BST_UNCHECKED
, (LPARAM
)0);
385 SendDlgItemMessage(hwndDlg
, IDC_MUTE_ICON
, STM_SETIMAGE
, IMAGE_ICON
, (LPARAM
)pGlobalData
->hIconUnMuted
);
388 case IDC_ADVANCED_BTN
:
389 LaunchSoundControl(hwndDlg
);
397 HWND hVolumeTrackbar
= GetDlgItem(hwndDlg
, IDC_VOLUME_TRACKBAR
);
398 if (hVolumeTrackbar
== (HWND
)lParam
)
400 pGlobalData
->volumeValue
= (DWORD
)SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_GETPOS
, 0, 0)*VOLUME_DIVIDER
;
401 SetVolumeValue(pGlobalData
);
402 SendDlgItemMessage(hwndDlg
, IDC_VOLUME_TRACKBAR
, TBM_SETSEL
, (WPARAM
)TRUE
,
403 (LPARAM
)MAKELONG(pGlobalData
->volumeMinimum
, pGlobalData
->volumeValue
/VOLUME_DIVIDER
));
409 mixerClose(pGlobalData
->hMixer
);
410 DestroyIcon(pGlobalData
->hIconMuted
);
411 DestroyIcon(pGlobalData
->hIconUnMuted
);
412 HeapFree(GetProcessHeap(), 0, pGlobalData
);