https://git.reactos.org/?p=reactos.git;a=commitdiff;h=c4e2826c73bb13ea9cdf6a9178f93866ab6db00f

commit c4e2826c73bb13ea9cdf6a9178f93866ab6db00f
Author:     Hermès Bélusca-Maïto <hermes.belusca-ma...@reactos.org>
AuthorDate: Sun May 19 22:55:14 2019 +0200
Commit:     Hermès Bélusca-Maïto <hermes.belusca-ma...@reactos.org>
CommitDate: Sun May 19 23:10:41 2019 +0200

    [CONSOLE][CONCFG][CONSRV] Provide support for specified additional TrueType 
fonts + bugfixes.
    CORE-12451 CORE-13182 CORE-13196
    
    - CONSOLE: Initialize the additional TrueType fonts cache.
      * Fix the font preview when a TrueType font has been selected.
      * Refresh the available fonts and the font preview when the selected
        code page is changed (Work In Progress), or when the OS pool of
        font resources has changed (WM_FONTCHANGE message).
    
    - CONCFG: Implement support for the additional TrueType fonts cache:
      the contents of the cache is enumerated under the registry key
        HKEY_LOCAL_MACHINE\Software\Microsoft\Windows 
NT\CurrentVersion\Console\TrueTypeFont
    
      * Add helper functions and macros.
      * In CreateConsoleFontEx(), set the mandatory font pitch&family flags
        and remove those that we do not support.
      * In IsValidConsoleFont2(), update the validity checks and the
        documentation links.
    
    - CONSRV: Load/refresh the additional TrueType fonts cache when needed.
---
 dll/cpl/console/console.c                          |  12 +-
 dll/cpl/console/console.h                          |  17 ++
 dll/cpl/console/font.c                             | 169 ++++++++----
 dll/cpl/console/layout.c                           |   6 +
 dll/cpl/console/options.c                          |  20 +-
 win32ss/user/winsrv/concfg/font.c                  | 293 ++++++++++++++++++---
 win32ss/user/winsrv/concfg/font.h                  |  71 ++++-
 win32ss/user/winsrv/concfg/precomp.h               |   3 +-
 win32ss/user/winsrv/consrv/frontends/gui/conwnd.c  |   2 +-
 .../user/winsrv/consrv/frontends/gui/guisettings.c |   4 +-
 win32ss/user/winsrv/consrv/frontends/gui/guiterm.c |  10 +-
 11 files changed, 498 insertions(+), 109 deletions(-)

diff --git a/dll/cpl/console/console.c b/dll/cpl/console/console.c
index 89e9ac57fb3..09c204c630c 100644
--- a/dll/cpl/console/console.c
+++ b/dll/cpl/console/console.c
@@ -227,11 +227,9 @@ InitApplet(HANDLE hSectionOrWnd)
         InitDefaultConsoleInfo(ConInfo);
     }
 
-    /* Initialize the font support */
-    FontPreview.hFont = CreateConsoleFont(ConInfo);
-    if (FontPreview.hFont == NULL)
-        DPRINT1("InitApplet: CreateConsoleFont() failed\n");
-    GetFontCellSize(NULL, FontPreview.hFont, &FontPreview.CharHeight, 
&FontPreview.CharWidth);
+    /* Initialize the font support -- additional TrueType fonts cache and 
current preview font */
+    InitTTFontCache();
+    RefreshFontPreview(&FontPreview, ConInfo);
 
     /* Initialize the property sheet structure */
     ZeroMemory(&psh, sizeof(psh));
@@ -274,8 +272,8 @@ InitApplet(HANDLE hSectionOrWnd)
     UnRegisterWinPrevClass(hApplet);
 
     /* Clear the font support */
-    if (FontPreview.hFont) DeleteObject(FontPreview.hFont);
-    FontPreview.hFont = NULL;
+    ResetFontPreview(&FontPreview);
+    ClearTTFontCache();
 
     /* Save the console settings */
     if (SetConsoleInfo)
diff --git a/dll/cpl/console/console.h b/dll/cpl/console/console.h
index 0f7b5924943..fbd7a81368d 100644
--- a/dll/cpl/console/console.h
+++ b/dll/cpl/console/console.h
@@ -49,6 +49,23 @@ extern FONT_PREVIEW FontPreview;
 
 VOID ApplyConsoleInfo(HWND hwndDlg);
 
