1 /* 2 * MLANG Class Factory 3 * 4 * Copyright 2002 Lionel Ulmer 5 * Copyright 2003,2004 Mike McCormack 6 * Copyright 2004,2005 Dmitry Timoshkov 7 * Copyright 2009 Detlef Riekenberg 8 * 9 * This library is free software; you can redistribute it and/or 10 * modify it under the terms of the GNU Lesser General Public 11 * License as published by the Free Software Foundation; either 12 * version 2.1 of the License, or (at your option) any later version. 13 * 14 * This library is distributed in the hope that it will be useful, 15 * but WITHOUT ANY WARRANTY; without even the implied warranty of 16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 17 * Lesser General Public License for more details. 18 * 19 * You should have received a copy of the GNU Lesser General Public 20 * License along with this library; if not, write to the Free Software 21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA 22 */ 23 24 25 #include <stdarg.h> 26 #include <stdio.h> 27 28 #define COBJMACROS 29 30 #include "windef.h" 31 #include "winbase.h" 32 #include "wingdi.h" 33 #include "winuser.h" 34 #include "ole2.h" 35 #include "objbase.h" 36 #include "rpcproxy.h" 37 #include "mlang.h" 38 #include "mimeole.h" 39 40 #include "wine/debug.h" 41 #include "wine/list.h" 42 43 WINE_DEFAULT_DEBUG_CHANNEL(mlang); 44 45 #include "initguid.h" 46 47 static HRESULT MultiLanguage_create(IUnknown *pUnkOuter, LPVOID *ppObj); 48 static HRESULT MLangConvertCharset_create(IUnknown *outer, void **obj); 49 static HRESULT EnumRfc1766_create(LANGID LangId, IEnumRfc1766 **ppEnum); 50 51 static HINSTANCE instance; 52 static DWORD MLANG_tls_index; /* to store various per thead data */ 53 54 /* FIXME: 55 * Under what circumstances HKEY_CLASSES_ROOT\MIME\Database\Codepage and 56 * HKEY_CLASSES_ROOT\MIME\Database\Charset are used? 57 */ 58 59 typedef struct 60 { 61 const char *description; 62 UINT cp; 63 DWORD flags; 64 const char *web_charset; 65 const char *header_charset; 66 const char *body_charset; 67 const WCHAR *alias; 68 } MIME_CP_INFO; 69 70 /* These data are based on the codepage info in libs/unicode/cpmap.pl */ 71 /* FIXME: Add 28604 (Celtic), 28606 (Balkan) */ 72 73 static const MIME_CP_INFO arabic_cp[] = 74 { 75 { "Arabic (864)", 76 864, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 77 MIMECONTF_MIME_LATEST, 78 "ibm864", "ibm864", "ibm864" }, 79 { "Arabic (1006)", 80 1006, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 81 MIMECONTF_MIME_LATEST, 82 "ibm1006", "ibm1006", "ibm1006" }, 83 { "Arabic (Windows)", 84 1256, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 85 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 86 MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 87 "windows-1256", "windows-1256", "windows-1256" }, 88 { "Arabic (ISO)", 89 28596, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 90 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 91 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | 92 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 93 "iso-8859-6", "iso-8859-6", "iso-8859-6" } 94 }; 95 static const MIME_CP_INFO baltic_cp[] = 96 { 97 { "Baltic (DOS)", 98 775, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 99 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 100 "ibm775", "ibm775", "ibm775" }, 101 { "Baltic (Windows)", 102 1257, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 103 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 104 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID | 105 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 106 "windows-1257", "windows-1257", "windows-1257" }, 107 { "Baltic (ISO)", 108 28594, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 109 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 110 MIMECONTF_EXPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 111 MIMECONTF_MIME_LATEST, 112 "iso-8859-4", "iso-8859-4", "iso-8859-4" }, 113 { "Estonian (ISO)", 114 28603, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 115 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 116 "iso-8859-13", "iso-8859-13", "iso-8859-13" } 117 }; 118 static const MIME_CP_INFO chinese_simplified_cp[] = 119 { 120 { "Chinese Simplified (Auto-Select)", 121 50936, MIMECONTF_IMPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 122 MIMECONTF_MIME_LATEST, 123 "_autodetect_chs", "_autodetect_chs", "_autodetect_chs" }, 124 { "Chinese Simplified (GB2312)", 125 936, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 126 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_VALID | 127 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 128 MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 129 "gb2312", "gb2312", "gb2312" }, 130 { "Chinese Simplified (GB2312-80)", 131 20936, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 132 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 133 "x-cp20936", "x-cp20936", "x-cp20936" }, 134 { "Chinese Simplified (HZ)", 135 52936, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 136 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | 137 MIMECONTF_VALID | MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | 138 MIMECONTF_MIME_LATEST, 139 "hz-gb-2312", "hz-gb-2312", "hz-gb-2312" }, 140 { "Chinese Simplified (GB18030)", 141 54936, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 142 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 143 MIMECONTF_EXPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 144 MIMECONTF_MIME_LATEST, 145 "GB18030", "GB18030", "GB18030" }, 146 { "Chinese Simplified (GBK)", 147 936, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 148 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 149 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 150 MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 151 "gbk", "gbk", "gbk" } 152 }; 153 static const MIME_CP_INFO chinese_traditional_cp[] = 154 { 155 { "Chinese Traditional (Auto-Select)", 156 50950, MIMECONTF_IMPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 157 MIMECONTF_MIME_LATEST, 158 "_autodetect_cht", "_autodetect_cht", "_autodetect_cht" }, 159 { "Chinese Traditional (Big5)", 160 950, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 161 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 162 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID | 163 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 164 "big5", "big5", "big5" }, 165 { "Chinese Traditional (CNS)", 166 20000, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 167 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 168 "x-Chinese-CNS", "x-Chinese-CNS", "x-Chinese-CNS" } 169 }; 170 static const MIME_CP_INFO central_european_cp[] = 171 { 172 { "Central European (DOS)", 173 852, MIMECONTF_BROWSER | MIMECONTF_IMPORT | MIMECONTF_SAVABLE_BROWSER | 174 MIMECONTF_EXPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 175 MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 176 "ibm852", "ibm852", "ibm852" }, 177 { "Central European (Windows)", 178 1250, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 179 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 180 MIMECONTF_EXPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 181 MIMECONTF_MIME_LATEST, 182 "windows-1250", "windows-1250", "windows-1250" }, 183 { "Central European (Mac)", 184 10029, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 185 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 186 "x-mac-ce", "x-mac-ce", "x-mac-ce" }, 187 { "Central European (ISO)", 188 28592, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 189 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 190 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID | 191 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 192 "iso-8859-2", "iso-8859-2", "iso-8859-2" } 193 }; 194 static const MIME_CP_INFO cyrillic_cp[] = 195 { 196 { "OEM Cyrillic", 197 855, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 198 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 199 "ibm855", "ibm855", "ibm855" }, 200 { "Cyrillic (DOS)", 201 866, MIMECONTF_BROWSER | MIMECONTF_IMPORT | MIMECONTF_SAVABLE_BROWSER | 202 MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | 203 MIMECONTF_MIME_LATEST, 204 "cp866", "cp866", "cp866" }, 205 #if 0 /* Windows has 20866 as an official code page for KOI8-R */ 206 { "Cyrillic (KOI8-R)", 207 878, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 208 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 209 "koi8-r", "koi8-r", "koi8-r" }, 210 #endif 211 { "Cyrillic (Windows)", 212 1251, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 213 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 214 MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 215 "windows-1251", "windows-1251", "windows-1251" }, 216 { "Cyrillic (Mac)", 217 10007, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 218 MIMECONTF_MIME_LATEST, 219 "x-mac-cyrillic", "x-mac-cyrillic", "x-mac-cyrillic" }, 220 { "Cyrillic (KOI8-R)", 221 20866, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 222 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 223 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | 224 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 225 "koi8-r", "koi8-r", "koi8-r" }, 226 { "Cyrillic (KOI8-U)", 227 21866, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 228 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 229 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | 230 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 231 "koi8-u", "koi8-u", "koi8-u" }, 232 { "Cyrillic (ISO)", 233 28595, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 234 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 235 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | 236 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 237 "iso-8859-5", "iso-8859-5", "iso-8859-5" } 238 }; 239 static const MIME_CP_INFO greek_cp[] = 240 { 241 { "Greek (DOS)", 242 737, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 243 MIMECONTF_MIME_LATEST, 244 "ibm737", "ibm737", "ibm737" }, 245 { "Greek, Modern (DOS)", 246 869, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 247 MIMECONTF_MIME_LATEST, 248 "ibm869", "ibm869", "ibm869" }, 249 { "IBM EBCDIC (Greek Modern)", 250 875, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 251 MIMECONTF_MIME_LATEST, 252 "cp875", "cp875", "cp875" }, 253 { "Greek (Windows)", 254 1253, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 255 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 256 MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 257 "windows-1253", "windows-1253", "windows-1253" }, 258 { "Greek (Mac)", 259 10006, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 260 MIMECONTF_MIME_LATEST, 261 "x-mac-greek", "x-mac-greek", "x-mac-greek" }, 262 { "Greek (ISO)", 263 28597, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 264 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 265 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | 266 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 267 "iso-8859-7", "iso-8859-7", "iso-8859-7" } 268 }; 269 static const MIME_CP_INFO hebrew_cp[] = 270 { 271 { "Hebrew (424)", 272 424, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 273 MIMECONTF_MIME_LATEST, 274 "ibm424", "ibm424", "ibm424" }, 275 { "Hebrew (856)", 276 856, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 277 MIMECONTF_MIME_LATEST, 278 "cp856", "cp856", "cp856" }, 279 { "Hebrew (DOS)", 280 862, MIMECONTF_BROWSER | MIMECONTF_MINIMAL | MIMECONTF_IMPORT | 281 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 282 MIMECONTF_MIME_LATEST, 283 "dos-862", "dos-862", "dos-862" }, 284 { "Hebrew (Windows)", 285 1255, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 286 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 287 MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 288 "windows-1255", "windows-1255", "windows-1255" }, 289 { "Hebrew (ISO-Visual)", 290 28598, MIMECONTF_BROWSER | MIMECONTF_MINIMAL | MIMECONTF_IMPORT | 291 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | 292 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 293 "iso-8859-8", "iso-8859-8", "iso-8859-8" } 294 }; 295 static const MIME_CP_INFO japanese_cp[] = 296 { 297 { "Japanese (Auto-Select)", 298 50932, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 299 MIMECONTF_IMPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 300 MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 301 "_autodetect", "_autodetect", "_autodetect" }, 302 { "Japanese (EUC)", 303 51932, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 304 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 305 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID | 306 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 307 "euc-jp", "euc-jp", "euc-jp" }, 308 { "Japanese (JIS)", 309 50220, MIMECONTF_IMPORT | MIMECONTF_MAILNEWS | MIMECONTF_EXPORT | 310 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_VALID_NLS | 311 MIMECONTF_PRIVCONVERTER | MIMECONTF_MIME_LATEST | 312 MIMECONTF_MIME_IE4, 313 "iso-2022-jp","iso-2022-jp","iso-2022-jp"}, 314 { "Japanese (JIS 0208-1990 and 0212-1990)", 315 20932, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 316 MIMECONTF_VALID | MIMECONTF_PRIVCONVERTER | MIMECONTF_MIME_LATEST, 317 "EUC-JP","EUC-JP","EUC-JP"}, 318 { "Japanese (JIS-Allow 1 byte Kana)", 319 50221, MIMECONTF_MAILNEWS | MIMECONTF_EXPORT | MIMECONTF_SAVABLE_BROWSER | 320 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_VALID_NLS | 321 MIMECONTF_VALID | MIMECONTF_PRIVCONVERTER | MIMECONTF_MIME_LATEST, 322 "csISO2022JP","iso-2022-jp","iso-2022-jp"}, 323 { "Japanese (JIS-Allow 1 byte Kana - SO/SI)", 324 50222, MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | MIMECONTF_VALID | 325 MIMECONTF_PRIVCONVERTER | MIMECONTF_MIME_LATEST, 326 "iso-2022-jp","iso-2022-jp","iso-2022-jp"}, 327 { "Japanese (Mac)", 328 10001, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 329 MIMECONTF_VALID | MIMECONTF_PRIVCONVERTER | MIMECONTF_MIME_LATEST, 330 "x-mac-japanese","x-mac-japanese","x-mac-japanese"}, 331 { "Japanese (Shift-JIS)", 332 932, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 333 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 334 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID | 335 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 336 "shift_jis", "iso-2022-jp", "iso-2022-jp" } 337 }; 338 static const MIME_CP_INFO korean_cp[] = 339 { 340 { "Korean", 341 949, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 342 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 343 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | 344 MIMECONTF_MIME_LATEST, 345 "ks_c_5601-1987", "ks_c_5601-1987", "ks_c_5601-1987" } 346 }; 347 static const MIME_CP_INFO thai_cp[] = 348 { 349 { "Thai (Windows)", 350 874, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_MIME_LATEST, 351 "ibm-thai", "ibm-thai", "ibm-thai" } 352 }; 353 static const MIME_CP_INFO turkish_cp[] = 354 { 355 { "Turkish (DOS)", 356 857, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 357 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 358 "ibm857", "ibm857", "ibm857" }, 359 { "IBM EBCDIC (Turkish Latin-5)", 360 1026, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 361 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 362 "ibm1026", "ibm1026", "ibm1026" }, 363 { "Turkish (Windows)", 364 1254, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 365 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 366 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID | 367 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 368 "windows-1254", "windows-1254", "windows-1254" }, 369 { "Turkish (Mac)", 370 10081, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 371 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 372 "x-mac-turkish", "x-mac-turkish", "x-mac-turkish" }, 373 { "Latin 3 (ISO)", 374 28593, MIMECONTF_MAILNEWS | MIMECONTF_IMPORT | 375 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_EXPORT | MIMECONTF_VALID | 376 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 377 "iso-8859-3", "iso-8859-3", "iso-8859-3" }, 378 { "Turkish (ISO)", 379 28599, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 380 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 381 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID | 382 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 383 "iso-8859-9", "iso-8859-9", "iso-8859-9" } 384 }; 385 static const MIME_CP_INFO vietnamese_cp[] = 386 { 387 { "Vietnamese (Windows)", 388 1258, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 389 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 390 MIMECONTF_EXPORT | MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | 391 MIMECONTF_MIME_LATEST, 392 "windows-1258", "windows-1258", "windows-1258" } 393 }; 394 395 static const WCHAR asciiW[] = {'a','s','c','i','i',0}; 396 397 static const MIME_CP_INFO western_cp[] = 398 { 399 { "IBM EBCDIC (US-Canada)", 400 37, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 401 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 402 "ibm037", "ibm037", "ibm037" }, 403 { "OEM United States", 404 437, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 405 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 406 "ibm437", "ibm437", "ibm437" }, 407 { "IBM EBCDIC (International)", 408 500, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 409 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 410 "ibm500", "ibm500", "ibm500" }, 411 { "Western European (DOS)", 412 850, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 413 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 414 "ibm850", "ibm850", "ibm850" }, 415 { "Portuguese (DOS)", 416 860, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 417 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 418 "ibm860", "ibm860", "ibm860" }, 419 { "Icelandic (DOS)", 420 861, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 421 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 422 "ibm861", "ibm861", "ibm861" }, 423 { "French Canadian (DOS)", 424 863, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 425 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 426 "ibm863", "ibm863", "ibm863" }, 427 { "Nordic (DOS)", 428 865, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 429 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 430 "ibm865", "ibm865", "ibm865" }, 431 { "Western European (Windows)", 432 1252, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_MINIMAL | 433 MIMECONTF_IMPORT | MIMECONTF_SAVABLE_MAILNEWS | 434 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | MIMECONTF_VALID | 435 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 436 "windows-1252", "windows-1252", "iso-8859-1" }, 437 { "Western European (Mac)", 438 10000, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 439 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 440 "macintosh", "macintosh", "macintosh" }, 441 { "Icelandic (Mac)", 442 10079, MIMECONTF_IMPORT | MIMECONTF_EXPORT | MIMECONTF_VALID | 443 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 444 "x-mac-icelandic", "x-mac-icelandic", "x-mac-icelandic" }, 445 { "US-ASCII", 446 20127, MIMECONTF_MAILNEWS | MIMECONTF_IMPORT | MIMECONTF_EXPORT | 447 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_VALID | 448 MIMECONTF_VALID_NLS | MIMECONTF_MIME_LATEST, 449 "us-ascii", "us-ascii", "us-ascii", asciiW }, 450 { "Western European (ISO)", 451 28591, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 452 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 453 MIMECONTF_EXPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 454 MIMECONTF_MIME_LATEST, 455 "iso-8859-1", "iso-8859-1", "iso-8859-1" }, 456 { "Latin 9 (ISO)", 457 28605, MIMECONTF_MAILNEWS | MIMECONTF_IMPORT | 458 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 459 MIMECONTF_EXPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 460 MIMECONTF_MIME_LATEST, 461 "iso-8859-15", "iso-8859-15", "iso-8859-15" } 462 }; 463 static const MIME_CP_INFO unicode_cp[] = 464 { 465 { "Unicode", 466 CP_UNICODE, MIMECONTF_MINIMAL | MIMECONTF_IMPORT | 467 MIMECONTF_SAVABLE_BROWSER | MIMECONTF_EXPORT | 468 MIMECONTF_VALID | MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | 469 MIMECONTF_MIME_LATEST, 470 "unicode", "unicode", "unicode" }, 471 { "Unicode (UTF-7)", 472 CP_UTF7, MIMECONTF_MAILNEWS | MIMECONTF_IMPORT | 473 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_EXPORT | MIMECONTF_VALID | 474 MIMECONTF_VALID_NLS | MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 475 "utf-7", "utf-7", "utf-7" }, 476 { "Unicode (UTF-8)", 477 CP_UTF8, MIMECONTF_MAILNEWS | MIMECONTF_BROWSER | MIMECONTF_IMPORT | 478 MIMECONTF_SAVABLE_MAILNEWS | MIMECONTF_SAVABLE_BROWSER | 479 MIMECONTF_EXPORT | MIMECONTF_VALID | MIMECONTF_VALID_NLS | 480 MIMECONTF_MIME_IE4 | MIMECONTF_MIME_LATEST, 481 "utf-8", "utf-8", "utf-8" } 482 }; 483 484 static const struct mlang_data 485 { 486 const char *description; 487 UINT family_codepage; 488 UINT number_of_cp; 489 const MIME_CP_INFO *mime_cp_info; 490 const char *fixed_font; 491 const char *proportional_font; 492 SCRIPT_ID sid; 493 } mlang_data[] = 494 { 495 { "Arabic", 1256, ARRAY_SIZE(arabic_cp), arabic_cp, 496 "Simplified Arabic Fixed","Simplified Arabic", sidArabic }, 497 { "Baltic", 1257, ARRAY_SIZE(baltic_cp), baltic_cp, 498 "Courier New","Arial", sidAsciiLatin }, 499 { "Chinese Simplified", 936, ARRAY_SIZE(chinese_simplified_cp), chinese_simplified_cp, 500 "Simsun","Simsun", sidHan }, 501 { "Chinese Traditional", 950, ARRAY_SIZE(chinese_traditional_cp), chinese_traditional_cp, 502 "MingLiu","New MingLiu", sidBopomofo }, 503 { "Central European", 1250, ARRAY_SIZE(central_european_cp), central_european_cp, 504 "Courier New","Arial", sidAsciiLatin }, 505 { "Cyrillic", 1251, ARRAY_SIZE(cyrillic_cp), cyrillic_cp, 506 "Courier New","Arial", sidCyrillic }, 507 { "Greek", 1253, ARRAY_SIZE(greek_cp), greek_cp, 508 "Courier New","Arial", sidGreek }, 509 { "Hebrew", 1255, ARRAY_SIZE(hebrew_cp), hebrew_cp, 510 "Miriam Fixed","David", sidHebrew }, 511 { "Japanese", 932, ARRAY_SIZE(japanese_cp), japanese_cp, 512 "MS Gothic","MS PGothic", sidKana }, 513 { "Korean", 949, ARRAY_SIZE(korean_cp), korean_cp, 514 "GulimChe","Gulim", sidHangul }, 515 { "Thai", 874, ARRAY_SIZE(thai_cp), thai_cp, 516 "Tahoma","Tahoma", sidThai }, 517 { "Turkish", 1254, ARRAY_SIZE(turkish_cp), turkish_cp, 518 "Courier New","Arial", sidAsciiLatin }, 519 { "Vietnamese", 1258, ARRAY_SIZE(vietnamese_cp), vietnamese_cp, 520 "Courier New","Arial", sidAsciiLatin }, 521 { "Western European", 1252, ARRAY_SIZE(western_cp), western_cp, 522 "Courier New","Arial", sidAsciiLatin }, 523 { "Unicode", CP_UNICODE, ARRAY_SIZE(unicode_cp), unicode_cp, 524 "Courier New","Arial" } 525 }; 526 527 struct font_list 528 { 529 struct list list_entry; 530 HFONT base_font; 531 HFONT font; 532 UINT charset; 533 }; 534 535 static struct list font_cache = LIST_INIT(font_cache); 536 static CRITICAL_SECTION font_cache_critical; 537 static CRITICAL_SECTION_DEBUG font_cache_critical_debug = 538 { 539 0, 0, &font_cache_critical, 540 { &font_cache_critical_debug.ProcessLocksList, &font_cache_critical_debug.ProcessLocksList }, 541 0, 0, { (DWORD_PTR)(__FILE__ ": font_cache_critical") } 542 }; 543 static CRITICAL_SECTION font_cache_critical = { &font_cache_critical_debug, -1, 0, 0, 0, 0 }; 544 545 static void fill_cp_info(const struct mlang_data *ml_data, UINT index, MIMECPINFO *mime_cp_info); 546 547 static LONG dll_count; 548 549 /* 550 * Japanese Detection and Conversion Functions 551 */ 552 553 #define HANKATA(A) ((A >= 161) && (A <= 223)) 554 #define ISEUC(A) ((A >= 161) && (A <= 254)) 555 #define NOTEUC(A,B) (((A >= 129) && (A <= 159)) && ((B >= 64) && (B <= 160))) 556 #define SJIS1(A) (((A >= 129) && (A <= 159)) || ((A >= 224) && (A <= 239))) 557 #define SJIS2(A) ((A >= 64) && (A <= 252)) 558 #define ISMARU(A) ((A >= 202) && (A <= 206)) 559 #define ISNIGORI(A) (((A >= 182) && (A <= 196)) || ((A >= 202) && (A <= 206))) 560 561 static UINT DetectJapaneseCode(LPCSTR input, DWORD count) 562 { 563 UINT code = 0; 564 DWORD i = 0; 565 unsigned char c1,c2; 566 567 while ((code == 0 || code == 51932) && i < count) 568 { 569 c1 = input[i]; 570 if (c1 == 0x1b /* ESC */) 571 { 572 i++; 573 if (i >= count) 574 return code; 575 c1 = input[i]; 576 if (c1 == '$') 577 { 578 i++; 579 if (i >= count) 580 return code; 581 c1 = input[i]; 582 if (c1 =='B' || c1 == '@') 583 code = 50220; 584 } 585 if (c1 == 'K') 586 code = 50220; 587 } 588 else if (c1 >= 129) 589 { 590 i++; 591 if (i >= count) 592 return code; 593 c2 = input[i]; 594 if NOTEUC(c1,c2) 595 code = 932; 596 else if (ISEUC(c1) && ISEUC(c2)) 597 code = 51932; 598 else if (((c1 == 142)) && HANKATA(c2)) 599 code = 51932; 600 } 601 i++; 602 } 603 return code; 604 } 605 606 static inline void jis2sjis(unsigned char *p1, unsigned char *p2) 607 { 608 unsigned char c1 = *p1; 609 unsigned char c2 = *p2; 610 int row = c1 < 95 ? 112 : 176; 611 int cell = c1 % 2 ? 31 + (c2 > 95) : 126; 612 613 *p1 = ((c1 + 1) >> 1) + row; 614 *p2 = c2 + cell; 615 } 616 617 static inline void sjis2jis(unsigned char *p1, unsigned char *p2) 618 { 619 unsigned char c1 = *p1; 620 unsigned char c2 = *p2; 621 int shift = c2 < 159; 622 int row = c1 < 160 ? 112 : 176; 623 int cell = shift ? (31 + (c2 > 127)): 126; 624 625 *p1 = ((c1 - row) << 1) - shift; 626 *p2 -= cell; 627 } 628 629 static int han2zen(unsigned char *p1, unsigned char *p2) 630 { 631 BOOL maru = FALSE; 632 BOOL nigori = FALSE; 633 static const unsigned char char1[] = {129,129,129,129,129,131,131,131,131, 634 131,131,131,131,131,131,129,131,131,131,131,131,131,131,131,131,131, 635 131,131,131,131,131,131,131,131,131,131,131,131,131,131,131,131,131, 636 131,131,131,131,131,131,131,131,131,131,131,131,131,131,131,131,131, 637 131,129,129 }; 638 static const unsigned char char2[] = {66,117,118,65,69,146,64,66,68,70, 639 72,131,133,135,98,91,65,67,69,71,73,74,76,78,80,82,84,86,88,90,92,94, 640 96,99,101,103,105,106,107,108,109,110,113,116,119,122,125,126,128, 641 129,130,132,134,136,137,138,139,140,141,143,147,74,75}; 642 643 if (( *p2 == 222) && ((ISNIGORI(*p1) || (*p1 == 179)))) 644 nigori = TRUE; 645 else if ((*p2 == 223) && (ISMARU(*p1))) 646 maru = TRUE; 647 648 if (*p1 >= 161 && *p1 <= 223) 649 { 650 unsigned char index = *p1 - 161; 651 *p1 = char1[index]; 652 *p2 = char2[index]; 653 } 654 655 if (maru || nigori) 656 { 657 if (nigori) 658 { 659 if (((*p2 >= 74) && (*p2 <= 103)) || ((*p2 >= 110) && (*p2 <= 122))) 660 (*p2)++; 661 else if ((*p1 == 131) && (*p2 == 69)) 662 *p2 = 148; 663 } 664 else if ((maru) && ((*p2 >= 110) && (*p2 <= 122))) 665 *p2+= 2; 666 667 return 1; 668 } 669 670 return 0; 671 } 672 673 674 static UINT ConvertJIS2SJIS(LPCSTR input, DWORD count, LPSTR output) 675 { 676 DWORD i = 0; 677 int j = 0; 678 unsigned char p2,p; 679 BOOL shifted = FALSE; 680 681 while (i < count) 682 { 683 p = input[i]; 684 if (p == 0x1b /* ESC */) 685 { 686 i++; 687 if (i >= count) 688 return 0; 689 p2 = input[i]; 690 if (p2 == '$' || p2 =='(') 691 i++; 692 if (p2 == 'K' || p2 =='$') 693 shifted = TRUE; 694 else 695 shifted = FALSE; 696 } 697 else 698 { 699 if (shifted) 700 { 701 i++; 702 if (i >= count) 703 return 0; 704 p2 = input[i]; 705 jis2sjis(&p,&p2); 706 output[j++]=p; 707 output[j++]=p2; 708 } 709 else 710 { 711 output[j++] = p; 712 } 713 } 714 i++; 715 } 716 return j; 717 } 718 719 static inline int exit_shift(LPSTR out, int c) 720 { 721 if (out) 722 { 723 out[c] = 0x1b; 724 out[c+1] = '('; 725 out[c+2] = 'B'; 726 } 727 return 3; 728 } 729 730 static inline int enter_shift(LPSTR out, int c) 731 { 732 if (out) 733 { 734 out[c] = 0x1b; 735 out[c+1] = '$'; 736 out[c+2] = 'B'; 737 } 738 return 3; 739 } 740 741 742 static UINT ConvertSJIS2JIS(LPCSTR input, DWORD count, LPSTR output) 743 { 744 DWORD i = 0; 745 int j = 0; 746 unsigned char p2,p; 747 BOOL shifted = FALSE; 748 749 while (i < count) 750 { 751 p = input[i] & 0xff; 752 if (p == 10 || p == 13) /* NL and CR */ 753 { 754 if (shifted) 755 { 756 shifted = FALSE; 757 j += exit_shift(output,j); 758 } 759 if (output) 760 output[j++] = p; 761 else 762 j++; 763 } 764 else 765 { 766 if (SJIS1(p)) 767 { 768 i++; 769 if (i >= count) 770 return 0; 771 p2 = input[i] & 0xff; 772 if (SJIS2(p2)) 773 { 774 sjis2jis(&p,&p2); 775 if (!shifted) 776 { 777 shifted = TRUE; 778 j+=enter_shift(output,j); 779 } 780 } 781 782 if (output) 783 { 784 output[j++]=p; 785 output[j++]=p2; 786 } 787 else 788 j+=2; 789 } 790 else 791 { 792 if (HANKATA(p)) 793 { 794 if ((i+1) >= count) 795 return 0; 796 p2 = input[i+1] & 0xff; 797 i+=han2zen(&p,&p2); 798 sjis2jis(&p,&p2); 799 if (!shifted) 800 { 801 shifted = TRUE; 802 j+=enter_shift(output,j); 803 } 804 if (output) 805 { 806 output[j++]=p; 807 output[j++]=p2; 808 } 809 else 810 j+=2; 811 } 812 else 813 { 814 if (shifted) 815 { 816 shifted = FALSE; 817 j += exit_shift(output,j); 818 } 819 if (output) 820 output[j++]=p; 821 else 822 j++; 823 } 824 } 825 } 826 i++; 827 } 828 if (shifted) 829 j += exit_shift(output,j); 830 return j; 831 } 832 833 static UINT ConvertJISJapaneseToUnicode(LPCSTR input, DWORD count, 834 LPWSTR output, DWORD out_count) 835 { 836 CHAR *sjis_string; 837 UINT rc = 0; 838 sjis_string = HeapAlloc(GetProcessHeap(),0,count); 839 rc = ConvertJIS2SJIS(input,count,sjis_string); 840 if (rc) 841 { 842 TRACE("%s\n",debugstr_an(sjis_string,rc)); 843 if (output) 844 rc = MultiByteToWideChar(932,0,sjis_string,rc,output,out_count); 845 else 846 rc = MultiByteToWideChar(932,0,sjis_string,rc,0,0); 847 } 848 HeapFree(GetProcessHeap(),0,sjis_string); 849 return rc; 850 851 } 852 853 static UINT ConvertUnknownJapaneseToUnicode(LPCSTR input, DWORD count, 854 LPWSTR output, DWORD out_count) 855 { 856 CHAR *sjis_string; 857 UINT rc = 0; 858 int code = DetectJapaneseCode(input,count); 859 TRACE("Japanese code %i\n",code); 860 861 switch (code) 862 { 863 case 0: 864 if (output) 865 rc = MultiByteToWideChar(CP_ACP,0,input,count,output,out_count); 866 else 867 rc = MultiByteToWideChar(CP_ACP,0,input,count,0,0); 868 break; 869 870 case 932: 871 if (output) 872 rc = MultiByteToWideChar(932,0,input,count,output,out_count); 873 else 874 rc = MultiByteToWideChar(932,0,input,count,0,0); 875 break; 876 877 case 51932: 878 if (output) 879 rc = MultiByteToWideChar(20932,0,input,count,output,out_count); 880 else 881 rc = MultiByteToWideChar(20932,0,input,count,0,0); 882 break; 883 884 case 50220: 885 sjis_string = HeapAlloc(GetProcessHeap(),0,count); 886 rc = ConvertJIS2SJIS(input,count,sjis_string); 887 if (rc) 888 { 889 TRACE("%s\n",debugstr_an(sjis_string,rc)); 890 if (output) 891 rc = MultiByteToWideChar(932,0,sjis_string,rc,output,out_count); 892 else 893 rc = MultiByteToWideChar(932,0,sjis_string,rc,0,0); 894 } 895 HeapFree(GetProcessHeap(),0,sjis_string); 896 break; 897 } 898 return rc; 899 } 900 901 static UINT ConvertJapaneseUnicodeToJIS(LPCWSTR input, DWORD count, 902 LPSTR output, DWORD out_count) 903 { 904 CHAR *sjis_string; 905 INT len; 906 UINT rc = 0; 907 908 len = WideCharToMultiByte(932,0,input,count,0,0,NULL,NULL); 909 sjis_string = HeapAlloc(GetProcessHeap(),0,len); 910 WideCharToMultiByte(932,0,input,count,sjis_string,len,NULL,NULL); 911 TRACE("%s\n",debugstr_an(sjis_string,len)); 912 913 rc = ConvertSJIS2JIS(sjis_string, len, NULL); 914 if (out_count >= rc) 915 { 916 ConvertSJIS2JIS(sjis_string, len, output); 917 } 918 HeapFree(GetProcessHeap(),0,sjis_string); 919 return rc; 920 921 } 922 923 /* 924 * Dll lifetime tracking declaration 925 */ 926 static void LockModule(void) 927 { 928 InterlockedIncrement(&dll_count); 929 } 930 931 static void UnlockModule(void) 932 { 933 InterlockedDecrement(&dll_count); 934 } 935 936 BOOL WINAPI DllMain(HINSTANCE hInstDLL, DWORD fdwReason, LPVOID lpv) 937 { 938 switch(fdwReason) { 939 case DLL_PROCESS_ATTACH: 940 instance = hInstDLL; 941 MLANG_tls_index = TlsAlloc(); 942 DisableThreadLibraryCalls(hInstDLL); 943 break; 944 case DLL_PROCESS_DETACH: 945 if (lpv) break; 946 TlsFree(MLANG_tls_index); 947 break; 948 } 949 return TRUE; 950 } 951 952 HRESULT WINAPI ConvertINetMultiByteToUnicode( 953 LPDWORD pdwMode, 954 DWORD dwEncoding, 955 LPCSTR pSrcStr, 956 LPINT pcSrcSize, 957 LPWSTR pDstStr, 958 LPINT pcDstSize) 959 { 960 INT src_len = -1; 961 962 TRACE("%p %d %s %p %p %p\n", pdwMode, dwEncoding, 963 debugstr_a(pSrcStr), pcSrcSize, pDstStr, pcDstSize); 964 965 if (!pcDstSize) 966 return E_FAIL; 967 968 if (!pcSrcSize) 969 pcSrcSize = &src_len; 970 971 if (!*pcSrcSize) 972 { 973 *pcDstSize = 0; 974 return S_OK; 975 } 976 977 /* forwarding euc-jp to EUC-JP */ 978 if (dwEncoding == 51932) 979 dwEncoding = 20932; 980 981 switch (dwEncoding) 982 { 983 case CP_UNICODE: 984 if (*pcSrcSize == -1) 985 *pcSrcSize = lstrlenW((LPCWSTR)pSrcStr); 986 *pcDstSize = min(*pcSrcSize, *pcDstSize); 987 *pcSrcSize *= sizeof(WCHAR); 988 if (pDstStr) 989 memmove(pDstStr, pSrcStr, *pcDstSize * sizeof(WCHAR)); 990 break; 991 992 case 50220: 993 case 50221: 994 case 50222: 995 *pcDstSize = ConvertJISJapaneseToUnicode(pSrcStr,*pcSrcSize,pDstStr,*pcDstSize); 996 break; 997 case 50932: 998 *pcDstSize = ConvertUnknownJapaneseToUnicode(pSrcStr,*pcSrcSize,pDstStr,*pcDstSize); 999 break; 1000 1001 default: 1002 if (*pcSrcSize == -1) 1003 *pcSrcSize = lstrlenA(pSrcStr); 1004 1005 if (pDstStr) 1006 *pcDstSize = MultiByteToWideChar(dwEncoding, 0, pSrcStr, *pcSrcSize, pDstStr, *pcDstSize); 1007 else 1008 *pcDstSize = MultiByteToWideChar(dwEncoding, 0, pSrcStr, *pcSrcSize, NULL, 0); 1009 break; 1010 } 1011 1012 if (!*pcDstSize) 1013 return E_FAIL; 1014 1015 return S_OK; 1016 } 1017 1018 HRESULT WINAPI ConvertINetUnicodeToMultiByte( 1019 LPDWORD pdwMode, 1020 DWORD dwEncoding, 1021 LPCWSTR pSrcStr, 1022 LPINT pcSrcSize, 1023 LPSTR pDstStr, 1024 LPINT pcDstSize) 1025 { 1026 INT destsz, size; 1027 INT src_len = -1; 1028 1029 TRACE("%p %d %s %p %p %p\n", pdwMode, dwEncoding, 1030 debugstr_w(pSrcStr), pcSrcSize, pDstStr, pcDstSize); 1031 1032 if (!pcDstSize) 1033 return S_OK; 1034 1035 if (!pcSrcSize) 1036 pcSrcSize = &src_len; 1037 1038 destsz = (pDstStr) ? *pcDstSize : 0; 1039 *pcDstSize = 0; 1040 1041 if (!pSrcStr || !*pcSrcSize) 1042 return S_OK; 1043 1044 if (*pcSrcSize == -1) 1045 *pcSrcSize = lstrlenW(pSrcStr); 1046 1047 /* forwarding euc-jp to EUC-JP */ 1048 if (dwEncoding == 51932) 1049 dwEncoding = 20932; 1050 1051 if (dwEncoding == CP_UNICODE) 1052 { 1053 if (*pcSrcSize == -1) 1054 *pcSrcSize = lstrlenW(pSrcStr); 1055 1056 size = min(*pcSrcSize, destsz) * sizeof(WCHAR); 1057 if (pDstStr) 1058 memmove(pDstStr, pSrcStr, size); 1059 1060 if (size >= destsz) 1061 goto fail; 1062 } 1063 else if (dwEncoding == 50220 || dwEncoding == 50221 || dwEncoding == 50222) 1064 { 1065 size = ConvertJapaneseUnicodeToJIS(pSrcStr, *pcSrcSize, NULL, 0); 1066 if (!size) 1067 goto fail; 1068 1069 if (pDstStr) 1070 { 1071 size = ConvertJapaneseUnicodeToJIS(pSrcStr, *pcSrcSize, pDstStr, 1072 destsz); 1073 if (!size) 1074 goto fail; 1075 } 1076 1077 } 1078 else 1079 { 1080 size = WideCharToMultiByte(dwEncoding, 0, pSrcStr, *pcSrcSize, 1081 NULL, 0, NULL, NULL); 1082 if (!size) 1083 goto fail; 1084 1085 if (pDstStr) 1086 { 1087 size = WideCharToMultiByte(dwEncoding, 0, pSrcStr, *pcSrcSize, 1088 pDstStr, destsz, NULL, NULL); 1089 if (!size) 1090 goto fail; 1091 } 1092 } 1093 1094 *pcDstSize = size; 1095 return S_OK; 1096 1097 fail: 1098 *pcSrcSize = 0; 1099 *pcDstSize = 0; 1100 return E_FAIL; 1101 } 1102 1103 HRESULT WINAPI ConvertINetString( 1104 LPDWORD pdwMode, 1105 DWORD dwSrcEncoding, 1106 DWORD dwDstEncoding, 1107 LPCSTR pSrcStr, 1108 LPINT pcSrcSize, 1109 LPSTR pDstStr, 1110 LPINT pcDstSize 1111 ) 1112 { 1113 TRACE("%p %d %d %s %p %p %p\n", pdwMode, dwSrcEncoding, dwDstEncoding, 1114 debugstr_a(pSrcStr), pcSrcSize, pDstStr, pcDstSize); 1115 1116 if (dwSrcEncoding == CP_UNICODE) 1117 { 1118 INT cSrcSizeW; 1119 if (pcSrcSize && *pcSrcSize != -1) 1120 { 1121 cSrcSizeW = *pcSrcSize / sizeof(WCHAR); 1122 pcSrcSize = &cSrcSizeW; 1123 } 1124 return ConvertINetUnicodeToMultiByte(pdwMode, dwDstEncoding, (LPCWSTR)pSrcStr, pcSrcSize, pDstStr, pcDstSize); 1125 } 1126 else if (dwDstEncoding == CP_UNICODE) 1127 { 1128 HRESULT hr = ConvertINetMultiByteToUnicode(pdwMode, dwSrcEncoding, pSrcStr, pcSrcSize, (LPWSTR)pDstStr, pcDstSize); 1129 *pcDstSize *= sizeof(WCHAR); 1130 return hr; 1131 } 1132 else 1133 { 1134 INT cDstSizeW; 1135 LPWSTR pDstStrW; 1136 HRESULT hr; 1137 1138 TRACE("convert %s from %d to %d\n", debugstr_a(pSrcStr), dwSrcEncoding, dwDstEncoding); 1139 1140 hr = ConvertINetMultiByteToUnicode(pdwMode, dwSrcEncoding, pSrcStr, pcSrcSize, NULL, &cDstSizeW); 1141 if (hr != S_OK) 1142 return hr; 1143 1144 pDstStrW = HeapAlloc(GetProcessHeap(), 0, cDstSizeW * sizeof(WCHAR)); 1145 hr = ConvertINetMultiByteToUnicode(pdwMode, dwSrcEncoding, pSrcStr, pcSrcSize, pDstStrW, &cDstSizeW); 1146 if (hr == S_OK) 1147 hr = ConvertINetUnicodeToMultiByte(pdwMode, dwDstEncoding, pDstStrW, &cDstSizeW, pDstStr, pcDstSize); 1148 1149 HeapFree(GetProcessHeap(), 0, pDstStrW); 1150 return hr; 1151 } 1152 } 1153 1154 static HRESULT GetFamilyCodePage( 1155 UINT uiCodePage, 1156 UINT* puiFamilyCodePage) 1157 { 1158 UINT i, n; 1159 1160 TRACE("%u %p\n", uiCodePage, puiFamilyCodePage); 1161 1162 if (!puiFamilyCodePage) return S_FALSE; 1163 1164 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 1165 { 1166 for (n = 0; n < mlang_data[i].number_of_cp; n++) 1167 { 1168 if (mlang_data[i].mime_cp_info[n].cp == uiCodePage) 1169 { 1170 *puiFamilyCodePage = mlang_data[i].family_codepage; 1171 return S_OK; 1172 } 1173 } 1174 } 1175 1176 return S_FALSE; 1177 } 1178 1179 HRESULT WINAPI IsConvertINetStringAvailable( 1180 DWORD dwSrcEncoding, 1181 DWORD dwDstEncoding) 1182 { 1183 UINT src_family, dst_family; 1184 1185 TRACE("%d %d\n", dwSrcEncoding, dwDstEncoding); 1186 1187 if (GetFamilyCodePage(dwSrcEncoding, &src_family) != S_OK || 1188 GetFamilyCodePage(dwDstEncoding, &dst_family) != S_OK) 1189 return S_FALSE; 1190 1191 if (src_family == dst_family) return S_OK; 1192 1193 /* we can convert any codepage to/from unicode */ 1194 if (src_family == CP_UNICODE || dst_family == CP_UNICODE) return S_OK; 1195 1196 return S_FALSE; 1197 } 1198 1199 static inline HRESULT lcid_to_rfc1766A( LCID lcid, LPSTR rfc1766, INT len ) 1200 { 1201 CHAR buffer[MAX_RFC1766_NAME]; 1202 INT n = GetLocaleInfoA(lcid, LOCALE_SISO639LANGNAME, buffer, MAX_RFC1766_NAME); 1203 INT i; 1204 1205 if (n) 1206 { 1207 i = PRIMARYLANGID(lcid); 1208 if ((((i == LANG_ENGLISH) || (i == LANG_CHINESE) || (i == LANG_ARABIC)) && 1209 (SUBLANGID(lcid) == SUBLANG_DEFAULT)) || 1210 (SUBLANGID(lcid) > SUBLANG_DEFAULT)) { 1211 1212 buffer[n - 1] = '-'; 1213 i = GetLocaleInfoA(lcid, LOCALE_SISO3166CTRYNAME, buffer + n, MAX_RFC1766_NAME - n); 1214 if (!i) 1215 buffer[n - 1] = '\0'; 1216 } 1217 else 1218 i = 0; 1219 1220 LCMapStringA( LOCALE_USER_DEFAULT, LCMAP_LOWERCASE, buffer, n + i, rfc1766, len ); 1221 return ((n + i) > len) ? E_INVALIDARG : S_OK; 1222 } 1223 return E_FAIL; 1224 } 1225 1226 static inline HRESULT lcid_to_rfc1766W( LCID lcid, LPWSTR rfc1766, INT len ) 1227 { 1228 WCHAR buffer[MAX_RFC1766_NAME]; 1229 INT n = GetLocaleInfoW(lcid, LOCALE_SISO639LANGNAME, buffer, MAX_RFC1766_NAME); 1230 INT i; 1231 1232 if (n) 1233 { 1234 i = PRIMARYLANGID(lcid); 1235 if ((((i == LANG_ENGLISH) || (i == LANG_CHINESE) || (i == LANG_ARABIC)) && 1236 (SUBLANGID(lcid) == SUBLANG_DEFAULT)) || 1237 (SUBLANGID(lcid) > SUBLANG_DEFAULT)) { 1238 1239 buffer[n - 1] = '-'; 1240 i = GetLocaleInfoW(lcid, LOCALE_SISO3166CTRYNAME, buffer + n, MAX_RFC1766_NAME - n); 1241 if (!i) 1242 buffer[n - 1] = '\0'; 1243 } 1244 else 1245 i = 0; 1246 1247 LCMapStringW(LOCALE_USER_DEFAULT, LCMAP_LOWERCASE, buffer, n + i, rfc1766, len); 1248 return ((n + i) > len) ? E_INVALIDARG : S_OK; 1249 } 1250 return E_FAIL; 1251 } 1252 1253 HRESULT WINAPI LcidToRfc1766A( 1254 LCID lcid, 1255 LPSTR pszRfc1766, 1256 INT nChar) 1257 { 1258 TRACE("%04x %p %u\n", lcid, pszRfc1766, nChar); 1259 if (!pszRfc1766) 1260 return E_INVALIDARG; 1261 1262 return lcid_to_rfc1766A(lcid, pszRfc1766, nChar); 1263 } 1264 1265 HRESULT WINAPI LcidToRfc1766W( 1266 LCID lcid, 1267 LPWSTR pszRfc1766, 1268 INT nChar) 1269 { 1270 TRACE("%04x %p %u\n", lcid, pszRfc1766, nChar); 1271 if (!pszRfc1766) 1272 return E_INVALIDARG; 1273 1274 return lcid_to_rfc1766W(lcid, pszRfc1766, nChar); 1275 } 1276 1277 static HRESULT lcid_from_rfc1766(IEnumRfc1766 *iface, LCID *lcid, LPCWSTR rfc1766) 1278 { 1279 RFC1766INFO info; 1280 ULONG num; 1281 1282 while (IEnumRfc1766_Next(iface, 1, &info, &num) == S_OK) 1283 { 1284 if (!wcsicmp(info.wszRfc1766, rfc1766)) 1285 { 1286 *lcid = info.lcid; 1287 return S_OK; 1288 } 1289 if (lstrlenW(rfc1766) == 2 && !memcmp(info.wszRfc1766, rfc1766, 2 * sizeof(WCHAR))) 1290 { 1291 *lcid = PRIMARYLANGID(info.lcid); 1292 return S_OK; 1293 } 1294 } 1295 1296 return E_FAIL; 1297 } 1298 1299 HRESULT WINAPI Rfc1766ToLcidW(LCID *pLocale, LPCWSTR pszRfc1766) 1300 { 1301 IEnumRfc1766 *enumrfc1766; 1302 HRESULT hr; 1303 1304 TRACE("(%p, %s)\n", pLocale, debugstr_w(pszRfc1766)); 1305 1306 if (!pLocale || !pszRfc1766) 1307 return E_INVALIDARG; 1308 1309 hr = EnumRfc1766_create(0, &enumrfc1766); 1310 if (FAILED(hr)) 1311 return hr; 1312 1313 hr = lcid_from_rfc1766(enumrfc1766, pLocale, pszRfc1766); 1314 IEnumRfc1766_Release(enumrfc1766); 1315 1316 return hr; 1317 } 1318 1319 HRESULT WINAPI Rfc1766ToLcidA(LCID *lcid, LPCSTR rfc1766A) 1320 { 1321 WCHAR rfc1766W[MAX_RFC1766_NAME + 1]; 1322 1323 if (!rfc1766A) 1324 return E_INVALIDARG; 1325 1326 MultiByteToWideChar(CP_ACP, 0, rfc1766A, -1, rfc1766W, MAX_RFC1766_NAME); 1327 rfc1766W[MAX_RFC1766_NAME] = 0; 1328 1329 return Rfc1766ToLcidW(lcid, rfc1766W); 1330 } 1331 1332 static HRESULT map_font(HDC hdc, DWORD codepages, HFONT src_font, HFONT *dst_font) 1333 { 1334 struct font_list *font_list_entry; 1335 CHARSETINFO charset_info; 1336 HFONT new_font, old_font; 1337 LOGFONTW font_attr; 1338 DWORD mask, Csb[2]; 1339 BOOL found_cached; 1340 UINT charset; 1341 BOOL ret; 1342 UINT i; 1343 1344 if (hdc == NULL || src_font == NULL) return E_FAIL; 1345 1346 for (i = 0; i < 32; i++) 1347 { 1348 mask = (DWORD)(1 << i); 1349 if (codepages & mask) 1350 { 1351 Csb[0] = mask; 1352 Csb[1] = 0x0; 1353 ret = TranslateCharsetInfo(Csb, &charset_info, TCI_SRCFONTSIG); 1354 if (!ret) continue; 1355 1356 /* use cached font if possible */ 1357 found_cached = FALSE; 1358 EnterCriticalSection(&font_cache_critical); 1359 LIST_FOR_EACH_ENTRY(font_list_entry, &font_cache, struct font_list, list_entry) 1360 { 1361 if (font_list_entry->charset == charset_info.ciCharset && 1362 font_list_entry->base_font == src_font) 1363 { 1364 if (dst_font != NULL) 1365 *dst_font = font_list_entry->font; 1366 found_cached = TRUE; 1367 } 1368 } 1369 LeaveCriticalSection(&font_cache_critical); 1370 if (found_cached) return S_OK; 1371 1372 GetObjectW(src_font, sizeof(font_attr), &font_attr); 1373 font_attr.lfCharSet = (BYTE)charset_info.ciCharset; 1374 font_attr.lfWidth = 0; 1375 font_attr.lfFaceName[0] = 0; 1376 new_font = CreateFontIndirectW(&font_attr); 1377 if (new_font == NULL) continue; 1378 1379 old_font = SelectObject(hdc, new_font); 1380 charset = GetTextCharset(hdc); 1381 SelectObject(hdc, old_font); 1382 if (charset == charset_info.ciCharset) 1383 { 1384 font_list_entry = HeapAlloc(GetProcessHeap(), 0, sizeof(*font_list_entry)); 1385 if (font_list_entry == NULL) return E_OUTOFMEMORY; 1386 1387 font_list_entry->base_font = src_font; 1388 font_list_entry->font = new_font; 1389 font_list_entry->charset = charset; 1390 1391 EnterCriticalSection(&font_cache_critical); 1392 list_add_tail(&font_cache, &font_list_entry->list_entry); 1393 LeaveCriticalSection(&font_cache_critical); 1394 1395 if (dst_font != NULL) 1396 *dst_font = new_font; 1397 return S_OK; 1398 } 1399 } 1400 } 1401 1402 return E_FAIL; 1403 } 1404 1405 static HRESULT release_font(HFONT font) 1406 { 1407 struct font_list *font_list_entry; 1408 HRESULT hr; 1409 1410 hr = E_FAIL; 1411 EnterCriticalSection(&font_cache_critical); 1412 LIST_FOR_EACH_ENTRY(font_list_entry, &font_cache, struct font_list, list_entry) 1413 { 1414 if (font_list_entry->font == font) 1415 { 1416 list_remove(&font_list_entry->list_entry); 1417 DeleteObject(font); 1418 HeapFree(GetProcessHeap(), 0, font_list_entry); 1419 hr = S_OK; 1420 break; 1421 } 1422 } 1423 LeaveCriticalSection(&font_cache_critical); 1424 1425 return hr; 1426 } 1427 1428 static HRESULT clear_font_cache(void) 1429 { 1430 struct font_list *font_list_entry; 1431 struct font_list *font_list_entry2; 1432 1433 EnterCriticalSection(&font_cache_critical); 1434 LIST_FOR_EACH_ENTRY_SAFE(font_list_entry, font_list_entry2, &font_cache, struct font_list, list_entry) 1435 { 1436 list_remove(&font_list_entry->list_entry); 1437 DeleteObject(font_list_entry->font); 1438 HeapFree(GetProcessHeap(), 0, font_list_entry); 1439 } 1440 LeaveCriticalSection(&font_cache_critical); 1441 1442 return S_OK; 1443 } 1444 1445 /****************************************************************************** 1446 * MLANG ClassFactory 1447 */ 1448 typedef struct { 1449 IClassFactory IClassFactory_iface; 1450 LONG ref; 1451 HRESULT (*pfnCreateInstance)(IUnknown *pUnkOuter, LPVOID *ppObj); 1452 } IClassFactoryImpl; 1453 1454 static inline IClassFactoryImpl *impl_from_IClassFactory(IClassFactory *iface) 1455 { 1456 return CONTAINING_RECORD(iface, IClassFactoryImpl, IClassFactory_iface); 1457 } 1458 1459 struct object_creation_info 1460 { 1461 const CLSID *clsid; 1462 LPCSTR szClassName; 1463 HRESULT (*pfnCreateInstance)(IUnknown *pUnkOuter, LPVOID *ppObj); 1464 }; 1465 1466 static const struct object_creation_info object_creation[] = 1467 { 1468 { &CLSID_CMultiLanguage, "CLSID_CMultiLanguage", MultiLanguage_create }, 1469 { &CLSID_CMLangConvertCharset, "CLSID_CMLangConvertCharset", MLangConvertCharset_create } 1470 }; 1471 1472 static HRESULT WINAPI MLANGCF_QueryInterface(IClassFactory *iface, REFIID riid, void **ppobj) 1473 { 1474 TRACE("%s\n", debugstr_guid(riid) ); 1475 1476 if (IsEqualGUID(riid, &IID_IUnknown) 1477 || IsEqualGUID(riid, &IID_IClassFactory)) 1478 { 1479 IClassFactory_AddRef(iface); 1480 *ppobj = iface; 1481 return S_OK; 1482 } 1483 1484 *ppobj = NULL; 1485 WARN("(%p)->(%s,%p), not found\n", iface, debugstr_guid(riid), ppobj); 1486 return E_NOINTERFACE; 1487 } 1488 1489 static ULONG WINAPI MLANGCF_AddRef(IClassFactory *iface) 1490 { 1491 IClassFactoryImpl *This = impl_from_IClassFactory(iface); 1492 return InterlockedIncrement(&This->ref); 1493 } 1494 1495 static ULONG WINAPI MLANGCF_Release(IClassFactory *iface) 1496 { 1497 IClassFactoryImpl *This = impl_from_IClassFactory(iface); 1498 ULONG ref = InterlockedDecrement(&This->ref); 1499 1500 if (ref == 0) 1501 { 1502 TRACE("Destroying %p\n", This); 1503 HeapFree(GetProcessHeap(), 0, This); 1504 } 1505 1506 return ref; 1507 } 1508 1509 static HRESULT WINAPI MLANGCF_CreateInstance(IClassFactory *iface, IUnknown *pOuter, 1510 REFIID riid, void **ppobj) 1511 { 1512 IClassFactoryImpl *This = impl_from_IClassFactory(iface); 1513 HRESULT hres; 1514 LPUNKNOWN punk; 1515 1516 TRACE("(%p)->(%p,%s,%p)\n",This,pOuter,debugstr_guid(riid),ppobj); 1517 1518 *ppobj = NULL; 1519 hres = This->pfnCreateInstance(pOuter, (LPVOID *) &punk); 1520 if (SUCCEEDED(hres)) { 1521 hres = IUnknown_QueryInterface(punk, riid, ppobj); 1522 IUnknown_Release(punk); 1523 } 1524 TRACE("returning (%p) -> %x\n", *ppobj, hres); 1525 return hres; 1526 } 1527 1528 static HRESULT WINAPI MLANGCF_LockServer(IClassFactory *iface, BOOL dolock) 1529 { 1530 if (dolock) 1531 LockModule(); 1532 else 1533 UnlockModule(); 1534 1535 return S_OK; 1536 } 1537 1538 static const IClassFactoryVtbl MLANGCF_Vtbl = 1539 { 1540 MLANGCF_QueryInterface, 1541 MLANGCF_AddRef, 1542 MLANGCF_Release, 1543 MLANGCF_CreateInstance, 1544 MLANGCF_LockServer 1545 }; 1546 1547 /****************************************************************** 1548 * DllGetClassObject (MLANG.@) 1549 */ 1550 HRESULT WINAPI DllGetClassObject(REFCLSID rclsid, REFIID iid, LPVOID *ppv) 1551 { 1552 unsigned int i; 1553 IClassFactoryImpl *factory; 1554 1555 TRACE("%s %s %p\n",debugstr_guid(rclsid), debugstr_guid(iid), ppv); 1556 1557 if ( !IsEqualGUID( &IID_IClassFactory, iid ) 1558 && ! IsEqualGUID( &IID_IUnknown, iid) ) 1559 return E_NOINTERFACE; 1560 1561 for (i = 0; i < ARRAY_SIZE(object_creation); i++) 1562 { 1563 if (IsEqualGUID(object_creation[i].clsid, rclsid)) 1564 break; 1565 } 1566 1567 if (i == ARRAY_SIZE(object_creation)) 1568 { 1569 FIXME("%s: no class found.\n", debugstr_guid(rclsid)); 1570 return CLASS_E_CLASSNOTAVAILABLE; 1571 } 1572 1573 TRACE("Creating a class factory for %s\n",object_creation[i].szClassName); 1574 1575 factory = HeapAlloc(GetProcessHeap(), 0, sizeof(*factory)); 1576 if (factory == NULL) return E_OUTOFMEMORY; 1577 1578 factory->IClassFactory_iface.lpVtbl = &MLANGCF_Vtbl; 1579 factory->ref = 1; 1580 1581 factory->pfnCreateInstance = object_creation[i].pfnCreateInstance; 1582 1583 *ppv = &factory->IClassFactory_iface; 1584 1585 TRACE("(%p) <- %p\n", ppv, &factory->IClassFactory_iface); 1586 1587 return S_OK; 1588 } 1589 1590 1591 /******************************************************************************/ 1592 1593 typedef struct tagMLang_impl 1594 { 1595 IMLangFontLink IMLangFontLink_iface; 1596 IMultiLanguage IMultiLanguage_iface; 1597 IMultiLanguage3 IMultiLanguage3_iface; 1598 IMLangFontLink2 IMLangFontLink2_iface; 1599 IMLangLineBreakConsole IMLangLineBreakConsole_iface; 1600 LONG ref; 1601 DWORD total_cp, total_scripts; 1602 } MLang_impl; 1603 1604 /******************************************************************************/ 1605 1606 typedef struct tagEnumCodePage_impl 1607 { 1608 IEnumCodePage IEnumCodePage_iface; 1609 LONG ref; 1610 MIMECPINFO *cpinfo; 1611 DWORD total, pos; 1612 } EnumCodePage_impl; 1613 1614 static inline EnumCodePage_impl *impl_from_IEnumCodePage( IEnumCodePage *iface ) 1615 { 1616 return CONTAINING_RECORD( iface, EnumCodePage_impl, IEnumCodePage_iface ); 1617 } 1618 1619 static HRESULT WINAPI fnIEnumCodePage_QueryInterface( 1620 IEnumCodePage* iface, 1621 REFIID riid, 1622 void** ppvObject) 1623 { 1624 EnumCodePage_impl *This = impl_from_IEnumCodePage( iface ); 1625 1626 TRACE("%p -> %s\n", This, debugstr_guid(riid) ); 1627 1628 if (IsEqualGUID(riid, &IID_IUnknown) 1629 || IsEqualGUID(riid, &IID_IEnumCodePage)) 1630 { 1631 IEnumCodePage_AddRef(iface); 1632 TRACE("Returning IID_IEnumCodePage %p ref = %d\n", This, This->ref); 1633 *ppvObject = &This->IEnumCodePage_iface; 1634 return S_OK; 1635 } 1636 1637 WARN("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject); 1638 return E_NOINTERFACE; 1639 } 1640 1641 static ULONG WINAPI fnIEnumCodePage_AddRef( 1642 IEnumCodePage* iface) 1643 { 1644 EnumCodePage_impl *This = impl_from_IEnumCodePage( iface ); 1645 return InterlockedIncrement(&This->ref); 1646 } 1647 1648 static ULONG WINAPI fnIEnumCodePage_Release( 1649 IEnumCodePage* iface) 1650 { 1651 EnumCodePage_impl *This = impl_from_IEnumCodePage( iface ); 1652 ULONG ref = InterlockedDecrement(&This->ref); 1653 1654 TRACE("%p ref = %d\n", This, ref); 1655 if (ref == 0) 1656 { 1657 TRACE("Destroying %p\n", This); 1658 HeapFree(GetProcessHeap(), 0, This->cpinfo); 1659 HeapFree(GetProcessHeap(), 0, This); 1660 } 1661 1662 return ref; 1663 } 1664 1665 static HRESULT WINAPI fnIEnumCodePage_Clone( 1666 IEnumCodePage* iface, 1667 IEnumCodePage** ppEnum) 1668 { 1669 EnumCodePage_impl *This = impl_from_IEnumCodePage( iface ); 1670 FIXME("%p %p\n", This, ppEnum); 1671 return E_NOTIMPL; 1672 } 1673 1674 static HRESULT WINAPI fnIEnumCodePage_Next( 1675 IEnumCodePage* iface, 1676 ULONG celt, 1677 PMIMECPINFO rgelt, 1678 ULONG* pceltFetched) 1679 { 1680 ULONG i; 1681 EnumCodePage_impl *This = impl_from_IEnumCodePage( iface ); 1682 1683 TRACE("%p %u %p %p\n", This, celt, rgelt, pceltFetched); 1684 1685 if (!pceltFetched) return S_FALSE; 1686 *pceltFetched = 0; 1687 1688 if (!rgelt) return S_FALSE; 1689 1690 if (This->pos + celt > This->total) 1691 celt = This->total - This->pos; 1692 1693 if (!celt) return S_FALSE; 1694 1695 memcpy(rgelt, This->cpinfo + This->pos, celt * sizeof(MIMECPINFO)); 1696 *pceltFetched = celt; 1697 This->pos += celt; 1698 1699 for (i = 0; i < celt; i++) 1700 { 1701 TRACE("#%u: %08x %u %u %s %s %s %s %s %s %d\n", 1702 i, rgelt[i].dwFlags, rgelt[i].uiCodePage, 1703 rgelt[i].uiFamilyCodePage, 1704 wine_dbgstr_w(rgelt[i].wszDescription), 1705 wine_dbgstr_w(rgelt[i].wszWebCharset), 1706 wine_dbgstr_w(rgelt[i].wszHeaderCharset), 1707 wine_dbgstr_w(rgelt[i].wszBodyCharset), 1708 wine_dbgstr_w(rgelt[i].wszFixedWidthFont), 1709 wine_dbgstr_w(rgelt[i].wszProportionalFont), 1710 rgelt[i].bGDICharset); 1711 } 1712 return S_OK; 1713 } 1714 1715 static HRESULT WINAPI fnIEnumCodePage_Reset( 1716 IEnumCodePage* iface) 1717 { 1718 EnumCodePage_impl *This = impl_from_IEnumCodePage( iface ); 1719 1720 TRACE("%p\n", This); 1721 1722 This->pos = 0; 1723 return S_OK; 1724 } 1725 1726 static HRESULT WINAPI fnIEnumCodePage_Skip( 1727 IEnumCodePage* iface, 1728 ULONG celt) 1729 { 1730 EnumCodePage_impl *This = impl_from_IEnumCodePage( iface ); 1731 1732 TRACE("%p %u\n", This, celt); 1733 1734 if (celt >= This->total) return S_FALSE; 1735 1736 This->pos += celt; 1737 return S_OK; 1738 } 1739 1740 static const IEnumCodePageVtbl IEnumCodePage_vtbl = 1741 { 1742 fnIEnumCodePage_QueryInterface, 1743 fnIEnumCodePage_AddRef, 1744 fnIEnumCodePage_Release, 1745 fnIEnumCodePage_Clone, 1746 fnIEnumCodePage_Next, 1747 fnIEnumCodePage_Reset, 1748 fnIEnumCodePage_Skip 1749 }; 1750 1751 static HRESULT EnumCodePage_create( MLang_impl* mlang, DWORD grfFlags, 1752 LANGID LangId, IEnumCodePage** ppEnumCodePage ) 1753 { 1754 EnumCodePage_impl *ecp; 1755 MIMECPINFO *cpinfo; 1756 UINT i, n; 1757 1758 TRACE("%p, %08x, %04x, %p\n", mlang, grfFlags, LangId, ppEnumCodePage); 1759 1760 if (!grfFlags) /* enumerate internal data base of encodings */ 1761 grfFlags = MIMECONTF_MIME_LATEST; 1762 1763 ecp = HeapAlloc( GetProcessHeap(), 0, sizeof (EnumCodePage_impl) ); 1764 ecp->IEnumCodePage_iface.lpVtbl = &IEnumCodePage_vtbl; 1765 ecp->ref = 1; 1766 ecp->pos = 0; 1767 ecp->total = 0; 1768 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 1769 { 1770 for (n = 0; n < mlang_data[i].number_of_cp; n++) 1771 { 1772 if (mlang_data[i].mime_cp_info[n].flags & grfFlags) 1773 ecp->total++; 1774 } 1775 } 1776 1777 ecp->cpinfo = HeapAlloc(GetProcessHeap(), 0, 1778 sizeof(MIMECPINFO) * ecp->total); 1779 cpinfo = ecp->cpinfo; 1780 1781 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 1782 { 1783 for (n = 0; n < mlang_data[i].number_of_cp; n++) 1784 { 1785 if (mlang_data[i].mime_cp_info[n].flags & grfFlags) 1786 fill_cp_info(&mlang_data[i], n, cpinfo++); 1787 } 1788 } 1789 1790 TRACE("enumerated %d codepages with flags %08x\n", ecp->total, grfFlags); 1791 1792 *ppEnumCodePage = &ecp->IEnumCodePage_iface; 1793 1794 return S_OK; 1795 } 1796 1797 /******************************************************************************/ 1798 1799 typedef struct tagEnumScript_impl 1800 { 1801 IEnumScript IEnumScript_iface; 1802 LONG ref; 1803 SCRIPTINFO *script_info; 1804 DWORD total, pos; 1805 } EnumScript_impl; 1806 1807 static inline EnumScript_impl *impl_from_IEnumScript( IEnumScript *iface ) 1808 { 1809 return CONTAINING_RECORD( iface, EnumScript_impl, IEnumScript_iface ); 1810 } 1811 1812 static HRESULT WINAPI fnIEnumScript_QueryInterface( 1813 IEnumScript* iface, 1814 REFIID riid, 1815 void** ppvObject) 1816 { 1817 EnumScript_impl *This = impl_from_IEnumScript( iface ); 1818 1819 TRACE("%p -> %s\n", This, debugstr_guid(riid) ); 1820 1821 if (IsEqualGUID(riid, &IID_IUnknown) 1822 || IsEqualGUID(riid, &IID_IEnumScript)) 1823 { 1824 IEnumScript_AddRef(iface); 1825 TRACE("Returning IID_IEnumScript %p ref = %d\n", This, This->ref); 1826 *ppvObject = &This->IEnumScript_iface; 1827 return S_OK; 1828 } 1829 1830 WARN("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject); 1831 return E_NOINTERFACE; 1832 } 1833 1834 static ULONG WINAPI fnIEnumScript_AddRef( 1835 IEnumScript* iface) 1836 { 1837 EnumScript_impl *This = impl_from_IEnumScript( iface ); 1838 return InterlockedIncrement(&This->ref); 1839 } 1840 1841 static ULONG WINAPI fnIEnumScript_Release( 1842 IEnumScript* iface) 1843 { 1844 EnumScript_impl *This = impl_from_IEnumScript( iface ); 1845 ULONG ref = InterlockedDecrement(&This->ref); 1846 1847 TRACE("%p ref = %d\n", This, ref); 1848 if (ref == 0) 1849 { 1850 TRACE("Destroying %p\n", This); 1851 HeapFree(GetProcessHeap(), 0, This->script_info); 1852 HeapFree(GetProcessHeap(), 0, This); 1853 } 1854 1855 return ref; 1856 } 1857 1858 static HRESULT WINAPI fnIEnumScript_Clone( 1859 IEnumScript* iface, 1860 IEnumScript** ppEnum) 1861 { 1862 EnumScript_impl *This = impl_from_IEnumScript( iface ); 1863 FIXME("%p %p: stub!\n", This, ppEnum); 1864 return E_NOTIMPL; 1865 } 1866 1867 static HRESULT WINAPI fnIEnumScript_Next( 1868 IEnumScript* iface, 1869 ULONG celt, 1870 PSCRIPTINFO rgelt, 1871 ULONG* pceltFetched) 1872 { 1873 EnumScript_impl *This = impl_from_IEnumScript( iface ); 1874 1875 TRACE("%p %u %p %p\n", This, celt, rgelt, pceltFetched); 1876 1877 if (!pceltFetched || !rgelt) return E_FAIL; 1878 1879 *pceltFetched = 0; 1880 1881 if (This->pos + celt > This->total) 1882 celt = This->total - This->pos; 1883 1884 if (!celt) return S_FALSE; 1885 1886 memcpy(rgelt, This->script_info + This->pos, celt * sizeof(SCRIPTINFO)); 1887 *pceltFetched = celt; 1888 This->pos += celt; 1889 1890 return S_OK; 1891 } 1892 1893 static HRESULT WINAPI fnIEnumScript_Reset( 1894 IEnumScript* iface) 1895 { 1896 EnumScript_impl *This = impl_from_IEnumScript( iface ); 1897 1898 TRACE("%p\n", This); 1899 1900 This->pos = 0; 1901 return S_OK; 1902 } 1903 1904 static HRESULT WINAPI fnIEnumScript_Skip( 1905 IEnumScript* iface, 1906 ULONG celt) 1907 { 1908 EnumScript_impl *This = impl_from_IEnumScript( iface ); 1909 1910 TRACE("%p %u\n", This, celt); 1911 1912 if (celt >= This->total) return S_FALSE; 1913 1914 This->pos += celt; 1915 return S_OK; 1916 } 1917 1918 static const IEnumScriptVtbl IEnumScript_vtbl = 1919 { 1920 fnIEnumScript_QueryInterface, 1921 fnIEnumScript_AddRef, 1922 fnIEnumScript_Release, 1923 fnIEnumScript_Clone, 1924 fnIEnumScript_Next, 1925 fnIEnumScript_Reset, 1926 fnIEnumScript_Skip 1927 }; 1928 1929 static HRESULT EnumScript_create( MLang_impl* mlang, DWORD dwFlags, 1930 LANGID LangId, IEnumScript** ppEnumScript ) 1931 { 1932 EnumScript_impl *es; 1933 UINT i; 1934 1935 TRACE("%p, %08x, %04x, %p\n", mlang, dwFlags, LangId, ppEnumScript); 1936 1937 if (!dwFlags) /* enumerate all available scripts */ 1938 dwFlags = SCRIPTCONTF_SCRIPT_USER | SCRIPTCONTF_SCRIPT_HIDE | SCRIPTCONTF_SCRIPT_SYSTEM; 1939 1940 es = HeapAlloc( GetProcessHeap(), 0, sizeof (EnumScript_impl) ); 1941 es->IEnumScript_iface.lpVtbl = &IEnumScript_vtbl; 1942 es->ref = 1; 1943 es->pos = 0; 1944 /* do not enumerate unicode flavours */ 1945 es->total = ARRAY_SIZE(mlang_data) - 1; 1946 es->script_info = HeapAlloc(GetProcessHeap(), 0, sizeof(SCRIPTINFO) * es->total); 1947 1948 for (i = 0; i < es->total; i++) 1949 { 1950 es->script_info[i].ScriptId = i; 1951 es->script_info[i].uiCodePage = mlang_data[i].family_codepage; 1952 MultiByteToWideChar(CP_ACP, 0, mlang_data[i].description, -1, 1953 es->script_info[i].wszDescription, MAX_SCRIPT_NAME); 1954 MultiByteToWideChar(CP_ACP, 0, mlang_data[i].fixed_font, -1, 1955 es->script_info[i].wszFixedWidthFont, MAX_MIMEFACE_NAME); 1956 MultiByteToWideChar(CP_ACP, 0, mlang_data[i].proportional_font, -1, 1957 es->script_info[i].wszProportionalFont, MAX_MIMEFACE_NAME); 1958 } 1959 1960 TRACE("enumerated %d scripts with flags %08x\n", es->total, dwFlags); 1961 1962 *ppEnumScript = &es->IEnumScript_iface; 1963 1964 return S_OK; 1965 } 1966 1967 /******************************************************************************/ 1968 1969 static inline MLang_impl *impl_from_IMLangFontLink( IMLangFontLink *iface ) 1970 { 1971 return CONTAINING_RECORD( iface, MLang_impl, IMLangFontLink_iface ); 1972 } 1973 1974 static HRESULT WINAPI fnIMLangFontLink_QueryInterface( 1975 IMLangFontLink* iface, 1976 REFIID riid, 1977 void** ppvObject) 1978 { 1979 MLang_impl *This = impl_from_IMLangFontLink( iface ); 1980 return IMultiLanguage3_QueryInterface( &This->IMultiLanguage3_iface, riid, ppvObject ); 1981 } 1982 1983 static ULONG WINAPI fnIMLangFontLink_AddRef( 1984 IMLangFontLink* iface) 1985 { 1986 MLang_impl *This = impl_from_IMLangFontLink( iface ); 1987 return IMultiLanguage3_AddRef( &This->IMultiLanguage3_iface ); 1988 } 1989 1990 static ULONG WINAPI fnIMLangFontLink_Release( 1991 IMLangFontLink* iface) 1992 { 1993 MLang_impl *This = impl_from_IMLangFontLink( iface ); 1994 return IMultiLanguage3_Release( &This->IMultiLanguage3_iface ); 1995 } 1996 1997 static HRESULT WINAPI fnIMLangFontLink_GetCharCodePages( 1998 IMLangFontLink* iface, 1999 WCHAR ch_src, 2000 DWORD* codepages) 2001 { 2002 MLang_impl *This = impl_from_IMLangFontLink( iface ); 2003 return IMLangFontLink2_GetCharCodePages(&This->IMLangFontLink2_iface, ch_src, codepages); 2004 } 2005 2006 static HRESULT WINAPI fnIMLangFontLink_GetStrCodePages( 2007 IMLangFontLink* iface, 2008 const WCHAR* src, 2009 LONG src_len, 2010 DWORD priority_cp, 2011 DWORD* codepages, 2012 LONG* ret_len) 2013 { 2014 MLang_impl *This = impl_from_IMLangFontLink( iface ); 2015 return IMLangFontLink2_GetStrCodePages(&This->IMLangFontLink2_iface, src, src_len, priority_cp, 2016 codepages, ret_len); 2017 } 2018 2019 static HRESULT WINAPI fnIMLangFontLink_CodePageToCodePages( 2020 IMLangFontLink* iface, 2021 UINT codepage, 2022 DWORD* codepages) 2023 { 2024 MLang_impl *This = impl_from_IMLangFontLink( iface ); 2025 return IMLangFontLink2_CodePageToCodePages(&This->IMLangFontLink2_iface, codepage, codepages); 2026 } 2027 2028 static HRESULT WINAPI fnIMLangFontLink_CodePagesToCodePage( 2029 IMLangFontLink* iface, 2030 DWORD codepages, 2031 UINT def_codepage, 2032 UINT* codepage) 2033 { 2034 MLang_impl *This = impl_from_IMLangFontLink(iface); 2035 return IMLangFontLink2_CodePagesToCodePage(&This->IMLangFontLink2_iface, codepages, 2036 def_codepage, codepage); 2037 } 2038 2039 static HRESULT WINAPI fnIMLangFontLink_GetFontCodePages( 2040 IMLangFontLink* iface, 2041 HDC hdc, 2042 HFONT hfont, 2043 DWORD* codepages) 2044 { 2045 MLang_impl *This = impl_from_IMLangFontLink(iface); 2046 return IMLangFontLink2_GetFontCodePages(&This->IMLangFontLink2_iface, hdc, hfont, codepages); 2047 } 2048 2049 static HRESULT WINAPI fnIMLangFontLink_MapFont( 2050 IMLangFontLink* iface, 2051 HDC hDC, 2052 DWORD dwCodePages, 2053 HFONT hSrcFont, 2054 HFONT* phDestFont) 2055 { 2056 TRACE("(%p)->%p %08x %p %p\n",iface, hDC, dwCodePages, hSrcFont, phDestFont); 2057 2058 return map_font(hDC, dwCodePages, hSrcFont, phDestFont); 2059 } 2060 2061 static HRESULT WINAPI fnIMLangFontLink_ReleaseFont( 2062 IMLangFontLink* iface, 2063 HFONT hFont) 2064 { 2065 TRACE("(%p)->%p\n",iface, hFont); 2066 2067 return release_font(hFont); 2068 } 2069 2070 static HRESULT WINAPI fnIMLangFontLink_ResetFontMapping( 2071 IMLangFontLink* iface) 2072 { 2073 TRACE("(%p)\n",iface); 2074 2075 return clear_font_cache(); 2076 } 2077 2078 2079 static const IMLangFontLinkVtbl IMLangFontLink_vtbl = 2080 { 2081 fnIMLangFontLink_QueryInterface, 2082 fnIMLangFontLink_AddRef, 2083 fnIMLangFontLink_Release, 2084 fnIMLangFontLink_GetCharCodePages, 2085 fnIMLangFontLink_GetStrCodePages, 2086 fnIMLangFontLink_CodePageToCodePages, 2087 fnIMLangFontLink_CodePagesToCodePage, 2088 fnIMLangFontLink_GetFontCodePages, 2089 fnIMLangFontLink_MapFont, 2090 fnIMLangFontLink_ReleaseFont, 2091 fnIMLangFontLink_ResetFontMapping, 2092 }; 2093 2094 /******************************************************************************/ 2095 2096 static inline MLang_impl *impl_from_IMultiLanguage( IMultiLanguage *iface ) 2097 { 2098 return CONTAINING_RECORD( iface, MLang_impl, IMultiLanguage_iface ); 2099 } 2100 2101 static HRESULT WINAPI fnIMultiLanguage_QueryInterface( 2102 IMultiLanguage* iface, 2103 REFIID riid, 2104 void** obj) 2105 { 2106 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2107 return IMultiLanguage3_QueryInterface(&This->IMultiLanguage3_iface, riid, obj); 2108 } 2109 2110 static ULONG WINAPI fnIMultiLanguage_AddRef( IMultiLanguage* iface ) 2111 { 2112 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2113 return IMultiLanguage3_AddRef(&This->IMultiLanguage3_iface); 2114 } 2115 2116 static ULONG WINAPI fnIMultiLanguage_Release( IMultiLanguage* iface ) 2117 { 2118 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2119 return IMultiLanguage3_Release(&This->IMultiLanguage3_iface); 2120 } 2121 2122 static HRESULT WINAPI fnIMultiLanguage_GetNumberOfCodePageInfo( 2123 IMultiLanguage* iface, 2124 UINT* cp) 2125 { 2126 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2127 TRACE("(%p, %p)\n", This, cp); 2128 return IMultiLanguage3_GetNumberOfCodePageInfo(&This->IMultiLanguage3_iface, cp); 2129 } 2130 2131 static HRESULT WINAPI fnIMultiLanguage_GetCodePageInfo( 2132 IMultiLanguage* iface, 2133 UINT uiCodePage, 2134 PMIMECPINFO pCodePageInfo) 2135 { 2136 UINT i, n; 2137 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2138 2139 TRACE("%p, %u, %p\n", This, uiCodePage, pCodePageInfo); 2140 2141 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 2142 { 2143 for (n = 0; n < mlang_data[i].number_of_cp; n++) 2144 { 2145 if (mlang_data[i].mime_cp_info[n].cp == uiCodePage) 2146 { 2147 fill_cp_info(&mlang_data[i], n, pCodePageInfo); 2148 return S_OK; 2149 } 2150 } 2151 } 2152 2153 return S_FALSE; 2154 } 2155 2156 static HRESULT WINAPI fnIMultiLanguage_GetFamilyCodePage( 2157 IMultiLanguage* iface, 2158 UINT cp, 2159 UINT* family_cp) 2160 { 2161 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2162 return IMultiLanguage3_GetFamilyCodePage(&This->IMultiLanguage3_iface, cp, family_cp); 2163 } 2164 2165 static HRESULT WINAPI fnIMultiLanguage_EnumCodePages( 2166 IMultiLanguage* iface, 2167 DWORD grfFlags, 2168 IEnumCodePage** ppEnumCodePage) 2169 { 2170 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2171 2172 TRACE("%p %08x %p\n", This, grfFlags, ppEnumCodePage); 2173 2174 return EnumCodePage_create( This, grfFlags, 0, ppEnumCodePage ); 2175 } 2176 2177 static HRESULT WINAPI fnIMultiLanguage_GetCharsetInfo( 2178 IMultiLanguage* iface, 2179 BSTR Charset, 2180 PMIMECSETINFO pCharsetInfo) 2181 { 2182 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2183 return IMultiLanguage3_GetCharsetInfo( &This->IMultiLanguage3_iface, Charset, pCharsetInfo ); 2184 } 2185 2186 static HRESULT WINAPI fnIMultiLanguage_IsConvertible( 2187 IMultiLanguage* iface, 2188 DWORD src_enc, 2189 DWORD dst_enc) 2190 { 2191 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2192 return IMultiLanguage3_IsConvertible(&This->IMultiLanguage3_iface, src_enc, dst_enc); 2193 } 2194 2195 static HRESULT WINAPI fnIMultiLanguage_ConvertString( 2196 IMultiLanguage* iface, 2197 DWORD* mode, 2198 DWORD src_enc, 2199 DWORD dst_enc, 2200 BYTE* src, 2201 UINT* src_size, 2202 BYTE* dest, 2203 UINT* dest_size) 2204 { 2205 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2206 return IMultiLanguage3_ConvertString(&This->IMultiLanguage3_iface, mode, src_enc, 2207 dst_enc, src, src_size, dest, dest_size); 2208 } 2209 2210 static HRESULT WINAPI fnIMultiLanguage_ConvertStringToUnicode( 2211 IMultiLanguage* iface, 2212 DWORD* mode, 2213 DWORD src_enc, 2214 CHAR* src, 2215 UINT* src_size, 2216 WCHAR* dest, 2217 UINT* dest_size) 2218 { 2219 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2220 return IMultiLanguage3_ConvertStringToUnicode(&This->IMultiLanguage3_iface, 2221 mode, src_enc, src, src_size, dest, dest_size); 2222 } 2223 2224 static HRESULT WINAPI fnIMultiLanguage_ConvertStringFromUnicode( 2225 IMultiLanguage* iface, 2226 DWORD* mode, 2227 DWORD encoding, 2228 WCHAR* src, 2229 UINT* src_size, 2230 CHAR* dest, 2231 UINT* dest_size) 2232 { 2233 MLang_impl *This = impl_from_IMultiLanguage(iface); 2234 return IMultiLanguage3_ConvertStringFromUnicode(&This->IMultiLanguage3_iface, 2235 mode, encoding, src, src_size, dest, dest_size); 2236 } 2237 2238 static HRESULT WINAPI fnIMultiLanguage_ConvertStringReset( 2239 IMultiLanguage* iface) 2240 { 2241 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2242 return IMultiLanguage3_ConvertStringReset(&This->IMultiLanguage3_iface); 2243 } 2244 2245 static HRESULT WINAPI fnIMultiLanguage_GetRfc1766FromLcid( 2246 IMultiLanguage* iface, 2247 LCID lcid, 2248 BSTR* pbstrRfc1766) 2249 { 2250 MLang_impl *This = impl_from_IMultiLanguage(iface); 2251 return IMultiLanguage3_GetRfc1766FromLcid(&This->IMultiLanguage3_iface, lcid, pbstrRfc1766); 2252 } 2253 2254 static HRESULT WINAPI fnIMultiLanguage_GetLcidFromRfc1766( 2255 IMultiLanguage* iface, 2256 LCID* locale, 2257 BSTR rfc1766) 2258 { 2259 MLang_impl *This = impl_from_IMultiLanguage(iface); 2260 return IMultiLanguage3_GetLcidFromRfc1766(&This->IMultiLanguage3_iface, locale, rfc1766); 2261 } 2262 2263 /******************************************************************************/ 2264 2265 typedef struct tagEnumRfc1766_impl 2266 { 2267 IEnumRfc1766 IEnumRfc1766_iface; 2268 LONG ref; 2269 RFC1766INFO *info; 2270 DWORD total, pos; 2271 } EnumRfc1766_impl; 2272 2273 static inline EnumRfc1766_impl *impl_from_IEnumRfc1766( IEnumRfc1766 *iface ) 2274 { 2275 return CONTAINING_RECORD( iface, EnumRfc1766_impl, IEnumRfc1766_iface ); 2276 } 2277 2278 static HRESULT WINAPI fnIEnumRfc1766_QueryInterface( 2279 IEnumRfc1766 *iface, 2280 REFIID riid, 2281 void** ppvObject) 2282 { 2283 EnumRfc1766_impl *This = impl_from_IEnumRfc1766( iface ); 2284 2285 TRACE("%p -> %s\n", This, debugstr_guid(riid) ); 2286 2287 if (IsEqualGUID(riid, &IID_IUnknown) 2288 || IsEqualGUID(riid, &IID_IEnumRfc1766)) 2289 { 2290 IEnumRfc1766_AddRef(iface); 2291 TRACE("Returning IID_IEnumRfc1766 %p ref = %d\n", This, This->ref); 2292 *ppvObject = &This->IEnumRfc1766_iface; 2293 return S_OK; 2294 } 2295 2296 WARN("(%p) -> (%s,%p), not found\n",This,debugstr_guid(riid),ppvObject); 2297 return E_NOINTERFACE; 2298 } 2299 2300 static ULONG WINAPI fnIEnumRfc1766_AddRef( 2301 IEnumRfc1766 *iface) 2302 { 2303 EnumRfc1766_impl *This = impl_from_IEnumRfc1766( iface ); 2304 return InterlockedIncrement(&This->ref); 2305 } 2306 2307 static ULONG WINAPI fnIEnumRfc1766_Release( 2308 IEnumRfc1766 *iface) 2309 { 2310 EnumRfc1766_impl *This = impl_from_IEnumRfc1766( iface ); 2311 ULONG ref = InterlockedDecrement(&This->ref); 2312 2313 TRACE("%p ref = %d\n", This, ref); 2314 if (ref == 0) 2315 { 2316 TRACE("Destroying %p\n", This); 2317 HeapFree(GetProcessHeap(), 0, This->info); 2318 HeapFree(GetProcessHeap(), 0, This); 2319 } 2320 return ref; 2321 } 2322 2323 static HRESULT WINAPI fnIEnumRfc1766_Clone( 2324 IEnumRfc1766 *iface, 2325 IEnumRfc1766 **ppEnum) 2326 { 2327 EnumRfc1766_impl *This = impl_from_IEnumRfc1766( iface ); 2328 2329 FIXME("%p %p\n", This, ppEnum); 2330 return E_NOTIMPL; 2331 } 2332 2333 static HRESULT WINAPI fnIEnumRfc1766_Next( 2334 IEnumRfc1766 *iface, 2335 ULONG celt, 2336 PRFC1766INFO rgelt, 2337 ULONG *pceltFetched) 2338 { 2339 ULONG i; 2340 EnumRfc1766_impl *This = impl_from_IEnumRfc1766( iface ); 2341 2342 TRACE("%p %u %p %p\n", This, celt, rgelt, pceltFetched); 2343 2344 if (!pceltFetched) return S_FALSE; 2345 *pceltFetched = 0; 2346 2347 if (!rgelt) return S_FALSE; 2348 2349 if (This->pos + celt > This->total) 2350 celt = This->total - This->pos; 2351 2352 if (!celt) return S_FALSE; 2353 2354 memcpy(rgelt, This->info + This->pos, celt * sizeof(RFC1766INFO)); 2355 *pceltFetched = celt; 2356 This->pos += celt; 2357 2358 for (i = 0; i < celt; i++) 2359 { 2360 TRACE("#%u: %08x %s %s\n", 2361 i, rgelt[i].lcid, 2362 wine_dbgstr_w(rgelt[i].wszRfc1766), 2363 wine_dbgstr_w(rgelt[i].wszLocaleName)); 2364 } 2365 return S_OK; 2366 } 2367 2368 static HRESULT WINAPI fnIEnumRfc1766_Reset( 2369 IEnumRfc1766 *iface) 2370 { 2371 EnumRfc1766_impl *This = impl_from_IEnumRfc1766( iface ); 2372 2373 TRACE("%p\n", This); 2374 2375 This->pos = 0; 2376 return S_OK; 2377 } 2378 2379 static HRESULT WINAPI fnIEnumRfc1766_Skip( 2380 IEnumRfc1766 *iface, 2381 ULONG celt) 2382 { 2383 EnumRfc1766_impl *This = impl_from_IEnumRfc1766( iface ); 2384 2385 TRACE("%p %u\n", This, celt); 2386 2387 if (celt >= This->total) return S_FALSE; 2388 2389 This->pos += celt; 2390 return S_OK; 2391 } 2392 2393 static const IEnumRfc1766Vtbl IEnumRfc1766_vtbl = 2394 { 2395 fnIEnumRfc1766_QueryInterface, 2396 fnIEnumRfc1766_AddRef, 2397 fnIEnumRfc1766_Release, 2398 fnIEnumRfc1766_Clone, 2399 fnIEnumRfc1766_Next, 2400 fnIEnumRfc1766_Reset, 2401 fnIEnumRfc1766_Skip 2402 }; 2403 2404 struct enum_locales_data 2405 { 2406 RFC1766INFO *info; 2407 DWORD total, allocated; 2408 }; 2409 2410 static BOOL CALLBACK enum_locales_proc(LPWSTR locale) 2411 { 2412 WCHAR *end; 2413 struct enum_locales_data *data = TlsGetValue(MLANG_tls_index); 2414 RFC1766INFO *info; 2415 2416 TRACE("%s\n", debugstr_w(locale)); 2417 2418 if (data->total >= data->allocated) 2419 { 2420 data->allocated += 32; 2421 data->info = HeapReAlloc(GetProcessHeap(), 0, data->info, data->allocated * sizeof(RFC1766INFO)); 2422 if (!data->info) return FALSE; 2423 } 2424 2425 info = &data->info[data->total]; 2426 2427 info->lcid = wcstol(locale, &end, 16); 2428 if (*end) /* invalid number */ 2429 return FALSE; 2430 2431 info->wszRfc1766[0] = 0; 2432 lcid_to_rfc1766W( info->lcid, info->wszRfc1766, MAX_RFC1766_NAME ); 2433 2434 info->wszLocaleName[0] = 0; 2435 GetLocaleInfoW(info->lcid, LOCALE_SLANGUAGE, info->wszLocaleName, MAX_LOCALE_NAME); 2436 TRACE("ISO639: %s SLANGUAGE: %s\n", wine_dbgstr_w(info->wszRfc1766), wine_dbgstr_w(info->wszLocaleName)); 2437 2438 data->total++; 2439 2440 return TRUE; 2441 } 2442 2443 static HRESULT EnumRfc1766_create(LANGID LangId, IEnumRfc1766 **ppEnum) 2444 { 2445 EnumRfc1766_impl *rfc; 2446 struct enum_locales_data data; 2447 2448 TRACE("%04x, %p\n", LangId, ppEnum); 2449 2450 rfc = HeapAlloc( GetProcessHeap(), 0, sizeof(EnumRfc1766_impl) ); 2451 rfc->IEnumRfc1766_iface.lpVtbl = &IEnumRfc1766_vtbl; 2452 rfc->ref = 1; 2453 rfc->pos = 0; 2454 rfc->total = 0; 2455 2456 data.total = 0; 2457 data.allocated = 160; 2458 data.info = HeapAlloc(GetProcessHeap(), 0, data.allocated * sizeof(RFC1766INFO)); 2459 if (!data.info) 2460 { 2461 HeapFree(GetProcessHeap(), 0, rfc); 2462 return E_OUTOFMEMORY; 2463 } 2464 2465 TlsSetValue(MLANG_tls_index, &data); 2466 EnumSystemLocalesW(enum_locales_proc, 0/*LOCALE_SUPPORTED*/); 2467 TlsSetValue(MLANG_tls_index, NULL); 2468 2469 TRACE("enumerated %d rfc1766 structures\n", data.total); 2470 2471 if (!data.total) 2472 { 2473 HeapFree(GetProcessHeap(), 0, data.info); 2474 HeapFree(GetProcessHeap(), 0, rfc); 2475 return E_FAIL; 2476 } 2477 2478 rfc->info = data.info; 2479 rfc->total = data.total; 2480 2481 *ppEnum = &rfc->IEnumRfc1766_iface; 2482 return S_OK; 2483 } 2484 2485 static HRESULT WINAPI fnIMultiLanguage_EnumRfc1766( 2486 IMultiLanguage *iface, 2487 IEnumRfc1766 **ppEnumRfc1766) 2488 { 2489 MLang_impl *This = impl_from_IMultiLanguage( iface ); 2490 2491 TRACE("%p %p\n", This, ppEnumRfc1766); 2492 2493 return EnumRfc1766_create(0, ppEnumRfc1766); 2494 } 2495 2496 /******************************************************************************/ 2497 2498 static HRESULT WINAPI fnIMultiLanguage_GetRfc1766Info( 2499 IMultiLanguage* iface, 2500 LCID Locale, 2501 PRFC1766INFO pRfc1766Info) 2502 { 2503 LCTYPE type = LOCALE_SLANGUAGE; 2504 2505 TRACE("(%p, 0x%04x, %p)\n", iface, Locale, pRfc1766Info); 2506 2507 if (!pRfc1766Info) 2508 return E_INVALIDARG; 2509 2510 if ((PRIMARYLANGID(Locale) == LANG_ENGLISH) || 2511 (PRIMARYLANGID(Locale) == LANG_CHINESE) || 2512 (PRIMARYLANGID(Locale) == LANG_ARABIC)) { 2513 2514 if (!SUBLANGID(Locale)) 2515 type = LOCALE_SENGLANGUAGE; /* suppress country */ 2516 } 2517 else 2518 { 2519 if (!SUBLANGID(Locale)) { 2520 TRACE("SUBLANGID missing in 0x%04x\n", Locale); 2521 return E_FAIL; 2522 } 2523 } 2524 2525 pRfc1766Info->lcid = Locale; 2526 pRfc1766Info->wszRfc1766[0] = 0; 2527 pRfc1766Info->wszLocaleName[0] = 0; 2528 2529 if ((!lcid_to_rfc1766W(Locale, pRfc1766Info->wszRfc1766, MAX_RFC1766_NAME)) && 2530 (GetLocaleInfoW(Locale, type, pRfc1766Info->wszLocaleName, MAX_LOCALE_NAME) > 0)) 2531 return S_OK; 2532 2533 /* Locale not supported */ 2534 return E_INVALIDARG; 2535 } 2536 2537 static HRESULT WINAPI fnIMultiLanguage_CreateConvertCharset( 2538 IMultiLanguage* iface, 2539 UINT src_cp, 2540 UINT dst_cp, 2541 DWORD prop, 2542 IMLangConvertCharset** convert_charset) 2543 { 2544 MLang_impl *This = impl_from_IMultiLanguage(iface); 2545 return IMultiLanguage3_CreateConvertCharset(&This->IMultiLanguage3_iface, src_cp, dst_cp, prop, convert_charset); 2546 } 2547 2548 static const IMultiLanguageVtbl IMultiLanguage_vtbl = 2549 { 2550 fnIMultiLanguage_QueryInterface, 2551 fnIMultiLanguage_AddRef, 2552 fnIMultiLanguage_Release, 2553 fnIMultiLanguage_GetNumberOfCodePageInfo, 2554 fnIMultiLanguage_GetCodePageInfo, 2555 fnIMultiLanguage_GetFamilyCodePage, 2556 fnIMultiLanguage_EnumCodePages, 2557 fnIMultiLanguage_GetCharsetInfo, 2558 fnIMultiLanguage_IsConvertible, 2559 fnIMultiLanguage_ConvertString, 2560 fnIMultiLanguage_ConvertStringToUnicode, 2561 fnIMultiLanguage_ConvertStringFromUnicode, 2562 fnIMultiLanguage_ConvertStringReset, 2563 fnIMultiLanguage_GetRfc1766FromLcid, 2564 fnIMultiLanguage_GetLcidFromRfc1766, 2565 fnIMultiLanguage_EnumRfc1766, 2566 fnIMultiLanguage_GetRfc1766Info, 2567 fnIMultiLanguage_CreateConvertCharset, 2568 }; 2569 2570 2571 /******************************************************************************/ 2572 2573 static inline MLang_impl *impl_from_IMultiLanguage3( IMultiLanguage3 *iface ) 2574 { 2575 return CONTAINING_RECORD( iface, MLang_impl, IMultiLanguage3_iface ); 2576 } 2577 2578 static HRESULT WINAPI fnIMultiLanguage3_QueryInterface( 2579 IMultiLanguage3* iface, 2580 REFIID riid, 2581 void** obj) 2582 { 2583 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 2584 2585 TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), obj); 2586 2587 if (IsEqualGUID(riid, &IID_IUnknown) || 2588 IsEqualGUID(riid, &IID_IMultiLanguage)) 2589 { 2590 *obj = &This->IMultiLanguage_iface; 2591 } 2592 else if (IsEqualGUID(riid, &IID_IMLangCodePages) || 2593 IsEqualGUID(riid, &IID_IMLangFontLink)) 2594 { 2595 *obj = &This->IMLangFontLink_iface; 2596 } 2597 else if (IsEqualGUID(riid, &IID_IMLangFontLink2)) 2598 { 2599 *obj = &This->IMLangFontLink2_iface; 2600 } 2601 else if (IsEqualGUID(riid, &IID_IMultiLanguage2) || 2602 IsEqualGUID(riid, &IID_IMultiLanguage3)) 2603 { 2604 *obj = &This->IMultiLanguage3_iface; 2605 } 2606 else if (IsEqualGUID(riid, &IID_IMLangLineBreakConsole)) 2607 { 2608 *obj = &This->IMLangLineBreakConsole_iface; 2609 } 2610 else 2611 { 2612 WARN("(%p)->(%s,%p),not found\n", This, debugstr_guid(riid), obj); 2613 *obj = NULL; 2614 return E_NOINTERFACE; 2615 } 2616 2617 IMultiLanguage3_AddRef(iface); 2618 return S_OK; 2619 } 2620 2621 static ULONG WINAPI fnIMultiLanguage3_AddRef( IMultiLanguage3* iface ) 2622 { 2623 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 2624 return InterlockedIncrement(&This->ref); 2625 } 2626 2627 static ULONG WINAPI fnIMultiLanguage3_Release( IMultiLanguage3* iface ) 2628 { 2629 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 2630 ULONG ref = InterlockedDecrement(&This->ref); 2631 2632 TRACE("(%p)->(%d)\n", This, ref); 2633 if (ref == 0) 2634 { 2635 HeapFree(GetProcessHeap(), 0, This); 2636 UnlockModule(); 2637 } 2638 2639 return ref; 2640 } 2641 2642 static HRESULT WINAPI fnIMultiLanguage3_GetNumberOfCodePageInfo( 2643 IMultiLanguage3* iface, 2644 UINT* pcCodePage) 2645 { 2646 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 2647 2648 TRACE("%p, %p\n", This, pcCodePage); 2649 2650 if (!pcCodePage) return E_INVALIDARG; 2651 2652 *pcCodePage = This->total_cp; 2653 return S_OK; 2654 } 2655 2656 static void fill_cp_info(const struct mlang_data *ml_data, UINT index, MIMECPINFO *mime_cp_info) 2657 { 2658 CHARSETINFO csi; 2659 2660 if (TranslateCharsetInfo((DWORD*)(DWORD_PTR)ml_data->family_codepage, &csi, 2661 TCI_SRCCODEPAGE)) 2662 mime_cp_info->bGDICharset = csi.ciCharset; 2663 else 2664 mime_cp_info->bGDICharset = DEFAULT_CHARSET; 2665 2666 mime_cp_info->dwFlags = ml_data->mime_cp_info[index].flags; 2667 mime_cp_info->uiCodePage = ml_data->mime_cp_info[index].cp; 2668 mime_cp_info->uiFamilyCodePage = ml_data->family_codepage; 2669 MultiByteToWideChar(CP_ACP, 0, ml_data->mime_cp_info[index].description, -1, 2670 mime_cp_info->wszDescription, ARRAY_SIZE(mime_cp_info->wszDescription)); 2671 MultiByteToWideChar(CP_ACP, 0, ml_data->mime_cp_info[index].web_charset, -1, 2672 mime_cp_info->wszWebCharset, ARRAY_SIZE(mime_cp_info->wszWebCharset)); 2673 MultiByteToWideChar(CP_ACP, 0, ml_data->mime_cp_info[index].header_charset, -1, 2674 mime_cp_info->wszHeaderCharset, ARRAY_SIZE(mime_cp_info->wszHeaderCharset)); 2675 MultiByteToWideChar(CP_ACP, 0, ml_data->mime_cp_info[index].body_charset, -1, 2676 mime_cp_info->wszBodyCharset, ARRAY_SIZE(mime_cp_info->wszBodyCharset)); 2677 2678 MultiByteToWideChar(CP_ACP, 0, ml_data->fixed_font, -1, 2679 mime_cp_info->wszFixedWidthFont, ARRAY_SIZE(mime_cp_info->wszFixedWidthFont)); 2680 MultiByteToWideChar(CP_ACP, 0, ml_data->proportional_font, -1, 2681 mime_cp_info->wszProportionalFont, ARRAY_SIZE(mime_cp_info->wszProportionalFont)); 2682 2683 TRACE("%08x %u %u %s %s %s %s %s %s %d\n", 2684 mime_cp_info->dwFlags, mime_cp_info->uiCodePage, 2685 mime_cp_info->uiFamilyCodePage, 2686 wine_dbgstr_w(mime_cp_info->wszDescription), 2687 wine_dbgstr_w(mime_cp_info->wszWebCharset), 2688 wine_dbgstr_w(mime_cp_info->wszHeaderCharset), 2689 wine_dbgstr_w(mime_cp_info->wszBodyCharset), 2690 wine_dbgstr_w(mime_cp_info->wszFixedWidthFont), 2691 wine_dbgstr_w(mime_cp_info->wszProportionalFont), 2692 mime_cp_info->bGDICharset); 2693 } 2694 2695 static HRESULT WINAPI fnIMultiLanguage3_GetCodePageInfo( 2696 IMultiLanguage3* iface, 2697 UINT uiCodePage, 2698 LANGID LangId, 2699 PMIMECPINFO pCodePageInfo) 2700 { 2701 UINT i, n; 2702 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 2703 2704 TRACE("%p, %u, %04x, %p\n", This, uiCodePage, LangId, pCodePageInfo); 2705 2706 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 2707 { 2708 for (n = 0; n < mlang_data[i].number_of_cp; n++) 2709 { 2710 if (mlang_data[i].mime_cp_info[n].cp == uiCodePage) 2711 { 2712 fill_cp_info(&mlang_data[i], n, pCodePageInfo); 2713 return S_OK; 2714 } 2715 } 2716 } 2717 2718 return S_FALSE; 2719 } 2720 2721 static HRESULT WINAPI fnIMultiLanguage3_GetFamilyCodePage( 2722 IMultiLanguage3* iface, 2723 UINT uiCodePage, 2724 UINT* puiFamilyCodePage) 2725 { 2726 return GetFamilyCodePage(uiCodePage, puiFamilyCodePage); 2727 } 2728 2729 static HRESULT WINAPI fnIMultiLanguage3_EnumCodePages( 2730 IMultiLanguage3* iface, 2731 DWORD grfFlags, 2732 LANGID LangId, 2733 IEnumCodePage** ppEnumCodePage) 2734 { 2735 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 2736 2737 TRACE("%p %08x %04x %p\n", This, grfFlags, LangId, ppEnumCodePage); 2738 2739 return EnumCodePage_create( This, grfFlags, LangId, ppEnumCodePage ); 2740 } 2741 2742 static HRESULT WINAPI fnIMultiLanguage3_GetCharsetInfo( 2743 IMultiLanguage3* iface, 2744 BSTR Charset, 2745 PMIMECSETINFO pCharsetInfo) 2746 { 2747 UINT i, n; 2748 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 2749 2750 TRACE("%p %s %p\n", This, debugstr_w(Charset), pCharsetInfo); 2751 2752 if (!pCharsetInfo) return E_FAIL; 2753 2754 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 2755 { 2756 for (n = 0; n < mlang_data[i].number_of_cp; n++) 2757 { 2758 WCHAR csetW[MAX_MIMECSET_NAME]; 2759 2760 MultiByteToWideChar(CP_ACP, 0, mlang_data[i].mime_cp_info[n].web_charset, -1, csetW, MAX_MIMECSET_NAME); 2761 if (!lstrcmpiW(Charset, csetW)) 2762 { 2763 pCharsetInfo->uiCodePage = mlang_data[i].family_codepage; 2764 pCharsetInfo->uiInternetEncoding = mlang_data[i].mime_cp_info[n].cp; 2765 lstrcpyW(pCharsetInfo->wszCharset, csetW); 2766 return S_OK; 2767 } 2768 if (mlang_data[i].mime_cp_info[n].alias && !lstrcmpiW(Charset, mlang_data[i].mime_cp_info[n].alias)) 2769 { 2770 pCharsetInfo->uiCodePage = mlang_data[i].family_codepage; 2771 pCharsetInfo->uiInternetEncoding = mlang_data[i].mime_cp_info[n].cp; 2772 lstrcpyW(pCharsetInfo->wszCharset, mlang_data[i].mime_cp_info[n].alias); 2773 return S_OK; 2774 } 2775 } 2776 } 2777 2778 /* FIXME: 2779 * Since we do not support charsets like iso-2022-jp and do not have 2780 * them in our database as a primary (web_charset) encoding this loop 2781 * does an attempt to 'approximate' charset name by header_charset. 2782 */ 2783 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 2784 { 2785 for (n = 0; n < mlang_data[i].number_of_cp; n++) 2786 { 2787 WCHAR csetW[MAX_MIMECSET_NAME]; 2788 2789 MultiByteToWideChar(CP_ACP, 0, mlang_data[i].mime_cp_info[n].header_charset, -1, csetW, MAX_MIMECSET_NAME); 2790 if (!lstrcmpiW(Charset, csetW)) 2791 { 2792 pCharsetInfo->uiCodePage = mlang_data[i].family_codepage; 2793 pCharsetInfo->uiInternetEncoding = mlang_data[i].mime_cp_info[n].cp; 2794 lstrcpyW(pCharsetInfo->wszCharset, csetW); 2795 return S_OK; 2796 } 2797 } 2798 } 2799 2800 return E_FAIL; 2801 } 2802 2803 static HRESULT WINAPI fnIMultiLanguage3_IsConvertible( 2804 IMultiLanguage3* iface, 2805 DWORD dwSrcEncoding, 2806 DWORD dwDstEncoding) 2807 { 2808 return IsConvertINetStringAvailable(dwSrcEncoding, dwDstEncoding); 2809 } 2810 2811 static HRESULT WINAPI fnIMultiLanguage3_ConvertString( 2812 IMultiLanguage3* iface, 2813 DWORD* pdwMode, 2814 DWORD dwSrcEncoding, 2815 DWORD dwDstEncoding, 2816 BYTE* pSrcStr, 2817 UINT* pcSrcSize, 2818 BYTE* pDstStr, 2819 UINT* pcDstSize) 2820 { 2821 return ConvertINetString(pdwMode, dwSrcEncoding, dwDstEncoding, 2822 (LPCSTR)pSrcStr, (LPINT)pcSrcSize, (LPSTR)pDstStr, (LPINT)pcDstSize); 2823 } 2824 2825 static HRESULT WINAPI fnIMultiLanguage3_ConvertStringToUnicode( 2826 IMultiLanguage3* iface, 2827 DWORD* pdwMode, 2828 DWORD dwEncoding, 2829 CHAR* pSrcStr, 2830 UINT* pcSrcSize, 2831 WCHAR* pDstStr, 2832 UINT* pcDstSize) 2833 { 2834 return ConvertINetMultiByteToUnicode(pdwMode, dwEncoding, 2835 pSrcStr, (LPINT)pcSrcSize, pDstStr, (LPINT)pcDstSize); 2836 } 2837 2838 static HRESULT WINAPI fnIMultiLanguage3_ConvertStringFromUnicode( 2839 IMultiLanguage3* iface, 2840 DWORD* pdwMode, 2841 DWORD dwEncoding, 2842 WCHAR* pSrcStr, 2843 UINT* pcSrcSize, 2844 CHAR* pDstStr, 2845 UINT* pcDstSize) 2846 { 2847 return ConvertINetUnicodeToMultiByte(pdwMode, dwEncoding, 2848 pSrcStr, (LPINT)pcSrcSize, pDstStr, (LPINT)pcDstSize); 2849 } 2850 2851 static HRESULT WINAPI fnIMultiLanguage3_ConvertStringReset( 2852 IMultiLanguage3* iface) 2853 { 2854 FIXME("\n"); 2855 return E_NOTIMPL; 2856 } 2857 2858 static HRESULT WINAPI fnIMultiLanguage3_GetRfc1766FromLcid( 2859 IMultiLanguage3* iface, 2860 LCID lcid, 2861 BSTR* pbstrRfc1766) 2862 { 2863 WCHAR buf[MAX_RFC1766_NAME]; 2864 2865 TRACE("%p %04x %p\n", iface, lcid, pbstrRfc1766); 2866 if (!pbstrRfc1766) 2867 return E_INVALIDARG; 2868 2869 if (!lcid_to_rfc1766W( lcid, buf, MAX_RFC1766_NAME )) 2870 { 2871 *pbstrRfc1766 = SysAllocString( buf ); 2872 return S_OK; 2873 } 2874 return E_FAIL; 2875 } 2876 2877 static HRESULT WINAPI fnIMultiLanguage3_GetLcidFromRfc1766( 2878 IMultiLanguage3* iface, 2879 LCID* pLocale, 2880 BSTR bstrRfc1766) 2881 { 2882 HRESULT hr; 2883 IEnumRfc1766 *rfc1766; 2884 2885 TRACE("%p %p %s\n", iface, pLocale, debugstr_w(bstrRfc1766)); 2886 2887 if (!pLocale || !bstrRfc1766) 2888 return E_INVALIDARG; 2889 2890 hr = IMultiLanguage3_EnumRfc1766(iface, 0, &rfc1766); 2891 if (FAILED(hr)) 2892 return hr; 2893 2894 hr = lcid_from_rfc1766(rfc1766, pLocale, bstrRfc1766); 2895 2896 IEnumRfc1766_Release(rfc1766); 2897 return hr; 2898 } 2899 2900 static HRESULT WINAPI fnIMultiLanguage3_EnumRfc1766( 2901 IMultiLanguage3* iface, 2902 LANGID LangId, 2903 IEnumRfc1766** ppEnumRfc1766) 2904 { 2905 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 2906 2907 TRACE("%p %p\n", This, ppEnumRfc1766); 2908 2909 return EnumRfc1766_create(LangId, ppEnumRfc1766); 2910 } 2911 2912 static HRESULT WINAPI fnIMultiLanguage3_GetRfc1766Info( 2913 IMultiLanguage3* iface, 2914 LCID Locale, 2915 LANGID LangId, 2916 PRFC1766INFO pRfc1766Info) 2917 { 2918 static LANGID last_lang = -1; 2919 LCTYPE type = LOCALE_SLANGUAGE; 2920 2921 TRACE("(%p, 0x%04x, 0x%04x, %p)\n", iface, Locale, LangId, pRfc1766Info); 2922 2923 if (!pRfc1766Info) 2924 return E_INVALIDARG; 2925 2926 if ((PRIMARYLANGID(Locale) == LANG_ENGLISH) || 2927 (PRIMARYLANGID(Locale) == LANG_CHINESE) || 2928 (PRIMARYLANGID(Locale) == LANG_ARABIC)) { 2929 2930 if (!SUBLANGID(Locale)) 2931 type = LOCALE_SENGLANGUAGE; /* suppress country */ 2932 } 2933 else 2934 { 2935 if (!SUBLANGID(Locale)) { 2936 TRACE("SUBLANGID missing in 0x%04x\n", Locale); 2937 return E_FAIL; 2938 } 2939 } 2940 2941 pRfc1766Info->lcid = Locale; 2942 pRfc1766Info->wszRfc1766[0] = 0; 2943 pRfc1766Info->wszLocaleName[0] = 0; 2944 2945 if ((PRIMARYLANGID(LangId) != LANG_ENGLISH) && 2946 (last_lang != LangId)) { 2947 FIXME("Only English names supported (requested: 0x%04x)\n", LangId); 2948 last_lang = LangId; 2949 } 2950 2951 if ((!lcid_to_rfc1766W(Locale, pRfc1766Info->wszRfc1766, MAX_RFC1766_NAME)) && 2952 (GetLocaleInfoW(Locale, type, pRfc1766Info->wszLocaleName, MAX_LOCALE_NAME) > 0)) 2953 return S_OK; 2954 2955 /* Locale not supported */ 2956 return E_INVALIDARG; 2957 } 2958 2959 static HRESULT WINAPI fnIMultiLanguage3_CreateConvertCharset( 2960 IMultiLanguage3* iface, 2961 UINT src_cp, 2962 UINT dst_cp, 2963 DWORD prop, 2964 IMLangConvertCharset** convert_charset) 2965 { 2966 HRESULT hr; 2967 2968 TRACE("(%u %u 0x%08x %p)\n", src_cp, dst_cp, prop, convert_charset); 2969 2970 hr = MLangConvertCharset_create(NULL, (void**)convert_charset); 2971 if (FAILED(hr)) return hr; 2972 2973 return IMLangConvertCharset_Initialize(*convert_charset, src_cp, dst_cp, prop); 2974 } 2975 2976 static HRESULT WINAPI fnIMultiLanguage3_ConvertStringInIStream( 2977 IMultiLanguage3* iface, 2978 DWORD* pdwMode, 2979 DWORD dwFlag, 2980 WCHAR* lpFallBack, 2981 DWORD dwSrcEncoding, 2982 DWORD dwDstEncoding, 2983 IStream* pstmIn, 2984 IStream* pstmOut) 2985 { 2986 char *src, *dst = NULL; 2987 INT srclen, dstlen; 2988 STATSTG stat; 2989 HRESULT hr; 2990 2991 TRACE("%p %0x8 %s %u %u %p %p\n", 2992 pdwMode, dwFlag, debugstr_w(lpFallBack), dwSrcEncoding, dwDstEncoding, pstmIn, pstmOut); 2993 2994 FIXME("dwFlag and lpFallBack not handled\n"); 2995 2996 hr = IStream_Stat(pstmIn, &stat, STATFLAG_NONAME); 2997 if (FAILED(hr)) return hr; 2998 2999 if (stat.cbSize.QuadPart > MAXLONG) return E_INVALIDARG; 3000 if (!(src = HeapAlloc(GetProcessHeap(), 0, stat.cbSize.QuadPart))) return E_OUTOFMEMORY; 3001 3002 hr = IStream_Read(pstmIn, src, stat.cbSize.QuadPart, (ULONG *)&srclen); 3003 if (FAILED(hr)) goto exit; 3004 3005 hr = ConvertINetString(pdwMode, dwSrcEncoding, dwDstEncoding, src, &srclen, NULL, &dstlen); 3006 if (FAILED(hr)) goto exit; 3007 3008 if (!(dst = HeapAlloc(GetProcessHeap(), 0, dstlen))) 3009 { 3010 hr = E_OUTOFMEMORY; 3011 goto exit; 3012 } 3013 hr = ConvertINetString(pdwMode, dwSrcEncoding, dwDstEncoding, src, &srclen, dst, &dstlen); 3014 if (FAILED(hr)) goto exit; 3015 3016 hr = IStream_Write(pstmOut, dst, dstlen, NULL); 3017 3018 exit: 3019 HeapFree(GetProcessHeap(), 0, src); 3020 HeapFree(GetProcessHeap(), 0, dst); 3021 return hr; 3022 } 3023 3024 static HRESULT WINAPI fnIMultiLanguage3_ConvertStringToUnicodeEx( 3025 IMultiLanguage3* iface, 3026 DWORD* pdwMode, 3027 DWORD dwEncoding, 3028 CHAR* pSrcStr, 3029 UINT* pcSrcSize, 3030 WCHAR* pDstStr, 3031 UINT* pcDstSize, 3032 DWORD dwFlag, 3033 WCHAR* lpFallBack) 3034 { 3035 if (dwFlag || lpFallBack) 3036 FIXME("Ignoring dwFlag (0x%x/%d) and lpFallBack (%p)\n", 3037 dwFlag, dwFlag, lpFallBack); 3038 3039 return ConvertINetMultiByteToUnicode(pdwMode, dwEncoding, 3040 pSrcStr, (LPINT)pcSrcSize, pDstStr, (LPINT)pcDstSize); 3041 } 3042 3043 /***************************************************************************** 3044 * MultiLanguage2::ConvertStringToUnicodeEx 3045 * 3046 * Translates the multibyte string from the specified code page to Unicode. 3047 * 3048 * PARAMS 3049 * see ConvertStringToUnicode 3050 * dwFlag 3051 * lpFallBack if dwFlag contains MLCONVCHARF_USEDEFCHAR, lpFallBack string used 3052 * instead unconvertible characters. 3053 * 3054 * RETURNS 3055 * S_OK Success. 3056 * S_FALSE The conversion is not supported. 3057 * E_FAIL Some error has occurred. 3058 * 3059 * TODO: handle dwFlag and lpFallBack 3060 */ 3061 static HRESULT WINAPI fnIMultiLanguage3_ConvertStringFromUnicodeEx( 3062 IMultiLanguage3* This, 3063 DWORD* pdwMode, 3064 DWORD dwEncoding, 3065 WCHAR* pSrcStr, 3066 UINT* pcSrcSize, 3067 CHAR* pDstStr, 3068 UINT* pcDstSize, 3069 DWORD dwFlag, 3070 WCHAR* lpFallBack) 3071 { 3072 FIXME("\n"); 3073 return ConvertINetUnicodeToMultiByte(pdwMode, dwEncoding, 3074 pSrcStr, (LPINT)pcSrcSize, pDstStr, (LPINT)pcDstSize); 3075 } 3076 3077 static HRESULT WINAPI fnIMultiLanguage3_DetectCodepageInIStream( 3078 IMultiLanguage3* iface, 3079 DWORD dwFlag, 3080 DWORD dwPrefWinCodePage, 3081 IStream* pstmIn, 3082 DetectEncodingInfo* lpEncoding, 3083 INT* pnScores) 3084 { 3085 FIXME("\n"); 3086 return E_NOTIMPL; 3087 } 3088 3089 static HRESULT WINAPI fnIMultiLanguage3_DetectInputCodepage( 3090 IMultiLanguage3* iface, 3091 DWORD dwFlag, 3092 DWORD dwPrefWinCodePage, 3093 CHAR* pSrcStr, 3094 INT* pcSrcSize, 3095 DetectEncodingInfo* lpEncoding, 3096 INT* pnScores) 3097 { 3098 FIXME("\n"); 3099 return E_NOTIMPL; 3100 } 3101 3102 static HRESULT WINAPI fnIMultiLanguage3_ValidateCodePage( 3103 IMultiLanguage3* iface, 3104 UINT uiCodePage, 3105 HWND hwnd) 3106 { 3107 return IMultiLanguage3_ValidateCodePageEx(iface,uiCodePage,hwnd,0); 3108 } 3109 3110 static HRESULT WINAPI fnIMultiLanguage3_GetCodePageDescription( 3111 IMultiLanguage3* iface, 3112 UINT uiCodePage, 3113 LCID lcid, 3114 LPWSTR lpWideCharStr, 3115 int cchWideChar) 3116 { 3117 /* Find first instance */ 3118 unsigned int i,n; 3119 3120 TRACE ("%u, %04x, %p, %d\n", uiCodePage, lcid, lpWideCharStr, cchWideChar); 3121 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 3122 { 3123 for (n = 0; n < mlang_data[i].number_of_cp; n++) 3124 { 3125 if (mlang_data[i].mime_cp_info[n].cp == uiCodePage) 3126 { 3127 MultiByteToWideChar(CP_ACP, 0, 3128 mlang_data[i].mime_cp_info[n].description, 3129 -1, lpWideCharStr, cchWideChar); 3130 return S_OK; 3131 } 3132 } 3133 } 3134 3135 return S_FALSE; 3136 } 3137 3138 static HRESULT WINAPI fnIMultiLanguage3_IsCodePageInstallable( 3139 IMultiLanguage3* iface, 3140 UINT uiCodePage) 3141 { 3142 TRACE("%u\n", uiCodePage); 3143 3144 /* FIXME: the installable set is usually larger than the set of valid codepages */ 3145 return IMultiLanguage3_ValidateCodePageEx(iface, uiCodePage, NULL, CPIOD_PEEK); 3146 } 3147 3148 static HRESULT WINAPI fnIMultiLanguage3_SetMimeDBSource( 3149 IMultiLanguage3* iface, 3150 MIMECONTF dwSource) 3151 { 3152 FIXME("0x%08x\n", dwSource); 3153 return S_OK; 3154 } 3155 3156 static HRESULT WINAPI fnIMultiLanguage3_GetNumberOfScripts( 3157 IMultiLanguage3* iface, 3158 UINT* pnScripts) 3159 { 3160 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 3161 3162 TRACE("%p %p\n", This, pnScripts); 3163 3164 if (!pnScripts) return S_FALSE; 3165 3166 *pnScripts = This->total_scripts; 3167 return S_OK; 3168 } 3169 3170 static HRESULT WINAPI fnIMultiLanguage3_EnumScripts( 3171 IMultiLanguage3* iface, 3172 DWORD dwFlags, 3173 LANGID LangId, 3174 IEnumScript** ppEnumScript) 3175 { 3176 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 3177 3178 TRACE("%p %08x %04x %p\n", This, dwFlags, LangId, ppEnumScript); 3179 3180 return EnumScript_create( This, dwFlags, LangId, ppEnumScript ); 3181 } 3182 3183 static HRESULT WINAPI fnIMultiLanguage3_ValidateCodePageEx( 3184 IMultiLanguage3* iface, 3185 UINT uiCodePage, 3186 HWND hwnd, 3187 DWORD dwfIODControl) 3188 { 3189 unsigned int i; 3190 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 3191 3192 TRACE("%p %u %p %08x\n", This, uiCodePage, hwnd, dwfIODControl); 3193 3194 /* quick check for kernel32 supported code pages */ 3195 if (IsValidCodePage(uiCodePage)) 3196 return S_OK; 3197 3198 /* check for mlang supported code pages */ 3199 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 3200 { 3201 UINT n; 3202 for (n = 0; n < mlang_data[i].number_of_cp; n++) 3203 { 3204 if (mlang_data[i].mime_cp_info[n].cp == uiCodePage) 3205 return S_OK; 3206 } 3207 } 3208 3209 if (dwfIODControl != CPIOD_PEEK) 3210 FIXME("Request to install codepage language pack not handled\n"); 3211 3212 return S_FALSE; 3213 } 3214 3215 static HRESULT WINAPI fnIMultiLanguage3_DetectOutboundCodePage( 3216 IMultiLanguage3 *iface, 3217 DWORD dwFlags, 3218 LPCWSTR lpWideCharStr, 3219 UINT cchWideChar, 3220 UINT *puiPreferredCodePages, 3221 UINT nPreferredCodePages, 3222 UINT *puiDetectedCodePages, 3223 UINT *pnDetectedCodePages, 3224 WCHAR *lpSpecialChar) 3225 { 3226 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 3227 3228 FIXME("(%p)->(%08x %s %p %u %p %p(%u) %s)\n", This, dwFlags, debugstr_w(lpWideCharStr), 3229 puiPreferredCodePages, nPreferredCodePages, puiDetectedCodePages, 3230 pnDetectedCodePages, pnDetectedCodePages ? *pnDetectedCodePages : 0, 3231 debugstr_w(lpSpecialChar)); 3232 3233 if (!puiDetectedCodePages || !pnDetectedCodePages || !*pnDetectedCodePages) 3234 return E_INVALIDARG; 3235 3236 puiDetectedCodePages[0] = CP_UTF8; 3237 *pnDetectedCodePages = 1; 3238 return S_OK; 3239 } 3240 3241 static HRESULT WINAPI fnIMultiLanguage3_DetectOutboundCodePageInIStream( 3242 IMultiLanguage3 *iface, 3243 DWORD dwFlags, 3244 IStream *pStrIn, 3245 UINT *puiPreferredCodePages, 3246 UINT nPreferredCodePages, 3247 UINT *puiDetectedCodePages, 3248 UINT *pnDetectedCodePages, 3249 WCHAR *lpSpecialChar) 3250 { 3251 MLang_impl *This = impl_from_IMultiLanguage3( iface ); 3252 3253 FIXME("(%p)->(%08x %p %p %u %p %p(%u) %s)\n", This, dwFlags, pStrIn, 3254 puiPreferredCodePages, nPreferredCodePages, puiDetectedCodePages, 3255 pnDetectedCodePages, pnDetectedCodePages ? *pnDetectedCodePages : 0, 3256 debugstr_w(lpSpecialChar)); 3257 3258 if (!puiDetectedCodePages || !pnDetectedCodePages || !*pnDetectedCodePages) 3259 return E_INVALIDARG; 3260 3261 puiDetectedCodePages[0] = CP_UTF8; 3262 *pnDetectedCodePages = 1; 3263 return S_OK; 3264 } 3265 3266 static const IMultiLanguage3Vtbl IMultiLanguage3_vtbl = 3267 { 3268 fnIMultiLanguage3_QueryInterface, 3269 fnIMultiLanguage3_AddRef, 3270 fnIMultiLanguage3_Release, 3271 fnIMultiLanguage3_GetNumberOfCodePageInfo, 3272 fnIMultiLanguage3_GetCodePageInfo, 3273 fnIMultiLanguage3_GetFamilyCodePage, 3274 fnIMultiLanguage3_EnumCodePages, 3275 fnIMultiLanguage3_GetCharsetInfo, 3276 fnIMultiLanguage3_IsConvertible, 3277 fnIMultiLanguage3_ConvertString, 3278 fnIMultiLanguage3_ConvertStringToUnicode, 3279 fnIMultiLanguage3_ConvertStringFromUnicode, 3280 fnIMultiLanguage3_ConvertStringReset, 3281 fnIMultiLanguage3_GetRfc1766FromLcid, 3282 fnIMultiLanguage3_GetLcidFromRfc1766, 3283 fnIMultiLanguage3_EnumRfc1766, 3284 fnIMultiLanguage3_GetRfc1766Info, 3285 fnIMultiLanguage3_CreateConvertCharset, 3286 fnIMultiLanguage3_ConvertStringInIStream, 3287 fnIMultiLanguage3_ConvertStringToUnicodeEx, 3288 fnIMultiLanguage3_ConvertStringFromUnicodeEx, 3289 fnIMultiLanguage3_DetectCodepageInIStream, 3290 fnIMultiLanguage3_DetectInputCodepage, 3291 fnIMultiLanguage3_ValidateCodePage, 3292 fnIMultiLanguage3_GetCodePageDescription, 3293 fnIMultiLanguage3_IsCodePageInstallable, 3294 fnIMultiLanguage3_SetMimeDBSource, 3295 fnIMultiLanguage3_GetNumberOfScripts, 3296 fnIMultiLanguage3_EnumScripts, 3297 fnIMultiLanguage3_ValidateCodePageEx, 3298 fnIMultiLanguage3_DetectOutboundCodePage, 3299 fnIMultiLanguage3_DetectOutboundCodePageInIStream 3300 }; 3301 3302 /******************************************************************************/ 3303 3304 static inline MLang_impl *impl_from_IMLangFontLink2( IMLangFontLink2 *iface ) 3305 { 3306 return CONTAINING_RECORD( iface, MLang_impl, IMLangFontLink2_iface ); 3307 } 3308 3309 static HRESULT WINAPI fnIMLangFontLink2_QueryInterface( 3310 IMLangFontLink2 * iface, 3311 REFIID riid, 3312 void** ppvObject) 3313 { 3314 MLang_impl *This = impl_from_IMLangFontLink2( iface ); 3315 return IMultiLanguage3_QueryInterface( &This->IMultiLanguage3_iface, riid, ppvObject ); 3316 } 3317 3318 static ULONG WINAPI fnIMLangFontLink2_AddRef( IMLangFontLink2* iface ) 3319 { 3320 MLang_impl *This = impl_from_IMLangFontLink2( iface ); 3321 return IMultiLanguage3_AddRef( &This->IMultiLanguage3_iface ); 3322 } 3323 3324 static ULONG WINAPI fnIMLangFontLink2_Release( IMLangFontLink2* iface ) 3325 { 3326 MLang_impl *This = impl_from_IMLangFontLink2( iface ); 3327 return IMultiLanguage3_Release( &This->IMultiLanguage3_iface ); 3328 } 3329 3330 static HRESULT WINAPI fnIMLangFontLink2_GetCharCodePages( IMLangFontLink2* iface, 3331 WCHAR ch_src, DWORD *ret_codepages) 3332 { 3333 MLang_impl *This = impl_from_IMLangFontLink2(iface); 3334 unsigned int i; 3335 3336 TRACE("(%p)->(%s %p)\n", This, debugstr_wn(&ch_src, 1), ret_codepages); 3337 3338 *ret_codepages = 0; 3339 3340 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 3341 { 3342 BOOL used_dc; 3343 CHAR buf; 3344 3345 WideCharToMultiByte(mlang_data[i].family_codepage, WC_NO_BEST_FIT_CHARS, 3346 &ch_src, 1, &buf, 1, NULL, &used_dc); 3347 3348 /* If default char is not used, current codepage include the given symbol */ 3349 if (!used_dc) 3350 { 3351 DWORD codepages; 3352 3353 IMLangFontLink2_CodePageToCodePages(iface, 3354 mlang_data[i].family_codepage, &codepages); 3355 *ret_codepages |= codepages; 3356 } 3357 } 3358 return S_OK; 3359 } 3360 3361 static HRESULT WINAPI fnIMLangFontLink2_GetStrCodePages( IMLangFontLink2* iface, 3362 const WCHAR *src, LONG src_len, DWORD priority_cp, 3363 DWORD *codepages, LONG *ret_len) 3364 { 3365 MLang_impl *This = impl_from_IMLangFontLink2(iface); 3366 LONG i; 3367 DWORD cps = 0; 3368 3369 TRACE("(%p)->(%s:%d %x %p %p)\n", This, debugstr_wn(src, src_len), src_len, priority_cp, 3370 codepages, ret_len); 3371 3372 if (codepages) *codepages = 0; 3373 if (ret_len) *ret_len = 0; 3374 3375 if (!src || !src_len || src_len < 0) 3376 return E_INVALIDARG; 3377 3378 for (i = 0; i < src_len; i++) 3379 { 3380 DWORD cp; 3381 HRESULT ret; 3382 3383 ret = IMLangFontLink2_GetCharCodePages(iface, src[i], &cp); 3384 if (ret != S_OK) return E_FAIL; 3385 3386 if (!cps) cps = cp; 3387 else cps &= cp; 3388 3389 /* FIXME: not tested */ 3390 if (priority_cp & cps) break; 3391 } 3392 3393 if (codepages) *codepages = cps; 3394 if (ret_len) *ret_len = min( i + 1, src_len ); 3395 return S_OK; 3396 } 3397 3398 static HRESULT WINAPI fnIMLangFontLink2_CodePageToCodePages(IMLangFontLink2* iface, 3399 UINT codepage, 3400 DWORD *codepages) 3401 { 3402 MLang_impl *This = impl_from_IMLangFontLink2(iface); 3403 CHARSETINFO cs; 3404 BOOL rc; 3405 3406 TRACE("(%p)->(%u %p)\n", This, codepage, codepages); 3407 3408 rc = TranslateCharsetInfo((DWORD*)(DWORD_PTR)codepage, &cs, TCI_SRCCODEPAGE); 3409 if (rc) 3410 { 3411 *codepages = cs.fs.fsCsb[0]; 3412 TRACE("resulting codepages 0x%x\n", *codepages); 3413 return S_OK; 3414 } 3415 3416 TRACE("codepage not found\n"); 3417 *codepages = 0; 3418 return E_FAIL; 3419 } 3420 3421 static HRESULT WINAPI fnIMLangFontLink2_CodePagesToCodePage(IMLangFontLink2* iface, 3422 DWORD codepages, UINT def_codepage, UINT *codepage) 3423 { 3424 MLang_impl *This = impl_from_IMLangFontLink2(iface); 3425 DWORD mask = 0; 3426 CHARSETINFO cs; 3427 BOOL rc; 3428 UINT i; 3429 3430 TRACE("(%p)->(0x%x %u %p)\n", This, codepages, def_codepage, codepage); 3431 3432 *codepage = 0; 3433 3434 rc = TranslateCharsetInfo((DWORD*)(DWORD_PTR)def_codepage, &cs, TCI_SRCCODEPAGE); 3435 if (rc && (codepages & cs.fs.fsCsb[0])) 3436 { 3437 TRACE("Found Default Codepage\n"); 3438 *codepage = def_codepage; 3439 return S_OK; 3440 } 3441 3442 for (i = 0; i < 32; i++) 3443 { 3444 mask = 1 << i; 3445 if (codepages & mask) 3446 { 3447 DWORD Csb[2]; 3448 Csb[0] = mask; 3449 Csb[1] = 0x0; 3450 rc = TranslateCharsetInfo(Csb, &cs, TCI_SRCFONTSIG); 3451 if (!rc) 3452 continue; 3453 3454 TRACE("Falling back to least significant found CodePage %u\n", 3455 cs.ciACP); 3456 *codepage = cs.ciACP; 3457 return S_OK; 3458 } 3459 } 3460 3461 TRACE("no codepage found\n"); 3462 return E_FAIL; 3463 } 3464 3465 static HRESULT WINAPI fnIMLangFontLink2_GetFontCodePages(IMLangFontLink2 *iface, 3466 HDC hdc, HFONT hfont, DWORD *codepages) 3467 { 3468 MLang_impl *This = impl_from_IMLangFontLink2(iface); 3469 FONTSIGNATURE fontsig; 3470 HFONT old_font; 3471 3472 TRACE("(%p)->(%p %p %p)\n", This, hdc, hfont, codepages); 3473 3474 old_font = SelectObject(hdc, hfont); 3475 GetTextCharsetInfo(hdc, &fontsig, 0); 3476 SelectObject(hdc, old_font); 3477 3478 *codepages = fontsig.fsCsb[0]; 3479 TRACE("ret 0x%x\n", fontsig.fsCsb[0]); 3480 3481 return S_OK; 3482 } 3483 3484 static HRESULT WINAPI fnIMLangFontLink2_ReleaseFont(IMLangFontLink2* This, 3485 HFONT hFont) 3486 { 3487 TRACE("(%p)->%p\n",This, hFont); 3488 3489 return release_font(hFont); 3490 } 3491 3492 static HRESULT WINAPI fnIMLangFontLink2_ResetFontMapping(IMLangFontLink2* This) 3493 { 3494 TRACE("(%p)\n",This); 3495 3496 return clear_font_cache(); 3497 } 3498 3499 static HRESULT WINAPI fnIMLangFontLink2_MapFont(IMLangFontLink2* This, 3500 HDC hDC, DWORD dwCodePages, WCHAR chSrc, HFONT *pFont) 3501 { 3502 HFONT old_font; 3503 3504 TRACE("(%p)->%p %08x %04x %p\n",This, hDC, dwCodePages, chSrc, pFont); 3505 3506 if (!hDC) return E_FAIL; 3507 3508 if (dwCodePages != 0) 3509 { 3510 old_font = GetCurrentObject(hDC, OBJ_FONT); 3511 return map_font(hDC, dwCodePages, old_font, pFont); 3512 } 3513 else 3514 { 3515 if (pFont == NULL) return E_INVALIDARG; 3516 FIXME("the situation where dwCodepages is set to zero is not implemented\n"); 3517 return E_FAIL; 3518 } 3519 } 3520 3521 static HRESULT WINAPI fnIMLangFontLink2_GetFontUnicodeRanges(IMLangFontLink2* This, 3522 HDC hDC, UINT *puiRanges, UNICODERANGE *pUranges) 3523 { 3524 DWORD size; 3525 GLYPHSET *gs; 3526 3527 TRACE("(%p)->%p %p %p\n", This, hDC, puiRanges, pUranges); 3528 3529 if (!puiRanges) return E_INVALIDARG; 3530 if (!(size = GetFontUnicodeRanges(hDC, NULL))) return E_FAIL; 3531 if (!(gs = HeapAlloc(GetProcessHeap(), 0, size))) return E_OUTOFMEMORY; 3532 3533 GetFontUnicodeRanges(hDC, gs); 3534 *puiRanges = gs->cRanges; 3535 if (pUranges) 3536 { 3537 UINT i; 3538 for (i = 0; i < gs->cRanges; i++) 3539 { 3540 if (i >= *puiRanges) break; 3541 pUranges[i].wcFrom = gs->ranges[i].wcLow; 3542 pUranges[i].wcTo = gs->ranges[i].wcLow + gs->ranges[i].cGlyphs; 3543 } 3544 *puiRanges = i; 3545 } 3546 HeapFree(GetProcessHeap(), 0, gs); 3547 return S_OK; 3548 } 3549 3550 static HRESULT WINAPI fnIMLangFontLink2_GetScriptFontInfo(IMLangFontLink2* This, 3551 SCRIPT_ID sid, DWORD dwFlags, UINT *puiFonts, 3552 SCRIPTFONTINFO *pScriptFont) 3553 { 3554 UINT i, j; 3555 3556 TRACE("(%p)->%u %x %p %p\n", This, sid, dwFlags, puiFonts, pScriptFont); 3557 3558 if (!dwFlags) dwFlags = SCRIPTCONTF_PROPORTIONAL_FONT; 3559 3560 for (i = 0, j = 0; i < ARRAY_SIZE(mlang_data); i++) 3561 { 3562 if (sid == mlang_data[i].sid) 3563 { 3564 if (pScriptFont) 3565 { 3566 if (j >= *puiFonts) break; 3567 3568 pScriptFont[j].scripts = 1 << mlang_data[i].sid; 3569 if (dwFlags == SCRIPTCONTF_FIXED_FONT) 3570 { 3571 MultiByteToWideChar(CP_ACP, 0, mlang_data[i].fixed_font, -1, 3572 pScriptFont[j].wszFont, MAX_MIMEFACE_NAME); 3573 } 3574 else if (dwFlags == SCRIPTCONTF_PROPORTIONAL_FONT) 3575 { 3576 MultiByteToWideChar(CP_ACP, 0, mlang_data[i].proportional_font, -1, 3577 pScriptFont[j].wszFont, MAX_MIMEFACE_NAME); 3578 } 3579 } 3580 j++; 3581 } 3582 } 3583 *puiFonts = j; 3584 return S_OK; 3585 } 3586 3587 static HRESULT WINAPI fnIMLangFontLink2_CodePageToScriptID(IMLangFontLink2* This, 3588 UINT uiCodePage, SCRIPT_ID *pSid) 3589 { 3590 UINT i; 3591 3592 TRACE("(%p)->%i %p\n", This, uiCodePage, pSid); 3593 3594 if (uiCodePage == CP_UNICODE) return E_FAIL; 3595 3596 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 3597 { 3598 if (uiCodePage == mlang_data[i].family_codepage) 3599 { 3600 if (pSid) *pSid = mlang_data[i].sid; 3601 return S_OK; 3602 } 3603 } 3604 return E_FAIL; 3605 } 3606 3607 static const IMLangFontLink2Vtbl IMLangFontLink2_vtbl = 3608 { 3609 fnIMLangFontLink2_QueryInterface, 3610 fnIMLangFontLink2_AddRef, 3611 fnIMLangFontLink2_Release, 3612 fnIMLangFontLink2_GetCharCodePages, 3613 fnIMLangFontLink2_GetStrCodePages, 3614 fnIMLangFontLink2_CodePageToCodePages, 3615 fnIMLangFontLink2_CodePagesToCodePage, 3616 fnIMLangFontLink2_GetFontCodePages, 3617 fnIMLangFontLink2_ReleaseFont, 3618 fnIMLangFontLink2_ResetFontMapping, 3619 fnIMLangFontLink2_MapFont, 3620 fnIMLangFontLink2_GetFontUnicodeRanges, 3621 fnIMLangFontLink2_GetScriptFontInfo, 3622 fnIMLangFontLink2_CodePageToScriptID 3623 }; 3624 3625 /******************************************************************************/ 3626 3627 static inline MLang_impl *impl_from_IMLangLineBreakConsole( IMLangLineBreakConsole *iface ) 3628 { 3629 return CONTAINING_RECORD( iface, MLang_impl, IMLangLineBreakConsole_iface ); 3630 } 3631 3632 static HRESULT WINAPI fnIMLangLineBreakConsole_QueryInterface( 3633 IMLangLineBreakConsole* iface, 3634 REFIID riid, 3635 void** ppvObject) 3636 { 3637 MLang_impl *This = impl_from_IMLangLineBreakConsole( iface ); 3638 return IMultiLanguage3_QueryInterface( &This->IMultiLanguage3_iface, riid, ppvObject ); 3639 } 3640 3641 static ULONG WINAPI fnIMLangLineBreakConsole_AddRef( 3642 IMLangLineBreakConsole* iface ) 3643 { 3644 MLang_impl *This = impl_from_IMLangLineBreakConsole( iface ); 3645 return IMultiLanguage3_AddRef( &This->IMultiLanguage3_iface ); 3646 } 3647 3648 static ULONG WINAPI fnIMLangLineBreakConsole_Release( 3649 IMLangLineBreakConsole* iface ) 3650 { 3651 MLang_impl *This = impl_from_IMLangLineBreakConsole( iface ); 3652 return IMultiLanguage3_Release( &This->IMultiLanguage3_iface ); 3653 } 3654 3655 static HRESULT WINAPI fnIMLangLineBreakConsole_BreakLineML( 3656 IMLangLineBreakConsole* iface, 3657 IMLangString* pSrcMLStr, 3658 LONG lSrcPos, 3659 LONG lSrcLen, 3660 LONG cMinColumns, 3661 LONG cMaxColumns, 3662 LONG* plLineLen, 3663 LONG* plSkipLen) 3664 { 3665 FIXME("(%p)->%p %i %i %i %i %p %p\n", iface, pSrcMLStr, lSrcPos, lSrcLen, cMinColumns, cMaxColumns, plLineLen, plSkipLen); 3666 return E_NOTIMPL; 3667 } 3668 3669 static HRESULT WINAPI fnIMLangLineBreakConsole_BreakLineW( 3670 IMLangLineBreakConsole* iface, 3671 LCID locale, 3672 const WCHAR* pszSrc, 3673 LONG cchSrc, 3674 LONG cMaxColumns, 3675 LONG* pcchLine, 3676 LONG* pcchSkip ) 3677 { 3678 FIXME("(%p)->%i %s %i %i %p %p\n", iface, locale, debugstr_wn(pszSrc,cchSrc), cchSrc, cMaxColumns, pcchLine, pcchSkip); 3679 3680 *pcchLine = cchSrc; 3681 *pcchSkip = 0; 3682 return S_OK; 3683 } 3684 3685 static HRESULT WINAPI fnIMLangLineBreakConsole_BreakLineA( 3686 IMLangLineBreakConsole* iface, 3687 LCID locale, 3688 UINT uCodePage, 3689 const CHAR* pszSrc, 3690 LONG cchSrc, 3691 LONG cMaxColumns, 3692 LONG* pcchLine, 3693 LONG* pcchSkip) 3694 { 3695 LONG i, line = cchSrc, skip = 0; 3696 3697 FIXME("(%p)->%i %i %s %i %i %p %p\n", iface, locale, uCodePage, debugstr_an(pszSrc,cchSrc), cchSrc, cMaxColumns, pcchLine, pcchSkip); 3698 3699 if (uCodePage == CP_USASCII && cchSrc > cMaxColumns) 3700 { 3701 for (line = cMaxColumns, i = cMaxColumns - 1; i >= 0; i--) 3702 { 3703 if (pszSrc[i] == ' ') 3704 { 3705 while (i >= 0 && pszSrc[i] == ' ') 3706 { 3707 i--; 3708 line--; 3709 skip++; 3710 } 3711 break; 3712 } 3713 } 3714 } 3715 *pcchLine = line; 3716 *pcchSkip = skip; 3717 return S_OK; 3718 } 3719 3720 static const IMLangLineBreakConsoleVtbl IMLangLineBreakConsole_vtbl = 3721 { 3722 fnIMLangLineBreakConsole_QueryInterface, 3723 fnIMLangLineBreakConsole_AddRef, 3724 fnIMLangLineBreakConsole_Release, 3725 fnIMLangLineBreakConsole_BreakLineML, 3726 fnIMLangLineBreakConsole_BreakLineW, 3727 fnIMLangLineBreakConsole_BreakLineA 3728 }; 3729 3730 struct convert_charset { 3731 IMLangConvertCharset IMLangConvertCharset_iface; 3732 LONG ref; 3733 3734 UINT src_cp; 3735 UINT dst_cp; 3736 }; 3737 3738 static inline struct convert_charset *impl_from_IMLangConvertCharset(IMLangConvertCharset *iface) 3739 { 3740 return CONTAINING_RECORD(iface, struct convert_charset, IMLangConvertCharset_iface); 3741 } 3742 3743 static HRESULT WINAPI MLangConvertCharset_QueryInterface(IMLangConvertCharset *iface, REFIID riid, void **obj) 3744 { 3745 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3746 3747 TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), obj); 3748 3749 if (IsEqualIID(riid, &IID_IUnknown) || IsEqualIID(riid, &IID_IMLangConvertCharset)) 3750 { 3751 *obj = &This->IMLangConvertCharset_iface; 3752 IMLangConvertCharset_AddRef(iface); 3753 return S_OK; 3754 } 3755 3756 *obj = NULL; 3757 return E_NOINTERFACE; 3758 } 3759 3760 static ULONG WINAPI MLangConvertCharset_AddRef(IMLangConvertCharset *iface) 3761 { 3762 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3763 ULONG ref = InterlockedIncrement(&This->ref); 3764 TRACE("(%p)->(%u)\n", This, ref); 3765 return ref; 3766 } 3767 3768 static ULONG WINAPI MLangConvertCharset_Release(IMLangConvertCharset *iface) 3769 { 3770 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3771 ULONG ref = InterlockedDecrement(&This->ref); 3772 3773 TRACE("(%p)->(%u)\n", This, ref); 3774 if (!ref) 3775 { 3776 HeapFree(GetProcessHeap(), 0, This); 3777 UnlockModule(); 3778 } 3779 3780 return ref; 3781 } 3782 3783 static HRESULT WINAPI MLangConvertCharset_Initialize(IMLangConvertCharset *iface, 3784 UINT src_cp, UINT dst_cp, DWORD prop) 3785 { 3786 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3787 3788 TRACE("(%p)->(%u %u 0x%08x)\n", This, src_cp, dst_cp, prop); 3789 3790 prop &= ~MLCONVCHARF_USEDEFCHAR; 3791 if (prop) 3792 FIXME("property 0x%08x not supported\n", prop); 3793 3794 This->src_cp = src_cp; 3795 This->dst_cp = dst_cp; 3796 3797 return S_OK; 3798 } 3799 3800 static HRESULT WINAPI MLangConvertCharset_GetSourceCodePage(IMLangConvertCharset *iface, UINT *src_cp) 3801 { 3802 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3803 3804 TRACE("(%p)->(%p)\n", This, src_cp); 3805 3806 if (!src_cp) return E_INVALIDARG; 3807 *src_cp = This->src_cp; 3808 return S_OK; 3809 } 3810 3811 static HRESULT WINAPI MLangConvertCharset_GetDestinationCodePage(IMLangConvertCharset *iface, UINT *dst_cp) 3812 { 3813 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3814 3815 TRACE("(%p)->(%p)\n", This, dst_cp); 3816 3817 if (!dst_cp) return E_INVALIDARG; 3818 *dst_cp = This->dst_cp; 3819 return S_OK; 3820 } 3821 3822 static HRESULT WINAPI MLangConvertCharset_GetProperty(IMLangConvertCharset *iface, DWORD *prop) 3823 { 3824 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3825 FIXME("(%p)->(%p): stub\n", This, prop); 3826 return E_NOTIMPL; 3827 } 3828 3829 static HRESULT WINAPI MLangConvertCharset_DoConversion(IMLangConvertCharset *iface, BYTE *src, 3830 UINT *src_size, BYTE *dest, UINT *dest_size) 3831 { 3832 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3833 FIXME("(%p)->(%p %p %p %p): stub\n", This, src, src_size, dest, dest_size); 3834 return E_NOTIMPL; 3835 } 3836 3837 static HRESULT WINAPI MLangConvertCharset_DoConversionToUnicode(IMLangConvertCharset *iface, CHAR *src, 3838 UINT *src_size, WCHAR *dest, UINT *dest_size) 3839 { 3840 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3841 TRACE("(%p)->(%p %p %p %p)\n", This, src, src_size, dest, dest_size); 3842 return ConvertINetMultiByteToUnicode(NULL, This->src_cp, src, (INT*)src_size, dest, (INT*)dest_size); 3843 } 3844 3845 static HRESULT WINAPI MLangConvertCharset_DoConversionFromUnicode(IMLangConvertCharset *iface, 3846 WCHAR *src, UINT *src_size, CHAR *dest, UINT *dest_size) 3847 { 3848 struct convert_charset *This = impl_from_IMLangConvertCharset(iface); 3849 TRACE("(%p)->(%p %p %p %p)\n", This, src, src_size, dest, dest_size); 3850 return ConvertINetUnicodeToMultiByte(NULL, This->dst_cp, src, (INT*)src_size, dest, (INT*)dest_size); 3851 } 3852 3853 static const IMLangConvertCharsetVtbl MLangConvertCharsetVtbl = 3854 { 3855 MLangConvertCharset_QueryInterface, 3856 MLangConvertCharset_AddRef, 3857 MLangConvertCharset_Release, 3858 MLangConvertCharset_Initialize, 3859 MLangConvertCharset_GetSourceCodePage, 3860 MLangConvertCharset_GetDestinationCodePage, 3861 MLangConvertCharset_GetProperty, 3862 MLangConvertCharset_DoConversion, 3863 MLangConvertCharset_DoConversionToUnicode, 3864 MLangConvertCharset_DoConversionFromUnicode 3865 }; 3866 3867 static HRESULT MultiLanguage_create(IUnknown *pUnkOuter, LPVOID *ppObj) 3868 { 3869 MLang_impl *mlang; 3870 UINT i; 3871 3872 TRACE("Creating MultiLanguage object\n"); 3873 3874 if( pUnkOuter ) 3875 return CLASS_E_NOAGGREGATION; 3876 3877 mlang = HeapAlloc( GetProcessHeap(), 0, sizeof (MLang_impl) ); 3878 mlang->IMLangFontLink_iface.lpVtbl = &IMLangFontLink_vtbl; 3879 mlang->IMultiLanguage_iface.lpVtbl = &IMultiLanguage_vtbl; 3880 mlang->IMultiLanguage3_iface.lpVtbl = &IMultiLanguage3_vtbl; 3881 mlang->IMLangFontLink2_iface.lpVtbl = &IMLangFontLink2_vtbl; 3882 mlang->IMLangLineBreakConsole_iface.lpVtbl = &IMLangLineBreakConsole_vtbl; 3883 3884 mlang->total_cp = 0; 3885 for (i = 0; i < ARRAY_SIZE(mlang_data); i++) 3886 mlang->total_cp += mlang_data[i].number_of_cp; 3887 3888 /* do not enumerate unicode flavours */ 3889 mlang->total_scripts = ARRAY_SIZE(mlang_data) - 1; 3890 3891 mlang->ref = 1; 3892 *ppObj = &mlang->IMultiLanguage_iface; 3893 TRACE("returning %p\n", mlang); 3894 3895 LockModule(); 3896 3897 return S_OK; 3898 } 3899 3900 static HRESULT MLangConvertCharset_create(IUnknown *outer, void **obj) 3901 { 3902 struct convert_charset *convert; 3903 3904 if (outer) 3905 return CLASS_E_NOAGGREGATION; 3906 3907 *obj = NULL; 3908 3909 convert = HeapAlloc(GetProcessHeap(), 0, sizeof(struct convert_charset)); 3910 if (!convert) return E_OUTOFMEMORY; 3911 3912 convert->IMLangConvertCharset_iface.lpVtbl = &MLangConvertCharsetVtbl; 3913 convert->ref = 1; 3914 3915 *obj = &convert->IMLangConvertCharset_iface; 3916 3917 LockModule(); 3918 3919 return S_OK; 3920 } 3921 3922 /******************************************************************************/ 3923 3924 HRESULT WINAPI DllCanUnloadNow(void) 3925 { 3926 return dll_count == 0 ? S_OK : S_FALSE; 3927 } 3928 3929 static BOOL register_codepages(void) 3930 { 3931 const struct mlang_data *family; 3932 const MIME_CP_INFO *info; 3933 HKEY db_key, key; 3934 WCHAR buf[32]; 3935 LSTATUS status; 3936 3937 static const WCHAR db_key_nameW[] = { 3938 'M','I','M','E', 3939 '\\','D','a','t','a','b','a','s','e', 3940 '\\','C','o','d','e','p','a','g','e',0}; 3941 static const WCHAR familyW[] = {'F','a','m','i','l','y',0}; 3942 static const WCHAR formatW[] = {'%','u',0}; 3943 3944 status = RegCreateKeyW(HKEY_CLASSES_ROOT, db_key_nameW, &db_key); 3945 if (status != ERROR_SUCCESS) 3946 return FALSE; 3947 3948 for (family = mlang_data; family < mlang_data + ARRAY_SIZE(mlang_data); family++) 3949 { 3950 for (info = family->mime_cp_info; info < family->mime_cp_info + family->number_of_cp; info++) 3951 { 3952 swprintf(buf, formatW, info->cp); 3953 status = RegCreateKeyW(db_key, buf, &key); 3954 if (status != ERROR_SUCCESS) 3955 continue; 3956 3957 RegSetValueExA(key, "BodyCharset", 0, REG_SZ, (BYTE*)info->body_charset, 3958 strlen(info->body_charset) + 1); 3959 3960 if (info->cp == family->family_codepage) 3961 { 3962 RegSetValueExA(key, "FixedWidthFont", 0, REG_SZ, (BYTE*)family->fixed_font, 3963 strlen(family->fixed_font) + 1); 3964 RegSetValueExA(key, "ProportionalFont", 0, REG_SZ, (BYTE*)family->proportional_font, 3965 strlen(family->proportional_font) + 1); 3966 } 3967 else 3968 { 3969 RegSetValueExW(key, familyW, 0, REG_DWORD, (BYTE*)&family->family_codepage, 3970 sizeof(family->family_codepage)); 3971 } 3972 3973 RegCloseKey(key); 3974 } 3975 } 3976 3977 RegCloseKey(db_key); 3978 return TRUE; 3979 } 3980 3981 /*********************************************************************** 3982 * DllRegisterServer (MLANG.@) 3983 */ 3984 HRESULT WINAPI DllRegisterServer(void) 3985 { 3986 if(!register_codepages()) 3987 return E_FAIL; 3988 return __wine_register_resources( instance ); 3989 } 3990 3991 /*********************************************************************** 3992 * DllUnregisterServer (MLANG.@) 3993 */ 3994 HRESULT WINAPI DllUnregisterServer(void) 3995 { 3996 return __wine_unregister_resources( instance ); 3997 } 3998 3999 HRESULT WINAPI GetGlobalFontLinkObject(void **unknown) 4000 { 4001 if (!unknown) return E_INVALIDARG; 4002 4003 FIXME("%p: stub\n", unknown); 4004 4005 return S_FALSE; 4006 } 4007