user32: Implement GetWindowModuleFileName.
[wine] / programs / winecfg / appdefaults.c
1 /*
2  * WineCfg app settings tabsheet
3  *
4  * Copyright 2004 Robert van Herk
5  * Copyright 2004 Chris Morgan
6  * Copyright 2004 Mike Hearn
7  *
8  * This library is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Lesser General Public
10  * License as published by the Free Software Foundation; either
11  * version 2.1 of the License, or (at your option) any later version.
12  *
13  * This library is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * Lesser General Public License for more details.
17  *
18  * You should have received a copy of the GNU Lesser General Public
19  * License along with this library; if not, write to the Free Software
20  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21  *
22  */
23
24 #define WIN32_LEAN_AND_MEAN
25 #define NONAMELESSUNION
26 #include <windows.h>
27 #include <commdlg.h>
28 #include <wine/debug.h>
29 #include <stdio.h>
30 #include <stdlib.h>
31 #include <assert.h>
32 #include "winecfg.h"
33 #include "resource.h"
34
35 WINE_DEFAULT_DEBUG_CHANNEL(winecfg);
36
37 static const struct
38 {
39     const char *szVersion;
40     const char *szDescription;
41     DWORD       dwMajorVersion;
42     DWORD       dwMinorVersion;
43     DWORD       dwBuildNumber;
44     DWORD       dwPlatformId;
45     const char *szCSDVersion;
46     WORD        wServicePackMajor;
47     WORD        wServicePackMinor;
48     const char *szProductType;
49 } win_versions[] =
50 {
51     { "win2008", "Windows 2008",   6,  0, 0x1771,VER_PLATFORM_WIN32_NT, "Service Pack 1", 0, 0, "ServerNT"},
52     { "vista",   "Windows Vista",  6,  0, 0x1770,VER_PLATFORM_WIN32_NT, " ", 0, 0, "WinNT"},
53     { "win2003", "Windows 2003",   5,  2, 0xECE, VER_PLATFORM_WIN32_NT, "Service Pack 1", 1, 0, "ServerNT"},
54     { "winxp",   "Windows XP",     5,  1, 0xA28, VER_PLATFORM_WIN32_NT, "Service Pack 2", 2, 0, "WinNT"},
55     { "win2k",   "Windows 2000",   5,  0, 0x893, VER_PLATFORM_WIN32_NT, "Service Pack 4", 4, 0, "WinNT"},
56     { "winme",   "Windows ME",     4, 90, 0xBB8, VER_PLATFORM_WIN32_WINDOWS, " ", 0, 0, ""},
57     { "win98",   "Windows 98",     4, 10, 0x8AE, VER_PLATFORM_WIN32_WINDOWS, " A ", 0, 0, ""},
58     { "win95",   "Windows 95",     4,  0, 0x3B6, VER_PLATFORM_WIN32_WINDOWS, "", 0, 0, ""},
59     { "nt40",    "Windows NT 4.0", 4,  0, 0x565, VER_PLATFORM_WIN32_NT, "Service Pack 6a", 6, 0, "WinNT"},
60     { "nt351",   "Windows NT 3.5", 3, 51, 0x421, VER_PLATFORM_WIN32_NT, "Service Pack 2", 0, 0, "WinNT"},
61     { "win31",   "Windows 3.1",    2, 10,     0, VER_PLATFORM_WIN32s, "Win32s 1.3", 0, 0, ""},
62     { "win30",   "Windows 3.0",    3,  0,     0, VER_PLATFORM_WIN32s, "Win32s 1.3", 0, 0, ""},
63     { "win20",   "Windows 2.0",    2,  0,     0, VER_PLATFORM_WIN32s, "Win32s 1.3", 0, 0, ""}
64 };
65
66 #define NB_VERSIONS (sizeof(win_versions)/sizeof(win_versions[0]))
67
68 static const char szKey9x[] = "Software\\Microsoft\\Windows\\CurrentVersion";
69 static const char szKeyNT[] = "Software\\Microsoft\\Windows NT\\CurrentVersion";
70
71 static int get_registry_version(void)
72 {
73     int i, best = -1, platform, major, minor = 0, build = 0;
74     char *p, *ver;
75
76     if ((ver = get_reg_key( HKEY_LOCAL_MACHINE, szKeyNT, "CurrentVersion", NULL )))
77     {
78         char *build_str;
79
80         platform = VER_PLATFORM_WIN32_NT;
81
82         build_str = get_reg_key( HKEY_LOCAL_MACHINE, szKeyNT, "CurrentBuildNumber", NULL );
83         build = atoi(build_str);
84     }
85     else if ((ver = get_reg_key( HKEY_LOCAL_MACHINE, szKey9x, "VersionNumber", NULL )))
86         platform = VER_PLATFORM_WIN32_WINDOWS;
87     else
88         return -1;
89
90     if ((p = strchr( ver, '.' )))
91     {
92         char *minor_str = p;
93         *minor_str++ = 0;
94         if ((p = strchr( minor_str, '.' )))
95         {
96             char *build_str = p;
97             *build_str++ = 0;
98             build = atoi(build_str);
99         }
100         minor = atoi(minor_str);
101     }
102     major = atoi(ver);
103
104     for (i = 0; i < NB_VERSIONS; i++)
105     {
106         if (win_versions[i].dwPlatformId != platform) continue;
107         if (win_versions[i].dwMajorVersion != major) continue;
108         best = i;
109         if ((win_versions[i].dwMinorVersion == minor) &&
110             (win_versions[i].dwBuildNumber == build))
111             return i;
112     }
113     return best;
114 }
115
116 static void update_comboboxes(HWND dialog)
117 {
118     int i, ver;
119     char *winver;
120
121     /* retrieve the registry values */
122     winver = get_reg_key(config_key, keypath(""), "Version", "");
123     ver = get_registry_version();
124
125     if (!winver || !winver[0])
126     {
127         HeapFree(GetProcessHeap(), 0, winver);
128
129         if (current_app) /* no explicit setting */
130         {
131             WINE_TRACE("setting winver combobox to default\n");
132             SendDlgItemMessage (dialog, IDC_WINVER, CB_SETCURSEL, 0, 0);
133             return;
134         }
135         if (ver != -1) winver = strdupA( win_versions[ver].szVersion );
136         else winver = strdupA("win2k");
137     }
138     WINE_TRACE("winver is %s\n", winver);
139
140     /* normalize the version strings */
141     for (i = 0; i < NB_VERSIONS; i++)
142     {
143         if (!strcasecmp (win_versions[i].szVersion, winver))
144         {
145             SendDlgItemMessage (dialog, IDC_WINVER, CB_SETCURSEL,
146                                 (WPARAM) i + (current_app?1:0), 0);
147             WINE_TRACE("match with %s\n", win_versions[i].szVersion);
148             break;
149         }
150     }
151
152     HeapFree(GetProcessHeap(), 0, winver);
153 }
154
155 static void
156 init_comboboxes (HWND dialog)
157 {
158     int i;
159
160     SendDlgItemMessage(dialog, IDC_WINVER, CB_RESETCONTENT, 0, 0);
161
162     /* add the default entries (automatic) which correspond to no setting  */
163     if (current_app)
164     {
165         WCHAR str[256];
166         LoadStringW (GetModuleHandle (NULL), IDS_USE_GLOBAL_SETTINGS, str,
167             sizeof(str)/sizeof(str[0]));
168         SendDlgItemMessageW (dialog, IDC_WINVER, CB_ADDSTRING, 0, (LPARAM)str);
169     }
170
171     for (i = 0; i < NB_VERSIONS; i++)
172     {
173       SendDlgItemMessage (dialog, IDC_WINVER, CB_ADDSTRING,
174                           0, (LPARAM) win_versions[i].szDescription);
175     }
176 }
177
178 static void add_listview_item(HWND listview, WCHAR *text, void *association)
179 {
180   LVITEMW item;
181
182   item.mask = LVIF_TEXT | LVIF_PARAM;
183   item.pszText = text;
184   item.cchTextMax = lstrlenW(text);
185   item.lParam = (LPARAM) association;
186   item.iItem = ListView_GetItemCount(listview);
187   item.iSubItem = 0;
188
189   SendMessage(listview, LVM_INSERTITEMW, 0, (LPARAM) &item);
190 }
191
192 /* Called when the application is initialized (cannot reinit!)  */
193 static void init_appsheet(HWND dialog)
194 {
195   HWND listview;
196   HKEY key;
197   int i;
198   DWORD size;
199   WCHAR appname[1024];
200
201   WINE_TRACE("()\n");
202
203   listview = GetDlgItem(dialog, IDC_APP_LISTVIEW);
204
205   /* we use the lparam field of the item so we can alter the presentation later and not change code
206    * for instance, to use the tile view or to display the EXEs embedded 'display name' */
207   LoadStringW (GetModuleHandle (NULL), IDS_DEFAULT_SETTINGS, appname,
208       sizeof(appname)/sizeof(appname[0]));
209   add_listview_item(listview, appname, NULL);
210
211   /* because this list is only populated once, it's safe to bypass the settings list here  */
212   if (RegOpenKey(config_key, "AppDefaults", &key) == ERROR_SUCCESS)
213   {
214       i = 0;
215       size = sizeof(appname)/sizeof(appname[0]);
216       while (RegEnumKeyExW (key, i, appname, &size, NULL, NULL, NULL, NULL) == ERROR_SUCCESS)
217       {
218           add_listview_item(listview, appname, strdupW(appname));
219
220           i++;
221           size = sizeof(appname)/sizeof(appname[0]);
222       }
223
224       RegCloseKey(key);
225   }
226
227   init_comboboxes(dialog);
228   
229   /* Select the default settings listview item  */
230   {
231       LVITEM item;
232       
233       item.iItem = 0;
234       item.iSubItem = 0;
235       item.mask = LVIF_STATE;
236       item.state = LVIS_SELECTED | LVIS_FOCUSED;
237       item.stateMask = LVIS_SELECTED | LVIS_FOCUSED;
238
239       SendMessage(listview, LVM_SETITEM, 0, (LPARAM) &item);
240   }
241   
242 }
243
244 /* there has to be an easier way than this  */
245 static int get_listview_selection(HWND listview)
246 {
247   int count = ListView_GetItemCount(listview);
248   int i;
249   
250   for (i = 0; i < count; i++)
251   {
252     if (ListView_GetItemState(listview, i, LVIS_SELECTED)) return i;
253   }
254
255   return -1;
256 }
257
258
259 /* called when the user selects a different application */
260 static void on_selection_change(HWND dialog, HWND listview)
261 {
262   LVITEM item;
263   WCHAR* oldapp = current_app;
264
265   WINE_TRACE("()\n");
266
267   item.iItem = get_listview_selection(listview);
268   item.iSubItem = 0;
269   item.mask = LVIF_PARAM;
270
271   WINE_TRACE("item.iItem=%d\n", item.iItem);
272   
273   if (item.iItem == -1) return;
274   
275   SendMessage(listview, LVM_GETITEM, 0, (LPARAM) &item);
276
277   current_app = (WCHAR*) item.lParam;
278
279   if (current_app)
280   {
281       WINE_TRACE("current_app is now %s\n", wine_dbgstr_w (current_app));
282       enable(IDC_APP_REMOVEAPP);
283   }
284   else
285   {
286       WINE_TRACE("current_app=NULL, editing global settings\n");
287       /* focus will never be on the button in this callback so it's safe  */
288       disable(IDC_APP_REMOVEAPP);
289   }
290
291   /* reset the combo boxes if we changed from/to global/app-specific  */
292
293   if ((oldapp && !current_app) || (!oldapp && current_app))
294       init_comboboxes(dialog);
295   
296   update_comboboxes(dialog);
297
298   set_window_title(dialog);
299 }
300
301 static BOOL list_contains_file(HWND listview, WCHAR *filename)
302 {
303   LVFINDINFOW find_info = { LVFI_STRING, filename, 0, {0, 0}, 0 };
304   int index;
305
306   index = ListView_FindItemW(listview, -1, &find_info);
307
308   return (index != -1);
309 }
310
311 static void on_add_app_click(HWND dialog)
312 {
313   WCHAR filetitle[MAX_PATH];
314   WCHAR file[MAX_PATH];
315   WCHAR programsFilter[100];
316   WCHAR selectExecutableStr[100];
317   static const WCHAR pathC[] = { 'c',':','\\',0 };
318
319   OPENFILENAMEW ofn = { sizeof(OPENFILENAMEW),
320                        0, /*hInst*/0, 0, NULL, 0, 0, NULL,
321                        0, NULL, 0, pathC, 0,
322                        OFN_SHOWHELP | OFN_HIDEREADONLY, 0, 0, NULL, 0, NULL };
323
324   LoadStringW (GetModuleHandle (NULL), IDS_SELECT_EXECUTABLE, selectExecutableStr,
325       sizeof(selectExecutableStr)/sizeof(selectExecutableStr[0]));
326   LoadStringW (GetModuleHandle (NULL), IDS_EXECUTABLE_FILTER, programsFilter,
327       sizeof(programsFilter)/sizeof(programsFilter[0]));
328
329   ofn.lpstrTitle = selectExecutableStr;
330   ofn.lpstrFilter = programsFilter;
331   ofn.lpstrFileTitle = filetitle;
332   ofn.lpstrFileTitle[0] = '\0';
333   ofn.nMaxFileTitle = sizeof(filetitle)/sizeof(filetitle[0]);
334   ofn.lpstrFile = file;
335   ofn.lpstrFile[0] = '\0';
336   ofn.nMaxFile = sizeof(file)/sizeof(file[0]);
337
338   if (GetOpenFileNameW (&ofn))
339   {
340       HWND listview = GetDlgItem(dialog, IDC_APP_LISTVIEW);
341       int count = ListView_GetItemCount(listview);
342       WCHAR* new_app;
343       
344       if (list_contains_file(listview, filetitle))
345           return;
346       
347       new_app = strdupW(filetitle);
348
349       WINE_TRACE("adding %s\n", wine_dbgstr_w (new_app));
350       
351       add_listview_item(listview, new_app, new_app);
352
353       ListView_SetItemState(listview, count, LVIS_SELECTED | LVIS_FOCUSED, LVIS_SELECTED | LVIS_FOCUSED);
354
355       SetFocus(listview);
356   }
357   else WINE_TRACE("user cancelled\n");
358 }
359
360 static void on_remove_app_click(HWND dialog)
361 {
362     HWND listview = GetDlgItem(dialog, IDC_APP_LISTVIEW);
363     int selection = get_listview_selection(listview);
364     char *section = keypath(""); /* AppDefaults\\whatever.exe\\ */
365     LVITEMW item;
366
367     item.iItem = selection;
368     item.iSubItem = 0;
369     item.mask = LVIF_PARAM;
370
371     WINE_TRACE("selection=%d, section=%s\n", selection, section);
372     
373     assert( selection != 0 ); /* user cannot click this button when "default settings" is selected  */
374
375     section[strlen(section)] = '\0'; /* remove last backslash  */
376     set_reg_key(config_key, section, NULL, NULL); /* delete the section  */
377     SendMessage(listview, LVM_GETITEMW, 0, (LPARAM) &item);
378     HeapFree (GetProcessHeap(), 0, (void*)item.lParam);
379     SendMessage(listview, LVM_DELETEITEM, selection, 0);
380     ListView_SetItemState(listview, selection - 1, LVIS_SELECTED | LVIS_FOCUSED, LVIS_SELECTED | LVIS_FOCUSED);
381
382     SetFocus(listview);
383     
384     SendMessage(GetParent(dialog), PSM_CHANGED, (WPARAM) dialog, 0);        
385 }
386
387 static void on_winver_change(HWND dialog)
388 {
389     int selection = SendDlgItemMessage(dialog, IDC_WINVER, CB_GETCURSEL, 0, 0);
390
391     if (current_app)
392     {
393         if (!selection)
394         {
395             WINE_TRACE("default selected so removing current setting\n");
396             set_reg_key(config_key, keypath(""), "Version", NULL);
397         }
398         else
399         {
400             WINE_TRACE("setting Version key to value '%s'\n", win_versions[selection-1].szVersion);
401             set_reg_key(config_key, keypath(""), "Version", win_versions[selection-1].szVersion);
402         }
403     }
404     else /* global version only */
405     {
406         static const char szKeyProdNT[] = "System\\CurrentControlSet\\Control\\ProductOptions";
407         static const char szKeyWindNT[] = "System\\CurrentControlSet\\Control\\Windows";
408         static const char szKeyEnvNT[]  = "System\\CurrentControlSet\\Control\\Session Manager\\Environment";
409         char Buffer[40];
410
411         switch (win_versions[selection].dwPlatformId)
412         {
413         case VER_PLATFORM_WIN32_WINDOWS:
414             snprintf(Buffer, sizeof(Buffer), "%d.%d.%d", win_versions[selection].dwMajorVersion,
415                      win_versions[selection].dwMinorVersion, win_versions[selection].dwBuildNumber);
416             set_reg_key(HKEY_LOCAL_MACHINE, szKey9x, "VersionNumber", Buffer);
417             set_reg_key(HKEY_LOCAL_MACHINE, szKey9x, "SubVersionNumber", win_versions[selection].szCSDVersion);
418
419             set_reg_key(HKEY_LOCAL_MACHINE, szKeyNT, "CSDVersion", NULL);
420             set_reg_key(HKEY_LOCAL_MACHINE, szKeyNT, "CurrentVersion", NULL);
421             set_reg_key(HKEY_LOCAL_MACHINE, szKeyNT, "CurrentBuildNumber", NULL);
422             set_reg_key(HKEY_LOCAL_MACHINE, szKeyProdNT, "ProductType", NULL);
423             set_reg_key(HKEY_LOCAL_MACHINE, szKeyWindNT, "CSDVersion", NULL);
424             set_reg_key(HKEY_LOCAL_MACHINE, szKeyEnvNT, "OS", NULL);
425             set_reg_key(config_key, keypath(""), "Version", NULL);
426             break;
427
428         case VER_PLATFORM_WIN32_NT:
429             snprintf(Buffer, sizeof(Buffer), "%d.%d", win_versions[selection].dwMajorVersion,
430                      win_versions[selection].dwMinorVersion);
431             set_reg_key(HKEY_LOCAL_MACHINE, szKeyNT, "CurrentVersion", Buffer);
432             set_reg_key(HKEY_LOCAL_MACHINE, szKeyNT, "CSDVersion", win_versions[selection].szCSDVersion);
433             snprintf(Buffer, sizeof(Buffer), "%d", win_versions[selection].dwBuildNumber);
434             set_reg_key(HKEY_LOCAL_MACHINE, szKeyNT, "CurrentBuildNumber", Buffer);
435             set_reg_key(HKEY_LOCAL_MACHINE, szKeyProdNT, "ProductType", win_versions[selection].szProductType);
436             set_reg_key_dword(HKEY_LOCAL_MACHINE, szKeyWindNT, "CSDVersion",
437                               MAKEWORD( win_versions[selection].wServicePackMinor,
438                                         win_versions[selection].wServicePackMajor ));
439             set_reg_key(HKEY_LOCAL_MACHINE, szKeyEnvNT, "OS", "Windows_NT");
440
441             set_reg_key(HKEY_LOCAL_MACHINE, szKey9x, "VersionNumber", NULL);
442             set_reg_key(HKEY_LOCAL_MACHINE, szKey9x, "SubVersionNumber", NULL);
443             set_reg_key(config_key, keypath(""), "Version", NULL);
444             break;
445
446         case VER_PLATFORM_WIN32s:
447             set_reg_key(HKEY_LOCAL_MACHINE, szKeyNT, "CSDVersion", NULL);
448             set_reg_key(HKEY_LOCAL_MACHINE, szKeyNT, "CurrentVersion", NULL);
449             set_reg_key(HKEY_LOCAL_MACHINE, szKeyNT, "CurrentBuildNumber", NULL);
450             set_reg_key(HKEY_LOCAL_MACHINE, szKeyProdNT, "ProductType", NULL);
451             set_reg_key(HKEY_LOCAL_MACHINE, szKeyWindNT, "CSDVersion", NULL);
452             set_reg_key(HKEY_LOCAL_MACHINE, szKeyEnvNT, "OS", NULL);
453             set_reg_key(HKEY_LOCAL_MACHINE, szKey9x, "VersionNumber", NULL);
454             set_reg_key(HKEY_LOCAL_MACHINE, szKey9x, "SubVersionNumber", NULL);
455             set_reg_key(config_key, keypath(""), "Version", win_versions[selection].szVersion);
456             break;
457         }
458     }
459
460     /* enable the apply button  */
461     SendMessage(GetParent(dialog), PSM_CHANGED, (WPARAM) dialog, 0);
462 }
463
464 INT_PTR CALLBACK
465 AppDlgProc (HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
466 {
467   switch (uMsg)
468   {
469     case WM_INITDIALOG:
470         init_appsheet(hDlg);
471         break;
472
473     case WM_SHOWWINDOW:
474         set_window_title(hDlg);
475         break;
476
477     case WM_NOTIFY:
478       switch (((LPNMHDR)lParam)->code)
479       {
480         case LVN_ITEMCHANGED:
481             on_selection_change(hDlg, GetDlgItem(hDlg, IDC_APP_LISTVIEW));
482             break;
483         case PSN_APPLY:
484             apply();
485             SetWindowLongPtr(hDlg, DWLP_MSGRESULT, PSNRET_NOERROR);
486             break;
487       }
488       
489       break;
490     
491     case WM_COMMAND:
492       switch(HIWORD(wParam))
493       {
494         case CBN_SELCHANGE:
495           switch(LOWORD(wParam))
496           {
497             case IDC_WINVER:
498               on_winver_change(hDlg);
499               break;
500           }
501         case BN_CLICKED:
502           switch(LOWORD(wParam))
503           {
504             case IDC_APP_ADDAPP:
505               on_add_app_click(hDlg);
506               break;
507             case IDC_APP_REMOVEAPP:
508               on_remove_app_click(hDlg);
509               break;
510           }
511           break;
512       }
513
514       break;
515   }
516   
517   return 0;
518 }