mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
50d260a7f0
Switching themes during initial setup broke at some point relatively recently - following said regression, selecting a theme simply has no effect at all. This fixes that.
3336 lines
102 KiB
C
3336 lines
102 KiB
C
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: System setup
|
|
* FILE: dll/win32/syssetup/wizard.c
|
|
* PURPOSE: GUI controls
|
|
* PROGRAMMERS: Eric Kohl
|
|
* Pierre Schweitzer <heis_spiter@hotmail.com>
|
|
* Ismael Ferreras Morezuelas <swyterzone+ros@gmail.com>
|
|
* Katayama Hirofumi MZ <katayama.hirofumi.mz@gmail.com>
|
|
* Oleg Dubinskiy <oleg.dubinskij30@gmail.com>
|
|
*/
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
#include "precomp.h"
|
|
|
|
#include <stdlib.h>
|
|
#include <time.h>
|
|
#include <winnls.h>
|
|
#include <windowsx.h>
|
|
#include <wincon.h>
|
|
#include <shlobj.h>
|
|
#include <shlwapi.h>
|
|
#include <tzlib.h>
|
|
#include <strsafe.h>
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
#define PM_REGISTRATION_NOTIFY (WM_APP + 1)
|
|
/* Private Message used to communicate progress from the background
|
|
registration thread to the main thread.
|
|
wParam = 0 Registration in progress
|
|
= 1 Registration completed
|
|
lParam = Pointer to a REGISTRATIONNOTIFY structure */
|
|
|
|
#define PM_ITEM_START (WM_APP + 2)
|
|
#define PM_ITEM_END (WM_APP + 3)
|
|
#define PM_STEP_START (WM_APP + 4)
|
|
#define PM_STEP_END (WM_APP + 5)
|
|
#define PM_ITEMS_DONE (WM_APP + 6)
|
|
|
|
typedef struct _REGISTRATIONNOTIFY
|
|
{
|
|
ULONG Progress;
|
|
UINT ActivityID;
|
|
LPCWSTR CurrentItem;
|
|
LPCWSTR ErrorMessage;
|
|
UINT MessageID;
|
|
DWORD LastError;
|
|
} REGISTRATIONNOTIFY, *PREGISTRATIONNOTIFY;
|
|
|
|
typedef struct _ITEMSDATA
|
|
{
|
|
HWND hwndDlg;
|
|
} ITEMSDATA, *PITEMSDATA;
|
|
|
|
typedef struct _REGISTRATIONDATA
|
|
{
|
|
HWND hwndDlg;
|
|
ULONG DllCount;
|
|
ULONG Registered;
|
|
PVOID DefaultContext;
|
|
} REGISTRATIONDATA, *PREGISTRATIONDATA;
|
|
|
|
typedef struct _TIMEZONE_ENTRY
|
|
{
|
|
struct _TIMEZONE_ENTRY *Prev;
|
|
struct _TIMEZONE_ENTRY *Next;
|
|
WCHAR Description[128]; /* 'Display' */
|
|
WCHAR StandardName[32]; /* 'Std' */
|
|
WCHAR DaylightName[32]; /* 'Dlt' */
|
|
REG_TZI_FORMAT TimezoneInfo; /* 'TZI' */
|
|
ULONG Index;
|
|
} TIMEZONE_ENTRY, *PTIMEZONE_ENTRY;
|
|
|
|
|
|
/* FUNCTIONS ****************************************************************/
|
|
|
|
extern void WINAPI Control_RunDLLW(HWND hWnd, HINSTANCE hInst, LPCWSTR cmd, DWORD nCmdShow);
|
|
|
|
|
|
static VOID
|
|
CenterWindow(HWND hWnd)
|
|
{
|
|
HWND hWndParent;
|
|
RECT rcParent;
|
|
RECT rcWindow;
|
|
|
|
hWndParent = GetParent(hWnd);
|
|
if (hWndParent == NULL)
|
|
hWndParent = GetDesktopWindow();
|
|
|
|
GetWindowRect(hWndParent, &rcParent);
|
|
GetWindowRect(hWnd, &rcWindow);
|
|
|
|
SetWindowPos(hWnd,
|
|
HWND_TOP,
|
|
((rcParent.right - rcParent.left) - (rcWindow.right - rcWindow.left)) / 2,
|
|
((rcParent.bottom - rcParent.top) - (rcWindow.bottom - rcWindow.top)) / 2,
|
|
0,
|
|
0,
|
|
SWP_NOSIZE);
|
|
}
|
|
|
|
|
|
static HFONT
|
|
CreateTitleFont(VOID)
|
|
{
|
|
LOGFONTW LogFont = {0};
|
|
HDC hdc;
|
|
HFONT hFont;
|
|
|
|
LogFont.lfWeight = FW_BOLD;
|
|
wcscpy(LogFont.lfFaceName, L"MS Shell Dlg");
|
|
|
|
hdc = GetDC(NULL);
|
|
LogFont.lfHeight = -MulDiv(12, GetDeviceCaps(hdc, LOGPIXELSY), 72);
|
|
|
|
hFont = CreateFontIndirectW(&LogFont);
|
|
|
|
ReleaseDC(NULL, hdc);
|
|
|
|
return hFont;
|
|
}
|
|
|
|
|
|
static HFONT
|
|
CreateBoldFont(VOID)
|
|
{
|
|
LOGFONTW tmpFont = {0};
|
|
HFONT hBoldFont;
|
|
HDC hDc;
|
|
|
|
/* Grabs the Drawing Context */
|
|
hDc = GetDC(NULL);
|
|
|
|
tmpFont.lfHeight = -MulDiv(8, GetDeviceCaps(hDc, LOGPIXELSY), 72);
|
|
tmpFont.lfWeight = FW_BOLD;
|
|
wcscpy(tmpFont.lfFaceName, L"MS Shell Dlg");
|
|
|
|
hBoldFont = CreateFontIndirectW(&tmpFont);
|
|
|
|
ReleaseDC(NULL, hDc);
|
|
|
|
return hBoldFont;
|
|
}
|
|
|
|
static INT_PTR CALLBACK
|
|
GplDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
HRSRC GplTextResource;
|
|
HGLOBAL GplTextMem;
|
|
PVOID GplTextLocked;
|
|
PCHAR GplText;
|
|
DWORD Size;
|
|
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
GplTextResource = FindResourceW(hDllInstance, MAKEINTRESOURCE(IDR_GPL), L"RT_TEXT");
|
|
if (NULL == GplTextResource)
|
|
{
|
|
break;
|
|
}
|
|
Size = SizeofResource(hDllInstance, GplTextResource);
|
|
if (0 == Size)
|
|
{
|
|
break;
|
|
}
|
|
GplText = HeapAlloc(GetProcessHeap(), 0, Size + 1);
|
|
if (NULL == GplText)
|
|
{
|
|
break;
|
|
}
|
|
GplTextMem = LoadResource(hDllInstance, GplTextResource);
|
|
if (NULL == GplTextMem)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, GplText);
|
|
break;
|
|
}
|
|
GplTextLocked = LockResource(GplTextMem);
|
|
if (NULL == GplTextLocked)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, GplText);
|
|
break;
|
|
}
|
|
memcpy(GplText, GplTextLocked, Size);
|
|
GplText[Size] = '\0';
|
|
SendMessageA(GetDlgItem(hwndDlg, IDC_GPL_TEXT), WM_SETTEXT, 0, (LPARAM) GplText);
|
|
HeapFree(GetProcessHeap(), 0, GplText);
|
|
SetFocus(GetDlgItem(hwndDlg, IDOK));
|
|
return FALSE;
|
|
|
|
case WM_CLOSE:
|
|
EndDialog(hwndDlg, IDCANCEL);
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
if (HIWORD(wParam) == BN_CLICKED && IDOK == LOWORD(wParam))
|
|
{
|
|
EndDialog(hwndDlg, IDOK);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static INT_PTR CALLBACK
|
|
WelcomeDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
PSETUPDATA pSetupData;
|
|
|
|
pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
HWND hwndControl;
|
|
DWORD dwStyle;
|
|
|
|
/* Get pointer to the global setup data */
|
|
pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
|
|
|
|
hwndControl = GetParent(hwndDlg);
|
|
|
|
/* Center the wizard window */
|
|
CenterWindow (hwndControl);
|
|
|
|
/* Hide the system menu */
|
|
dwStyle = GetWindowLongPtr(hwndControl, GWL_STYLE);
|
|
SetWindowLongPtr(hwndControl, GWL_STYLE, dwStyle & ~WS_SYSMENU);
|
|
|
|
/* Hide and disable the 'Cancel' button */
|
|
hwndControl = GetDlgItem(GetParent(hwndDlg), IDCANCEL);
|
|
ShowWindow (hwndControl, SW_HIDE);
|
|
EnableWindow (hwndControl, FALSE);
|
|
|
|
/* Set title font */
|
|
SendDlgItemMessage(hwndDlg,
|
|
IDC_WELCOMETITLE,
|
|
WM_SETFONT,
|
|
(WPARAM)pSetupData->hTitleFont,
|
|
(LPARAM)TRUE);
|
|
}
|
|
break;
|
|
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
LPNMHDR lpnm = (LPNMHDR)lParam;
|
|
|
|
switch (lpnm->code)
|
|
{
|
|
case PSN_SETACTIVE:
|
|
LogItem(L"BEGIN", L"WelcomePage");
|
|
/* Enable the Next button */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_NEXT);
|
|
if (pSetupData->UnattendSetup)
|
|
{
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_ACKPAGE);
|
|
return TRUE;
|
|
}
|
|
break;
|
|
|
|
case PSN_WIZNEXT:
|
|
LogItem(L"END", L"WelcomePage");
|
|
break;
|
|
|
|
case PSN_WIZBACK:
|
|
pSetupData->UnattendSetup = FALSE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static INT_PTR CALLBACK
|
|
AckPageDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
LPNMHDR lpnm;
|
|
PWCHAR Projects;
|
|
PWCHAR End, CurrentProject;
|
|
INT ProjectsSize, ProjectsCount;
|
|
PSETUPDATA pSetupData;
|
|
|
|
pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
|
|
|
|
Projects = NULL;
|
|
ProjectsSize = 256;
|
|
while (TRUE)
|
|
{
|
|
Projects = HeapAlloc(GetProcessHeap(), 0, ProjectsSize * sizeof(WCHAR));
|
|
if (NULL == Projects)
|
|
{
|
|
return FALSE;
|
|
}
|
|
ProjectsCount = LoadStringW(hDllInstance, IDS_ACKPROJECTS, Projects, ProjectsSize);
|
|
if (0 == ProjectsCount)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, Projects);
|
|
return FALSE;
|
|
}
|
|
if (ProjectsCount < ProjectsSize - 1)
|
|
{
|
|
break;
|
|
}
|
|
HeapFree(GetProcessHeap(), 0, Projects);
|
|
ProjectsSize *= 2;
|
|
}
|
|
|
|
CurrentProject = Projects;
|
|
while (*CurrentProject != L'\0')
|
|
{
|
|
End = wcschr(CurrentProject, L'\n');
|
|
if (NULL != End)
|
|
{
|
|
*End = L'\0';
|
|
}
|
|
(void)ListBox_AddString(GetDlgItem(hwndDlg, IDC_PROJECTS), CurrentProject);
|
|
if (NULL != End)
|
|
{
|
|
CurrentProject = End + 1;
|
|
}
|
|
else
|
|
{
|
|
CurrentProject += wcslen(CurrentProject);
|
|
}
|
|
}
|
|
HeapFree(GetProcessHeap(), 0, Projects);
|
|
}
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
if (HIWORD(wParam) == BN_CLICKED && IDC_VIEWGPL == LOWORD(wParam))
|
|
{
|
|
DialogBox(hDllInstance, MAKEINTRESOURCE(IDD_GPL), NULL, GplDlgProc);
|
|
SetForegroundWindow(GetParent(hwndDlg));
|
|
}
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
lpnm = (LPNMHDR)lParam;
|
|
|
|
switch (lpnm->code)
|
|
{
|
|
case PSN_SETACTIVE:
|
|
/* Enable the Back and Next buttons */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
|
|
if (pSetupData->UnattendSetup)
|
|
{
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_PRODUCT);
|
|
return TRUE;
|
|
}
|
|
break;
|
|
|
|
case PSN_WIZBACK:
|
|
pSetupData->UnattendSetup = FALSE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static const WCHAR s_szProductOptions[] = L"SYSTEM\\CurrentControlSet\\Control\\ProductOptions";
|
|
static const WCHAR s_szRosVersion[] = L"SYSTEM\\CurrentControlSet\\Control\\ReactOS\\Settings\\Version";
|
|
static const WCHAR s_szControlWindows[] = L"SYSTEM\\CurrentControlSet\\Control\\Windows";
|
|
static const WCHAR s_szWinlogon[] = L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon";
|
|
static const WCHAR s_szDefaultSoundEvents[] = L"AppEvents\\Schemes\\Apps\\.Default";
|
|
static const WCHAR s_szExplorerSoundEvents[] = L"AppEvents\\Schemes\\Apps\\Explorer";
|
|
|
|
typedef struct _PRODUCT_OPTION_DATA
|
|
{
|
|
LPCWSTR ProductSuite;
|
|
LPCWSTR ProductType;
|
|
DWORD ReportAsWorkstation;
|
|
DWORD CSDVersion;
|
|
DWORD LogonType;
|
|
} PRODUCT_OPTION_DATA;
|
|
|
|
static const PRODUCT_OPTION_DATA s_ProductOptionData[] =
|
|
{
|
|
{ L"Terminal Server\0", L"ServerNT", 0, 0x200, 0 },
|
|
{ L"\0", L"WinNT", 1, 0x300, 1 }
|
|
};
|
|
|
|
static const WCHAR* s_DefaultSoundEvents[][2] =
|
|
{
|
|
{ L".Default", L"%SystemRoot%\\Media\\ReactOS_Default.wav" },
|
|
{ L"AppGPFault", L"" },
|
|
{ L"Close", L"" },
|
|
{ L"CriticalBatteryAlarm", L"%SystemRoot%\\Media\\ReactOS_Battery_Critical.wav" },
|
|
{ L"DeviceConnect", L"%SystemRoot%\\Media\\ReactOS_Hardware_Insert.wav" },
|
|
{ L"DeviceDisconnect", L"%SystemRoot%\\Media\\ReactOS_Hardware_Remove.wav" },
|
|
{ L"DeviceFail", L"%SystemRoot%\\Media\\ReactOS_Hardware_Fail.wav" },
|
|
{ L"LowBatteryAlarm", L"%SystemRoot%\\Media\\ReactOS_Battery_Low.wav" },
|
|
{ L"MailBeep", L"%SystemRoot%\\Media\\ReactOS_Notify.wav" },
|
|
{ L"Maximize", L"%SystemRoot%\\Media\\ReactOS_Restore.wav" },
|
|
{ L"MenuCommand", L"%SystemRoot%\\Media\\ReactOS_Menu_Command.wav" },
|
|
{ L"MenuPopup", L"" },
|
|
{ L"Minimize", L"%SystemRoot%\\Media\\ReactOS_Minimize.wav" },
|
|
{ L"Open", L"" },
|
|
{ L"PrintComplete", L"%SystemRoot%\\Media\\ReactOS_Print_Complete.wav" },
|
|
{ L"RestoreDown", L"" },
|
|
{ L"RestoreUp", L"" },
|
|
{ L"SystemAsterisk", L"%SystemRoot%\\Media\\ReactOS_Ding.wav" },
|
|
{ L"SystemExclamation", L"%SystemRoot%\\Media\\ReactOS_Exclamation.wav" },
|
|
{ L"SystemExit", L"%SystemRoot%\\Media\\ReactOS_Shutdown.wav" },
|
|
{ L"SystemHand", L"%SystemRoot%\\Media\\ReactOS_Critical_Stop.wav" },
|
|
{ L"SystemNotification", L"%SystemRoot%\\Media\\ReactOS_Balloon.wav" },
|
|
{ L"SystemQuestion", L"%SystemRoot%\\Media\\ReactOS_Ding.wav" },
|
|
{ L"SystemStart", L"%SystemRoot%\\Media\\ReactOS_Startup.wav" },
|
|
{ L"WindowsLogoff", L"%SystemRoot%\\Media\\ReactOS_LogOff.wav" }
|
|
/* Logon sound is already set by default for both Server and Workstation */
|
|
};
|
|
|
|
static const WCHAR* s_ExplorerSoundEvents[][2] =
|
|
{
|
|
{ L"EmptyRecycleBin", L"%SystemRoot%\\Media\\ReactOS_Recycle.wav" },
|
|
{ L"Navigating", L"%SystemRoot%\\Media\\ReactOS_Start.wav" }
|
|
};
|
|
|
|
static BOOL
|
|
DoWriteSoundEvents(HKEY hKey,
|
|
LPCWSTR lpSubkey,
|
|
LPCWSTR lpEventsArray[][2],
|
|
DWORD dwSize)
|
|
{
|
|
HKEY hRootKey, hEventKey, hDefaultKey;
|
|
LONG error;
|
|
ULONG i;
|
|
WCHAR szDest[MAX_PATH];
|
|
DWORD dwAttribs;
|
|
DWORD cbData;
|
|
|
|
/* Open the sound events key */
|
|
error = RegOpenKeyExW(hKey, lpSubkey, 0, KEY_READ, &hRootKey);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegOpenKeyExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
/* Set each sound event */
|
|
for (i = 0; i < dwSize; i++)
|
|
{
|
|
/*
|
|
* Verify that the sound file exists and is an actual file.
|
|
*/
|
|
|
|
/* Expand the sound file path */
|
|
if (!ExpandEnvironmentStringsW(lpEventsArray[i][1], szDest, _countof(szDest)))
|
|
{
|
|
/* Failed to expand, continue with the next sound event */
|
|
continue;
|
|
}
|
|
|
|
/* Check if the sound file exists and isn't a directory */
|
|
dwAttribs = GetFileAttributesW(szDest);
|
|
if ((dwAttribs == INVALID_FILE_ATTRIBUTES) ||
|
|
(dwAttribs & FILE_ATTRIBUTE_DIRECTORY))
|
|
{
|
|
/* It does not, just continue with the next sound event */
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* Create the sound event entry.
|
|
*/
|
|
|
|
/* Open the sound event subkey */
|
|
error = RegOpenKeyExW(hRootKey, lpEventsArray[i][0], 0, KEY_READ, &hEventKey);
|
|
if (error)
|
|
{
|
|
/* Failed to open, continue with next sound event */
|
|
continue;
|
|
}
|
|
|
|
/* Open .Default subkey */
|
|
error = RegOpenKeyExW(hEventKey, L".Default", 0, KEY_WRITE, &hDefaultKey);
|
|
RegCloseKey(hEventKey);
|
|
if (error)
|
|
{
|
|
/* Failed to open, continue with next sound event */
|
|
continue;
|
|
}
|
|
|
|
/* Associate the sound file to this sound event */
|
|
cbData = (lstrlenW(lpEventsArray[i][1]) + 1) * sizeof(WCHAR);
|
|
error = RegSetValueExW(hDefaultKey, NULL, 0, REG_EXPAND_SZ, (const BYTE *)lpEventsArray[i][1], cbData);
|
|
RegCloseKey(hDefaultKey);
|
|
if (error)
|
|
{
|
|
/* Failed to set the value, continue with next sound event */
|
|
continue;
|
|
}
|
|
}
|
|
|
|
Error:
|
|
if (hRootKey)
|
|
RegCloseKey(hRootKey);
|
|
|
|
return error == ERROR_SUCCESS;
|
|
}
|
|
|
|
static BOOL
|
|
DoWriteProductOption(PRODUCT_OPTION nOption)
|
|
{
|
|
HKEY hKey;
|
|
LONG error;
|
|
LPCWSTR pszData;
|
|
DWORD dwValue, cbData;
|
|
const PRODUCT_OPTION_DATA *pData = &s_ProductOptionData[nOption];
|
|
ASSERT(0 <= nOption && nOption < _countof(s_ProductOptionData));
|
|
|
|
/* open ProductOptions key */
|
|
error = RegOpenKeyExW(HKEY_LOCAL_MACHINE, s_szProductOptions, 0, KEY_WRITE, &hKey);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegOpenKeyExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
/* write ProductSuite */
|
|
pszData = pData->ProductSuite;
|
|
cbData = (lstrlenW(pszData) + 2) * sizeof(WCHAR);
|
|
error = RegSetValueExW(hKey, L"ProductSuite", 0, REG_MULTI_SZ, (const BYTE *)pszData, cbData);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegSetValueExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
/* write ProductType */
|
|
pszData = pData->ProductType;
|
|
cbData = (lstrlenW(pszData) + 1) * sizeof(WCHAR);
|
|
error = RegSetValueExW(hKey, L"ProductType", 0, REG_SZ, (const BYTE *)pszData, cbData);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegSetValueExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
/* open ReactOS version key */
|
|
error = RegOpenKeyExW(HKEY_LOCAL_MACHINE, s_szRosVersion, 0, KEY_WRITE, &hKey);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegOpenKeyExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
/* write ReportAsWorkstation */
|
|
dwValue = pData->ReportAsWorkstation;
|
|
cbData = sizeof(dwValue);
|
|
error = RegSetValueExW(hKey, L"ReportAsWorkstation", 0, REG_DWORD, (const BYTE *)&dwValue, cbData);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegSetValueExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
/* open Control Windows key */
|
|
error = RegOpenKeyExW(HKEY_LOCAL_MACHINE, s_szControlWindows, 0, KEY_WRITE, &hKey);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegOpenKeyExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
/* write Control Windows CSDVersion */
|
|
dwValue = pData->CSDVersion;
|
|
cbData = sizeof(dwValue);
|
|
error = RegSetValueExW(hKey, L"CSDVersion", 0, REG_DWORD, (const BYTE *)&dwValue, cbData);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegSetValueExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
/* open Winlogon key */
|
|
error = RegOpenKeyExW(HKEY_LOCAL_MACHINE, s_szWinlogon, 0, KEY_WRITE, &hKey);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegOpenKeyExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
/* write LogonType */
|
|
dwValue = pData->LogonType;
|
|
cbData = sizeof(dwValue);
|
|
error = RegSetValueExW(hKey, L"LogonType", 0, REG_DWORD, (const BYTE *)&dwValue, cbData);
|
|
if (error)
|
|
{
|
|
DPRINT1("RegSetValueExW failed\n");
|
|
goto Error;
|
|
}
|
|
|
|
if (nOption == PRODUCT_OPTION_WORKSTATION)
|
|
{
|
|
/* Write system sound events values for Workstation */
|
|
DoWriteSoundEvents(HKEY_CURRENT_USER, s_szDefaultSoundEvents, s_DefaultSoundEvents, _countof(s_DefaultSoundEvents));
|
|
DoWriteSoundEvents(HKEY_CURRENT_USER, s_szExplorerSoundEvents, s_ExplorerSoundEvents, _countof(s_ExplorerSoundEvents));
|
|
}
|
|
|
|
Error:
|
|
if (hKey)
|
|
RegCloseKey(hKey);
|
|
|
|
return error == ERROR_SUCCESS;
|
|
}
|
|
|
|
static void
|
|
OnChooseOption(HWND hwndDlg, PRODUCT_OPTION nOption)
|
|
{
|
|
WCHAR szText[256];
|
|
ASSERT(0 <= nOption && nOption < _countof(s_ProductOptionData));
|
|
|
|
switch (nOption)
|
|
{
|
|
case PRODUCT_OPTION_SERVER:
|
|
LoadStringW(hDllInstance, IDS_PRODUCTSERVERINFO, szText, _countof(szText));
|
|
break;
|
|
|
|
case PRODUCT_OPTION_WORKSTATION:
|
|
LoadStringW(hDllInstance, IDS_PRODUCTWORKSTATIONINFO, szText, _countof(szText));
|
|
break;
|
|
|
|
default:
|
|
return;
|
|
}
|
|
|
|
SetDlgItemTextW(hwndDlg, IDC_PRODUCT_DESCRIPTION, szText);
|
|
}
|
|
|
|
static INT_PTR CALLBACK
|
|
ProductPageDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LPNMHDR lpnm;
|
|
PSETUPDATA pSetupData;
|
|
INT iItem;
|
|
WCHAR szText[64], szDefault[64];
|
|
HICON hIcon;
|
|
|
|
pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
|
|
|
|
LoadStringW(hDllInstance, IDS_DEFAULT, szDefault, _countof(szDefault));
|
|
|
|
LoadStringW(hDllInstance, IDS_PRODUCTSERVERNAME, szText, _countof(szText));
|
|
if (PRODUCT_OPTION_DEFAULT == PRODUCT_OPTION_SERVER)
|
|
{
|
|
StringCchCatW(szText, _countof(szText), L" ");
|
|
StringCchCatW(szText, _countof(szText), szDefault);
|
|
}
|
|
SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_ADDSTRING, 0, (LPARAM)szText);
|
|
|
|
LoadStringW(hDllInstance, IDS_PRODUCTWORKSTATIONNAME, szText, _countof(szText));
|
|
if (PRODUCT_OPTION_DEFAULT == PRODUCT_OPTION_WORKSTATION)
|
|
{
|
|
StringCchCatW(szText, _countof(szText), L" ");
|
|
StringCchCatW(szText, _countof(szText), szDefault);
|
|
}
|
|
SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_ADDSTRING, 0, (LPARAM)szText);
|
|
|
|
SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_SETCURSEL, PRODUCT_OPTION_DEFAULT, 0);
|
|
OnChooseOption(hwndDlg, PRODUCT_OPTION_DEFAULT);
|
|
|
|
hIcon = LoadIcon(NULL, IDI_WINLOGO);
|
|
SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_ICON, STM_SETICON, (WPARAM)hIcon, 0);
|
|
return TRUE;
|
|
}
|
|
|
|
case WM_COMMAND:
|
|
if (HIWORD(wParam) == CBN_SELCHANGE && IDC_PRODUCT_OPTIONS == LOWORD(wParam))
|
|
{
|
|
iItem = SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_GETCURSEL, 0, 0);
|
|
OnChooseOption(hwndDlg, (PRODUCT_OPTION)iItem);
|
|
}
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
lpnm = (LPNMHDR)lParam;
|
|
|
|
switch (lpnm->code)
|
|
{
|
|
case PSN_SETACTIVE:
|
|
/* Enable the Back and Next buttons */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
|
|
if (pSetupData->UnattendSetup)
|
|
{
|
|
OnChooseOption(hwndDlg, pSetupData->ProductOption);
|
|
DoWriteProductOption(pSetupData->ProductOption);
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_LOCALEPAGE);
|
|
return TRUE;
|
|
}
|
|
break;
|
|
|
|
case PSN_WIZNEXT:
|
|
iItem = SendDlgItemMessageW(hwndDlg, IDC_PRODUCT_OPTIONS, CB_GETCURSEL, 0, 0);
|
|
pSetupData->ProductOption = (PRODUCT_OPTION)iItem;
|
|
DoWriteProductOption(pSetupData->ProductOption);
|
|
break;
|
|
|
|
case PSN_WIZBACK:
|
|
pSetupData->UnattendSetup = FALSE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static
|
|
BOOL
|
|
WriteOwnerSettings(WCHAR * OwnerName,
|
|
WCHAR * OwnerOrganization)
|
|
{
|
|
HKEY hKey;
|
|
LONG res;
|
|
|
|
res = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
L"Software\\Microsoft\\Windows NT\\CurrentVersion",
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
&hKey);
|
|
|
|
if (res != ERROR_SUCCESS)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
res = RegSetValueExW(hKey,
|
|
L"RegisteredOwner",
|
|
0,
|
|
REG_SZ,
|
|
(LPBYTE)OwnerName,
|
|
(wcslen(OwnerName) + 1) * sizeof(WCHAR));
|
|
|
|
if (res != ERROR_SUCCESS)
|
|
{
|
|
RegCloseKey(hKey);
|
|
return FALSE;
|
|
}
|
|
|
|
res = RegSetValueExW(hKey,
|
|
L"RegisteredOrganization",
|
|
0,
|
|
REG_SZ,
|
|
(LPBYTE)OwnerOrganization,
|
|
(wcslen(OwnerOrganization) + 1) * sizeof(WCHAR));
|
|
|
|
RegCloseKey(hKey);
|
|
return (res == ERROR_SUCCESS);
|
|
}
|
|
|
|
static INT_PTR CALLBACK
|
|
OwnerPageDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
WCHAR OwnerName[51];
|
|
WCHAR OwnerOrganization[51];
|
|
WCHAR Title[64];
|
|
WCHAR ErrorName[256];
|
|
LPNMHDR lpnm;
|
|
PSETUPDATA pSetupData;
|
|
|
|
pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
|
|
|
|
/* set a localized ('Owner') placeholder string as default */
|
|
if (LoadStringW(hDllInstance, IDS_MACHINE_OWNER_NAME, OwnerName, _countof(OwnerName)))
|
|
{
|
|
SendDlgItemMessage(hwndDlg, IDC_OWNERNAME, WM_SETTEXT, 0, (LPARAM)OwnerName);
|
|
}
|
|
|
|
SendDlgItemMessage(hwndDlg, IDC_OWNERNAME, EM_LIMITTEXT, 50, 0);
|
|
SendDlgItemMessage(hwndDlg, IDC_OWNERORGANIZATION, EM_LIMITTEXT, 50, 0);
|
|
|
|
/* Set focus to owner name */
|
|
SetFocus(GetDlgItem(hwndDlg, IDC_OWNERNAME));
|
|
|
|
/* Select the default text to quickly overwrite it by typing */
|
|
SendDlgItemMessage(hwndDlg, IDC_OWNERNAME, EM_SETSEL, 0, -1);
|
|
}
|
|
break;
|
|
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
lpnm = (LPNMHDR)lParam;
|
|
|
|
switch (lpnm->code)
|
|
{
|
|
case PSN_SETACTIVE:
|
|
/* Enable the Back and Next buttons */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
|
|
if (pSetupData->UnattendSetup)
|
|
{
|
|
SendMessage(GetDlgItem(hwndDlg, IDC_OWNERNAME), WM_SETTEXT, 0, (LPARAM)pSetupData->OwnerName);
|
|
SendMessage(GetDlgItem(hwndDlg, IDC_OWNERORGANIZATION), WM_SETTEXT, 0, (LPARAM)pSetupData->OwnerOrganization);
|
|
if (WriteOwnerSettings(pSetupData->OwnerName, pSetupData->OwnerOrganization))
|
|
{
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_COMPUTERPAGE);
|
|
return TRUE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case PSN_WIZNEXT:
|
|
OwnerName[0] = 0;
|
|
if (GetDlgItemTextW(hwndDlg, IDC_OWNERNAME, OwnerName, 50) == 0)
|
|
{
|
|
if (0 == LoadStringW(hDllInstance, IDS_REACTOS_SETUP, Title, ARRAYSIZE(Title)))
|
|
{
|
|
wcscpy(Title, L"ReactOS Setup");
|
|
}
|
|
if (0 == LoadStringW(hDllInstance, IDS_WZD_NAME, ErrorName, ARRAYSIZE(ErrorName)))
|
|
{
|
|
wcscpy(ErrorName, L"Setup cannot continue until you enter your name.");
|
|
}
|
|
MessageBoxW(hwndDlg, ErrorName, Title, MB_ICONERROR | MB_OK);
|
|
|
|
SetFocus(GetDlgItem(hwndDlg, IDC_OWNERNAME));
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
OwnerOrganization[0] = 0;
|
|
GetDlgItemTextW(hwndDlg, IDC_OWNERORGANIZATION, OwnerOrganization, 50);
|
|
|
|
if (!WriteOwnerSettings(OwnerName, OwnerOrganization))
|
|
{
|
|
SetFocus(GetDlgItem(hwndDlg, IDC_OWNERNAME));
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
|
|
return TRUE;
|
|
}
|
|
|
|
case PSN_WIZBACK:
|
|
pSetupData->UnattendSetup = FALSE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static
|
|
BOOL
|
|
WriteComputerSettings(WCHAR * ComputerName, HWND hwndDlg)
|
|
{
|
|
WCHAR Title[64];
|
|
WCHAR ErrorComputerName[256];
|
|
LONG lError;
|
|
HKEY hKey = NULL;
|
|
|
|
if (!SetComputerNameW(ComputerName))
|
|
{
|
|
if (hwndDlg != NULL)
|
|
{
|
|
if (0 == LoadStringW(hDllInstance, IDS_REACTOS_SETUP, Title, ARRAYSIZE(Title)))
|
|
{
|
|
wcscpy(Title, L"ReactOS Setup");
|
|
}
|
|
if (0 == LoadStringW(hDllInstance, IDS_WZD_SETCOMPUTERNAME, ErrorComputerName,
|
|
ARRAYSIZE(ErrorComputerName)))
|
|
{
|
|
wcscpy(ErrorComputerName, L"Setup failed to set the computer name.");
|
|
}
|
|
MessageBoxW(hwndDlg, ErrorComputerName, Title, MB_ICONERROR | MB_OK);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* Set the physical DNS domain */
|
|
SetComputerNameExW(ComputerNamePhysicalDnsDomain, L"");
|
|
|
|
/* Set the physical DNS hostname */
|
|
SetComputerNameExW(ComputerNamePhysicalDnsHostname, ComputerName);
|
|
|
|
/* Set the accounts domain name */
|
|
SetAccountsDomainSid(NULL, ComputerName);
|
|
|
|
/* Now we need to set the Hostname */
|
|
lError = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
L"SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",
|
|
0,
|
|
KEY_SET_VALUE,
|
|
&hKey);
|
|
if (lError != ERROR_SUCCESS)
|
|
{
|
|
DPRINT1("RegOpenKeyExW for Tcpip\\Parameters failed (%08lX)\n", lError);
|
|
return TRUE;
|
|
}
|
|
|
|
lError = RegSetValueEx(hKey,
|
|
L"Hostname",
|
|
0,
|
|
REG_SZ,
|
|
(LPBYTE)ComputerName,
|
|
(wcslen(ComputerName) + 1) * sizeof(WCHAR));
|
|
if (lError != ERROR_SUCCESS)
|
|
{
|
|
DPRINT1("RegSetValueEx(\"Hostname\") failed (%08lX)\n", lError);
|
|
}
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static
|
|
BOOL
|
|
WriteDefaultLogonData(LPWSTR Domain)
|
|
{
|
|
WCHAR szAdministratorName[256];
|
|
HKEY hKey = NULL;
|
|
LONG lError;
|
|
|
|
if (LoadStringW(hDllInstance,
|
|
IDS_ADMINISTRATOR_NAME,
|
|
szAdministratorName,
|
|
ARRAYSIZE(szAdministratorName)) == 0)
|
|
{
|
|
wcscpy(szAdministratorName, L"Administrator");
|
|
}
|
|
|
|
lError = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon",
|
|
0,
|
|
KEY_SET_VALUE,
|
|
&hKey);
|
|
if (lError != ERROR_SUCCESS)
|
|
return FALSE;
|
|
|
|
lError = RegSetValueEx(hKey,
|
|
L"DefaultDomainName",
|
|
0,
|
|
REG_SZ,
|
|
(LPBYTE)Domain,
|
|
(wcslen(Domain)+ 1) * sizeof(WCHAR));
|
|
if (lError != ERROR_SUCCESS)
|
|
{
|
|
DPRINT1("RegSetValueEx(\"DefaultDomainName\") failed!\n");
|
|
}
|
|
|
|
lError = RegSetValueEx(hKey,
|
|
L"DefaultUserName",
|
|
0,
|
|
REG_SZ,
|
|
(LPBYTE)szAdministratorName,
|
|
(wcslen(szAdministratorName)+ 1) * sizeof(WCHAR));
|
|
if (lError != ERROR_SUCCESS)
|
|
{
|
|
DPRINT1("RegSetValueEx(\"DefaultUserName\") failed!\n");
|
|
}
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/* lpBuffer will be filled with a 15-char string (plus the null terminator) */
|
|
static void
|
|
GenerateComputerName(LPWSTR lpBuffer)
|
|
{
|
|
static const WCHAR Chars[] = L"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
|
static const unsigned cChars = sizeof(Chars) / sizeof(WCHAR) - 1;
|
|
unsigned i;
|
|
|
|
wcscpy(lpBuffer, L"REACTOS-");
|
|
|
|
srand(GetTickCount());
|
|
|
|
/* fill in 7 characters */
|
|
for (i = 8; i < 15; i++)
|
|
lpBuffer[i] = Chars[rand() % cChars];
|
|
|
|
lpBuffer[15] = UNICODE_NULL; /* NULL-terminate */
|
|
}
|
|
|
|
static INT_PTR CALLBACK
|
|
ComputerPageDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
WCHAR ComputerName[MAX_COMPUTERNAME_LENGTH + 1];
|
|
WCHAR Password1[128];
|
|
WCHAR Password2[128];
|
|
PWCHAR Password;
|
|
WCHAR Title[64];
|
|
WCHAR EmptyComputerName[256], NotMatchPassword[256], WrongPassword[256];
|
|
LPNMHDR lpnm;
|
|
PSETUPDATA pSetupData;
|
|
|
|
pSetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, DWLP_USER);
|
|
|
|
if (0 == LoadStringW(hDllInstance, IDS_REACTOS_SETUP, Title, ARRAYSIZE(Title)))
|
|
{
|
|
wcscpy(Title, L"ReactOS Setup");
|
|
}
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
pSetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pSetupData);
|
|
|
|
/* Generate a new pseudo-random computer name */
|
|
GenerateComputerName(ComputerName);
|
|
|
|
/* Display current computer name */
|
|
SetDlgItemTextW(hwndDlg, IDC_COMPUTERNAME, ComputerName);
|
|
|
|
/* Set text limits */
|
|
SendDlgItemMessage(hwndDlg, IDC_COMPUTERNAME, EM_LIMITTEXT, MAX_COMPUTERNAME_LENGTH, 0);
|
|
SendDlgItemMessage(hwndDlg, IDC_ADMINPASSWORD1, EM_LIMITTEXT, 127, 0);
|
|
SendDlgItemMessage(hwndDlg, IDC_ADMINPASSWORD2, EM_LIMITTEXT, 127, 0);
|
|
|
|
/* Set focus to computer name */
|
|
SetFocus(GetDlgItem(hwndDlg, IDC_COMPUTERNAME));
|
|
if (pSetupData->UnattendSetup)
|
|
{
|
|
SendMessage(GetDlgItem(hwndDlg, IDC_COMPUTERNAME), WM_SETTEXT, 0, (LPARAM)pSetupData->ComputerName);
|
|
SendMessage(GetDlgItem(hwndDlg, IDC_ADMINPASSWORD1), WM_SETTEXT, 0, (LPARAM)pSetupData->AdminPassword);
|
|
SendMessage(GetDlgItem(hwndDlg, IDC_ADMINPASSWORD2), WM_SETTEXT, 0, (LPARAM)pSetupData->AdminPassword);
|
|
WriteComputerSettings(pSetupData->ComputerName, NULL);
|
|
SetAdministratorPassword(pSetupData->AdminPassword);
|
|
}
|
|
|
|
/* Store the administrator account name as the default user name */
|
|
WriteDefaultLogonData(pSetupData->ComputerName);
|
|
break;
|
|
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
lpnm = (LPNMHDR)lParam;
|
|
|
|
switch (lpnm->code)
|
|
{
|
|
case PSN_SETACTIVE:
|
|
/* Enable the Back and Next buttons */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
|
|
if (pSetupData->UnattendSetup && WriteComputerSettings(pSetupData->ComputerName, hwndDlg))
|
|
{
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_THEMEPAGE);
|
|
return TRUE;
|
|
}
|
|
break;
|
|
|
|
case PSN_WIZNEXT:
|
|
if (0 == GetDlgItemTextW(hwndDlg, IDC_COMPUTERNAME, ComputerName, MAX_COMPUTERNAME_LENGTH + 1))
|
|
{
|
|
if (0 == LoadStringW(hDllInstance, IDS_WZD_COMPUTERNAME, EmptyComputerName,
|
|
ARRAYSIZE(EmptyComputerName)))
|
|
{
|
|
wcscpy(EmptyComputerName, L"Setup cannot continue until you enter the name of your computer.");
|
|
}
|
|
MessageBoxW(hwndDlg, EmptyComputerName, Title, MB_ICONERROR | MB_OK);
|
|
SetFocus(GetDlgItem(hwndDlg, IDC_COMPUTERNAME));
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
|
|
return TRUE;
|
|
}
|
|
|
|
/* No need to check computer name for invalid characters,
|
|
* SetComputerName() will do it for us */
|
|
|
|
if (!WriteComputerSettings(ComputerName, hwndDlg))
|
|
{
|
|
SetFocus(GetDlgItem(hwndDlg, IDC_COMPUTERNAME));
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
|
|
return TRUE;
|
|
}
|
|
|
|
#ifdef PASSWORDS_MANDATORY
|
|
/* Check if admin passwords have been entered */
|
|
if ((GetDlgItemText(hwndDlg, IDC_ADMINPASSWORD1, Password1, 128) == 0) ||
|
|
(GetDlgItemText(hwndDlg, IDC_ADMINPASSWORD2, Password2, 128) == 0))
|
|
{
|
|
if (0 == LoadStringW(hDllInstance, IDS_WZD_PASSWORDEMPTY, EmptyPassword,
|
|
ARRAYSIZE(EmptyPassword)))
|
|
{
|
|
wcscpy(EmptyPassword, L"You must enter a password !");
|
|
}
|
|
MessageBoxW(hwndDlg, EmptyPassword, Title, MB_ICONERROR | MB_OK);
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
|
|
return TRUE;
|
|
}
|
|
#else
|
|
GetDlgItemTextW(hwndDlg, IDC_ADMINPASSWORD1, Password1, 128);
|
|
GetDlgItemTextW(hwndDlg, IDC_ADMINPASSWORD2, Password2, 128);
|
|
#endif
|
|
/* Check if passwords match */
|
|
if (wcscmp(Password1, Password2))
|
|
{
|
|
if (0 == LoadStringW(hDllInstance, IDS_WZD_PASSWORDMATCH, NotMatchPassword,
|
|
ARRAYSIZE(NotMatchPassword)))
|
|
{
|
|
wcscpy(NotMatchPassword, L"The passwords you entered do not match. Please enter the desired password again.");
|
|
}
|
|
MessageBoxW(hwndDlg, NotMatchPassword, Title, MB_ICONERROR | MB_OK);
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
|
|
return TRUE;
|
|
}
|
|
|
|
/* Check password for invalid characters */
|
|
Password = (PWCHAR)Password1;
|
|
while (*Password)
|
|
{
|
|
if (!isprint(*Password))
|
|
{
|
|
if (0 == LoadStringW(hDllInstance, IDS_WZD_PASSWORDCHAR, WrongPassword,
|
|
ARRAYSIZE(WrongPassword)))
|
|
{
|
|
wcscpy(WrongPassword, L"The password you entered contains invalid characters. Please enter a cleaned password.");
|
|
}
|
|
MessageBoxW(hwndDlg, WrongPassword, Title, MB_ICONERROR | MB_OK);
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
|
|
return TRUE;
|
|
}
|
|
Password++;
|
|
}
|
|
|
|
/* Set admin password */
|
|
SetAdministratorPassword(Password1);
|
|
break;
|
|
|
|
case PSN_WIZBACK:
|
|
pSetupData->UnattendSetup = FALSE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static VOID
|
|
SetUserLocaleName(HWND hwnd)
|
|
{
|
|
WCHAR CurLocale[256] = L"";
|
|
WCHAR CurGeo[256] = L"";
|
|
WCHAR ResText[256] = L"";
|
|
WCHAR LocaleText[256 * 2];
|
|
|
|
GetLocaleInfoW(GetUserDefaultLCID(), LOCALE_SLANGUAGE, CurLocale, ARRAYSIZE(CurLocale));
|
|
GetGeoInfoW(GetUserGeoID(GEOCLASS_NATION), GEO_FRIENDLYNAME, CurGeo, ARRAYSIZE(CurGeo), GetThreadLocale());
|
|
|
|
LoadStringW(hDllInstance, IDS_LOCALETEXT, ResText, ARRAYSIZE(ResText));
|
|
StringCchPrintfW(LocaleText, ARRAYSIZE(LocaleText), ResText, CurLocale, CurGeo);
|
|
|
|
SetWindowTextW(hwnd, LocaleText);
|
|
}
|
|
|
|
static VOID
|
|
SetKeyboardLayoutName(HWND hwnd)
|
|
{
|
|
HKL hkl;
|
|
BOOL LayoutSpecial = FALSE;
|
|
WCHAR LayoutPath[256];
|
|
WCHAR LocaleName[32];
|
|
WCHAR SpecialId[5] = L"";
|
|
WCHAR ResText[256] = L"";
|
|
DWORD dwValueSize;
|
|
HKEY hKey;
|
|
UINT i;
|
|
|
|
/* Get the default input language and method */
|
|
if (!SystemParametersInfoW(SPI_GETDEFAULTINPUTLANG, 0, (LPDWORD)&hkl, 0))
|
|
{
|
|
hkl = GetKeyboardLayout(0);
|
|
}
|
|
|
|
if ((HIWORD(hkl) & 0xF000) == 0xF000)
|
|
{
|
|
/* Process keyboard layout with special id */
|
|
StringCchPrintfW(SpecialId, ARRAYSIZE(SpecialId), L"%04x", (HIWORD(hkl) & 0x0FFF));
|
|
LayoutSpecial = TRUE;
|
|
}
|
|
|
|
#define MAX_LAYOUTS_PER_LANGID 0x10000
|
|
for (i = 0; i < (LayoutSpecial ? MAX_LAYOUTS_PER_LANGID : 1); i++)
|
|
{
|
|
/* Generate a hexadecimal identifier for keyboard layout registry key */
|
|
StringCchPrintfW(LocaleName, ARRAYSIZE(LocaleName), L"%08lx", (i << 16) | LOWORD(hkl));
|
|
|
|
StringCchCopyW(LayoutPath, ARRAYSIZE(LayoutPath), L"SYSTEM\\CurrentControlSet\\Control\\Keyboard Layouts\\");
|
|
StringCchCatW(LayoutPath, ARRAYSIZE(LayoutPath), LocaleName);
|
|
*LocaleName = UNICODE_NULL;
|
|
|
|
if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
LayoutPath,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
&hKey) == ERROR_SUCCESS)
|
|
{
|
|
/* Make sure the keyboard layout key we opened is the one we need.
|
|
* If the layout has no special id, just pass this check. */
|
|
dwValueSize = sizeof(LocaleName);
|
|
if (!LayoutSpecial ||
|
|
((RegQueryValueExW(hKey,
|
|
L"Layout Id",
|
|
NULL,
|
|
NULL,
|
|
(PVOID)&LocaleName,
|
|
&dwValueSize) == ERROR_SUCCESS) &&
|
|
(wcscmp(LocaleName, SpecialId) == 0)))
|
|
{
|
|
*LocaleName = UNICODE_NULL;
|
|
dwValueSize = sizeof(LocaleName);
|
|
RegQueryValueExW(hKey,
|
|
L"Layout Text",
|
|
NULL,
|
|
NULL,
|
|
(PVOID)&LocaleName,
|
|
&dwValueSize);
|
|
/* Let the loop know where to stop */
|
|
i = MAX_LAYOUTS_PER_LANGID;
|
|
}
|
|
RegCloseKey(hKey);
|
|
}
|
|
else
|
|
{
|
|
/* Keyboard layout registry keys are expected to go in order without gaps */
|
|
break;
|
|
}
|
|
}
|
|
#undef MAX_LAYOUTS_PER_LANGID
|
|
|
|
LoadStringW(hDllInstance, IDS_LAYOUTTEXT, ResText, ARRAYSIZE(ResText));
|
|
StringCchPrintfW(LayoutPath, ARRAYSIZE(LayoutPath), ResText, LocaleName);
|
|
|
|
SetWindowTextW(hwnd, LayoutPath);
|
|
}
|
|
|
|
|
|
static BOOL
|
|
RunControlPanelApplet(HWND hwnd, PCWSTR pwszCPLParameters)
|
|
{
|
|
MSG msg;
|
|
HWND MainWindow = GetParent(hwnd);
|
|
STARTUPINFOW StartupInfo;
|
|
PROCESS_INFORMATION ProcessInformation;
|
|
WCHAR CmdLine[MAX_PATH] = L"rundll32.exe shell32.dll,Control_RunDLL ";
|
|
|
|
if (!pwszCPLParameters)
|
|
{
|
|
MessageBoxW(hwnd, L"Error: Failed to launch the Control Panel Applet.", NULL, MB_ICONERROR);
|
|
return FALSE;
|
|
}
|
|
|
|
ZeroMemory(&StartupInfo, sizeof(StartupInfo));
|
|
StartupInfo.cb = sizeof(StartupInfo);
|
|
ZeroMemory(&ProcessInformation, sizeof(ProcessInformation));
|
|
|
|
ASSERT(_countof(CmdLine) > wcslen(CmdLine) + wcslen(pwszCPLParameters));
|
|
wcscat(CmdLine, pwszCPLParameters);
|
|
|
|
if (!CreateProcessW(NULL,
|
|
CmdLine,
|
|
NULL,
|
|
NULL,
|
|
FALSE,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
&StartupInfo,
|
|
&ProcessInformation))
|
|
{
|
|
MessageBoxW(hwnd, L"Error: Failed to launch the Control Panel Applet.", NULL, MB_ICONERROR);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Disable the Back and Next buttons and the main window
|
|
* while we're interacting with the control panel applet */
|
|
PropSheet_SetWizButtons(MainWindow, 0);
|
|
EnableWindow(MainWindow, FALSE);
|
|
|
|
while ((MsgWaitForMultipleObjects(1, &ProcessInformation.hProcess, FALSE, INFINITE, QS_ALLINPUT|QS_ALLPOSTMESSAGE )) != WAIT_OBJECT_0)
|
|
{
|
|
/* We still need to process main window messages to avoid freeze */
|
|
while (PeekMessageW(&msg, NULL, 0, 0, PM_REMOVE))
|
|
{
|
|
TranslateMessage(&msg);
|
|
DispatchMessageW(&msg);
|
|
}
|
|
}
|
|
CloseHandle(ProcessInformation.hThread);
|
|
CloseHandle(ProcessInformation.hProcess);
|
|
|
|
/* Enable the Back and Next buttons and the main window again */
|
|
PropSheet_SetWizButtons(MainWindow, PSWIZB_BACK | PSWIZB_NEXT);
|
|
EnableWindow(MainWindow, TRUE);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static VOID
|
|
WriteUserLocale(VOID)
|
|
{
|
|
HKEY hKey;
|
|
LCID lcid;
|
|
WCHAR Locale[12];
|
|
|
|
lcid = GetSystemDefaultLCID();
|
|
|
|
if (GetLocaleInfoW(MAKELCID(lcid, SORT_DEFAULT), LOCALE_ILANGUAGE, Locale, ARRAYSIZE(Locale)) != 0)
|
|
{
|
|
if (RegCreateKeyExW(HKEY_CURRENT_USER, L"Control Panel\\International",
|
|
0, NULL, REG_OPTION_NON_VOLATILE,
|
|
KEY_WRITE, NULL, &hKey, NULL) == ERROR_SUCCESS)
|
|
{
|
|
RegSetValueExW(hKey, L"Locale", 0, REG_SZ, (LPBYTE)Locale, (wcslen(Locale) + 1) * sizeof(WCHAR));
|
|
RegCloseKey(hKey);
|
|
}
|
|
}
|
|
}
|
|
|
|
static INT_PTR CALLBACK
|
|
LocalePageDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
PSETUPDATA SetupData;
|
|
|
|
/* Retrieve pointer to the global setup data */
|
|
SetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
/* Save pointer to the global setup data */
|
|
SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (DWORD_PTR)SetupData);
|
|
WriteUserLocale();
|
|
|
|
SetUserLocaleName(GetDlgItem(hwndDlg, IDC_LOCALETEXT));
|
|
SetKeyboardLayoutName(GetDlgItem(hwndDlg, IDC_LAYOUTTEXT));
|
|
}
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
if (HIWORD(wParam) == BN_CLICKED)
|
|
{
|
|
switch (LOWORD(wParam))
|
|
{
|
|
case IDC_CUSTOMLOCALE:
|
|
RunControlPanelApplet(hwndDlg, L"intl.cpl,,5");
|
|
SetUserLocaleName(GetDlgItem(hwndDlg, IDC_LOCALETEXT));
|
|
break;
|
|
|
|
case IDC_CUSTOMLAYOUT:
|
|
RunControlPanelApplet(hwndDlg, L"input.dll,@1");
|
|
SetKeyboardLayoutName(GetDlgItem(hwndDlg, IDC_LAYOUTTEXT));
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
LPNMHDR lpnm = (LPNMHDR)lParam;
|
|
|
|
switch (lpnm->code)
|
|
{
|
|
case PSN_SETACTIVE:
|
|
/* Enable the Back and Next buttons */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
|
|
if (SetupData->UnattendSetup)
|
|
{
|
|
// if (!*SetupData->SourcePath)
|
|
{
|
|
RunControlPanelApplet(hwndDlg, L"intl.cpl,,/f:\"$winnt$.inf\""); // Should be in System32
|
|
}
|
|
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, IDD_OWNERPAGE);
|
|
return TRUE;
|
|
}
|
|
break;
|
|
|
|
case PSN_WIZNEXT:
|
|
break;
|
|
|
|
case PSN_WIZBACK:
|
|
SetupData->UnattendSetup = FALSE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static PTIMEZONE_ENTRY
|
|
GetLargerTimeZoneEntry(PSETUPDATA SetupData, DWORD Index)
|
|
{
|
|
PTIMEZONE_ENTRY Entry;
|
|
|
|
Entry = SetupData->TimeZoneListHead;
|
|
while (Entry != NULL)
|
|
{
|
|
if (Entry->Index >= Index)
|
|
return Entry;
|
|
|
|
Entry = Entry->Next;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static LONG
|
|
RetrieveTimeZone(
|
|
IN HKEY hZoneKey,
|
|
IN PVOID Context)
|
|
{
|
|
LONG lError;
|
|
PSETUPDATA SetupData = (PSETUPDATA)Context;
|
|
PTIMEZONE_ENTRY Entry;
|
|
PTIMEZONE_ENTRY Current;
|
|
ULONG DescriptionSize;
|
|
ULONG StandardNameSize;
|
|
ULONG DaylightNameSize;
|
|
|
|
Entry = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(TIMEZONE_ENTRY));
|
|
if (Entry == NULL)
|
|
{
|
|
return ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
|
|
DescriptionSize = sizeof(Entry->Description);
|
|
StandardNameSize = sizeof(Entry->StandardName);
|
|
DaylightNameSize = sizeof(Entry->DaylightName);
|
|
|
|
lError = QueryTimeZoneData(hZoneKey,
|
|
&Entry->Index,
|
|
&Entry->TimezoneInfo,
|
|
Entry->Description,
|
|
&DescriptionSize,
|
|
Entry->StandardName,
|
|
&StandardNameSize,
|
|
Entry->DaylightName,
|
|
&DaylightNameSize);
|
|
if (lError != ERROR_SUCCESS)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, Entry);
|
|
return lError;
|
|
}
|
|
|
|
if (SetupData->TimeZoneListHead == NULL &&
|
|
SetupData->TimeZoneListTail == NULL)
|
|
{
|
|
Entry->Prev = NULL;
|
|
Entry->Next = NULL;
|
|
SetupData->TimeZoneListHead = Entry;
|
|
SetupData->TimeZoneListTail = Entry;
|
|
}
|
|
else
|
|
{
|
|
Current = GetLargerTimeZoneEntry(SetupData, Entry->Index);
|
|
if (Current != NULL)
|
|
{
|
|
if (Current == SetupData->TimeZoneListHead)
|
|
{
|
|
/* Prepend to head */
|
|
Entry->Prev = NULL;
|
|
Entry->Next = SetupData->TimeZoneListHead;
|
|
SetupData->TimeZoneListHead->Prev = Entry;
|
|
SetupData->TimeZoneListHead = Entry;
|
|
}
|
|
else
|
|
{
|
|
/* Insert before current */
|
|
Entry->Prev = Current->Prev;
|
|
Entry->Next = Current;
|
|
Current->Prev->Next = Entry;
|
|
Current->Prev = Entry;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Append to tail */
|
|
Entry->Prev = SetupData->TimeZoneListTail;
|
|
Entry->Next = NULL;
|
|
SetupData->TimeZoneListTail->Next = Entry;
|
|
SetupData->TimeZoneListTail = Entry;
|
|
}
|
|
}
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
static VOID
|
|
CreateTimeZoneList(PSETUPDATA SetupData)
|
|
{
|
|
EnumerateTimeZoneList(RetrieveTimeZone, SetupData);
|
|
}
|
|
|
|
static VOID
|
|
DestroyTimeZoneList(PSETUPDATA SetupData)
|
|
{
|
|
PTIMEZONE_ENTRY Entry;
|
|
|
|
while (SetupData->TimeZoneListHead != NULL)
|
|
{
|
|
Entry = SetupData->TimeZoneListHead;
|
|
|
|
SetupData->TimeZoneListHead = Entry->Next;
|
|
if (SetupData->TimeZoneListHead != NULL)
|
|
{
|
|
SetupData->TimeZoneListHead->Prev = NULL;
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, Entry);
|
|
}
|
|
|
|
SetupData->TimeZoneListTail = NULL;
|
|
}
|
|
|
|
|
|
static VOID
|
|
ShowTimeZoneList(HWND hwnd, PSETUPDATA SetupData, DWORD dwEntryIndex)
|
|
{
|
|
PTIMEZONE_ENTRY Entry;
|
|
DWORD dwIndex = 0;
|
|
DWORD dwCount;
|
|
|
|
GetTimeZoneListIndex(&dwEntryIndex);
|
|
|
|
Entry = SetupData->TimeZoneListHead;
|
|
while (Entry != NULL)
|
|
{
|
|
dwCount = SendMessage(hwnd,
|
|
CB_ADDSTRING,
|
|
0,
|
|
(LPARAM)Entry->Description);
|
|
|
|
if (dwEntryIndex != 0 && dwEntryIndex == Entry->Index)
|
|
dwIndex = dwCount;
|
|
|
|
Entry = Entry->Next;
|
|
}
|
|
|
|
SendMessage(hwnd,
|
|
CB_SETCURSEL,
|
|
(WPARAM)dwIndex,
|
|
0);
|
|
}
|
|
|
|
|
|
static VOID
|
|
SetLocalTimeZone(HWND hwnd, PSETUPDATA SetupData)
|
|
{
|
|
TIME_ZONE_INFORMATION TimeZoneInformation;
|
|
PTIMEZONE_ENTRY Entry;
|
|
DWORD dwIndex;
|
|
DWORD i;
|
|
|
|
dwIndex = SendMessage(hwnd,
|
|
CB_GETCURSEL,
|
|
0,
|
|
0);
|
|
|
|
i = 0;
|
|
Entry = SetupData->TimeZoneListHead;
|
|
while (i < dwIndex)
|
|
{
|
|
if (Entry == NULL)
|
|
return;
|
|
|
|
i++;
|
|
Entry = Entry->Next;
|
|
}
|
|
|
|
wcscpy(TimeZoneInformation.StandardName,
|
|
Entry->StandardName);
|
|
wcscpy(TimeZoneInformation.DaylightName,
|
|
Entry->DaylightName);
|
|
|
|
TimeZoneInformation.Bias = Entry->TimezoneInfo.Bias;
|
|
TimeZoneInformation.StandardBias = Entry->TimezoneInfo.StandardBias;
|
|
TimeZoneInformation.DaylightBias = Entry->TimezoneInfo.DaylightBias;
|
|
|
|
memcpy(&TimeZoneInformation.StandardDate,
|
|
&Entry->TimezoneInfo.StandardDate,
|
|
sizeof(SYSTEMTIME));
|
|
memcpy(&TimeZoneInformation.DaylightDate,
|
|
&Entry->TimezoneInfo.DaylightDate,
|
|
sizeof(SYSTEMTIME));
|
|
|
|
/* Set time zone information */
|
|
SetTimeZoneInformation(&TimeZoneInformation);
|
|
}
|
|
|
|
|
|
static BOOL
|
|
GetLocalSystemTime(HWND hwnd, PSETUPDATA SetupData)
|
|
{
|
|
SYSTEMTIME Date;
|
|
SYSTEMTIME Time;
|
|
|
|
if (DateTime_GetSystemtime(GetDlgItem(hwnd, IDC_DATEPICKER), &Date) != GDT_VALID)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if (DateTime_GetSystemtime(GetDlgItem(hwnd, IDC_TIMEPICKER), &Time) != GDT_VALID)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
SetupData->SystemTime.wYear = Date.wYear;
|
|
SetupData->SystemTime.wMonth = Date.wMonth;
|
|
SetupData->SystemTime.wDayOfWeek = Date.wDayOfWeek;
|
|
SetupData->SystemTime.wDay = Date.wDay;
|
|
SetupData->SystemTime.wHour = Time.wHour;
|
|
SetupData->SystemTime.wMinute = Time.wMinute;
|
|
SetupData->SystemTime.wSecond = Time.wSecond;
|
|
SetupData->SystemTime.wMilliseconds = Time.wMilliseconds;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static BOOL
|
|
SetSystemLocalTime(HWND hwnd, PSETUPDATA SetupData)
|
|
{
|
|
BOOL Ret = FALSE;
|
|
|
|
/*
|
|
* Call SetLocalTime twice to ensure correct results
|
|
*/
|
|
Ret = SetLocalTime(&SetupData->SystemTime) &&
|
|
SetLocalTime(&SetupData->SystemTime);
|
|
|
|
return Ret;
|
|
}
|
|
|
|
|
|
static VOID
|
|
UpdateLocalSystemTime(HWND hwnd, SYSTEMTIME LocalTime)
|
|
{
|
|
DateTime_SetSystemtime(GetDlgItem(hwnd, IDC_DATEPICKER), GDT_VALID, &LocalTime);
|
|
DateTime_SetSystemtime(GetDlgItem(hwnd, IDC_TIMEPICKER), GDT_VALID, &LocalTime);
|
|
}
|
|
|
|
|
|
static BOOL
|
|
WriteDateTimeSettings(HWND hwndDlg, PSETUPDATA SetupData)
|
|
{
|
|
WCHAR Title[64];
|
|
WCHAR ErrorLocalTime[256];
|
|
|
|
GetLocalSystemTime(hwndDlg, SetupData);
|
|
SetLocalTimeZone(GetDlgItem(hwndDlg, IDC_TIMEZONELIST),
|
|
SetupData);
|
|
|
|
SetAutoDaylight(SendDlgItemMessage(hwndDlg, IDC_AUTODAYLIGHT,
|
|
BM_GETCHECK, 0, 0) != BST_UNCHECKED);
|
|
if (!SetSystemLocalTime(hwndDlg, SetupData))
|
|
{
|
|
if (0 == LoadStringW(hDllInstance, IDS_REACTOS_SETUP, Title, ARRAYSIZE(Title)))
|
|
{
|
|
wcscpy(Title, L"ReactOS Setup");
|
|
}
|
|
if (0 == LoadStringW(hDllInstance, IDS_WZD_LOCALTIME, ErrorLocalTime,
|
|
ARRAYSIZE(ErrorLocalTime)))
|
|
{
|
|
wcscpy(ErrorLocalTime, L"Setup was unable to set the local time.");
|
|
}
|
|
MessageBoxW(hwndDlg, ErrorLocalTime, Title, MB_ICONWARNING | MB_OK);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static INT_PTR CALLBACK
|
|
DateTimePageDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
PSETUPDATA SetupData;
|
|
|
|
/* Retrieve pointer to the global setup data */
|
|
SetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
/* Save pointer to the global setup data */
|
|
SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (DWORD_PTR)SetupData);
|
|
|
|
CreateTimeZoneList(SetupData);
|
|
|
|
if (SetupData->UnattendSetup)
|
|
{
|
|
ShowTimeZoneList(GetDlgItem(hwndDlg, IDC_TIMEZONELIST),
|
|
SetupData, SetupData->TimeZoneIndex);
|
|
|
|
if (!SetupData->DisableAutoDaylightTimeSet)
|
|
{
|
|
SendDlgItemMessage(hwndDlg, IDC_AUTODAYLIGHT, BM_SETCHECK, (WPARAM)BST_CHECKED, 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ShowTimeZoneList(GetDlgItem(hwndDlg, IDC_TIMEZONELIST),
|
|
SetupData, -1);
|
|
|
|
SendDlgItemMessage(hwndDlg, IDC_AUTODAYLIGHT, BM_SETCHECK, (WPARAM)BST_CHECKED, 0);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case WM_TIMER:
|
|
{
|
|
SYSTEMTIME LocalTime;
|
|
|
|
GetLocalTime(&LocalTime);
|
|
UpdateLocalSystemTime(hwndDlg, LocalTime);
|
|
|
|
// Reset timeout.
|
|
SetTimer(hwndDlg, 1, 1000 - LocalTime.wMilliseconds, NULL);
|
|
break;
|
|
}
|
|
|
|
case WM_NOTIFY:
|
|
switch (((LPNMHDR)lParam)->code)
|
|
{
|
|
case PSN_SETACTIVE:
|
|
{
|
|
SYSTEMTIME LocalTime;
|
|
|
|
GetLocalTime(&LocalTime);
|
|
UpdateLocalSystemTime(hwndDlg, LocalTime);
|
|
|
|
/* Enable the Back and Next buttons */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
|
|
|
|
if (SetupData->UnattendSetup && WriteDateTimeSettings(hwndDlg, SetupData))
|
|
{
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, SetupData->uFirstNetworkWizardPage);
|
|
return TRUE;
|
|
}
|
|
|
|
SetTimer(hwndDlg, 1, 1000 - LocalTime.wMilliseconds, NULL);
|
|
break;
|
|
}
|
|
|
|
case PSN_KILLACTIVE:
|
|
case DTN_DATETIMECHANGE:
|
|
// NB: Not re-set until changing page (PSN_SETACTIVE).
|
|
KillTimer(hwndDlg, 1);
|
|
break;
|
|
|
|
case PSN_WIZNEXT:
|
|
WriteDateTimeSettings(hwndDlg, SetupData);
|
|
break;
|
|
|
|
case PSN_WIZBACK:
|
|
SetupData->UnattendSetup = FALSE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case WM_DESTROY:
|
|
DestroyTimeZoneList(SetupData);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static struct ThemeInfo
|
|
{
|
|
LPCWSTR PreviewBitmap;
|
|
UINT DisplayName;
|
|
LPCWSTR ThemeFile;
|
|
|
|
} Themes[] = {
|
|
{ MAKEINTRESOURCE(IDB_CLASSIC), IDS_CLASSIC, NULL },
|
|
{ MAKEINTRESOURCE(IDB_LAUTUS), IDS_LAUTUS, L"themes\\lautus\\lautus.msstyles" },
|
|
{ MAKEINTRESOURCE(IDB_LUNAR), IDS_LUNAR, L"themes\\lunar\\lunar.msstyles" },
|
|
{ MAKEINTRESOURCE(IDB_MIZU), IDS_MIZU, L"themes\\mizu\\mizu.msstyles"},
|
|
};
|
|
|
|
static INT_PTR CALLBACK
|
|
ThemePageDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
PSETUPDATA SetupData;
|
|
LPNMLISTVIEW pnmv;
|
|
|
|
/* Retrieve pointer to the global setup data */
|
|
SetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
HWND hListView;
|
|
HIMAGELIST himl;
|
|
DWORD n;
|
|
LVITEM lvi = {0};
|
|
|
|
/* Save pointer to the global setup data */
|
|
SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (DWORD_PTR)SetupData);
|
|
|
|
hListView = GetDlgItem(hwndDlg, IDC_THEMEPICKER);
|
|
|
|
/* Common */
|
|
himl = ImageList_Create(180, 163, ILC_COLOR32 | ILC_MASK, ARRAYSIZE(Themes), 1);
|
|
lvi.mask = LVIF_TEXT | LVIF_IMAGE |LVIF_STATE;
|
|
|
|
for (n = 0; n < ARRAYSIZE(Themes); ++n)
|
|
{
|
|
WCHAR DisplayName[100] = {0};
|
|
/* Load the bitmap */
|
|
HANDLE image = LoadImageW(hDllInstance, Themes[n].PreviewBitmap, IMAGE_BITMAP, 0, 0, LR_CREATEDIBSECTION);
|
|
ImageList_AddMasked(himl, image, RGB(255,0,255));
|
|
|
|
/* Load the string */
|
|
LoadStringW(hDllInstance, Themes[n].DisplayName, DisplayName, ARRAYSIZE(DisplayName));
|
|
DisplayName[ARRAYSIZE(DisplayName)-1] = UNICODE_NULL;
|
|
|
|
/* Add the listview item */
|
|
lvi.iItem = n;
|
|
lvi.iImage = n;
|
|
lvi.pszText = DisplayName;
|
|
ListView_InsertItem(hListView, &lvi);
|
|
}
|
|
|
|
/* Register the imagelist */
|
|
ListView_SetImageList(hListView, himl, LVSIL_NORMAL);
|
|
/* Transparant background */
|
|
ListView_SetBkColor(hListView, CLR_NONE);
|
|
ListView_SetTextBkColor(hListView, CLR_NONE);
|
|
/* Reduce the size between the items */
|
|
ListView_SetIconSpacing(hListView, 190, 173);
|
|
break;
|
|
}
|
|
case WM_NOTIFY:
|
|
switch (((LPNMHDR)lParam)->code)
|
|
{
|
|
//case LVN_ITEMCHANGING:
|
|
case LVN_ITEMCHANGED:
|
|
pnmv = (LPNMLISTVIEW)lParam;
|
|
if ((pnmv->uChanged & LVIF_STATE) && (pnmv->uNewState & LVIS_SELECTED))
|
|
{
|
|
int iTheme = pnmv->iItem;
|
|
DPRINT1("Selected theme: %u\n", Themes[iTheme].DisplayName);
|
|
|
|
if (Themes[iTheme].ThemeFile)
|
|
{
|
|
WCHAR wszParams[1024];
|
|
WCHAR wszTheme[MAX_PATH];
|
|
WCHAR* format = L"desk.cpl,,2 /Action:ActivateMSTheme /file:\"%s\"";
|
|
|
|
SHGetFolderPathAndSubDirW(0, CSIDL_RESOURCES, NULL, SHGFP_TYPE_DEFAULT, Themes[iTheme].ThemeFile, wszTheme);
|
|
swprintf(wszParams, format, wszTheme);
|
|
RunControlPanelApplet(hwndDlg, wszParams);
|
|
}
|
|
else
|
|
{
|
|
RunControlPanelApplet(hwndDlg, L"desk.cpl,,2 /Action:ActivateMSTheme");
|
|
}
|
|
}
|
|
break;
|
|
case PSN_SETACTIVE:
|
|
/* Enable the Back and Next buttons */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_NEXT);
|
|
if (SetupData->UnattendSetup)
|
|
{
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, SetupData->uFirstNetworkWizardPage);
|
|
return TRUE;
|
|
}
|
|
break;
|
|
|
|
case PSN_WIZNEXT:
|
|
break;
|
|
|
|
case PSN_WIZBACK:
|
|
SetupData->UnattendSetup = FALSE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static UINT CALLBACK
|
|
RegistrationNotificationProc(PVOID Context,
|
|
UINT Notification,
|
|
UINT_PTR Param1,
|
|
UINT_PTR Param2)
|
|
{
|
|
PREGISTRATIONDATA RegistrationData;
|
|
REGISTRATIONNOTIFY RegistrationNotify;
|
|
PSP_REGISTER_CONTROL_STATUSW StatusInfo;
|
|
UINT MessageID;
|
|
|
|
RegistrationData = (PREGISTRATIONDATA)Context;
|
|
|
|
if (Notification == SPFILENOTIFY_STARTREGISTRATION ||
|
|
Notification == SPFILENOTIFY_ENDREGISTRATION)
|
|
{
|
|
StatusInfo = (PSP_REGISTER_CONTROL_STATUSW) Param1;
|
|
RegistrationNotify.CurrentItem = wcsrchr(StatusInfo->FileName, L'\\');
|
|
if (RegistrationNotify.CurrentItem == NULL)
|
|
{
|
|
RegistrationNotify.CurrentItem = StatusInfo->FileName;
|
|
}
|
|
else
|
|
{
|
|
RegistrationNotify.CurrentItem++;
|
|
}
|
|
|
|
if (Notification == SPFILENOTIFY_STARTREGISTRATION)
|
|
{
|
|
DPRINT("Received SPFILENOTIFY_STARTREGISTRATION notification for %S\n",
|
|
StatusInfo->FileName);
|
|
RegistrationNotify.ErrorMessage = NULL;
|
|
RegistrationNotify.Progress = RegistrationData->Registered;
|
|
SendMessage(RegistrationData->hwndDlg, PM_STEP_START, 0, (LPARAM)&RegistrationNotify);
|
|
}
|
|
else
|
|
{
|
|
DPRINT("Received SPFILENOTIFY_ENDREGISTRATION notification for %S\n",
|
|
StatusInfo->FileName);
|
|
DPRINT("Win32Error %u FailureCode %u\n", StatusInfo->Win32Error,
|
|
StatusInfo->FailureCode);
|
|
if (StatusInfo->FailureCode != SPREG_SUCCESS)
|
|
{
|
|
switch (StatusInfo->FailureCode)
|
|
{
|
|
case SPREG_LOADLIBRARY:
|
|
MessageID = IDS_LOADLIBRARY_FAILED;
|
|
break;
|
|
case SPREG_GETPROCADDR:
|
|
MessageID = IDS_GETPROCADDR_FAILED;
|
|
break;
|
|
case SPREG_REGSVR:
|
|
MessageID = IDS_REGSVR_FAILED;
|
|
break;
|
|
case SPREG_DLLINSTALL:
|
|
MessageID = IDS_DLLINSTALL_FAILED;
|
|
break;
|
|
case SPREG_TIMEOUT:
|
|
MessageID = IDS_TIMEOUT;
|
|
break;
|
|
default:
|
|
MessageID = IDS_REASON_UNKNOWN;
|
|
break;
|
|
}
|
|
|
|
RegistrationNotify.MessageID = MessageID;
|
|
RegistrationNotify.LastError = StatusInfo->Win32Error;
|
|
}
|
|
else
|
|
{
|
|
RegistrationNotify.MessageID = 0;
|
|
RegistrationNotify.LastError = ERROR_SUCCESS;
|
|
}
|
|
|
|
if (RegistrationData->Registered < RegistrationData->DllCount)
|
|
{
|
|
RegistrationData->Registered++;
|
|
}
|
|
|
|
RegistrationNotify.Progress = RegistrationData->Registered;
|
|
SendMessage(RegistrationData->hwndDlg, PM_STEP_END, 0, (LPARAM)&RegistrationNotify);
|
|
}
|
|
|
|
return FILEOP_DOIT;
|
|
}
|
|
else
|
|
{
|
|
DPRINT1("Received unexpected notification %u\n", Notification);
|
|
return SetupDefaultQueueCallback(RegistrationData->DefaultContext,
|
|
Notification, Param1, Param2);
|
|
}
|
|
}
|
|
|
|
|
|
static
|
|
DWORD
|
|
RegisterDlls(
|
|
PITEMSDATA pItemsData)
|
|
{
|
|
REGISTRATIONDATA RegistrationData;
|
|
WCHAR SectionName[512];
|
|
INFCONTEXT Context;
|
|
LONG DllCount = 0;
|
|
DWORD LastError = NO_ERROR;
|
|
|
|
ZeroMemory(&RegistrationData, sizeof(REGISTRATIONDATA));
|
|
RegistrationData.hwndDlg = pItemsData->hwndDlg;
|
|
RegistrationData.Registered = 0;
|
|
|
|
if (!SetupFindFirstLineW(hSysSetupInf, L"RegistrationPhase2",
|
|
L"RegisterDlls", &Context))
|
|
{
|
|
DPRINT1("No RegistrationPhase2 section found\n");
|
|
return FALSE;
|
|
}
|
|
|
|
if (!SetupGetStringFieldW(&Context, 1, SectionName,
|
|
ARRAYSIZE(SectionName),
|
|
NULL))
|
|
{
|
|
DPRINT1("Unable to retrieve section name\n");
|
|
return FALSE;
|
|
}
|
|
|
|
DllCount = SetupGetLineCountW(hSysSetupInf, SectionName);
|
|
DPRINT1("SectionName %S DllCount %ld\n", SectionName, DllCount);
|
|
if (DllCount < 0)
|
|
{
|
|
SetLastError(STATUS_NOT_FOUND);
|
|
return FALSE;
|
|
}
|
|
|
|
RegistrationData.DllCount = (ULONG)DllCount;
|
|
RegistrationData.DefaultContext = SetupInitDefaultQueueCallback(RegistrationData.hwndDlg);
|
|
|
|
SendMessage(pItemsData->hwndDlg, PM_ITEM_START, 0, (LPARAM)RegistrationData.DllCount);
|
|
|
|
_SEH2_TRY
|
|
{
|
|
if (!SetupInstallFromInfSectionW(GetParent(RegistrationData.hwndDlg),
|
|
hSysSetupInf,
|
|
L"RegistrationPhase2",
|
|
SPINST_REGISTRY | SPINST_REGISTERCALLBACKAWARE | SPINST_REGSVR,
|
|
0,
|
|
NULL,
|
|
0,
|
|
RegistrationNotificationProc,
|
|
&RegistrationData,
|
|
NULL,
|
|
NULL))
|
|
{
|
|
LastError = GetLastError();
|
|
}
|
|
}
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
DPRINT("Catching exception\n");
|
|
LastError = RtlNtStatusToDosError(_SEH2_GetExceptionCode());
|
|
}
|
|
_SEH2_END;
|
|
|
|
SetupTermDefaultQueueCallback(RegistrationData.DefaultContext);
|
|
|
|
SendMessage(pItemsData->hwndDlg, PM_ITEM_END, 0, LastError);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static
|
|
DWORD
|
|
CALLBACK
|
|
ItemCompletionThread(
|
|
LPVOID Parameter)
|
|
{
|
|
PITEMSDATA pItemsData;
|
|
HWND hwndDlg;
|
|
|
|
pItemsData = (PITEMSDATA)Parameter;
|
|
hwndDlg = pItemsData->hwndDlg;
|
|
|
|
RegisterDlls(pItemsData);
|
|
|
|
RegisterTypeLibraries(hSysSetupInf, L"TypeLibraries");
|
|
|
|
/* FIXME: Add completion steps here! */
|
|
|
|
// FIXME: Move this call to a separate cleanup page!
|
|
RtlCreateBootStatusDataFile();
|
|
|
|
/* Free the items data */
|
|
HeapFree(GetProcessHeap(), 0, pItemsData);
|
|
|
|
/* Tell the wizard page that we are done */
|
|
PostMessage(hwndDlg, PM_ITEMS_DONE, 0, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static
|
|
BOOL
|
|
RunItemCompletionThread(
|
|
_In_ HWND hwndDlg)
|
|
{
|
|
HANDLE hCompletionThread;
|
|
PITEMSDATA pItemsData;
|
|
|
|
pItemsData = HeapAlloc(GetProcessHeap(), 0, sizeof(ITEMSDATA));
|
|
if (pItemsData == NULL)
|
|
return FALSE;
|
|
|
|
pItemsData->hwndDlg = hwndDlg;
|
|
|
|
hCompletionThread = CreateThread(NULL,
|
|
0,
|
|
ItemCompletionThread,
|
|
pItemsData,
|
|
0,
|
|
NULL);
|
|
if (hCompletionThread == NULL)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, pItemsData);
|
|
}
|
|
else
|
|
{
|
|
CloseHandle(hCompletionThread);
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static
|
|
VOID
|
|
ShowItemError(
|
|
HWND hwndDlg,
|
|
DWORD LastError)
|
|
{
|
|
LPWSTR ErrorMessage = NULL;
|
|
WCHAR UnknownError[84];
|
|
WCHAR Title[64];
|
|
|
|
if (FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
|
|
NULL, LastError, 0, ErrorMessage, 0, NULL) == 0)
|
|
{
|
|
if (LoadStringW(hDllInstance, IDS_UNKNOWN_ERROR,
|
|
UnknownError,
|
|
ARRAYSIZE(UnknownError) - 20) == 0)
|
|
{
|
|
wcscpy(UnknownError, L"Unknown error");
|
|
}
|
|
wcscat(UnknownError, L" ");
|
|
_ultow(LastError, UnknownError + wcslen(UnknownError), 10);
|
|
ErrorMessage = UnknownError;
|
|
}
|
|
|
|
if (ErrorMessage != NULL)
|
|
{
|
|
if (LoadStringW(hDllInstance, IDS_REACTOS_SETUP,
|
|
Title, ARRAYSIZE(Title)) == 0)
|
|
{
|
|
wcscpy(Title, L"ReactOS Setup");
|
|
}
|
|
|
|
MessageBoxW(hwndDlg, ErrorMessage, Title, MB_ICONERROR | MB_OK);
|
|
}
|
|
|
|
if (ErrorMessage != NULL &&
|
|
ErrorMessage != UnknownError)
|
|
{
|
|
LocalFree(ErrorMessage);
|
|
}
|
|
}
|
|
|
|
|
|
static
|
|
VOID
|
|
ShowStepError(
|
|
HWND hwndDlg,
|
|
PREGISTRATIONNOTIFY RegistrationNotify)
|
|
{
|
|
WCHAR ErrorMessage[128];
|
|
WCHAR Title[64];
|
|
|
|
if (LoadStringW(hDllInstance, RegistrationNotify->MessageID,
|
|
ErrorMessage,
|
|
ARRAYSIZE(ErrorMessage)) == 0)
|
|
{
|
|
ErrorMessage[0] = L'\0';
|
|
}
|
|
|
|
if (RegistrationNotify->MessageID != IDS_TIMEOUT)
|
|
{
|
|
FormatMessageW(FORMAT_MESSAGE_FROM_SYSTEM, NULL,
|
|
RegistrationNotify->LastError, 0,
|
|
ErrorMessage + wcslen(ErrorMessage),
|
|
ARRAYSIZE(ErrorMessage) - wcslen(ErrorMessage),
|
|
NULL);
|
|
}
|
|
|
|
if (ErrorMessage[0] != L'\0')
|
|
{
|
|
if (LoadStringW(hDllInstance, IDS_REACTOS_SETUP,
|
|
Title, ARRAYSIZE(Title)) == 0)
|
|
{
|
|
wcscpy(Title, L"ReactOS Setup");
|
|
}
|
|
|
|
MessageBoxW(hwndDlg, ErrorMessage,
|
|
Title, MB_ICONERROR | MB_OK);
|
|
}
|
|
}
|
|
|
|
|
|
static INT_PTR CALLBACK
|
|
ProcessPageDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
PSETUPDATA SetupData;
|
|
PREGISTRATIONNOTIFY RegistrationNotify;
|
|
|
|
/* Retrieve pointer to the global setup data */
|
|
SetupData = (PSETUPDATA)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
/* Save pointer to the global setup data */
|
|
SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (DWORD_PTR)SetupData);
|
|
ShowWindow(GetDlgItem(hwndDlg, IDC_TASKTEXT2), SW_HIDE);
|
|
ShowWindow(GetDlgItem(hwndDlg, IDC_TASKTEXT3), SW_HIDE);
|
|
ShowWindow(GetDlgItem(hwndDlg, IDC_TASKTEXT4), SW_HIDE);
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
switch (((LPNMHDR)lParam)->code)
|
|
{
|
|
case PSN_SETACTIVE:
|
|
/* Disable the Back and Next buttons */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), 0);
|
|
RunItemCompletionThread(hwndDlg);
|
|
break;
|
|
|
|
case PSN_WIZNEXT:
|
|
break;
|
|
|
|
case PSN_WIZBACK:
|
|
SetupData->UnattendSetup = FALSE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case PM_ITEM_START:
|
|
DPRINT("PM_ITEM_START %lu\n", (ULONG)lParam);
|
|
SendDlgItemMessage(hwndDlg, IDC_PROCESSPROGRESS, PBM_SETRANGE, 0, MAKELPARAM(0, (ULONG)lParam));
|
|
SendDlgItemMessage(hwndDlg, IDC_PROCESSPROGRESS, PBM_SETPOS, 0, 0);
|
|
SendDlgItemMessage(hwndDlg, IDC_TASKTEXT1 + wParam, WM_SETFONT, (WPARAM)SetupData->hBoldFont, (LPARAM)TRUE);
|
|
break;
|
|
|
|
case PM_ITEM_END:
|
|
DPRINT("PM_ITEM_END\n");
|
|
if (lParam == ERROR_SUCCESS)
|
|
{
|
|
}
|
|
else
|
|
{
|
|
ShowItemError(hwndDlg, (DWORD)lParam);
|
|
}
|
|
break;
|
|
|
|
case PM_STEP_START:
|
|
DPRINT("PM_STEP_START\n");
|
|
RegistrationNotify = (PREGISTRATIONNOTIFY)lParam;
|
|
SendDlgItemMessage(hwndDlg, IDC_ITEM, WM_SETTEXT, 0,
|
|
(LPARAM)((RegistrationNotify->CurrentItem != NULL)? RegistrationNotify->CurrentItem : L""));
|
|
break;
|
|
|
|
case PM_STEP_END:
|
|
DPRINT("PM_STEP_END\n");
|
|
RegistrationNotify = (PREGISTRATIONNOTIFY)lParam;
|
|
SendDlgItemMessage(hwndDlg, IDC_PROCESSPROGRESS, PBM_SETPOS, RegistrationNotify->Progress, 0);
|
|
if (RegistrationNotify->LastError != ERROR_SUCCESS)
|
|
{
|
|
ShowStepError(hwndDlg, RegistrationNotify);
|
|
}
|
|
break;
|
|
|
|
case PM_ITEMS_DONE:
|
|
DPRINT("PM_ITEMS_DONE\n");
|
|
/* Enable the Back and Next buttons */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_NEXT);
|
|
PropSheet_PressButton(GetParent(hwndDlg), PSBTN_NEXT);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static VOID
|
|
SetInstallationCompleted(VOID)
|
|
{
|
|
HKEY hKey = 0;
|
|
DWORD InProgress = 0;
|
|
DWORD InstallDate;
|
|
|
|
if (RegOpenKeyExW( HKEY_LOCAL_MACHINE,
|
|
L"SYSTEM\\Setup",
|
|
0,
|
|
KEY_WRITE,
|
|
&hKey ) == ERROR_SUCCESS)
|
|
{
|
|
RegSetValueExW( hKey, L"SystemSetupInProgress", 0, REG_DWORD, (LPBYTE)&InProgress, sizeof(InProgress) );
|
|
RegCloseKey( hKey );
|
|
}
|
|
|
|
if (RegOpenKeyExW( HKEY_LOCAL_MACHINE,
|
|
L"Software\\Microsoft\\Windows NT\\CurrentVersion",
|
|
0,
|
|
KEY_WRITE,
|
|
&hKey ) == ERROR_SUCCESS)
|
|
{
|
|
InstallDate = (DWORD)time(NULL);
|
|
RegSetValueExW( hKey, L"InstallDate", 0, REG_DWORD, (LPBYTE)&InstallDate, sizeof(InstallDate) );
|
|
RegCloseKey( hKey );
|
|
}
|
|
}
|
|
|
|
static INT_PTR CALLBACK
|
|
FinishDlgProc(HWND hwndDlg,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
/* Get pointer to the global setup data */
|
|
PSETUPDATA SetupData = (PSETUPDATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
|
|
if (!SetupData->UnattendSetup || !SetupData->DisableGeckoInst)
|
|
{
|
|
/* Run the Wine Gecko prompt */
|
|
Control_RunDLLW(hwndDlg, 0, L"appwiz.cpl,,install_gecko", SW_SHOW);
|
|
}
|
|
|
|
/* Set title font */
|
|
SendDlgItemMessage(hwndDlg,
|
|
IDC_FINISHTITLE,
|
|
WM_SETFONT,
|
|
(WPARAM)SetupData->hTitleFont,
|
|
(LPARAM)TRUE);
|
|
if (SetupData->UnattendSetup)
|
|
{
|
|
KillTimer(hwndDlg, 1);
|
|
SetInstallationCompleted();
|
|
PostQuitMessage(0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_DESTROY:
|
|
{
|
|
SetInstallationCompleted();
|
|
PostQuitMessage(0);
|
|
return TRUE;
|
|
}
|
|
|
|
case WM_TIMER:
|
|
{
|
|
INT Position;
|
|
HWND hWndProgress;
|
|
|
|
hWndProgress = GetDlgItem(hwndDlg, IDC_RESTART_PROGRESS);
|
|
Position = SendMessage(hWndProgress, PBM_GETPOS, 0, 0);
|
|
if (Position == 300)
|
|
{
|
|
KillTimer(hwndDlg, 1);
|
|
PropSheet_PressButton(GetParent(hwndDlg), PSBTN_FINISH);
|
|
}
|
|
else
|
|
{
|
|
SendMessage(hWndProgress, PBM_SETPOS, Position + 1, 0);
|
|
}
|
|
}
|
|
return TRUE;
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
LPNMHDR lpnm = (LPNMHDR)lParam;
|
|
|
|
switch (lpnm->code)
|
|
{
|
|
case PSN_SETACTIVE:
|
|
/* Enable the correct buttons on for the active page */
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_FINISH);
|
|
|
|
SendDlgItemMessage(hwndDlg, IDC_RESTART_PROGRESS, PBM_SETRANGE, 0,
|
|
MAKELPARAM(0, 300));
|
|
SendDlgItemMessage(hwndDlg, IDC_RESTART_PROGRESS, PBM_SETPOS, 0, 0);
|
|
SetTimer(hwndDlg, 1, 50, NULL);
|
|
break;
|
|
|
|
case PSN_WIZFINISH:
|
|
DestroyWindow(GetParent(hwndDlg));
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
* GetInstallSourceWin32 retrieves the path to the ReactOS installation medium
|
|
* in Win32 format, for later use by syssetup and storage in the registry.
|
|
*/
|
|
static BOOL
|
|
GetInstallSourceWin32(
|
|
OUT PWSTR pwszPath,
|
|
IN DWORD cchPathMax,
|
|
IN PCWSTR pwszNTPath)
|
|
{
|
|
WCHAR wszDrives[512];
|
|
WCHAR wszNTPath[512]; // MAX_PATH ?
|
|
DWORD cchDrives;
|
|
PWCHAR pwszDrive;
|
|
|
|
*pwszPath = UNICODE_NULL;
|
|
|
|
cchDrives = GetLogicalDriveStringsW(_countof(wszDrives) - 1, wszDrives);
|
|
if (cchDrives == 0 || cchDrives >= _countof(wszDrives))
|
|
{
|
|
/* Buffer too small or failure */
|
|
LogItem(NULL, L"GetLogicalDriveStringsW failed");
|
|
return FALSE;
|
|
}
|
|
|
|
for (pwszDrive = wszDrives; *pwszDrive; pwszDrive += wcslen(pwszDrive) + 1)
|
|
{
|
|
WCHAR wszBuf[MAX_PATH];
|
|
|
|
/* Retrieve the NT path corresponding to the current Win32 DOS path */
|
|
pwszDrive[2] = UNICODE_NULL; // Temporarily remove the backslash
|
|
QueryDosDeviceW(pwszDrive, wszNTPath, _countof(wszNTPath));
|
|
pwszDrive[2] = L'\\'; // Restore the backslash
|
|
|
|
wcscat(wszNTPath, L"\\"); // Concat a backslash
|
|
|
|
/* Logging */
|
|
wsprintf(wszBuf, L"Testing '%s' --> '%s' %s a CD",
|
|
pwszDrive, wszNTPath,
|
|
(GetDriveTypeW(pwszDrive) == DRIVE_CDROM) ? L"is" : L"is not");
|
|
LogItem(NULL, wszBuf);
|
|
|
|
/* Check whether the NT path corresponds to the NT installation source path */
|
|
if (!_wcsicmp(wszNTPath, pwszNTPath))
|
|
{
|
|
/* Found it! */
|
|
wcscpy(pwszPath, pwszDrive); // cchPathMax
|
|
|
|
/* Logging */
|
|
wsprintf(wszBuf, L"GetInstallSourceWin32: %s", pwszPath);
|
|
LogItem(NULL, wszBuf);
|
|
wcscat(wszBuf, L"\n");
|
|
OutputDebugStringW(wszBuf);
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
VOID
|
|
ProcessUnattendSection(
|
|
IN OUT PSETUPDATA pSetupData)
|
|
{
|
|
INFCONTEXT InfContext;
|
|
WCHAR szName[256];
|
|
WCHAR szValue[MAX_PATH];
|
|
DWORD LineLength;
|
|
HKEY hKey;
|
|
|
|
if (!SetupFindFirstLineW(pSetupData->hSetupInf,
|
|
L"Unattend",
|
|
L"UnattendSetupEnabled",
|
|
&InfContext))
|
|
{
|
|
DPRINT1("Error: Cannot find UnattendSetupEnabled Key! %d\n", GetLastError());
|
|
return;
|
|
}
|
|
|
|
if (!SetupGetStringFieldW(&InfContext,
|
|
1,
|
|
szValue,
|
|
ARRAYSIZE(szValue),
|
|
&LineLength))
|
|
{
|
|
DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
|
|
return;
|
|
}
|
|
|
|
if (_wcsicmp(szValue, L"yes") != 0)
|
|
{
|
|
DPRINT("Unattend setup was disabled by UnattendSetupEnabled key.\n");
|
|
return;
|
|
}
|
|
|
|
pSetupData->UnattendSetup = TRUE;
|
|
|
|
if (!SetupFindFirstLineW(pSetupData->hSetupInf,
|
|
L"Unattend",
|
|
NULL,
|
|
&InfContext))
|
|
{
|
|
DPRINT1("Error: SetupFindFirstLine failed %d\n", GetLastError());
|
|
return;
|
|
}
|
|
|
|
do
|
|
{
|
|
if (!SetupGetStringFieldW(&InfContext,
|
|
0,
|
|
szName,
|
|
ARRAYSIZE(szName),
|
|
&LineLength))
|
|
{
|
|
DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
|
|
return;
|
|
}
|
|
|
|
if (!SetupGetStringFieldW(&InfContext,
|
|
1,
|
|
szValue,
|
|
ARRAYSIZE(szValue),
|
|
&LineLength))
|
|
{
|
|
DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
|
|
return;
|
|
}
|
|
DPRINT1("Name %S Value %S\n", szName, szValue);
|
|
if (!_wcsicmp(szName, L"FullName"))
|
|
{
|
|
if (ARRAYSIZE(pSetupData->OwnerName) > LineLength)
|
|
{
|
|
wcscpy(pSetupData->OwnerName, szValue);
|
|
}
|
|
}
|
|
else if (!_wcsicmp(szName, L"OrgName"))
|
|
{
|
|
if (ARRAYSIZE(pSetupData->OwnerOrganization) > LineLength)
|
|
{
|
|
wcscpy(pSetupData->OwnerOrganization, szValue);
|
|
}
|
|
}
|
|
else if (!_wcsicmp(szName, L"ComputerName"))
|
|
{
|
|
if (ARRAYSIZE(pSetupData->ComputerName) > LineLength)
|
|
{
|
|
wcscpy(pSetupData->ComputerName, szValue);
|
|
}
|
|
}
|
|
else if (!_wcsicmp(szName, L"AdminPassword"))
|
|
{
|
|
if (ARRAYSIZE(pSetupData->AdminPassword) > LineLength)
|
|
{
|
|
wcscpy(pSetupData->AdminPassword, szValue);
|
|
}
|
|
}
|
|
else if (!_wcsicmp(szName, L"TimeZoneIndex"))
|
|
{
|
|
pSetupData->TimeZoneIndex = _wtoi(szValue);
|
|
}
|
|
else if (!_wcsicmp(szName, L"DisableAutoDaylightTimeSet"))
|
|
{
|
|
pSetupData->DisableAutoDaylightTimeSet = _wtoi(szValue);
|
|
}
|
|
else if (!_wcsicmp(szName, L"DisableGeckoInst"))
|
|
{
|
|
if (!_wcsicmp(szValue, L"yes"))
|
|
pSetupData->DisableGeckoInst = TRUE;
|
|
else
|
|
pSetupData->DisableGeckoInst = FALSE;
|
|
}
|
|
else if (!_wcsicmp(szName, L"ProductOption"))
|
|
{
|
|
pSetupData->ProductOption = (PRODUCT_OPTION)_wtoi(szValue);
|
|
}
|
|
} while (SetupFindNextLine(&InfContext, &InfContext));
|
|
|
|
if (SetupFindFirstLineW(pSetupData->hSetupInf,
|
|
L"Display",
|
|
NULL,
|
|
&InfContext))
|
|
{
|
|
DEVMODEW dm = { { 0 } };
|
|
dm.dmSize = sizeof(dm);
|
|
if (EnumDisplaySettingsW(NULL, ENUM_CURRENT_SETTINGS, &dm))
|
|
{
|
|
do
|
|
{
|
|
int iValue;
|
|
if (!SetupGetStringFieldW(&InfContext,
|
|
0,
|
|
szName,
|
|
ARRAYSIZE(szName),
|
|
&LineLength))
|
|
{
|
|
DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
|
|
return;
|
|
}
|
|
|
|
if (!SetupGetStringFieldW(&InfContext,
|
|
1,
|
|
szValue,
|
|
ARRAYSIZE(szValue),
|
|
&LineLength))
|
|
{
|
|
DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
|
|
return;
|
|
}
|
|
iValue = _wtoi(szValue);
|
|
DPRINT1("Name %S Value %i\n", szName, iValue);
|
|
|
|
if (!iValue)
|
|
continue;
|
|
|
|
if (!_wcsicmp(szName, L"BitsPerPel"))
|
|
{
|
|
dm.dmFields |= DM_BITSPERPEL;
|
|
dm.dmBitsPerPel = iValue;
|
|
}
|
|
else if (!_wcsicmp(szName, L"XResolution"))
|
|
{
|
|
dm.dmFields |= DM_PELSWIDTH;
|
|
dm.dmPelsWidth = iValue;
|
|
}
|
|
else if (!_wcsicmp(szName, L"YResolution"))
|
|
{
|
|
dm.dmFields |= DM_PELSHEIGHT;
|
|
dm.dmPelsHeight = iValue;
|
|
}
|
|
else if (!_wcsicmp(szName, L"VRefresh"))
|
|
{
|
|
dm.dmFields |= DM_DISPLAYFREQUENCY;
|
|
dm.dmDisplayFrequency = iValue;
|
|
}
|
|
} while (SetupFindNextLine(&InfContext, &InfContext));
|
|
|
|
ChangeDisplaySettingsW(&dm, CDS_UPDATEREGISTRY);
|
|
}
|
|
}
|
|
|
|
if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\RunOnce",
|
|
0,
|
|
KEY_SET_VALUE,
|
|
&hKey) != ERROR_SUCCESS)
|
|
{
|
|
DPRINT1("Error: failed to open HKLM\\Software\\Microsoft\\Windows\\CurrentVersion\\RunOnce\n");
|
|
return;
|
|
}
|
|
|
|
if (SetupFindFirstLineW(pSetupData->hSetupInf,
|
|
L"GuiRunOnce",
|
|
NULL,
|
|
&InfContext))
|
|
{
|
|
int i = 0;
|
|
do
|
|
{
|
|
if (SetupGetStringFieldW(&InfContext,
|
|
0,
|
|
szValue,
|
|
ARRAYSIZE(szValue),
|
|
NULL))
|
|
{
|
|
WCHAR szPath[MAX_PATH];
|
|
swprintf(szName, L"%d", i);
|
|
DPRINT("szName %S szValue %S\n", szName, szValue);
|
|
|
|
if (ExpandEnvironmentStringsW(szValue, szPath, MAX_PATH))
|
|
{
|
|
DPRINT("value %S\n", szPath);
|
|
if (RegSetValueExW(hKey,
|
|
szName,
|
|
0,
|
|
REG_SZ,
|
|
(const BYTE*)szPath,
|
|
(wcslen(szPath) + 1) * sizeof(WCHAR)) == ERROR_SUCCESS)
|
|
{
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
} while (SetupFindNextLine(&InfContext, &InfContext));
|
|
}
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
if (SetupFindFirstLineW(pSetupData->hSetupInf,
|
|
L"Env",
|
|
NULL,
|
|
&InfContext))
|
|
{
|
|
if (RegCreateKeyExW(
|
|
HKEY_LOCAL_MACHINE, L"SYSTEM\\CurrentControlSet\\Control\\Session Manager\\Environment", 0, NULL,
|
|
REG_OPTION_NON_VOLATILE, KEY_WRITE | KEY_READ, NULL, &hKey, NULL) != ERROR_SUCCESS)
|
|
{
|
|
DPRINT1("Error: failed to open HKLM\\SYSTEM\\CurrentControlSet\\Control\\Session Manager\\Environment\n");
|
|
return;
|
|
}
|
|
do
|
|
{
|
|
if (!SetupGetStringFieldW(&InfContext,
|
|
0,
|
|
szName,
|
|
ARRAYSIZE(szName),
|
|
&LineLength))
|
|
{
|
|
DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
|
|
return;
|
|
}
|
|
|
|
if (!SetupGetStringFieldW(&InfContext,
|
|
1,
|
|
szValue,
|
|
ARRAYSIZE(szValue),
|
|
&LineLength))
|
|
{
|
|
DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
|
|
return;
|
|
}
|
|
DPRINT1("[ENV] %S=%S\n", szName, szValue);
|
|
|
|
DWORD dwType = wcschr(szValue, '%') != NULL ? REG_EXPAND_SZ : REG_SZ;
|
|
|
|
if (RegSetValueExW(hKey, szName, 0, dwType, (const BYTE*)szValue, (DWORD)(wcslen(szValue) + 1) * sizeof(TCHAR)) != ERROR_SUCCESS)
|
|
{
|
|
DPRINT1(" - Error %d\n", GetLastError());
|
|
}
|
|
|
|
} while (SetupFindNextLine(&InfContext, &InfContext));
|
|
|
|
RegCloseKey(hKey);
|
|
}
|
|
}
|
|
|
|
static BOOL
|
|
PathIsEqual(
|
|
IN LPCWSTR lpPath1,
|
|
IN LPCWSTR lpPath2)
|
|
{
|
|
WCHAR szPath1[MAX_PATH];
|
|
WCHAR szPath2[MAX_PATH];
|
|
|
|
/* If something goes wrong, better return TRUE,
|
|
* so the calling function returns early.
|
|
*/
|
|
if (!PathCanonicalizeW(szPath1, lpPath1))
|
|
return TRUE;
|
|
|
|
if (!PathAddBackslashW(szPath1))
|
|
return TRUE;
|
|
|
|
if (!PathCanonicalizeW(szPath2, lpPath2))
|
|
return TRUE;
|
|
|
|
if (!PathAddBackslashW(szPath2))
|
|
return TRUE;
|
|
|
|
return (_wcsicmp(szPath1, szPath2) == 0);
|
|
}
|
|
|
|
static VOID
|
|
AddInstallationSource(
|
|
IN HKEY hKey,
|
|
IN LPWSTR lpPath)
|
|
{
|
|
LONG res;
|
|
DWORD dwRegType;
|
|
DWORD dwPathLength = 0;
|
|
DWORD dwNewLength = 0;
|
|
LPWSTR Buffer = NULL;
|
|
LPWSTR Path;
|
|
|
|
res = RegQueryValueExW(
|
|
hKey,
|
|
L"Installation Sources",
|
|
NULL,
|
|
&dwRegType,
|
|
NULL,
|
|
&dwPathLength);
|
|
|
|
if (res != ERROR_SUCCESS ||
|
|
dwRegType != REG_MULTI_SZ ||
|
|
dwPathLength == 0 ||
|
|
dwPathLength % sizeof(WCHAR) != 0)
|
|
{
|
|
dwPathLength = 0;
|
|
goto set;
|
|
}
|
|
|
|
/* Reserve space for existing data + new string */
|
|
dwNewLength = dwPathLength + (wcslen(lpPath) + 1) * sizeof(WCHAR);
|
|
Buffer = HeapAlloc(GetProcessHeap(), 0, dwNewLength);
|
|
if (!Buffer)
|
|
return;
|
|
|
|
ZeroMemory(Buffer, dwNewLength);
|
|
|
|
res = RegQueryValueExW(
|
|
hKey,
|
|
L"Installation Sources",
|
|
NULL,
|
|
NULL,
|
|
(LPBYTE)Buffer,
|
|
&dwPathLength);
|
|
|
|
if (res != ERROR_SUCCESS)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, Buffer);
|
|
dwPathLength = 0;
|
|
goto set;
|
|
}
|
|
|
|
/* Sanity check, these should already be zeros */
|
|
Buffer[dwPathLength / sizeof(WCHAR) - 2] = UNICODE_NULL;
|
|
Buffer[dwPathLength / sizeof(WCHAR) - 1] = UNICODE_NULL;
|
|
|
|
for (Path = Buffer; *Path; Path += wcslen(Path) + 1)
|
|
{
|
|
/* Check if path is already added */
|
|
if (PathIsEqual(Path, lpPath))
|
|
goto cleanup;
|
|
}
|
|
|
|
Path = Buffer + dwPathLength / sizeof(WCHAR) - 1;
|
|
|
|
set:
|
|
if (dwPathLength == 0)
|
|
{
|
|
dwNewLength = (wcslen(lpPath) + 1 + 1) * sizeof(WCHAR);
|
|
Buffer = HeapAlloc(GetProcessHeap(), 0, dwNewLength);
|
|
if (!Buffer)
|
|
return;
|
|
|
|
Path = Buffer;
|
|
}
|
|
|
|
StringCbCopyW(Path, dwNewLength - (Path - Buffer) * sizeof(WCHAR), lpPath);
|
|
Buffer[dwNewLength / sizeof(WCHAR) - 1] = UNICODE_NULL;
|
|
|
|
RegSetValueExW(
|
|
hKey,
|
|
L"Installation Sources",
|
|
0,
|
|
REG_MULTI_SZ,
|
|
(LPBYTE)Buffer,
|
|
dwNewLength);
|
|
|
|
cleanup:
|
|
HeapFree(GetProcessHeap(), 0, Buffer);
|
|
}
|
|
|
|
VOID
|
|
ProcessSetupInf(
|
|
IN OUT PSETUPDATA pSetupData)
|
|
{
|
|
WCHAR szPath[MAX_PATH];
|
|
WCHAR szValue[MAX_PATH];
|
|
INFCONTEXT InfContext;
|
|
DWORD LineLength;
|
|
HKEY hKey;
|
|
LONG res;
|
|
|
|
pSetupData->hSetupInf = INVALID_HANDLE_VALUE;
|
|
|
|
/* Retrieve the path of the setup INF */
|
|
GetSystemDirectoryW(szPath, _countof(szPath));
|
|
wcscat(szPath, L"\\$winnt$.inf");
|
|
|
|
/* Open the setup INF */
|
|
pSetupData->hSetupInf = SetupOpenInfFileW(szPath,
|
|
NULL,
|
|
INF_STYLE_OLDNT,
|
|
NULL);
|
|
if (pSetupData->hSetupInf == INVALID_HANDLE_VALUE)
|
|
{
|
|
DPRINT1("Error: Cannot open the setup information file %S with error %d\n", szPath, GetLastError());
|
|
return;
|
|
}
|
|
|
|
|
|
/* Retrieve the NT source path from which the 1st-stage installer was run */
|
|
if (!SetupFindFirstLineW(pSetupData->hSetupInf,
|
|
L"data",
|
|
L"sourcepath",
|
|
&InfContext))
|
|
{
|
|
DPRINT1("Error: Cannot find sourcepath Key! %d\n", GetLastError());
|
|
return;
|
|
}
|
|
|
|
if (!SetupGetStringFieldW(&InfContext,
|
|
1,
|
|
szValue,
|
|
ARRAYSIZE(szValue),
|
|
&LineLength))
|
|
{
|
|
DPRINT1("Error: SetupGetStringField failed with %d\n", GetLastError());
|
|
return;
|
|
}
|
|
|
|
*pSetupData->SourcePath = UNICODE_NULL;
|
|
|
|
/* Close the setup INF as we are going to modify it manually */
|
|
if (pSetupData->hSetupInf != INVALID_HANDLE_VALUE)
|
|
SetupCloseInfFile(pSetupData->hSetupInf);
|
|
|
|
|
|
/* Find the installation source path in Win32 format */
|
|
if (!GetInstallSourceWin32(pSetupData->SourcePath,
|
|
_countof(pSetupData->SourcePath),
|
|
szValue))
|
|
{
|
|
*pSetupData->SourcePath = UNICODE_NULL;
|
|
}
|
|
|
|
/* Save the path in Win32 format in the setup INF */
|
|
swprintf(szValue, L"\"%s\"", pSetupData->SourcePath);
|
|
WritePrivateProfileStringW(L"data", L"dospath", szValue, szPath);
|
|
|
|
/*
|
|
* Save it also in the registry, in the following keys:
|
|
* - HKLM\Software\Microsoft\Windows\CurrentVersion\Setup ,
|
|
* values "SourcePath" and "ServicePackSourcePath" (REG_SZ);
|
|
* - HKLM\Software\Microsoft\Windows NT\CurrentVersion ,
|
|
* value "SourcePath" (REG_SZ); set to the full path (e.g. D:\I386).
|
|
*/
|
|
#if 0
|
|
res = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
L"Software\\Microsoft\\Windows NT\\CurrentVersion",
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
&hKey);
|
|
|
|
if (res != ERROR_SUCCESS)
|
|
{
|
|
return FALSE;
|
|
}
|
|
#endif
|
|
|
|
res = RegCreateKeyExW(HKEY_LOCAL_MACHINE,
|
|
L"Software\\Microsoft\\Windows\\CurrentVersion\\Setup",
|
|
0, NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_ALL_ACCESS, // KEY_WRITE
|
|
NULL,
|
|
&hKey,
|
|
NULL);
|
|
if (res == ERROR_SUCCESS)
|
|
{
|
|
AddInstallationSource(hKey, pSetupData->SourcePath);
|
|
|
|
res = RegSetValueExW(hKey,
|
|
L"SourcePath",
|
|
0,
|
|
REG_SZ,
|
|
(LPBYTE)pSetupData->SourcePath,
|
|
(wcslen(pSetupData->SourcePath) + 1) * sizeof(WCHAR));
|
|
|
|
res = RegSetValueExW(hKey,
|
|
L"ServicePackSourcePath",
|
|
0,
|
|
REG_SZ,
|
|
(LPBYTE)pSetupData->SourcePath,
|
|
(wcslen(pSetupData->SourcePath) + 1) * sizeof(WCHAR));
|
|
|
|
RegCloseKey(hKey);
|
|
}
|
|
|
|
|
|
/* Now, re-open the setup INF (this must succeed) */
|
|
pSetupData->hSetupInf = SetupOpenInfFileW(szPath,
|
|
NULL,
|
|
INF_STYLE_OLDNT,
|
|
NULL);
|
|
if (pSetupData->hSetupInf == INVALID_HANDLE_VALUE)
|
|
{
|
|
DPRINT1("Error: Cannot open the setup information file %S with error %d\n", szPath, GetLastError());
|
|
return;
|
|
}
|
|
|
|
/* Process the unattended section of the setup file */
|
|
ProcessUnattendSection(pSetupData);
|
|
}
|
|
|
|
typedef DWORD(WINAPI *PFNREQUESTWIZARDPAGES)(PDWORD, HPROPSHEETPAGE *, PSETUPDATA);
|
|
|
|
VOID
|
|
InstallWizard(VOID)
|
|
{
|
|
PROPSHEETHEADER psh = {0};
|
|
HPROPSHEETPAGE *phpage = NULL;
|
|
PROPSHEETPAGE psp = {0};
|
|
UINT nPages = 0;
|
|
HWND hWnd;
|
|
MSG msg;
|
|
PSETUPDATA pSetupData = NULL;
|
|
HMODULE hNetShell = NULL;
|
|
PFNREQUESTWIZARDPAGES pfn = NULL;
|
|
DWORD dwPageCount = 10, dwNetworkPageCount = 0;
|
|
|
|
LogItem(L"BEGIN_SECTION", L"InstallWizard");
|
|
|
|
/* Allocate setup data */
|
|
pSetupData = HeapAlloc(GetProcessHeap(),
|
|
HEAP_ZERO_MEMORY,
|
|
sizeof(SETUPDATA));
|
|
if (pSetupData == NULL)
|
|
{
|
|
LogItem(NULL, L"SetupData allocation failed!");
|
|
MessageBoxW(NULL,
|
|
L"Setup failed to allocate global data!",
|
|
L"ReactOS Setup",
|
|
MB_ICONERROR | MB_OK);
|
|
goto done;
|
|
}
|
|
pSetupData->ProductOption = PRODUCT_OPTION_DEFAULT;
|
|
|
|
hNetShell = LoadLibraryW(L"netshell.dll");
|
|
if (hNetShell != NULL)
|
|
{
|
|
DPRINT("Netshell.dll loaded!\n");
|
|
|
|
pfn = (PFNREQUESTWIZARDPAGES)GetProcAddress(hNetShell,
|
|
"NetSetupRequestWizardPages");
|
|
if (pfn != NULL)
|
|
{
|
|
pfn(&dwNetworkPageCount, NULL, NULL);
|
|
dwPageCount += dwNetworkPageCount;
|
|
}
|
|
}
|
|
|
|
DPRINT("PageCount: %lu\n", dwPageCount);
|
|
|
|
phpage = HeapAlloc(GetProcessHeap(),
|
|
HEAP_ZERO_MEMORY,
|
|
dwPageCount * sizeof(HPROPSHEETPAGE));
|
|
if (phpage == NULL)
|
|
{
|
|
LogItem(NULL, L"Page array allocation failed!");
|
|
MessageBoxW(NULL,
|
|
L"Setup failed to allocate page array!",
|
|
L"ReactOS Setup",
|
|
MB_ICONERROR | MB_OK);
|
|
goto done;
|
|
}
|
|
|
|
/* Process the $winnt$.inf setup file */
|
|
ProcessSetupInf(pSetupData);
|
|
|
|
/* Create the Welcome page */
|
|
psp.dwSize = sizeof(PROPSHEETPAGE);
|
|
psp.dwFlags = PSP_DEFAULT | PSP_HIDEHEADER;
|
|
psp.hInstance = hDllInstance;
|
|
psp.lParam = (LPARAM)pSetupData;
|
|
psp.pfnDlgProc = WelcomeDlgProc;
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_WELCOMEPAGE);
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
/* Create the Acknowledgements page */
|
|
psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
|
|
psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_ACKTITLE);
|
|
psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_ACKSUBTITLE);
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_ACKPAGE);
|
|
psp.pfnDlgProc = AckPageDlgProc;
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
/* Create the Product page */
|
|
psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
|
|
psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_PRODUCTTITLE);
|
|
psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_PRODUCTSUBTITLE);
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_PRODUCT);
|
|
psp.pfnDlgProc = ProductPageDlgProc;
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
/* Create the Locale page */
|
|
psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
|
|
psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_LOCALETITLE);
|
|
psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_LOCALESUBTITLE);
|
|
psp.pfnDlgProc = LocalePageDlgProc;
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_LOCALEPAGE);
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
/* Create the Owner page */
|
|
psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
|
|
psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_OWNERTITLE);
|
|
psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_OWNERSUBTITLE);
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_OWNERPAGE);
|
|
psp.pfnDlgProc = OwnerPageDlgProc;
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
/* Create the Computer page */
|
|
psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
|
|
psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_COMPUTERTITLE);
|
|
psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_COMPUTERSUBTITLE);
|
|
psp.pfnDlgProc = ComputerPageDlgProc;
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_COMPUTERPAGE);
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
/* Create the DateTime page */
|
|
psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
|
|
psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_DATETIMETITLE);
|
|
psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_DATETIMESUBTITLE);
|
|
psp.pfnDlgProc = DateTimePageDlgProc;
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_DATETIMEPAGE);
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
/* Create the theme selection page */
|
|
psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
|
|
psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_THEMESELECTIONTITLE);
|
|
psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_THEMESELECTIONSUBTITLE);
|
|
psp.pfnDlgProc = ThemePageDlgProc;
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_THEMEPAGE);
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
pSetupData->uFirstNetworkWizardPage = IDD_PROCESSPAGE;
|
|
pSetupData->uPostNetworkWizardPage = IDD_PROCESSPAGE;
|
|
|
|
if (pfn)
|
|
{
|
|
pfn(&dwNetworkPageCount, &phpage[nPages], pSetupData);
|
|
nPages += dwNetworkPageCount;
|
|
}
|
|
|
|
/* Create the Process page */
|
|
psp.dwFlags = PSP_DEFAULT | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
|
|
psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_PROCESSTITLE);
|
|
psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_PROCESSSUBTITLE);
|
|
psp.pfnDlgProc = ProcessPageDlgProc;
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_PROCESSPAGE);
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
/* Create the Finish page */
|
|
psp.dwFlags = PSP_DEFAULT | PSP_HIDEHEADER;
|
|
psp.pfnDlgProc = FinishDlgProc;
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_FINISHPAGE);
|
|
phpage[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
ASSERT(nPages == dwPageCount);
|
|
|
|
/* Create the property sheet */
|
|
psh.dwSize = sizeof(PROPSHEETHEADER);
|
|
psh.dwFlags = PSH_WIZARD97 | PSH_WATERMARK | PSH_HEADER | PSH_MODELESS;
|
|
psh.hInstance = hDllInstance;
|
|
psh.hwndParent = NULL;
|
|
psh.nPages = nPages;
|
|
psh.nStartPage = 0;
|
|
psh.phpage = phpage;
|
|
psh.pszbmWatermark = MAKEINTRESOURCE(IDB_WATERMARK);
|
|
psh.pszbmHeader = MAKEINTRESOURCE(IDB_HEADER);
|
|
|
|
/* Create title font */
|
|
pSetupData->hTitleFont = CreateTitleFont();
|
|
pSetupData->hBoldFont = CreateBoldFont();
|
|
|
|
/* Display the wizard */
|
|
hWnd = (HWND)PropertySheet(&psh);
|
|
ShowWindow(hWnd, SW_SHOW);
|
|
|
|
while (GetMessage(&msg, NULL, 0, 0))
|
|
{
|
|
if (!IsDialogMessage(hWnd, &msg))
|
|
{
|
|
TranslateMessage(&msg);
|
|
DispatchMessage(&msg);
|
|
}
|
|
}
|
|
|
|
DeleteObject(pSetupData->hBoldFont);
|
|
DeleteObject(pSetupData->hTitleFont);
|
|
|
|
if (pSetupData->hSetupInf != INVALID_HANDLE_VALUE)
|
|
SetupCloseInfFile(pSetupData->hSetupInf);
|
|
|
|
done:
|
|
if (phpage != NULL)
|
|
HeapFree(GetProcessHeap(), 0, phpage);
|
|
|
|
if (hNetShell != NULL)
|
|
FreeLibrary(hNetShell);
|
|
|
|
if (pSetupData != NULL)
|
|
HeapFree(GetProcessHeap(), 0, pSetupData);
|
|
|
|
LogItem(L"END_SECTION", L"InstallWizard");
|
|
}
|
|
|
|
/* EOF */
|