mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
611 lines
18 KiB
C
611 lines
18 KiB
C
/*
|
|
* ReactOS kernel
|
|
* Copyright (C) 2008 ReactOS Team
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS text-mode setup
|
|
* FILE: base/setup/usetup/mui.c
|
|
* PURPOSE: Text-mode setup
|
|
* PROGRAMMER:
|
|
*/
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
#include "precomp.h"
|
|
#include "mui.h"
|
|
#include "muifonts.h"
|
|
#include "muilanguages.h"
|
|
#include "registry.h"
|
|
#include "substset.h"
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
|
|
/* FUNCTIONS ****************************************************************/
|
|
|
|
static
|
|
ULONG
|
|
FindLanguageIndex(
|
|
IN PCWSTR LanguageId)
|
|
{
|
|
ULONG lngIndex = 0;
|
|
|
|
if (LanguageId == NULL)
|
|
{
|
|
/* Default to en-US */
|
|
// return 0; // FIXME!!
|
|
LanguageId = L"00000409";
|
|
}
|
|
|
|
while (MUILanguageList[lngIndex].LanguageID != NULL)
|
|
{
|
|
if (_wcsicmp(MUILanguageList[lngIndex].LanguageID, LanguageId) == 0)
|
|
{
|
|
return lngIndex;
|
|
}
|
|
|
|
lngIndex++;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
BOOLEAN
|
|
IsLanguageAvailable(
|
|
IN PCWSTR LanguageId)
|
|
{
|
|
ULONG lngIndex = 0;
|
|
|
|
while (MUILanguageList[lngIndex].LanguageID != NULL)
|
|
{
|
|
if (_wcsicmp(MUILanguageList[lngIndex].LanguageID, LanguageId) == 0)
|
|
return TRUE;
|
|
|
|
lngIndex++;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
KLID
|
|
MUIDefaultKeyboardLayout(
|
|
IN PCWSTR LanguageId)
|
|
{
|
|
ULONG lngIndex = FindLanguageIndex(LanguageId);
|
|
return MUILanguageList[lngIndex].MuiLayouts[0].LayoutID;
|
|
}
|
|
|
|
UINT
|
|
MUIGetOEMCodePage(
|
|
IN PCWSTR LanguageId)
|
|
{
|
|
ULONG lngIndex = FindLanguageIndex(LanguageId);
|
|
return MUILanguageList[lngIndex].OEMCPage;
|
|
}
|
|
|
|
GEOID
|
|
MUIGetGeoID(
|
|
IN PCWSTR LanguageId)
|
|
{
|
|
ULONG lngIndex = FindLanguageIndex(LanguageId);
|
|
return MUILanguageList[lngIndex].GeoID;
|
|
}
|
|
|
|
const MUI_LAYOUTS*
|
|
MUIGetLayoutsList(
|
|
IN PCWSTR LanguageId)
|
|
{
|
|
ULONG lngIndex = FindLanguageIndex(LanguageId);
|
|
return MUILanguageList[lngIndex].MuiLayouts;
|
|
}
|
|
|
|
|
|
static
|
|
BOOLEAN
|
|
AddHotkeySettings(
|
|
IN PCWSTR Hotkey,
|
|
IN PCWSTR LangHotkey,
|
|
IN PCWSTR LayoutHotkey)
|
|
{
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
UNICODE_STRING KeyName;
|
|
UNICODE_STRING ValueName;
|
|
HANDLE KeyHandle;
|
|
ULONG Disposition;
|
|
NTSTATUS Status;
|
|
|
|
RtlInitUnicodeString(&KeyName,
|
|
L".DEFAULT\\Keyboard Layout\\Toggle");
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
&KeyName,
|
|
OBJ_CASE_INSENSITIVE,
|
|
GetRootKeyByPredefKey(HKEY_USERS, NULL),
|
|
NULL);
|
|
|
|
Status = NtCreateKey(&KeyHandle,
|
|
KEY_SET_VALUE,
|
|
&ObjectAttributes,
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
&Disposition);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtCreateKey() failed (Status %lx)\n", Status);
|
|
return FALSE;
|
|
}
|
|
|
|
RtlInitUnicodeString(&ValueName,
|
|
L"Hotkey");
|
|
|
|
Status = NtSetValueKey(KeyHandle,
|
|
&ValueName,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)Hotkey,
|
|
(1 + 1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status %lx)\n", Status);
|
|
NtClose(KeyHandle);
|
|
return FALSE;
|
|
}
|
|
|
|
RtlInitUnicodeString(&ValueName,
|
|
L"Language Hotkey");
|
|
|
|
Status = NtSetValueKey(KeyHandle,
|
|
&ValueName,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)LangHotkey,
|
|
(1 + 1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status %lx)\n", Status);
|
|
NtClose(KeyHandle);
|
|
return FALSE;
|
|
}
|
|
|
|
RtlInitUnicodeString(&ValueName,
|
|
L"Layout Hotkey");
|
|
|
|
Status = NtSetValueKey(KeyHandle,
|
|
&ValueName,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)LayoutHotkey,
|
|
(1 + 1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status %lx)\n", Status);
|
|
NtClose(KeyHandle);
|
|
return FALSE;
|
|
}
|
|
|
|
NtClose(KeyHandle);
|
|
return TRUE;
|
|
}
|
|
|
|
BOOLEAN
|
|
AddKbLayoutsToRegistry(
|
|
_In_ const MUI_LAYOUTS* MuiLayouts)
|
|
{
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
UNICODE_STRING KeyName;
|
|
UNICODE_STRING ValueName;
|
|
HANDLE KeyHandle;
|
|
HANDLE SubKeyHandle;
|
|
NTSTATUS Status;
|
|
ULONG Disposition;
|
|
ULONG uIndex;
|
|
ULONG uCount;
|
|
WCHAR szKeyName[48] = L".DEFAULT\\Keyboard Layout";
|
|
WCHAR szValueName[3 + 1];
|
|
WCHAR szSubstID[8 + 1];
|
|
WCHAR szLayoutID[8 + 1];
|
|
|
|
/* Open the keyboard layout key */
|
|
RtlInitUnicodeString(&KeyName, szKeyName);
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
&KeyName,
|
|
OBJ_CASE_INSENSITIVE,
|
|
GetRootKeyByPredefKey(HKEY_USERS, NULL),
|
|
NULL);
|
|
|
|
Status = NtCreateKey(&KeyHandle,
|
|
KEY_CREATE_SUB_KEY,
|
|
&ObjectAttributes,
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
&Disposition);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtCreateKey() failed (Status %lx)\n", Status);
|
|
return FALSE;
|
|
}
|
|
|
|
NtClose(KeyHandle);
|
|
|
|
KeyName.MaximumLength = sizeof(szKeyName);
|
|
Status = RtlAppendUnicodeToString(&KeyName, L"\\Preload");
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("RtlAppend() failed (%lx), string is '%wZ'\n", Status, &KeyName);
|
|
return FALSE;
|
|
}
|
|
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
&KeyName,
|
|
OBJ_CASE_INSENSITIVE,
|
|
GetRootKeyByPredefKey(HKEY_USERS, NULL),
|
|
NULL);
|
|
|
|
Status = NtCreateKey(&KeyHandle,
|
|
KEY_SET_VALUE,
|
|
&ObjectAttributes,
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
&Disposition);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtCreateKey() failed (Status %lx)\n", Status);
|
|
return FALSE;
|
|
}
|
|
|
|
RtlInitUnicodeString(&KeyName, L".DEFAULT\\Keyboard Layout\\Substitutes");
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
&KeyName,
|
|
OBJ_CASE_INSENSITIVE,
|
|
GetRootKeyByPredefKey(HKEY_USERS, NULL),
|
|
NULL);
|
|
|
|
Status = NtCreateKey(&SubKeyHandle,
|
|
KEY_SET_VALUE,
|
|
&ObjectAttributes,
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
&Disposition);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtCreateKey() failed (Status %lx)\n", Status);
|
|
goto Quit;
|
|
}
|
|
|
|
uCount = 0;
|
|
for (uIndex = 0; (uIndex <= 19) && (MuiLayouts[uIndex].LangID != 0); ++uIndex)
|
|
{
|
|
RtlStringCchPrintfW(szValueName, _countof(szValueName), L"%u", uIndex + 1);
|
|
RtlInitUnicodeString(&ValueName, szValueName);
|
|
|
|
RtlStringCchPrintfW(szLayoutID, _countof(szLayoutID), L"%08lx", MuiLayouts[uIndex].LayoutID);
|
|
|
|
if ((KLID)MuiLayouts[uIndex].LangID == MuiLayouts[uIndex].LayoutID)
|
|
{
|
|
/* Main keyboard layout */
|
|
Status = NtSetValueKey(KeyHandle,
|
|
&ValueName,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)szLayoutID,
|
|
(wcslen(szLayoutID)+1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status = %lx, uIndex = %u)\n", Status, uIndex);
|
|
goto Quit;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Generate a substitute keyboard layout ID */
|
|
RtlStringCchPrintfW(szSubstID, _countof(szSubstID), L"%08lx",
|
|
(0xD0000000/*SUBST_MASK*/ | ((USHORT)uCount << 4) | MuiLayouts[uIndex].LangID));
|
|
Status = NtSetValueKey(KeyHandle,
|
|
&ValueName,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)szSubstID,
|
|
(wcslen(szSubstID)+1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status = %lx, uIndex = %u)\n", Status, uIndex);
|
|
goto Quit;
|
|
}
|
|
|
|
/* Link the substitute layout with the original one */
|
|
RtlInitUnicodeString(&ValueName, szSubstID);
|
|
Status = NtSetValueKey(SubKeyHandle,
|
|
&ValueName,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)szLayoutID,
|
|
(wcslen(szLayoutID)+1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status = %lx, uIndex = %u)\n", Status, uIndex);
|
|
goto Quit;
|
|
}
|
|
|
|
++uCount;
|
|
}
|
|
}
|
|
|
|
AddHotkeySettings(L"1", L"1", L"2");
|
|
|
|
Quit:
|
|
NtClose(SubKeyHandle);
|
|
NtClose(KeyHandle);
|
|
return NT_SUCCESS(Status);
|
|
}
|
|
|
|
BOOLEAN
|
|
AddKeyboardLayouts(
|
|
IN PCWSTR LanguageId)
|
|
{
|
|
ULONG lngIndex = 0;
|
|
|
|
while (MUILanguageList[lngIndex].LanguageID != NULL)
|
|
{
|
|
if (_wcsicmp(MUILanguageList[lngIndex].LanguageID, LanguageId) == 0)
|
|
{
|
|
return AddKbLayoutsToRegistry(MUILanguageList[lngIndex].MuiLayouts);
|
|
}
|
|
|
|
lngIndex++;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static
|
|
BOOLEAN
|
|
AddCodepageToRegistry(
|
|
_In_ UINT ACPage,
|
|
_In_ UINT OEMCPage,
|
|
_In_ UINT MACCPage)
|
|
{
|
|
NTSTATUS Status;
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
UNICODE_STRING Name;
|
|
HANDLE KeyHandle;
|
|
/*
|
|
* Buffer big enough to hold the NULL-terminated string L"4294967295",
|
|
* corresponding to the literal 0xFFFFFFFF (MAXULONG) in decimal.
|
|
*/
|
|
WCHAR Value[sizeof("4294967295")];
|
|
|
|
/* Open the NLS CodePage key */
|
|
RtlInitUnicodeString(&Name,
|
|
L"SYSTEM\\CurrentControlSet\\Control\\NLS\\CodePage");
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
&Name,
|
|
OBJ_CASE_INSENSITIVE,
|
|
GetRootKeyByPredefKey(HKEY_LOCAL_MACHINE, NULL),
|
|
NULL);
|
|
Status = NtOpenKey(&KeyHandle,
|
|
KEY_WRITE,
|
|
&ObjectAttributes);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtOpenKey() failed (Status %lx)\n", Status);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Set ANSI codepage */
|
|
Status = RtlStringCchPrintfW(Value, _countof(Value), L"%lu", ACPage);
|
|
ASSERT(NT_SUCCESS(Status));
|
|
|
|
RtlInitUnicodeString(&Name, L"ACP");
|
|
Status = NtSetValueKey(KeyHandle,
|
|
&Name,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)Value,
|
|
(wcslen(Value)+1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status %lx)\n", Status);
|
|
goto Quit;
|
|
}
|
|
|
|
/* Set OEM codepage */
|
|
Status = RtlStringCchPrintfW(Value, _countof(Value), L"%lu", OEMCPage);
|
|
ASSERT(NT_SUCCESS(Status));
|
|
|
|
RtlInitUnicodeString(&Name, L"OEMCP");
|
|
Status = NtSetValueKey(KeyHandle,
|
|
&Name,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)Value,
|
|
(wcslen(Value)+1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status %lx)\n", Status);
|
|
goto Quit;
|
|
}
|
|
|
|
/* Set MAC codepage */
|
|
Status = RtlStringCchPrintfW(Value, _countof(Value), L"%lu", MACCPage);
|
|
ASSERT(NT_SUCCESS(Status));
|
|
|
|
RtlInitUnicodeString(&Name, L"MACCP");
|
|
Status = NtSetValueKey(KeyHandle,
|
|
&Name,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)Value,
|
|
(wcslen(Value)+1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status %lx)\n", Status);
|
|
goto Quit;
|
|
}
|
|
|
|
Quit:
|
|
NtClose(KeyHandle);
|
|
return NT_SUCCESS(Status);
|
|
}
|
|
|
|
static
|
|
BOOLEAN
|
|
AddFontsSettingsToRegistry(
|
|
IN const MUI_SUBFONT * MuiSubFonts)
|
|
{
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
UNICODE_STRING KeyName;
|
|
UNICODE_STRING ValueName;
|
|
HANDLE KeyHandle;
|
|
NTSTATUS Status;
|
|
ULONG uIndex = 0;
|
|
|
|
RtlInitUnicodeString(&KeyName,
|
|
L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\FontSubstitutes");
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
&KeyName,
|
|
OBJ_CASE_INSENSITIVE,
|
|
GetRootKeyByPredefKey(HKEY_LOCAL_MACHINE, NULL),
|
|
NULL);
|
|
Status = NtOpenKey(&KeyHandle,
|
|
KEY_WRITE,
|
|
&ObjectAttributes);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtOpenKey() failed (Status %lx)\n", Status);
|
|
return FALSE;
|
|
}
|
|
|
|
while (MuiSubFonts[uIndex].FontName != NULL)
|
|
{
|
|
RtlInitUnicodeString(&ValueName, MuiSubFonts[uIndex].FontName);
|
|
if (MuiSubFonts[uIndex].SubFontName)
|
|
{
|
|
Status = NtSetValueKey(KeyHandle,
|
|
&ValueName,
|
|
0,
|
|
REG_SZ,
|
|
(PVOID)MuiSubFonts[uIndex].SubFontName,
|
|
(wcslen(MuiSubFonts[uIndex].SubFontName)+1) * sizeof(WCHAR));
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtSetValueKey() failed (Status = %lx, uIndex = %d)\n", Status, uIndex);
|
|
NtClose(KeyHandle);
|
|
return FALSE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Status = NtDeleteValueKey(KeyHandle, &ValueName);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("NtDeleteValueKey failed, Status = %lx\n", Status);
|
|
}
|
|
}
|
|
|
|
uIndex++;
|
|
}
|
|
|
|
NtClose(KeyHandle);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOLEAN
|
|
AddCodePage(
|
|
IN PCWSTR LanguageId)
|
|
{
|
|
ULONG lngIndex = 0;
|
|
|
|
while (MUILanguageList[lngIndex].LanguageID != NULL)
|
|
{
|
|
if (_wcsicmp(MUILanguageList[lngIndex].LanguageID, LanguageId) == 0)
|
|
{
|
|
if (AddCodepageToRegistry(MUILanguageList[lngIndex].ACPage,
|
|
MUILanguageList[lngIndex].OEMCPage,
|
|
MUILanguageList[lngIndex].MACCPage) &&
|
|
AddFontsSettingsToRegistry(MUILanguageList[lngIndex].MuiSubFonts))
|
|
{
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
lngIndex++;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
#ifdef __REACTOS__ /* HACK */
|
|
BOOL
|
|
DoRegistryFontFixup(PFONTSUBSTSETTINGS pSettings, LANGID LangID)
|
|
{
|
|
if (pSettings->bFoundFontMINGLIU)
|
|
AddFontsSettingsToRegistry(FontFixupMINGLIU);
|
|
if (pSettings->bFoundFontSIMSUN)
|
|
AddFontsSettingsToRegistry(FontFixupSIMSUN);
|
|
if (pSettings->bFoundFontMSSONG)
|
|
AddFontsSettingsToRegistry(FontFixupMSSONG);
|
|
if (pSettings->bFoundFontMSGOTHIC)
|
|
AddFontsSettingsToRegistry(FontFixupMSGOTHIC);
|
|
if (pSettings->bFoundFontMSMINCHO)
|
|
AddFontsSettingsToRegistry(FontFixupMSMINCHO);
|
|
if (pSettings->bFoundFontGULIM)
|
|
AddFontsSettingsToRegistry(FontFixupGULIM);
|
|
if (pSettings->bFoundFontBATANG)
|
|
AddFontsSettingsToRegistry(FontFixupBATANG);
|
|
|
|
switch (PRIMARYLANGID(LangID))
|
|
{
|
|
case LANG_CHINESE:
|
|
if (SUBLANGID(LangID) == SUBLANG_CHINESE_SIMPLIFIED)
|
|
{
|
|
if (pSettings->bFoundFontSIMSUN)
|
|
AddFontsSettingsToRegistry(SimplifiedChineseFontFixup);
|
|
}
|
|
else
|
|
{
|
|
if (pSettings->bFoundFontMINGLIU)
|
|
AddFontsSettingsToRegistry(TraditionalChineseFontFixup);
|
|
}
|
|
break;
|
|
|
|
case LANG_JAPANESE:
|
|
if (pSettings->bFoundFontMSGOTHIC)
|
|
AddFontsSettingsToRegistry(JapaneseFontFixup);
|
|
break;
|
|
|
|
case LANG_KOREAN:
|
|
if (pSettings->bFoundFontBATANG)
|
|
AddFontsSettingsToRegistry(KoreanFontFixup);
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
#endif /* HACK */
|
|
|
|
/* EOF */
|