2019-01-20 14:57:34 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: ReactOS Multimedia Control Panel
|
|
|
|
* FILE: dll/cpl/mmsys/speakervolume.c
|
|
|
|
* PURPOSE: ReactOS Multimedia Control Panel
|
|
|
|
* PROGRAMMER: Eric Kohl <eric.kohl@reactos.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "mmsys.h"
|
|
|
|
|
2019-01-20 20:59:45 +00:00
|
|
|
typedef struct _PAGE_DATA
|
|
|
|
{
|
|
|
|
HMIXER hMixer;
|
|
|
|
DWORD volumeControlID;
|
|
|
|
DWORD volumeChannels;
|
|
|
|
DWORD volumeMinimum;
|
|
|
|
DWORD volumeMaximum;
|
|
|
|
DWORD volumeStep;
|
|
|
|
PMIXERCONTROLDETAILS_UNSIGNED volumeValues;
|
2019-01-20 23:25:17 +00:00
|
|
|
BOOL volumeSync;
|
2019-01-20 20:59:45 +00:00
|
|
|
} PAGE_DATA, *PPAGE_DATA;
|
|
|
|
|
|
|
|
|
2019-01-20 14:57:34 +00:00
|
|
|
static
|
|
|
|
BOOL
|
|
|
|
OnInitDialog(
|
2019-01-20 20:59:45 +00:00
|
|
|
PPAGE_DATA pPageData,
|
|
|
|
HWND hwndDlg)
|
2019-01-20 14:57:34 +00:00
|
|
|
{
|
2022-09-14 17:06:22 +00:00
|
|
|
WCHAR szBuffer[256];
|
|
|
|
MIXERLINEW mxln;
|
|
|
|
MIXERCONTROLW mxc;
|
|
|
|
MIXERLINECONTROLSW mxlctrl;
|
2019-01-20 14:57:34 +00:00
|
|
|
MIXERCONTROLDETAILS mxcd;
|
|
|
|
INT i, j;
|
|
|
|
|
2019-01-20 20:59:45 +00:00
|
|
|
/* Open the mixer */
|
|
|
|
if (mixerOpen(&pPageData->hMixer, 0, PtrToUlong(hwndDlg), 0, MIXER_OBJECTF_MIXER | CALLBACK_WINDOW) != MMSYSERR_NOERROR)
|
|
|
|
{
|
2022-09-14 17:06:22 +00:00
|
|
|
MessageBoxW(hwndDlg, L"Cannot open mixer", NULL, MB_OK);
|
2019-01-20 20:59:45 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
2019-01-20 14:57:34 +00:00
|
|
|
|
2019-01-20 20:59:45 +00:00
|
|
|
/* Retrieve the mixer information */
|
2022-09-14 17:06:22 +00:00
|
|
|
mxln.cbStruct = sizeof(MIXERLINEW);
|
2019-01-20 14:57:34 +00:00
|
|
|
mxln.dwComponentType = MIXERLINE_COMPONENTTYPE_DST_SPEAKERS;
|
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
if (mixerGetLineInfoW((HMIXEROBJ)pPageData->hMixer, &mxln, MIXER_OBJECTF_HMIXER | MIXER_GETLINEINFOF_COMPONENTTYPE) != MMSYSERR_NOERROR)
|
2019-01-20 14:57:34 +00:00
|
|
|
return FALSE;
|
|
|
|
|
2019-01-20 20:59:45 +00:00
|
|
|
pPageData->volumeChannels = mxln.cChannels;
|
|
|
|
|
|
|
|
/* Retrieve the line information */
|
2022-09-14 17:06:22 +00:00
|
|
|
mxlctrl.cbStruct = sizeof(MIXERLINECONTROLSW);
|
2019-01-20 14:57:34 +00:00
|
|
|
mxlctrl.dwLineID = mxln.dwLineID;
|
|
|
|
mxlctrl.dwControlType = MIXERCONTROL_CONTROLTYPE_VOLUME;
|
|
|
|
mxlctrl.cControls = 1;
|
2022-09-14 17:06:22 +00:00
|
|
|
mxlctrl.cbmxctrl = sizeof(MIXERCONTROLW);
|
2019-01-20 14:57:34 +00:00
|
|
|
mxlctrl.pamxctrl = &mxc;
|
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
if (mixerGetLineControlsW((HMIXEROBJ)pPageData->hMixer, &mxlctrl, MIXER_OBJECTF_HMIXER | MIXER_GETLINECONTROLSF_ONEBYTYPE) != MMSYSERR_NOERROR)
|
2019-01-20 14:57:34 +00:00
|
|
|
return FALSE;
|
|
|
|
|
2019-01-20 20:59:45 +00:00
|
|
|
pPageData->volumeControlID = mxc.dwControlID;
|
|
|
|
pPageData->volumeMinimum = mxc.Bounds.dwMinimum;
|
|
|
|
pPageData->volumeMaximum = mxc.Bounds.dwMaximum;
|
|
|
|
pPageData->volumeStep = (pPageData->volumeMaximum - pPageData->volumeMinimum) / (VOLUME_MAX - VOLUME_MIN);
|
2019-01-20 14:57:34 +00:00
|
|
|
|
2019-01-20 20:59:45 +00:00
|
|
|
/* Allocate a buffer for all channel volume values */
|
|
|
|
pPageData->volumeValues = HeapAlloc(GetProcessHeap(),
|
|
|
|
0,
|
|
|
|
mxln.cChannels * sizeof(MIXERCONTROLDETAILS_UNSIGNED));
|
|
|
|
if (pPageData->volumeValues == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* Retrieve the channel volume values */
|
2019-01-20 14:57:34 +00:00
|
|
|
mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
|
|
|
|
mxcd.dwControlID = mxc.dwControlID;
|
|
|
|
mxcd.cChannels = mxln.cChannels;
|
|
|
|
mxcd.cMultipleItems = 0;
|
|
|
|
mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_UNSIGNED);
|
2019-01-20 20:59:45 +00:00
|
|
|
mxcd.paDetails = pPageData->volumeValues;
|
2019-01-20 14:57:34 +00:00
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
if (mixerGetControlDetailsW((HMIXEROBJ)pPageData->hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_GETCONTROLDETAILSF_VALUE) != MMSYSERR_NOERROR)
|
2019-01-20 14:57:34 +00:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* Initialize the channels */
|
|
|
|
for (i = 0; i < min(mxln.cChannels, 5); i++)
|
|
|
|
{
|
|
|
|
j = i * 4;
|
2019-01-20 20:59:45 +00:00
|
|
|
|
|
|
|
/* Set the channel name */
|
2022-09-14 17:06:22 +00:00
|
|
|
LoadStringW(hApplet, IDS_SPEAKER_LEFT + i, szBuffer, _countof(szBuffer));
|
|
|
|
SetWindowTextW(GetDlgItem(hwndDlg, 9472 + j), szBuffer);
|
2019-01-20 14:57:34 +00:00
|
|
|
|
2019-01-20 20:59:45 +00:00
|
|
|
/* Initialize the channel trackbar */
|
2022-09-14 17:06:22 +00:00
|
|
|
SendDlgItemMessageW(hwndDlg, 9475 + j, TBM_SETRANGE, (WPARAM)TRUE, MAKELPARAM(VOLUME_MIN, VOLUME_MAX));
|
|
|
|
SendDlgItemMessageW(hwndDlg, 9475 + j, TBM_SETTICFREQ, VOLUME_TICFREQ, 0);
|
|
|
|
SendDlgItemMessageW(hwndDlg, 9475 + j, TBM_SETPAGESIZE, 0, VOLUME_PAGESIZE);
|
|
|
|
SendDlgItemMessageW(hwndDlg, 9475 + j, TBM_SETPOS, (WPARAM)TRUE, (LPARAM)(pPageData->volumeValues[i].dwValue - pPageData->volumeMinimum) / pPageData->volumeStep);
|
2019-01-20 14:57:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Hide the unused controls */
|
|
|
|
for (i = mxln.cChannels; i < 8; i++)
|
|
|
|
{
|
|
|
|
j = i * 4;
|
|
|
|
ShowWindow(GetDlgItem(hwndDlg, 9472 + j), SW_HIDE);
|
|
|
|
ShowWindow(GetDlgItem(hwndDlg, 9473 + j), SW_HIDE);
|
|
|
|
ShowWindow(GetDlgItem(hwndDlg, 9474 + j), SW_HIDE);
|
|
|
|
ShowWindow(GetDlgItem(hwndDlg, 9475 + j), SW_HIDE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-01-20 20:59:45 +00:00
|
|
|
static
|
|
|
|
VOID
|
|
|
|
OnMixerControlChange(
|
|
|
|
PPAGE_DATA pPageData,
|
|
|
|
HWND hwndDlg)
|
|
|
|
{
|
|
|
|
MIXERCONTROLDETAILS mxcd;
|
|
|
|
INT i, j;
|
|
|
|
|
|
|
|
/* Retrieve the channel volume values */
|
|
|
|
mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
|
|
|
|
mxcd.dwControlID = pPageData->volumeControlID;
|
|
|
|
mxcd.cChannels = pPageData->volumeChannels;
|
|
|
|
mxcd.cMultipleItems = 0;
|
|
|
|
mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_UNSIGNED);
|
|
|
|
mxcd.paDetails = pPageData->volumeValues;
|
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
if (mixerGetControlDetailsW((HMIXEROBJ)pPageData->hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_GETCONTROLDETAILSF_VALUE) != MMSYSERR_NOERROR)
|
2019-01-20 20:59:45 +00:00
|
|
|
return;
|
|
|
|
|
2019-01-20 23:25:17 +00:00
|
|
|
for (i = 0; i < pPageData->volumeChannels; i++)
|
2019-01-20 20:59:45 +00:00
|
|
|
{
|
|
|
|
j = i * 4;
|
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
SendDlgItemMessageW(hwndDlg, 9475 + j, TBM_SETPOS, (WPARAM)TRUE, (LPARAM)(pPageData->volumeValues[i].dwValue - pPageData->volumeMinimum) / pPageData->volumeStep);
|
2019-01-20 20:59:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static
|
|
|
|
VOID
|
|
|
|
OnHScroll(
|
|
|
|
PPAGE_DATA pPageData,
|
|
|
|
HWND hwndDlg,
|
|
|
|
WPARAM wParam,
|
|
|
|
LPARAM lParam)
|
|
|
|
{
|
|
|
|
MIXERCONTROLDETAILS mxcd;
|
2019-02-04 22:54:04 +00:00
|
|
|
DWORD dwValue, dwPosition;
|
2019-01-20 23:25:17 +00:00
|
|
|
INT id, idx, i, j;
|
2019-01-20 20:59:45 +00:00
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
id = (INT)GetWindowLongPtrW((HWND)lParam, GWLP_ID);
|
2020-05-25 17:57:52 +00:00
|
|
|
if (id < 9475 || id > 9503)
|
2019-01-20 20:59:45 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
if ((id - 9475) % 4 != 0)
|
|
|
|
return;
|
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
dwPosition = (DWORD)SendDlgItemMessageW(hwndDlg, id, TBM_GETPOS, 0, 0);
|
2019-02-04 22:54:04 +00:00
|
|
|
|
|
|
|
if (dwPosition == VOLUME_MIN)
|
|
|
|
dwValue = pPageData->volumeMinimum;
|
|
|
|
else if (dwPosition == VOLUME_MAX)
|
|
|
|
dwValue = pPageData->volumeMaximum;
|
|
|
|
else
|
|
|
|
dwValue = (dwPosition * pPageData->volumeStep) + pPageData->volumeMinimum;
|
2019-01-20 20:59:45 +00:00
|
|
|
|
2019-01-20 23:25:17 +00:00
|
|
|
if (pPageData->volumeSync)
|
|
|
|
{
|
|
|
|
for (i = 0; i < pPageData->volumeChannels; i++)
|
|
|
|
{
|
|
|
|
j = 9475 + (i * 4);
|
|
|
|
if (j != id)
|
2022-09-14 17:06:22 +00:00
|
|
|
SendDlgItemMessageW(hwndDlg, j, TBM_SETPOS, (WPARAM)TRUE, dwPosition);
|
2019-01-20 23:25:17 +00:00
|
|
|
|
|
|
|
pPageData->volumeValues[i].dwValue = dwValue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
idx = (id - 9475) / 4;
|
|
|
|
pPageData->volumeValues[idx].dwValue = dwValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
|
|
|
|
mxcd.dwControlID = pPageData->volumeControlID;
|
|
|
|
mxcd.cChannels = pPageData->volumeChannels;
|
|
|
|
mxcd.cMultipleItems = 0;
|
|
|
|
mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_UNSIGNED);
|
|
|
|
mxcd.paDetails = pPageData->volumeValues;
|
|
|
|
|
|
|
|
if (mixerSetControlDetails((HMIXEROBJ)pPageData->hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_SETCONTROLDETAILSF_VALUE) != MMSYSERR_NOERROR)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static
|
|
|
|
VOID
|
|
|
|
OnSetDefaults(
|
|
|
|
PPAGE_DATA pPageData,
|
|
|
|
HWND hwndDlg)
|
|
|
|
{
|
|
|
|
MIXERCONTROLDETAILS mxcd;
|
|
|
|
DWORD dwValue, i;
|
|
|
|
|
|
|
|
dwValue = ((VOLUME_MAX - VOLUME_MIN) / 2 * pPageData->volumeStep) + pPageData->volumeMinimum;
|
|
|
|
|
|
|
|
for (i = 0; i < pPageData->volumeChannels; i++)
|
|
|
|
pPageData->volumeValues[i].dwValue = dwValue;
|
2019-01-20 20:59:45 +00:00
|
|
|
|
|
|
|
mxcd.cbStruct = sizeof(MIXERCONTROLDETAILS);
|
|
|
|
mxcd.dwControlID = pPageData->volumeControlID;
|
|
|
|
mxcd.cChannels = pPageData->volumeChannels;
|
|
|
|
mxcd.cMultipleItems = 0;
|
|
|
|
mxcd.cbDetails = sizeof(MIXERCONTROLDETAILS_UNSIGNED);
|
|
|
|
mxcd.paDetails = pPageData->volumeValues;
|
|
|
|
|
|
|
|
if (mixerSetControlDetails((HMIXEROBJ)pPageData->hMixer, &mxcd, MIXER_OBJECTF_HMIXER | MIXER_SETCONTROLDETAILSF_VALUE) != MMSYSERR_NOERROR)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-01-20 14:57:34 +00:00
|
|
|
INT_PTR
|
|
|
|
CALLBACK
|
|
|
|
SpeakerVolumeDlgProc(
|
|
|
|
HWND hwndDlg,
|
|
|
|
UINT uMsg,
|
|
|
|
WPARAM wParam,
|
|
|
|
LPARAM lParam)
|
|
|
|
{
|
2019-01-20 20:59:45 +00:00
|
|
|
PPAGE_DATA pPageData;
|
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
pPageData = (PPAGE_DATA)GetWindowLongPtrW(hwndDlg, DWLP_USER);
|
2019-01-20 14:57:34 +00:00
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
switch (uMsg)
|
2019-01-20 14:57:34 +00:00
|
|
|
{
|
|
|
|
case WM_INITDIALOG:
|
2019-01-20 20:59:45 +00:00
|
|
|
pPageData = (PPAGE_DATA)HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(PAGE_DATA));
|
2022-09-14 17:06:22 +00:00
|
|
|
SetWindowLongPtrW(hwndDlg, DWLP_USER, (LONG_PTR)pPageData);
|
2019-01-20 20:59:45 +00:00
|
|
|
|
|
|
|
if (pPageData)
|
|
|
|
{
|
|
|
|
OnInitDialog(pPageData, hwndDlg);
|
|
|
|
}
|
2019-01-20 14:57:34 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_DESTROY:
|
2019-01-20 20:59:45 +00:00
|
|
|
if (pPageData)
|
|
|
|
{
|
|
|
|
if (pPageData->volumeValues)
|
|
|
|
HeapFree(GetProcessHeap(), 0, pPageData->volumeValues);
|
|
|
|
|
|
|
|
if (pPageData->hMixer)
|
|
|
|
mixerClose(pPageData->hMixer);
|
|
|
|
|
|
|
|
HeapFree(GetProcessHeap(), 0, pPageData);
|
|
|
|
pPageData = NULL;
|
2022-09-14 17:06:22 +00:00
|
|
|
SetWindowLongPtrW(hwndDlg, DWLP_USER, (LONG_PTR)NULL);
|
2019-01-20 20:59:45 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_HSCROLL:
|
|
|
|
if (pPageData)
|
2019-01-20 23:25:17 +00:00
|
|
|
{
|
2019-01-20 20:59:45 +00:00
|
|
|
OnHScroll(pPageData, hwndDlg, wParam, lParam);
|
2019-01-20 23:25:17 +00:00
|
|
|
PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_COMMAND:
|
|
|
|
switch (LOWORD(wParam))
|
|
|
|
{
|
|
|
|
case 9504:
|
|
|
|
if (HIWORD(wParam) == BN_CLICKED)
|
2022-09-14 17:06:22 +00:00
|
|
|
pPageData->volumeSync = (SendDlgItemMessageW(hwndDlg, 9504, BM_GETCHECK, 0, 0) == BST_CHECKED);
|
2019-01-20 23:25:17 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 9505:
|
|
|
|
OnSetDefaults(pPageData, hwndDlg);
|
|
|
|
PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
|
|
|
|
break;
|
|
|
|
}
|
2019-01-20 14:57:34 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_NOTIFY:
|
|
|
|
return TRUE;
|
2019-01-20 20:59:45 +00:00
|
|
|
|
|
|
|
case MM_MIXM_CONTROL_CHANGE:
|
|
|
|
if (pPageData)
|
|
|
|
OnMixerControlChange(pPageData, hwndDlg);
|
|
|
|
break;
|
2019-01-20 14:57:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
INT_PTR
|
|
|
|
SpeakerVolume(
|
2019-01-20 20:59:45 +00:00
|
|
|
HWND hwndDlg)
|
2019-01-20 14:57:34 +00:00
|
|
|
{
|
2022-09-14 17:06:22 +00:00
|
|
|
PROPSHEETPAGEW psp[1];
|
|
|
|
PROPSHEETHEADERW psh;
|
2019-01-20 14:57:34 +00:00
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
ZeroMemory(&psh, sizeof(PROPSHEETHEADERW));
|
|
|
|
psh.dwSize = sizeof(PROPSHEETHEADERW);
|
2019-01-20 14:57:34 +00:00
|
|
|
psh.dwFlags = PSH_PROPSHEETPAGE;
|
2019-01-20 20:59:45 +00:00
|
|
|
psh.hwndParent = hwndDlg;
|
2019-01-20 14:57:34 +00:00
|
|
|
psh.hInstance = hApplet;
|
2022-08-27 14:01:55 +00:00
|
|
|
psh.pszCaption = MAKEINTRESOURCE(IDS_SPEAKER_VOLUME);
|
2022-09-14 17:06:22 +00:00
|
|
|
psh.nPages = _countof(psp);
|
2019-01-20 14:57:34 +00:00
|
|
|
psh.nStartPage = 0;
|
|
|
|
psh.ppsp = psp;
|
|
|
|
|
2019-01-20 20:59:45 +00:00
|
|
|
InitPropSheetPage(&psp[0], IDD_MULTICHANNEL, SpeakerVolumeDlgProc);
|
2019-01-20 23:25:17 +00:00
|
|
|
psp[0].dwFlags |= PSP_USETITLE;
|
2022-08-27 14:01:55 +00:00
|
|
|
psp[0].hInstance = hApplet;
|
|
|
|
psp[0].pszTitle = MAKEINTRESOURCE(IDS_SPEAKER_VOLUME);
|
2019-01-20 14:57:34 +00:00
|
|
|
|
2022-09-14 17:06:22 +00:00
|
|
|
return (LONG)(PropertySheetW(&psh) != -1);
|
2019-01-20 14:57:34 +00:00
|
|
|
}
|