uxtheme: Check LoadedBefore registry key before loading theme.

Only save unthemed system metrics to the registry when the LoadedBefore
registry key value is '0'. This avoids saving themed system metrics
when two processes are trying to activate theming at the same time.

Signed-off-by: Zhiyi Zhang <zzhang@codeweavers.com>
Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
Zhiyi Zhang 2021-10-21 10:06:26 +08:00 committed by Alexandre Julliard
parent aefcd057b1
commit d290362c8c
2 changed files with 29 additions and 12 deletions

View File

@ -185,7 +185,7 @@ static void UXTHEME_LoadTheme(void)
lstrcpynW(szCurrentColor, pt->pszSelectedColor, ARRAY_SIZE(szCurrentColor)); lstrcpynW(szCurrentColor, pt->pszSelectedColor, ARRAY_SIZE(szCurrentColor));
lstrcpynW(szCurrentSize, pt->pszSelectedSize, ARRAY_SIZE(szCurrentSize)); lstrcpynW(szCurrentSize, pt->pszSelectedSize, ARRAY_SIZE(szCurrentSize));
MSSTYLES_SetActiveTheme(pt, FALSE); UXTHEME_SetActiveTheme(pt);
TRACE("Theme active: %s %s %s\n", debugstr_w(szCurrentTheme), TRACE("Theme active: %s %s %s\n", debugstr_w(szCurrentTheme),
debugstr_w(szCurrentColor), debugstr_w(szCurrentSize)); debugstr_w(szCurrentColor), debugstr_w(szCurrentSize));
MSSTYLES_CloseThemeFile(pt); MSSTYLES_CloseThemeFile(pt);
@ -419,27 +419,38 @@ static void UXTHEME_SaveSystemMetrics(struct system_metrics *metrics, BOOL send_
* *
* Change the current active theme * Change the current active theme
*/ */
static HRESULT UXTHEME_SetActiveTheme(PTHEME_FILE tf) HRESULT UXTHEME_SetActiveTheme(PTHEME_FILE tf)
{ {
BOOL ret, loaded_before = FALSE;
struct system_metrics metrics; struct system_metrics metrics;
DWORD size;
HKEY hKey; HKEY hKey;
WCHAR tmp[2]; WCHAR tmp[2];
HRESULT hr; HRESULT hr;
if (tf && !bThemeActive)
{
if (UXTHEME_GetSystemMetrics(&metrics))
UXTHEME_SaveUnthemedSystemMetrics(&metrics);
}
hr = MSSTYLES_SetActiveTheme(tf, TRUE);
if(FAILED(hr))
return hr;
if(tf) { if(tf) {
bThemeActive = TRUE; bThemeActive = TRUE;
lstrcpynW(szCurrentTheme, tf->szThemeFile, ARRAY_SIZE(szCurrentTheme)); lstrcpynW(szCurrentTheme, tf->szThemeFile, ARRAY_SIZE(szCurrentTheme));
lstrcpynW(szCurrentColor, tf->pszSelectedColor, ARRAY_SIZE(szCurrentColor)); lstrcpynW(szCurrentColor, tf->pszSelectedColor, ARRAY_SIZE(szCurrentColor));
lstrcpynW(szCurrentSize, tf->pszSelectedSize, ARRAY_SIZE(szCurrentSize)); lstrcpynW(szCurrentSize, tf->pszSelectedSize, ARRAY_SIZE(szCurrentSize));
ret = UXTHEME_GetSystemMetrics(&metrics);
/* Check if theming is already active */
if (!RegOpenKeyW(HKEY_CURRENT_USER, szThemeManager, &hKey))
{
size = sizeof(tmp);
if (!RegQueryValueExW(hKey, L"LoadedBefore", NULL, NULL, (BYTE *)tmp, &size))
loaded_before = (tmp[0] != '0');
else
WARN("Failed to get LoadedBefore: %d\n", GetLastError());
RegCloseKey(hKey);
}
if (loaded_before)
return MSSTYLES_SetActiveTheme(tf, FALSE);
if (ret)
UXTHEME_SaveUnthemedSystemMetrics(&metrics);
} }
else { else {
bThemeActive = FALSE; bThemeActive = FALSE;
@ -460,18 +471,20 @@ static HRESULT UXTHEME_SetActiveTheme(PTHEME_FILE tf)
(lstrlenW(szCurrentSize)+1)*sizeof(WCHAR)); (lstrlenW(szCurrentSize)+1)*sizeof(WCHAR));
RegSetValueExW(hKey, L"DllName", 0, REG_SZ, (const BYTE*)szCurrentTheme, RegSetValueExW(hKey, L"DllName", 0, REG_SZ, (const BYTE*)szCurrentTheme,
(lstrlenW(szCurrentTheme)+1)*sizeof(WCHAR)); (lstrlenW(szCurrentTheme)+1)*sizeof(WCHAR));
RegSetValueExW(hKey, L"LoadedBefore", 0, REG_SZ, (const BYTE *)tmp, sizeof(WCHAR) * 2);
} }
else { else {
RegDeleteValueW(hKey, L"ColorName"); RegDeleteValueW(hKey, L"ColorName");
RegDeleteValueW(hKey, L"SizeName"); RegDeleteValueW(hKey, L"SizeName");
RegDeleteValueW(hKey, L"DllName"); RegDeleteValueW(hKey, L"DllName");
RegDeleteValueW(hKey, L"LoadedBefore");
} }
RegCloseKey(hKey); RegCloseKey(hKey);
} }
else else
TRACE("Failed to open theme registry key\n"); TRACE("Failed to open theme registry key\n");
hr = MSSTYLES_SetActiveTheme(tf, TRUE);
if (bThemeActive) if (bThemeActive)
{ {
if (UXTHEME_GetSystemMetrics(&metrics)) if (UXTHEME_GetSystemMetrics(&metrics))

View File

@ -21,7 +21,10 @@
#ifndef __WINE_UXTHEMEDLL_H #ifndef __WINE_UXTHEMEDLL_H
#define __WINE_UXTHEMEDLL_H #define __WINE_UXTHEMEDLL_H
#include <wingdi.h>
#include <winuser.h> #include <winuser.h>
#include <uxtheme.h>
#include <msstyles.h>
typedef HANDLE HTHEMEFILE; typedef HANDLE HTHEMEFILE;
@ -99,6 +102,7 @@ BOOL WINAPI ThemeHooksInstall(void) DECLSPEC_HIDDEN;
BOOL WINAPI ThemeHooksRemove(void) DECLSPEC_HIDDEN; BOOL WINAPI ThemeHooksRemove(void) DECLSPEC_HIDDEN;
extern void UXTHEME_InitSystem(HINSTANCE hInst) DECLSPEC_HIDDEN; extern void UXTHEME_InitSystem(HINSTANCE hInst) DECLSPEC_HIDDEN;
extern HRESULT UXTHEME_SetActiveTheme(PTHEME_FILE tf) DECLSPEC_HIDDEN;
extern void UXTHEME_UninitSystem(void) DECLSPEC_HIDDEN; extern void UXTHEME_UninitSystem(void) DECLSPEC_HIDDEN;
extern struct user_api_hook user_api DECLSPEC_HIDDEN; extern struct user_api_hook user_api DECLSPEC_HIDDEN;