Fix some hackish usages of WideCharToMultiByte.
[wine.git] / programs / winecfg / theme.c
blobcdab531d2ce78250657b04f264e6e2ed9b5fcac7
1 /*
2 * Desktop Integration
3 * - Theme configuration code
4 * - User Shell Folder mapping
6 * Copyright (c) 2005 by Frank Richter
7 * Copyright (c) 2006 by Michael Jung
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
25 #include "config.h"
26 #include "wine/port.h"
28 #include <stdarg.h>
29 #include <stdlib.h>
30 #include <stdio.h>
31 #ifdef HAVE_SYS_STAT_H
32 #include <sys/stat.h>
33 #endif
34 #ifdef HAVE_UNISTD_H
35 #include <unistd.h>
36 #endif
37 #ifdef HAVE_DIRECT_H
38 #include <direct.h>
39 #endif
41 #define COBJMACROS
43 #include <windows.h>
44 #include <uxtheme.h>
45 #include <tmschema.h>
46 #include <shlobj.h>
47 #include <shlwapi.h>
48 #include <wine/debug.h>
49 #include <wine/unicode.h>
51 #include "resource.h"
52 #include "winecfg.h"
54 WINE_DEFAULT_DEBUG_CHANNEL(winecfg);
56 /* UXTHEME functions not in the headers */
58 typedef struct tagTHEMENAMES
60 WCHAR szName[MAX_PATH+1];
61 WCHAR szDisplayName[MAX_PATH+1];
62 WCHAR szTooltip[MAX_PATH+1];
63 } THEMENAMES, *PTHEMENAMES;
65 typedef void* HTHEMEFILE;
66 typedef BOOL (CALLBACK *EnumThemeProc)(LPVOID lpReserved,
67 LPCWSTR pszThemeFileName,
68 LPCWSTR pszThemeName,
69 LPCWSTR pszToolTip, LPVOID lpReserved2,
70 LPVOID lpData);
72 HRESULT WINAPI EnumThemeColors (LPCWSTR pszThemeFileName, LPWSTR pszSizeName,
73 DWORD dwColorNum, PTHEMENAMES pszColorNames);
74 HRESULT WINAPI EnumThemeSizes (LPCWSTR pszThemeFileName, LPWSTR pszColorName,
75 DWORD dwSizeNum, PTHEMENAMES pszSizeNames);
76 HRESULT WINAPI ApplyTheme (HTHEMEFILE hThemeFile, char* unknown, HWND hWnd);
77 HRESULT WINAPI OpenThemeFile (LPCWSTR pszThemeFileName, LPCWSTR pszColorName,
78 LPCWSTR pszSizeName, HTHEMEFILE* hThemeFile,
79 DWORD unknown);
80 HRESULT WINAPI CloseThemeFile (HTHEMEFILE hThemeFile);
81 HRESULT WINAPI EnumThemes (LPCWSTR pszThemePath, EnumThemeProc callback,
82 LPVOID lpData);
84 /* A struct to keep both the internal and "fancy" name of a color or size */
85 typedef struct
87 WCHAR* name;
88 WCHAR* fancyName;
89 } ThemeColorOrSize;
91 /* wrapper around DSA that also keeps an item count */
92 typedef struct
94 HDSA dsa;
95 int count;
96 } WrappedDsa;
98 /* Some helper functions to deal with ThemeColorOrSize structs in WrappedDSAs */
100 static void color_or_size_dsa_add (WrappedDsa* wdsa, const WCHAR* name,
101 const WCHAR* fancyName)
103 ThemeColorOrSize item;
105 item.name = HeapAlloc (GetProcessHeap(), 0,
106 (lstrlenW (name) + 1) * sizeof(WCHAR));
107 lstrcpyW (item.name, name);
109 item.fancyName = HeapAlloc (GetProcessHeap(), 0,
110 (lstrlenW (fancyName) + 1) * sizeof(WCHAR));
111 lstrcpyW (item.fancyName, fancyName);
113 DSA_InsertItem (wdsa->dsa, wdsa->count, &item);
114 wdsa->count++;
117 static int CALLBACK dsa_destroy_callback (LPVOID p, LPVOID pData)
119 ThemeColorOrSize* item = (ThemeColorOrSize*)p;
120 HeapFree (GetProcessHeap(), 0, item->name);
121 HeapFree (GetProcessHeap(), 0, item->fancyName);
122 return 1;
125 static void free_color_or_size_dsa (WrappedDsa* wdsa)
127 DSA_DestroyCallback (wdsa->dsa, dsa_destroy_callback, NULL);
130 static void create_color_or_size_dsa (WrappedDsa* wdsa)
132 wdsa->dsa = DSA_Create (sizeof (ThemeColorOrSize), 1);
133 wdsa->count = 0;
136 static ThemeColorOrSize* color_or_size_dsa_get (WrappedDsa* wdsa, int index)
138 return (ThemeColorOrSize*)DSA_GetItemPtr (wdsa->dsa, index);
141 static int color_or_size_dsa_find (WrappedDsa* wdsa, const WCHAR* name)
143 int i = 0;
144 for (; i < wdsa->count; i++)
146 ThemeColorOrSize* item = color_or_size_dsa_get (wdsa, i);
147 if (lstrcmpiW (item->name, name) == 0) break;
149 return i;
152 /* A theme file, contains file name, display name, color and size scheme names */
153 typedef struct
155 WCHAR* themeFileName;
156 WCHAR* fancyName;
157 WrappedDsa colors;
158 WrappedDsa sizes;
159 } ThemeFile;
161 static HDSA themeFiles = NULL;
162 static int themeFilesCount = 0;
164 static int CALLBACK theme_dsa_destroy_callback (LPVOID p, LPVOID pData)
166 ThemeFile* item = (ThemeFile*)p;
167 HeapFree (GetProcessHeap(), 0, item->themeFileName);
168 HeapFree (GetProcessHeap(), 0, item->fancyName);
169 free_color_or_size_dsa (&item->colors);
170 free_color_or_size_dsa (&item->sizes);
171 return 1;
174 /* Free memory occupied by the theme list */
175 static void free_theme_files(void)
177 if (themeFiles == NULL) return;
179 DSA_DestroyCallback (themeFiles , theme_dsa_destroy_callback, NULL);
180 themeFiles = NULL;
181 themeFilesCount = 0;
184 typedef HRESULT (WINAPI * EnumTheme) (LPCWSTR, LPWSTR, DWORD, PTHEMENAMES);
186 /* fill a string list with either colors or sizes of a theme */
187 static void fill_theme_string_array (const WCHAR* filename,
188 WrappedDsa* wdsa,
189 EnumTheme enumTheme)
191 DWORD index = 0;
192 THEMENAMES names;
194 WINE_TRACE ("%s %p %p\n", wine_dbgstr_w (filename), wdsa, enumTheme);
196 while (SUCCEEDED (enumTheme (filename, NULL, index++, &names)))
198 WINE_TRACE ("%s: %s\n", wine_dbgstr_w (names.szName),
199 wine_dbgstr_w (names.szDisplayName));
200 color_or_size_dsa_add (wdsa, names.szName, names.szDisplayName);
204 /* Theme enumeration callback, adds theme to theme list */
205 static BOOL CALLBACK myEnumThemeProc (LPVOID lpReserved,
206 LPCWSTR pszThemeFileName,
207 LPCWSTR pszThemeName,
208 LPCWSTR pszToolTip,
209 LPVOID lpReserved2, LPVOID lpData)
211 ThemeFile newEntry;
213 /* fill size/color lists */
214 create_color_or_size_dsa (&newEntry.colors);
215 fill_theme_string_array (pszThemeFileName, &newEntry.colors, EnumThemeColors);
216 create_color_or_size_dsa (&newEntry.sizes);
217 fill_theme_string_array (pszThemeFileName, &newEntry.sizes, EnumThemeSizes);
219 newEntry.themeFileName = HeapAlloc (GetProcessHeap(), 0,
220 (lstrlenW (pszThemeFileName) + 1) * sizeof(WCHAR));
221 lstrcpyW (newEntry.themeFileName, pszThemeFileName);
223 newEntry.fancyName = HeapAlloc (GetProcessHeap(), 0,
224 (lstrlenW (pszThemeName) + 1) * sizeof(WCHAR));
225 lstrcpyW (newEntry.fancyName, pszThemeName);
227 /*list_add_tail (&themeFiles, &newEntry->entry);*/
228 DSA_InsertItem (themeFiles, themeFilesCount, &newEntry);
229 themeFilesCount++;
231 return TRUE;
234 /* Scan for themes */
235 static void scan_theme_files(void)
237 static const WCHAR themesSubdir[] = { '\\','T','h','e','m','e','s',0 };
238 WCHAR themesPath[MAX_PATH];
240 free_theme_files();
242 if (FAILED (SHGetFolderPathW (NULL, CSIDL_RESOURCES, NULL,
243 SHGFP_TYPE_CURRENT, themesPath))) return;
245 themeFiles = DSA_Create (sizeof (ThemeFile), 1);
246 lstrcatW (themesPath, themesSubdir);
248 EnumThemes (themesPath, myEnumThemeProc, 0);
251 /* fill the color & size combo boxes for a given theme */
252 static void fill_color_size_combos (ThemeFile* theme, HWND comboColor,
253 HWND comboSize)
255 int i;
257 SendMessageW (comboColor, CB_RESETCONTENT, 0, 0);
258 for (i = 0; i < theme->colors.count; i++)
260 ThemeColorOrSize* item = color_or_size_dsa_get (&theme->colors, i);
261 SendMessageW (comboColor, CB_ADDSTRING, 0, (LPARAM)item->fancyName);
264 SendMessageW (comboSize, CB_RESETCONTENT, 0, 0);
265 for (i = 0; i < theme->sizes.count; i++)
267 ThemeColorOrSize* item = color_or_size_dsa_get (&theme->sizes, i);
268 SendMessageW (comboSize, CB_ADDSTRING, 0, (LPARAM)item->fancyName);
272 /* Select the item of a combo box that matches a theme's color and size
273 * scheme. */
274 static void select_color_and_size (ThemeFile* theme,
275 const WCHAR* colorName, HWND comboColor,
276 const WCHAR* sizeName, HWND comboSize)
278 SendMessageW (comboColor, CB_SETCURSEL,
279 color_or_size_dsa_find (&theme->colors, colorName), 0);
280 SendMessageW (comboSize, CB_SETCURSEL,
281 color_or_size_dsa_find (&theme->sizes, sizeName), 0);
284 /* Fill theme, color and sizes combo boxes with the know themes and select
285 * the entries matching the currently active theme. */
286 static BOOL fill_theme_list (HWND comboTheme, HWND comboColor, HWND comboSize)
288 WCHAR textNoTheme[256];
289 int themeIndex = 0;
290 BOOL ret = TRUE;
291 int i;
292 WCHAR currentTheme[MAX_PATH];
293 WCHAR currentColor[MAX_PATH];
294 WCHAR currentSize[MAX_PATH];
295 ThemeFile* theme = NULL;
297 LoadStringW (GetModuleHandle (NULL), IDS_NOTHEME, textNoTheme,
298 sizeof(textNoTheme) / sizeof(WCHAR));
300 SendMessageW (comboTheme, CB_RESETCONTENT, 0, 0);
301 SendMessageW (comboTheme, CB_ADDSTRING, 0, (LPARAM)textNoTheme);
303 for (i = 0; i < themeFilesCount; i++)
305 ThemeFile* item = (ThemeFile*)DSA_GetItemPtr (themeFiles, i);
306 SendMessageW (comboTheme, CB_ADDSTRING, 0,
307 (LPARAM)item->fancyName);
310 if (IsThemeActive () && SUCCEEDED (GetCurrentThemeName (currentTheme,
311 sizeof(currentTheme) / sizeof(WCHAR),
312 currentColor, sizeof(currentColor) / sizeof(WCHAR),
313 currentSize, sizeof(currentSize) / sizeof(WCHAR))))
315 /* Determine the index of the currently active theme. */
316 BOOL found = FALSE;
317 for (i = 0; i < themeFilesCount; i++)
319 theme = (ThemeFile*)DSA_GetItemPtr (themeFiles, i);
320 if (lstrcmpiW (theme->themeFileName, currentTheme) == 0)
322 found = TRUE;
323 themeIndex = i+1;
324 break;
327 if (!found)
329 /* Current theme not found?... add to the list, then... */
330 WINE_TRACE("Theme %s not in list of enumerated themes\n",
331 wine_dbgstr_w (currentTheme));
332 myEnumThemeProc (NULL, currentTheme, currentTheme,
333 currentTheme, NULL, NULL);
334 themeIndex = themeFilesCount;
335 theme = (ThemeFile*)DSA_GetItemPtr (themeFiles,
336 themeFilesCount-1);
338 fill_color_size_combos (theme, comboColor, comboSize);
339 select_color_and_size (theme, currentColor, comboColor,
340 currentSize, comboSize);
342 else
344 /* No theme selected */
345 ret = FALSE;
348 SendMessageW (comboTheme, CB_SETCURSEL, themeIndex, 0);
349 return ret;
352 /* Update the color & size combo boxes when the selection of the theme
353 * combo changed. Selects the current color and size scheme if the theme
354 * is currently active, otherwise the first color and size. */
355 static BOOL update_color_and_size (int themeIndex, HWND comboColor,
356 HWND comboSize)
358 if (themeIndex == 0)
360 return FALSE;
362 else
364 WCHAR currentTheme[MAX_PATH];
365 WCHAR currentColor[MAX_PATH];
366 WCHAR currentSize[MAX_PATH];
367 ThemeFile* theme =
368 (ThemeFile*)DSA_GetItemPtr (themeFiles, themeIndex - 1);
370 fill_color_size_combos (theme, comboColor, comboSize);
372 if ((SUCCEEDED (GetCurrentThemeName (currentTheme,
373 sizeof(currentTheme) / sizeof(WCHAR),
374 currentColor, sizeof(currentColor) / sizeof(WCHAR),
375 currentSize, sizeof(currentSize) / sizeof(WCHAR))))
376 && (lstrcmpiW (currentTheme, theme->themeFileName) == 0))
378 select_color_and_size (theme, currentColor, comboColor,
379 currentSize, comboSize);
381 else
383 SendMessageW (comboColor, CB_SETCURSEL, 0, 0);
384 SendMessageW (comboSize, CB_SETCURSEL, 0, 0);
387 return TRUE;
390 /* Apply a theme from a given theme, color and size combo box item index. */
391 static void do_apply_theme (int themeIndex, int colorIndex, int sizeIndex)
393 static char b[] = "\0";
395 if (themeIndex == 0)
397 /* no theme */
398 ApplyTheme (NULL, b, NULL);
400 else
402 ThemeFile* theme =
403 (ThemeFile*)DSA_GetItemPtr (themeFiles, themeIndex-1);
404 const WCHAR* themeFileName = theme->themeFileName;
405 const WCHAR* colorName = NULL;
406 const WCHAR* sizeName = NULL;
407 HTHEMEFILE hTheme;
408 ThemeColorOrSize* item;
410 item = color_or_size_dsa_get (&theme->colors, colorIndex);
411 colorName = item->name;
413 item = color_or_size_dsa_get (&theme->sizes, sizeIndex);
414 sizeName = item->name;
416 if (SUCCEEDED (OpenThemeFile (themeFileName, colorName, sizeName,
417 &hTheme, 0)))
419 ApplyTheme (hTheme, b, NULL);
420 CloseThemeFile (hTheme);
422 else
424 ApplyTheme (NULL, b, NULL);
429 int updating_ui;
430 BOOL theme_dirty;
432 static void enable_size_and_color_controls (HWND dialog, BOOL enable)
434 EnableWindow (GetDlgItem (dialog, IDC_THEME_COLORCOMBO), enable);
435 EnableWindow (GetDlgItem (dialog, IDC_THEME_COLORTEXT), enable);
436 EnableWindow (GetDlgItem (dialog, IDC_THEME_SIZECOMBO), enable);
437 EnableWindow (GetDlgItem (dialog, IDC_THEME_SIZETEXT), enable);
440 static void init_dialog (HWND dialog)
442 updating_ui = TRUE;
444 scan_theme_files();
445 if (!fill_theme_list (GetDlgItem (dialog, IDC_THEME_THEMECOMBO),
446 GetDlgItem (dialog, IDC_THEME_COLORCOMBO),
447 GetDlgItem (dialog, IDC_THEME_SIZECOMBO)))
449 SendMessageW (GetDlgItem (dialog, IDC_THEME_COLORCOMBO), CB_SETCURSEL, (WPARAM)-1, 0);
450 SendMessageW (GetDlgItem (dialog, IDC_THEME_SIZECOMBO), CB_SETCURSEL, (WPARAM)-1, 0);
451 enable_size_and_color_controls (dialog, FALSE);
453 else
455 enable_size_and_color_controls (dialog, TRUE);
457 theme_dirty = FALSE;
459 SendDlgItemMessageW(dialog, IDC_SYSPARAM_SIZE_UD, UDM_SETBUDDY, (WPARAM)GetDlgItem(dialog, IDC_SYSPARAM_SIZE), 0);
460 SendDlgItemMessageW(dialog, IDC_SYSPARAM_SIZE_UD, UDM_SETRANGE, 0, MAKELONG(100, 8));
462 updating_ui = FALSE;
465 static void on_theme_changed(HWND dialog) {
466 int index = SendMessageW (GetDlgItem (dialog, IDC_THEME_THEMECOMBO),
467 CB_GETCURSEL, 0, 0);
468 if (!update_color_and_size (index, GetDlgItem (dialog, IDC_THEME_COLORCOMBO),
469 GetDlgItem (dialog, IDC_THEME_SIZECOMBO)))
471 SendMessageW (GetDlgItem (dialog, IDC_THEME_COLORCOMBO), CB_SETCURSEL, (WPARAM)-1, 0);
472 SendMessageW (GetDlgItem (dialog, IDC_THEME_SIZECOMBO), CB_SETCURSEL, (WPARAM)-1, 0);
473 enable_size_and_color_controls (dialog, FALSE);
475 else
477 enable_size_and_color_controls (dialog, TRUE);
479 theme_dirty = TRUE;
482 static void apply_theme(HWND dialog)
484 int themeIndex, colorIndex, sizeIndex;
486 if (!theme_dirty) return;
488 themeIndex = SendMessageW (GetDlgItem (dialog, IDC_THEME_THEMECOMBO),
489 CB_GETCURSEL, 0, 0);
490 colorIndex = SendMessageW (GetDlgItem (dialog, IDC_THEME_COLORCOMBO),
491 CB_GETCURSEL, 0, 0);
492 sizeIndex = SendMessageW (GetDlgItem (dialog, IDC_THEME_SIZECOMBO),
493 CB_GETCURSEL, 0, 0);
495 do_apply_theme (themeIndex, colorIndex, sizeIndex);
496 theme_dirty = FALSE;
499 static struct
501 int sm_idx, color_idx;
502 const char *color_reg;
503 int size;
504 COLORREF color;
505 LOGFONTW lf;
506 } metrics[] =
508 {-1, COLOR_BTNFACE, "ButtonFace" }, /* IDC_SYSPARAMS_BUTTON */
509 {-1, COLOR_BTNTEXT, "ButtonText" }, /* IDC_SYSPARAMS_BUTTON_TEXT */
510 {-1, COLOR_BACKGROUND, "Background" }, /* IDC_SYSPARAMS_DESKTOP */
511 {SM_CXMENUSIZE, COLOR_MENU, "Menu" }, /* IDC_SYSPARAMS_MENU */
512 {-1, COLOR_MENUTEXT, "MenuText" }, /* IDC_SYSPARAMS_MENU_TEXT */
513 {SM_CXVSCROLL, COLOR_SCROLLBAR, "Scrollbar" }, /* IDC_SYSPARAMS_SCROLLBAR */
514 {-1, COLOR_HIGHLIGHT, "Hilight" }, /* IDC_SYSPARAMS_SELECTION */
515 {-1, COLOR_HIGHLIGHTTEXT, "HilightText" }, /* IDC_SYSPARAMS_SELECTION_TEXT */
516 {-1, COLOR_INFOBK, "InfoWindow" }, /* IDC_SYSPARAMS_TOOLTIP */
517 {-1, COLOR_INFOTEXT, "InfoText" }, /* IDC_SYSPARAMS_TOOLTIP_TEXT */
518 {-1, COLOR_WINDOW, "Window" }, /* IDC_SYSPARAMS_WINDOW */
519 {-1, COLOR_WINDOWTEXT, "WindowText" }, /* IDC_SYSPARAMS_WINDOW_TEXT */
520 {SM_CXSIZE, COLOR_ACTIVECAPTION, "ActiveTitle" }, /* IDC_SYSPARAMS_ACTIVE_TITLE */
521 {-1, COLOR_CAPTIONTEXT, "TitleText" }, /* IDC_SYSPARAMS_ACTIVE_TITLE_TEXT */
522 {-1, COLOR_INACTIVECAPTION, "InactiveTitle" }, /* IDC_SYSPARAMS_INACTIVE_TITLE */
523 {-1, COLOR_INACTIVECAPTIONTEXT,"InactiveTitleText" }, /* IDC_SYSPARAMS_INACTIVE_TITLE_TEXT */
524 {-1, -1, "MsgBoxText" }, /* IDC_SYSPARAMS_MSGBOX_TEXT */
525 {-1, COLOR_APPWORKSPACE, "AppWorkSpace" }, /* IDC_SYSPARAMS_APPWORKSPACE */
526 {-1, COLOR_WINDOWFRAME, "WindowFrame" }, /* IDC_SYSPARAMS_WINDOW_FRAME */
527 {-1, COLOR_ACTIVEBORDER, "ActiveBorder" }, /* IDC_SYSPARAMS_ACTIVE_BORDER */
528 {-1, COLOR_INACTIVEBORDER, "InactiveBorder" }, /* IDC_SYSPARAMS_INACTIVE_BORDER */
529 {-1, COLOR_BTNSHADOW, "ButtonShadow" }, /* IDC_SYSPARAMS_BUTTON_SHADOW */
530 {-1, COLOR_GRAYTEXT, "GrayText" }, /* IDC_SYSPARAMS_GRAY_TEXT */
531 {-1, COLOR_BTNHILIGHT, "ButtonHilight" }, /* IDC_SYSPARAMS_BUTTON_HILIGHT */
532 {-1, COLOR_3DDKSHADOW, "ButtonDkShadow" }, /* IDC_SYSPARAMS_BUTTON_DARK_SHADOW */
533 {-1, COLOR_3DLIGHT, "ButtonLight" }, /* IDC_SYSPARAMS_BUTTON_LIGHT */
534 {-1, COLOR_ALTERNATEBTNFACE, "ButtonAlternateFace" }, /* IDC_SYSPARAMS_BUTTON_ALTERNATE */
535 {-1, COLOR_HOTLIGHT, "HotTrackingColor" }, /* IDC_SYSPARAMS_HOT_TRACKING */
536 {-1, COLOR_GRADIENTACTIVECAPTION, "GradientActiveTitle" }, /* IDC_SYSPARAMS_ACTIVE_TITLE_GRADIENT */
537 {-1, COLOR_GRADIENTINACTIVECAPTION, "GradientInactiveTitle" }, /* IDC_SYSPARAMS_INACTIVE_TITLE_GRADIENT */
538 {-1, COLOR_MENUHILIGHT, "MenuHilight" }, /* IDC_SYSPARAMS_MENU_HILIGHT */
539 {-1, COLOR_MENUBAR, "MenuBar" }, /* IDC_SYSPARAMS_MENUBAR */
542 static void save_sys_color(int idx, COLORREF clr)
544 char buffer[13];
546 sprintf(buffer, "%d %d %d", GetRValue (clr), GetGValue (clr), GetBValue (clr));
547 set_reg_key(HKEY_CURRENT_USER, "Control Panel\\Colors", metrics[idx].color_reg, buffer);
550 static void set_color_from_theme(WCHAR *keyName, COLORREF color)
552 char *keyNameA = NULL;
553 int keyNameSize=0, i=0;
555 keyNameSize = WideCharToMultiByte(CP_ACP, 0, keyName, -1, keyNameA, 0, NULL, NULL);
556 keyNameA = HeapAlloc(GetProcessHeap(), 0, keyNameSize);
557 WideCharToMultiByte(CP_ACP, 0, keyName, -1, keyNameA, keyNameSize, NULL, NULL);
559 for (i=0; i<sizeof(metrics)/sizeof(metrics[0]); i++)
561 if (lstrcmpiA(metrics[i].color_reg, keyNameA)==0)
563 metrics[i].color = color;
564 save_sys_color(i, color);
565 break;
568 HeapFree(GetProcessHeap(), 0, keyNameA);
571 static void do_parse_theme(WCHAR *file)
573 static const WCHAR colorSect[] = {
574 'C','o','n','t','r','o','l',' ','P','a','n','e','l','\\',
575 'C','o','l','o','r','s',0};
576 WCHAR keyName[MAX_PATH], keyNameValue[MAX_PATH];
577 WCHAR *keyNamePtr = NULL;
578 char *keyNameValueA = NULL;
579 int keyNameValueSize = 0;
580 int red = 0, green = 0, blue = 0;
581 COLORREF color;
583 WINE_TRACE("%s\n", wine_dbgstr_w(file));
585 GetPrivateProfileStringW(colorSect, NULL, NULL, keyName,
586 MAX_PATH*sizeof(WCHAR), file);
588 keyNamePtr = keyName;
589 while (*keyNamePtr!=0) {
590 GetPrivateProfileStringW(colorSect, keyNamePtr, NULL, keyNameValue,
591 MAX_PATH*sizeof(WCHAR), file);
593 keyNameValueSize = WideCharToMultiByte(CP_ACP, 0, keyNameValue, -1,
594 keyNameValueA, 0, NULL, NULL);
595 keyNameValueA = HeapAlloc(GetProcessHeap(), 0, keyNameValueSize);
596 WideCharToMultiByte(CP_ACP, 0, keyNameValue, -1, keyNameValueA, keyNameValueSize, NULL, NULL);
598 WINE_TRACE("parsing key: %s with value: %s\n",
599 wine_dbgstr_w(keyNamePtr), wine_dbgstr_w(keyNameValue));
601 sscanf(keyNameValueA, "%d %d %d", &red, &green, &blue);
603 color = RGB((BYTE)red, (BYTE)green, (BYTE)blue);
605 HeapFree(GetProcessHeap(), 0, keyNameValueA);
607 set_color_from_theme(keyNamePtr, color);
609 keyNamePtr+=lstrlenW(keyNamePtr);
610 keyNamePtr++;
614 static void on_theme_install(HWND dialog)
616 static const WCHAR filterMask[] = {0,'*','.','m','s','s','t','y','l','e','s',';',
617 '*','.','t','h','e','m','e',0,0};
618 static const WCHAR themeExt[] = {'.','T','h','e','m','e',0};
619 const int filterMaskLen = sizeof(filterMask)/sizeof(filterMask[0]);
620 OPENFILENAMEW ofn;
621 WCHAR filetitle[MAX_PATH];
622 WCHAR file[MAX_PATH];
623 WCHAR filter[100];
624 WCHAR title[100];
626 LoadStringW (GetModuleHandle (NULL), IDS_THEMEFILE,
627 filter, sizeof (filter) / sizeof (filter[0]) - filterMaskLen);
628 memcpy (filter + lstrlenW (filter), filterMask,
629 filterMaskLen * sizeof (WCHAR));
630 LoadStringW (GetModuleHandle (NULL), IDS_THEMEFILE_SELECT,
631 title, sizeof (title) / sizeof (title[0]));
633 ofn.lStructSize = sizeof(OPENFILENAMEW);
634 ofn.hwndOwner = 0;
635 ofn.hInstance = 0;
636 ofn.lpstrFilter = filter;
637 ofn.lpstrCustomFilter = NULL;
638 ofn.nMaxCustFilter = 0;
639 ofn.nFilterIndex = 0;
640 ofn.lpstrFile = file;
641 ofn.lpstrFile[0] = '\0';
642 ofn.nMaxFile = sizeof(file)/sizeof(filetitle[0]);
643 ofn.lpstrFileTitle = filetitle;
644 ofn.lpstrFileTitle[0] = '\0';
645 ofn.nMaxFileTitle = sizeof(filetitle)/sizeof(filetitle[0]);
646 ofn.lpstrInitialDir = NULL;
647 ofn.lpstrTitle = title;
648 ofn.Flags = OFN_FILEMUSTEXIST | OFN_PATHMUSTEXIST | OFN_HIDEREADONLY;
649 ofn.nFileOffset = 0;
650 ofn.nFileExtension = 0;
651 ofn.lpstrDefExt = NULL;
652 ofn.lCustData = 0;
653 ofn.lpfnHook = NULL;
654 ofn.lpTemplateName = NULL;
656 if (GetOpenFileNameW(&ofn))
658 static const WCHAR themesSubdir[] = { '\\','T','h','e','m','e','s',0 };
659 static const WCHAR backslash[] = { '\\',0 };
660 WCHAR themeFilePath[MAX_PATH];
661 SHFILEOPSTRUCTW shfop;
663 if (FAILED (SHGetFolderPathW (NULL, CSIDL_RESOURCES|CSIDL_FLAG_CREATE, NULL,
664 SHGFP_TYPE_CURRENT, themeFilePath))) return;
666 if (lstrcmpiW(PathFindExtensionW(filetitle), themeExt)==0)
668 do_parse_theme(file);
669 SendMessage(GetParent(dialog), PSM_CHANGED, 0, 0);
670 return;
673 PathRemoveExtensionW (filetitle);
675 /* Construct path into which the theme file goes */
676 lstrcatW (themeFilePath, themesSubdir);
677 lstrcatW (themeFilePath, backslash);
678 lstrcatW (themeFilePath, filetitle);
680 /* Create the directory */
681 SHCreateDirectoryExW (dialog, themeFilePath, NULL);
683 /* Append theme file name itself */
684 lstrcatW (themeFilePath, backslash);
685 lstrcatW (themeFilePath, PathFindFileNameW (file));
686 /* SHFileOperation() takes lists as input, so double-nullterminate */
687 themeFilePath[lstrlenW (themeFilePath)+1] = 0;
688 file[lstrlenW (file)+1] = 0;
690 /* Do the copying */
691 WINE_TRACE("copying: %s -> %s\n", wine_dbgstr_w (file),
692 wine_dbgstr_w (themeFilePath));
693 shfop.hwnd = dialog;
694 shfop.wFunc = FO_COPY;
695 shfop.pFrom = file;
696 shfop.pTo = themeFilePath;
697 shfop.fFlags = FOF_NOCONFIRMMKDIR;
698 if (SHFileOperationW (&shfop) == 0)
700 scan_theme_files();
701 if (!fill_theme_list (GetDlgItem (dialog, IDC_THEME_THEMECOMBO),
702 GetDlgItem (dialog, IDC_THEME_COLORCOMBO),
703 GetDlgItem (dialog, IDC_THEME_SIZECOMBO)))
705 SendMessageW (GetDlgItem (dialog, IDC_THEME_COLORCOMBO), CB_SETCURSEL, (WPARAM)-1, 0);
706 SendMessageW (GetDlgItem (dialog, IDC_THEME_SIZECOMBO), CB_SETCURSEL, (WPARAM)-1, 0);
707 enable_size_and_color_controls (dialog, FALSE);
709 else
711 enable_size_and_color_controls (dialog, TRUE);
714 else
715 WINE_TRACE("copy operation failed\n");
717 else WINE_TRACE("user cancelled\n");
720 /* Information about symbolic link targets of certain User Shell Folders. */
721 struct ShellFolderInfo {
722 int nFolder;
723 char szLinkTarget[FILENAME_MAX]; /* in unix locale */
726 static struct ShellFolderInfo asfiInfo[] = {
727 { CSIDL_DESKTOP, "" },
728 { CSIDL_PERSONAL, "" },
729 { CSIDL_MYPICTURES, "" },
730 { CSIDL_MYMUSIC, "" },
731 { CSIDL_MYVIDEO, "" }
734 static struct ShellFolderInfo *psfiSelected = NULL;
736 #define NUM_ELEMS(x) (sizeof(x)/sizeof(*(x)))
738 /* create a unicode string from a string in Unix locale */
739 static WCHAR *strdupU2W(const char *unix_str)
741 WCHAR *unicode_str;
742 int lenW;
744 lenW = MultiByteToWideChar(CP_UNIXCP, 0, unix_str, -1, NULL, 0);
745 unicode_str = HeapAlloc(GetProcessHeap(), 0, lenW * sizeof(WCHAR));
746 if (unicode_str)
747 MultiByteToWideChar(CP_UNIXCP, 0, unix_str, -1, unicode_str, lenW);
748 return unicode_str;
751 static void init_shell_folder_listview_headers(HWND dialog) {
752 LVCOLUMN listColumn;
753 RECT viewRect;
754 char szShellFolder[64] = "Shell Folder";
755 char szLinksTo[64] = "Links to";
756 int width;
758 LoadString(GetModuleHandle(NULL), IDS_SHELL_FOLDER, szShellFolder, sizeof(szShellFolder));
759 LoadString(GetModuleHandle(NULL), IDS_LINKS_TO, szLinksTo, sizeof(szLinksTo));
761 GetClientRect(GetDlgItem(dialog, IDC_LIST_SFPATHS), &viewRect);
762 width = (viewRect.right - viewRect.left) / 4;
764 listColumn.mask = LVCF_TEXT | LVCF_WIDTH | LVCF_SUBITEM;
765 listColumn.pszText = szShellFolder;
766 listColumn.cchTextMax = lstrlen(listColumn.pszText);
767 listColumn.cx = width;
769 SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_INSERTCOLUMN, 0, (LPARAM) &listColumn);
771 listColumn.pszText = szLinksTo;
772 listColumn.cchTextMax = lstrlen(listColumn.pszText);
773 listColumn.cx = viewRect.right - viewRect.left - width - 1;
775 SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_INSERTCOLUMN, 1, (LPARAM) &listColumn);
778 /* Reads the currently set shell folder symbol link targets into asfiInfo. */
779 static void read_shell_folder_link_targets(void) {
780 WCHAR wszPath[MAX_PATH];
781 HRESULT hr;
782 int i;
784 for (i=0; i<NUM_ELEMS(asfiInfo); i++) {
785 asfiInfo[i].szLinkTarget[0] = '\0';
786 hr = SHGetFolderPathW(NULL, asfiInfo[i].nFolder|CSIDL_FLAG_DONT_VERIFY, NULL,
787 SHGFP_TYPE_CURRENT, wszPath);
788 if (SUCCEEDED(hr)) {
789 char *pszUnixPath = wine_get_unix_file_name(wszPath);
790 if (pszUnixPath) {
791 struct stat statPath;
792 if (!lstat(pszUnixPath, &statPath) && S_ISLNK(statPath.st_mode)) {
793 int cLen = readlink(pszUnixPath, asfiInfo[i].szLinkTarget, FILENAME_MAX-1);
794 if (cLen >= 0) asfiInfo[i].szLinkTarget[cLen] = '\0';
796 HeapFree(GetProcessHeap(), 0, pszUnixPath);
802 static void update_shell_folder_listview(HWND dialog) {
803 int i;
804 LVITEMW item;
805 LONG lSelected = SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_GETNEXTITEM, (WPARAM)-1,
806 MAKELPARAM(LVNI_SELECTED,0));
808 SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_DELETEALLITEMS, 0, 0);
810 for (i=0; i<NUM_ELEMS(asfiInfo); i++) {
811 WCHAR buffer[MAX_PATH];
812 HRESULT hr;
813 LPITEMIDLIST pidlCurrent;
815 /* Some acrobatic to get the localized name of the shell folder */
816 hr = SHGetFolderLocation(dialog, asfiInfo[i].nFolder, NULL, 0, &pidlCurrent);
817 if (SUCCEEDED(hr)) {
818 LPSHELLFOLDER psfParent;
819 LPCITEMIDLIST pidlLast;
820 hr = SHBindToParent(pidlCurrent, &IID_IShellFolder, (LPVOID*)&psfParent, &pidlLast);
821 if (SUCCEEDED(hr)) {
822 STRRET strRet;
823 hr = IShellFolder_GetDisplayNameOf(psfParent, pidlLast, SHGDN_FORADDRESSBAR, &strRet);
824 if (SUCCEEDED(hr)) {
825 hr = StrRetToBufW(&strRet, pidlLast, buffer, MAX_PATH);
827 IShellFolder_Release(psfParent);
829 ILFree(pidlCurrent);
832 /* If there's a dangling symlink for the current shell folder, SHGetFolderLocation
833 * will fail above. We fall back to the (non-verified) path of the shell folder. */
834 if (FAILED(hr)) {
835 hr = SHGetFolderPathW(dialog, asfiInfo[i].nFolder|CSIDL_FLAG_DONT_VERIFY, NULL,
836 SHGFP_TYPE_CURRENT, buffer);
839 item.mask = LVIF_TEXT | LVIF_PARAM;
840 item.iItem = i;
841 item.iSubItem = 0;
842 item.pszText = buffer;
843 item.lParam = (LPARAM)&asfiInfo[i];
844 SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_INSERTITEMW, 0, (LPARAM)&item);
846 item.mask = LVIF_TEXT;
847 item.iItem = i;
848 item.iSubItem = 1;
849 item.pszText = strdupU2W(asfiInfo[i].szLinkTarget);
850 SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_SETITEMW, 0, (LPARAM)&item);
851 HeapFree(GetProcessHeap(), 0, item.pszText);
854 /* Ensure that the previously selected item is selected again. */
855 if (lSelected >= 0) {
856 item.mask = LVIF_STATE;
857 item.state = LVIS_SELECTED;
858 item.stateMask = LVIS_SELECTED;
859 SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_SETITEMSTATE, (WPARAM)lSelected,
860 (LPARAM)&item);
864 static void on_shell_folder_selection_changed(HWND hDlg, LPNMLISTVIEW lpnm) {
865 if (lpnm->uNewState & LVIS_SELECTED) {
866 psfiSelected = (struct ShellFolderInfo *)lpnm->lParam;
867 EnableWindow(GetDlgItem(hDlg, IDC_LINK_SFPATH), 1);
868 if (strlen(psfiSelected->szLinkTarget)) {
869 WCHAR *link;
870 CheckDlgButton(hDlg, IDC_LINK_SFPATH, BST_CHECKED);
871 EnableWindow(GetDlgItem(hDlg, IDC_EDIT_SFPATH), 1);
872 EnableWindow(GetDlgItem(hDlg, IDC_BROWSE_SFPATH), 1);
873 link = strdupU2W(psfiSelected->szLinkTarget);
874 set_textW(hDlg, IDC_EDIT_SFPATH, link);
875 HeapFree(GetProcessHeap(), 0, link);
876 } else {
877 CheckDlgButton(hDlg, IDC_LINK_SFPATH, BST_UNCHECKED);
878 EnableWindow(GetDlgItem(hDlg, IDC_EDIT_SFPATH), 0);
879 EnableWindow(GetDlgItem(hDlg, IDC_BROWSE_SFPATH), 0);
880 set_text(hDlg, IDC_EDIT_SFPATH, "");
882 } else {
883 psfiSelected = NULL;
884 CheckDlgButton(hDlg, IDC_LINK_SFPATH, BST_UNCHECKED);
885 set_text(hDlg, IDC_EDIT_SFPATH, "");
886 EnableWindow(GetDlgItem(hDlg, IDC_LINK_SFPATH), 0);
887 EnableWindow(GetDlgItem(hDlg, IDC_EDIT_SFPATH), 0);
888 EnableWindow(GetDlgItem(hDlg, IDC_BROWSE_SFPATH), 0);
892 /* Keep the contents of the edit control, the listview control and the symlink
893 * information in sync. */
894 static void on_shell_folder_edit_changed(HWND hDlg) {
895 LVITEMW item;
896 WCHAR *text = get_textW(hDlg, IDC_EDIT_SFPATH);
897 LONG iSel = SendDlgItemMessage(hDlg, IDC_LIST_SFPATHS, LVM_GETNEXTITEM, -1,
898 MAKELPARAM(LVNI_SELECTED,0));
900 if (!text || !psfiSelected || iSel < 0) {
901 HeapFree(GetProcessHeap(), 0, text);
902 return;
905 WideCharToMultiByte(CP_UNIXCP, 0, text, -1,
906 psfiSelected->szLinkTarget, FILENAME_MAX, NULL, NULL);
908 item.mask = LVIF_TEXT;
909 item.iItem = iSel;
910 item.iSubItem = 1;
911 item.pszText = text;
912 SendDlgItemMessage(hDlg, IDC_LIST_SFPATHS, LVM_SETITEMW, 0, (LPARAM)&item);
914 HeapFree(GetProcessHeap(), 0, text);
916 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
919 static void apply_shell_folder_changes(void) {
920 WCHAR wszPath[MAX_PATH];
921 char szBackupPath[FILENAME_MAX], szUnixPath[FILENAME_MAX], *pszUnixPath = NULL;
922 int i, cUnixPathLen;
923 struct stat statPath;
924 HRESULT hr;
926 for (i=0; i<NUM_ELEMS(asfiInfo); i++) {
927 /* Ignore nonexistent link targets */
928 if (asfiInfo[i].szLinkTarget[0] && stat(asfiInfo[i].szLinkTarget, &statPath))
929 continue;
931 hr = SHGetFolderPathW(NULL, asfiInfo[i].nFolder|CSIDL_FLAG_CREATE, NULL,
932 SHGFP_TYPE_CURRENT, wszPath);
933 if (FAILED(hr)) continue;
935 /* Retrieve the corresponding unix path. */
936 pszUnixPath = wine_get_unix_file_name(wszPath);
937 if (!pszUnixPath) continue;
938 lstrcpyA(szUnixPath, pszUnixPath);
939 HeapFree(GetProcessHeap(), 0, pszUnixPath);
941 /* Derive name for folder backup. */
942 cUnixPathLen = lstrlenA(szUnixPath);
943 lstrcpyA(szBackupPath, szUnixPath);
944 lstrcatA(szBackupPath, ".winecfg");
946 if (lstat(szUnixPath, &statPath)) continue;
948 /* Move old folder/link out of the way. */
949 if (S_ISLNK(statPath.st_mode)) {
950 if (unlink(szUnixPath)) continue; /* Unable to remove link. */
951 } else {
952 if (!*asfiInfo[i].szLinkTarget) {
953 continue; /* We are done. Old was real folder, as new shall be. */
954 } else {
955 if (rename(szUnixPath, szBackupPath)) { /* Move folder out of the way. */
956 continue; /* Unable to move old folder. */
961 /* Create new link/folder. */
962 if (*asfiInfo[i].szLinkTarget) {
963 symlink(asfiInfo[i].szLinkTarget, szUnixPath);
964 } else {
965 /* If there's a backup folder, restore it. Else create new folder. */
966 if (!lstat(szBackupPath, &statPath) && S_ISDIR(statPath.st_mode)) {
967 rename(szBackupPath, szUnixPath);
968 } else {
969 mkdir(szUnixPath, 0777);
975 static void read_sysparams(HWND hDlg)
977 WCHAR buffer[256];
978 HWND list = GetDlgItem(hDlg, IDC_SYSPARAM_COMBO);
979 NONCLIENTMETRICSW nonclient_metrics;
980 int i, idx;
982 for (i = 0; i < sizeof(metrics) / sizeof(metrics[0]); i++)
984 LoadStringW(GetModuleHandle(NULL), i + IDC_SYSPARAMS_BUTTON, buffer,
985 sizeof(buffer) / sizeof(buffer[0]));
986 idx = SendMessageW(list, CB_ADDSTRING, 0, (LPARAM)buffer);
987 if (idx != CB_ERR) SendMessageW(list, CB_SETITEMDATA, idx, i);
989 if (metrics[i].sm_idx != -1)
990 metrics[i].size = GetSystemMetrics(metrics[i].sm_idx);
991 if (metrics[i].color_idx != -1)
992 metrics[i].color = GetSysColor(metrics[i].color_idx);
995 nonclient_metrics.cbSize = sizeof(NONCLIENTMETRICSW);
996 SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof(NONCLIENTMETRICSW), &nonclient_metrics, 0);
998 memcpy(&(metrics[IDC_SYSPARAMS_MENU_TEXT - IDC_SYSPARAMS_BUTTON].lf),
999 &(nonclient_metrics.lfMenuFont), sizeof(LOGFONTW));
1000 memcpy(&(metrics[IDC_SYSPARAMS_ACTIVE_TITLE_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1001 &(nonclient_metrics.lfCaptionFont), sizeof(LOGFONTW));
1002 memcpy(&(metrics[IDC_SYSPARAMS_TOOLTIP_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1003 &(nonclient_metrics.lfStatusFont), sizeof(LOGFONTW));
1004 memcpy(&(metrics[IDC_SYSPARAMS_MSGBOX_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1005 &(nonclient_metrics.lfMessageFont), sizeof(LOGFONTW));
1008 static void apply_sysparams(void)
1010 NONCLIENTMETRICSW nonclient_metrics;
1011 int i, cnt = 0;
1012 int colors_idx[sizeof(metrics) / sizeof(metrics[0])];
1013 COLORREF colors[sizeof(metrics) / sizeof(metrics[0])];
1015 nonclient_metrics.cbSize = sizeof(nonclient_metrics);
1016 SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof(nonclient_metrics), &nonclient_metrics, 0);
1018 nonclient_metrics.iMenuWidth = nonclient_metrics.iMenuHeight =
1019 metrics[IDC_SYSPARAMS_MENU - IDC_SYSPARAMS_BUTTON].size;
1020 nonclient_metrics.iCaptionWidth = nonclient_metrics.iCaptionHeight =
1021 metrics[IDC_SYSPARAMS_ACTIVE_TITLE - IDC_SYSPARAMS_BUTTON].size;
1022 nonclient_metrics.iScrollWidth = nonclient_metrics.iScrollHeight =
1023 metrics[IDC_SYSPARAMS_SCROLLBAR - IDC_SYSPARAMS_BUTTON].size;
1025 memcpy(&(nonclient_metrics.lfMenuFont),
1026 &(metrics[IDC_SYSPARAMS_MENU_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1027 sizeof(LOGFONTW));
1028 memcpy(&(nonclient_metrics.lfCaptionFont),
1029 &(metrics[IDC_SYSPARAMS_ACTIVE_TITLE_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1030 sizeof(LOGFONTW));
1031 memcpy(&(nonclient_metrics.lfStatusFont),
1032 &(metrics[IDC_SYSPARAMS_TOOLTIP_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1033 sizeof(LOGFONTW));
1034 memcpy(&(nonclient_metrics.lfMessageFont),
1035 &(metrics[IDC_SYSPARAMS_MSGBOX_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1036 sizeof(LOGFONTW));
1038 SystemParametersInfoW(SPI_SETNONCLIENTMETRICS, sizeof(nonclient_metrics), &nonclient_metrics,
1039 SPIF_UPDATEINIFILE | SPIF_SENDCHANGE);
1041 for (i = 0; i < sizeof(metrics) / sizeof(metrics[0]); i++)
1042 if (metrics[i].color_idx != -1)
1044 colors_idx[cnt] = metrics[i].color_idx;
1045 colors[cnt++] = metrics[i].color;
1047 SetSysColors(cnt, colors_idx, colors);
1050 static void on_sysparam_change(HWND hDlg)
1052 int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1054 index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1056 updating_ui = TRUE;
1058 EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_COLOR_TEXT), metrics[index].color_idx != -1);
1059 EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_COLOR), metrics[index].color_idx != -1);
1060 InvalidateRect(GetDlgItem(hDlg, IDC_SYSPARAM_COLOR), NULL, TRUE);
1062 EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_SIZE_TEXT), metrics[index].sm_idx != -1);
1063 EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_SIZE), metrics[index].sm_idx != -1);
1064 EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_SIZE_UD), metrics[index].sm_idx != -1);
1065 if (metrics[index].sm_idx != -1)
1066 SendDlgItemMessageW(hDlg, IDC_SYSPARAM_SIZE_UD, UDM_SETPOS, 0, MAKELONG(metrics[index].size, 0));
1067 else
1068 set_text(hDlg, IDC_SYSPARAM_SIZE, "");
1070 EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_FONT),
1071 index == IDC_SYSPARAMS_MENU_TEXT-IDC_SYSPARAMS_BUTTON ||
1072 index == IDC_SYSPARAMS_ACTIVE_TITLE_TEXT-IDC_SYSPARAMS_BUTTON ||
1073 index == IDC_SYSPARAMS_TOOLTIP_TEXT-IDC_SYSPARAMS_BUTTON ||
1074 index == IDC_SYSPARAMS_MSGBOX_TEXT-IDC_SYSPARAMS_BUTTON
1077 updating_ui = FALSE;
1080 static void on_draw_item(HWND hDlg, WPARAM wParam, LPARAM lParam)
1082 static HBRUSH black_brush = 0;
1083 LPDRAWITEMSTRUCT draw_info = (LPDRAWITEMSTRUCT)lParam;
1085 if (!black_brush) black_brush = CreateSolidBrush(0);
1087 if (draw_info->CtlID == IDC_SYSPARAM_COLOR)
1089 UINT state = DFCS_ADJUSTRECT | DFCS_BUTTONPUSH;
1091 if (draw_info->itemState & ODS_DISABLED)
1092 state |= DFCS_INACTIVE;
1093 else
1094 state |= draw_info->itemState & ODS_SELECTED ? DFCS_PUSHED : 0;
1096 DrawFrameControl(draw_info->hDC, &draw_info->rcItem, DFC_BUTTON, state);
1098 if (!(draw_info->itemState & ODS_DISABLED))
1100 HBRUSH brush;
1101 int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1103 index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1104 brush = CreateSolidBrush(metrics[index].color);
1106 InflateRect(&draw_info->rcItem, -1, -1);
1107 FrameRect(draw_info->hDC, &draw_info->rcItem, black_brush);
1108 InflateRect(&draw_info->rcItem, -1, -1);
1109 FillRect(draw_info->hDC, &draw_info->rcItem, brush);
1110 DeleteObject(brush);
1115 static void on_select_font(HWND hDlg)
1117 CHOOSEFONTW cf;
1118 int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1119 index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1121 ZeroMemory(&cf, sizeof(cf));
1122 cf.lStructSize = sizeof(CHOOSEFONTW);
1123 cf.hwndOwner = hDlg;
1124 cf.lpLogFont = &(metrics[index].lf);
1125 cf.Flags = CF_SCREENFONTS | CF_INITTOLOGFONTSTRUCT | CF_NOSCRIPTSEL;
1127 ChooseFontW(&cf);
1130 INT_PTR CALLBACK
1131 ThemeDlgProc (HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
1133 switch (uMsg) {
1134 case WM_INITDIALOG:
1135 read_shell_folder_link_targets();
1136 init_shell_folder_listview_headers(hDlg);
1137 update_shell_folder_listview(hDlg);
1138 read_sysparams(hDlg);
1139 break;
1141 case WM_DESTROY:
1142 free_theme_files();
1143 break;
1145 case WM_SHOWWINDOW:
1146 set_window_title(hDlg);
1147 break;
1149 case WM_COMMAND:
1150 switch(HIWORD(wParam)) {
1151 case CBN_SELCHANGE: {
1152 if (updating_ui) break;
1153 switch (LOWORD(wParam))
1155 case IDC_THEME_THEMECOMBO: on_theme_changed(hDlg); break;
1156 case IDC_THEME_COLORCOMBO: /* fall through */
1157 case IDC_THEME_SIZECOMBO: theme_dirty = TRUE; break;
1158 case IDC_SYSPARAM_COMBO: on_sysparam_change(hDlg); return FALSE;
1160 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1161 break;
1163 case EN_CHANGE: {
1164 if (updating_ui) break;
1165 switch (LOWORD(wParam))
1167 case IDC_EDIT_SFPATH: on_shell_folder_edit_changed(hDlg); break;
1168 case IDC_SYSPARAM_SIZE:
1170 char *text = get_text(hDlg, IDC_SYSPARAM_SIZE);
1171 int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1173 index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1174 metrics[index].size = atoi(text);
1175 HeapFree(GetProcessHeap(), 0, text);
1177 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1178 break;
1181 break;
1183 case BN_CLICKED:
1184 switch (LOWORD(wParam))
1186 case IDC_THEME_INSTALL:
1187 on_theme_install (hDlg);
1188 break;
1190 case IDC_SYSPARAM_FONT:
1191 on_select_font(hDlg);
1192 break;
1194 case IDC_BROWSE_SFPATH:
1196 WCHAR link[FILENAME_MAX];
1197 if (browse_for_unix_folder(hDlg, link)) {
1198 WideCharToMultiByte(CP_UNIXCP, 0, link, -1,
1199 psfiSelected->szLinkTarget, FILENAME_MAX,
1200 NULL, NULL);
1201 update_shell_folder_listview(hDlg);
1202 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1204 break;
1207 case IDC_LINK_SFPATH:
1208 if (IsDlgButtonChecked(hDlg, IDC_LINK_SFPATH)) {
1209 WCHAR link[FILENAME_MAX];
1210 if (browse_for_unix_folder(hDlg, link)) {
1211 WideCharToMultiByte(CP_UNIXCP, 0, link, -1,
1212 psfiSelected->szLinkTarget, FILENAME_MAX,
1213 NULL, NULL);
1214 update_shell_folder_listview(hDlg);
1215 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1216 } else {
1217 CheckDlgButton(hDlg, IDC_LINK_SFPATH, BST_UNCHECKED);
1219 } else {
1220 psfiSelected->szLinkTarget[0] = '\0';
1221 update_shell_folder_listview(hDlg);
1222 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1224 break;
1226 case IDC_SYSPARAM_COLOR:
1228 static COLORREF user_colors[16];
1229 CHOOSECOLORW c_color;
1230 int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1232 index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1234 memset(&c_color, 0, sizeof(c_color));
1235 c_color.lStructSize = sizeof(c_color);
1236 c_color.lpCustColors = user_colors;
1237 c_color.rgbResult = metrics[index].color;
1238 c_color.Flags = CC_ANYCOLOR | CC_RGBINIT;
1239 c_color.hwndOwner = hDlg;
1240 if (ChooseColorW(&c_color))
1242 metrics[index].color = c_color.rgbResult;
1243 save_sys_color(index, metrics[index].color);
1244 InvalidateRect(GetDlgItem(hDlg, IDC_SYSPARAM_COLOR), NULL, TRUE);
1245 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1247 break;
1250 break;
1252 break;
1254 case WM_NOTIFY:
1255 switch (((LPNMHDR)lParam)->code) {
1256 case PSN_KILLACTIVE: {
1257 SetWindowLongPtr(hDlg, DWLP_MSGRESULT, FALSE);
1258 break;
1260 case PSN_APPLY: {
1261 apply();
1262 apply_theme(hDlg);
1263 apply_shell_folder_changes();
1264 apply_sysparams();
1265 read_shell_folder_link_targets();
1266 update_shell_folder_listview(hDlg);
1267 SetWindowLongPtr(hDlg, DWLP_MSGRESULT, PSNRET_NOERROR);
1268 break;
1270 case LVN_ITEMCHANGED: {
1271 if (wParam == IDC_LIST_SFPATHS)
1272 on_shell_folder_selection_changed(hDlg, (LPNMLISTVIEW)lParam);
1273 break;
1275 case PSN_SETACTIVE: {
1276 init_dialog (hDlg);
1277 break;
1280 break;
1282 case WM_DRAWITEM:
1283 on_draw_item(hDlg, wParam, lParam);
1284 break;
1286 default:
1287 break;
1289 return FALSE;