xref: /reactos/dll/win32/syssetup/wizard.c (revision 57b9c6b2)
1 /*
2  * COPYRIGHT:       See COPYING in the top level directory
3  * PROJECT:         System setup
4  * FILE:            dll/win32/syssetup/wizard.c
5  * PURPOSE:         GUI controls
6  * PROGRAMMERS:     Eric Kohl
7  *                  Pierre Schweitzer <heis_spiter@hotmail.com>
8  *                  Ismael Ferreras Morezuelas <swyterzone+ros@gmail.com>
9  *                  Katayama Hirofumi MZ <katayama.hirofumi.mz@gmail.com>
10  *                  Oleg Dubinskiy <oleg.dubinskij30@gmail.com>
11  */
12 
13 /* INCLUDES *****************************************************************/
14 
15 #include "precomp.h"
16 
17 #include <stdlib.h>
18 #include <time.h>
19 #include <winnls.h>
20 #include <windowsx.h>
21 #include <wincon.h>
22 #include <shlobj.h>
23 #include <tzlib.h>
24 #include <strsafe.h>
25 
26 #define NDEBUG
27 #include <debug.h>
28 
29 #define PM_REGISTRATION_NOTIFY (WM_APP + 1)
30 /* Private Message used to communicate progress from the background
31    registration thread to the main thread.
32    wParam = 0 Registration in progress
33           = 1 Registration completed
34    lParam = Pointer to a REGISTRATIONNOTIFY structure */
35 
36 #define PM_ITEM_START (WM_APP + 2)
37 #define PM_ITEM_END   (WM_APP + 3)
38 #define PM_STEP_START (WM_APP + 4)
39 #define PM_STEP_END   (WM_APP + 5)
40 #define PM_ITEMS_DONE (WM_APP + 6)
41 
42 typedef struct _REGISTRATIONNOTIFY
43 {
44     ULONG Progress;
45     UINT ActivityID;
46     LPCWSTR CurrentItem;
47     LPCWSTR ErrorMessage;
48     UINT MessageID;
49     DWORD LastError;
50 } REGISTRATIONNOTIFY, *PREGISTRATIONNOTIFY;
51 
52 typedef struct _ITEMSDATA
53 {
54     HWND hwndDlg;
55 } ITEMSDATA, *PITEMSDATA;
56 
57 typedef struct _REGISTRATIONDATA
58 {
59     HWND hwndDlg;
60     ULONG DllCount;
61     ULONG Registered;
62     PVOID DefaultContext;
63 } REGISTRATIONDATA, *PREGISTRATIONDATA;
64 
65 typedef struct _TIMEZONE_ENTRY
66 {
67     struct _TIMEZONE_ENTRY *Prev;
68     struct _TIMEZONE_ENTRY *Next;
69     WCHAR Description[128]; /* 'Display' */
70     WCHAR StandardName[32]; /* 'Std' */
71     WCHAR DaylightName[32]; /* 'Dlt' */
72     REG_TZI_FORMAT TimezoneInfo; /* 'TZI' */
73     ULONG Index;
74 } TIMEZONE_ENTRY, *PTIMEZONE_ENTRY;
75 
76 
77 /* FUNCTIONS ****************************************************************/
78 
79 extern void WINAPI Control_RunDLLW(HWND hWnd, HINSTANCE hInst, LPCWSTR cmd, DWORD nCmdShow);
80 
81 
82 static VOID
83 CenterWindow(HWND hWnd)
84 {
85     HWND hWndParent;
86     RECT rcParent;
87     RECT rcWindow;
88 
89     hWndParent = GetParent(hWnd);
90     if (hWndParent == NULL)
91         hWndParent = GetDesktopWindow();
92 
93     GetWindowRect(hWndParent, &rcParent);
94     GetWindowRect(hWnd, &rcWindow);
95 
96     SetWindowPos(hWnd,
97                  HWND_TOP,
98                  ((rcParent.right - rcParent.left) - (rcWindow.right - rcWindow.left)) / 2,
99                  ((rcParent.bottom - rcParent.top) - (rcWindow.bottom - rcWindow.top)) / 2,
100                  0,
101                  0,
102                  SWP_NOSIZE);
103 }
104 
105 
106 static HFONT
107 CreateTitleFont(VOID)
108 {
109     LOGFONTW LogFont = {0};
110     HDC hdc;
111     HFONT hFont;
112 
113     LogFont.lfWeight = FW_BOLD;
114     wcscpy(LogFont.lfFaceName, L"MS Shell Dlg");
115 
116     hdc = GetDC(NULL);
117     LogFont.lfHeight = -MulDiv(12, GetDeviceCaps(hdc, LOGPIXELSY), 72);
118 
119     hFont = CreateFontIndirectW(&LogFont);
120 
121     ReleaseDC(NULL, hdc);
122 
123     return hFont;
124 }
125 
126 
127 static HFONT
128 CreateBoldFont(VOID)
129 {
130     LOGFONTW tmpFont = {0};
131     HFONT hBoldFont;
132     HDC hDc;
133 
134     /* Grabs the Drawing Context */
135     hDc = GetDC(NULL);
136 
137     tmpFont.lfHeight = -MulDiv(8, GetDeviceCaps(hDc, LOGPIXELSY), 72);
138     tmpFont.lfWeight = FW_BOLD;
139     wcscpy(tmpFont.lfFaceName, L"MS Shell Dlg");
140 
141     hBoldFont = CreateFontIndirectW(&tmpFont);
142 
143     ReleaseDC(NULL, hDc);
144 
145     return hBoldFont;
146 }
147 
148 static INT_PTR CALLBACK
149 GplDlgProc(HWND hwndDlg,
150            UINT uMsg,
151            WPARAM wParam,
152            LPARAM lParam)
153 {
154     HRSRC GplTextResource;
155     HGLOBAL GplTextMem;
156     PVOID GplTextLocked;
157     PCHAR GplText;
158     DWORD Size;
159 
160 
161     switch (uMsg)
162     {
163         case WM_INITDIALOG:
164             GplTextResource = FindResourceW(hDllInstance, MAKEINTRESOURCE(IDR_GPL), L"RT_TEXT");
165             if (NULL == GplTextResource)
166             {
167                 break;
168             }
169             Size = SizeofResource(hDllInstance, GplTextResource);
170             if (0 == Size)
171             {
172                 break;
173             }
174             GplText = HeapAlloc(GetProcessHeap(), 0, Size + 1);
175             if (NULL == GplText)
176             {
177                 break;
178             }
179             GplTextMem = LoadResource(hDllInstance, GplTextResource);
180             if (NULL == GplTextMem)
181             {
182                 HeapFree(GetProcessHeap(), 0, GplText);
183                 break;
184             }
185             GplTextLocked = LockResource(GplTextMem);
186             if (NULL == GplTextLocked)
187             {
188                 HeapFree(GetProcessHeap(), 0, GplText);
189                 break;
190             }
191             memcpy(GplText, GplTextLocked, Size);
192             GplText[Size] = '\0';
193             SendMessageA(GetDlgItem(hwndDlg, IDC_GPL_TEXT), WM_SETTEXT, 0, (LPARAM) GplText);
194             HeapFree(GetProcessHeap(), 0, GplText);
195             SetFocus(GetDlgItem(hwndDlg, IDOK));
196             return FALSE;
197 
198         case WM_CLOSE:
199             EndDialog(hwndDlg, IDCANCEL);
200             break;
201 
202         case WM_COMMAND:
203             if (HIWORD(wParam) == BN_CLICKED && IDOK == LOWORD(wParam))
204             {
205                 EndDialog(hwndDlg, IDOK);
206             }
207             break;
208 
209         default:
210             break;
211     }
212 
213     return FALSE;
214 }
215 
216 
217 static INT_PTR CALLBACK
218 WelcomeDlgProc(HWND hwndDlg,
219                UINT uMsg,
220                WPARAM wParam,
221                LPARAM lParam)
222 {
223     PSETUPDATA pSetupData;
224 
225     pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
226 
227     switch (uMsg)
228     {
229         case WM_INITDIALOG:
230         {
231             HWND hwndControl;
232             DWORD dwStyle;
233 
234             /* Get pointer to the global setup data */
235             pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
236             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
237 
238             hwndControl = GetParent(hwndDlg);
239 
240             /* Center the wizard window */
241             CenterWindow (hwndControl);
242 
243             /* Hide the system menu */
244             dwStyle = GetWindowLongPtr(hwndControl, GWL_STYLE);
245             SetWindowLongPtr(hwndControl, GWL_STYLE, dwStyle & ~WS_SYSMENU);
246 
247             /* Hide and disable the 'Cancel' button */
248             hwndControl = GetDlgItem(GetParent(hwndDlg), IDCANCEL);
249             ShowWindow (hwndControl, SW_HIDE);
250             EnableWindow (hwndControl, FALSE);
251 
252             /* Set title font */
253             SendDlgItemMessage(hwndDlg,
254                                IDC_WELCOMETITLE,
255                                WM_SETFONT,
256                                (WPARAM)pSetupData->hTitleFont,
257                                (LPARAM)TRUE);
258         }
259         break;
260 
261 
262         case WM_NOTIFY:
263         {
264             LPNMHDR lpnm = (LPNMHDR)lParam;
265 
266             switch (lpnm->code)
267             {
268                 case PSN_SETACTIVE:
269                     LogItem(L"BEGIN", L"WelcomePage");
270                     /* Enable the Next button */
271                     PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_NEXT);
272                     if (pSetupData->UnattendSetup)
273                     {
274                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_ACKPAGE);
275                         return TRUE;
276                     }
277                     break;
278 
279                 case PSN_WIZNEXT:
280                     LogItem(L"END", L"WelcomePage");
281                     break;
282 
283                 case PSN_WIZBACK:
284                     pSetupData->UnattendSetup = FALSE;
285                     break;
286 
287                 default:
288                     break;
289             }
290         }
291         break;
292 
293         default:
294             break;
295     }
296 
297     return FALSE;
298 }
299 
300 
301 static INT_PTR CALLBACK
302 AckPageDlgProc(HWND hwndDlg,
303                UINT uMsg,
304                WPARAM wParam,
305                LPARAM lParam)
306 {
307     LPNMHDR lpnm;
308     PWCHAR Projects;
309     PWCHAR End, CurrentProject;
310     INT ProjectsSize, ProjectsCount;
311     PSETUPDATA pSetupData;
312 
313     pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
314 
315     switch (uMsg)
316     {
317         case WM_INITDIALOG:
318         {
319             pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
320             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
321 
322             Projects = NULL;
323             ProjectsSize = 256;
324             while (TRUE)
325             {
326                 Projects = HeapAlloc(GetProcessHeap(), 0, ProjectsSize * sizeof(WCHAR));
327                 if (NULL == Projects)
328                 {
329                     return FALSE;
330                 }
331                 ProjectsCount =  LoadStringW(hDllInstance, IDS_ACKPROJECTS, Projects, ProjectsSize);
332                 if (0 == ProjectsCount)
333                 {
334                     HeapFree(GetProcessHeap(), 0, Projects);
335                     return FALSE;
336                 }
337                 if (ProjectsCount < ProjectsSize - 1)
338                 {
339                     break;
340                 }
341                 HeapFree(GetProcessHeap(), 0, Projects);
342                 ProjectsSize *= 2;
343             }
344 
345             CurrentProject = Projects;
346             while (*CurrentProject != L'\0')
347             {
348                 End = wcschr(CurrentProject, L'\n');
349                 if (NULL != End)
350                 {
351                     *End = L'\0';
352                 }
353                 (void)ListBox_AddString(GetDlgItem(hwndDlg, IDC_PROJECTS), CurrentProject);
354                 if (NULL != End)
355                 {
356                     CurrentProject = End + 1;
357                 }
358                 else
359                 {
360                     CurrentProject += wcslen(CurrentProject);
361                 }
362             }
363             HeapFree(GetProcessHeap(), 0, Projects);
364         }
365         break;
366 
367         case WM_COMMAND:
368             if (HIWORD(wParam) == BN_CLICKED && IDC_VIEWGPL == LOWORD(wParam))
369             {
370                 DialogBox(hDllInstance, MAKEINTRESOURCE(IDD_GPL), NULL, GplDlgProc);
371                 SetForegroundWindow(GetParent(hwndDlg));
372             }
373             break;
374 
375         case WM_NOTIFY:
376         {
377             lpnm = (LPNMHDR)lParam;
378 
379             switch (lpnm->code)
380             {
381                 case PSN_SETACTIVE:
382                     /* Enable the Back and Next buttons */
383                     PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
384                     if (pSetupData->UnattendSetup)
385                     {
386                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_PRODUCT);
387                         return TRUE;
388                     }
389                     break;
390 
391                 case PSN_WIZBACK:
392                     pSetupData->UnattendSetup = FALSE;
393                     break;
394 
395                 default:
396                     break;
397             }
398         }
399         break;
400 
401         default:
402             break;
403     }
404 
405     return FALSE;
406 }
407 
408 static const WCHAR s_szProductOptions[] = L"SYSTEM\\CurrentControlSet\\Control\\ProductOptions";
409 static const WCHAR s_szRosVersion[] = L"SYSTEM\\CurrentControlSet\\Control\\ReactOS\\Settings\\Version";
410 static const WCHAR s_szControlWindows[] = L"SYSTEM\\CurrentControlSet\\Control\\Windows";
411 static const WCHAR s_szWinlogon[] = L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon";
412 static const WCHAR s_szDefaultSoundEvents[] = L"AppEvents\\Schemes\\Apps\\.Default";
413 static const WCHAR s_szExplorerSoundEvents[] = L"AppEvents\\Schemes\\Apps\\Explorer";
414 
415 typedef struct _PRODUCT_OPTION_DATA
416 {
417     LPCWSTR ProductSuite;
418     LPCWSTR ProductType;
419     DWORD ReportAsWorkstation;
420     DWORD CSDVersion;
421     DWORD LogonType;
422 } PRODUCT_OPTION_DATA;
423 
424 static const PRODUCT_OPTION_DATA s_ProductOptionData[] =
425 {
426     { L"Terminal Server\0", L"ServerNT", 0, 0x200, 0 },
427     { L"\0", L"WinNT", 1, 0x300, 1 }
428 };
429 
430 static const WCHAR* s_DefaultSoundEvents[][2] =
431 {
432     { L".Default", L"%SystemRoot%\\Media\\ReactOS_Default.wav" },
433     { L"AppGPFault", L"" },
434     { L"Close", L"" },
435     { L"CriticalBatteryAlarm", L"%SystemRoot%\\Media\\ReactOS_Battery_Critical.wav" },
436     { L"DeviceConnect",  L"%SystemRoot%\\Media\\ReactOS_Hardware_Insert.wav" },
437     { L"DeviceDisconnect", L"%SystemRoot%\\Media\\ReactOS_Hardware_Remove.wav" },
438     { L"DeviceFail", L"%SystemRoot%\\Media\\ReactOS_Hardware_Fail.wav" },
439     { L"LowBatteryAlarm", L"%SystemRoot%\\Media\\ReactOS_Battery_Low.wav" },
440     { L"MailBeep", L"%SystemRoot%\\Media\\ReactOS_Notify.wav" },
441     { L"Maximize", L"%SystemRoot%\\Media\\ReactOS_Restore.wav" },
442     { L"MenuCommand", L"%SystemRoot%\\Media\\ReactOS_Menu_Command.wav" },
443     { L"MenuPopup", L"" },
444     { L"Minimize", L"%SystemRoot%\\Media\\ReactOS_Minimize.wav" },
445     { L"Open", L"" },
446     { L"PrintComplete", L"%SystemRoot%\\Media\\ReactOS_Print_Complete.wav" },
447     { L"RestoreDown", L"" },
448     { L"RestoreUp", L"" },
449     { L"SystemAsterisk", L"%SystemRoot%\\Media\\ReactOS_Ding.wav" },
450     { L"SystemExclamation", L"%SystemRoot%\\Media\\ReactOS_Exclamation.wav" },
451     { L"SystemExit", L"%SystemRoot%\\Media\\ReactOS_Shutdown.wav" },
452     { L"SystemHand", L"%SystemRoot%\\Media\\ReactOS_Critical_Stop.wav" },
453     { L"SystemNotification", L"%SystemRoot%\\Media\\ReactOS_Balloon.wav" },
454     { L"SystemQuestion", L"%SystemRoot%\\Media\\ReactOS_Ding.wav" },
455     { L"SystemStart", L"%SystemRoot%\\Media\\ReactOS_Startup.wav" },
456     { L"WindowsLogoff", L"%SystemRoot%\\Media\\ReactOS_LogOff.wav" }
457 /* Logon sound is already set by default for both Server and Workstation */
458 };
459 
460 static const WCHAR* s_ExplorerSoundEvents[][2] =
461 {
462     { L"EmptyRecycleBin", L"%SystemRoot%\\Media\\ReactOS_Recycle.wav" },
463     { L"Navigating", L"%SystemRoot%\\Media\\ReactOS_Start.wav" }
464 };
465 
466 static BOOL
467 DoWriteSoundEvents(HKEY hKey,
468                    LPCWSTR lpSubkey,
469                    LPCWSTR lpEventsArray[][2],
470                    DWORD dwSize)
471 {
472     HKEY hRootKey, hEventKey, hDefaultKey;
473     LONG error;
474     ULONG i;
475     WCHAR szDest[MAX_PATH];
476     DWORD dwAttribs;
477     DWORD cbData;
478 
479     /* Open the sound events key */
480     error = RegOpenKeyExW(hKey, lpSubkey, 0, KEY_READ, &hRootKey);
481     if (error)
482     {
483         DPRINT1("RegOpenKeyExW failed\n");
484         goto Error;
485     }
486 
487     /* Set each sound event */
488     for (i = 0; i < dwSize; i++)
489     {
490         /*
491          * Verify that the sound file exists and is an actual file.
492          */
493 
494         /* Expand the sound file path */
495         if (!ExpandEnvironmentStringsW(lpEventsArray[i][1], szDest, _countof(szDest)))
496         {
497             /* Failed to expand, continue with the next sound event */
498             continue;
499         }
500 
501         /* Check if the sound file exists and isn't a directory */
502         dwAttribs = GetFileAttributesW(szDest);
503         if ((dwAttribs == INVALID_FILE_ATTRIBUTES) ||
504             (dwAttribs & FILE_ATTRIBUTE_DIRECTORY))
505         {
506             /* It does not, just continue with the next sound event */
507             continue;
508         }
509 
510         /*
511          * Create the sound event entry.
512          */
513 
514         /* Open the sound event subkey */
515         error = RegOpenKeyExW(hRootKey, lpEventsArray[i][0], 0, KEY_READ, &hEventKey);
516         if (error)
517         {
518             /* Failed to open, continue with next sound event */
519             continue;
520         }
521 
522         /* Open .Default subkey */
523         error = RegOpenKeyExW(hEventKey, L".Default", 0, KEY_WRITE, &hDefaultKey);
524         RegCloseKey(hEventKey);
525         if (error)
526         {
527             /* Failed to open, continue with next sound event */
528             continue;
529         }
530 
531         /* Associate the sound file to this sound event */
532         cbData = (lstrlenW(lpEventsArray[i][1]) + 1) * sizeof(WCHAR);
533         error = RegSetValueExW(hDefaultKey, NULL, 0, REG_EXPAND_SZ, (const BYTE *)lpEventsArray[i][1], cbData);
534         RegCloseKey(hDefaultKey);
535         if (error)
536         {
537             /* Failed to set the value, continue with next sound event */
538             continue;
539         }
540     }
541 
542 Error:
543     if (hRootKey)
544         RegCloseKey(hRootKey);
545 
546     return error == ERROR_SUCCESS;
547 }
548 
549 static BOOL
550 DoWriteProductOption(PRODUCT_OPTION nOption)
551 {
552     HKEY hKey;
553     LONG error;
554     LPCWSTR pszData;
555     DWORD dwValue, cbData;
556     const PRODUCT_OPTION_DATA *pData = &s_ProductOptionData[nOption];
557     ASSERT(0 <= nOption && nOption < _countof(s_ProductOptionData));
558 
559     /* open ProductOptions key */
560     error = RegOpenKeyExW(HKEY_LOCAL_MACHINE, s_szProductOptions, 0, KEY_WRITE, &hKey);
561     if (error)
562     {
563         DPRINT1("RegOpenKeyExW failed\n");
564         goto Error;
565     }
566 
567     /* write ProductSuite */
568     pszData = pData->ProductSuite;
569     cbData = (lstrlenW(pszData) + 2) * sizeof(WCHAR);
570     error = RegSetValueExW(hKey, L"ProductSuite", 0, REG_MULTI_SZ, (const BYTE *)pszData, cbData);
571     if (error)
572     {
573         DPRINT1("RegSetValueExW failed\n");
574         goto Error;
575     }
576 
577     /* write ProductType */
578     pszData = pData->ProductType;
579     cbData = (lstrlenW(pszData) + 1) * sizeof(WCHAR);
580     error = RegSetValueExW(hKey, L"ProductType", 0, REG_SZ, (const BYTE *)pszData, cbData);
581     if (error)
582     {
583         DPRINT1("RegSetValueExW failed\n");
584         goto Error;
585     }
586 
587     RegCloseKey(hKey);
588 
589     /* open ReactOS version key */
590     error = RegOpenKeyExW(HKEY_LOCAL_MACHINE, s_szRosVersion, 0, KEY_WRITE, &hKey);
591     if (error)
592     {
593         DPRINT1("RegOpenKeyExW failed\n");
594         goto Error;
595     }
596 
597     /* write ReportAsWorkstation */
598     dwValue = pData->ReportAsWorkstation;
599     cbData = sizeof(dwValue);
600     error = RegSetValueExW(hKey, L"ReportAsWorkstation", 0, REG_DWORD, (const BYTE *)&dwValue, cbData);
601     if (error)
602     {
603         DPRINT1("RegSetValueExW failed\n");
604         goto Error;
605     }
606 
607     RegCloseKey(hKey);
608 
609     /* open Control Windows key */
610     error = RegOpenKeyExW(HKEY_LOCAL_MACHINE, s_szControlWindows, 0, KEY_WRITE, &hKey);
611     if (error)
612     {
613         DPRINT1("RegOpenKeyExW failed\n");
614         goto Error;
615     }
616 
617     /* write Control Windows CSDVersion */
618     dwValue = pData->CSDVersion;
619     cbData = sizeof(dwValue);
620     error = RegSetValueExW(hKey, L"CSDVersion", 0, REG_DWORD, (const BYTE *)&dwValue, cbData);
621     if (error)
622     {
623         DPRINT1("RegSetValueExW failed\n");
624         goto Error;
625     }
626 
627     RegCloseKey(hKey);
628 
629     /* open Winlogon key */
630     error = RegOpenKeyExW(HKEY_LOCAL_MACHINE, s_szWinlogon, 0, KEY_WRITE, &hKey);
631     if (error)
632     {
633         DPRINT1("RegOpenKeyExW failed\n");
634         goto Error;
635     }
636 
637     /* write LogonType */
638     dwValue = pData->LogonType;
639     cbData = sizeof(dwValue);
640     error = RegSetValueExW(hKey, L"LogonType", 0, REG_DWORD, (const BYTE *)&dwValue, cbData);
641     if (error)
642     {
643         DPRINT1("RegSetValueExW failed\n");
644         goto Error;
645     }
646 
647     if (nOption == PRODUCT_OPTION_WORKSTATION)
648     {
649         /* Write system sound events values for Workstation */
650         DoWriteSoundEvents(HKEY_CURRENT_USER, s_szDefaultSoundEvents, s_DefaultSoundEvents, _countof(s_DefaultSoundEvents));
651         DoWriteSoundEvents(HKEY_CURRENT_USER, s_szExplorerSoundEvents, s_ExplorerSoundEvents, _countof(s_ExplorerSoundEvents));
652     }
653 
654 Error:
655     if (hKey)
656         RegCloseKey(hKey);
657 
658     return error == ERROR_SUCCESS;
659 }
660 
661 static void
662 OnChooseOption(HWND hwndDlg, PRODUCT_OPTION nOption)
663 {
664     WCHAR szText[256];
665     ASSERT(0 <= nOption && nOption < _countof(s_ProductOptionData));
666 
667     switch (nOption)
668     {
669         case PRODUCT_OPTION_SERVER:
670             LoadStringW(hDllInstance, IDS_PRODUCTSERVERINFO, szText, _countof(szText));
671             break;
672 
673         case PRODUCT_OPTION_WORKSTATION:
674             LoadStringW(hDllInstance, IDS_PRODUCTWORKSTATIONINFO, szText, _countof(szText));
675             break;
676 
677         default:
678             return;
679     }
680 
681     SetDlgItemTextW(hwndDlg, IDC_PRODUCT_DESCRIPTION, szText);
682 }
683 
684 static INT_PTR CALLBACK
685 ProductPageDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
686 {
687     LPNMHDR lpnm;
688     PSETUPDATA pSetupData;
689     INT iItem;
690     WCHAR szText[64], szDefault[64];
691     HICON hIcon;
692 
693     pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
694 
695     switch (uMsg)
696     {
697         case WM_INITDIALOG:
698         {
699             pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
700             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
701 
702             LoadStringW(hDllInstance, IDS_DEFAULT, szDefault, _countof(szDefault));
703 
704             LoadStringW(hDllInstance, IDS_PRODUCTSERVERNAME, szText, _countof(szText));
705             if (PRODUCT_OPTION_DEFAULT == PRODUCT_OPTION_SERVER)
706             {
707                 StringCchCatW(szText, _countof(szText), L" ");
708                 StringCchCatW(szText, _countof(szText), szDefault);
709             }
710             SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_ADDSTRING, 0, (LPARAM)szText);
711 
712             LoadStringW(hDllInstance, IDS_PRODUCTWORKSTATIONNAME, szText, _countof(szText));
713             if (PRODUCT_OPTION_DEFAULT == PRODUCT_OPTION_WORKSTATION)
714             {
715                 StringCchCatW(szText, _countof(szText), L" ");
716                 StringCchCatW(szText, _countof(szText), szDefault);
717             }
718             SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_ADDSTRING, 0, (LPARAM)szText);
719 
720             SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_SETCURSEL, PRODUCT_OPTION_DEFAULT, 0);
721             OnChooseOption(hwndDlg, PRODUCT_OPTION_DEFAULT);
722 
723             hIcon = LoadIcon(NULL, IDI_WINLOGO);
724             SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_ICON, STM_SETICON, (WPARAM)hIcon, 0);
725             return TRUE;
726         }
727 
728         case WM_COMMAND:
729             if (HIWORD(wParam) == CBN_SELCHANGE && IDC_PRODUCT_OPTIONS == LOWORD(wParam))
730             {
731                 iItem = SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_GETCURSEL, 0, 0);
732                 OnChooseOption(hwndDlg, (PRODUCT_OPTION)iItem);
733             }
734             break;
735 
736         case WM_NOTIFY:
737         {
738             lpnm = (LPNMHDR)lParam;
739 
740             switch (lpnm->code)
741             {
742                 case PSN_SETACTIVE:
743                     /* Enable the Back and Next buttons */
744                     PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
745                     if (pSetupData->UnattendSetup)
746                     {
747                         OnChooseOption(hwndDlg, pSetupData->ProductOption);
748                         DoWriteProductOption(pSetupData->ProductOption);
749                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_LOCALEPAGE);
750                         return TRUE;
751                     }
752                     break;
753 
754                 case PSN_WIZNEXT:
755                     iItem = SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_GETCURSEL, 0, 0);
756                     pSetupData->ProductOption = (PRODUCT_OPTION)iItem;
757                     DoWriteProductOption(pSetupData->ProductOption);
758                     break;
759 
760                 case PSN_WIZBACK:
761                     pSetupData->UnattendSetup = FALSE;
762                     break;
763 
764                 default:
765                     break;
766             }
767         }
768         break;
769 
770         default:
771             break;
772     }
773 
774     return FALSE;
775 }
776 
777 static
778 BOOL
779 WriteOwnerSettings(WCHAR * OwnerName,
780                    WCHAR * OwnerOrganization)
781 {
782     HKEY hKey;
783     LONG res;
784 
785     res = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
786                         L"Software\\Microsoft\\Windows NT\\CurrentVersion",
787                         0,
788                         KEY_ALL_ACCESS,
789                         &hKey);
790 
791     if (res != ERROR_SUCCESS)
792     {
793         return FALSE;
794     }
795 
796     res = RegSetValueExW(hKey,
797                          L"RegisteredOwner",
798                          0,
799                          REG_SZ,
800                          (LPBYTE)OwnerName,
801                          (wcslen(OwnerName) + 1) * sizeof(WCHAR));
802 
803     if (res != ERROR_SUCCESS)
804     {
805         RegCloseKey(hKey);
806         return FALSE;
807     }
808 
809     res = RegSetValueExW(hKey,
810                          L"RegisteredOrganization",
811                          0,
812                          REG_SZ,
813                          (LPBYTE)OwnerOrganization,
814                          (wcslen(OwnerOrganization) + 1) * sizeof(WCHAR));
815 
816     RegCloseKey(hKey);
817     return (res == ERROR_SUCCESS);
818 }
819 
820 static INT_PTR CALLBACK
821 OwnerPageDlgProc(HWND hwndDlg,
822                  UINT uMsg,
823                  WPARAM wParam,
824                  LPARAM lParam)
825 {
826     WCHAR OwnerName[51];
827     WCHAR OwnerOrganization[51];
828     WCHAR Title[64];
829     WCHAR ErrorName[256];
830     LPNMHDR lpnm;
831     PSETUPDATA pSetupData;
832 
833     pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
834 
835     switch (uMsg)
836     {
837         case WM_INITDIALOG:
838         {
839             pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
840             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
841 
842             /* set a localized ('Owner') placeholder string as default */
843             if (LoadStringW(hDllInstance, IDS_MACHINE_OWNER_NAME, OwnerName, _countof(OwnerName)))
844             {
845                 SendDlgItemMessage(hwndDlg, IDC_OWNERNAME, WM_SETTEXT, 0, (LPARAM)OwnerName);
846             }
847 
848             SendDlgItemMessage(hwndDlg, IDC_OWNERNAME, EM_LIMITTEXT, 50, 0);
849             SendDlgItemMessage(hwndDlg, IDC_OWNERORGANIZATION, EM_LIMITTEXT, 50, 0);
850 
851             /* Set focus to owner name */
852             SetFocus(GetDlgItem(hwndDlg, IDC_OWNERNAME));
853 
854             /* Select the default text to quickly overwrite it by typing */
855             SendDlgItemMessage(hwndDlg, IDC_OWNERNAME, EM_SETSEL, 0, -1);
856         }
857         break;
858 
859 
860         case WM_NOTIFY:
861         {
862             lpnm = (LPNMHDR)lParam;
863 
864             switch (lpnm->code)
865             {
866                 case PSN_SETACTIVE:
867                     /* Enable the Back and Next buttons */
868                     PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
869                     if (pSetupData->UnattendSetup)
870                     {
871                         SendMessage(GetDlgItem(hwndDlg, IDC_OWNERNAME), WM_SETTEXT, 0, (LPARAM)pSetupData->OwnerName);
872                         SendMessage(GetDlgItem(hwndDlg, IDC_OWNERORGANIZATION), WM_SETTEXT, 0, (LPARAM)pSetupData->OwnerOrganization);
873                         if (WriteOwnerSettings(pSetupData->OwnerName, pSetupData->OwnerOrganization))
874                         {
875                             SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_COMPUTERPAGE);
876                             return TRUE;
877                         }
878                     }
879                     break;
880 
881                 case PSN_WIZNEXT:
882                     OwnerName[0] = 0;
883                     if (GetDlgItemTextW(hwndDlg, IDC_OWNERNAME, OwnerName, 50) == 0)
884                     {
885                         if (0 == LoadStringW(hDllInstance, IDS_REACTOS_SETUP, Title, ARRAYSIZE(Title)))
886                         {
887                             wcscpy(Title, L"ReactOS Setup");
888                         }
889                         if (0 == LoadStringW(hDllInstance, IDS_WZD_NAME, ErrorName, ARRAYSIZE(ErrorName)))
890                         {
891                             wcscpy(ErrorName, L"Setup cannot continue until you enter your name.");
892                         }
893                         MessageBoxW(hwndDlg, ErrorName, Title, MB_ICONERROR | MB_OK);
894 
895                         SetFocus(GetDlgItem(hwndDlg, IDC_OWNERNAME));
896                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
897 
898                         return TRUE;
899                     }
900 
901                     OwnerOrganization[0] = 0;
902                     GetDlgItemTextW(hwndDlg, IDC_OWNERORGANIZATION, OwnerOrganization, 50);
903 
904                     if (!WriteOwnerSettings(OwnerName, OwnerOrganization))
905                     {
906                         SetFocus(GetDlgItem(hwndDlg, IDC_OWNERNAME));
907                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
908                         return TRUE;
909                     }
910 
911                 case PSN_WIZBACK:
912                     pSetupData->UnattendSetup = FALSE;
913                     break;
914 
915                 default:
916                     break;
917             }
918         }
919         break;
920 
921         default:
922             break;
923     }
924 
925     return FALSE;
926 }
927 
928 static
929 BOOL
930 WriteComputerSettings(WCHAR * ComputerName, HWND hwndDlg)
931 {
932     WCHAR Title[64];
933     WCHAR ErrorComputerName[256];
934     LONG lError;
935     HKEY hKey = NULL;
936 
937     if (!SetComputerNameW(ComputerName))
938     {
939         if (hwndDlg != NULL)
940         {
941             if (0 == LoadStringW(hDllInstance, IDS_REACTOS_SETUP, Title, ARRAYSIZE(Title)))
942             {
943                 wcscpy(Title, L"ReactOS Setup");
944             }
945             if (0 == LoadStringW(hDllInstance, IDS_WZD_SETCOMPUTERNAME, ErrorComputerName,
946                                  ARRAYSIZE(ErrorComputerName)))
947             {
948                 wcscpy(ErrorComputerName, L"Setup failed to set the computer name.");
949             }
950             MessageBoxW(hwndDlg, ErrorComputerName, Title, MB_ICONERROR | MB_OK);
951         }
952 
953         return FALSE;
954     }
955 
956     /* Set the physical DNS domain */
957     SetComputerNameExW(ComputerNamePhysicalDnsDomain, L"");
958 
959     /* Set the physical DNS hostname */
960     SetComputerNameExW(ComputerNamePhysicalDnsHostname, ComputerName);
961 
962     /* Set the accounts domain name */
963     SetAccountsDomainSid(NULL, ComputerName);
964 
965     /* Now we need to set the Hostname */
966     lError = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
967                            L"SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",
968                            0,
969                            KEY_SET_VALUE,
970                            &hKey);
971     if (lError != ERROR_SUCCESS)
972     {
973         DPRINT1("RegOpenKeyExW for Tcpip\\Parameters failed (%08lX)\n", lError);
974         return TRUE;
975     }
976 
977     lError = RegSetValueEx(hKey,
978                            L"Hostname",
979                            0,
980                            REG_SZ,
981                            (LPBYTE)ComputerName,
982                            (wcslen(ComputerName) + 1) * sizeof(WCHAR));
983     if (lError != ERROR_SUCCESS)
984     {
985         DPRINT1("RegSetValueEx(\"Hostname\") failed (%08lX)\n", lError);
986     }
987 
988     RegCloseKey(hKey);
989 
990     return TRUE;
991 }
992 
993 
994 static
995 BOOL
996 WriteDefaultLogonData(LPWSTR Domain)
997 {
998     WCHAR szAdministratorName[256];
999     HKEY hKey = NULL;
1000     LONG lError;
1001 
1002     if (LoadStringW(hDllInstance,
1003                     IDS_ADMINISTRATOR_NAME,
1004                     szAdministratorName,
1005                     ARRAYSIZE(szAdministratorName)) == 0)
1006     {
1007         wcscpy(szAdministratorName, L"Administrator");
1008     }
1009 
1010     lError = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
1011                            L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon",
1012                            0,
1013                            KEY_SET_VALUE,
1014                            &hKey);
1015     if (lError != ERROR_SUCCESS)
1016         return FALSE;
1017 
1018     lError = RegSetValueEx(hKey,
1019                            L"DefaultDomainName",
1020                            0,
1021                            REG_SZ,
1022                            (LPBYTE)Domain,
1023                            (wcslen(Domain)+ 1) * sizeof(WCHAR));
1024     if (lError != ERROR_SUCCESS)
1025     {
1026         DPRINT1("RegSetValueEx(\"DefaultDomainName\") failed!\n");
1027     }
1028 
1029     lError = RegSetValueEx(hKey,
1030                            L"DefaultUserName",
1031                            0,
1032                            REG_SZ,
1033                            (LPBYTE)szAdministratorName,
1034                            (wcslen(szAdministratorName)+ 1) * sizeof(WCHAR));
1035     if (lError != ERROR_SUCCESS)
1036     {
1037         DPRINT1("RegSetValueEx(\"DefaultUserName\") failed!\n");
1038     }
1039 
1040     RegCloseKey(hKey);
1041 
1042     return TRUE;
1043 }
1044 
1045 
1046 /* lpBuffer will be filled with a 15-char string (plus the null terminator) */
1047 static void
1048 GenerateComputerName(LPWSTR lpBuffer)
1049 {
1050     static const WCHAR Chars[] = L"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ";
1051     static const unsigned cChars = sizeof(Chars) / sizeof(WCHAR) - 1;
1052     unsigned i;
1053 
1054     wcscpy(lpBuffer, L"REACTOS-");
1055 
1056     srand(GetTickCount());
1057 
1058     /* fill in 7 characters */
1059     for (i = 8; i < 15; i++)
1060         lpBuffer[i] = Chars[rand() % cChars];
1061 
1062     lpBuffer[15] = UNICODE_NULL; /* NULL-terminate */
1063 }
1064 
1065 static INT_PTR CALLBACK
1066 ComputerPageDlgProc(HWND hwndDlg,
1067                     UINT uMsg,
1068                     WPARAM wParam,
1069                     LPARAM lParam)
1070 {
1071     WCHAR ComputerName[MAX_COMPUTERNAME_LENGTH + 1];
1072     WCHAR Password1[128];
1073     WCHAR Password2[128];
1074     PWCHAR Password;
1075     WCHAR Title[64];
1076     WCHAR EmptyComputerName[256], NotMatchPassword[256], WrongPassword[256];
1077     LPNMHDR lpnm;
1078     PSETUPDATA pSetupData;
1079 
1080     pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
1081 
1082     if (0 == LoadStringW(hDllInstance, IDS_REACTOS_SETUP, Title, ARRAYSIZE(Title)))
1083     {
1084         wcscpy(Title, L"ReactOS Setup");
1085     }
1086 
1087     switch (uMsg)
1088     {
1089         case WM_INITDIALOG:
1090             pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
1091             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
1092 
1093             /* Generate a new pseudo-random computer name */
1094             GenerateComputerName(ComputerName);
1095 
1096             /* Display current computer name */
1097             SetDlgItemTextW(hwndDlg, IDC_COMPUTERNAME, ComputerName);
1098 
1099             /* Set text limits */
1100             SendDlgItemMessage(hwndDlg, IDC_COMPUTERNAME, EM_LIMITTEXT, MAX_COMPUTERNAME_LENGTH, 0);
1101             SendDlgItemMessage(hwndDlg, IDC_ADMINPASSWORD1, EM_LIMITTEXT, 127, 0);
1102             SendDlgItemMessage(hwndDlg, IDC_ADMINPASSWORD2, EM_LIMITTEXT, 127, 0);
1103 
1104             /* Set focus to computer name */
1105             SetFocus(GetDlgItem(hwndDlg, IDC_COMPUTERNAME));
1106             if (pSetupData->UnattendSetup)
1107             {
1108                 SendMessage(GetDlgItem(hwndDlg, IDC_COMPUTERNAME), WM_SETTEXT, 0, (LPARAM)pSetupData->ComputerName);
1109                 SendMessage(GetDlgItem(hwndDlg, IDC_ADMINPASSWORD1), WM_SETTEXT, 0, (LPARAM)pSetupData->AdminPassword);
1110                 SendMessage(GetDlgItem(hwndDlg, IDC_ADMINPASSWORD2), WM_SETTEXT, 0, (LPARAM)pSetupData->AdminPassword);
1111                 WriteComputerSettings(pSetupData->ComputerName, NULL);
1112                 SetAdministratorPassword(pSetupData->AdminPassword);
1113             }
1114 
1115             /* Store the administrator account name as the default user name */
1116             WriteDefaultLogonData(pSetupData->ComputerName);
1117             break;
1118 
1119 
1120         case WM_NOTIFY:
1121         {
1122             lpnm = (LPNMHDR)lParam;
1123 
1124             switch (lpnm->code)
1125             {
1126                 case PSN_SETACTIVE:
1127                     /* Enable the Back and Next buttons */
1128                     PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
1129                     if (pSetupData->UnattendSetup && WriteComputerSettings(pSetupData->ComputerName, hwndDlg))
1130                     {
1131                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_THEMEPAGE);
1132                         return TRUE;
1133                     }
1134                     break;
1135 
1136                 case PSN_WIZNEXT:
1137                     if (0 == GetDlgItemTextW(hwndDlg, IDC_COMPUTERNAME, ComputerName, MAX_COMPUTERNAME_LENGTH + 1))
1138                     {
1139                         if (0 == LoadStringW(hDllInstance, IDS_WZD_COMPUTERNAME, EmptyComputerName,
1140                                              ARRAYSIZE(EmptyComputerName)))
1141                         {
1142                             wcscpy(EmptyComputerName, L"Setup cannot continue until you enter the name of your computer.");
1143                         }
1144                         MessageBoxW(hwndDlg, EmptyComputerName, Title, MB_ICONERROR | MB_OK);
1145                         SetFocus(GetDlgItem(hwndDlg, IDC_COMPUTERNAME));
1146                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
1147                         return TRUE;
1148                     }
1149 
1150                     /* No need to check computer name for invalid characters,
1151                      * SetComputerName() will do it for us */
1152 
1153                     if (!WriteComputerSettings(ComputerName, hwndDlg))
1154                     {
1155                         SetFocus(GetDlgItem(hwndDlg, IDC_COMPUTERNAME));
1156                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
1157                         return TRUE;
1158                     }
1159 
1160 #ifdef PASSWORDS_MANDATORY
1161                     /* Check if admin passwords have been entered */
1162                     if ((GetDlgItemText(hwndDlg, IDC_ADMINPASSWORD1, Password1, 128) == 0) ||
1163                         (GetDlgItemText(hwndDlg, IDC_ADMINPASSWORD2, Password2, 128) == 0))
1164                     {
1165                         if (0 == LoadStringW(hDllInstance, IDS_WZD_PASSWORDEMPTY, EmptyPassword,
1166                                              ARRAYSIZE(EmptyPassword)))
1167                         {
1168                             wcscpy(EmptyPassword, L"You must enter a password !");
1169                         }
1170                         MessageBoxW(hwndDlg, EmptyPassword, Title, MB_ICONERROR | MB_OK);
1171                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
1172                         return TRUE;
1173                     }
1174 #else
1175                     GetDlgItemTextW(hwndDlg, IDC_ADMINPASSWORD1, Password1, 128);
1176                     GetDlgItemTextW(hwndDlg, IDC_ADMINPASSWORD2, Password2, 128);
1177 #endif
1178                     /* Check if passwords match */
1179                     if (wcscmp(Password1, Password2))
1180                     {
1181                         if (0 == LoadStringW(hDllInstance, IDS_WZD_PASSWORDMATCH, NotMatchPassword,
1182                                              ARRAYSIZE(NotMatchPassword)))
1183                         {
1184                             wcscpy(NotMatchPassword, L"The passwords you entered do not match. Please enter the desired password again.");
1185                         }
1186                         MessageBoxW(hwndDlg, NotMatchPassword, Title, MB_ICONERROR | MB_OK);
1187                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
1188                         return TRUE;
1189                     }
1190 
1191                     /* Check password for invalid characters */
1192                     Password = (PWCHAR)Password1;
1193                     while (*Password)
1194                     {
1195                         if (!isprint(*Password))
1196                         {
1197                             if (0 == LoadStringW(hDllInstance, IDS_WZD_PASSWORDCHAR, WrongPassword,
1198                                                  ARRAYSIZE(WrongPassword)))
1199                             {
1200                                 wcscpy(WrongPassword, L"The password you entered contains invalid characters. Please enter a cleaned password.");
1201                             }
1202                             MessageBoxW(hwndDlg, WrongPassword, Title, MB_ICONERROR | MB_OK);
1203                             SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
1204                             return TRUE;
1205                         }
1206                         Password++;
1207                     }
1208 
1209                     /* Set admin password */
1210                     SetAdministratorPassword(Password1);
1211                     break;
1212 
1213                 case PSN_WIZBACK:
1214                     pSetupData->UnattendSetup = FALSE;
1215                     break;
1216 
1217                 default:
1218                     break;
1219             }
1220         }
1221         break;
1222 
1223         default:
1224             break;
1225     }
1226 
1227     return FALSE;
1228 }
1229 
1230 
1231 static VOID
1232 SetUserLocaleName(HWND hwnd)
1233 {
1234     WCHAR CurLocale[256] = L"";
1235     WCHAR CurGeo[256] = L"";
1236     WCHAR ResText[256] = L"";
1237     WCHAR LocaleText[256 * 2];
1238 
1239     GetLocaleInfoW(GetUserDefaultLCID(), LOCALE_SLANGUAGE, CurLocale, ARRAYSIZE(CurLocale));
1240     GetGeoInfoW(GetUserGeoID(GEOCLASS_NATION), GEO_FRIENDLYNAME, CurGeo, ARRAYSIZE(CurGeo), GetThreadLocale());
1241 
1242     LoadStringW(hDllInstance, IDS_LOCALETEXT, ResText, ARRAYSIZE(ResText));
1243     StringCchPrintfW(LocaleText, ARRAYSIZE(LocaleText), ResText, CurLocale, CurGeo);
1244 
1245     SetWindowTextW(hwnd, LocaleText);
1246 }
1247 
1248 static VOID
1249 SetKeyboardLayoutName(HWND hwnd)
1250 {
1251     HKL hkl;
1252     BOOL LayoutSpecial = FALSE;
1253     WCHAR LayoutPath[256];
1254     WCHAR LocaleName[32];
1255     WCHAR SpecialId[5] = L"";
1256     WCHAR ResText[256] = L"";
1257     DWORD dwValueSize;
1258     HKEY hKey;
1259     UINT i;
1260 
1261     /* Get the default input language and method */
1262     if (!SystemParametersInfoW(SPI_GETDEFAULTINPUTLANG, 0, (LPDWORD)&hkl, 0))
1263     {
1264         hkl = GetKeyboardLayout(0);
1265     }
1266 
1267     if ((HIWORD(hkl) & 0xF000) == 0xF000)
1268     {
1269         /* Process keyboard layout with special id */
1270         StringCchPrintfW(SpecialId, ARRAYSIZE(SpecialId), L"%04x", (HIWORD(hkl) & 0x0FFF));
1271         LayoutSpecial = TRUE;
1272     }
1273 
1274 #define MAX_LAYOUTS_PER_LANGID 0x10000
1275     for (i = 0; i < (LayoutSpecial ? MAX_LAYOUTS_PER_LANGID : 1); i++)
1276     {
1277         /* Generate a hexadecimal identifier for keyboard layout registry key */
1278         StringCchPrintfW(LocaleName, ARRAYSIZE(LocaleName), L"%08lx", (i << 16) | LOWORD(hkl));
1279 
1280         StringCchCopyW(LayoutPath, ARRAYSIZE(LayoutPath), L"SYSTEM\\CurrentControlSet\\Control\\Keyboard Layouts\\");
1281         StringCchCatW(LayoutPath, ARRAYSIZE(LayoutPath), LocaleName);
1282         *LocaleName = UNICODE_NULL;
1283 
1284         if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
1285                           LayoutPath,
1286                           0,
1287                           KEY_ALL_ACCESS,
1288                           &hKey) == ERROR_SUCCESS)
1289         {
1290             /* Make sure the keyboard layout key we opened is the one we need.
1291              * If the layout has no special id, just pass this check. */
1292             dwValueSize = sizeof(LocaleName);
1293             if (!LayoutSpecial ||
1294                 ((RegQueryValueExW(hKey,
1295                                    L"Layout Id",
1296                                    NULL,
1297                                    NULL,
1298                                    (PVOID)&LocaleName,
1299                                    &dwValueSize) == ERROR_SUCCESS) &&
1300                 (wcscmp(LocaleName, SpecialId) == 0)))
1301             {
1302                 *LocaleName = UNICODE_NULL;
1303                 dwValueSize = sizeof(LocaleName);
1304                 RegQueryValueExW(hKey,
1305                                  L"Layout Text",
1306                                  NULL,
1307                                  NULL,
1308                                  (PVOID)&LocaleName,
1309                                  &dwValueSize);
1310                 /* Let the loop know where to stop */
1311                 i = MAX_LAYOUTS_PER_LANGID;
1312             }
1313             RegCloseKey(hKey);
1314         }
1315         else
1316         {
1317             /* Keyboard layout registry keys are expected to go in order without gaps */
1318             break;
1319         }
1320     }
1321 #undef MAX_LAYOUTS_PER_LANGID
1322 
1323     LoadStringW(hDllInstance, IDS_LAYOUTTEXT, ResText, ARRAYSIZE(ResText));
1324     StringCchPrintfW(LayoutPath, ARRAYSIZE(LayoutPath), ResText, LocaleName);
1325 
1326     SetWindowTextW(hwnd, LayoutPath);
1327 }
1328 
1329 
1330 static BOOL
1331 RunControlPanelApplet(HWND hwnd, PCWSTR pwszCPLParameters)
1332 {
1333     MSG msg;
1334     HWND MainWindow = GetParent(hwnd);
1335     STARTUPINFOW StartupInfo;
1336     PROCESS_INFORMATION ProcessInformation;
1337     WCHAR CmdLine[MAX_PATH] = L"rundll32.exe shell32.dll,Control_RunDLL ";
1338 
1339     if (!pwszCPLParameters)
1340     {
1341         MessageBoxW(hwnd, L"Error: Failed to launch the Control Panel Applet.", NULL, MB_ICONERROR);
1342         return FALSE;
1343     }
1344 
1345     ZeroMemory(&StartupInfo, sizeof(StartupInfo));
1346     StartupInfo.cb = sizeof(StartupInfo);
1347     ZeroMemory(&ProcessInformation, sizeof(ProcessInformation));
1348 
1349     ASSERT(_countof(CmdLine) > wcslen(CmdLine) + wcslen(pwszCPLParameters));
1350     wcscat(CmdLine, pwszCPLParameters);
1351 
1352     if (!CreateProcessW(NULL,
1353                         CmdLine,
1354                         NULL,
1355                         NULL,
1356                         FALSE,
1357                         0,
1358                         NULL,
1359                         NULL,
1360                         &StartupInfo,
1361                         &ProcessInformation))
1362     {
1363         MessageBoxW(hwnd, L"Error: Failed to launch the Control Panel Applet.", NULL, MB_ICONERROR);
1364         return FALSE;
1365     }
1366 
1367     /* Disable the Back and Next buttons and the main window
1368      * while we're interacting with the control panel applet */
1369     PropSheet_SetWizButtons(MainWindow, 0);
1370     EnableWindow(MainWindow, FALSE);
1371 
1372     while ((MsgWaitForMultipleObjects(1, &ProcessInformation.hProcess, FALSE, INFINITE, QS_ALLINPUT|QS_ALLPOSTMESSAGE )) != WAIT_OBJECT_0)
1373     {
1374        /* We still need to process main window messages to avoid freeze */
1375        while (PeekMessageW(&msg, NULL, 0, 0, PM_REMOVE))
1376        {
1377            TranslateMessage(&msg);
1378            DispatchMessageW(&msg);
1379        }
1380     }
1381     CloseHandle(ProcessInformation.hThread);
1382     CloseHandle(ProcessInformation.hProcess);
1383 
1384     /* Enable the Back and Next buttons and the main window again */
1385     PropSheet_SetWizButtons(MainWindow, PSWIZB_BACK | PSWIZB_NEXT);
1386     EnableWindow(MainWindow, TRUE);
1387 
1388     return TRUE;
1389 }
1390 
1391 static VOID
1392 WriteUserLocale(VOID)
1393 {
1394     HKEY hKey;
1395     LCID lcid;
1396     WCHAR Locale[12];
1397 
1398     lcid = GetSystemDefaultLCID();
1399 
1400     if (GetLocaleInfoW(MAKELCID(lcid, SORT_DEFAULT), LOCALE_ILANGUAGE, Locale, ARRAYSIZE(Locale)) != 0)
1401     {
1402         if (RegCreateKeyExW(HKEY_CURRENT_USER, L"Control Panel\\International",
1403                             0, NULL, REG_OPTION_NON_VOLATILE,
1404                             KEY_WRITE, NULL, &hKey, NULL) == ERROR_SUCCESS)
1405         {
1406             RegSetValueExW(hKey, L"Locale", 0, REG_SZ, (LPBYTE)Locale, (wcslen(Locale) + 1) * sizeof(WCHAR));
1407             RegCloseKey(hKey);
1408         }
1409     }
1410 }
1411 
1412 static INT_PTR CALLBACK
1413 LocalePageDlgProc(HWND hwndDlg,
1414                   UINT uMsg,
1415                   WPARAM wParam,
1416                   LPARAM lParam)
1417 {
1418     PSETUPDATA SetupData;
1419 
1420     /* Retrieve pointer to the global setup data */
1421     SetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
1422 
1423     switch (uMsg)
1424     {
1425         case WM_INITDIALOG:
1426         {
1427             /* Save pointer to the global setup data */
1428             SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
1429             SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (DWORD_PTR)SetupData);
1430             WriteUserLocale();
1431 
1432             SetUserLocaleName(GetDlgItem(hwndDlg, IDC_LOCALETEXT));
1433             SetKeyboardLayoutName(GetDlgItem(hwndDlg, IDC_LAYOUTTEXT));
1434         }
1435         break;
1436 
1437         case WM_COMMAND:
1438             if (HIWORD(wParam) == BN_CLICKED)
1439             {
1440                 switch (LOWORD(wParam))
1441                 {
1442                     case IDC_CUSTOMLOCALE:
1443                         RunControlPanelApplet(hwndDlg, L"intl.cpl,,5");
1444                         SetUserLocaleName(GetDlgItem(hwndDlg, IDC_LOCALETEXT));
1445                         break;
1446 
1447                     case IDC_CUSTOMLAYOUT:
1448                         RunControlPanelApplet(hwndDlg, L"input.dll,@1");
1449                         SetKeyboardLayoutName(GetDlgItem(hwndDlg, IDC_LAYOUTTEXT));
1450                         break;
1451                 }
1452             }
1453             break;
1454 
1455         case WM_NOTIFY:
1456         {
1457             LPNMHDR lpnm = (LPNMHDR)lParam;
1458 
1459             switch (lpnm->code)
1460             {
1461                 case PSN_SETACTIVE:
1462                     /* Enable the Back and Next buttons */
1463                     PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
1464                     if (SetupData->UnattendSetup)
1465                     {
1466                         // if (!*SetupData->SourcePath)
1467                         {
1468                             RunControlPanelApplet(hwndDlg, L"intl.cpl,,/f:\"$winnt$.inf\""); // Should be in System32
1469                         }
1470 
1471                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_OWNERPAGE);
1472                         return TRUE;
1473                     }
1474                     break;
1475 
1476                 case PSN_WIZNEXT:
1477                     break;
1478 
1479                 case PSN_WIZBACK:
1480                     SetupData->UnattendSetup = FALSE;
1481                     break;
1482 
1483                 default:
1484                     break;
1485             }
1486         }
1487         break;
1488 
1489         default:
1490             break;
1491     }
1492 
1493     return FALSE;
1494 }
1495 
1496 
1497 static PTIMEZONE_ENTRY
1498 GetLargerTimeZoneEntry(PSETUPDATA SetupData, DWORD Index)
1499 {
1500     PTIMEZONE_ENTRY Entry;
1501 
1502     Entry = SetupData->TimeZoneListHead;
1503     while (Entry != NULL)
1504     {
1505         if (Entry->Index >= Index)
1506             return Entry;
1507 
1508         Entry = Entry->Next;
1509     }
1510 
1511     return NULL;
1512 }
1513 
1514 static LONG
1515 RetrieveTimeZone(
1516     IN HKEY hZoneKey,
1517     IN PVOID Context)
1518 {
1519     LONG lError;
1520     PSETUPDATA SetupData = (PSETUPDATA)Context;
1521     PTIMEZONE_ENTRY Entry;
1522     PTIMEZONE_ENTRY Current;
1523     ULONG DescriptionSize;
1524     ULONG StandardNameSize;
1525     ULONG DaylightNameSize;
1526 
1527     Entry = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(TIMEZONE_ENTRY));
1528     if (Entry == NULL)
1529     {
1530         return ERROR_NOT_ENOUGH_MEMORY;
1531     }
1532 
1533     DescriptionSize  = sizeof(Entry->Description);
1534     StandardNameSize = sizeof(Entry->StandardName);
1535     DaylightNameSize = sizeof(Entry->DaylightName);
1536 
1537     lError = QueryTimeZoneData(hZoneKey,
1538                                &Entry->Index,
1539                                &Entry->TimezoneInfo,
1540                                Entry->Description,
1541                                &DescriptionSize,
1542                                Entry->StandardName,
1543                                &StandardNameSize,
1544                                Entry->DaylightName,
1545                                &DaylightNameSize);
1546     if (lError != ERROR_SUCCESS)
1547     {
1548         HeapFree(GetProcessHeap(), 0, Entry);
1549         return lError;
1550     }
1551 
1552     if (SetupData->TimeZoneListHead == NULL &&
1553         SetupData->TimeZoneListTail == NULL)
1554     {
1555         Entry->Prev = NULL;
1556         Entry->Next = NULL;
1557         SetupData->TimeZoneListHead = Entry;
1558         SetupData->TimeZoneListTail = Entry;
1559     }
1560     else
1561     {
1562         Current = GetLargerTimeZoneEntry(SetupData, Entry->Index);
1563         if (Current != NULL)
1564         {
1565             if (Current == SetupData->TimeZoneListHead)
1566             {
1567                 /* Prepend to head */
1568                 Entry->Prev = NULL;
1569                 Entry->Next = SetupData->TimeZoneListHead;
1570                 SetupData->TimeZoneListHead->Prev = Entry;
1571                 SetupData->TimeZoneListHead = Entry;
1572             }
1573             else
1574             {
1575                 /* Insert before current */
1576                 Entry->Prev = Current->Prev;
1577                 Entry->Next = Current;
1578                 Current->Prev->Next = Entry;
1579                 Current->Prev = Entry;
1580             }
1581         }
1582         else
1583         {
1584             /* Append to tail */
1585             Entry->Prev = SetupData->TimeZoneListTail;
1586             Entry->Next = NULL;
1587             SetupData->TimeZoneListTail->Next = Entry;
1588             SetupData->TimeZoneListTail = Entry;
1589         }
1590     }
1591 
1592     return ERROR_SUCCESS;
1593 }
1594 
1595 static VOID
1596 CreateTimeZoneList(PSETUPDATA SetupData)
1597 {
1598     EnumerateTimeZoneList(RetrieveTimeZone, SetupData);
1599 }
1600 
1601 static VOID
1602 DestroyTimeZoneList(PSETUPDATA SetupData)
1603 {
1604     PTIMEZONE_ENTRY Entry;
1605 
1606     while (SetupData->TimeZoneListHead != NULL)
1607     {
1608         Entry = SetupData->TimeZoneListHead;
1609 
1610         SetupData->TimeZoneListHead = Entry->Next;
1611         if (SetupData->TimeZoneListHead != NULL)
1612         {
1613             SetupData->TimeZoneListHead->Prev = NULL;
1614         }
1615 
1616         HeapFree(GetProcessHeap(), 0, Entry);
1617     }
1618 
1619     SetupData->TimeZoneListTail = NULL;
1620 }
1621 
1622 
1623 static VOID
1624 ShowTimeZoneList(HWND hwnd, PSETUPDATA SetupData, DWORD dwEntryIndex)
1625 {
1626     PTIMEZONE_ENTRY Entry;
1627     DWORD dwIndex = 0;
1628     DWORD dwCount;
1629 
1630     GetTimeZoneListIndex(&dwEntryIndex);
1631 
1632     Entry = SetupData->TimeZoneListHead;
1633     while (Entry != NULL)
1634     {
1635         dwCount = SendMessage(hwnd,
1636                               CB_ADDSTRING,
1637                               0,
1638                               (LPARAM)Entry->Description);
1639 
1640         if (dwEntryIndex != 0 && dwEntryIndex == Entry->Index)
1641             dwIndex = dwCount;
1642 
1643         Entry = Entry->Next;
1644     }
1645 
1646     SendMessage(hwnd,
1647                 CB_SETCURSEL,
1648                 (WPARAM)dwIndex,
1649                 0);
1650 }
1651 
1652 
1653 static VOID
1654 SetLocalTimeZone(HWND hwnd, PSETUPDATA SetupData)
1655 {
1656     TIME_ZONE_INFORMATION TimeZoneInformation;
1657     PTIMEZONE_ENTRY Entry;
1658     DWORD dwIndex;
1659     DWORD i;
1660 
1661     dwIndex = SendMessage(hwnd,
1662                           CB_GETCURSEL,
1663                           0,
1664                           0);
1665 
1666     i = 0;
1667     Entry = SetupData->TimeZoneListHead;
1668     while (i < dwIndex)
1669     {
1670         if (Entry == NULL)
1671             return;
1672 
1673         i++;
1674         Entry = Entry->Next;
1675     }
1676 
1677     wcscpy(TimeZoneInformation.StandardName,
1678            Entry->StandardName);
1679     wcscpy(TimeZoneInformation.DaylightName,
1680            Entry->DaylightName);
1681 
1682     TimeZoneInformation.Bias = Entry->TimezoneInfo.Bias;
1683     TimeZoneInformation.StandardBias = Entry->TimezoneInfo.StandardBias;
1684     TimeZoneInformation.DaylightBias = Entry->TimezoneInfo.DaylightBias;
1685 
1686     memcpy(&TimeZoneInformation.StandardDate,
1687            &Entry->TimezoneInfo.StandardDate,
1688            sizeof(SYSTEMTIME));
1689     memcpy(&TimeZoneInformation.DaylightDate,
1690            &Entry->TimezoneInfo.DaylightDate,
1691            sizeof(SYSTEMTIME));
1692 
1693     /* Set time zone information */
1694     SetTimeZoneInformation(&TimeZoneInformation);
1695 }
1696 
1697 
1698 static BOOL
1699 GetLocalSystemTime(HWND hwnd, PSETUPDATA SetupData)
1700 {
1701     SYSTEMTIME Date;
1702     SYSTEMTIME Time;
1703 
1704     if (DateTime_GetSystemtime(GetDlgItem(hwnd, IDC_DATEPICKER), &Date) != GDT_VALID)
1705     {
1706         return FALSE;
1707     }
1708 
1709     if (DateTime_GetSystemtime(GetDlgItem(hwnd, IDC_TIMEPICKER), &Time) != GDT_VALID)
1710     {
1711         return FALSE;
1712     }
1713 
1714     SetupData->SystemTime.wYear = Date.wYear;
1715     SetupData->SystemTime.wMonth = Date.wMonth;
1716     SetupData->SystemTime.wDayOfWeek = Date.wDayOfWeek;
1717     SetupData->SystemTime.wDay = Date.wDay;
1718     SetupData->SystemTime.wHour = Time.wHour;
1719     SetupData->SystemTime.wMinute = Time.wMinute;
1720     SetupData->SystemTime.wSecond = Time.wSecond;
1721     SetupData->SystemTime.wMilliseconds = Time.wMilliseconds;
1722 
1723     return TRUE;
1724 }
1725 
1726 
1727 static BOOL
1728 SetSystemLocalTime(HWND hwnd, PSETUPDATA SetupData)
1729 {
1730     BOOL Ret = FALSE;
1731 
1732     /*
1733      * Call SetLocalTime twice to ensure correct results
1734      */
1735     Ret = SetLocalTime(&SetupData->SystemTime) &&
1736           SetLocalTime(&SetupData->SystemTime);
1737 
1738     return Ret;
1739 }
1740 
1741 
1742 static VOID
1743 UpdateLocalSystemTime(HWND hwnd, SYSTEMTIME LocalTime)
1744 {
1745     DateTime_SetSystemtime(GetDlgItem(hwnd, IDC_DATEPICKER), GDT_VALID, &LocalTime);
1746     DateTime_SetSystemtime(GetDlgItem(hwnd, IDC_TIMEPICKER), GDT_VALID, &LocalTime);
1747 }
1748 
1749 
1750 static BOOL
1751 WriteDateTimeSettings(HWND hwndDlg, PSETUPDATA SetupData)
1752 {
1753     WCHAR Title[64];
1754     WCHAR ErrorLocalTime[256];
1755 
1756     GetLocalSystemTime(hwndDlg, SetupData);
1757     SetLocalTimeZone(GetDlgItem(hwndDlg, IDC_TIMEZONELIST),
1758                      SetupData);
1759 
1760     SetAutoDaylight(SendDlgItemMessage(hwndDlg, IDC_AUTODAYLIGHT,
1761                                        BM_GETCHECK, 0, 0) != BST_UNCHECKED);
1762     if (!SetSystemLocalTime(hwndDlg, SetupData))
1763     {
1764         if (0 == LoadStringW(hDllInstance, IDS_REACTOS_SETUP, Title, ARRAYSIZE(Title)))
1765         {
1766             wcscpy(Title, L"ReactOS Setup");
1767         }
1768         if (0 == LoadStringW(hDllInstance, IDS_WZD_LOCALTIME, ErrorLocalTime,
1769                              ARRAYSIZE(ErrorLocalTime)))
1770         {
1771             wcscpy(ErrorLocalTime, L"Setup was unable to set the local time.");
1772         }
1773         MessageBoxW(hwndDlg, ErrorLocalTime, Title, MB_ICONWARNING | MB_OK);
1774         return FALSE;
1775     }
1776 
1777     return TRUE;
1778 }
1779 
1780 
1781 static INT_PTR CALLBACK
1782 DateTimePageDlgProc(HWND hwndDlg,
1783                     UINT uMsg,
1784                     WPARAM wParam,
1785                     LPARAM lParam)
1786 {
1787     PSETUPDATA SetupData;
1788 
1789     /* Retrieve pointer to the global setup data */
1790     SetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
1791 
1792     switch (uMsg)
1793     {
1794         case WM_INITDIALOG:
1795         {
1796             /* Save pointer to the global setup data */
1797             SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
1798             SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (DWORD_PTR)SetupData);
1799 
1800             CreateTimeZoneList(SetupData);
1801 
1802             if (SetupData->UnattendSetup)
1803             {
1804                 ShowTimeZoneList(GetDlgItem(hwndDlg, IDC_TIMEZONELIST),
1805                                  SetupData, SetupData->TimeZoneIndex);
1806 
1807                 if (!SetupData->DisableAutoDaylightTimeSet)
1808                 {
1809                     SendDlgItemMessage(hwndDlg, IDC_AUTODAYLIGHT, BM_SETCHECK, (WPARAM)BST_CHECKED, 0);
1810                 }
1811             }
1812             else
1813             {
1814                 ShowTimeZoneList(GetDlgItem(hwndDlg, IDC_TIMEZONELIST),
1815                                  SetupData, -1);
1816 
1817                 SendDlgItemMessage(hwndDlg, IDC_AUTODAYLIGHT, BM_SETCHECK, (WPARAM)BST_CHECKED, 0);
1818             }
1819             break;
1820         }
1821 
1822         case WM_TIMER:
1823         {
1824             SYSTEMTIME LocalTime;
1825 
1826             GetLocalTime(&LocalTime);
1827             UpdateLocalSystemTime(hwndDlg, LocalTime);
1828 
1829             // Reset timeout.
1830             SetTimer(hwndDlg, 1, 1000 - LocalTime.wMilliseconds, NULL);
1831             break;
1832         }
1833 
1834         case WM_NOTIFY:
1835             switch (((LPNMHDR)lParam)->code)
1836             {
1837                 case PSN_SETACTIVE:
1838                 {
1839                     SYSTEMTIME LocalTime;
1840 
1841                     GetLocalTime(&LocalTime);
1842                     UpdateLocalSystemTime(hwndDlg, LocalTime);
1843 
1844                     /* Enable the Back and Next buttons */
1845                     PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
1846 
1847                     if (SetupData->UnattendSetup && WriteDateTimeSettings(hwndDlg, SetupData))
1848                     {
1849                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, SetupData->uFirstNetworkWizardPage);
1850                         return TRUE;
1851                     }
1852 
1853                     SetTimer(hwndDlg, 1, 1000 - LocalTime.wMilliseconds, NULL);
1854                     break;
1855                 }
1856 
1857                 case PSN_KILLACTIVE:
1858                 case DTN_DATETIMECHANGE:
1859                     // NB: Not re-set until changing page (PSN_SETACTIVE).
1860                     KillTimer(hwndDlg, 1);
1861                     break;
1862 
1863                 case PSN_WIZNEXT:
1864                     WriteDateTimeSettings(hwndDlg, SetupData);
1865                     break;
1866 
1867                 case PSN_WIZBACK:
1868                     SetupData->UnattendSetup = FALSE;
1869                     break;
1870 
1871                 default:
1872                     break;
1873             }
1874             break;
1875 
1876         case WM_DESTROY:
1877             DestroyTimeZoneList(SetupData);
1878             break;
1879 
1880         default:
1881             break;
1882     }
1883 
1884     return FALSE;
1885 }
1886 
1887 static struct ThemeInfo
1888 {
1889     LPCWSTR PreviewBitmap;
1890     UINT DisplayName;
1891     LPCWSTR ThemeFile;
1892 
1893 } Themes[] = {
1894     { MAKEINTRESOURCE(IDB_CLASSIC), IDS_CLASSIC, NULL },
1895     { MAKEINTRESOURCE(IDB_LAUTUS), IDS_LAUTUS, L"themes\\lautus\\lautus.msstyles" },
1896     { MAKEINTRESOURCE(IDB_LUNAR), IDS_LUNAR, L"themes\\lunar\\lunar.msstyles" },
1897     { MAKEINTRESOURCE(IDB_MIZU), IDS_MIZU, L"themes\\mizu\\mizu.msstyles"},
1898 };
1899 
1900 static INT_PTR CALLBACK
1901 ThemePageDlgProc(HWND hwndDlg,
1902                     UINT uMsg,
1903                     WPARAM wParam,
1904                     LPARAM lParam)
1905 {
1906     PSETUPDATA SetupData;
1907     LPNMLISTVIEW pnmv;
1908 
1909     /* Retrieve pointer to the global setup data */
1910     SetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
1911 
1912     switch (uMsg)
1913     {
1914         case WM_INITDIALOG:
1915         {
1916             HWND hListView;
1917             HIMAGELIST himl;
1918             DWORD n;
1919             LVITEM lvi = {0};
1920 
1921             /* Save pointer to the global setup data */
1922             SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
1923             SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (DWORD_PTR)SetupData);
1924 
1925             hListView = GetDlgItem(hwndDlg, IDC_THEMEPICKER);
1926 
1927             /* Common */
1928             himl = ImageList_Create(180, 163, ILC_COLOR32 | ILC_MASK, ARRAYSIZE(Themes), 1);
1929             lvi.mask = LVIF_TEXT | LVIF_IMAGE |LVIF_STATE;
1930 
1931             for (n = 0; n < ARRAYSIZE(Themes); ++n)
1932             {
1933                 WCHAR DisplayName[100] = {0};
1934                 /* Load the bitmap */
1935                 HANDLE image = LoadImageW(hDllInstance, Themes[n].PreviewBitmap, IMAGE_BITMAP, 0, 0, LR_CREATEDIBSECTION);
1936                 ImageList_AddMasked(himl, image, RGB(255,0,255));
1937 
1938                 /* Load the string */
1939                 LoadStringW(hDllInstance, Themes[n].DisplayName, DisplayName, ARRAYSIZE(DisplayName));
1940                 DisplayName[ARRAYSIZE(DisplayName)-1] = UNICODE_NULL;
1941 
1942                 /* Add the listview item */
1943                 lvi.iItem  = n;
1944                 lvi.iImage = n;
1945                 lvi.pszText = DisplayName;
1946                 ListView_InsertItem(hListView, &lvi);
1947             }
1948 
1949             /* Register the imagelist */
1950             ListView_SetImageList(hListView, himl, LVSIL_NORMAL);
1951             /* Transparant background */
1952             ListView_SetBkColor(hListView, CLR_NONE);
1953             ListView_SetTextBkColor(hListView, CLR_NONE);
1954             /* Reduce the size between the items */
1955             ListView_SetIconSpacing(hListView, 190, 173);
1956             break;
1957         }
1958         case WM_NOTIFY:
1959             switch (((LPNMHDR)lParam)->code)
1960             {
1961                 //case LVN_ITEMCHANGING:
1962                 case LVN_ITEMCHANGED:
1963                     pnmv = (LPNMLISTVIEW)lParam;
1964                     if ((pnmv->uChanged & LVIF_STATE) && (pnmv->uNewState & LVIS_SELECTED))
1965                     {
1966                         int iTheme = pnmv->iItem;
1967                         DPRINT1("Selected theme: %u\n", Themes[iTheme].DisplayName);
1968 
1969                         if (Themes[iTheme].ThemeFile)
1970                         {
1971                             WCHAR wszParams[1024];
1972                             WCHAR wszTheme[MAX_PATH];
1973                             WCHAR* format = L"desk.cpl desk,@Appearance /Action:ActivateMSTheme /file:\"%s\"";
1974 
1975                             SHGetFolderPathAndSubDirW(0, CSIDL_RESOURCES, NULL, SHGFP_TYPE_DEFAULT, Themes[iTheme].ThemeFile, wszTheme);
1976                             swprintf(wszParams, format, wszTheme);
1977                             RunControlPanelApplet(hwndDlg, wszParams);
1978                         }
1979                         else
1980                         {
1981                             RunControlPanelApplet(hwndDlg, L"desk.cpl desk,@Appearance /Action:ActivateMSTheme");
1982                         }
1983                     }
1984                     break;
1985                 case PSN_SETACTIVE:
1986                     /* Enable the Back and Next buttons */
1987                     PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
1988                     if (SetupData->UnattendSetup)
1989                     {
1990                         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, SetupData->uFirstNetworkWizardPage);
1991                         return TRUE;
1992                     }
1993                     break;
1994 
1995                 case PSN_WIZNEXT:
1996                     break;
1997 
1998                 case PSN_WIZBACK:
1999                     SetupData->UnattendSetup = FALSE;
2000                     break;
2001 
2002                 default:
2003                     break;
2004             }
2005             break;
2006 
2007         default:
2008             break;
2009     }
2010 
2011     return FALSE;
2012 }
2013 
2014 static UINT CALLBACK
2015 RegistrationNotificationProc(PVOID Context,
2016                              UINT Notification,
2017                              UINT_PTR Param1,
2018                              UINT_PTR Param2)
2019 {
2020     PREGISTRATIONDATA RegistrationData;
2021     REGISTRATIONNOTIFY RegistrationNotify;
2022     PSP_REGISTER_CONTROL_STATUSW StatusInfo;
2023     UINT MessageID;
2024 
2025     RegistrationData = (PREGISTRATIONDATA)Context;
2026 
2027     if (Notification == SPFILENOTIFY_STARTREGISTRATION ||
2028         Notification == SPFILENOTIFY_ENDREGISTRATION)
2029     {
2030         StatusInfo = (PSP_REGISTER_CONTROL_STATUSW) Param1;
2031         RegistrationNotify.CurrentItem = wcsrchr(StatusInfo->FileName, L'\\');
2032         if (RegistrationNotify.CurrentItem == NULL)
2033         {
2034             RegistrationNotify.CurrentItem = StatusInfo->FileName;
2035         }
2036         else
2037         {
2038             RegistrationNotify.CurrentItem++;
2039         }
2040 
2041         if (Notification == SPFILENOTIFY_STARTREGISTRATION)
2042         {
2043             DPRINT("Received SPFILENOTIFY_STARTREGISTRATION notification for %S\n",
2044                    StatusInfo->FileName);
2045             RegistrationNotify.ErrorMessage = NULL;
2046             RegistrationNotify.Progress = RegistrationData->Registered;
2047             SendMessage(RegistrationData->hwndDlg, PM_STEP_START, 0, (LPARAM)&RegistrationNotify);
2048         }
2049         else
2050         {
2051             DPRINT("Received SPFILENOTIFY_ENDREGISTRATION notification for %S\n",
2052                    StatusInfo->FileName);
2053             DPRINT("Win32Error %u FailureCode %u\n", StatusInfo->Win32Error,
2054                    StatusInfo->FailureCode);
2055             if (StatusInfo->FailureCode != SPREG_SUCCESS)
2056             {
2057                 switch (StatusInfo->FailureCode)
2058                 {
2059                     case SPREG_LOADLIBRARY:
2060                         MessageID = IDS_LOADLIBRARY_FAILED;
2061                         break;
2062                     case SPREG_GETPROCADDR:
2063                         MessageID = IDS_GETPROCADDR_FAILED;
2064                         break;
2065                     case SPREG_REGSVR:
2066                         MessageID = IDS_REGSVR_FAILED;
2067                         break;
2068                     case SPREG_DLLINSTALL:
2069                         MessageID = IDS_DLLINSTALL_FAILED;
2070                         break;
2071                     case SPREG_TIMEOUT:
2072                         MessageID = IDS_TIMEOUT;
2073                         break;
2074                     default:
2075                         MessageID = IDS_REASON_UNKNOWN;
2076                         break;
2077                 }
2078 
2079                 RegistrationNotify.MessageID = MessageID;
2080                 RegistrationNotify.LastError = StatusInfo->Win32Error;
2081             }
2082             else
2083             {
2084                 RegistrationNotify.MessageID = 0;
2085                 RegistrationNotify.LastError = ERROR_SUCCESS;
2086             }
2087 
2088             if (RegistrationData->Registered < RegistrationData->DllCount)
2089             {
2090                 RegistrationData->Registered++;
2091             }
2092 
2093             RegistrationNotify.Progress = RegistrationData->Registered;
2094             SendMessage(RegistrationData->hwndDlg, PM_STEP_END, 0, (LPARAM)&RegistrationNotify);
2095         }
2096 
2097         return FILEOP_DOIT;
2098     }
2099     else
2100     {
2101         DPRINT1("Received unexpected notification %u\n", Notification);
2102         return SetupDefaultQueueCallback(RegistrationData->DefaultContext,
2103                                          Notification, Param1, Param2);
2104     }
2105 }
2106 
2107 
2108 static
2109 DWORD
2110 RegisterDlls(
2111     PITEMSDATA pItemsData)
2112 {
2113     REGISTRATIONDATA RegistrationData;
2114     WCHAR SectionName[512];
2115     INFCONTEXT Context;
2116     LONG DllCount = 0;
2117     DWORD LastError = NO_ERROR;
2118 
2119     ZeroMemory(&RegistrationData, sizeof(REGISTRATIONDATA));
2120     RegistrationData.hwndDlg = pItemsData->hwndDlg;
2121     RegistrationData.Registered = 0;
2122 
2123     if (!SetupFindFirstLineW(hSysSetupInf, L"RegistrationPhase2",
2124                              L"RegisterDlls", &Context))
2125     {
2126         DPRINT1("No RegistrationPhase2 section found\n");
2127         return FALSE;
2128     }
2129 
2130     if (!SetupGetStringFieldW(&Context, 1, SectionName,
2131                               ARRAYSIZE(SectionName),
2132                               NULL))
2133     {
2134         DPRINT1("Unable to retrieve section name\n");
2135         return FALSE;
2136     }
2137 
2138     DllCount = SetupGetLineCountW(hSysSetupInf, SectionName);
2139     DPRINT1("SectionName %S DllCount %ld\n", SectionName, DllCount);
2140     if (DllCount < 0)
2141     {
2142         SetLastError(STATUS_NOT_FOUND);
2143         return FALSE;
2144     }
2145 
2146     RegistrationData.DllCount = (ULONG)DllCount;
2147     RegistrationData.DefaultContext = SetupInitDefaultQueueCallback(RegistrationData.hwndDlg);
2148 
2149     SendMessage(pItemsData->hwndDlg, PM_ITEM_START, 0, (LPARAM)RegistrationData.DllCount);
2150 
2151     _SEH2_TRY
2152     {
2153         if (!SetupInstallFromInfSectionW(GetParent(RegistrationData.hwndDlg),
2154                                          hSysSetupInf,
2155                                          L"RegistrationPhase2",
2156                                          SPINST_REGISTRY | SPINST_REGISTERCALLBACKAWARE | SPINST_REGSVR,
2157                                          0,
2158                                          NULL,
2159                                          0,
2160                                          RegistrationNotificationProc,
2161                                          &RegistrationData,
2162                                          NULL,
2163                                          NULL))
2164         {
2165             LastError = GetLastError();
2166         }
2167     }
2168     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
2169     {
2170         DPRINT("Catching exception\n");
2171         LastError = RtlNtStatusToDosError(_SEH2_GetExceptionCode());
2172     }
2173     _SEH2_END;
2174 
2175     SetupTermDefaultQueueCallback(RegistrationData.DefaultContext);
2176 
2177     SendMessage(pItemsData->hwndDlg, PM_ITEM_END, 0, LastError);
2178 
2179     return 0;
2180 }
2181 
2182 
2183 static
2184 DWORD
2185 CALLBACK
2186 ItemCompletionThread(
2187     LPVOID Parameter)
2188 {
2189     PITEMSDATA pItemsData;
2190     HWND hwndDlg;
2191 
2192     pItemsData = (PITEMSDATA)Parameter;
2193     hwndDlg = pItemsData->hwndDlg;
2194 
2195     RegisterDlls(pItemsData);
2196 
2197     RegisterTypeLibraries(hSysSetupInf, L"TypeLibraries");
2198 
2199     /* FIXME: Add completion steps here! */
2200 
2201     // FIXME: Move this call to a separate cleanup page!
2202     RtlCreateBootStatusDataFile();
2203 
2204     /* Free the items data */
2205     HeapFree(GetProcessHeap(), 0, pItemsData);
2206 
2207     /* Tell the wizard page that we are done */
2208     PostMessage(hwndDlg, PM_ITEMS_DONE, 0, 0);
2209 
2210     return 0;
2211 }
2212 
2213 
2214 static
2215 BOOL
2216 RunItemCompletionThread(
2217     _In_ HWND hwndDlg)
2218 {
2219     HANDLE hCompletionThread;
2220     PITEMSDATA pItemsData;
2221 
2222     pItemsData = HeapAlloc(GetProcessHeap(), 0, sizeof(ITEMSDATA));
2223     if (pItemsData == NULL)
2224         return FALSE;
2225 
2226     pItemsData->hwndDlg = hwndDlg;
2227 
2228     hCompletionThread = CreateThread(NULL,
2229                                      0,
2230                                      ItemCompletionThread,
2231                                      pItemsData,
2232                                      0,
2233                                      NULL);
2234     if (hCompletionThread == NULL)
2235     {
2236         HeapFree(GetProcessHeap(), 0, pItemsData);
2237     }
2238     else
2239     {
2240         CloseHandle(hCompletionThread);
2241         return TRUE;
2242     }
2243 
2244     return FALSE;
2245 }
2246 
2247 static
2248 VOID
2249 ShowItemError(
2250     HWND hwndDlg,
2251     DWORD LastError)
2252 {
2253     LPWSTR ErrorMessage = NULL;
2254     WCHAR UnknownError[84];
2255     WCHAR Title[64];
2256 
2257     if (FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
2258                        NULL, LastError, 0, ErrorMessage, 0, NULL) == 0)
2259     {
2260         if (LoadStringW(hDllInstance, IDS_UNKNOWN_ERROR,
2261                         UnknownError,
2262                         ARRAYSIZE(UnknownError) - 20) == 0)
2263         {
2264             wcscpy(UnknownError, L"Unknown error");
2265         }
2266         wcscat(UnknownError, L" ");
2267         _ultow(LastError, UnknownError + wcslen(UnknownError), 10);
2268         ErrorMessage = UnknownError;
2269     }
2270 
2271     if (ErrorMessage != NULL)
2272     {
2273         if (LoadStringW(hDllInstance, IDS_REACTOS_SETUP,
2274                         Title, ARRAYSIZE(Title)) == 0)
2275         {
2276             wcscpy(Title, L"ReactOS Setup");
2277         }
2278 
2279         MessageBoxW(hwndDlg, ErrorMessage, Title, MB_ICONERROR | MB_OK);
2280     }
2281 
2282     if (ErrorMessage != NULL &&
2283         ErrorMessage != UnknownError)
2284     {
2285         LocalFree(ErrorMessage);
2286     }
2287 }
2288 
2289 
2290 static
2291 VOID
2292 ShowStepError(
2293     HWND hwndDlg,
2294     PREGISTRATIONNOTIFY RegistrationNotify)
2295 {
2296     WCHAR ErrorMessage[128];
2297     WCHAR Title[64];
2298 
2299     if (LoadStringW(hDllInstance, RegistrationNotify->MessageID,
2300                     ErrorMessage,
2301                     ARRAYSIZE(ErrorMessage)) == 0)
2302     {
2303         ErrorMessage[0] = L'\0';
2304     }
2305 
2306     if (RegistrationNotify->MessageID != IDS_TIMEOUT)
2307     {
2308         FormatMessageW(FORMAT_MESSAGE_FROM_SYSTEM, NULL,
2309                        RegistrationNotify->LastError, 0,
2310                        ErrorMessage + wcslen(ErrorMessage),
2311                        ARRAYSIZE(ErrorMessage) - wcslen(ErrorMessage),
2312                        NULL);
2313     }
2314 
2315     if (ErrorMessage[0] != L'\0')
2316     {
2317         if (LoadStringW(hDllInstance, IDS_REACTOS_SETUP,
2318                         Title, ARRAYSIZE(Title)) == 0)
2319         {
2320             wcscpy(Title, L"ReactOS Setup");
2321         }
2322 
2323         MessageBoxW(hwndDlg, ErrorMessage,
2324                     Title, MB_ICONERROR | MB_OK);
2325     }
2326 }
2327 
2328 
2329 static INT_PTR CALLBACK
2330 ProcessPageDlgProc(HWND hwndDlg,
2331                    UINT uMsg,
2332                    WPARAM wParam,
2333                    LPARAM lParam)
2334 {
2335     PSETUPDATA SetupData;
2336     PREGISTRATIONNOTIFY RegistrationNotify;
2337 
2338     /* Retrieve pointer to the global setup data */
2339     SetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
2340 
2341     switch (uMsg)
2342     {
2343         case WM_INITDIALOG:
2344             /* Save pointer to the global setup data */
2345             SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
2346             SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (DWORD_PTR)SetupData);
2347             ShowWindow(GetDlgItem(hwndDlg, IDC_TASKTEXT2), SW_HIDE);
2348             ShowWindow(GetDlgItem(hwndDlg, IDC_TASKTEXT3), SW_HIDE);
2349             ShowWindow(GetDlgItem(hwndDlg, IDC_TASKTEXT4), SW_HIDE);
2350             break;
2351 
2352         case WM_NOTIFY:
2353             switch (((LPNMHDR)lParam)->code)
2354             {
2355                 case PSN_SETACTIVE:
2356                     /* Disable the Back and Next buttons */
2357                     PropSheet_SetWizButtons(GetParent(hwndDlg), 0);
2358                     RunItemCompletionThread(hwndDlg);
2359                     break;
2360 
2361                 case PSN_WIZNEXT:
2362                     break;
2363 
2364                 case PSN_WIZBACK:
2365                     SetupData->UnattendSetup = FALSE;
2366                     break;
2367 
2368                 default:
2369                     break;
2370             }
2371             break;
2372 
2373         case PM_ITEM_START:
2374             DPRINT("PM_ITEM_START %lu\n", (ULONG)lParam);
2375             SendDlgItemMessage(hwndDlg, IDC_PROCESSPROGRESS, PBM_SETRANGE, 0, MAKELPARAM(0, (ULONG)lParam));
2376             SendDlgItemMessage(hwndDlg, IDC_PROCESSPROGRESS, PBM_SETPOS, 0, 0);
2377             SendDlgItemMessage(hwndDlg, IDC_TASKTEXT1 + wParam, WM_SETFONT, (WPARAM)SetupData->hBoldFont, (LPARAM)TRUE);
2378             break;
2379 
2380         case PM_ITEM_END:
2381             DPRINT("PM_ITEM_END\n");
2382             if (lParam == ERROR_SUCCESS)
2383             {
2384             }
2385             else
2386             {
2387                 ShowItemError(hwndDlg, (DWORD)lParam);
2388             }
2389             break;
2390 
2391         case PM_STEP_START:
2392             DPRINT("PM_STEP_START\n");
2393             RegistrationNotify = (PREGISTRATIONNOTIFY)lParam;
2394             SendDlgItemMessage(hwndDlg, IDC_ITEM, WM_SETTEXT, 0,
2395                                (LPARAM)((RegistrationNotify->CurrentItem != NULL)? RegistrationNotify->CurrentItem : L""));
2396             break;
2397 
2398         case PM_STEP_END:
2399             DPRINT("PM_STEP_END\n");
2400             RegistrationNotify = (PREGISTRATIONNOTIFY)lParam;
2401             SendDlgItemMessage(hwndDlg, IDC_PROCESSPROGRESS, PBM_SETPOS, RegistrationNotify->Progress, 0);
2402             if (RegistrationNotify->LastError != ERROR_SUCCESS)
2403             {
2404                 ShowStepError(hwndDlg, RegistrationNotify);
2405             }
2406             break;
2407 
2408         case PM_ITEMS_DONE:
2409             DPRINT("PM_ITEMS_DONE\n");
2410             /* Enable the Back and Next buttons */
2411             PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_NEXT);
2412             PropSheet_PressButton(GetParent(hwndDlg), PSBTN_NEXT);
2413             break;
2414 
2415         default:
2416             break;
2417     }
2418 
2419     return FALSE;
2420 }
2421 
2422 
2423 static VOID
2424 SetInstallationCompleted(VOID)
2425 {
2426     HKEY hKey = 0;
2427     DWORD InProgress = 0;
2428     DWORD InstallDate;
2429 
2430     if (RegOpenKeyExW( HKEY_LOCAL_MACHINE,
2431                        L"SYSTEM\\Setup",
2432                        0,
2433                        KEY_WRITE,
2434                        &hKey ) == ERROR_SUCCESS)
2435     {
2436         RegSetValueExW( hKey, L"SystemSetupInProgress", 0, REG_DWORD, (LPBYTE)&InProgress, sizeof(InProgress) );
2437         RegCloseKey( hKey );
2438     }
2439 
2440     if (RegOpenKeyExW( HKEY_LOCAL_MACHINE,
2441                        L"Software\\Microsoft\\Windows NT\\CurrentVersion",
2442                        0,
2443                        KEY_WRITE,
2444                        &hKey ) == ERROR_SUCCESS)
2445     {
2446         InstallDate = (DWORD)time(NULL);
2447         RegSetValueExW( hKey, L"InstallDate", 0, REG_DWORD, (LPBYTE)&InstallDate, sizeof(InstallDate) );
2448         RegCloseKey( hKey );
2449     }
2450 }
2451 
2452 static INT_PTR CALLBACK
2453 FinishDlgProc(HWND hwndDlg,
2454               UINT uMsg,
2455               WPARAM wParam,
2456               LPARAM lParam)
2457 {
2458 
2459     switch (uMsg)
2460     {
2461         case WM_INITDIALOG:
2462         {
2463             /* Get pointer to the global setup data */
2464             PSETUPDATA SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
2465 
2466             if (!SetupData->UnattendSetup || !SetupData->DisableGeckoInst)
2467             {
2468                 /* Run the Wine Gecko prompt */
2469                 Control_RunDLLW(hwndDlg, 0, L"appwiz.cpl install_gecko", SW_SHOW);
2470             }
2471 
2472             /* Set title font */
2473             SendDlgItemMessage(hwndDlg,
2474                                IDC_FINISHTITLE,
2475                                WM_SETFONT,
2476                                (WPARAM)SetupData->hTitleFont,
2477                                (LPARAM)TRUE);
2478             if (SetupData->UnattendSetup)
2479             {
2480                 KillTimer(hwndDlg, 1);
2481                 SetInstallationCompleted();
2482                 PostQuitMessage(0);
2483             }
2484         }
2485         break;
2486 
2487         case WM_DESTROY:
2488         {
2489             SetInstallationCompleted();
2490             PostQuitMessage(0);
2491             return TRUE;
2492         }
2493 
2494         case WM_TIMER:
2495         {
2496             INT Position;
2497             HWND hWndProgress;
2498 
2499             hWndProgress = GetDlgItem(hwndDlg, IDC_RESTART_PROGRESS);
2500             Position = SendMessage(hWndProgress, PBM_GETPOS, 0, 0);
2501             if (Position == 300)
2502             {
2503                 KillTimer(hwndDlg, 1);
2504                 PropSheet_PressButton(GetParent(hwndDlg), PSBTN_FINISH);
2505             }
2506             else
2507             {
2508                 SendMessage(hWndProgress, PBM_SETPOS, Position + 1, 0);
2509             }
2510         }
2511         return TRUE;
2512 
2513         case WM_NOTIFY:
2514         {
2515             LPNMHDR lpnm = (LPNMHDR)lParam;
2516 
2517             switch (lpnm->code)
2518             {
2519                 case PSN_SETACTIVE:
2520                     /* Enable the correct buttons on for the active page */
2521                     PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_FINISH);
2522 
2523                     SendDlgItemMessage(hwndDlg, IDC_RESTART_PROGRESS, PBM_SETRANGE, 0,
2524                                        MAKELPARAM(0, 300));
2525                     SendDlgItemMessage(hwndDlg, IDC_RESTART_PROGRESS, PBM_SETPOS, 0, 0);
2526                     SetTimer(hwndDlg, 1, 50, NULL);
2527                     break;
2528 
2529                 case PSN_WIZFINISH:
2530                     DestroyWindow(GetParent(hwndDlg));
2531                     break;
2532 
2533                 default:
2534                     break;
2535             }
2536         }
2537         break;
2538 
2539         default:
2540             break;
2541     }
2542 
2543     return FALSE;
2544 }
2545 
2546 
2547 /*
2548  * GetInstallSourceWin32 retrieves the path to the ReactOS installation medium
2549  * in Win32 format, for later use by syssetup and storage in the registry.
2550  */
2551 static BOOL
2552 GetInstallSourceWin32(
2553     OUT PWSTR pwszPath,
2554     IN DWORD cchPathMax,
2555     IN PCWSTR pwszNTPath)
2556 {
2557     WCHAR wszDrives[512];
2558     WCHAR wszNTPath[512]; // MAX_PATH ?
2559     DWORD cchDrives;
2560     PWCHAR pwszDrive;
2561 
2562     *pwszPath = UNICODE_NULL;
2563 
2564     cchDrives = GetLogicalDriveStringsW(_countof(wszDrives) - 1, wszDrives);
2565     if (cchDrives == 0 || cchDrives >= _countof(wszDrives))
2566     {
2567         /* Buffer too small or failure */
2568         LogItem(NULL, L"GetLogicalDriveStringsW failed");
2569         return FALSE;
2570     }
2571 
2572     for (pwszDrive = wszDrives; *pwszDrive; pwszDrive += wcslen(pwszDrive) + 1)
2573     {
2574         WCHAR wszBuf[MAX_PATH];
2575 
2576         /* Retrieve the NT path corresponding to the current Win32 DOS path */
2577         pwszDrive[2] = UNICODE_NULL; // Temporarily remove the backslash
2578         QueryDosDeviceW(pwszDrive, wszNTPath, _countof(wszNTPath));
2579         pwszDrive[2] = L'\\';        // Restore the backslash
2580 
2581         wcscat(wszNTPath, L"\\");    // Concat a backslash
2582 
2583         /* Logging */
2584         wsprintf(wszBuf, L"Testing '%s' --> '%s' %s a CD",
2585                  pwszDrive, wszNTPath,
2586                  (GetDriveTypeW(pwszDrive) == DRIVE_CDROM) ? L"is" : L"is not");
2587         LogItem(NULL, wszBuf);
2588 
2589         /* Check whether the NT path corresponds to the NT installation source path */
2590         if (!_wcsicmp(wszNTPath, pwszNTPath))
2591         {
2592             /* Found it! */
2593             wcscpy(pwszPath, pwszDrive); // cchPathMax
2594 
2595             /* Logging */
2596             wsprintf(wszBuf, L"GetInstallSourceWin32: %s", pwszPath);
2597             LogItem(NULL, wszBuf);
2598             wcscat(wszBuf, L"\n");
2599             OutputDebugStringW(wszBuf);
2600 
2601             return TRUE;
2602         }
2603     }
2604 
2605     return FALSE;
2606 }
2607 
2608 VOID
2609 ProcessUnattendSection(
2610     IN OUT PSETUPDATA pSetupData)
2611 {
2612     INFCONTEXT InfContext;
2613     WCHAR szName[256];
2614     WCHAR szValue[MAX_PATH];
2615     DWORD LineLength;
2616     HKEY hKey;
2617 
2618     if (!SetupFindFirstLineW(pSetupData->hSetupInf,
2619                              L"Unattend",
2620                              L"UnattendSetupEnabled",
2621                              &InfContext))
2622     {
2623         DPRINT1("Error: Cannot find UnattendSetupEnabled Key! %d\n", GetLastError());
2624         return;
2625     }
2626 
2627     if (!SetupGetStringFieldW(&InfContext,
2628                               1,
2629                               szValue,
2630                               ARRAYSIZE(szValue),
2631                               &LineLength))
2632     {
2633         DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
2634         return;
2635     }
2636 
2637     if (_wcsicmp(szValue, L"yes") != 0)
2638     {
2639         DPRINT("Unattend setup was disabled by UnattendSetupEnabled key.\n");
2640         return;
2641     }
2642 
2643     pSetupData->UnattendSetup = TRUE;
2644 
2645     if (!SetupFindFirstLineW(pSetupData->hSetupInf,
2646                              L"Unattend",
2647                              NULL,
2648                              &InfContext))
2649     {
2650         DPRINT1("Error: SetupFindFirstLine failed %d\n", GetLastError());
2651         return;
2652     }
2653 
2654     do
2655     {
2656         if (!SetupGetStringFieldW(&InfContext,
2657                                   0,
2658                                   szName,
2659                                   ARRAYSIZE(szName),
2660                                   &LineLength))
2661         {
2662             DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
2663             return;
2664         }
2665 
2666         if (!SetupGetStringFieldW(&InfContext,
2667                                   1,
2668                                   szValue,
2669                                   ARRAYSIZE(szValue),
2670                                   &LineLength))
2671         {
2672             DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
2673             return;
2674         }
2675         DPRINT1("Name %S Value %S\n", szName, szValue);
2676         if (!_wcsicmp(szName, L"FullName"))
2677         {
2678             if (ARRAYSIZE(pSetupData->OwnerName) > LineLength)
2679             {
2680                 wcscpy(pSetupData->OwnerName, szValue);
2681             }
2682         }
2683         else if (!_wcsicmp(szName, L"OrgName"))
2684         {
2685             if (ARRAYSIZE(pSetupData->OwnerOrganization) > LineLength)
2686             {
2687                 wcscpy(pSetupData->OwnerOrganization, szValue);
2688             }
2689         }
2690         else if (!_wcsicmp(szName, L"ComputerName"))
2691         {
2692             if (ARRAYSIZE(pSetupData->ComputerName) > LineLength)
2693             {
2694                 wcscpy(pSetupData->ComputerName, szValue);
2695             }
2696         }
2697         else if (!_wcsicmp(szName, L"AdminPassword"))
2698         {
2699             if (ARRAYSIZE(pSetupData->AdminPassword) > LineLength)
2700             {
2701                 wcscpy(pSetupData->AdminPassword, szValue);
2702             }
2703         }
2704         else if (!_wcsicmp(szName, L"TimeZoneIndex"))
2705         {
2706             pSetupData->TimeZoneIndex = _wtoi(szValue);
2707         }
2708         else if (!_wcsicmp(szName, L"DisableAutoDaylightTimeSet"))
2709         {
2710             pSetupData->DisableAutoDaylightTimeSet = _wtoi(szValue);
2711         }
2712         else if (!_wcsicmp(szName, L"DisableGeckoInst"))
2713         {
2714             if (!_wcsicmp(szValue, L"yes"))
2715                 pSetupData->DisableGeckoInst = TRUE;
2716             else
2717                 pSetupData->DisableGeckoInst = FALSE;
2718         }
2719         else if (!_wcsicmp(szName, L"ProductOption"))
2720         {
2721             pSetupData->ProductOption = (PRODUCT_OPTION)_wtoi(szValue);
2722         }
2723     } while (SetupFindNextLine(&InfContext, &InfContext));
2724 
2725     if (SetupFindFirstLineW(pSetupData->hSetupInf,
2726                             L"Display",
2727                             NULL,
2728                             &InfContext))
2729     {
2730         DEVMODEW dm = { { 0 } };
2731         dm.dmSize = sizeof(dm);
2732         if (EnumDisplaySettingsW(NULL, ENUM_CURRENT_SETTINGS, &dm))
2733         {
2734             do
2735             {
2736                 int iValue;
2737                 if (!SetupGetStringFieldW(&InfContext,
2738                                           0,
2739                                           szName,
2740                                           ARRAYSIZE(szName),
2741                                           &LineLength))
2742                 {
2743                     DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
2744                     return;
2745                 }
2746 
2747                 if (!SetupGetStringFieldW(&InfContext,
2748                                           1,
2749                                           szValue,
2750                                           ARRAYSIZE(szValue),
2751                                           &LineLength))
2752                 {
2753                     DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
2754                     return;
2755                 }
2756                 iValue = _wtoi(szValue);
2757                 DPRINT1("Name %S Value %i\n", szName, iValue);
2758 
2759                 if (!iValue)
2760                     continue;
2761 
2762                 if (!_wcsicmp(szName, L"BitsPerPel"))
2763                 {
2764                     dm.dmFields |= DM_BITSPERPEL;
2765                     dm.dmBitsPerPel = iValue;
2766                 }
2767                 else if (!_wcsicmp(szName, L"XResolution"))
2768                 {
2769                     dm.dmFields |= DM_PELSWIDTH;
2770                     dm.dmPelsWidth = iValue;
2771                 }
2772                 else if (!_wcsicmp(szName, L"YResolution"))
2773                 {
2774                     dm.dmFields |= DM_PELSHEIGHT;
2775                     dm.dmPelsHeight = iValue;
2776                 }
2777                 else if (!_wcsicmp(szName, L"VRefresh"))
2778                 {
2779                     dm.dmFields |= DM_DISPLAYFREQUENCY;
2780                     dm.dmDisplayFrequency = iValue;
2781                 }
2782             } while (SetupFindNextLine(&InfContext, &InfContext));
2783 
2784             ChangeDisplaySettingsW(&dm, CDS_UPDATEREGISTRY);
2785         }
2786     }
2787 
2788     if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
2789                       L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\RunOnce",
2790                       0,
2791                       KEY_SET_VALUE,
2792                       &hKey) != ERROR_SUCCESS)
2793     {
2794         DPRINT1("Error: failed to open HKLM\\Software\\Microsoft\\Windows\\CurrentVersion\\RunOnce\n");
2795         return;
2796     }
2797 
2798     if (SetupFindFirstLineW(pSetupData->hSetupInf,
2799                             L"GuiRunOnce",
2800                             NULL,
2801                             &InfContext))
2802     {
2803         int i = 0;
2804         do
2805         {
2806             if (SetupGetStringFieldW(&InfContext,
2807                                      0,
2808                                      szValue,
2809                                      ARRAYSIZE(szValue),
2810                                      NULL))
2811             {
2812                 WCHAR szPath[MAX_PATH];
2813                 swprintf(szName, L"%d", i);
2814                 DPRINT("szName %S szValue %S\n", szName, szValue);
2815 
2816                 if (ExpandEnvironmentStringsW(szValue, szPath, MAX_PATH))
2817                 {
2818                     DPRINT("value %S\n", szPath);
2819                     if (RegSetValueExW(hKey,
2820                                        szName,
2821                                        0,
2822                                        REG_SZ,
2823                                        (const BYTE*)szPath,
2824                                        (wcslen(szPath) + 1) * sizeof(WCHAR)) == ERROR_SUCCESS)
2825                     {
2826                         i++;
2827                     }
2828                 }
2829             }
2830         } while (SetupFindNextLine(&InfContext, &InfContext));
2831     }
2832 
2833     RegCloseKey(hKey);
2834 }
2835 
2836 VOID
2837 ProcessSetupInf(
2838     IN OUT PSETUPDATA pSetupData)
2839 {
2840     WCHAR szPath[MAX_PATH];
2841     WCHAR szValue[MAX_PATH];
2842     INFCONTEXT InfContext;
2843     DWORD LineLength;
2844     HKEY hKey;
2845     LONG res;
2846 
2847     pSetupData->hSetupInf = INVALID_HANDLE_VALUE;
2848 
2849     /* Retrieve the path of the setup INF */
2850     GetSystemDirectoryW(szPath, _countof(szPath));
2851     wcscat(szPath, L"\\$winnt$.inf");
2852 
2853     /* Open the setup INF */
2854     pSetupData->hSetupInf = SetupOpenInfFileW(szPath,
2855                                               NULL,
2856                                               INF_STYLE_OLDNT,
2857                                               NULL);
2858     if (pSetupData->hSetupInf == INVALID_HANDLE_VALUE)
2859     {
2860         DPRINT1("Error: Cannot open the setup information file %S with error %d\n", szPath, GetLastError());
2861         return;
2862     }
2863 
2864 
2865     /* Retrieve the NT source path from which the 1st-stage installer was run */
2866     if (!SetupFindFirstLineW(pSetupData->hSetupInf,
2867                              L"data",
2868                              L"sourcepath",
2869                              &InfContext))
2870     {
2871         DPRINT1("Error: Cannot find sourcepath Key! %d\n", GetLastError());
2872         return;
2873     }
2874 
2875     if (!SetupGetStringFieldW(&InfContext,
2876                               1,
2877                               szValue,
2878                               ARRAYSIZE(szValue),
2879                               &LineLength))
2880     {
2881         DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
2882         return;
2883     }
2884 
2885     *pSetupData->SourcePath = UNICODE_NULL;
2886 
2887     /* Close the setup INF as we are going to modify it manually */
2888     if (pSetupData->hSetupInf != INVALID_HANDLE_VALUE)
2889         SetupCloseInfFile(pSetupData->hSetupInf);
2890 
2891 
2892     /* Find the installation source path in Win32 format */
2893     if (!GetInstallSourceWin32(pSetupData->SourcePath,
2894                                _countof(pSetupData->SourcePath),
2895                                szValue))
2896     {
2897         *pSetupData->SourcePath = UNICODE_NULL;
2898     }
2899 
2900     /* Save the path in Win32 format in the setup INF */
2901     swprintf(szValue, L"\"%s\"", pSetupData->SourcePath);
2902     WritePrivateProfileStringW(L"data", L"dospath", szValue, szPath);
2903 
2904     /*
2905      * Save it also in the registry, in the following keys:
2906      * - HKLM\Software\Microsoft\Windows\CurrentVersion\Setup ,
2907      *   values "SourcePath" and "ServicePackSourcePath" (REG_SZ);
2908      * - HKLM\Software\Microsoft\Windows NT\CurrentVersion ,
2909      *   value "SourcePath" (REG_SZ); set to the full path (e.g. D:\I386).
2910      */
2911 #if 0
2912     res = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
2913                         L"Software\\Microsoft\\Windows NT\\CurrentVersion",
2914                         0,
2915                         KEY_ALL_ACCESS,
2916                         &hKey);
2917 
2918     if (res != ERROR_SUCCESS)
2919     {
2920         return FALSE;
2921     }
2922 #endif
2923 
2924     res = RegCreateKeyExW(HKEY_LOCAL_MACHINE,
2925                           L"Software\\Microsoft\\Windows\\CurrentVersion\\Setup",
2926                           0, NULL,
2927                           REG_OPTION_NON_VOLATILE,
2928                           KEY_ALL_ACCESS, // KEY_WRITE
2929                           NULL,
2930                           &hKey,
2931                           NULL);
2932     if (res == ERROR_SUCCESS)
2933     {
2934         res = RegSetValueExW(hKey,
2935                              L"SourcePath",
2936                              0,
2937                              REG_SZ,
2938                              (LPBYTE)pSetupData->SourcePath,
2939                              (wcslen(pSetupData->SourcePath) + 1) * sizeof(WCHAR));
2940 
2941         res = RegSetValueExW(hKey,
2942                              L"ServicePackSourcePath",
2943                              0,
2944                              REG_SZ,
2945                              (LPBYTE)pSetupData->SourcePath,
2946                              (wcslen(pSetupData->SourcePath) + 1) * sizeof(WCHAR));
2947 
2948         RegCloseKey(hKey);
2949     }
2950 
2951 
2952     /* Now, re-open the setup INF (this must succeed) */
2953     pSetupData->hSetupInf = SetupOpenInfFileW(szPath,
2954                                               NULL,
2955                                               INF_STYLE_OLDNT,
2956                                               NULL);
2957     if (pSetupData->hSetupInf == INVALID_HANDLE_VALUE)
2958     {
2959         DPRINT1("Error: Cannot open the setup information file %S with error %d\n", szPath, GetLastError());
2960         return;
2961     }
2962 
2963     /* Process the unattended section of the setup file */
2964     ProcessUnattendSection(pSetupData);
2965 }
2966 
2967 typedef DWORD(WINAPI *PFNREQUESTWIZARDPAGES)(PDWORD, HPROPSHEETPAGE *, PSETUPDATA);
2968 
2969 VOID
2970 InstallWizard(VOID)
2971 {
2972     PROPSHEETHEADER psh = {0};
2973     HPROPSHEETPAGE *phpage = NULL;
2974     PROPSHEETPAGE psp = {0};
2975     UINT nPages = 0;
2976     HWND hWnd;
2977     MSG msg;
2978     PSETUPDATA pSetupData = NULL;
2979     HMODULE hNetShell = NULL;
2980     PFNREQUESTWIZARDPAGES pfn = NULL;
2981     DWORD dwPageCount = 10, dwNetworkPageCount = 0;
2982 
2983     LogItem(L"BEGIN_SECTION", L"InstallWizard");
2984 
2985     /* Allocate setup data */
2986     pSetupData = HeapAlloc(GetProcessHeap(),
2987                            HEAP_ZERO_MEMORY,
2988                            sizeof(SETUPDATA));
2989     if (pSetupData == NULL)
2990     {
2991         LogItem(NULL, L"SetupData allocation failed!");
2992         MessageBoxW(NULL,
2993                     L"Setup failed to allocate global data!",
2994                     L"ReactOS Setup",
2995                     MB_ICONERROR | MB_OK);
2996         goto done;
2997     }
2998     pSetupData->ProductOption = PRODUCT_OPTION_DEFAULT;
2999 
3000     hNetShell = LoadLibraryW(L"netshell.dll");
3001     if (hNetShell != NULL)
3002     {
3003         DPRINT("Netshell.dll loaded!\n");
3004 
3005         pfn = (PFNREQUESTWIZARDPAGES)GetProcAddress(hNetShell,
3006                                                     "NetSetupRequestWizardPages");
3007         if (pfn != NULL)
3008         {
3009             pfn(&dwNetworkPageCount, NULL, NULL);
3010             dwPageCount += dwNetworkPageCount;
3011         }
3012     }
3013 
3014     DPRINT("PageCount: %lu\n", dwPageCount);
3015 
3016     phpage = HeapAlloc(GetProcessHeap(),
3017                        HEAP_ZERO_MEMORY,
3018                        dwPageCount * sizeof(HPROPSHEETPAGE));
3019     if (phpage == NULL)
3020     {
3021         LogItem(NULL, L"Page array allocation failed!");
3022         MessageBoxW(NULL,
3023                     L"Setup failed to allocate page array!",
3024                     L"ReactOS Setup",
3025                     MB_ICONERROR | MB_OK);
3026         goto done;
3027     }
3028 
3029     /* Process the $winnt$.inf setup file */
3030     ProcessSetupInf(pSetupData);
3031 
3032     /* Create the Welcome page */
3033     psp.dwSize = sizeof(PROPSHEETPAGE);
3034     psp.dwFlags = PSP_DEFAULT | PSP_HIDEHEADER;
3035     psp.hInstance = hDllInstance;
3036     psp.lParam = (LPARAM)pSetupData;
3037     psp.pfnDlgProc = WelcomeDlgProc;
3038     psp.pszTemplate = MAKEINTRESOURCE(IDD_WELCOMEPAGE);
3039     phpage[nPages++] = CreatePropertySheetPage(&psp);
3040 
3041     /* Create the Acknowledgements page */
3042     psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
3043     psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_ACKTITLE);
3044     psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_ACKSUBTITLE);
3045     psp.pszTemplate = MAKEINTRESOURCE(IDD_ACKPAGE);
3046     psp.pfnDlgProc = AckPageDlgProc;
3047     phpage[nPages++] = CreatePropertySheetPage(&psp);
3048 
3049     /* Create the Product page */
3050     psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
3051     psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_PRODUCTTITLE);
3052     psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_PRODUCTSUBTITLE);
3053     psp.pszTemplate = MAKEINTRESOURCE(IDD_PRODUCT);
3054     psp.pfnDlgProc = ProductPageDlgProc;
3055     phpage[nPages++] = CreatePropertySheetPage(&psp);
3056 
3057     /* Create the Locale page */
3058     psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
3059     psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_LOCALETITLE);
3060     psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_LOCALESUBTITLE);
3061     psp.pfnDlgProc = LocalePageDlgProc;
3062     psp.pszTemplate = MAKEINTRESOURCE(IDD_LOCALEPAGE);
3063     phpage[nPages++] = CreatePropertySheetPage(&psp);
3064 
3065     /* Create the Owner page */
3066     psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
3067     psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_OWNERTITLE);
3068     psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_OWNERSUBTITLE);
3069     psp.pszTemplate = MAKEINTRESOURCE(IDD_OWNERPAGE);
3070     psp.pfnDlgProc = OwnerPageDlgProc;
3071     phpage[nPages++] = CreatePropertySheetPage(&psp);
3072 
3073     /* Create the Computer page */
3074     psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
3075     psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_COMPUTERTITLE);
3076     psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_COMPUTERSUBTITLE);
3077     psp.pfnDlgProc = ComputerPageDlgProc;
3078     psp.pszTemplate = MAKEINTRESOURCE(IDD_COMPUTERPAGE);
3079     phpage[nPages++] = CreatePropertySheetPage(&psp);
3080 
3081     /* Create the DateTime page */
3082     psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
3083     psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_DATETIMETITLE);
3084     psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_DATETIMESUBTITLE);
3085     psp.pfnDlgProc = DateTimePageDlgProc;
3086     psp.pszTemplate = MAKEINTRESOURCE(IDD_DATETIMEPAGE);
3087     phpage[nPages++] = CreatePropertySheetPage(&psp);
3088 
3089     /* Create the theme selection page */
3090     psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
3091     psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_THEMESELECTIONTITLE);
3092     psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_THEMESELECTIONSUBTITLE);
3093     psp.pfnDlgProc = ThemePageDlgProc;
3094     psp.pszTemplate = MAKEINTRESOURCE(IDD_THEMEPAGE);
3095     phpage[nPages++] = CreatePropertySheetPage(&psp);
3096 
3097     pSetupData->uFirstNetworkWizardPage = IDD_PROCESSPAGE;
3098     pSetupData->uPostNetworkWizardPage = IDD_PROCESSPAGE;
3099 
3100     if (pfn)
3101     {
3102         pfn(&dwNetworkPageCount, &phpage[nPages], pSetupData);
3103         nPages += dwNetworkPageCount;
3104     }
3105 
3106     /* Create the Process page */
3107     psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
3108     psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_PROCESSTITLE);
3109     psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_PROCESSSUBTITLE);
3110     psp.pfnDlgProc = ProcessPageDlgProc;
3111     psp.pszTemplate = MAKEINTRESOURCE(IDD_PROCESSPAGE);
3112     phpage[nPages++] = CreatePropertySheetPage(&psp);
3113 
3114     /* Create the Finish page */
3115     psp.dwFlags = PSP_DEFAULT | PSP_HIDEHEADER;
3116     psp.pfnDlgProc = FinishDlgProc;
3117     psp.pszTemplate = MAKEINTRESOURCE(IDD_FINISHPAGE);
3118     phpage[nPages++] = CreatePropertySheetPage(&psp);
3119 
3120     ASSERT(nPages == dwPageCount);
3121 
3122     /* Create the property sheet */
3123     psh.dwSize = sizeof(PROPSHEETHEADER);
3124     psh.dwFlags = PSH_WIZARD97 | PSH_WATERMARK | PSH_HEADER | PSH_MODELESS;
3125     psh.hInstance = hDllInstance;
3126     psh.hwndParent = NULL;
3127     psh.nPages = nPages;
3128     psh.nStartPage = 0;
3129     psh.phpage = phpage;
3130     psh.pszbmWatermark = MAKEINTRESOURCE(IDB_WATERMARK);
3131     psh.pszbmHeader = MAKEINTRESOURCE(IDB_HEADER);
3132 
3133     /* Create title font */
3134     pSetupData->hTitleFont = CreateTitleFont();
3135     pSetupData->hBoldFont  = CreateBoldFont();
3136 
3137     /* Display the wizard */
3138     hWnd = (HWND)PropertySheet(&psh);
3139     ShowWindow(hWnd, SW_SHOW);
3140 
3141     while (GetMessage(&msg, NULL, 0, 0))
3142     {
3143         if (!IsDialogMessage(hWnd, &msg))
3144         {
3145             TranslateMessage(&msg);
3146             DispatchMessage(&msg);
3147         }
3148     }
3149 
3150     DeleteObject(pSetupData->hBoldFont);
3151     DeleteObject(pSetupData->hTitleFont);
3152 
3153     if (pSetupData->hSetupInf != INVALID_HANDLE_VALUE)
3154         SetupCloseInfFile(pSetupData->hSetupInf);
3155 
3156 done:
3157     if (phpage != NULL)
3158         HeapFree(GetProcessHeap(), 0, phpage);
3159 
3160     if (hNetShell != NULL)
3161         FreeLibrary(hNetShell);
3162 
3163     if (pSetupData != NULL)
3164         HeapFree(GetProcessHeap(), 0, pSetupData);
3165 
3166     LogItem(L"END_SECTION", L"InstallWizard");
3167 }
3168 
3169 /* EOF */
3170