makefiles: Add a standard header for all makefiles to replace the common variable...
[wine] / programs / winecfg / theme.c
1 /*
2  * Desktop Integration
3  * - Theme configuration code
4  * - User Shell Folder mapping
5  *
6  * Copyright (c) 2005 by Frank Richter
7  * Copyright (c) 2006 by Michael Jung
8  *
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.
13  *
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.
18  *
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
22  *
23  */
24
25 #include "config.h"
26 #include "wine/port.h"
27
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
40
41 #define COBJMACROS
42
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>
50
51 #include "resource.h"
52 #include "winecfg.h"
53
54 WINE_DEFAULT_DEBUG_CHANNEL(winecfg);
55
56 /* UXTHEME functions not in the headers */
57
58 typedef struct tagTHEMENAMES
59 {
60     WCHAR szName[MAX_PATH+1];
61     WCHAR szDisplayName[MAX_PATH+1];
62     WCHAR szTooltip[MAX_PATH+1];
63 } THEMENAMES, *PTHEMENAMES;
64
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);
71
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);
83
84 /* A struct to keep both the internal and "fancy" name of a color or size */
85 typedef struct
86 {
87   WCHAR* name;
88   WCHAR* fancyName;
89 } ThemeColorOrSize;
90
91 /* wrapper around DSA that also keeps an item count */
92 typedef struct
93 {
94   HDSA dsa;
95   int count;
96 } WrappedDsa;
97
98 /* Some helper functions to deal with ThemeColorOrSize structs in WrappedDSAs */
99
100 static void color_or_size_dsa_add (WrappedDsa* wdsa, const WCHAR* name,
101                                    const WCHAR* fancyName)
102 {
103     ThemeColorOrSize item;
104     
105     item.name = HeapAlloc (GetProcessHeap(), 0, 
106         (lstrlenW (name) + 1) * sizeof(WCHAR));
107     lstrcpyW (item.name, name);
108
109     item.fancyName = HeapAlloc (GetProcessHeap(), 0, 
110         (lstrlenW (fancyName) + 1) * sizeof(WCHAR));
111     lstrcpyW (item.fancyName, fancyName);
112
113     DSA_InsertItem (wdsa->dsa, wdsa->count, &item);
114     wdsa->count++;
115 }
116
117 static int CALLBACK dsa_destroy_callback (LPVOID p, LPVOID pData)
118 {
119     ThemeColorOrSize* item = p;
120     HeapFree (GetProcessHeap(), 0, item->name);
121     HeapFree (GetProcessHeap(), 0, item->fancyName);
122     return 1;
123 }
124
125 static void free_color_or_size_dsa (WrappedDsa* wdsa)
126 {
127     DSA_DestroyCallback (wdsa->dsa, dsa_destroy_callback, NULL);
128 }
129
130 static void create_color_or_size_dsa (WrappedDsa* wdsa)
131 {
132     wdsa->dsa = DSA_Create (sizeof (ThemeColorOrSize), 1);
133     wdsa->count = 0;
134 }
135
136 static ThemeColorOrSize* color_or_size_dsa_get (WrappedDsa* wdsa, int index)
137 {
138     return DSA_GetItemPtr (wdsa->dsa, index);
139 }
140
141 static int color_or_size_dsa_find (WrappedDsa* wdsa, const WCHAR* name)
142 {
143     int i = 0;
144     for (; i < wdsa->count; i++)
145     {
146         ThemeColorOrSize* item = color_or_size_dsa_get (wdsa, i);
147         if (lstrcmpiW (item->name, name) == 0) break;
148     }
149     return i;
150 }
151
152 /* A theme file, contains file name, display name, color and size scheme names */
153 typedef struct
154 {
155     WCHAR* themeFileName;
156     WCHAR* fancyName;
157     WrappedDsa colors;
158     WrappedDsa sizes;
159 } ThemeFile;
160
161 static HDSA themeFiles = NULL;
162 static int themeFilesCount = 0;
163
164 static int CALLBACK theme_dsa_destroy_callback (LPVOID p, LPVOID pData)
165 {
166     ThemeFile* item = 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;
172 }
173
174 /* Free memory occupied by the theme list */
175 static void free_theme_files(void)
176 {
177     if (themeFiles == NULL) return;
178       
179     DSA_DestroyCallback (themeFiles , theme_dsa_destroy_callback, NULL);
180     themeFiles = NULL;
181     themeFilesCount = 0;
182 }
183
184 typedef HRESULT (WINAPI * EnumTheme) (LPCWSTR, LPWSTR, DWORD, PTHEMENAMES);
185
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)
190 {
191     DWORD index = 0;
192     THEMENAMES names;
193
194     WINE_TRACE ("%s %p %p\n", wine_dbgstr_w (filename), wdsa, enumTheme);
195
196     while (SUCCEEDED (enumTheme (filename, NULL, index++, &names)))
197     {
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);
201     }
202 }
203
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)
210 {
211     ThemeFile newEntry;
212
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);
218
219     newEntry.themeFileName = HeapAlloc (GetProcessHeap(), 0, 
220         (lstrlenW (pszThemeFileName) + 1) * sizeof(WCHAR));
221     lstrcpyW (newEntry.themeFileName, pszThemeFileName);
222   
223     newEntry.fancyName = HeapAlloc (GetProcessHeap(), 0, 
224         (lstrlenW (pszThemeName) + 1) * sizeof(WCHAR));
225     lstrcpyW (newEntry.fancyName, pszThemeName);
226   
227     /*list_add_tail (&themeFiles, &newEntry->entry);*/
228     DSA_InsertItem (themeFiles, themeFilesCount, &newEntry);
229     themeFilesCount++;
230
231     return TRUE;
232 }
233
234 /* Scan for themes */
235 static void scan_theme_files(void)
236 {
237     static const WCHAR themesSubdir[] = { '\\','T','h','e','m','e','s',0 };
238     WCHAR themesPath[MAX_PATH];
239
240     free_theme_files();
241
242     if (FAILED (SHGetFolderPathW (NULL, CSIDL_RESOURCES, NULL, 
243         SHGFP_TYPE_CURRENT, themesPath))) return;
244
245     themeFiles = DSA_Create (sizeof (ThemeFile), 1);
246     lstrcatW (themesPath, themesSubdir);
247
248     EnumThemes (themesPath, myEnumThemeProc, 0);
249 }
250
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)
254 {
255     int i;
256
257     SendMessageW (comboColor, CB_RESETCONTENT, 0, 0);
258     for (i = 0; i < theme->colors.count; i++)
259     {
260         ThemeColorOrSize* item = color_or_size_dsa_get (&theme->colors, i);
261         SendMessageW (comboColor, CB_ADDSTRING, 0, (LPARAM)item->fancyName);
262     }
263
264     SendMessageW (comboSize, CB_RESETCONTENT, 0, 0);
265     for (i = 0; i < theme->sizes.count; i++)
266     {
267         ThemeColorOrSize* item = color_or_size_dsa_get (&theme->sizes, i);
268         SendMessageW (comboSize, CB_ADDSTRING, 0, (LPARAM)item->fancyName);
269     }
270 }
271
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)
277 {
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);
282 }
283
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)
287 {
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;
296
297     LoadStringW (GetModuleHandle (NULL), IDS_NOTHEME, textNoTheme,
298         sizeof(textNoTheme) / sizeof(WCHAR));
299
300     SendMessageW (comboTheme, CB_RESETCONTENT, 0, 0);
301     SendMessageW (comboTheme, CB_ADDSTRING, 0, (LPARAM)textNoTheme);
302
303     for (i = 0; i < themeFilesCount; i++)
304     {
305         ThemeFile* item = DSA_GetItemPtr (themeFiles, i);
306         SendMessageW (comboTheme, CB_ADDSTRING, 0, 
307             (LPARAM)item->fancyName);
308     }
309   
310     if (IsThemeActive () && SUCCEEDED (GetCurrentThemeName (currentTheme, 
311             sizeof(currentTheme) / sizeof(WCHAR),
312             currentColor, sizeof(currentColor) / sizeof(WCHAR),
313             currentSize, sizeof(currentSize) / sizeof(WCHAR))))
314     {
315         /* Determine the index of the currently active theme. */
316         BOOL found = FALSE;
317         for (i = 0; i < themeFilesCount; i++)
318         {
319             theme = DSA_GetItemPtr (themeFiles, i);
320             if (lstrcmpiW (theme->themeFileName, currentTheme) == 0)
321             {
322                 found = TRUE;
323                 themeIndex = i+1;
324                 break;
325             }
326         }
327         if (!found)
328         {
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 = DSA_GetItemPtr (themeFiles, themeFilesCount-1);
336         }
337         fill_color_size_combos (theme, comboColor, comboSize);
338         select_color_and_size (theme, currentColor, comboColor,
339             currentSize, comboSize);
340     }
341     else
342     {
343         /* No theme selected */
344         ret = FALSE;
345     }
346
347     SendMessageW (comboTheme, CB_SETCURSEL, themeIndex, 0);
348     return ret;
349 }
350
351 /* Update the color & size combo boxes when the selection of the theme
352  * combo changed. Selects the current color and size scheme if the theme
353  * is currently active, otherwise the first color and size. */
354 static BOOL update_color_and_size (int themeIndex, HWND comboColor, 
355                                    HWND comboSize)
356 {
357     if (themeIndex == 0)
358     {
359         return FALSE;
360     }
361     else
362     {
363         WCHAR currentTheme[MAX_PATH];
364         WCHAR currentColor[MAX_PATH];
365         WCHAR currentSize[MAX_PATH];
366         ThemeFile* theme = DSA_GetItemPtr (themeFiles, themeIndex - 1);
367     
368         fill_color_size_combos (theme, comboColor, comboSize);
369       
370         if ((SUCCEEDED (GetCurrentThemeName (currentTheme, 
371             sizeof(currentTheme) / sizeof(WCHAR),
372             currentColor, sizeof(currentColor) / sizeof(WCHAR),
373             currentSize, sizeof(currentSize) / sizeof(WCHAR))))
374             && (lstrcmpiW (currentTheme, theme->themeFileName) == 0))
375         {
376             select_color_and_size (theme, currentColor, comboColor,
377                 currentSize, comboSize);
378         }
379         else
380         {
381             SendMessageW (comboColor, CB_SETCURSEL, 0, 0);
382             SendMessageW (comboSize, CB_SETCURSEL, 0, 0);
383         }
384     }
385     return TRUE;
386 }
387
388 /* Apply a theme from a given theme, color and size combo box item index. */
389 static void do_apply_theme (int themeIndex, int colorIndex, int sizeIndex)
390 {
391     static char b[] = "\0";
392
393     if (themeIndex == 0)
394     {
395         /* no theme */
396         ApplyTheme (NULL, b, NULL);
397     }
398     else
399     {
400         ThemeFile* theme = DSA_GetItemPtr (themeFiles, themeIndex-1);
401         const WCHAR* themeFileName = theme->themeFileName;
402         const WCHAR* colorName = NULL;
403         const WCHAR* sizeName = NULL;
404         HTHEMEFILE hTheme;
405         ThemeColorOrSize* item;
406     
407         item = color_or_size_dsa_get (&theme->colors, colorIndex);
408         colorName = item->name;
409         
410         item = color_or_size_dsa_get (&theme->sizes, sizeIndex);
411         sizeName = item->name;
412         
413         if (SUCCEEDED (OpenThemeFile (themeFileName, colorName, sizeName,
414             &hTheme, 0)))
415         {
416             ApplyTheme (hTheme, b, NULL);
417             CloseThemeFile (hTheme);
418         }
419         else
420         {
421             ApplyTheme (NULL, b, NULL);
422         }
423     }
424 }
425
426 int updating_ui;
427 BOOL theme_dirty;
428
429 static void enable_size_and_color_controls (HWND dialog, BOOL enable)
430 {
431     EnableWindow (GetDlgItem (dialog, IDC_THEME_COLORCOMBO), enable);
432     EnableWindow (GetDlgItem (dialog, IDC_THEME_COLORTEXT), enable);
433     EnableWindow (GetDlgItem (dialog, IDC_THEME_SIZECOMBO), enable);
434     EnableWindow (GetDlgItem (dialog, IDC_THEME_SIZETEXT), enable);
435 }
436   
437 static void init_dialog (HWND dialog)
438 {
439     updating_ui = TRUE;
440     
441     scan_theme_files();
442     if (!fill_theme_list (GetDlgItem (dialog, IDC_THEME_THEMECOMBO),
443         GetDlgItem (dialog, IDC_THEME_COLORCOMBO),
444         GetDlgItem (dialog, IDC_THEME_SIZECOMBO)))
445     {
446         SendMessageW (GetDlgItem (dialog, IDC_THEME_COLORCOMBO), CB_SETCURSEL, (WPARAM)-1, 0);
447         SendMessageW (GetDlgItem (dialog, IDC_THEME_SIZECOMBO), CB_SETCURSEL, (WPARAM)-1, 0);
448         enable_size_and_color_controls (dialog, FALSE);
449     }
450     else
451     {
452         enable_size_and_color_controls (dialog, TRUE);
453     }
454     theme_dirty = FALSE;
455
456     SendDlgItemMessageW(dialog, IDC_SYSPARAM_SIZE_UD, UDM_SETBUDDY, (WPARAM)GetDlgItem(dialog, IDC_SYSPARAM_SIZE), 0);
457     SendDlgItemMessageW(dialog, IDC_SYSPARAM_SIZE_UD, UDM_SETRANGE, 0, MAKELONG(100, 8));
458
459     updating_ui = FALSE;
460 }
461
462 static void on_theme_changed(HWND dialog) {
463     int index = SendMessageW (GetDlgItem (dialog, IDC_THEME_THEMECOMBO),
464         CB_GETCURSEL, 0, 0);
465     if (!update_color_and_size (index, GetDlgItem (dialog, IDC_THEME_COLORCOMBO),
466         GetDlgItem (dialog, IDC_THEME_SIZECOMBO)))
467     {
468         SendMessageW (GetDlgItem (dialog, IDC_THEME_COLORCOMBO), CB_SETCURSEL, -1, 0);
469         SendMessageW (GetDlgItem (dialog, IDC_THEME_SIZECOMBO), CB_SETCURSEL, -1, 0);
470         enable_size_and_color_controls (dialog, FALSE);
471     }
472     else
473     {
474         enable_size_and_color_controls (dialog, TRUE);
475     }
476     theme_dirty = TRUE;
477 }
478
479 static void apply_theme(HWND dialog)
480 {
481     int themeIndex, colorIndex, sizeIndex;
482
483     if (!theme_dirty) return;
484
485     themeIndex = SendMessageW (GetDlgItem (dialog, IDC_THEME_THEMECOMBO),
486         CB_GETCURSEL, 0, 0);
487     colorIndex = SendMessageW (GetDlgItem (dialog, IDC_THEME_COLORCOMBO),
488         CB_GETCURSEL, 0, 0);
489     sizeIndex = SendMessageW (GetDlgItem (dialog, IDC_THEME_SIZECOMBO),
490         CB_GETCURSEL, 0, 0);
491
492     do_apply_theme (themeIndex, colorIndex, sizeIndex);
493     theme_dirty = FALSE;
494 }
495
496 static struct
497 {
498     int sm_idx, color_idx;
499     const char *color_reg;
500     int size;
501     COLORREF color;
502     LOGFONTW lf;
503 } metrics[] =
504 {
505     {-1,                COLOR_BTNFACE,          "ButtonFace"    }, /* IDC_SYSPARAMS_BUTTON */
506     {-1,                COLOR_BTNTEXT,          "ButtonText"    }, /* IDC_SYSPARAMS_BUTTON_TEXT */
507     {-1,                COLOR_BACKGROUND,       "Background"    }, /* IDC_SYSPARAMS_DESKTOP */
508     {SM_CXMENUSIZE,     COLOR_MENU,             "Menu"          }, /* IDC_SYSPARAMS_MENU */
509     {-1,                COLOR_MENUTEXT,         "MenuText"      }, /* IDC_SYSPARAMS_MENU_TEXT */
510     {SM_CXVSCROLL,      COLOR_SCROLLBAR,        "Scrollbar"     }, /* IDC_SYSPARAMS_SCROLLBAR */
511     {-1,                COLOR_HIGHLIGHT,        "Hilight"       }, /* IDC_SYSPARAMS_SELECTION */
512     {-1,                COLOR_HIGHLIGHTTEXT,    "HilightText"   }, /* IDC_SYSPARAMS_SELECTION_TEXT */
513     {-1,                COLOR_INFOBK,           "InfoWindow"    }, /* IDC_SYSPARAMS_TOOLTIP */
514     {-1,                COLOR_INFOTEXT,         "InfoText"      }, /* IDC_SYSPARAMS_TOOLTIP_TEXT */
515     {-1,                COLOR_WINDOW,           "Window"        }, /* IDC_SYSPARAMS_WINDOW */
516     {-1,                COLOR_WINDOWTEXT,       "WindowText"    }, /* IDC_SYSPARAMS_WINDOW_TEXT */
517     {SM_CXSIZE,         COLOR_ACTIVECAPTION,    "ActiveTitle"   }, /* IDC_SYSPARAMS_ACTIVE_TITLE */
518     {-1,                COLOR_CAPTIONTEXT,      "TitleText"     }, /* IDC_SYSPARAMS_ACTIVE_TITLE_TEXT */
519     {-1,                COLOR_INACTIVECAPTION,  "InactiveTitle" }, /* IDC_SYSPARAMS_INACTIVE_TITLE */
520     {-1,                COLOR_INACTIVECAPTIONTEXT,"InactiveTitleText" }, /* IDC_SYSPARAMS_INACTIVE_TITLE_TEXT */
521     {-1,                -1,                     "MsgBoxText"    }, /* IDC_SYSPARAMS_MSGBOX_TEXT */
522     {-1,                COLOR_APPWORKSPACE,     "AppWorkSpace"  }, /* IDC_SYSPARAMS_APPWORKSPACE */
523     {-1,                COLOR_WINDOWFRAME,      "WindowFrame"   }, /* IDC_SYSPARAMS_WINDOW_FRAME */
524     {-1,                COLOR_ACTIVEBORDER,     "ActiveBorder"  }, /* IDC_SYSPARAMS_ACTIVE_BORDER */
525     {-1,                COLOR_INACTIVEBORDER,   "InactiveBorder" }, /* IDC_SYSPARAMS_INACTIVE_BORDER */
526     {-1,                COLOR_BTNSHADOW,        "ButtonShadow"  }, /* IDC_SYSPARAMS_BUTTON_SHADOW */
527     {-1,                COLOR_GRAYTEXT,         "GrayText"      }, /* IDC_SYSPARAMS_GRAY_TEXT */
528     {-1,                COLOR_BTNHILIGHT,       "ButtonHilight" }, /* IDC_SYSPARAMS_BUTTON_HILIGHT */
529     {-1,                COLOR_3DDKSHADOW,       "ButtonDkShadow" }, /* IDC_SYSPARAMS_BUTTON_DARK_SHADOW */
530     {-1,                COLOR_3DLIGHT,          "ButtonLight"   }, /* IDC_SYSPARAMS_BUTTON_LIGHT */
531     {-1,                COLOR_ALTERNATEBTNFACE, "ButtonAlternateFace" }, /* IDC_SYSPARAMS_BUTTON_ALTERNATE */
532     {-1,                COLOR_HOTLIGHT,         "HotTrackingColor" }, /* IDC_SYSPARAMS_HOT_TRACKING */
533     {-1,                COLOR_GRADIENTACTIVECAPTION, "GradientActiveTitle" }, /* IDC_SYSPARAMS_ACTIVE_TITLE_GRADIENT */
534     {-1,                COLOR_GRADIENTINACTIVECAPTION, "GradientInactiveTitle" }, /* IDC_SYSPARAMS_INACTIVE_TITLE_GRADIENT */
535     {-1,                COLOR_MENUHILIGHT,      "MenuHilight"   }, /* IDC_SYSPARAMS_MENU_HILIGHT */
536     {-1,                COLOR_MENUBAR,          "MenuBar"       }, /* IDC_SYSPARAMS_MENUBAR */
537 };
538
539 static void save_sys_color(int idx, COLORREF clr)
540 {
541     char buffer[13];
542
543     sprintf(buffer, "%d %d %d",  GetRValue (clr), GetGValue (clr), GetBValue (clr));
544     set_reg_key(HKEY_CURRENT_USER, "Control Panel\\Colors", metrics[idx].color_reg, buffer);
545 }
546
547 static void set_color_from_theme(WCHAR *keyName, COLORREF color)
548 {
549     char *keyNameA = NULL;
550     int keyNameSize=0, i=0;
551
552     keyNameSize = WideCharToMultiByte(CP_ACP, 0, keyName, -1, keyNameA, 0, NULL, NULL);
553     keyNameA = HeapAlloc(GetProcessHeap(), 0, keyNameSize);
554     WideCharToMultiByte(CP_ACP, 0, keyName, -1, keyNameA, keyNameSize, NULL, NULL);
555
556     for (i=0; i<sizeof(metrics)/sizeof(metrics[0]); i++)
557     {
558         if (lstrcmpiA(metrics[i].color_reg, keyNameA)==0)
559         {
560             metrics[i].color = color;
561             save_sys_color(i, color);
562             break;
563         }
564     }
565     HeapFree(GetProcessHeap(), 0, keyNameA);
566 }
567
568 static void do_parse_theme(WCHAR *file)
569 {
570     static const WCHAR colorSect[] = {
571         'C','o','n','t','r','o','l',' ','P','a','n','e','l','\\',
572         'C','o','l','o','r','s',0};
573     WCHAR keyName[MAX_PATH], keyNameValue[MAX_PATH];
574     WCHAR *keyNamePtr = NULL;
575     char *keyNameValueA = NULL;
576     int keyNameValueSize = 0;
577     int red = 0, green = 0, blue = 0;
578     COLORREF color;
579
580     WINE_TRACE("%s\n", wine_dbgstr_w(file));
581
582     GetPrivateProfileStringW(colorSect, NULL, NULL, keyName,
583                              MAX_PATH, file);
584
585     keyNamePtr = keyName;
586     while (*keyNamePtr!=0) {
587         GetPrivateProfileStringW(colorSect, keyNamePtr, NULL, keyNameValue,
588                                  MAX_PATH, file);
589
590         keyNameValueSize = WideCharToMultiByte(CP_ACP, 0, keyNameValue, -1,
591                                                keyNameValueA, 0, NULL, NULL);
592         keyNameValueA = HeapAlloc(GetProcessHeap(), 0, keyNameValueSize);
593         WideCharToMultiByte(CP_ACP, 0, keyNameValue, -1, keyNameValueA, keyNameValueSize, NULL, NULL);
594
595         WINE_TRACE("parsing key: %s with value: %s\n",
596                    wine_dbgstr_w(keyNamePtr), wine_dbgstr_w(keyNameValue));
597
598         sscanf(keyNameValueA, "%d %d %d", &red, &green, &blue);
599
600         color = RGB((BYTE)red, (BYTE)green, (BYTE)blue);
601
602         HeapFree(GetProcessHeap(), 0, keyNameValueA);
603
604         set_color_from_theme(keyNamePtr, color);
605
606         keyNamePtr+=lstrlenW(keyNamePtr);
607         keyNamePtr++;
608     }
609 }
610
611 static void on_theme_install(HWND dialog)
612 {
613   static const WCHAR filterMask[] = {0,'*','.','m','s','s','t','y','l','e','s',';',
614       '*','.','t','h','e','m','e',0,0};
615   static const WCHAR themeExt[] = {'.','T','h','e','m','e',0};
616   const int filterMaskLen = sizeof(filterMask)/sizeof(filterMask[0]);
617   OPENFILENAMEW ofn;
618   WCHAR filetitle[MAX_PATH];
619   WCHAR file[MAX_PATH];
620   WCHAR filter[100];
621   WCHAR title[100];
622
623   LoadStringW (GetModuleHandle (NULL), IDS_THEMEFILE, 
624       filter, sizeof (filter) / sizeof (filter[0]) - filterMaskLen);
625   memcpy (filter + lstrlenW (filter), filterMask, 
626       filterMaskLen * sizeof (WCHAR));
627   LoadStringW (GetModuleHandle (NULL), IDS_THEMEFILE_SELECT, 
628       title, sizeof (title) / sizeof (title[0]));
629
630   ofn.lStructSize = sizeof(OPENFILENAMEW);
631   ofn.hwndOwner = 0;
632   ofn.hInstance = 0;
633   ofn.lpstrFilter = filter;
634   ofn.lpstrCustomFilter = NULL;
635   ofn.nMaxCustFilter = 0;
636   ofn.nFilterIndex = 0;
637   ofn.lpstrFile = file;
638   ofn.lpstrFile[0] = '\0';
639   ofn.nMaxFile = sizeof(file)/sizeof(filetitle[0]);
640   ofn.lpstrFileTitle = filetitle;
641   ofn.lpstrFileTitle[0] = '\0';
642   ofn.nMaxFileTitle = sizeof(filetitle)/sizeof(filetitle[0]);
643   ofn.lpstrInitialDir = NULL;
644   ofn.lpstrTitle = title;
645   ofn.Flags = OFN_FILEMUSTEXIST | OFN_PATHMUSTEXIST | OFN_HIDEREADONLY | OFN_ENABLESIZING;
646   ofn.nFileOffset = 0;
647   ofn.nFileExtension = 0;
648   ofn.lpstrDefExt = NULL;
649   ofn.lCustData = 0;
650   ofn.lpfnHook = NULL;
651   ofn.lpTemplateName = NULL;
652
653   if (GetOpenFileNameW(&ofn))
654   {
655       static const WCHAR themesSubdir[] = { '\\','T','h','e','m','e','s',0 };
656       static const WCHAR backslash[] = { '\\',0 };
657       WCHAR themeFilePath[MAX_PATH];
658       SHFILEOPSTRUCTW shfop;
659
660       if (FAILED (SHGetFolderPathW (NULL, CSIDL_RESOURCES|CSIDL_FLAG_CREATE, NULL, 
661           SHGFP_TYPE_CURRENT, themeFilePath))) return;
662
663       if (lstrcmpiW(PathFindExtensionW(filetitle), themeExt)==0)
664       {
665           do_parse_theme(file);
666           SendMessage(GetParent(dialog), PSM_CHANGED, 0, 0);
667           return;
668       }
669
670       PathRemoveExtensionW (filetitle);
671
672       /* Construct path into which the theme file goes */
673       lstrcatW (themeFilePath, themesSubdir);
674       lstrcatW (themeFilePath, backslash);
675       lstrcatW (themeFilePath, filetitle);
676
677       /* Create the directory */
678       SHCreateDirectoryExW (dialog, themeFilePath, NULL);
679
680       /* Append theme file name itself */
681       lstrcatW (themeFilePath, backslash);
682       lstrcatW (themeFilePath, PathFindFileNameW (file));
683       /* SHFileOperation() takes lists as input, so double-nullterminate */
684       themeFilePath[lstrlenW (themeFilePath)+1] = 0;
685       file[lstrlenW (file)+1] = 0;
686
687       /* Do the copying */
688       WINE_TRACE("copying: %s -> %s\n", wine_dbgstr_w (file), 
689           wine_dbgstr_w (themeFilePath));
690       shfop.hwnd = dialog;
691       shfop.wFunc = FO_COPY;
692       shfop.pFrom = file;
693       shfop.pTo = themeFilePath;
694       shfop.fFlags = FOF_NOCONFIRMMKDIR;
695       if (SHFileOperationW (&shfop) == 0)
696       {
697           scan_theme_files();
698           if (!fill_theme_list (GetDlgItem (dialog, IDC_THEME_THEMECOMBO),
699               GetDlgItem (dialog, IDC_THEME_COLORCOMBO),
700               GetDlgItem (dialog, IDC_THEME_SIZECOMBO)))
701           {
702               SendMessageW (GetDlgItem (dialog, IDC_THEME_COLORCOMBO), CB_SETCURSEL, -1, 0);
703               SendMessageW (GetDlgItem (dialog, IDC_THEME_SIZECOMBO), CB_SETCURSEL, -1, 0);
704               enable_size_and_color_controls (dialog, FALSE);
705           }
706           else
707           {
708               enable_size_and_color_controls (dialog, TRUE);
709           }
710       }
711       else
712           WINE_TRACE("copy operation failed\n");
713   }
714   else WINE_TRACE("user cancelled\n");
715 }
716
717 /* Information about symbolic link targets of certain User Shell Folders. */
718 struct ShellFolderInfo {
719     int nFolder;
720     char szLinkTarget[FILENAME_MAX]; /* in unix locale */
721 };
722
723 static struct ShellFolderInfo asfiInfo[] = {
724     { CSIDL_DESKTOP,  "" },
725     { CSIDL_PERSONAL, "" },
726     { CSIDL_MYPICTURES, "" },
727     { CSIDL_MYMUSIC, "" },
728     { CSIDL_MYVIDEO, "" }
729 };
730
731 static struct ShellFolderInfo *psfiSelected = NULL;
732
733 #define NUM_ELEMS(x) (sizeof(x)/sizeof(*(x)))
734
735 static void init_shell_folder_listview_headers(HWND dialog) {
736     LVCOLUMN listColumn;
737     RECT viewRect;
738     char szShellFolder[64] = "Shell Folder";
739     char szLinksTo[64] = "Links to";
740     int width;
741
742     LoadString(GetModuleHandle(NULL), IDS_SHELL_FOLDER, szShellFolder, sizeof(szShellFolder));
743     LoadString(GetModuleHandle(NULL), IDS_LINKS_TO, szLinksTo, sizeof(szLinksTo));
744     
745     GetClientRect(GetDlgItem(dialog, IDC_LIST_SFPATHS), &viewRect);
746     width = (viewRect.right - viewRect.left) / 4;
747
748     listColumn.mask = LVCF_TEXT | LVCF_WIDTH | LVCF_SUBITEM;
749     listColumn.pszText = szShellFolder;
750     listColumn.cchTextMax = lstrlen(listColumn.pszText);
751     listColumn.cx = width;
752
753     SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_INSERTCOLUMN, 0, (LPARAM) &listColumn);
754
755     listColumn.pszText = szLinksTo;
756     listColumn.cchTextMax = lstrlen(listColumn.pszText);
757     listColumn.cx = viewRect.right - viewRect.left - width - 1;
758
759     SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_INSERTCOLUMN, 1, (LPARAM) &listColumn);
760 }
761
762 /* Reads the currently set shell folder symbol link targets into asfiInfo. */
763 static void read_shell_folder_link_targets(void) {
764     WCHAR wszPath[MAX_PATH];
765     HRESULT hr;
766     int i;
767    
768     for (i=0; i<NUM_ELEMS(asfiInfo); i++) {
769         asfiInfo[i].szLinkTarget[0] = '\0';
770         hr = SHGetFolderPathW(NULL, asfiInfo[i].nFolder|CSIDL_FLAG_DONT_VERIFY, NULL, 
771                               SHGFP_TYPE_CURRENT, wszPath);
772         if (SUCCEEDED(hr)) {
773             char *pszUnixPath = wine_get_unix_file_name(wszPath);
774             if (pszUnixPath) {
775                 struct stat statPath;
776                 if (!lstat(pszUnixPath, &statPath) && S_ISLNK(statPath.st_mode)) {
777                     int cLen = readlink(pszUnixPath, asfiInfo[i].szLinkTarget, FILENAME_MAX-1);
778                     if (cLen >= 0) asfiInfo[i].szLinkTarget[cLen] = '\0';
779                 }
780                 HeapFree(GetProcessHeap(), 0, pszUnixPath);
781             }
782         } 
783     }    
784 }
785
786 static void update_shell_folder_listview(HWND dialog) {
787     int i;
788     LVITEMW item;
789     LONG lSelected = SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_GETNEXTITEM, -1,
790                                         MAKELPARAM(LVNI_SELECTED,0));
791     
792     SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_DELETEALLITEMS, 0, 0);
793
794     for (i=0; i<NUM_ELEMS(asfiInfo); i++) {
795         WCHAR buffer[MAX_PATH];
796         HRESULT hr;
797         LPITEMIDLIST pidlCurrent;
798
799         /* Some acrobatic to get the localized name of the shell folder */
800         hr = SHGetFolderLocation(dialog, asfiInfo[i].nFolder, NULL, 0, &pidlCurrent);
801         if (SUCCEEDED(hr)) { 
802             LPSHELLFOLDER psfParent;
803             LPCITEMIDLIST pidlLast;
804             hr = SHBindToParent(pidlCurrent, &IID_IShellFolder, (LPVOID*)&psfParent, &pidlLast);
805             if (SUCCEEDED(hr)) {
806                 STRRET strRet;
807                 hr = IShellFolder_GetDisplayNameOf(psfParent, pidlLast, SHGDN_FORADDRESSBAR, &strRet);
808                 if (SUCCEEDED(hr)) {
809                     hr = StrRetToBufW(&strRet, pidlLast, buffer, MAX_PATH);
810                 }
811                 IShellFolder_Release(psfParent);
812             }
813             ILFree(pidlCurrent);
814         }
815
816         /* If there's a dangling symlink for the current shell folder, SHGetFolderLocation
817          * will fail above. We fall back to the (non-verified) path of the shell folder. */
818         if (FAILED(hr)) {
819             hr = SHGetFolderPathW(dialog, asfiInfo[i].nFolder|CSIDL_FLAG_DONT_VERIFY, NULL,
820                                  SHGFP_TYPE_CURRENT, buffer);
821         }
822     
823         item.mask = LVIF_TEXT | LVIF_PARAM;
824         item.iItem = i;
825         item.iSubItem = 0;
826         item.pszText = buffer;
827         item.lParam = (LPARAM)&asfiInfo[i];
828         SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_INSERTITEMW, 0, (LPARAM)&item);
829
830         item.mask = LVIF_TEXT;
831         item.iItem = i;
832         item.iSubItem = 1;
833         item.pszText = strdupU2W(asfiInfo[i].szLinkTarget);
834         SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_SETITEMW, 0, (LPARAM)&item);
835         HeapFree(GetProcessHeap(), 0, item.pszText);
836     }
837
838     /* Ensure that the previously selected item is selected again. */
839     if (lSelected >= 0) {
840         item.mask = LVIF_STATE;
841         item.state = LVIS_SELECTED;
842         item.stateMask = LVIS_SELECTED;
843         SendDlgItemMessage(dialog, IDC_LIST_SFPATHS, LVM_SETITEMSTATE, lSelected,
844                            (LPARAM)&item);
845     }
846 }
847
848 static void on_shell_folder_selection_changed(HWND hDlg, LPNMLISTVIEW lpnm) {
849     if (lpnm->uNewState & LVIS_SELECTED) {
850         psfiSelected = (struct ShellFolderInfo *)lpnm->lParam;
851         EnableWindow(GetDlgItem(hDlg, IDC_LINK_SFPATH), 1);
852         if (strlen(psfiSelected->szLinkTarget)) {
853             WCHAR *link;
854             CheckDlgButton(hDlg, IDC_LINK_SFPATH, BST_CHECKED);
855             EnableWindow(GetDlgItem(hDlg, IDC_EDIT_SFPATH), 1);
856             EnableWindow(GetDlgItem(hDlg, IDC_BROWSE_SFPATH), 1);
857             link = strdupU2W(psfiSelected->szLinkTarget);
858             set_textW(hDlg, IDC_EDIT_SFPATH, link);
859             HeapFree(GetProcessHeap(), 0, link);
860         } else {
861             CheckDlgButton(hDlg, IDC_LINK_SFPATH, BST_UNCHECKED);
862             EnableWindow(GetDlgItem(hDlg, IDC_EDIT_SFPATH), 0);
863             EnableWindow(GetDlgItem(hDlg, IDC_BROWSE_SFPATH), 0);
864             set_text(hDlg, IDC_EDIT_SFPATH, "");
865         }
866     } else {
867         psfiSelected = NULL;
868         CheckDlgButton(hDlg, IDC_LINK_SFPATH, BST_UNCHECKED);
869         set_text(hDlg, IDC_EDIT_SFPATH, "");
870         EnableWindow(GetDlgItem(hDlg, IDC_LINK_SFPATH), 0);
871         EnableWindow(GetDlgItem(hDlg, IDC_EDIT_SFPATH), 0);
872         EnableWindow(GetDlgItem(hDlg, IDC_BROWSE_SFPATH), 0);
873     }
874 }
875
876 /* Keep the contents of the edit control, the listview control and the symlink 
877  * information in sync. */
878 static void on_shell_folder_edit_changed(HWND hDlg) {
879     LVITEMW item;
880     WCHAR *text = get_textW(hDlg, IDC_EDIT_SFPATH);
881     LONG iSel = SendDlgItemMessage(hDlg, IDC_LIST_SFPATHS, LVM_GETNEXTITEM, -1,
882                                    MAKELPARAM(LVNI_SELECTED,0));
883     
884     if (!text || !psfiSelected || iSel < 0) {
885         HeapFree(GetProcessHeap(), 0, text);
886         return;
887     }
888
889     WideCharToMultiByte(CP_UNIXCP, 0, text, -1,
890                         psfiSelected->szLinkTarget, FILENAME_MAX, NULL, NULL);
891
892     item.mask = LVIF_TEXT;
893     item.iItem = iSel;
894     item.iSubItem = 1;
895     item.pszText = text;
896     SendDlgItemMessage(hDlg, IDC_LIST_SFPATHS, LVM_SETITEMW, 0, (LPARAM)&item);
897
898     HeapFree(GetProcessHeap(), 0, text);
899
900     SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
901 }
902
903 static void apply_shell_folder_changes(void) {
904     WCHAR wszPath[MAX_PATH];
905     char szBackupPath[FILENAME_MAX], szUnixPath[FILENAME_MAX], *pszUnixPath = NULL;
906     int i;
907     struct stat statPath;
908     HRESULT hr;
909
910     for (i=0; i<NUM_ELEMS(asfiInfo); i++) {
911         /* Ignore nonexistent link targets */
912         if (asfiInfo[i].szLinkTarget[0] && stat(asfiInfo[i].szLinkTarget, &statPath))
913             continue;
914         
915         hr = SHGetFolderPathW(NULL, asfiInfo[i].nFolder|CSIDL_FLAG_CREATE, NULL, 
916                               SHGFP_TYPE_CURRENT, wszPath);
917         if (FAILED(hr)) continue;
918
919         /* Retrieve the corresponding unix path. */
920         pszUnixPath = wine_get_unix_file_name(wszPath);
921         if (!pszUnixPath) continue;
922         lstrcpyA(szUnixPath, pszUnixPath);
923         HeapFree(GetProcessHeap(), 0, pszUnixPath);
924             
925         /* Derive name for folder backup. */
926         lstrcpyA(szBackupPath, szUnixPath);
927         lstrcatA(szBackupPath, ".winecfg");
928         
929         if (lstat(szUnixPath, &statPath)) continue;
930     
931         /* Move old folder/link out of the way. */
932         if (S_ISLNK(statPath.st_mode)) {
933             if (unlink(szUnixPath)) continue; /* Unable to remove link. */
934         } else { 
935             if (!*asfiInfo[i].szLinkTarget) {
936                 continue; /* We are done. Old was real folder, as new shall be. */
937             } else { 
938                 if (rename(szUnixPath, szBackupPath)) { /* Move folder out of the way. */
939                     continue; /* Unable to move old folder. */
940                 }
941             }
942         }
943     
944         /* Create new link/folder. */
945         if (*asfiInfo[i].szLinkTarget) {
946             symlink(asfiInfo[i].szLinkTarget, szUnixPath);
947         } else {
948             /* If there's a backup folder, restore it. Else create new folder. */
949             if (!lstat(szBackupPath, &statPath) && S_ISDIR(statPath.st_mode)) {
950                 rename(szBackupPath, szUnixPath);
951             } else {
952                 mkdir(szUnixPath, 0777);
953             }
954         }
955     }
956 }
957
958 static void read_sysparams(HWND hDlg)
959 {
960     WCHAR buffer[256];
961     HWND list = GetDlgItem(hDlg, IDC_SYSPARAM_COMBO);
962     NONCLIENTMETRICSW nonclient_metrics;
963     int i, idx;
964
965     for (i = 0; i < sizeof(metrics) / sizeof(metrics[0]); i++)
966     {
967         LoadStringW(GetModuleHandle(NULL), i + IDC_SYSPARAMS_BUTTON, buffer,
968                     sizeof(buffer) / sizeof(buffer[0]));
969         idx = SendMessageW(list, CB_ADDSTRING, 0, (LPARAM)buffer);
970         if (idx != CB_ERR) SendMessageW(list, CB_SETITEMDATA, idx, i);
971
972         if (metrics[i].sm_idx != -1)
973             metrics[i].size = GetSystemMetrics(metrics[i].sm_idx);
974         if (metrics[i].color_idx != -1)
975             metrics[i].color = GetSysColor(metrics[i].color_idx);
976     }
977
978     nonclient_metrics.cbSize = sizeof(NONCLIENTMETRICSW);
979     SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof(NONCLIENTMETRICSW), &nonclient_metrics, 0);
980
981     memcpy(&(metrics[IDC_SYSPARAMS_MENU_TEXT - IDC_SYSPARAMS_BUTTON].lf),
982            &(nonclient_metrics.lfMenuFont), sizeof(LOGFONTW));
983     memcpy(&(metrics[IDC_SYSPARAMS_ACTIVE_TITLE_TEXT - IDC_SYSPARAMS_BUTTON].lf),
984            &(nonclient_metrics.lfCaptionFont), sizeof(LOGFONTW));
985     memcpy(&(metrics[IDC_SYSPARAMS_TOOLTIP_TEXT - IDC_SYSPARAMS_BUTTON].lf),
986            &(nonclient_metrics.lfStatusFont), sizeof(LOGFONTW));
987     memcpy(&(metrics[IDC_SYSPARAMS_MSGBOX_TEXT - IDC_SYSPARAMS_BUTTON].lf),
988            &(nonclient_metrics.lfMessageFont), sizeof(LOGFONTW));
989 }
990
991 static void apply_sysparams(void)
992 {
993     NONCLIENTMETRICSW nonclient_metrics;
994     int i, cnt = 0;
995     int colors_idx[sizeof(metrics) / sizeof(metrics[0])];
996     COLORREF colors[sizeof(metrics) / sizeof(metrics[0])];
997
998     nonclient_metrics.cbSize = sizeof(nonclient_metrics);
999     SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof(nonclient_metrics), &nonclient_metrics, 0);
1000
1001     nonclient_metrics.iMenuWidth = nonclient_metrics.iMenuHeight =
1002             metrics[IDC_SYSPARAMS_MENU - IDC_SYSPARAMS_BUTTON].size;
1003     nonclient_metrics.iCaptionWidth = nonclient_metrics.iCaptionHeight =
1004             metrics[IDC_SYSPARAMS_ACTIVE_TITLE - IDC_SYSPARAMS_BUTTON].size;
1005     nonclient_metrics.iScrollWidth = nonclient_metrics.iScrollHeight =
1006             metrics[IDC_SYSPARAMS_SCROLLBAR - IDC_SYSPARAMS_BUTTON].size;
1007
1008     memcpy(&(nonclient_metrics.lfMenuFont),
1009            &(metrics[IDC_SYSPARAMS_MENU_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1010            sizeof(LOGFONTW));
1011     memcpy(&(nonclient_metrics.lfCaptionFont),
1012            &(metrics[IDC_SYSPARAMS_ACTIVE_TITLE_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1013            sizeof(LOGFONTW));
1014     memcpy(&(nonclient_metrics.lfStatusFont),
1015            &(metrics[IDC_SYSPARAMS_TOOLTIP_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1016            sizeof(LOGFONTW));
1017     memcpy(&(nonclient_metrics.lfMessageFont),
1018            &(metrics[IDC_SYSPARAMS_MSGBOX_TEXT - IDC_SYSPARAMS_BUTTON].lf),
1019            sizeof(LOGFONTW));
1020
1021     SystemParametersInfoW(SPI_SETNONCLIENTMETRICS, sizeof(nonclient_metrics), &nonclient_metrics,
1022                           SPIF_UPDATEINIFILE | SPIF_SENDCHANGE);
1023
1024     for (i = 0; i < sizeof(metrics) / sizeof(metrics[0]); i++)
1025         if (metrics[i].color_idx != -1)
1026         {
1027             colors_idx[cnt] = metrics[i].color_idx;
1028             colors[cnt++] = metrics[i].color;
1029         }
1030     SetSysColors(cnt, colors_idx, colors);
1031 }
1032
1033 static void on_sysparam_change(HWND hDlg)
1034 {
1035     int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1036
1037     index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1038
1039     updating_ui = TRUE;
1040
1041     EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_COLOR_TEXT), metrics[index].color_idx != -1);
1042     EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_COLOR), metrics[index].color_idx != -1);
1043     InvalidateRect(GetDlgItem(hDlg, IDC_SYSPARAM_COLOR), NULL, TRUE);
1044
1045     EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_SIZE_TEXT), metrics[index].sm_idx != -1);
1046     EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_SIZE), metrics[index].sm_idx != -1);
1047     EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_SIZE_UD), metrics[index].sm_idx != -1);
1048     if (metrics[index].sm_idx != -1)
1049         SendDlgItemMessageW(hDlg, IDC_SYSPARAM_SIZE_UD, UDM_SETPOS, 0, MAKELONG(metrics[index].size, 0));
1050     else
1051         set_text(hDlg, IDC_SYSPARAM_SIZE, "");
1052
1053     EnableWindow(GetDlgItem(hDlg, IDC_SYSPARAM_FONT),
1054         index == IDC_SYSPARAMS_MENU_TEXT-IDC_SYSPARAMS_BUTTON ||
1055         index == IDC_SYSPARAMS_ACTIVE_TITLE_TEXT-IDC_SYSPARAMS_BUTTON ||
1056         index == IDC_SYSPARAMS_TOOLTIP_TEXT-IDC_SYSPARAMS_BUTTON ||
1057         index == IDC_SYSPARAMS_MSGBOX_TEXT-IDC_SYSPARAMS_BUTTON
1058     );
1059
1060     updating_ui = FALSE;
1061 }
1062
1063 static void on_draw_item(HWND hDlg, WPARAM wParam, LPARAM lParam)
1064 {
1065     static HBRUSH black_brush = 0;
1066     LPDRAWITEMSTRUCT draw_info = (LPDRAWITEMSTRUCT)lParam;
1067
1068     if (!black_brush) black_brush = CreateSolidBrush(0);
1069
1070     if (draw_info->CtlID == IDC_SYSPARAM_COLOR)
1071     {
1072         UINT state = DFCS_ADJUSTRECT | DFCS_BUTTONPUSH;
1073
1074         if (draw_info->itemState & ODS_DISABLED)
1075             state |= DFCS_INACTIVE;
1076         else
1077             state |= draw_info->itemState & ODS_SELECTED ? DFCS_PUSHED : 0;
1078
1079         DrawFrameControl(draw_info->hDC, &draw_info->rcItem, DFC_BUTTON, state);
1080
1081         if (!(draw_info->itemState & ODS_DISABLED))
1082         {
1083             HBRUSH brush;
1084             int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1085
1086             index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1087             brush = CreateSolidBrush(metrics[index].color);
1088
1089             InflateRect(&draw_info->rcItem, -1, -1);
1090             FrameRect(draw_info->hDC, &draw_info->rcItem, black_brush);
1091             InflateRect(&draw_info->rcItem, -1, -1);
1092             FillRect(draw_info->hDC, &draw_info->rcItem, brush);
1093             DeleteObject(brush);
1094         }
1095     }
1096 }
1097
1098 static void on_select_font(HWND hDlg)
1099 {
1100     CHOOSEFONTW cf;
1101     int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1102     index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1103
1104     ZeroMemory(&cf, sizeof(cf));
1105     cf.lStructSize = sizeof(CHOOSEFONTW);
1106     cf.hwndOwner = hDlg;
1107     cf.lpLogFont = &(metrics[index].lf);
1108     cf.Flags = CF_SCREENFONTS | CF_INITTOLOGFONTSTRUCT | CF_NOSCRIPTSEL;
1109
1110     ChooseFontW(&cf);
1111 }
1112
1113 INT_PTR CALLBACK
1114 ThemeDlgProc (HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
1115 {
1116     switch (uMsg) {
1117         case WM_INITDIALOG:
1118             read_shell_folder_link_targets();
1119             init_shell_folder_listview_headers(hDlg);
1120             update_shell_folder_listview(hDlg);
1121             read_sysparams(hDlg);
1122             break;
1123         
1124         case WM_DESTROY:
1125             free_theme_files();
1126             break;
1127
1128         case WM_SHOWWINDOW:
1129             set_window_title(hDlg);
1130             break;
1131             
1132         case WM_COMMAND:
1133             switch(HIWORD(wParam)) {
1134                 case CBN_SELCHANGE: {
1135                     if (updating_ui) break;
1136                     switch (LOWORD(wParam))
1137                     {
1138                         case IDC_THEME_THEMECOMBO: on_theme_changed(hDlg); break;
1139                         case IDC_THEME_COLORCOMBO: /* fall through */
1140                         case IDC_THEME_SIZECOMBO: theme_dirty = TRUE; break;
1141                         case IDC_SYSPARAM_COMBO: on_sysparam_change(hDlg); return FALSE;
1142                     }
1143                     SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1144                     break;
1145                 }
1146                 case EN_CHANGE: {
1147                     if (updating_ui) break;
1148                     switch (LOWORD(wParam))
1149                     {
1150                         case IDC_EDIT_SFPATH: on_shell_folder_edit_changed(hDlg); break;
1151                         case IDC_SYSPARAM_SIZE:
1152                         {
1153                             char *text = get_text(hDlg, IDC_SYSPARAM_SIZE);
1154                             int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1155
1156                             index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1157                             metrics[index].size = atoi(text);
1158                             HeapFree(GetProcessHeap(), 0, text);
1159
1160                             SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1161                             break;
1162                         }
1163                     }
1164                     break;
1165                 }
1166                 case BN_CLICKED:
1167                     switch (LOWORD(wParam))
1168                     {
1169                         case IDC_THEME_INSTALL:
1170                             on_theme_install (hDlg);
1171                             break;
1172
1173                         case IDC_SYSPARAM_FONT:
1174                             on_select_font(hDlg);
1175                             break;
1176
1177                         case IDC_BROWSE_SFPATH:
1178                         {
1179                             WCHAR link[FILENAME_MAX];
1180                             if (browse_for_unix_folder(hDlg, link)) {
1181                                 WideCharToMultiByte(CP_UNIXCP, 0, link, -1,
1182                                                     psfiSelected->szLinkTarget, FILENAME_MAX,
1183                                                     NULL, NULL);
1184                                 update_shell_folder_listview(hDlg);
1185                                 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1186                             }
1187                             break;
1188                         }
1189
1190                         case IDC_LINK_SFPATH:
1191                             if (IsDlgButtonChecked(hDlg, IDC_LINK_SFPATH)) {
1192                                 WCHAR link[FILENAME_MAX];
1193                                 if (browse_for_unix_folder(hDlg, link)) {
1194                                     WideCharToMultiByte(CP_UNIXCP, 0, link, -1,
1195                                                         psfiSelected->szLinkTarget, FILENAME_MAX,
1196                                                         NULL, NULL);
1197                                     update_shell_folder_listview(hDlg);
1198                                     SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1199                                 } else {
1200                                     CheckDlgButton(hDlg, IDC_LINK_SFPATH, BST_UNCHECKED);
1201                                 }
1202                             } else {
1203                                 psfiSelected->szLinkTarget[0] = '\0';
1204                                 update_shell_folder_listview(hDlg);
1205                                 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1206                             }
1207                             break;    
1208
1209                         case IDC_SYSPARAM_COLOR:
1210                         {
1211                             static COLORREF user_colors[16];
1212                             CHOOSECOLORW c_color;
1213                             int index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETCURSEL, 0, 0);
1214
1215                             index = SendDlgItemMessageW(hDlg, IDC_SYSPARAM_COMBO, CB_GETITEMDATA, index, 0);
1216
1217                             memset(&c_color, 0, sizeof(c_color));
1218                             c_color.lStructSize = sizeof(c_color);
1219                             c_color.lpCustColors = user_colors;
1220                             c_color.rgbResult = metrics[index].color;
1221                             c_color.Flags = CC_ANYCOLOR | CC_RGBINIT;
1222                             c_color.hwndOwner = hDlg;
1223                             if (ChooseColorW(&c_color))
1224                             {
1225                                 metrics[index].color = c_color.rgbResult;
1226                                 save_sys_color(index, metrics[index].color);
1227                                 InvalidateRect(GetDlgItem(hDlg, IDC_SYSPARAM_COLOR), NULL, TRUE);
1228                                 SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0);
1229                             }
1230                             break;
1231                         }
1232                     }
1233                     break;
1234             }
1235             break;
1236         
1237         case WM_NOTIFY:
1238             switch (((LPNMHDR)lParam)->code) {
1239                 case PSN_KILLACTIVE: {
1240                     SetWindowLongPtr(hDlg, DWLP_MSGRESULT, FALSE);
1241                     break;
1242                 }
1243                 case PSN_APPLY: {
1244                     apply();
1245                     apply_theme(hDlg);
1246                     apply_shell_folder_changes();
1247                     apply_sysparams();
1248                     read_shell_folder_link_targets();
1249                     update_shell_folder_listview(hDlg);
1250                     SetWindowLongPtr(hDlg, DWLP_MSGRESULT, PSNRET_NOERROR);
1251                     break;
1252                 }
1253                 case LVN_ITEMCHANGED: { 
1254                     if (wParam == IDC_LIST_SFPATHS)  
1255                         on_shell_folder_selection_changed(hDlg, (LPNMLISTVIEW)lParam);
1256                     break;
1257                 }
1258                 case PSN_SETACTIVE: {
1259                     init_dialog (hDlg);
1260                     break;
1261                 }
1262             }
1263             break;
1264
1265         case WM_DRAWITEM:
1266             on_draw_item(hDlg, wParam, lParam);
1267             break;
1268
1269         default:
1270             break;
1271     }
1272     return FALSE;
1273 }