mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
fced1c1192
When FALSE is returned, Apply button ignores validation result and becomes disabled after clicking as if settings were applied. Fix it by setting the return value to TRUE. See https://docs.microsoft.com/en-us/windows/win32/controls/psn-apply
886 lines
25 KiB
C
886 lines
25 KiB
C
#include "precomp.h"
|
|
|
|
#include <tchar.h>
|
|
#include <winreg.h>
|
|
#include <cfgmgr32.h>
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
#include "resource.h"
|
|
|
|
static HINSTANCE hInstance;
|
|
static BOOL bFoundAdapter;
|
|
static DEVINST diAdapter;
|
|
|
|
#ifdef UNICODE
|
|
typedef INT_PTR (WINAPI *PDEVICEPROPERTIES)(HWND,LPCWSTR,LPCWSTR,BOOL);
|
|
#define FUNC_DEVICEPROPERTIES "DevicePropertiesW"
|
|
#else
|
|
typedef INT_PTR (WINAPI *PDEVICEPROPERTIES)(HWND,LPCSTR,LPCSTR,BOOL);
|
|
#define FUNC_DEVICEPROPERTIES "DevicePropertiesA"
|
|
#endif
|
|
|
|
/**
|
|
* @brief
|
|
* Converts a Hardware ID (DeviceID from EnumDisplayDevices)
|
|
* to an unique Device Instance ID.
|
|
*
|
|
* @param[in] lpDeviceID
|
|
* A pointer to a null-terminated string
|
|
* containing a Hardware ID concatenated with driver key.
|
|
* e.g. "Monitor\Default_Monitor\{4D36E96E-E325-11CE-BFC1-08002BE10318}\0000"
|
|
*
|
|
* @return
|
|
* A pointer to a null-terminated string
|
|
* containing an unique Device Instance ID
|
|
* or NULL in case of error.
|
|
* e.g. "DISPLAY\DEFAULT_MONITOR\4&2ABFAA30&0&00000001&00&02"
|
|
*
|
|
* @remarks
|
|
* The caller must free the returned string with LocalFree.
|
|
*/
|
|
static LPTSTR
|
|
GetMonitorDevInstID(LPCTSTR lpDeviceID)
|
|
{
|
|
CONFIGRET cr;
|
|
DEVINST diChild;
|
|
TCHAR szProperty[256];
|
|
DWORD dwSize;
|
|
LPTSTR lpDevInstId = NULL;
|
|
|
|
if (!bFoundAdapter)
|
|
return NULL;
|
|
|
|
/* Look for child monitor devices of selected video adapter */
|
|
cr = CM_Get_Child(&diChild, diAdapter, 0);
|
|
if (cr != CR_SUCCESS)
|
|
{
|
|
DPRINT1("CM_Get_Child failed: %d\n", cr);
|
|
return NULL;
|
|
}
|
|
|
|
do
|
|
{
|
|
/* Get Hardware ID for each of them */
|
|
dwSize = sizeof(szProperty) - sizeof(TCHAR);
|
|
|
|
cr = CM_Get_DevNode_Registry_Property(diChild,
|
|
CM_DRP_HARDWAREID,
|
|
NULL,
|
|
szProperty,
|
|
&dwSize,
|
|
0);
|
|
if (cr != CR_SUCCESS)
|
|
{
|
|
DPRINT1("CM_Get_DevNode_Registry_Property failed: %d\n", cr);
|
|
continue;
|
|
}
|
|
|
|
/* Concatenate with driver key */
|
|
_tcscat(szProperty, TEXT("\\"));
|
|
dwSize = sizeof(szProperty) - sizeof(TCHAR);
|
|
dwSize -= _tcslen(szProperty) * sizeof(TCHAR);
|
|
|
|
cr = CM_Get_DevNode_Registry_Property(diChild,
|
|
CM_DRP_DRIVER,
|
|
NULL,
|
|
szProperty + _tcslen(szProperty),
|
|
&dwSize,
|
|
0);
|
|
if (cr != CR_SUCCESS)
|
|
{
|
|
DPRINT1("CM_Get_DevNode_Registry_Property failed: %d\n", cr);
|
|
continue;
|
|
}
|
|
|
|
/* If the strings match, this is our monitor device node */
|
|
if (_tcscmp(szProperty, lpDeviceID) == 0)
|
|
{
|
|
cr = CM_Get_Device_ID_Size(&dwSize,
|
|
diChild,
|
|
0);
|
|
if (cr != CR_SUCCESS)
|
|
{
|
|
DPRINT1("CM_Get_Device_ID_Size failed: %d\n", cr);
|
|
break;
|
|
}
|
|
|
|
lpDevInstId = LocalAlloc(LMEM_FIXED,
|
|
(dwSize + 1) * sizeof(TCHAR));
|
|
if (lpDevInstId == NULL)
|
|
{
|
|
DPRINT1("LocalAlloc failed\n");
|
|
break;
|
|
}
|
|
|
|
cr = CM_Get_Device_ID(diChild,
|
|
lpDevInstId,
|
|
dwSize + 1,
|
|
0);
|
|
if (cr != CR_SUCCESS)
|
|
{
|
|
DPRINT1("CM_Get_Device_ID failed: %d\n", cr);
|
|
LocalFree((HLOCAL)lpDevInstId);
|
|
lpDevInstId = NULL;
|
|
}
|
|
|
|
break;
|
|
}
|
|
} while (CM_Get_Sibling(&diChild, diChild, 0) == CR_SUCCESS);
|
|
|
|
return lpDevInstId;
|
|
}
|
|
|
|
static VOID
|
|
ShowMonitorProperties(PDESKMONITOR This)
|
|
{
|
|
HMODULE hDevMgr;
|
|
PDEVICEPROPERTIES pDeviceProperties;
|
|
LPTSTR lpDevInstID;
|
|
|
|
if (This->SelMonitor != NULL)
|
|
{
|
|
lpDevInstID = GetMonitorDevInstID(This->SelMonitor->dd.DeviceID);
|
|
if (lpDevInstID != NULL)
|
|
{
|
|
hDevMgr = LoadLibrary(TEXT("devmgr.dll"));
|
|
if (hDevMgr != NULL)
|
|
{
|
|
pDeviceProperties = (PDEVICEPROPERTIES)GetProcAddress(hDevMgr,
|
|
FUNC_DEVICEPROPERTIES);
|
|
if (pDeviceProperties != NULL)
|
|
{
|
|
pDeviceProperties(This->hwndDlg,
|
|
NULL,
|
|
lpDevInstID,
|
|
FALSE);
|
|
}
|
|
|
|
FreeLibrary(hDevMgr);
|
|
}
|
|
|
|
LocalFree((HLOCAL)lpDevInstID);
|
|
}
|
|
}
|
|
}
|
|
|
|
static VOID
|
|
UpdateMonitorSelection(PDESKMONITOR This)
|
|
{
|
|
INT i;
|
|
LPTSTR lpDevInstID = NULL;
|
|
|
|
if (This->dwMonitorCount > 1)
|
|
{
|
|
This->SelMonitor = NULL;
|
|
|
|
i = (INT)SendDlgItemMessage(This->hwndDlg,
|
|
IDC_MONITORLIST,
|
|
LB_GETCURSEL,
|
|
0,
|
|
0);
|
|
if (i >= 0)
|
|
{
|
|
This->SelMonitor = (PDESKMONINFO)SendDlgItemMessage(This->hwndDlg,
|
|
IDC_MONITORLIST,
|
|
LB_GETITEMDATA,
|
|
(WPARAM)i,
|
|
0);
|
|
}
|
|
}
|
|
else
|
|
This->SelMonitor = This->Monitors;
|
|
|
|
if (This->SelMonitor != NULL)
|
|
lpDevInstID = GetMonitorDevInstID(This->SelMonitor->dd.DeviceID);
|
|
|
|
EnableWindow(GetDlgItem(This->hwndDlg,
|
|
IDC_MONITORPROPERTIES),
|
|
lpDevInstID != NULL && lpDevInstID[0] != TEXT('\0'));
|
|
|
|
if (lpDevInstID != NULL)
|
|
LocalFree((HLOCAL)lpDevInstID);
|
|
}
|
|
|
|
static VOID
|
|
UpdatePruningControls(PDESKMONITOR This)
|
|
{
|
|
EnableWindow(GetDlgItem(This->hwndDlg,
|
|
IDC_PRUNINGCHECK),
|
|
This->bModesPruned && !This->bKeyIsReadOnly);
|
|
CheckDlgButton(This->hwndDlg,
|
|
IDC_PRUNINGCHECK,
|
|
(This->bModesPruned && This->bPruningOn) ? BST_CHECKED : BST_UNCHECKED);
|
|
}
|
|
|
|
static VOID
|
|
GetPruningSettings(PDESKMONITOR This)
|
|
{
|
|
BOOL bModesPruned = FALSE, bKeyIsReadOnly = FALSE, bPruningOn = FALSE;
|
|
|
|
if (This->DeskExtInterface != NULL)
|
|
{
|
|
This->DeskExtInterface->GetPruningMode(This->DeskExtInterface->Context,
|
|
&bModesPruned,
|
|
&bKeyIsReadOnly,
|
|
&bPruningOn);
|
|
}
|
|
|
|
/* Check the boolean values against zero before assigning it to the bitfields! */
|
|
This->bModesPruned = (bModesPruned != FALSE);
|
|
This->bKeyIsReadOnly = (bKeyIsReadOnly != FALSE);
|
|
This->bPruningOn = (bPruningOn != FALSE);
|
|
|
|
UpdatePruningControls(This);
|
|
}
|
|
|
|
static VOID
|
|
UpdateRefreshFrequencyList(PDESKMONITOR This)
|
|
{
|
|
PDEVMODEW lpCurrentMode, lpMode;
|
|
TCHAR szBuffer[64];
|
|
DWORD dwIndex;
|
|
INT i;
|
|
BOOL bHasDef = FALSE;
|
|
BOOL bAdded = FALSE;
|
|
|
|
/* Fill the refresh rate combo box */
|
|
SendDlgItemMessage(This->hwndDlg,
|
|
IDC_REFRESHRATE,
|
|
CB_RESETCONTENT,
|
|
0,
|
|
0);
|
|
|
|
lpCurrentMode = This->DeskExtInterface->GetCurrentMode(This->DeskExtInterface->Context);
|
|
dwIndex = 0;
|
|
|
|
do
|
|
{
|
|
lpMode = This->DeskExtInterface->EnumAllModes(This->DeskExtInterface->Context,
|
|
dwIndex++);
|
|
if (lpMode != NULL &&
|
|
lpMode->dmBitsPerPel == lpCurrentMode->dmBitsPerPel &&
|
|
lpMode->dmPelsWidth == lpCurrentMode->dmPelsWidth &&
|
|
lpMode->dmPelsHeight == lpCurrentMode->dmPelsHeight)
|
|
{
|
|
/* We're only interested in refresh rates for the current resolution and color depth */
|
|
|
|
if (lpMode->dmDisplayFrequency <= 1)
|
|
{
|
|
/* Default hardware frequency */
|
|
if (bHasDef)
|
|
continue;
|
|
|
|
bHasDef = TRUE;
|
|
|
|
if (!LoadString(hInstance,
|
|
IDS_USEDEFFRQUENCY,
|
|
szBuffer,
|
|
sizeof(szBuffer) / sizeof(szBuffer[0])))
|
|
{
|
|
szBuffer[0] = TEXT('\0');
|
|
}
|
|
}
|
|
else
|
|
{
|
|
TCHAR szFmt[64];
|
|
|
|
if (!LoadString(hInstance,
|
|
IDS_FREQFMT,
|
|
szFmt,
|
|
sizeof(szFmt) / sizeof(szFmt[0])))
|
|
{
|
|
szFmt[0] = TEXT('\0');
|
|
}
|
|
|
|
_sntprintf(szBuffer,
|
|
sizeof(szBuffer) / sizeof(szBuffer[0]),
|
|
szFmt,
|
|
lpMode->dmDisplayFrequency);
|
|
}
|
|
|
|
i = (INT)SendDlgItemMessage(This->hwndDlg,
|
|
IDC_REFRESHRATE,
|
|
CB_ADDSTRING,
|
|
0,
|
|
(LPARAM)szBuffer);
|
|
if (i >= 0)
|
|
{
|
|
bAdded = TRUE;
|
|
|
|
SendDlgItemMessage(This->hwndDlg,
|
|
IDC_REFRESHRATE,
|
|
CB_SETITEMDATA,
|
|
(WPARAM)i,
|
|
(LPARAM)lpMode);
|
|
|
|
if (lpMode->dmDisplayFrequency == lpCurrentMode->dmDisplayFrequency)
|
|
{
|
|
SendDlgItemMessage(This->hwndDlg,
|
|
IDC_REFRESHRATE,
|
|
CB_SETCURSEL,
|
|
(WPARAM)i,
|
|
0);
|
|
}
|
|
}
|
|
}
|
|
|
|
} while (lpMode != NULL);
|
|
|
|
EnableWindow(GetDlgItem(This->hwndDlg,
|
|
IDS_MONITORSETTINGSGROUP),
|
|
bAdded);
|
|
EnableWindow(GetDlgItem(This->hwndDlg,
|
|
IDS_REFRESHRATELABEL),
|
|
bAdded);
|
|
EnableWindow(GetDlgItem(This->hwndDlg,
|
|
IDC_REFRESHRATE),
|
|
bAdded);
|
|
|
|
GetPruningSettings(This);
|
|
}
|
|
|
|
static VOID
|
|
InitMonitorDialog(PDESKMONITOR This)
|
|
{
|
|
PDESKMONINFO pmi, pminext, *pmilink;
|
|
LPTSTR lpDeviceId;
|
|
CONFIGRET cr;
|
|
DISPLAY_DEVICE dd;
|
|
BOOL bRet;
|
|
INT i;
|
|
DWORD dwIndex;
|
|
|
|
/* Free all allocated monitors */
|
|
pmi = This->Monitors;
|
|
This->Monitors = NULL;
|
|
while (pmi != NULL)
|
|
{
|
|
pminext = pmi->Next;
|
|
LocalFree((HLOCAL)pmi);
|
|
pmi = pminext;
|
|
}
|
|
|
|
This->SelMonitor = NULL;
|
|
This->dwMonitorCount = 0;
|
|
|
|
bFoundAdapter = FALSE;
|
|
lpDeviceId = QueryDeskCplString(This->pdtobj,
|
|
RegisterClipboardFormat(DESK_EXT_DISPLAYID));
|
|
|
|
if (lpDeviceId != NULL && lpDeviceId[0] != TEXT('\0'))
|
|
{
|
|
cr = CM_Locate_DevNode(&diAdapter,
|
|
lpDeviceId,
|
|
CM_LOCATE_DEVNODE_NORMAL);
|
|
bFoundAdapter = (cr == CR_SUCCESS);
|
|
|
|
if (!bFoundAdapter)
|
|
DPRINT1("CM_Locate_DevNode failed: %d\n", cr);
|
|
}
|
|
|
|
if (This->lpDisplayDevice != NULL)
|
|
LocalFree((HLOCAL)This->lpDisplayDevice);
|
|
|
|
This->lpDisplayDevice = QueryDeskCplString(This->pdtobj,
|
|
RegisterClipboardFormat(DESK_EXT_DISPLAYDEVICE));
|
|
|
|
if (This->DeskExtInterface != NULL)
|
|
{
|
|
if (This->lpDisplayDevice != NULL)
|
|
{
|
|
/* Enumerate all monitors */
|
|
dwIndex = 0;
|
|
pmilink = &This->Monitors;
|
|
|
|
do
|
|
{
|
|
dd.cb = sizeof(dd);
|
|
bRet = EnumDisplayDevices(This->lpDisplayDevice,
|
|
dwIndex++,
|
|
&dd,
|
|
0);
|
|
if (bRet)
|
|
{
|
|
pmi = LocalAlloc(LMEM_FIXED,
|
|
sizeof(*pmi));
|
|
if (pmi != NULL)
|
|
{
|
|
CopyMemory(&pmi->dd,
|
|
&dd,
|
|
sizeof(dd));
|
|
pmi->Next = NULL;
|
|
*pmilink = pmi;
|
|
pmilink = &pmi->Next;
|
|
|
|
This->dwMonitorCount++;
|
|
}
|
|
}
|
|
} while (bRet);
|
|
}
|
|
|
|
This->lpDevModeOnInit = This->DeskExtInterface->GetCurrentMode(This->DeskExtInterface->Context);
|
|
}
|
|
else
|
|
This->lpDevModeOnInit = NULL;
|
|
|
|
This->lpSelDevMode = This->lpDevModeOnInit;
|
|
|
|
/* Setup the UI depending on how many monitors are attached */
|
|
if (This->dwMonitorCount == 0)
|
|
{
|
|
LPTSTR lpMonitorName;
|
|
|
|
/* This is a fallback, let's hope that desk.cpl can provide us with a monitor name */
|
|
lpMonitorName = QueryDeskCplString(This->pdtobj,
|
|
RegisterClipboardFormat(DESK_EXT_MONITORNAME));
|
|
|
|
SetDlgItemText(This->hwndDlg,
|
|
IDC_MONITORNAME,
|
|
lpMonitorName);
|
|
|
|
if (lpMonitorName != NULL)
|
|
LocalFree((HLOCAL)lpMonitorName);
|
|
}
|
|
else if (This->dwMonitorCount == 1)
|
|
{
|
|
This->SelMonitor = This->Monitors;
|
|
SetDlgItemText(This->hwndDlg,
|
|
IDC_MONITORNAME,
|
|
This->Monitors->dd.DeviceString);
|
|
}
|
|
else
|
|
{
|
|
SendDlgItemMessage(This->hwndDlg,
|
|
IDC_MONITORLIST,
|
|
LB_RESETCONTENT,
|
|
0,
|
|
0);
|
|
|
|
pmi = This->Monitors;
|
|
while (pmi != NULL)
|
|
{
|
|
i = (INT)SendDlgItemMessage(This->hwndDlg,
|
|
IDC_MONITORLIST,
|
|
LB_ADDSTRING,
|
|
0,
|
|
(LPARAM)pmi->dd.DeviceString);
|
|
if (i >= 0)
|
|
{
|
|
SendDlgItemMessage(This->hwndDlg,
|
|
IDC_MONITORLIST,
|
|
LB_SETITEMDATA,
|
|
(WPARAM)i,
|
|
(LPARAM)pmi);
|
|
|
|
if (This->SelMonitor == NULL)
|
|
{
|
|
SendDlgItemMessage(This->hwndDlg,
|
|
IDC_MONITORLIST,
|
|
LB_SETCURSEL,
|
|
(WPARAM)i,
|
|
0);
|
|
|
|
This->SelMonitor = pmi;
|
|
}
|
|
}
|
|
|
|
pmi = pmi->Next;
|
|
}
|
|
}
|
|
|
|
/* Show/Hide controls */
|
|
ShowWindow(GetDlgItem(This->hwndDlg,
|
|
IDC_MONITORNAME),
|
|
(This->dwMonitorCount <= 1 ? SW_SHOW : SW_HIDE));
|
|
ShowWindow(GetDlgItem(This->hwndDlg,
|
|
IDC_MONITORLIST),
|
|
(This->dwMonitorCount > 1 ? SW_SHOW : SW_HIDE));
|
|
|
|
UpdateRefreshFrequencyList(This);
|
|
UpdateMonitorSelection(This);
|
|
}
|
|
|
|
static VOID
|
|
UpdatePruningSelection(PDESKMONITOR This)
|
|
{
|
|
BOOL bPruningOn;
|
|
|
|
if (This->DeskExtInterface != NULL && This->bModesPruned && !This->bKeyIsReadOnly)
|
|
{
|
|
bPruningOn = IsDlgButtonChecked(This->hwndDlg,
|
|
IDC_PRUNINGCHECK) != BST_UNCHECKED;
|
|
|
|
if (bPruningOn != This->bPruningOn)
|
|
{
|
|
/* Tell desk.cpl to turn on/off pruning mode */
|
|
This->bPruningOn = bPruningOn;
|
|
This->DeskExtInterface->SetPruningMode(This->DeskExtInterface->Context,
|
|
bPruningOn);
|
|
|
|
/* Fill the refresh rate combobox again, we now receive a filtered
|
|
or unfiltered device mode list from desk.cpl (depending on whether
|
|
pruning is active or not) */
|
|
UpdateRefreshFrequencyList(This);
|
|
|
|
(void)PropSheet_Changed(GetParent(This->hwndDlg),
|
|
This->hwndDlg);
|
|
}
|
|
}
|
|
}
|
|
|
|
static VOID
|
|
UpdateRefreshRateSelection(PDESKMONITOR This)
|
|
{
|
|
PDEVMODEW lpCurrentDevMode;
|
|
INT i;
|
|
|
|
if (This->DeskExtInterface != NULL)
|
|
{
|
|
i = (INT)SendDlgItemMessage(This->hwndDlg,
|
|
IDC_REFRESHRATE,
|
|
CB_GETCURSEL,
|
|
0,
|
|
0);
|
|
if (i >= 0)
|
|
{
|
|
lpCurrentDevMode = This->lpSelDevMode;
|
|
This->lpSelDevMode = (PDEVMODEW)SendDlgItemMessage(This->hwndDlg,
|
|
IDC_REFRESHRATE,
|
|
CB_GETITEMDATA,
|
|
(WPARAM)i,
|
|
0);
|
|
|
|
if (This->lpSelDevMode != NULL && This->lpSelDevMode != lpCurrentDevMode)
|
|
{
|
|
This->DeskExtInterface->SetCurrentMode(This->DeskExtInterface->Context,
|
|
This->lpSelDevMode);
|
|
|
|
UpdateRefreshFrequencyList(This);
|
|
|
|
(void)PropSheet_Changed(GetParent(This->hwndDlg),
|
|
This->hwndDlg);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static LONG
|
|
ApplyMonitorChanges(PDESKMONITOR This)
|
|
{
|
|
LONG lChangeRet;
|
|
|
|
if (This->DeskExtInterface != NULL)
|
|
{
|
|
/* Change the display settings through desk.cpl */
|
|
lChangeRet = DeskCplExtDisplaySaveSettings(This->DeskExtInterface,
|
|
This->hwndDlg);
|
|
if (lChangeRet == DISP_CHANGE_SUCCESSFUL)
|
|
{
|
|
/* Save the new mode */
|
|
This->lpDevModeOnInit = This->DeskExtInterface->GetCurrentMode(This->DeskExtInterface->Context);
|
|
This->lpSelDevMode = This->lpDevModeOnInit;
|
|
return PSNRET_NOERROR;
|
|
}
|
|
else if (lChangeRet == DISP_CHANGE_RESTART)
|
|
{
|
|
/* Notify desk.cpl that the user needs to reboot */
|
|
PropSheet_RestartWindows(GetParent(This->hwndDlg));
|
|
return PSNRET_NOERROR;
|
|
}
|
|
}
|
|
|
|
InitMonitorDialog(This);
|
|
|
|
return PSNRET_INVALID_NOCHANGEPAGE;
|
|
}
|
|
|
|
static VOID
|
|
ResetMonitorChanges(PDESKMONITOR This)
|
|
{
|
|
if (This->DeskExtInterface != NULL && This->lpDevModeOnInit != NULL)
|
|
{
|
|
This->DeskExtInterface->SetCurrentMode(This->DeskExtInterface->Context,
|
|
This->lpDevModeOnInit);
|
|
}
|
|
}
|
|
|
|
static INT_PTR CALLBACK
|
|
MonitorDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
PDESKMONITOR This;
|
|
INT_PTR Ret = 0;
|
|
|
|
if (uMsg != WM_INITDIALOG)
|
|
{
|
|
This = (PDESKMONITOR)GetWindowLongPtr(hwndDlg, DWLP_USER);
|
|
}
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
This = (PDESKMONITOR)((LPCPROPSHEETPAGE)lParam)->lParam;
|
|
This->hwndDlg = hwndDlg;
|
|
SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)This);
|
|
|
|
InitMonitorDialog(This);
|
|
Ret = TRUE;
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
switch (LOWORD(wParam))
|
|
{
|
|
case IDC_MONITORPROPERTIES:
|
|
ShowMonitorProperties(This);
|
|
break;
|
|
|
|
case IDC_MONITORLIST:
|
|
if (HIWORD(wParam) == LBN_SELCHANGE)
|
|
UpdateMonitorSelection(This);
|
|
break;
|
|
|
|
case IDC_PRUNINGCHECK:
|
|
if (HIWORD(wParam) == BN_CLICKED)
|
|
UpdatePruningSelection(This);
|
|
break;
|
|
|
|
case IDC_REFRESHRATE:
|
|
if (HIWORD(wParam) == CBN_SELCHANGE)
|
|
UpdateRefreshRateSelection(This);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
NMHDR *nmh = (NMHDR *)lParam;
|
|
|
|
switch (nmh->code)
|
|
{
|
|
case PSN_APPLY:
|
|
{
|
|
SetWindowLongPtr(hwndDlg,
|
|
DWLP_MSGRESULT,
|
|
ApplyMonitorChanges(This));
|
|
Ret = TRUE;
|
|
break;
|
|
}
|
|
|
|
case PSN_RESET:
|
|
ResetMonitorChanges(This);
|
|
break;
|
|
|
|
case PSN_SETACTIVE:
|
|
UpdateRefreshFrequencyList(This);
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
return Ret;
|
|
}
|
|
|
|
static VOID
|
|
IDeskMonitor_Destroy(PDESKMONITOR This)
|
|
{
|
|
PDESKMONINFO pmi, pminext;
|
|
|
|
if (This->pdtobj != NULL)
|
|
{
|
|
IDataObject_Release(This->pdtobj);
|
|
This->pdtobj = NULL;
|
|
}
|
|
|
|
if (This->DeskExtInterface != NULL)
|
|
{
|
|
LocalFree((HLOCAL)This->DeskExtInterface);
|
|
This->DeskExtInterface = NULL;
|
|
}
|
|
|
|
if (This->lpDisplayDevice != NULL)
|
|
{
|
|
LocalFree((HLOCAL)This->lpDisplayDevice);
|
|
This->lpDisplayDevice = NULL;
|
|
}
|
|
|
|
/* Free all monitors */
|
|
pmi = This->Monitors;
|
|
This->Monitors = NULL;
|
|
while (pmi != NULL)
|
|
{
|
|
pminext = pmi->Next;
|
|
LocalFree((HLOCAL)pmi);
|
|
pmi = pminext;
|
|
}
|
|
}
|
|
|
|
ULONG
|
|
IDeskMonitor_AddRef(PDESKMONITOR This)
|
|
{
|
|
ULONG ret;
|
|
|
|
ret = InterlockedIncrement((PLONG)&This->ref);
|
|
if (ret == 1)
|
|
InterlockedIncrement(&dll_refs);
|
|
|
|
return ret;
|
|
}
|
|
|
|
ULONG
|
|
IDeskMonitor_Release(PDESKMONITOR This)
|
|
{
|
|
ULONG ret;
|
|
|
|
ret = InterlockedDecrement((PLONG)&This->ref);
|
|
if (ret == 0)
|
|
{
|
|
IDeskMonitor_Destroy(This);
|
|
InterlockedDecrement(&dll_refs);
|
|
|
|
HeapFree(GetProcessHeap(),
|
|
0,
|
|
This);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE
|
|
IDeskMonitor_QueryInterface(PDESKMONITOR This,
|
|
REFIID iid,
|
|
PVOID *pvObject)
|
|
{
|
|
*pvObject = NULL;
|
|
|
|
if (IsEqualIID(iid,
|
|
&IID_IShellPropSheetExt) ||
|
|
IsEqualIID(iid,
|
|
&IID_IUnknown))
|
|
{
|
|
*pvObject = impl_to_interface(This, IShellPropSheetExt);
|
|
}
|
|
else if (IsEqualIID(iid,
|
|
&IID_IShellExtInit))
|
|
{
|
|
*pvObject = impl_to_interface(This, IShellExtInit);
|
|
}
|
|
else if (IsEqualIID(iid,
|
|
&IID_IClassFactory))
|
|
{
|
|
*pvObject = impl_to_interface(This, IClassFactory);
|
|
}
|
|
else
|
|
{
|
|
DPRINT1("IDeskMonitor::QueryInterface(%p,%p): E_NOINTERFACE\n", iid, pvObject);
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IDeskMonitor_AddRef(This);
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT
|
|
IDeskMonitor_Initialize(PDESKMONITOR This,
|
|
LPCITEMIDLIST pidlFolder,
|
|
IDataObject *pdtobj,
|
|
HKEY hkeyProgID)
|
|
{
|
|
DPRINT1("IDeskMonitor::Initialize(%p,%p,%p)\n", pidlFolder, pdtobj, hkeyProgID);
|
|
|
|
if (pdtobj != NULL)
|
|
{
|
|
IDataObject_AddRef(pdtobj);
|
|
This->pdtobj = pdtobj;
|
|
|
|
/* Get a copy of the desk.cpl extension interface */
|
|
This->DeskExtInterface = QueryDeskCplExtInterface(This->pdtobj);
|
|
if (This->DeskExtInterface != NULL)
|
|
return S_OK;
|
|
}
|
|
|
|
return S_FALSE;
|
|
}
|
|
|
|
HRESULT
|
|
IDeskMonitor_AddPages(PDESKMONITOR This,
|
|
LPFNADDPROPSHEETPAGE pfnAddPage,
|
|
LPARAM lParam)
|
|
{
|
|
HPROPSHEETPAGE hpsp;
|
|
PROPSHEETPAGE psp;
|
|
|
|
DPRINT1("IDeskMonitor::AddPages(%p,%p)\n", pfnAddPage, lParam);
|
|
|
|
psp.dwSize = sizeof(psp);
|
|
psp.dwFlags = PSP_DEFAULT;
|
|
psp.hInstance = hInstance;
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_MONITOR);
|
|
psp.pfnDlgProc = MonitorDlgProc;
|
|
psp.lParam = (LPARAM)This;
|
|
|
|
hpsp = CreatePropertySheetPage(&psp);
|
|
if (hpsp != NULL && pfnAddPage(hpsp, lParam))
|
|
return S_OK;
|
|
|
|
return S_FALSE;
|
|
}
|
|
|
|
HRESULT
|
|
IDeskMonitor_ReplacePage(PDESKMONITOR This,
|
|
EXPPS uPageID,
|
|
LPFNADDPROPSHEETPAGE pfnReplacePage,
|
|
LPARAM lParam)
|
|
{
|
|
DPRINT1("IDeskMonitor::ReplacePage(%u,%p,%p)\n", uPageID, pfnReplacePage, lParam);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
HRESULT
|
|
IDeskMonitor_Constructor(REFIID riid,
|
|
LPVOID *ppv)
|
|
{
|
|
PDESKMONITOR This;
|
|
HRESULT hRet = E_OUTOFMEMORY;
|
|
|
|
DPRINT1("IDeskMonitor::Constructor(%p,%p)\n", riid, ppv);
|
|
|
|
This = HeapAlloc(GetProcessHeap(),
|
|
0,
|
|
sizeof(*This));
|
|
if (This != NULL)
|
|
{
|
|
ZeroMemory(This,
|
|
sizeof(*This));
|
|
|
|
IDeskMonitor_InitIface(This);
|
|
|
|
hRet = IDeskMonitor_QueryInterface(This,
|
|
riid,
|
|
ppv);
|
|
if (!SUCCEEDED(hRet))
|
|
IDeskMonitor_Release(This);
|
|
}
|
|
|
|
return hRet;
|
|
}
|
|
|
|
BOOL WINAPI
|
|
DllMain(HINSTANCE hinstDLL,
|
|
DWORD dwReason,
|
|
LPVOID lpvReserved)
|
|
{
|
|
switch (dwReason)
|
|
{
|
|
case DLL_PROCESS_ATTACH:
|
|
hInstance = hinstDLL;
|
|
DisableThreadLibraryCalls(hInstance);
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|