[DESK] Add the current screensaver in the list if it isn't already present (#5766)

- Introduce AddScreenSaver[Item] helpers for adding screensavers in
  the list (either given by file path or by item structure).

- Rename "AddScreenSavers" to "EnumScreenSavers", and
  "ScreenSaverItem" type to "SCREEN_SAVER_ITEM".

- Ensure the stored "SCRNSAVE.EXE" value is NULL-terminated.

- Add the current screensaver, specified by the "SCRNSAVE.EXE" registry
  value, in the list if it isn't already present; otherwise select it
  in the list.

- Make the filename comparison case-insensitive.
This commit is contained in:
Hermès Bélusca-Maïto 2023-10-16 20:52:31 +02:00
parent b3a25bcf8b
commit 171941ad18
No known key found for this signature in database
GPG key ID: 3B2539C65E7B93D0

View file

@ -16,15 +16,15 @@ static const TCHAR szPreviewWndClass[] = TEXT("SSDemoParent");
typedef struct typedef struct
{ {
BOOL bIsScreenSaver; /* Is this background a wallpaper */ BOOL bIsScreenSaver; /* Is this a valid screensaver */
TCHAR szFilename[MAX_PATH]; TCHAR szFilename[MAX_PATH];
TCHAR szDisplayName[256]; TCHAR szDisplayName[256];
} ScreenSaverItem; } SCREEN_SAVER_ITEM;
typedef struct _DATA typedef struct _DATA
{ {
ScreenSaverItem ScreenSaverItems[MAX_SCREENSAVERS]; SCREEN_SAVER_ITEM ScreenSaverItems[MAX_SCREENSAVERS];
PROCESS_INFORMATION PrevWindowPi; PROCESS_INFORMATION PrevWindowPi;
int Selection; int Selection;
WNDPROC OldPreviewProc; WNDPROC OldPreviewProc;
@ -411,136 +411,149 @@ CheckRegScreenSaverIsSecure(HWND hwndDlg)
} }
static VOID static BOOL
SearchScreenSavers(HWND hwndScreenSavers, AddScreenSaverItem(
LPCTSTR pszSearchPath, _In_ HWND hwndScreenSavers,
PDATA pData) _In_ PDATA pData,
_In_ SCREEN_SAVER_ITEM* ScreenSaverItem)
{ {
WIN32_FIND_DATA fd; UINT i;
TCHAR szSearchPath[MAX_PATH];
HANDLE hFind;
ScreenSaverItem *ScreenSaverItem;
HANDLE hModule;
UINT i, ScreenSaverCount;
HRESULT hr;
ScreenSaverCount = pData->ScreenSaverCount; if (pData->ScreenSaverCount >= MAX_SCREENSAVERS)
return FALSE;
hr = StringCbCopy(szSearchPath, sizeof(szSearchPath), pszSearchPath);
if (FAILED(hr))
return;
hr = StringCbCat(szSearchPath, sizeof(szSearchPath), TEXT("\\*.scr"));
if (FAILED(hr))
return;
hFind = FindFirstFile(szSearchPath, &fd);
if (hFind == INVALID_HANDLE_VALUE)
return;
while (ScreenSaverCount < MAX_SCREENSAVERS)
{
/* Don't add any hidden screensavers */
if ((fd.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) == 0)
{
TCHAR filename[MAX_PATH];
hr = StringCbCopy(filename, sizeof(filename), pszSearchPath);
if (FAILED(hr))
{
FindClose(hFind);
return;
}
hr = StringCbCat(filename, sizeof(filename), _T("\\"));
if (FAILED(hr))
{
FindClose(hFind);
return;
}
hr = StringCbCat(filename, sizeof(filename), fd.cFileName);
if (FAILED(hr))
{
FindClose(hFind);
return;
}
ScreenSaverItem = pData->ScreenSaverItems + ScreenSaverCount;
ScreenSaverItem->bIsScreenSaver = TRUE;
hModule = LoadLibraryEx(filename,
NULL,
DONT_RESOLVE_DLL_REFERENCES | LOAD_LIBRARY_AS_DATAFILE);
if (hModule)
{
if (0 == LoadString(hModule,
1,
ScreenSaverItem->szDisplayName,
sizeof(ScreenSaverItem->szDisplayName) / sizeof(TCHAR)))
{
// If the string does not exists, copy the name of the file
hr = StringCbCopy(ScreenSaverItem->szDisplayName, sizeof(ScreenSaverItem->szDisplayName), fd.cFileName);
if (FAILED(hr))
{
FreeLibrary(hModule);
FindClose(hFind);
return;
}
ScreenSaverItem->szDisplayName[_tcslen(fd.cFileName)-4] = '\0';
}
FreeLibrary(hModule);
}
else
{
hr = StringCbCopy(ScreenSaverItem->szDisplayName, sizeof(ScreenSaverItem->szDisplayName), _T("Unknown"));
if (FAILED(hr))
{
FindClose(hFind);
return;
}
}
hr = StringCbCopy(ScreenSaverItem->szFilename, sizeof(ScreenSaverItem->szFilename), filename);
if (FAILED(hr))
{
FindClose(hFind);
return;
}
i = SendMessage(hwndScreenSavers, i = SendMessage(hwndScreenSavers,
CB_ADDSTRING, CB_ADDSTRING,
0, 0,
(LPARAM)ScreenSaverItem->szDisplayName); (LPARAM)ScreenSaverItem->szDisplayName);
if ((i == CB_ERR) || (i == CB_ERRSPACE))
return FALSE;
SendMessage(hwndScreenSavers, SendMessage(hwndScreenSavers,
CB_SETITEMDATA, CB_SETITEMDATA,
i, i,
(LPARAM)ScreenSaverCount); (LPARAM)pData->ScreenSaverCount);
ScreenSaverCount++; pData->ScreenSaverCount++;
return TRUE;
} }
if (!FindNextFile(hFind, &fd)) static BOOL
break; AddScreenSaver(
_In_ HWND hwndScreenSavers,
_In_ PDATA pData,
_In_ LPCTSTR pszFilePath,
_In_ LPCTSTR pszFileName)
{
SCREEN_SAVER_ITEM* ScreenSaverItem;
HANDLE hModule;
HRESULT hr;
if (pData->ScreenSaverCount >= MAX_SCREENSAVERS)
return FALSE;
ScreenSaverItem = pData->ScreenSaverItems + pData->ScreenSaverCount;
ScreenSaverItem->bIsScreenSaver = TRUE;
hModule = LoadLibraryEx(pszFilePath,
NULL,
DONT_RESOLVE_DLL_REFERENCES | LOAD_LIBRARY_AS_DATAFILE);
if (hModule)
{
if (LoadString(hModule,
1,
ScreenSaverItem->szDisplayName,
_countof(ScreenSaverItem->szDisplayName)) == 0)
{
/* If the string does not exist, copy the file name */
hr = StringCbCopy(ScreenSaverItem->szDisplayName,
sizeof(ScreenSaverItem->szDisplayName),
pszFileName);
if (FAILED(hr))
{
FreeLibrary(hModule);
return FALSE;
}
/* Remove the .scr extension */
ScreenSaverItem->szDisplayName[_tcslen(pszFileName)-4] = _T('\0');
}
FreeLibrary(hModule);
}
else
{
hr = StringCbCopy(ScreenSaverItem->szDisplayName,
sizeof(ScreenSaverItem->szDisplayName),
_T("Unknown"));
if (FAILED(hr))
return FALSE;
} }
FindClose(hFind); hr = StringCbCopy(ScreenSaverItem->szFilename,
sizeof(ScreenSaverItem->szFilename),
pszFilePath);
if (FAILED(hr))
return FALSE;
pData->ScreenSaverCount = ScreenSaverCount; return AddScreenSaverItem(hwndScreenSavers, pData, ScreenSaverItem);
} }
static VOID static VOID
AddScreenSavers(HWND hwndDlg, PDATA pData) SearchScreenSavers(
_In_ HWND hwndScreenSavers,
_In_ PDATA pData,
_In_ LPCTSTR pszSearchPath)
{ {
HWND hwndScreenSavers = GetDlgItem(hwndDlg, IDC_SCREENS_LIST); HRESULT hr;
WIN32_FIND_DATA fd;
HANDLE hFind;
TCHAR szFilePath[MAX_PATH];
hr = StringCbPrintf(szFilePath, sizeof(szFilePath),
TEXT("%s\\*.scr"), pszSearchPath);
if (FAILED(hr))
return;
hFind = FindFirstFile(szFilePath, &fd);
if (hFind == INVALID_HANDLE_VALUE)
return;
do
{
/* Don't add any hidden screensavers */
if (fd.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN)
continue;
if (pData->ScreenSaverCount >= MAX_SCREENSAVERS)
break;
hr = StringCbPrintf(szFilePath, sizeof(szFilePath),
TEXT("%s\\%s"), pszSearchPath, fd.cFileName);
if (FAILED(hr))
break;
if (!AddScreenSaver(hwndScreenSavers, pData, szFilePath, fd.cFileName))
break;
} while (FindNextFile(hFind, &fd));
FindClose(hFind);
}
static VOID
EnumScreenSavers(
_In_ HWND hwndScreenSavers,
_In_ PDATA pData)
{
SCREEN_SAVER_ITEM* ScreenSaverItem;
PTCHAR pBackSlash;
TCHAR szSearchPath[MAX_PATH]; TCHAR szSearchPath[MAX_PATH];
TCHAR szLocalPath[MAX_PATH]; TCHAR szLocalPath[MAX_PATH];
INT i;
ScreenSaverItem *ScreenSaverItem = NULL;
LPTSTR lpBackSlash;
/* Add the "None" item */ /* Initialize the number of list items */
pData->ScreenSaverCount = 0;
/* Add the "(None)" item */
ScreenSaverItem = pData->ScreenSaverItems; ScreenSaverItem = pData->ScreenSaverItems;
ScreenSaverItem->bIsScreenSaver = FALSE; ScreenSaverItem->bIsScreenSaver = FALSE;
@ -548,39 +561,28 @@ AddScreenSavers(HWND hwndDlg, PDATA pData)
LoadString(hApplet, LoadString(hApplet,
IDS_NONE, IDS_NONE,
ScreenSaverItem->szDisplayName, ScreenSaverItem->szDisplayName,
sizeof(ScreenSaverItem->szDisplayName) / sizeof(TCHAR)); _countof(ScreenSaverItem->szDisplayName));
i = SendMessage(hwndScreenSavers, AddScreenSaverItem(hwndScreenSavers, pData, ScreenSaverItem);
CB_ADDSTRING,
0,
(LPARAM)ScreenSaverItem->szDisplayName);
SendMessage(hwndScreenSavers, /* Add all the screensavers where the applet is stored */
CB_SETITEMDATA, GetModuleFileName(hApplet, szLocalPath, _countof(szLocalPath));
i, pBackSlash = _tcsrchr(szLocalPath, _T('\\'));
(LPARAM)0); if (pBackSlash != NULL)
// Initialize number of items into the list
pData->ScreenSaverCount = 1;
// Add all the screensavers where the applet is stored.
GetModuleFileName(hApplet, szLocalPath, MAX_PATH);
lpBackSlash = _tcsrchr(szLocalPath, _T('\\'));
if (lpBackSlash != NULL)
{ {
*lpBackSlash = '\0'; *pBackSlash = _T('\0');
SearchScreenSavers(hwndScreenSavers, szLocalPath, pData); SearchScreenSavers(hwndScreenSavers, pData, szLocalPath);
} }
// Add all the screensavers in the C:\ReactOS\System32 directory. /* Add all the screensavers in the C:\ReactOS\System32 directory */
GetSystemDirectory(szSearchPath, MAX_PATH); GetSystemDirectory(szSearchPath, _countof(szSearchPath));
if (lpBackSlash != NULL && _tcsicmp(szSearchPath, szLocalPath) != 0) if (pBackSlash != NULL && _tcsicmp(szSearchPath, szLocalPath) != 0)
SearchScreenSavers(hwndScreenSavers, szSearchPath, pData); SearchScreenSavers(hwndScreenSavers, pData, szSearchPath);
// Add all the screensavers in the C:\ReactOS directory. /* Add all the screensavers in the C:\ReactOS directory */
GetWindowsDirectory(szSearchPath, MAX_PATH); GetWindowsDirectory(szSearchPath, _countof(szSearchPath));
if (lpBackSlash != NULL && _tcsicmp(szSearchPath, szLocalPath) != 0) if (pBackSlash != NULL && _tcsicmp(szSearchPath, szLocalPath) != 0)
SearchScreenSavers(hwndScreenSavers, szSearchPath, pData); SearchScreenSavers(hwndScreenSavers, pData, szSearchPath);
} }
@ -606,7 +608,7 @@ SetScreenSaver(HWND hwndDlg, PDATA pData)
/* Set the screensaver */ /* Set the screensaver */
if (pData->ScreenSaverItems[pData->Selection].bIsScreenSaver) if (pData->ScreenSaverItems[pData->Selection].bIsScreenSaver)
{ {
SIZE_T Length = _tcslen(pData->ScreenSaverItems[pData->Selection].szFilename) * sizeof(TCHAR); SIZE_T Length = (_tcslen(pData->ScreenSaverItems[pData->Selection].szFilename) + 1) * sizeof(TCHAR);
RegSetValueEx(regKey, RegSetValueEx(regKey,
_T("SCRNSAVE.EXE"), _T("SCRNSAVE.EXE"),
0, 0,
@ -658,9 +660,9 @@ SetScreenSaver(HWND hwndDlg, PDATA pData)
static BOOL static BOOL
OnInitDialog(HWND hwndDlg, PDATA pData) OnInitDialog(HWND hwndDlg, PDATA pData)
{ {
LPTSTR lpCurSs;
HWND hwndSSCombo = GetDlgItem(hwndDlg, IDC_SCREENS_LIST); HWND hwndSSCombo = GetDlgItem(hwndDlg, IDC_SCREENS_LIST);
INT Num; LPTSTR pSsValue;
INT iCurSs;
WNDCLASS wc = {0}; WNDCLASS wc = {0};
pData = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(DATA)); pData = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(DATA));
@ -712,76 +714,67 @@ OnInitDialog(HWND hwndDlg, PDATA pData)
IDC_SCREENS_TIME, IDC_SCREENS_TIME,
UDM_SETRANGE, UDM_SETRANGE,
0, 0,
MAKELONG MAKELONG(240, 1));
((short) 240, (short) 1));
AddScreenSavers(hwndDlg, EnumScreenSavers(hwndSSCombo, pData);
pData);
CheckRegScreenSaverIsSecure(hwndDlg); CheckRegScreenSaverIsSecure(hwndDlg);
/* Set the current screensaver in the combo box */ /* Set the current screensaver in the combo box */
lpCurSs = GetCurrentScreenSaverValue(_T("SCRNSAVE.EXE")); iCurSs = 0; // Default to "(None)"
if (lpCurSs) pSsValue = GetCurrentScreenSaverValue(_T("SCRNSAVE.EXE"));
if (pSsValue)
{ {
BOOL bFound = FALSE; BOOL bFound = FALSE;
INT i; INT i;
for (i = 0; i < MAX_SCREENSAVERS; i++) /* Find whether the current screensaver is in the list */
for (i = 0; i < pData->ScreenSaverCount; i++)
{ {
if (!_tcscmp(lpCurSs, pData->ScreenSaverItems[i].szFilename)) if (!_tcsicmp(pSsValue, pData->ScreenSaverItems[i].szFilename))
{ {
bFound = TRUE; bFound = TRUE;
break; break;
} }
} }
if (!bFound)
{
/* The current screensaver is not in the list: add it */
// i = pData->ScreenSaverCount;
bFound = AddScreenSaver(hwndSSCombo, pData, pSsValue, _T("SCRNSAVE.EXE"));
if (bFound)
i = pData->ScreenSaverCount - 1;
}
HeapFree(GetProcessHeap(), 0, pSsValue);
if (bFound) if (bFound)
{ {
Num = SendMessage(hwndSSCombo, /* The current screensaver should be in the list: select it */
iCurSs = SendMessage(hwndSSCombo,
CB_FINDSTRINGEXACT, CB_FINDSTRINGEXACT,
-1, -1,
(LPARAM)pData->ScreenSaverItems[i].szDisplayName); (LPARAM)pData->ScreenSaverItems[i].szDisplayName);
if (Num != CB_ERR) if (iCurSs == CB_ERR)
SendMessage(hwndSSCombo, iCurSs = 0; // Default to "(None)"
CB_SETCURSEL,
Num,
0);
} }
else
{
SendMessage(hwndSSCombo,
CB_SETCURSEL,
0,
0);
}
HeapFree(GetProcessHeap(), 0, lpCurSs);
}
else
{
/* Set screensaver to (none) */
SendMessage(hwndSSCombo,
CB_SETCURSEL,
0,
0);
} }
SendMessage(hwndSSCombo, CB_SETCURSEL, iCurSs, 0);
/* Set the current timeout */ /* Set the current timeout */
lpCurSs = GetCurrentScreenSaverValue(_T("ScreenSaveTimeOut")); pSsValue = GetCurrentScreenSaverValue(_T("ScreenSaveTimeOut"));
if (lpCurSs) if (pSsValue)
{ {
UINT Time = _ttoi(lpCurSs); UINT Time = _ttoi(pSsValue) / 60;
Time /= 60; HeapFree(GetProcessHeap(), 0, pSsValue);
SendDlgItemMessage(hwndDlg, SendDlgItemMessage(hwndDlg,
IDC_SCREENS_TIME, IDC_SCREENS_TIME,
UDM_SETPOS32, UDM_SETPOS32,
0, 0,
Time); Time);
HeapFree(GetProcessHeap(), 0, lpCurSs);
} }
SelectionChanged(hwndDlg, pData); SelectionChanged(hwndDlg, pData);