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