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