Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

477 lines
18 KiB

  1. /*****************************************************************************\
  2. FILE: fontfix.cpp
  3. DESCRIPTION:
  4. This file will implement an API: FixFontsOnLanguageChange().
  5. The USER32 or Regional Settings code should own this API. The fact that it's
  6. in the shell is a hack and it should be moved to USER32. This font will
  7. be called when the MUI language changes so the fonts in the system metrics
  8. can be changed to valid values for the language.
  9. Contacts: EdwardP - International Font PM.
  10. Sankar ?/??/???? - Created for Win2k or before in desk.cpl.
  11. BryanSt 3/24/2000 - Make to be modular so it can be moved back into USER32.
  12. Made the code more robust. Removed creating custom appearance
  13. schemes in order to be compatible with new .theme and
  14. .msstyles support.
  15. Copyright (C) Microsoft Corp 2000-2000. All rights reserved.
  16. \*****************************************************************************/
  17. #include "priv.h"
  18. #include "AdvAppearPg.h"
  19. #include "fontfix.h"
  20. #define SZ_DEFAULT_FONT TEXT("Tahoma")
  21. /////////////////////////////////////////////////////////////////////
  22. // Private Functions
  23. /////////////////////////////////////////////////////////////////////
  24. BOOL FontFix_ReadCharsets(UINT uiCharsets[], int iCount)
  25. {
  26. HKEY hkAppearance;
  27. BOOL fSuccess = FALSE;
  28. if (RegOpenKeyEx(HKEY_CURRENT_USER, SZ_REGKEY_APPEARANCE, 0, KEY_READ, &hkAppearance) == ERROR_SUCCESS)
  29. {
  30. DWORD dwType = REG_BINARY;
  31. DWORD dwSize = iCount * sizeof(UINT);
  32. if (RegQueryValueEx(hkAppearance, SZ_REGVALUE_RECENTFOURCHARSETS, NULL, &dwType, (LPBYTE)uiCharsets, &dwSize) == ERROR_SUCCESS)
  33. fSuccess = TRUE;
  34. RegCloseKey(hkAppearance);
  35. }
  36. return fSuccess;
  37. }
  38. BOOL FontFix_SaveCharsets(UINT uiCharsets[], int iCount)
  39. {
  40. HKEY hkAppearance;
  41. BOOL fSuccess = FALSE;
  42. if(RegCreateKeyEx(HKEY_CURRENT_USER, SZ_REGKEY_APPEARANCE, 0, TEXT(""), 0, KEY_WRITE, NULL, &hkAppearance, NULL) == ERROR_SUCCESS)
  43. {
  44. if(RegSetValueEx(hkAppearance, SZ_REGVALUE_RECENTFOURCHARSETS, 0, REG_BINARY, (LPBYTE)uiCharsets, iCount * sizeof(UINT)) == ERROR_SUCCESS)
  45. fSuccess = TRUE;
  46. RegCloseKey(hkAppearance);
  47. }
  48. return fSuccess;
  49. }
  50. void FontFix_GetDefaultFontName(LPTSTR pszDefFontName, DWORD cchSize)
  51. {
  52. HKEY hkDefFont;
  53. //Value is not there in the registry; Use "Tahoma" as the default name.
  54. StrCpyN(pszDefFontName, SZ_DEFAULT_FONT, cchSize);
  55. // Read the "DefaultFontName" to be used.
  56. if (RegOpenKeyEx(HKEY_LOCAL_MACHINE,
  57. SZ_APPEARANCE_SCHEMES,
  58. 0,
  59. KEY_READ,
  60. &hkDefFont) == ERROR_SUCCESS)
  61. {
  62. DWORD dwType = REG_SZ;
  63. DWORD cbSize = (cchSize * sizeof(pszDefFontName[0]));
  64. if (RegQueryValueEx(hkDefFont,
  65. SZ_REGVALUE_DEFAULTFONTNAME,
  66. NULL,
  67. &dwType,
  68. (LPBYTE) pszDefFontName,
  69. &cbSize) != ERROR_SUCCESS)
  70. {
  71. // We already set a fallback value.
  72. }
  73. RegCloseKey(hkDefFont);
  74. }
  75. }
  76. BOOL FontFix_DoesFontSupportAllCharsets(HDC hdc, LPLOGFONT plf, UINT uiUniqueCharsets[], int iCountUniqueCharsets)
  77. {
  78. int j;
  79. //The given font supports the system charset; Let's check if it supports the other charsets
  80. for (j = 0; j < iCountUniqueCharsets; j++)
  81. {
  82. plf->lfCharSet = (BYTE)uiUniqueCharsets[j]; //Let's try the next charset in the array.
  83. if (EnumFontFamiliesEx(hdc, plf, (FONTENUMPROC)Font_EnumValidCharsets, (LPARAM)0, 0) != 0)
  84. {
  85. // EnumFontFamiliesEx would have returned a zero if Font_EnumValidCharsets was called
  86. // even once. In other words, it returned a non-zero because not even a single font existed
  87. // that supported the given charset.
  88. return FALSE;
  89. }
  90. }
  91. return TRUE; //Yes this font supports all the charsets we are interested in.
  92. }
  93. // Given an array of fonts and an array of unique charsets, this function checks if the fonts
  94. // support ALL these charsets.
  95. // Returns TRUE if these fonts support all the charsets.
  96. // In all other cases, this function will return TRUE. If the fonts need to be changed to support
  97. // the given charsets, this function does all those changes.
  98. //
  99. // lpszName is the name of the scheme to be used in the MessageBox that appears if fSilent is FALSE.
  100. BOOL FontFix_CheckFontsCharsets(LOGFONT lfUIFonts[], int iCountFonts,
  101. UINT uiCurUniqueCharsets[], int iCountCurUniqueCharsets,
  102. BOOL *pfDirty, LPCTSTR lpszName)
  103. {
  104. int i;
  105. TCHAR szDefaultFontFaceName[LF_FACESIZE];
  106. HDC hdc;
  107. *pfDirty = FALSE; //Assume that this scheme does not need to be saved.
  108. //Read the default font name from the registry (Mostly: Tahoma)
  109. FontFix_GetDefaultFontName(szDefaultFontFaceName, ARRAYSIZE(szDefaultFontFaceName));
  110. hdc = GetDC(NULL);
  111. //Check to see of the fonts support the system charset
  112. for (i = 0; i < iCountFonts; i++)
  113. {
  114. //Save the current charset because FontFix_DoesFontSupportAllCharsets() destroys this field.
  115. BYTE bCurCharset = lfUIFonts[i].lfCharSet;
  116. if (!FontFix_DoesFontSupportAllCharsets(hdc, &lfUIFonts[i], uiCurUniqueCharsets, iCountCurUniqueCharsets))
  117. {
  118. //Copy the default fontname to the font.
  119. lstrcpy(lfUIFonts[i].lfFaceName, szDefaultFontFaceName);
  120. *pfDirty = TRUE; //This scheme needs to be saved.
  121. }
  122. //Restore the charset because FontFix_DoesFontSupportAllCharsets() destroyed this field.
  123. lfUIFonts[i].lfCharSet = bCurCharset; // Restore the current charset.
  124. // Warning #1: The IconTitle font's Charset must always match the System Locale charset.
  125. // Warning #2: FoxPro's tooltips code expects the Status font's charset to the the System
  126. // Locale's charset.
  127. // As per intl guys, we set the charset of all the UI fonts to SYSTEM_LOCALE_CHARSET.
  128. if (lfUIFonts[i].lfCharSet != uiCurUniqueCharsets[SYSTEM_LOCALE_CHARSET])
  129. {
  130. lfUIFonts[i].lfCharSet = (BYTE)uiCurUniqueCharsets[SYSTEM_LOCALE_CHARSET];
  131. *pfDirty = TRUE;
  132. }
  133. } //For loop
  134. ReleaseDC(NULL, hdc);
  135. return TRUE; //The fonts have been modified as required.
  136. }
  137. void FontFix_GetUIFonts(NONCLIENTMETRICS *pncm, LOGFONT lfUIFonts[])
  138. {
  139. pncm->cbSize = sizeof(NONCLIENTMETRICS);
  140. ClassicSystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(NONCLIENTMETRICS),
  141. (void far *)(LPNONCLIENTMETRICS)pncm, FALSE);
  142. // Read the icon title font directly into the font array.
  143. ClassicSystemParametersInfo(SPI_GETICONTITLELOGFONT, sizeof(LOGFONT),
  144. (void far *)(LPLOGFONT)&(lfUIFonts[FONT_ICONTITLE]), FALSE);
  145. //Make a copy of the ncm fonts into fonts array.
  146. LF32toLF(&(pncm->lfCaptionFont), &(lfUIFonts[FONT_CAPTION]));
  147. LF32toLF(&(pncm->lfSmCaptionFont), &(lfUIFonts[FONT_SMCAPTION]));
  148. LF32toLF(&(pncm->lfMenuFont), &(lfUIFonts[FONT_MENU]));
  149. LF32toLF(&(pncm->lfStatusFont), &(lfUIFonts[FONT_STATUS]));
  150. LF32toLF(&(pncm->lfMessageFont), &(lfUIFonts[FONT_MSGBOX]));
  151. }
  152. void FontFix_SetUIFonts(NONCLIENTMETRICS *pncm, LOGFONT lfUIFonts[])
  153. {
  154. //Copy all fonts back into the ncm structure.
  155. LFtoLF32(&(lfUIFonts[FONT_CAPTION]), &(pncm->lfCaptionFont));
  156. LFtoLF32(&(lfUIFonts[FONT_SMCAPTION]), &(pncm->lfSmCaptionFont));
  157. LFtoLF32(&(lfUIFonts[FONT_MENU]), &(pncm->lfMenuFont));
  158. LFtoLF32(&(lfUIFonts[FONT_STATUS]), &(pncm->lfStatusFont));
  159. LFtoLF32(&(lfUIFonts[FONT_MSGBOX]), &(pncm->lfMessageFont));
  160. // FEATURE: Do we want a WININICHANGE HERE?
  161. // NOTE: We want to set a SPIF_SENDWININICHANGE, because we want to refresh.
  162. // Note we don't do this async. This should only happen when the user changes MUI languages,
  163. // and in that case, perf can suck.
  164. TraceMsg(TF_GENERAL, "desk.cpl: Calling SPI_SETNONCLIENTMETRICS");
  165. ClassicSystemParametersInfo(SPI_SETNONCLIENTMETRICS, sizeof(*pncm), (void far *)(LPNONCLIENTMETRICS)pncm, (SPIF_UPDATEINIFILE | SPIF_SENDWININICHANGE));
  166. TraceMsg(TF_GENERAL,"desk.cpl: Calling SPI_SETICONTITLELOGFONT");
  167. ClassicSystemParametersInfo(SPI_SETICONTITLELOGFONT, sizeof(LOGFONT),
  168. (void far *)(LPLOGFONT)&lfUIFonts[FONT_ICONTITLE], (SPIF_UPDATEINIFILE | SPIF_SENDWININICHANGE));
  169. }
  170. // ------------------------ manage system settings --------------------------
  171. // Given the Locale ID, this returns the corresponding charset
  172. UINT GetCharsetFromLCID(LCID lcid)
  173. {
  174. TCHAR szData[6+1]; // 6 chars are max allowed for this lctype
  175. UINT uiRet;
  176. DWORD dwError = 0;
  177. if (GetLocaleInfo(lcid, LOCALE_IDEFAULTANSICODEPAGE, szData, ARRAYSIZE(szData)) > 0)
  178. {
  179. UINT uiCp = (UINT)StrToInt(szData);
  180. CHARSETINFO csinfo = {0};
  181. if (!TranslateCharsetInfo((DWORD *)IntToPtr(uiCp), &csinfo, TCI_SRCCODEPAGE))
  182. {
  183. dwError = GetLastError();
  184. uiRet = DEFAULT_CHARSET;
  185. }
  186. else
  187. {
  188. uiRet = csinfo.ciCharset;
  189. }
  190. }
  191. else
  192. {
  193. // at worst non penalty for charset
  194. dwError = GetLastError();
  195. uiRet = DEFAULT_CHARSET;
  196. }
  197. return uiRet;
  198. }
  199. int FontFix_CompareUniqueCharsets(UINT uiCharset1[], int iCount1, UINT uiCharset2[], int iCount2)
  200. {
  201. if (iCount1 == iCount2)
  202. {
  203. int i, j;
  204. // The first items in the array is SYSTEM CHAR SET; It must match because system locale's
  205. // charset is always used by the Icon Title font; Icon Title font's charset is used by
  206. // comctl32 to do A/W conversion. In order that all ANSI applications run correctly,
  207. // the icon charset must always match current system locale.
  208. if (uiCharset1[SYSTEM_LOCALE_CHARSET] != uiCharset2[SYSTEM_LOCALE_CHARSET])
  209. return -1;
  210. //Now see if the arrays have the same elements.
  211. ASSERT(SYSTEM_LOCALE_CHARSET == 0);
  212. for (i = SYSTEM_LOCALE_CHARSET+1; i < iCount1; i++)
  213. {
  214. for (j = SYSTEM_LOCALE_CHARSET+1; j < iCount2; j++)
  215. {
  216. if(uiCharset1[i] == uiCharset2[j])
  217. break;
  218. }
  219. if (j == iCount2)
  220. return -1; // uiCharset1[i] is not found in the second array.
  221. }
  222. }
  223. return (iCount1 - iCount2); // Both the arrays have the same Charsets
  224. }
  225. // Given the Language ID, this gets the charset.
  226. UINT GetCharsetFromLang(LANGID wLang)
  227. {
  228. return(GetCharsetFromLCID(MAKELCID(wLang, SORT_DEFAULT)));
  229. }
  230. /////////////////////////////////////////////////////////////////////
  231. // Public Functions
  232. /////////////////////////////////////////////////////////////////////
  233. void Font_GetCurrentCharsets(UINT uiCharsets[], int iCount)
  234. {
  235. LCID lcid;
  236. LANGID langID;
  237. ASSERT(iCount == MAX_CHARSETS);
  238. // Get all the four charsets we are interested in.
  239. uiCharsets[0] = GetCharsetFromLCID(lcid = GetSystemDefaultLCID());
  240. AssertMsg(lcid, TEXT("GetSystemDefaultLCID() failed with %d"), GetLastError());
  241. uiCharsets[1] = GetCharsetFromLCID(lcid = GetUserDefaultLCID());
  242. AssertMsg(lcid, TEXT("GetUserDefaultLCID() failed with %d"), GetLastError());
  243. uiCharsets[2] = GetCharsetFromLang(langID = GetSystemDefaultUILanguage());
  244. AssertMsg(langID, TEXT("GetSystemDefaultUILanguage() failed with %d"), GetLastError());
  245. uiCharsets[3] = GetCharsetFromLang(langID = GetUserDefaultUILanguage());
  246. AssertMsg(langID, TEXT("GetUserDefaultUILanguage() failed with %d"), GetLastError());
  247. }
  248. void Font_GetUniqueCharsets(UINT uiCharsets[], UINT uiUniqueCharsets[], int iMaxCount, int *piCountUniqueCharsets)
  249. {
  250. int i, j;
  251. // Find the unique Charsets;
  252. *piCountUniqueCharsets = 0;
  253. for (i = 0; i < iMaxCount; i++)
  254. {
  255. uiUniqueCharsets[i] = DEFAULT_CHARSET; //Initialize it to default charset.
  256. for (j = 0; j < *piCountUniqueCharsets; j++)
  257. {
  258. if (uiUniqueCharsets[j] == uiCharsets[i])
  259. break; // This Charset is already in the array
  260. }
  261. if (j == *piCountUniqueCharsets)
  262. {
  263. // Yes! It is a unique char set; Save it!
  264. uiUniqueCharsets[j] = uiCharsets[i];
  265. (*piCountUniqueCharsets)++; //One more unique char set found.
  266. }
  267. }
  268. }
  269. int CALLBACK Font_EnumValidCharsets(LPENUMLOGFONTEX lpelf, LPNEWTEXTMETRIC lpntm, DWORD Type, LPARAM lData)
  270. {
  271. // The purpose of this function is to determine if a font supports a particular charset;
  272. // If this callback gets called even once, then that means that this font supports the given
  273. // charset. There is no need to enumerate all the other styles. We immediately return zero to
  274. // stop the enumeration. Since we return zero, the EnumFontFamiliesEx() also returns zero in
  275. // this case and that return value is used to determine if a given font supports a given
  276. // charset.
  277. return 0;
  278. }
  279. HRESULT FontFix_FixNonClientFonts(LOGFONT lfUIFonts[])
  280. {
  281. UINT uiCurCharsets[MAX_CHARSETS];
  282. UINT uiRegCharsets[MAX_CHARSETS];
  283. UINT uiCurUniqueCharsets[MAX_CHARSETS];
  284. int iCountCurUniqueCharsets = 0;
  285. UINT uiRegUniqueCharsets[MAX_CHARSETS];
  286. int iCountRegUniqueCharsets = 0;
  287. BOOL fRegCharsetsValid = FALSE;
  288. HRESULT hr = S_OK;
  289. // Get the current four charsets from system.
  290. Font_GetCurrentCharsets(uiCurCharsets, MAX_CHARSETS);
  291. //Get the charsets saved in the registry.
  292. fRegCharsetsValid = FontFix_ReadCharsets(uiRegCharsets, MAX_CHARSETS);
  293. // Get rid of the duplicate charsets and get only the unique Charsets from these arrays.
  294. Font_GetUniqueCharsets(uiCurCharsets, uiCurUniqueCharsets, MAX_CHARSETS, &iCountCurUniqueCharsets);
  295. if (fRegCharsetsValid)
  296. Font_GetUniqueCharsets(uiRegCharsets, uiRegUniqueCharsets, MAX_CHARSETS, &iCountRegUniqueCharsets);
  297. // Check if these two arrays have the same charsets.
  298. if (!fRegCharsetsValid || !(FontFix_CompareUniqueCharsets(uiCurUniqueCharsets, iCountCurUniqueCharsets, uiRegUniqueCharsets, iCountRegUniqueCharsets) == 0))
  299. {
  300. BOOL fDirty = FALSE;
  301. FontFix_CheckFontsCharsets(lfUIFonts, NUM_FONTS, uiCurUniqueCharsets, iCountCurUniqueCharsets, &fDirty, TEXT(""));
  302. // Save the cur charsets into the registry.
  303. FontFix_SaveCharsets(uiCurCharsets, MAX_CHARSETS);
  304. hr = (fDirty ? S_OK : S_FALSE);
  305. }
  306. else
  307. {
  308. hr = S_FALSE; // The charsets are the same; Nothing to do!
  309. }
  310. return hr; //Yes! We had to do some updates in connection with charsets and fonts.
  311. }
  312. //------------------------------------------------------------------------------------------------
  313. //
  314. // This is the function that needs to be called everytime a locale changes (or could have changed).
  315. //
  316. // It does the following:
  317. // 1. It checks if any of the four the charset settings has changed.
  318. // 2. If some charset has changed, it makes the corresponding changes in the 6 UI fonts.
  319. // 3. If a scheme is selected, it checks to see if the fonts support the new charsets and if not
  320. // changes the fonts and/or charsets and saves the scheme under a new name.
  321. // 4. Saves the new charsets in the registry sothat we don't have to do the same everytime this
  322. // function is called.
  323. //
  324. // NOTE: This is a private export from desk.cpl. This is called in two places:
  325. // 1. from regional control panel whenever it is run AND whenever it changes some locale.
  326. // 2. from desk.cpl itself whenever "Appearance" tab is created. This is required because it is
  327. // possible that an admin changes a system locale and then someother user logs-in. For this user
  328. // the locale changes will cause font changes only when he runs Regional options or Appearance tab.
  329. // The only other alternative would be to call this entry point whenever a user logs on, which will
  330. // be a boot time perf hit.
  331. //
  332. //-------------------------------------------------------------------------------------------------
  333. STDAPI FixFontsOnLanguageChange(void)
  334. {
  335. NONCLIENTMETRICS ncm;
  336. LOGFONT lfUIFonts[NUM_FONTS];
  337. // Get all the 6 UI fonts.
  338. FontFix_GetUIFonts(&ncm, lfUIFonts);
  339. if (S_OK == FontFix_FixNonClientFonts(lfUIFonts))
  340. {
  341. FontFix_SetUIFonts(&ncm, lfUIFonts);
  342. }
  343. return S_OK; //Yes! We had to do some updates in connection with charsets and fonts.
  344. }
  345. HRESULT FontFix_CheckSchemeCharsets(SYSTEMMETRICSALL * pSystemMetricsAll)
  346. {
  347. LOGFONT lfUIFonts[NUM_FONTS]; //Make a local copy of the fonts.
  348. // Make a copy of the ncm fonts into the local array.
  349. LF32toLF(&(pSystemMetricsAll->schemeData.ncm.lfCaptionFont), &(lfUIFonts[FONT_CAPTION]));
  350. LF32toLF(&(pSystemMetricsAll->schemeData.ncm.lfSmCaptionFont), &(lfUIFonts[FONT_SMCAPTION]));
  351. LF32toLF(&(pSystemMetricsAll->schemeData.ncm.lfMenuFont), &(lfUIFonts[FONT_MENU]));
  352. LF32toLF(&(pSystemMetricsAll->schemeData.ncm.lfStatusFont), &(lfUIFonts[FONT_STATUS]));
  353. LF32toLF(&(pSystemMetricsAll->schemeData.ncm.lfMessageFont), &(lfUIFonts[FONT_MSGBOX]));
  354. // Make a copy of the icon title font
  355. LF32toLF(&(pSystemMetricsAll->schemeData.lfIconTitle), &(lfUIFonts[FONT_ICONTITLE]));
  356. // Let's copy the data back into the SchemeData structure
  357. if (S_OK == FontFix_FixNonClientFonts(lfUIFonts))
  358. {
  359. LFtoLF32(&(lfUIFonts[FONT_CAPTION]), &(pSystemMetricsAll->schemeData.ncm.lfCaptionFont));
  360. LFtoLF32(&(lfUIFonts[FONT_SMCAPTION]), &(pSystemMetricsAll->schemeData.ncm.lfSmCaptionFont));
  361. LFtoLF32(&(lfUIFonts[FONT_MENU]), &(pSystemMetricsAll->schemeData.ncm.lfMenuFont));
  362. LFtoLF32(&(lfUIFonts[FONT_STATUS]), &(pSystemMetricsAll->schemeData.ncm.lfStatusFont));
  363. LFtoLF32(&(lfUIFonts[FONT_MSGBOX]), &(pSystemMetricsAll->schemeData.ncm.lfMessageFont));
  364. // Make a copy of the icon title font
  365. LFtoLF32(&(lfUIFonts[FONT_ICONTITLE]), &(pSystemMetricsAll->schemeData.lfIconTitle));
  366. }
  367. return S_OK; // It is alright to use this scheme stored in *psd.
  368. }