xref: /reactos/dll/cpl/mmsys/speakervolume.c (revision 9393fc32)
1 /*
2  * PROJECT:         ReactOS Multimedia Control Panel
3  * FILE:            dll/cpl/mmsys/speakervolume.c
4  * PURPOSE:         ReactOS Multimedia Control Panel
5  * PROGRAMMER:      Eric Kohl <eric.kohl@reactos.com>
6  */
7 
8 #include "mmsys.h"
9 
10 typedef struct _PAGE_DATA
11 {
12     HMIXER hMixer;
13     DWORD volumeControlID;
14     DWORD volumeChannels;
15     DWORD volumeMinimum;
16     DWORD volumeMaximum;
17     DWORD volumeStep;
18     PMIXERCONTROLDETAILS_UNSIGNED volumeValues;
19     BOOL volumeSync;
20 } PAGE_DATA, *PPAGE_DATA;
21 
22 
23 static
24 BOOL
25 OnInitDialog(
26     PPAGE_DATA pPageData,
27     HWND hwndDlg)
28 {
29     TCHAR szBuffer[256];
30     MIXERLINE mxln;
31     MIXERCONTROL mxc;
32     MIXERLINECONTROLS mxlctrl;
33     MIXERCONTROLDETAILS mxcd;
34     INT i, j;
35 
36     /* Open the mixer */
37     if (mixerOpen(&pPageData->hMixer, 0, PtrToUlong(hwndDlg), 0, MIXER_OBJECTF_MIXER | CALLBACK_WINDOW) != MMSYSERR_NOERROR)
38     {
39         MessageBox(hwndDlg, _T("Cannot open mixer"), NULL, MB_OK);
40         return FALSE;
41     }
42 
43     /* Retrieve the mixer information */
44     mxln.cbStruct = sizeof(MIXERLINE);
45     mxln.dwComponentType = MIXERLINE_COMPONENTTYPE_DST_SPEAKERS;
46 
47     if (mixerGetLineInfo((HMIXEROBJ)pPageData->hMixer, &mxln, MIXER_OBJECTF_HMIXER | MIXER_GETLINEINFOF_COMPONENTTYPE) != MMSYSERR_NOERROR)
48         return FALSE;
49 
50     pPageData->volumeChannels = mxln.cChannels;
51 
52     /* Retrieve the line information */
53     mxlctrl.cbStruct = sizeof(MIXERLINECONTROLS);
54     mxlctrl.dwLineID = mxln.dwLineID;
55     mxlctrl.dwControlType = MIXERCONTROL_CONTROLTYPE_VOLUME;
56     mxlctrl.cControls = 1;
57     mxlctrl.cbmxctrl = sizeof(MIXERCONTROL);
58     mxlctrl.pamxctrl = &mxc;
59 
60     if (mixerGetLineControls((HMIXEROBJ)pPageData->hMixer, &mxlctrl, MIXER_OBJECTF_HMIXER | MIXER_GETLINECONTROLSF_ONEBYTYPE) != MMSYSERR_NOERROR)
61         return FALSE;
62 
63     pPageData->volumeControlID = mxc.dwControlID;
64     pPageData->volumeMinimum = mxc.Bounds.dwMinimum;
65     pPageData->volumeMaximum = mxc.Bounds.dwMaximum;
66     pPageData->volumeStep = (pPageData->volumeMaximum - pPageData->volumeMinimum) / (VOLUME_MAX - VOLUME_MIN);
67 
68     /* Allocate a buffer for all channel volume values */
69     pPageData->volumeValues = HeapAlloc(GetProcessHeap(),
70                                         0,
71                                         mxln.cChannels * sizeof(MIXERCONTROLDETAILS_UNSIGNED));
72     if (pPageData->volumeValues == NULL)
73         return FALSE;
74 
75     /* Retrieve the channel volume values */
76     mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
77     mxcd.dwControlID = mxc.dwControlID;
78     mxcd.cChannels = mxln.cChannels;
79     mxcd.cMultipleItems = 0;
80     mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_UNSIGNED);
81     mxcd.paDetails = pPageData->volumeValues;
82 
83     if (mixerGetControlDetails((HMIXEROBJ)pPageData->hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_GETCONTROLDETAILSF_VALUE) != MMSYSERR_NOERROR)
84         return FALSE;
85 
86     /* Initialize the channels */
87     for (i = 0; i < min(mxln.cChannels, 5); i++)
88     {
89         j = i * 4;
90 
91         /* Set the channel name */
92         LoadString(hApplet, IDS_SPEAKER_LEFT + i, szBuffer, _countof(szBuffer));
93         SetWindowText(GetDlgItem(hwndDlg, 9472 + j), szBuffer);
94 
95         /* Initialize the channel trackbar */
96         SendDlgItemMessage(hwndDlg, 9475 + j, TBM_SETRANGE, (WPARAM)TRUE, (LPARAM)MAKELONG(VOLUME_MIN, VOLUME_MAX));
97         SendDlgItemMessage(hwndDlg, 9475 + j, TBM_SETTICFREQ, VOLUME_TICFREQ, 0);
98         SendDlgItemMessage(hwndDlg, 9475 + j, TBM_SETPAGESIZE, 0, VOLUME_PAGESIZE);
99         SendDlgItemMessage(hwndDlg, 9475 + j, TBM_SETPOS, (WPARAM)TRUE, (LPARAM)(pPageData->volumeValues[i].dwValue - pPageData->volumeMinimum) / pPageData->volumeStep);
100     }
101 
102     /* Hide the unused controls */
103     for (i = mxln.cChannels; i < 8; i++)
104     {
105         j = i * 4;
106         ShowWindow(GetDlgItem(hwndDlg, 9472 + j), SW_HIDE);
107         ShowWindow(GetDlgItem(hwndDlg, 9473 + j), SW_HIDE);
108         ShowWindow(GetDlgItem(hwndDlg, 9474 + j), SW_HIDE);
109         ShowWindow(GetDlgItem(hwndDlg, 9475 + j), SW_HIDE);
110     }
111 
112     return TRUE;
113 }
114 
115 
116 static
117 VOID
118 OnMixerControlChange(
119     PPAGE_DATA pPageData,
120     HWND hwndDlg)
121 {
122     MIXERCONTROLDETAILS mxcd;
123     INT i, j;
124 
125     /* Retrieve the channel volume values */
126     mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
127     mxcd.dwControlID = pPageData->volumeControlID;
128     mxcd.cChannels = pPageData->volumeChannels;
129     mxcd.cMultipleItems = 0;
130     mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_UNSIGNED);
131     mxcd.paDetails = pPageData->volumeValues;
132 
133     if (mixerGetControlDetails((HMIXEROBJ)pPageData->hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_GETCONTROLDETAILSF_VALUE) != MMSYSERR_NOERROR)
134         return;
135 
136     for (i = 0; i < pPageData->volumeChannels; i++)
137     {
138         j = i * 4;
139 
140         SendDlgItemMessage(hwndDlg, 9475 + j, TBM_SETPOS, (WPARAM)TRUE, (LPARAM)(pPageData->volumeValues[i].dwValue - pPageData->volumeMinimum) / pPageData->volumeStep);
141     }
142 }
143 
144 
145 static
146 VOID
147 OnHScroll(
148     PPAGE_DATA pPageData,
149     HWND hwndDlg,
150     WPARAM wParam,
151     LPARAM lParam)
152 {
153     MIXERCONTROLDETAILS mxcd;
154     DWORD dwValue, dwPosition;
155     INT id, idx, i, j;
156 
157     id = (INT)GetWindowLongPtr((HWND)lParam, GWLP_ID);
158     if (id < 9475 || id > 9503)
159         return;
160 
161     if ((id - 9475) % 4 != 0)
162         return;
163 
164     dwPosition = (DWORD)SendDlgItemMessage(hwndDlg, id, TBM_GETPOS, 0, 0);
165 
166     if (dwPosition == VOLUME_MIN)
167         dwValue = pPageData->volumeMinimum;
168     else if (dwPosition == VOLUME_MAX)
169         dwValue = pPageData->volumeMaximum;
170     else
171         dwValue = (dwPosition * pPageData->volumeStep) + pPageData->volumeMinimum;
172 
173     if (pPageData->volumeSync)
174     {
175         for (i = 0; i < pPageData->volumeChannels; i++)
176         {
177             j = 9475 + (i * 4);
178             if (j != id)
179                 SendDlgItemMessage(hwndDlg, j, TBM_SETPOS, (WPARAM)TRUE, dwPosition);
180 
181             pPageData->volumeValues[i].dwValue = dwValue;
182         }
183     }
184     else
185     {
186         idx = (id - 9475) / 4;
187         pPageData->volumeValues[idx].dwValue = dwValue;
188     }
189 
190     mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
191     mxcd.dwControlID = pPageData->volumeControlID;
192     mxcd.cChannels = pPageData->volumeChannels;
193     mxcd.cMultipleItems = 0;
194     mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_UNSIGNED);
195     mxcd.paDetails = pPageData->volumeValues;
196 
197     if (mixerSetControlDetails((HMIXEROBJ)pPageData->hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_SETCONTROLDETAILSF_VALUE) != MMSYSERR_NOERROR)
198         return;
199 }
200 
201 
202 static
203 VOID
204 OnSetDefaults(
205     PPAGE_DATA pPageData,
206     HWND hwndDlg)
207 {
208     MIXERCONTROLDETAILS mxcd;
209     DWORD dwValue, i;
210 
211     dwValue = ((VOLUME_MAX - VOLUME_MIN) / 2 * pPageData->volumeStep) + pPageData->volumeMinimum;
212 
213     for (i = 0; i < pPageData->volumeChannels; i++)
214         pPageData->volumeValues[i].dwValue = dwValue;
215 
216     mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
217     mxcd.dwControlID = pPageData->volumeControlID;
218     mxcd.cChannels = pPageData->volumeChannels;
219     mxcd.cMultipleItems = 0;
220     mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_UNSIGNED);
221     mxcd.paDetails = pPageData->volumeValues;
222 
223     if (mixerSetControlDetails((HMIXEROBJ)pPageData->hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_SETCONTROLDETAILSF_VALUE) != MMSYSERR_NOERROR)
224         return;
225 }
226 
227 
228 INT_PTR
229 CALLBACK
230 SpeakerVolumeDlgProc(
231     HWND hwndDlg,
232     UINT uMsg,
233     WPARAM wParam,
234     LPARAM lParam)
235 {
236     PPAGE_DATA pPageData;
237 
238     UNREFERENCED_PARAMETER(wParam);
239 
240     pPageData = (PPAGE_DATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
241 
242     switch(uMsg)
243     {
244         case WM_INITDIALOG:
245             pPageData = (PPAGE_DATA)HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(PAGE_DATA));
246             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pPageData);
247 
248             if (pPageData)
249             {
250                 OnInitDialog(pPageData, hwndDlg);
251             }
252             break;
253 
254         case WM_DESTROY:
255             if (pPageData)
256             {
257                 if (pPageData->volumeValues)
258                     HeapFree(GetProcessHeap(), 0, pPageData->volumeValues);
259 
260                 if (pPageData->hMixer)
261                     mixerClose(pPageData->hMixer);
262 
263                 HeapFree(GetProcessHeap(), 0, pPageData);
264                 pPageData = NULL;
265                 SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)NULL);
266             }
267             break;
268 
269         case WM_HSCROLL:
270             if (pPageData)
271             {
272                 OnHScroll(pPageData, hwndDlg, wParam, lParam);
273                 PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
274             }
275             break;
276 
277         case WM_COMMAND:
278             switch (LOWORD(wParam))
279             {
280                 case 9504:
281                     if (HIWORD(wParam) == BN_CLICKED)
282                         pPageData->volumeSync = (SendDlgItemMessage(hwndDlg, 9504, BM_GETCHECK, 0, 0) == BST_CHECKED);
283                     break;
284 
285                 case 9505:
286                     OnSetDefaults(pPageData, hwndDlg);
287                     PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
288                     break;
289             }
290             break;
291 
292         case WM_NOTIFY:
293             return TRUE;
294 
295         case MM_MIXM_CONTROL_CHANGE:
296             if (pPageData)
297                 OnMixerControlChange(pPageData, hwndDlg);
298             break;
299     }
300 
301     return FALSE;
302 }
303 
304 
305 INT_PTR
306 SpeakerVolume(
307     HWND hwndDlg)
308 {
309     PROPSHEETPAGE psp[1];
310     PROPSHEETHEADER psh;
311     TCHAR Caption[256];
312 
313     LoadString(hApplet, IDS_SPEAKER_VOLUME, Caption, _countof(Caption));
314 
315     ZeroMemory(&psh, sizeof(PROPSHEETHEADER));
316     psh.dwSize = sizeof(PROPSHEETHEADER);
317     psh.dwFlags =  PSH_PROPSHEETPAGE;
318     psh.hwndParent = hwndDlg;
319     psh.hInstance = hApplet;
320     psh.pszCaption = Caption;
321     psh.nPages = sizeof(psp) / sizeof(PROPSHEETPAGE);
322     psh.nStartPage = 0;
323     psh.ppsp = psp;
324 
325     InitPropSheetPage(&psp[0], IDD_MULTICHANNEL, SpeakerVolumeDlgProc);
326     psp[0].dwFlags |= PSP_USETITLE;
327     psp[0].pszTitle = Caption;
328 
329     return (LONG)(PropertySheet(&psh) != -1);
330 }
331