+
+VOID
+RefreshFontPreview(
+    IN FONT_PREVIEW* Preview,
+    IN PCONSOLE_STATE_INFO pConInfo);
+
+VOID
+UpdateFontPreview(
+    IN FONT_PREVIEW* Preview,
+    IN HFONT hFont,
+    IN UINT  CharWidth,
+    IN UINT  CharHeight);
+
+#define ResetFontPreview(Preview)   \
+    UpdateFontPreview((Preview), NULL, 0, 0)
+
+
 /* Preview Windows */
 BOOL
 RegisterWinPrevClass(
diff --git a/dll/cpl/console/font.c b/dll/cpl/console/font.c
index 66c9cc6a417..522a6fbaaf4 100644
--- a/dll/cpl/console/font.c
+++ b/dll/cpl/console/font.c
@@ -45,6 +45,31 @@ static INT   CurrentSelFont  = LB_ERR;
 static DWORD CurrentFontType = (DWORD)-1;   // Invalid font type
 
 
+VOID
+RefreshFontPreview(
+    IN FONT_PREVIEW* Preview,
+    IN PCONSOLE_STATE_INFO pConInfo)
+{
+    if (Preview->hFont) DeleteObject(Preview->hFont);
+    Preview->hFont = CreateConsoleFont(pConInfo);
+    if (Preview->hFont == NULL)
+        DPRINT1("RefreshFontPreview: CreateConsoleFont() failed\n");
+    GetFontCellSize(NULL, Preview->hFont, &Preview->CharHeight, 
&Preview->CharWidth);
+}
+
+VOID
+UpdateFontPreview(
+    IN FONT_PREVIEW* Preview,
+    IN HFONT hFont,
+    IN UINT  CharWidth,
+    IN UINT  CharHeight)
+{
+    if (Preview->hFont) DeleteObject(Preview->hFont);
+    Preview->hFont = hFont;
+    Preview->CharWidth  = CharWidth;
+    Preview->CharHeight = CharHeight;
+}
+
 // PLIST_GETCOUNT
 static INT
 RasterSizeList_GetCount(
@@ -284,13 +309,6 @@ EnumFaceNamesProc(
 {
     PFACE_NAMES_PROC_PARAM Param = (PFACE_NAMES_PROC_PARAM)lParam;
 
-    /*
-     * To install additional TrueType fonts to be available for the console,
-     * add entries of type REG_SZ named "0", "00" etc... in:
-     * HKEY_LOCAL_MACHINE\Software\Microsoft\Windows 
NT\CurrentVersion\Console\TrueTypeFont
-     * The names of the fonts listed there should match those in:
-     * HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Fonts
-     */
     if (IsValidConsoleFont2(lplf, lpntm, FontType, Param->CodePage))
     {
         /* Add the font to the list */
@@ -374,7 +392,7 @@ FaceNameList_Initialize(
 
     ZeroMemory(&lf, sizeof(lf));
     lf.lfCharSet = DEFAULT_CHARSET; // CodePageToCharSet(CodePage);
-    // lf.lfPitchAndFamily = FIXED_PITCH | FF_DONTCARE;
+    // lf.lfPitchAndFamily = FIXED_PITCH | FF_MODERN;
 
     hDC = GetDC(NULL);
     EnumFontFamiliesExW(hDC, &lf, (FONTENUMPROCW)EnumFaceNamesProc, 
(LPARAM)&Param, 0);
@@ -397,9 +415,14 @@ FaceNameList_Initialize(
 }
 
 static VOID
-FaceNameList_SelectFaceName(
+FaceNameList_SelectFont(
+    IN HWND hDlg,
     IN HWND hWndList,
-    IN LPCWSTR FaceName)
+    IN PFONTSIZE_LIST_CTL SizeList,
+    IN LPCWSTR FaceName,
+    IN ULONG FontFamily,
+    IN ULONG FontWeight,
+    IN COORD FontSize)
 {
     INT iItem;
 
@@ -410,6 +433,21 @@ FaceNameList_SelectFaceName(
         iItem = 0;
     SendMessageW(hWndList, LB_SETCURSEL, (WPARAM)iItem, 0);
 
+    if (FontWeight >= FW_BOLD)
+        CheckDlgButton(hDlg, IDC_CHECK_BOLD_FONTS, BST_CHECKED);
+    else
+        CheckDlgButton(hDlg, IDC_CHECK_BOLD_FONTS, BST_UNCHECKED);
+
+    /* Select the current font size */
+    /*
+     * Format:
+     * Width  = FontSize.X = LOWORD(FontSize);
+     * Height = FontSize.Y = HIWORD(FontSize);
+     */
+    SizeList->CurrentRasterSize = MAKELONG(FontSize.X, FontSize.Y);
+    SizeList->CurrentTTSize = FontSize.Y;
+    // FontSizeList_SelectFontSize(SizeList, SizeList->CurrentRasterSize);
+
     // return iItem;
 }
 
@@ -486,7 +524,6 @@ FontTypeChange(
 {
     HWND hListBox = GetDlgItem(hDlg, IDC_LBOX_FONTTYPE);
     INT Length, nSel;
-    LOGFONTW lf;
     LPWSTR FaceName;
     DWORD FontType;
     LPCWSTR FontGrpBoxLabelTpl = NULL;
@@ -518,12 +555,7 @@ FontTypeChange(
     FaceName[Length] = L'\0';
 
     StringCchCopyW(pConInfo->FaceName, ARRAYSIZE(pConInfo->FaceName), 
FaceName);
-    DPRINT1("pConInfo->FaceName = '%S'\n", pConInfo->FaceName);
-
-    ZeroMemory(&lf, sizeof(lf));
-    lf.lfCharSet = DEFAULT_CHARSET; // CodePageToCharSet(pConInfo->CodePage);
-    // lf.lfPitchAndFamily = FIXED_PITCH | FF_DONTCARE;
-    StringCchCopyW(lf.lfFaceName, ARRAYSIZE(lf.lfFaceName), FaceName);
+    DPRINT("pConInfo->FaceName = '%S'\n", pConInfo->FaceName);
 
     /*
      * Retrieve the read-only font group box label string template,
@@ -560,6 +592,7 @@ FontTypeChange(
         if ((CurrentFontType != FontType) ||
             (FontType == RASTER_FONTTYPE && CurrentSelFont != nSel))
         {
+            LOGFONTW lf;
             HDC hDC;
 
             if (SizeList->UseRasterOrTTList)
@@ -567,6 +600,11 @@ FontTypeChange(
             else
                 SendMessageW(SizeList->hWndTTSizeList, CB_RESETCONTENT, 0, 0);
 
+            ZeroMemory(&lf, sizeof(lf));
+            lf.lfCharSet = DEFAULT_CHARSET; // 
CodePageToCharSet(pConInfo->CodePage);
+            // lf.lfPitchAndFamily = FIXED_PITCH | FF_MODERN;
+            StringCchCopyW(lf.lfFaceName, ARRAYSIZE(lf.lfFaceName), 
pConInfo->FaceName);
+
             hDC = GetDC(NULL);
             EnumFontFamiliesExW(hDC, &lf, (FONTENUMPROCW)EnumFontSizesProc, 
(LPARAM)SizeList, 0);
             ReleaseDC(NULL, hDC);
@@ -597,7 +635,9 @@ FontSizeChange(
     IN PFONTSIZE_LIST_CTL SizeList,
     IN OUT PCONSOLE_STATE_INFO pConInfo)
 {
-    LONG FontSize, CharWidth, CharHeight;
+    LONG FontSize;
+    UINT CharWidth, CharHeight;
+    HFONT hFont;
     WCHAR szFontSize[100];
 
     /*
@@ -610,35 +650,44 @@ FontSizeChange(
     if (FontSize == 0)
         return FALSE; // We have got an invalid font size...
 
-    CharHeight = (SizeList->UseRasterOrTTList ? (LONG)HIWORD(FontSize) : 
FontSize);
-    CharWidth  = (SizeList->UseRasterOrTTList ? (LONG)LOWORD(FontSize) : 0);
+    /*
+     * For TrueType fonts we set the requested width to zero
+     * so as to obtain a default aspect-ratio width.
+     */
+    CharHeight = (UINT)(SizeList->UseRasterOrTTList ? HIWORD(FontSize) : 
FontSize);
+    CharWidth  = (UINT)(SizeList->UseRasterOrTTList ? LOWORD(FontSize) : 0);
 
-    if (FontPreview.hFont) DeleteObject(FontPreview.hFont);
-    FontPreview.hFont = CreateConsoleFont2(CharHeight, CharWidth, pConInfo);
-    if (FontPreview.hFont == NULL)
+    hFont = CreateConsoleFont2((LONG)CharHeight, (LONG)CharWidth, pConInfo);
+    if (hFont == NULL)
         DPRINT1("FontSizeChange: CreateConsoleFont2() failed\n");
 
     /* Retrieve the real character size in pixels */
-    GetFontCellSize(NULL, FontPreview.hFont, &FontPreview.CharHeight, 
&FontPreview.CharWidth);
+    GetFontCellSize(NULL, hFont, &CharHeight, &CharWidth);
+
+    /*
+     * Update the font preview as well, and store the font handle. It will be
+     * freed at later update or when the font preview is refreshed or reset.
+     * For TrueType fonts, the preview will show the actual character width.
+     */
+    UpdateFontPreview(&FontPreview, hFont, CharWidth, CharHeight);
 
     /*
      * Format:
      * Width  = FontSize.X = LOWORD(FontSize);
      * Height = FontSize.Y = HIWORD(FontSize);
      */
-    pConInfo->FontSize.X = (SHORT)(SizeList->UseRasterOrTTList ? 
FontPreview.CharWidth : 0);
-    pConInfo->FontSize.Y = (SHORT)FontPreview.CharHeight;
+    pConInfo->FontSize.X = (SHORT)(SizeList->UseRasterOrTTList ? CharWidth : 
0);
+    pConInfo->FontSize.Y = (SHORT)CharHeight;
 
-    DPRINT1("pConInfo->FontSize = (%d x %d) ; (CharWidth x CharHeight) = (%d x 
%d)\n",
-            pConInfo->FontSize.X, pConInfo->FontSize.Y,
-            FontPreview.CharWidth, FontPreview.CharHeight);
+    DPRINT("pConInfo->FontSize = (%d x %d) ; (CharWidth x CharHeight) = (%d x 
%d)\n",
+           pConInfo->FontSize.X, pConInfo->FontSize.Y, CharWidth, CharHeight);
 
     InvalidateRect(GetDlgItem(hDlg, IDC_STATIC_FONT_WINDOW_PREVIEW), NULL, 
TRUE);
     InvalidateRect(GetDlgItem(hDlg, IDC_STATIC_SELECT_FONT_PREVIEW), NULL, 
TRUE);
 
-    StringCchPrintfW(szFontSize, ARRAYSIZE(szFontSize), L"%d", 
FontPreview.CharWidth);
+    StringCchPrintfW(szFontSize, ARRAYSIZE(szFontSize), L"%d", CharWidth);
     SetDlgItemText(hDlg, IDC_FONT_SIZE_X, szFontSize);
-    StringCchPrintfW(szFontSize, ARRAYSIZE(szFontSize), L"%d", 
FontPreview.CharHeight);
+    StringCchPrintfW(szFontSize, ARRAYSIZE(szFontSize), L"%d", CharHeight);
     SetDlgItemText(hDlg, IDC_FONT_SIZE_Y, szFontSize);
 
     return TRUE;
@@ -689,22 +738,12 @@ FontProc(HWND hDlg,
 
             /* Select the current font */
             DPRINT1("ConInfo->FaceName = '%S'\n", ConInfo->FaceName);
-            FaceNameList_SelectFaceName(hFontList, ConInfo->FaceName);
-
-            if (ConInfo->FontWeight >= FW_BOLD)
-                CheckDlgButton(hDlg, IDC_CHECK_BOLD_FONTS, BST_CHECKED);
-            else
-                CheckDlgButton(hDlg, IDC_CHECK_BOLD_FONTS, BST_UNCHECKED);
-
-            /* Select the current font size */
-            /*
-             * Format:
-             * Width  = FontSize.X = LOWORD(FontSize);
-             * Height = FontSize.Y = HIWORD(FontSize);
-             */
-            SizeList->CurrentRasterSize = MAKELONG(ConInfo->FontSize.X, 
ConInfo->FontSize.Y);
-            SizeList->CurrentTTSize = ConInfo->FontSize.Y;
-            // FontSizeList_SelectFontSize(SizeList, 
SizeList->CurrentRasterSize);
+            FaceNameList_SelectFont(hDlg, hFontList,
+                                    SizeList,
+                                    ConInfo->FaceName,
+                                    ConInfo->FontFamily,
+                                    ConInfo->FontWeight,
+                                    ConInfo->FontSize);
 
             /* Refresh everything */
             FontTypeChange(hDlg, SizeList, ConInfo);
@@ -737,6 +776,42 @@ FontProc(HWND hDlg,
             break;
         }
 
+#if 0
+        case PSM_QUERYSIBLINGS:
+        {
+            /*
+             * If this is a notification from the "Options" dialog because we
+             * changed the code page, treat it using the WM_FONTCHANGE case,
+             * otherwise ignore it.
+             */
+            if (wParam != IDL_CODEPAGE)
+                return FALSE;
+
+            /* Fall through */
+        }
+#endif
+
+        case WM_FONTCHANGE:
+        {
+            /* The pool of font resources has changed, re-enumerate the fonts 
*/
+            HWND hFontList = GetDlgItem(hDlg, IDC_LBOX_FONTTYPE);
+
+            /* Initialize the font list */
+            FaceNameList_Initialize(hFontList, ConInfo->CodePage);
+
+            /* Select the current font */
+            FaceNameList_SelectFont(hDlg, hFontList,
+                                    SizeList,
+                                    ConInfo->FaceName,
+                                    ConInfo->FontFamily,
+                                    ConInfo->FontWeight,
+                                    ConInfo->FontSize);
+
+            /* Refresh everything */
+            FontTypeChange(hDlg, SizeList, ConInfo);
+            break;
+        }
+
         case WM_NOTIFY:
         {
             switch (((LPNMHDR)lParam)->code)
diff --git a/dll/cpl/console/layout.c b/dll/cpl/console/layout.c
index f8d998625c3..a94bcb53241 100644
--- a/dll/cpl/console/layout.c
+++ b/dll/cpl/console/layout.c
@@ -203,6 +203,9 @@ WinPrev_OnDraw(
      * Compute the console window layout
      */
 
+    if (FontPreview.hFont == NULL)
+        RefreshFontPreview(&FontPreview, pConInfo);
+
     /* We start with the console client area, rescaled for the preview */
     SetRect(&rcWin, 0, 0,
             pConInfo->WindowSize.X * FontPreview.CharWidth,
@@ -489,6 +492,9 @@ PaintText(
     hBrush = CreateSolidBrush(nbkColor);
     if (!hBrush) return;
 
+    if (FontPreview.hFont == NULL)
+        RefreshFontPreview(&FontPreview, pConInfo);
+
     hOldFont = SelectObject(drawItem->hDC, FontPreview.hFont);
     //if (hOldFont == NULL)
     //{
diff --git a/dll/cpl/console/options.c b/dll/cpl/console/options.c
index 689553d516b..e35a2479439 100644
--- a/dll/cpl/console/options.c
+++ b/dll/cpl/console/options.c
@@ -72,7 +72,8 @@ AddCodePage(
 
 static VOID
 BuildCodePageList(
-    IN HWND hDlg)
+    IN HWND hDlg,
+    IN UINT CurrentCodePage)
 {
     LIST_CTL ListCtl;
     HKEY hKey;
@@ -126,7 +127,7 @@ BuildCodePageList(
     AddCodePage(&ListCtl, CP_UTF8);
 
     /* Find and select the current code page in the sorted list */
-    if (BisectListSortedByValue(&ListCtl, ConInfo->CodePage, &CodePage, FALSE) 
== CB_ERR ||
+    if (BisectListSortedByValue(&ListCtl, CurrentCodePage, &CodePage, FALSE) 
== CB_ERR ||
         CodePage == CB_ERR)
     {
         /* Not found, select the first element */
@@ -212,7 +213,7 @@ OptionsProc(HWND hDlg,
     {
         case WM_INITDIALOG:
         {
-            BuildCodePageList(hDlg);
+            BuildCodePageList(hDlg, ConInfo->CodePage);
             UpdateDialogElements(hDlg, ConInfo);
             return TRUE;
         }
@@ -332,6 +333,7 @@ OptionsProc(HWND hDlg,
                 }
             }
             else
+            // (HIWORD(wParam) == CBN_KILLFOCUS)
             if ((HIWORD(wParam) == CBN_SELCHANGE || HIWORD(wParam) == 
CBN_SELENDOK) &&
                 (LOWORD(wParam) == IDL_CODEPAGE))
             {
@@ -347,11 +349,15 @@ OptionsProc(HWND hDlg,
                 if (CodePage == CB_ERR)
                     break;
 
-                ConInfo->CodePage = CodePage;
-
-                /* Change the property sheet state only if the user validated 
*/
-                if (HIWORD(wParam) == CBN_SELENDOK)
+                /* If the user validated a different code page... */
+                if ((HIWORD(wParam) == CBN_SELENDOK) && (CodePage != 
ConInfo->CodePage))
+                {
+                    /* ... update the code page, notify the siblings and 
change the property sheet state */
+                    ConInfo->CodePage = CodePage;
+                    // PropSheet_QuerySiblings(GetParent(hDlg), IDL_CODEPAGE, 
0);
+                    ResetFontPreview(&FontPreview);
                     PropSheet_Changed(GetParent(hDlg), hDlg);
+                }
             }
 
             break;
diff --git a/win32ss/user/winsrv/concfg/font.c 
b/win32ss/user/winsrv/concfg/font.c
index fdc4e15f84b..6aa75b074e0 100644
--- a/win32ss/user/winsrv/concfg/font.c
+++ b/win32ss/user/winsrv/concfg/font.c
@@ -20,6 +20,11 @@
 #include <debug.h>
 
 
+/* GLOBALS 
********************************************************************/
+
+// RTL_STATIC_LIST_HEAD(TTFontCache);
+LIST_ENTRY TTFontCache = {&TTFontCache, &TTFontCache};
+
 /* FUNCTIONS 
******************************************************************/
 
 /* Retrieves the character set associated with a given code page */
@@ -58,7 +63,10 @@ CreateConsoleFontEx(
     lf.lfOutPrecision  = OUT_DEFAULT_PRECIS;
     lf.lfClipPrecision = CLIP_DEFAULT_PRECIS;
     lf.lfQuality = DEFAULT_QUALITY;
-    lf.lfPitchAndFamily = (BYTE)(FIXED_PITCH | FontFamily);
+
+    /* Set the mandatory flags and remove those that we do not support */
+    lf.lfPitchAndFamily = (BYTE)( (FIXED_PITCH | FF_MODERN | FontFamily) &
+                                 ~(VARIABLE_PITCH | FF_DECORATIVE | FF_ROMAN | 
FF_SCRIPT | FF_SWISS));
 
     if (!IsValidConsoleFont(FaceName, CodePage))
         StringCchCopyW(FaceName, LF_FACESIZE, L"Terminal");
@@ -179,13 +187,10 @@ IsValidConsoleFont2(
 {
     LPCWSTR FaceName = lplf->lfFaceName;
 
-    /* Record the font's attributes (Fixedwidth and Truetype) */
-    // BOOL fFixed    = ((lplf->lfPitchAndFamily & 0x03) == FIXED_PITCH);
-    // BOOL fTrueType = (lplf->lfOutPrecision == OUT_STROKE_PRECIS);
-
     /*
-     * According to: http://support.microsoft.com/kb/247815
-     * the criteria for console-eligible fonts are:
+     * According to: 
https://web.archive.org/web/20140901124501/http://support.microsoft.com/kb/247815
+     * "Necessary criteria for fonts to be available in a command window",
+     * the criteria for console-eligible fonts are as follows:
      * - The font must be a fixed-pitch font.
      * - The font cannot be an italic font.
      * - The font cannot have a negative A or C space.
@@ -198,6 +203,10 @@ IsValidConsoleFont2(
      * - If it is not a TrueType font, the face name must be "Terminal".
      * - If it is an Asian TrueType font, it must also be an Asian character 
set.
      *
+     * See also Raymond Chen's blog: 
https://devblogs.microsoft.com/oldnewthing/?p=26843
+     * and MIT-licensed Microsoft Terminal source code: 
https://github.com/microsoft/Terminal/blob/master/src/propsheet/misc.cpp
+     * for other details.
+     *
      * To install additional TrueType fonts to be available for the console,
      * add entries of type REG_SZ named "0", "00" etc... in:
      * HKEY_LOCAL_MACHINE\Software\Microsoft\Windows 
NT\CurrentVersion\Console\TrueTypeFont
@@ -205,27 +214,30 @@ IsValidConsoleFont2(
      * HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Fonts
      */
 
-     /*
-      * In ReactOS we relax some of the criteria:
-      * - We allow fixed-pitch FF_MODERN (Monospace) TrueType fonts
-      *   that can be italic or have negative A or C space.
-      * - If it is not a TrueType font, it can be from another character set
-      *   than OEM_CHARSET.
-      * - We do not look into the magic registry key mentioned above.
-      */
-
-    /* Reject variable width fonts */
-    if (((lplf->lfPitchAndFamily & 0x03) != FIXED_PITCH)
-#if 0 /* Reject italic and TrueType fonts with negative A or C space */
-        || (lplf->lfItalic)
-        || !(lpntm->ntmFlags & NTM_NONNEGATIVE_AC)
+    /*
+     * In ReactOS we relax some of the criteria:
+     * - We allow fixed-pitch FF_MODERN (Monospace) TrueType fonts
+     *   that can be italic or have negative A or C space.
+     * - If it is not a TrueType font, it can be from another character set
+     *   than OEM_CHARSET. When an Asian codepage is active however, we require
+     *   that this non-TrueType font has an Asian character set.
+     */
+
+    /* Reject variable-width fonts ... */
+    if ( ( ((lplf->lfPitchAndFamily & 0x03) != FIXED_PITCH)
+#if 0 /* Reject italic and TrueType fonts with negative A or C space ... */
+           || (lplf->lfItalic)
+           || !(lpntm->ntmFlags & NTM_NONNEGATIVE_AC)
 #endif
-        )
+         ) &&
+        /* ... if they are not in the list of additional TrueType fonts to 
include */
+         !IsAdditionalTTFont(FaceName) )
     {
-        DPRINT1("Font '%S' rejected because it%s (lfPitchAndFamily = %d).\n",
-                FaceName, !(lplf->lfPitchAndFamily & FIXED_PITCH) ? "'s not 
FIXED_PITCH"
-                                                                  : 
(!(lpntm->ntmFlags & NTM_NONNEGATIVE_AC) ? " has negative A or C space"
-                                                                               
                              : " is broken"),
+        DPRINT1("Font '%S' rejected because it%s (lfPitchAndFamily = %d)\n",
+                FaceName,
+                !(lplf->lfPitchAndFamily & FIXED_PITCH) ? "'s not FIXED_PITCH"
+                    : (!(lpntm->ntmFlags & NTM_NONNEGATIVE_AC) ? " has 
negative A or C space"
+                                                               : " is broken"),
                 lplf->lfPitchAndFamily);
         return FALSE;
     }
@@ -238,25 +250,66 @@ IsValidConsoleFont2(
         return FALSE;
     }
 
+    /* Reject vertical fonts (tategaki) */
+    if (FaceName[0] == L'@')
+    {
+        DPRINT1("Font '%S' rejected because it's vertical\n", FaceName);
+        return FALSE;
+    }
+
     /* Is the current code page Chinese, Japanese or Korean? */
     if (IsCJKCodePage(CodePage))
     {
-        /* It's Asian */
+        /* It's CJK */
+
         if (FontType == TRUETYPE_FONTTYPE)
         {
-            if (lplf->lfCharSet != CodePageToCharSet(CodePage))
+            /*
+             * Here we are inclusive and check for any CJK character set,
+             * instead of looking just at the current one via 
CodePageToCharSet().
+             */
+            if (!IsCJKCharSet(lplf->lfCharSet)
+#if 1 // FIXME: Temporary HACK!
+                && wcscmp(FaceName, L"Terminal") != 0
+#endif
+               )
             {
-                DPRINT1("TrueType font '%S' rejected because it's not user 
Asian charset (lfCharSet = %d)\n",
+                DPRINT1("TrueType font '%S' rejected because it's not Asian 
charset (lfCharSet = %d)\n",
+                        FaceName, lplf->lfCharSet);
+                return FALSE;
+            }
+
+            /*
+             * If this is a cached TrueType font that is used only for certain
+             * code pages, verify that the charset it claims is the correct 
one.
+             *
+             * Since there may be multiple entries for a cached TrueType font,
+             * a general one (code page == 0) and one or more for explicit
+             * code pages, we need to perform two search queries instead of
+             * just one and retrieving the code page for this entry.
+             */
+            if (IsAdditionalTTFont(FaceName) && 
!IsAdditionalTTFontCP(FaceName, 0) &&
+                !IsCJKCharSet(lplf->lfCharSet))
+            {
+                DPRINT1("Cached TrueType font '%S' rejected because it claims 
a code page that is not Asian charset (lfCharSet = %d)\n",
                         FaceName, lplf->lfCharSet);
                 return FALSE;
             }
         }
         else
         {
+            /* Reject non-TrueType fonts that do not have an Asian character 
set */
+            if (!IsCJKCharSet(lplf->lfCharSet) && (lplf->lfCharSet != 
OEM_CHARSET))
+            {
+                DPRINT1("Non-TrueType font '%S' rejected because it's not 
Asian charset or OEM_CHARSET (lfCharSet = %d)\n",
+                        FaceName, lplf->lfCharSet);
+                return FALSE;
+            }
+
             /* Reject non-TrueType fonts that are not Terminal */
             if (wcscmp(FaceName, L"Terminal") != 0)
             {
-                DPRINT1("Non-TrueType font '%S' rejected because it's not 
Terminal\n", FaceName);
+                DPRINT1("Non-TrueType font '%S' rejected because it's not 
'Terminal'\n", FaceName);
                 return FALSE;
             }
         }
@@ -264,6 +317,8 @@ IsValidConsoleFont2(
     else
     {
         /* Not CJK */
+
+        /* Reject non-TrueType fonts that are not OEM or similar */
         if ((FontType != TRUETYPE_FONTTYPE) &&
             (lplf->lfCharSet != ANSI_CHARSET) &&
             (lplf->lfCharSet != DEFAULT_CHARSET) &&
@@ -275,13 +330,6 @@ IsValidConsoleFont2(
         }
     }
 
-    /* Reject fonts that are vertical (tategaki) */
-    if (FaceName[0] == L'@')
-    {
-        DPRINT1("Font '%S' rejected because it's vertical\n", FaceName);
-        return FALSE;
-    }
-
     /* All good */
     return TRUE;
 }
@@ -320,7 +368,7 @@ IsValidConsoleFont(
 
     RtlZeroMemory(&lf, sizeof(lf));
     lf.lfCharSet = DEFAULT_CHARSET; // CodePageToCharSet(CodePage);
-    // lf.lfPitchAndFamily = FIXED_PITCH | FF_DONTCARE;
+    // lf.lfPitchAndFamily = FIXED_PITCH | FF_MODERN;
     StringCchCopyW(lf.lfFaceName, ARRAYSIZE(lf.lfFaceName), FaceName);
 
     hDC = GetDC(NULL);
@@ -330,4 +378,173 @@ IsValidConsoleFont(
     return Param.IsValidFont;
 }
 
+/*
+ * To install additional TrueType fonts to be available for the console,
+ * add entries of type REG_SZ named "0", "00" etc... in:
+ * HKEY_LOCAL_MACHINE\Software\Microsoft\Windows 
NT\CurrentVersion\Console\TrueTypeFont
+ * The names of the fonts listed there should match those in:
+ * HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Fonts
+ *
+ * This function initializes the cache of the fonts listed there.
+ */
+VOID
+InitTTFontCache(VOID)
+{
+    BOOLEAN Success;
+    HKEY  hKeyTTFonts; // hKey;
+    DWORD dwNumValues = 0;
+    DWORD dwIndex;
+    DWORD dwType;
+    WCHAR szValueName[MAX_PATH];
+    DWORD dwValueName;
+    WCHAR szValue[LF_FACESIZE] = L"";
+    DWORD dwValue;
+    PTT_FONT_ENTRY FontEntry;
+    PWCHAR pszNext = NULL;
+    UINT CodePage;
+
+    if (!IsListEmpty(&TTFontCache))
+        return;
+    // InitializeListHead(&TTFontCache);
+
+    /* Open the key */
+    // "\\Registry\\Machine\\Software\\Microsoft\\Windows 
NT\\CurrentVersion\\Console\\TrueTypeFont"
+    Success = (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
+                             L"Software\\Microsoft\\Windows 
NT\\CurrentVersion\\Console\\TrueTypeFont",
+                             0,
+                             KEY_READ,
+                             &hKeyTTFonts) == ERROR_SUCCESS);
+    if (!Success)
+        return;
+
+    /* Enumerate each value */
+    if (RegQueryInfoKeyW(hKeyTTFonts, NULL, NULL, NULL, NULL, NULL, NULL,
+                         &dwNumValues, NULL, NULL, NULL, NULL) != 
ERROR_SUCCESS)
+    {
+        DPRINT("ConCfgReadUserSettings: RegQueryInfoKeyW failed\n");
+        RegCloseKey(hKeyTTFonts);
+        return;
+    }
+
+    for (dwIndex = 0; dwIndex < dwNumValues; dwIndex++)
+    {
+        dwValue = sizeof(szValue);
+        dwValueName = ARRAYSIZE(szValueName);
+        if (RegEnumValueW(hKeyTTFonts, dwIndex, szValueName, &dwValueName, 
NULL, &dwType, (BYTE*)szValue, &dwValue) != ERROR_SUCCESS)
+        {
+            DPRINT1("InitTTFontCache: RegEnumValueW failed, continuing...\n");
+            continue;
+        }
+        /* Only (multi-)string values are supported */
+        if ((dwType != REG_SZ) && (dwType != REG_MULTI_SZ))
+            continue;
+
+        /* The value name is a code page (in decimal), validate it */
+        CodePage = wcstoul(szValueName, &pszNext, 10);
+        if (*pszNext)
+            continue; // Non-numerical garbage followed...
+        // IsValidCodePage(CodePage);
+
+        FontEntry = RtlAllocateHeap(RtlGetProcessHeap(), HEAP_ZERO_MEMORY, 
sizeof(*FontEntry));
+        if (!FontEntry)
+        {
+            DPRINT1("InitTTFontCache: Failed to allocate memory, 
continuing...\n");
+            continue;
+        }
+
+        FontEntry->CodePage = CodePage;
+
+        pszNext = szValue;
+
+        /* Check whether bold is disabled for this font */
+        if (*pszNext == L'*')
+        {
+            FontEntry->DisableBold = TRUE;
+            ++pszNext;
+        }
+        else
+        {
+            FontEntry->DisableBold = FALSE;
+        }
+
+        /* Copy the font name */
+        StringCchCopyNW(FontEntry->FaceName, ARRAYSIZE(FontEntry->FaceName),
+                        pszNext, wcslen(pszNext));
+
+        if (dwType == REG_MULTI_SZ)
+        {
+            /* There may be an alternate face name as the second string */
+            pszNext += wcslen(pszNext) + 1;
+
+            /* Check whether bold is disabled for this font */
+            if (*pszNext == L'*')
+            {
+                FontEntry->DisableBold = TRUE;
+                ++pszNext;
+            }
+            // else, keep the original setting.
+
+            /* Copy the alternate font name */
+            StringCchCopyNW(FontEntry->FaceNameAlt, 
ARRAYSIZE(FontEntry->FaceNameAlt),
+                            pszNext, wcslen(pszNext));
+        }
+
+        InsertTailList(&TTFontCache, &FontEntry->Entry);
+    }
+
+    /* Close the key and quit */
+    RegCloseKey(hKeyTTFonts);
+}
+
+VOID
+ClearTTFontCache(VOID)
+{
+    PLIST_ENTRY Entry;
+    PTT_FONT_ENTRY FontEntry;
+
+    while (!IsListEmpty(&TTFontCache))
+    {
+        Entry = RemoveHeadList(&TTFontCache);
+        FontEntry = CONTAINING_RECORD(Entry, TT_FONT_ENTRY, Entry);
+        RtlFreeHeap(RtlGetProcessHeap(), 0, FontEntry);
+    }
+    InitializeListHead(&TTFontCache);
+}
+
+VOID
+RefreshTTFontCache(VOID)
+{
+    ClearTTFontCache();
+    InitTTFontCache();
+}
+
+PTT_FONT_ENTRY
+FindCachedTTFont(
+    IN LPCWSTR FaceName,
+    IN UINT CodePage)
+{
+    PLIST_ENTRY Entry;
+    PTT_FONT_ENTRY FontEntry;
+
+    /* Search for the font in the cache */
+    for (Entry = TTFontCache.Flink;
+         Entry != &TTFontCache;
+         Entry = Entry->Flink)
+    {
+        FontEntry = CONTAINING_RECORD(Entry, TT_FONT_ENTRY, Entry);
+
+        /* NOTE: The font face names are case-sensitive */
+        if ((wcscmp(FontEntry->FaceName   , FaceName) == 0) ||
+            (wcscmp(FontEntry->FaceNameAlt, FaceName) == 0))
+        {
+            /* Return a match if we don't look at the code pages, or when they 
match */
+            if ((CodePage == INVALID_CP) || (CodePage == FontEntry->CodePage))
+            {
+                return FontEntry;
+            }
+        }
+    }
+    return NULL;
+}
+
 /* EOF */
diff --git a/win32ss/user/winsrv/concfg/font.h 
b/win32ss/user/winsrv/concfg/font.h
index 09b076aff80..ee1aa92736c 100644
--- a/win32ss/user/winsrv/concfg/font.h
+++ b/win32ss/user/winsrv/concfg/font.h
@@ -11,16 +11,50 @@
 
 /* DEFINES 
********************************************************************/
 
-#define CP_SHIFTJIS 932 // Japanese Shift-JIS
-#define CP_HANGUL   949 // Korean Hangul
-#define CP_GB2312   936 // Chinese Simplified (GB2312)
-#define CP_BIG5     950 // Chinese Traditional (Big5)
+#define INVALID_CP  ((UINT)-1)
+
+#define CP_SHIFTJIS 932  // Japanese Shift-JIS
+#define CP_HANGUL   949  // Korean Hangul/Wansung
+#define CP_JOHAB    1361 // Korean Johab
+#define CP_GB2312   936  // Chinese Simplified (GB2312)
+#define CP_BIG5     950  // Chinese Traditional (Big5)
 
 /* IsFarEastCP(CodePage) */
 #define IsCJKCodePage(CodePage) \
     ((CodePage) == CP_SHIFTJIS || (CodePage) == CP_HANGUL || \
+  /* (CodePage) == CP_JOHAB || */ \
      (CodePage) == CP_BIG5     || (CodePage) == CP_GB2312)
 
+#if !defined(_WINGDI_) || defined(NOGDI)
+#define SHIFTJIS_CHARSET    128
+#define HANGEUL_CHARSET     129
+#define HANGUL_CHARSET      129 // HANGEUL_CHARSET
+#if(WINVER >= 0x0400)
+#define JOHAB_CHARSET       130
+#endif /* WINVER */
+#define GB2312_CHARSET      134
+#define CHINESEBIG5_CHARSET 136
+#endif /* !defined(_WINGDI_) || defined(NOGDI) */
+
+/* IsAnyDBCSCharSet(CharSet) */
+#define IsCJKCharSet(CharSet)   \
+    ((CharSet) == SHIFTJIS_CHARSET || (CharSet) == HANGUL_CHARSET || \
+  /* (CharSet) == JOHAB_CHARSET || */ \
+     (CharSet) == GB2312_CHARSET   || (CharSet) == CHINESEBIG5_CHARSET)
+
+#define IsBoldFont(Weight)  \
+    ((Weight) >= FW_SEMIBOLD) /* Sometimes, just > FW_MEDIUM */
+
+typedef struct _TT_FONT_ENTRY
+{
+    LIST_ENTRY Entry;
+    UINT CodePage;
+    BOOL DisableBold;
+    WCHAR FaceName[LF_FACESIZE];
+    WCHAR FaceNameAlt[LF_FACESIZE];
+} TT_FONT_ENTRY, *PTT_FONT_ENTRY;
+
+
 /* FUNCTIONS 
******************************************************************/
 
 BYTE
@@ -65,4 +99,33 @@ IsValidConsoleFont(
     IN LPCWSTR FaceName,
     IN UINT CodePage);
 
+/*
+ * To install additional TrueType fonts to be available for the console,
+ * add entries of type REG_SZ named "0", "00" etc... in:
+ * HKEY_LOCAL_MACHINE\Software\Microsoft\Windows 
NT\CurrentVersion\Console\TrueTypeFont
+ * The names of the fonts listed there should match those in:
+ * HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Fonts
+ *
+ * This function initializes the cache of the fonts listed there.
+ */
+VOID
+InitTTFontCache(VOID);
+
+VOID
+ClearTTFontCache(VOID);
+
+VOID
+RefreshTTFontCache(VOID);
+
+PTT_FONT_ENTRY
+FindCachedTTFont(
+    IN LPCWSTR FaceName,
+    IN UINT CodePage);
+
+#define IsAdditionalTTFont(FaceName) \
+    (FindCachedTTFont((FaceName), INVALID_CP) != NULL)
+
+#define IsAdditionalTTFontCP(FaceName, CodePage) \
+    (FindCachedTTFont((FaceName), (CodePage)) != NULL)
+
 /* EOF */
diff --git a/win32ss/user/winsrv/concfg/precomp.h 
b/win32ss/user/winsrv/concfg/precomp.h
index cfa0519ca36..5f6ca4a333e 100644
--- a/win32ss/user/winsrv/concfg/precomp.h
+++ b/win32ss/user/winsrv/concfg/precomp.h
@@ -31,7 +31,8 @@
 #include <ndk/obfuncs.h>
 #include <ndk/rtlfuncs.h>
 
-#include <stdio.h> // for swprintf
+#include <stdio.h>  // For swprintf()
+#include <stdlib.h> // For wcstoul()
 #include <strsafe.h>
 
 /* EOF */
diff --git a/win32ss/user/winsrv/consrv/frontends/gui/conwnd.c 
b/win32ss/user/winsrv/consrv/frontends/gui/conwnd.c
index 13f5da948f8..6e3603fa360 100644
--- a/win32ss/user/winsrv/consrv/frontends/gui/conwnd.c
+++ b/win32ss/user/winsrv/consrv/frontends/gui/conwnd.c
@@ -20,7 +20,7 @@
 #define NDEBUG
 #include <debug.h>
 
-#include "font.h"
+#include "concfg/font.h"
 #include "guiterm.h"
 #include "resource.h"
 
diff --git a/win32ss/user/winsrv/consrv/frontends/gui/guisettings.c 
b/win32ss/user/winsrv/consrv/frontends/gui/guisettings.c
index e317334790a..1ccee596beb 100644
--- a/win32ss/user/winsrv/consrv/frontends/gui/guisettings.c
+++ b/win32ss/user/winsrv/consrv/frontends/gui/guisettings.c
@@ -14,6 +14,7 @@
 #define NDEBUG
 #include <debug.h>
 
+#include "concfg/font.h"
 #include "guiterm.h"
 #include "guisettings.h"
 
@@ -331,7 +332,8 @@ GuiApplyUserSettings(PGUI_CONSOLE_DATA GuiData,
 
         /* Set the terminal informations */
 
-        /* Change the font */
+        /* Refresh the additional TrueType fonts cache and change the font */
+        RefreshTTFontCache();
         InitFonts(GuiData,
                   pConInfo->FaceName,
                   pConInfo->FontFamily,
diff --git a/win32ss/user/winsrv/consrv/frontends/gui/guiterm.c 
b/win32ss/user/winsrv/consrv/frontends/gui/guiterm.c
index c811c4c030c..afca59331a6 100644
--- a/win32ss/user/winsrv/consrv/frontends/gui/guiterm.c
+++ b/win32ss/user/winsrv/consrv/frontends/gui/guiterm.c
@@ -16,6 +16,7 @@
 #define NDEBUG
 #include <debug.h>
 
+#include "concfg/font.h"
 #include "guiterm.h"
 #include "resource.h"
 
@@ -294,12 +295,15 @@ GuiInit(IN PCONSOLE_INIT_INFO ConsoleInitInfo,
     HANDLE hInputThread;
     CLIENT_ID ClientId;
 
-    /*
-     * Initialize and register the console window class, if needed.
-     */
+    /* Perform one-time initialization */
     if (!ConsInitialized)
     {
+        /* Initialize and register the console window class */
         if (!RegisterConWndClass(ConSrvDllInstance)) return FALSE;
+
+        /* Initialize the font support -- additional TrueType fonts cache */
+        InitTTFontCache();
+
         ConsInitialized = TRUE;
     }
 

Reply via email to