reactos/base/applications/sndvol32/misc.c

412 lines
11 KiB
C

/*
* ReactOS Sound Volume Control
* Copyright (C) 2004-2005 Thomas Weidenmueller
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
/*
* COPYRIGHT: See COPYING in the top level directory
* PROJECT: ReactOS Sound Volume Control
* FILE: base/applications/sndvol32/misc.c
* PROGRAMMERS: Thomas Weidenmueller <w3seek@reactos.com>
*/
#include "sndvol32.h"
#include <winreg.h>
static INT
LengthOfStrResource(IN HINSTANCE hInst,
IN UINT uID)
{
HRSRC hrSrc;
HGLOBAL hRes;
LPWSTR lpName, lpStr;
if (hInst == NULL)
{
return -1;
}
/* There are always blocks of 16 strings */
lpName = (LPWSTR)MAKEINTRESOURCE((uID >> 4) + 1);
/* Find the string table block */
if ((hrSrc = FindResourceW(hInst,
lpName,
(LPWSTR)RT_STRING)) &&
(hRes = LoadResource(hInst,
hrSrc)) &&
(lpStr = LockResource(hRes)))
{
UINT x;
/* Find the string we're looking for */
uID &= 0xF; /* position in the block, same as % 16 */
for (x = 0; x < uID; x++)
{
lpStr += (*lpStr) + 1;
}
/* Found the string */
return (int)(*lpStr);
}
return -1;
}
INT
AllocAndLoadString(OUT LPWSTR *lpTarget,
IN HINSTANCE hInst,
IN UINT uID)
{
INT ln;
ln = LengthOfStrResource(hInst,
uID);
if (ln++ > 0)
{
(*lpTarget) = (LPWSTR)LocalAlloc(LMEM_FIXED,
ln * sizeof(WCHAR));
if ((*lpTarget) != NULL)
{
INT Ret = LoadStringW(hInst,
uID,
*lpTarget,
ln);
if (!Ret)
{
LocalFree((HLOCAL)(*lpTarget));
}
return Ret;
}
}
return 0;
}
DWORD
LoadAndFormatString(IN HINSTANCE hInstance,
IN UINT uID,
OUT LPWSTR *lpTarget,
...)
{
DWORD Ret = 0;
LPWSTR lpFormat;
va_list lArgs;
if (AllocAndLoadString(&lpFormat,
hInstance,
uID) > 0)
{
va_start(lArgs, lpTarget);
/* let's use FormatMessage to format it because it has the ability to
allocate memory automatically */
Ret = FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_STRING,
lpFormat,
0,
0,
(LPWSTR)lpTarget,
0,
&lArgs);
va_end(lArgs);
LocalFree((HLOCAL)lpFormat);
}
return Ret;
}
static const TCHAR AppRegSettings[] = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Applets\\Volume Control");
static const TCHAR AppOptionsKey[] = TEXT("Options");
static const TCHAR LineStatesValue[] = TEXT("LineStates");
static const TCHAR StyleValue[] = TEXT("Style");
HKEY hAppSettingsKey = NULL;
BOOL
InitAppConfig(VOID)
{
return RegCreateKeyEx(HKEY_CURRENT_USER,
AppRegSettings,
0,
NULL,
REG_OPTION_NON_VOLATILE,
KEY_READ | KEY_WRITE,
NULL,
&hAppSettingsKey,
NULL) == ERROR_SUCCESS;
}
VOID
CloseAppConfig(VOID)
{
if (hAppSettingsKey != NULL)
{
RegCloseKey(hAppSettingsKey);
hAppSettingsKey = NULL;
}
}
BOOL
LoadXYCoordWnd(IN PPREFERENCES_CONTEXT PrefContext)
{
HKEY hKey;
LONG lResult;
TCHAR DeviceMixerSettings[256];
DWORD dwData;
DWORD cbData = sizeof(dwData);
/* Append the registry key path and device name key into one single string */
StringCchPrintf(DeviceMixerSettings, _countof(DeviceMixerSettings), TEXT("%s\\%s"), AppRegSettings, PrefContext->DeviceName);
lResult = RegOpenKeyEx(HKEY_CURRENT_USER,
DeviceMixerSettings,
0,
KEY_READ,
&hKey);
if (lResult != ERROR_SUCCESS)
{
return FALSE;
}
lResult = RegQueryValueEx(hKey,
TEXT("X"),
0,
0,
(LPBYTE)&dwData,
&cbData);
if (lResult != ERROR_SUCCESS)
{
RegCloseKey(hKey);
return FALSE;
}
/* Cache the X coordinate point */
PrefContext->MixerWindow->WndPosX = dwData;
lResult = RegQueryValueEx(hKey,
TEXT("Y"),
0,
0,
(LPBYTE)&dwData,
&cbData);
if (lResult != ERROR_SUCCESS)
{
RegCloseKey(hKey);
return FALSE;
}
/* Cache the Y coordinate point */
PrefContext->MixerWindow->WndPosY = dwData;
RegCloseKey(hKey);
return TRUE;
}
BOOL
SaveXYCoordWnd(IN HWND hWnd,
IN PPREFERENCES_CONTEXT PrefContext)
{
HKEY hKey;
LONG lResult;
TCHAR DeviceMixerSettings[256];
WINDOWPLACEMENT wp;
/* Get the placement coordinate data from the window */
wp.length = sizeof(WINDOWPLACEMENT);
GetWindowPlacement(hWnd, &wp);
/* Append the registry key path and device name key into one single string */
StringCchPrintf(DeviceMixerSettings, _countof(DeviceMixerSettings), TEXT("%s\\%s"), AppRegSettings, PrefContext->DeviceName);
lResult = RegOpenKeyEx(HKEY_CURRENT_USER,
DeviceMixerSettings,
0,
KEY_SET_VALUE,
&hKey);
if (lResult != ERROR_SUCCESS)
{
return FALSE;
}
lResult = RegSetValueEx(hKey,
TEXT("X"),
0,
REG_DWORD,
(LPBYTE)&wp.rcNormalPosition.left,
sizeof(wp.rcNormalPosition.left));
if (lResult != ERROR_SUCCESS)
{
RegCloseKey(hKey);
return FALSE;
}
lResult = RegSetValueEx(hKey,
TEXT("Y"),
0,
REG_DWORD,
(LPBYTE)&wp.rcNormalPosition.top,
sizeof(wp.rcNormalPosition.top));
if (lResult != ERROR_SUCCESS)
{
RegCloseKey(hKey);
return FALSE;
}
RegCloseKey(hKey);
return TRUE;
}
BOOL
WriteLineConfig(IN LPTSTR szDeviceName,
IN LPTSTR szLineName,
IN PSNDVOL_REG_LINESTATE LineState,
IN DWORD cbSize)
{
HKEY hLineKey;
TCHAR szDevRegKey[MAX_PATH];
BOOL Ret = FALSE;
_stprintf(szDevRegKey,
TEXT("%s\\%s"),
szDeviceName,
szLineName);
if (RegCreateKeyEx(hAppSettingsKey,
szDevRegKey,
0,
NULL,
REG_OPTION_NON_VOLATILE,
KEY_READ | KEY_WRITE,
NULL,
&hLineKey,
NULL) == ERROR_SUCCESS)
{
/* now update line states */
RegSetValueEx(hLineKey, LineStatesValue, 0, REG_BINARY, (const BYTE*)LineState, cbSize);
Ret = TRUE;
RegCloseKey(hLineKey);
}
return Ret;
}
BOOL
ReadLineConfig(IN LPTSTR szDeviceName,
IN LPTSTR szLineName,
IN LPTSTR szControlName,
OUT DWORD *Flags)
{
HKEY hLineKey;
DWORD Type;
DWORD i, Size = 0;
PSNDVOL_REG_LINESTATE LineStates = NULL;
TCHAR szDevRegKey[MAX_PATH];
BOOL Ret = FALSE;
_stprintf(szDevRegKey,
TEXT("%s\\%s"),
szDeviceName,
szLineName);
if (RegCreateKeyEx(hAppSettingsKey,
szDevRegKey,
0,
NULL,
REG_OPTION_NON_VOLATILE,
KEY_READ | KEY_WRITE,
NULL,
&hLineKey,
NULL) == ERROR_SUCCESS)
{
if (RegQueryValueEx(hLineKey,
LineStatesValue,
NULL,
&Type,
NULL,
&Size) != ERROR_SUCCESS ||
Type != REG_BINARY ||
Size == 0 || (Size % sizeof(SNDVOL_REG_LINESTATE) != 0))
{
goto ExitClose;
}
LineStates = HeapAlloc(GetProcessHeap(),
HEAP_ZERO_MEMORY,
Size);
if (LineStates != NULL)
{
if (RegQueryValueEx(hLineKey,
LineStatesValue,
NULL,
&Type,
(LPBYTE)LineStates,
&Size) != ERROR_SUCCESS ||
Type != REG_BINARY ||
Size == 0 || (Size % sizeof(SNDVOL_REG_LINESTATE) != 0))
{
goto ExitClose;
}
/* try to find the control */
for (i = 0; i < Size / sizeof(SNDVOL_REG_LINESTATE); i++)
{
if (!_tcscmp(szControlName,
LineStates[i].LineName))
{
*Flags = LineStates[i].Flags;
Ret = TRUE;
break;
}
}
}
ExitClose:
HeapFree(GetProcessHeap(),
0,
LineStates);
RegCloseKey(hLineKey);
}
return Ret;
}
DWORD
GetStyleValue(VOID)
{
HKEY hOptionsKey;
DWORD dwStyle = 0, dwSize;
if (RegOpenKeyEx(hAppSettingsKey,
AppOptionsKey,
0,
KEY_READ,
&hOptionsKey) == ERROR_SUCCESS)
{
dwSize = sizeof(DWORD);
RegQueryValueEx(hOptionsKey,
StyleValue,
NULL,
NULL,
(LPBYTE)&dwStyle,
&dwSize);
RegCloseKey(hOptionsKey);
}
return dwStyle;
}