// Copyright (C) Microsoft Corporation, 1994-1995 // File: unicwrap.cpp // Contents: Wrappers for all Unicode functions used in MSHTML. // Any Unicode parameters/structure fields/buffers are converted // to ANSI, and then the corresponding ANSI version of the function // is called. #include "ctlspriv.h" #ifdef UNICODE #ifndef WINNT #include "unicwrap.h" #undef TextOutW #undef ExtTextOutW int MbcsFromUnicode(LPSTR pstr, int cch, LPCWSTR pwstr, int cwch = -1); int UnicodeFromMbcs(LPWSTR pwstr, int cwch, LPCSTR pstr, int cch = -1); // Member: CConvertStr::Free // Synopsis: Frees string if alloc'd and initializes to NULL. void CConvertStr::Free() { if (_pstr != _ach && HIWORD64(_pstr) != 0) { delete[] _pstr; } _pstr = NULL; } // Member: CConvertStrW::Free // Synopsis: Frees string if alloc'd and initializes to NULL. void CConvertStrW::Free() { if (_pwstr != _awch && HIWORD64(_pwstr) != 0) { delete[] _pwstr; } _pwstr = NULL; } // Member: CStrInW::Init // Synopsis: Converts a LPSTR function argument to a LPWSTR. // Arguments: [pstr] -- The function argument. May be NULL or an atom // (HIWORD(pwstr) == 0). // [cch] -- The number of characters in the string to // convert. If -1, the string is assumed to be // NULL terminated and its length is calculated. // Modifies: [this] void CStrInW::Init(LPCSTR pstr, int cch) { int cchBufReq; _cwchLen = 0; // Check if string is NULL or an atom. if (HIWORD64(pstr) == 0) { _pwstr = (LPWSTR)pstr; return; } ASSERT(cch == -1 || cch > 0); // Convert string to preallocated buffer, and return if successful. _cwchLen = MultiByteToWideChar(CP_ACP, 0, pstr, cch, _awch, ARRAYSIZE(_awch)); if (_cwchLen > 0) { if (_awch[_cwchLen - 1] == 0) _cwchLen--; // account for terminator _pwstr = _awch; return; } // Alloc space on heap for buffer. cchBufReq = MultiByteToWideChar(CP_ACP, 0, pstr, cch, NULL, 0); ASSERT(cchBufReq > 0); _pwstr = new WCHAR[cchBufReq]; if (!_pwstr) { // On failure, the argument will point to the empty string. _awch[0] = 0; _pwstr = _awch; return; } ASSERT(HIWORD64(_pwstr)); _cwchLen = -1 + MultiByteToWideChar(CP_ACP, 0, pstr, cch, _pwstr, cchBufReq); ASSERT(_cwchLen >= 0); } // Member: CStrIn::CStrIn // Synopsis: Inits the class. // NOTE: Don't inline this function or you'll increase code size // by pushing -1 on the stack for each call. CStrIn::CStrIn(LPCWSTR pwstr) : CConvertStr(CP_ACP) { Init(pwstr, -1); } CStrIn::CStrIn(UINT uCP, LPCWSTR pwstr) : CConvertStr(uCP) { Init(pwstr, -1); } // Member: CStrIn::Init // Synopsis: Converts a LPWSTR function argument to a LPSTR. // Arguments: [pwstr] -- The function argument. May be NULL or an atom // (HIWORD(pwstr) == 0). // [cwch] -- The number of characters in the string to // convert. If -1, the string is assumed to be // NULL terminated and its length is calculated. // Modifies: [this] void CStrIn::Init(LPCWSTR pwstr, int cwch) { int cchBufReq; #if DBG == 1 /* { */ int errcode; #endif /* } */ _cchLen = 0; // Check if string is NULL or an atom. if (HIWORD64(pwstr) == 0) { _pstr = (LPSTR)pwstr; return; } if (cwch == 0) { *_ach = '\0'; _pstr = _ach; return; } ASSERT(cwch == -1 || cwch > 0); // Convert string to preallocated buffer, and return if successful. _cchLen = WideCharToMultiByte(_uCP, 0, pwstr, cwch, _ach, ARRAYSIZE(_ach), NULL, NULL); if (_cchLen > 0) { if (_ach[_cchLen - 1] == 0) _cchLen--; // account for terminator _pstr = _ach; return; } cchBufReq = WideCharToMultiByte(CP_ACP, 0, pwstr, cwch, NULL, 0, NULL, NULL); ASSERT(cchBufReq > 0); _pstr = new char[cchBufReq]; if (!_pstr) { // On failure, the argument will point to the empty string. _ach[0] = 0; _pstr = _ach; return; } ASSERT(HIWORD64(_pstr)); _cchLen = -1 + WideCharToMultiByte(_uCP, 0, pwstr, cwch, _pstr, cchBufReq, NULL, NULL); #if DBG == 1 /* { */ if (_cchLen < 0) { errcode = GetLastError(); ASSERT(0 && "WideCharToMultiByte failed in unicode wrapper."); } #endif /* } */ } // Member: CStrInMulti::CStrInMulti // Synopsis: Converts mulitple LPWSTRs to a multiple LPSTRs. // Arguments: [pwstr] -- The strings to convert. // Modifies: [this] CStrInMulti::CStrInMulti(LPCWSTR pwstr) { LPCWSTR pwstrT; // We don't handle atoms because we don't need to. ASSERT(HIWORD64(pwstr)); // Count number of characters to convert. pwstrT = pwstr; if (pwstr) { do { while (*pwstrT++) ; } while (*pwstrT++); } Init(pwstr, (int)(pwstrT - pwstr)); } // Member: CStrOut::CStrOut // Synopsis: Allocates enough space for an out buffer. // Arguments: [pwstr] -- The Unicode buffer to convert to when destroyed. // May be NULL. // [cwchBuf] -- The size of the buffer in characters. // Modifies: [this]. CStrOut::CStrOut(LPWSTR pwstr, int cwchBuf) : CConvertStr(CP_ACP) { ASSERT(cwchBuf >= 0); if (!cwchBuf) pwstr = NULL; _pwstr = pwstr; _cwchBuf = cwchBuf; if (!pwstr) { ASSERT(cwchBuf == 0); _pstr = NULL; return; } ASSERT(HIWORD64(pwstr)); // Initialize buffer in case Windows API returns an error. _ach[0] = 0; // Use preallocated buffer if big enough. if (cwchBuf * 2 <= ARRAYSIZE(_ach)) { _pstr = _ach; return; } // Allocate buffer. _pstr = new char[cwchBuf * 2]; if (!_pstr) { // On failure, the argument will point to a zero-sized buffer initialized // to the empty string. This should cause the Windows API to fail. ASSERT(cwchBuf > 0); _pwstr[0] = 0; _cwchBuf = 0; _pstr = _ach; return; } ASSERT(HIWORD64(_pstr)); _pstr[0] = 0; } // Member: CStrOut::ConvertIncludingNul // Synopsis: Converts the buffer from MBCS to Unicode // Return: Character count INCLUDING the trailing '\0' int CStrOut::ConvertIncludingNul() { int cwch; if (!_pstr) return 0; ASSERT(_cwchBuf); // Preinit to null string in case of horrible catastrophe _pwstr[0] = TEXT('\0'); cwch = MultiByteToWideChar(_uCP, 0, _pstr, -1, _pwstr, _cwchBuf); if (!cwch) { // Output buffer was short. Must double-buffer (yuck) int cwchNeeded = MultiByteToWideChar(_uCP, 0, _pstr, -1, NULL, 0); if (cwchNeeded) { LPWSTR pwsz = (LPWSTR)LocalAlloc(LMEM_FIXED, cwchNeeded * SIZEOF(WCHAR)); if (pwsz) { cwch = MultiByteToWideChar(_uCP, 0, _pstr, -1, pwsz, cwchNeeded); if (cwch) { StrCpyNW(_pwstr, pwsz, _cwchBuf); cwch = _cwchBuf; } LocalFree(pwsz); } } else { #if DBG == 1 /* { */ DWORD errcode = GetLastError(); ASSERT(0 && "MultiByteToWideChar failed in unicode wrapper."); #endif /* } */ } } Free(); return cwch; } // Member: CStrOut::ConvertExcludingNul // Synopsis: Converts the buffer from MBCS to Unicode // Return: Character count EXCLUDING the trailing '\0' int CStrOut::ConvertExcludingNul() { int ret = ConvertIncludingNul(); if (ret) { ret -= 1; } return ret; } // Member: CStrOut::~CStrOut // Synopsis: Converts the buffer from MBCS to Unicode. // Note: Don't inline this function, or you'll increase code size as // both ConvertIncludingNul() and CConvertStr::~CConvertStr will be // called inline. CStrOut::~CStrOut() { ConvertIncludingNul(); } // Function: MbcsFromUnicode // Synopsis: Converts a string to MBCS from Unicode. // Arguments: [pstr] -- The buffer for the MBCS string. // [cch] -- The size of the MBCS buffer, including space for // NULL terminator. // [pwstr] -- The Unicode string to convert. // [cwch] -- The number of characters in the Unicode string to // convert, including NULL terminator. If this // number is -1, the string is assumed to be // NULL terminated. -1 is supplied as a // default argument. // Returns: If [pstr] is NULL or [cch] is 0, 0 is returned. Otherwise, // the number of characters converted, including the terminating // NULL, is returned (note that converting the empty string will // return 1). If the conversion fails, 0 is returned. // Modifies: [pstr]. int MbcsFromUnicode(LPSTR pstr, int cch, LPCWSTR pwstr, int cwch) { int ret; #if DBG == 1 /* { */ int errcode; #endif /* } */ ASSERT(cch >= 0); if (!pstr || cch == 0) return 0; ASSERT(pwstr); ASSERT(cwch == -1 || cwch > 0); ret = WideCharToMultiByte(CP_ACP, 0, pwstr, cwch, pstr, cch, NULL, NULL); #if DBG == 1 /* { */ if (ret <= 0) { errcode = GetLastError(); ASSERT(0 && "WideCharToMultiByte failed in unicode wrapper."); } #endif /* } */ return ret; } // Function: UnicodeFromMbcs // Synopsis: Converts a string to Unicode from MBCS. // Arguments: [pwstr] -- The buffer for the Unicode string. // [cwch] -- The size of the Unicode buffer, including space for // NULL terminator. // [pstr] -- The MBCS string to convert. // [cch] -- The number of characters in the MBCS string to // convert, including NULL terminator. If this // number is -1, the string is assumed to be // NULL terminated. -1 is supplied as a // default argument. // Returns: If [pwstr] is NULL or [cwch] is 0, 0 is returned. Otherwise, // the number of characters converted, including the terminating // NULL, is returned (note that converting the empty string will // return 1). If the conversion fails, 0 is returned. // Modifies: [pwstr]. int UnicodeFromMbcs(LPWSTR pwstr, int cwch, LPCSTR pstr, int cch) { int ret; #if DBG == 1 /* { */ int errcode; #endif /* } */ ASSERT(cwch >= 0); if (!pwstr || cwch == 0) return 0; ASSERT(pstr); ASSERT(cch == -1 || cch > 0); ret = MultiByteToWideChar(CP_ACP, 0, pstr, cch, pwstr, cwch); #if DBG == 1 /* { */ if (ret <= 0) { errcode = GetLastError(); ASSERT(0 && "MultiByteToWideChar failed in unicode wrapper."); } #endif /* } */ return ret; } // Contents: widechar character type function (CT_CTYPE1) and (CT_CTYPE3) // Synopsis: We do not have wide char support for IsChar functions // under Win95. The Unicode-Wrapper functions we have // in core\wrappers all convert to CP_ACP and then call // the A version, which means we will have invalid results // for any characters which aren't in CP_ACP. // The solution is to roll our own, which result in these // unfortunately large tables. Here's how it works: // bits: fedc ba98 7654 3210 // pppp pppp iiib bbbb // The 'b' bits form a 32-bit bit mask into our data. The data // entrys boolean, and are thus 4-bytes long. Of the 2^32 // possible combinations, we in fact have only 218 distinct // values of data. These are stored in adwData. // The 'p' bits represent a page. Each page has eight // possible entries, represent by 'i'. In most pages, the // bitfields and data are both uniform. // adwData[abIndex[abType[page]][index]] represents the data // 1 << bits represents the bitmask. #define __BIT_SHIFT 0 #define __INDEX_SHIFT 5 #define __PAGE_SHIFT 8 #define __BIT_MASK 31 #define __INDEX_MASK 7 // straight lookup functions are inlined. #define ISCHARFUNC(type, wch) \ (adwData[abIndex[abType1##type[wch>>__PAGE_SHIFT]] \ [(wch>>__INDEX_SHIFT)&__INDEX_MASK]] \ >> (wch&__BIT_MASK)) & 1 // To avoid header file conflicts with IsCharAlphaW, IsCharAlphaNumericW, ... defined in // winuser.h, the functions names end in "Wrap". SHLWAPI.DEF exports these functions with // the correct name. STDAPI_(BOOL) IsCharAlphaWrap(WCHAR wch); STDAPI_(BOOL) IsCharAlphaNumericWrap(WCHAR wch); STDAPI_(BOOL) IsCharUpperWrap(WCHAR wch); STDAPI_(BOOL) IsCharLowerWrap(WCHAR wch); const DWORD adwData[218] = { 0x00000000, 0x07fffffe, 0xff7fffff, 0xffffffff, // 0x00-0x03 0xfc3fffff, 0x00ffffff, 0xffff0000, 0x000001ff, // 0x04-0x07 0xffffd740, 0xfffffffb, 0x547f7fff, 0x000ffffd, // 0x08-0x0b 0xffffdffe, 0xdffeffff, 0xffff0003, 0xffff199f, // 0x0c-0x0f 0x033fcfff, 0xfffe0000, 0x007fffff, 0xfffffffe, // 0x10-0x13 0x000000ff, 0x000707ff, 0x000007fe, 0x7cffffff, // 0x14-0x17 0x002f7fff, 0xffffffe0, 0x03ffffff, 0xff000000, // 0x18-0x1b 0x00000003, 0xfff99fe0, 0x03c5fdff, 0xb0000000, // 0x1c-0x1f 0x00030003, 0xfff987e0, 0x036dfdff, 0x5e000000, // 0x20-0x23 0xfffbafe0, 0x03edfdff, 0x00000001, 0x03cdfdff, // 0x24-0x27 0xd63dc7e0, 0x03bfc718, 0xfffddfe0, 0x03effdff, // 0x28-0x2b 0x40000000, 0x03fffdff, 0x000d7fff, 0x0000003f, // 0x2c-0x2f 0xfef02596, 0x00006cae, 0x30000000, 0xffff003f, // 0x30-0x33 0x83ffffff, 0xffffff07, 0x07ffffff, 0x3f3fffff, // 0x34-0x37 0xaaff3f3f, 0x3fffffff, 0x1fdfffff, 0x0fcf1fdc, // 0x38-0x3b 0x1fdc1fff, 0xf0000000, 0x000003ff, 0x00000020, // 0x3c-0x3f 0x781fffff, 0x77ffffff, 0xfffe1fff, 0x00007fff, // 0x40-0x43 0x0000000f, 0x00003fff, 0x80f8007f, 0x5f7fffff, // 0x44-0x47 0xffffffdb, 0x0003ffff, 0xfff80000, 0xfffffdff, // 0x48-0x4b 0xfffffffd, 0xfffcffff, 0x0fff0000, 0x1fffffff, // 0x4c-0x4f 0xffffffc0, 0x7ffffffe, 0x1cfcfcfc, 0x00003e00, // 0x50-0x53 0x00000fff, 0x80000000, 0xfc00fffe, 0xf8000001, // 0x54-0x57 0x78000001, 0x00800000, 0x00040000, 0x7fffffff, // 0x58-0x5b 0x44300003, 0x000000b0, 0x0000007c, 0xfe000000, // 0x5c-0x5f 0x00000200, 0x00180000, 0x88001000, 0x0007f801, // 0x60-0x63 0x00013c00, 0xffd00000, 0x0000000e, 0x001f3fff, // 0x64-0x67 0x0001003c, 0xd0000000, 0x0080399f, 0x07fc000c, // 0x68-0x6b 0x00000004, 0x00003987, 0x001f0000, 0x00013bbf, // 0x6c-0x6f 0x00c0398f, 0x00010000, 0x0000000c, 0xc0000000, // 0x70-0x73 0x00803dc7, 0x00603ddf, 0x00803dcf, 0x87f28000, // 0x74-0x77 0x0c00ffc0, 0x3bff8000, 0x00003f5f, 0x08000000, // 0x78-0x7b 0xe0000000, 0xe000e003, 0x6000e000, 0xffff7fff, // 0x7c-0x7f 0x0000007f, 0xfc00fc00, 0x00007c00, 0x01ffffff, // 0x80-0x83 0xffff0007, 0x000007ff, 0x0000001f, 0x003fffff, // 0x84-0x87 0xffffdfff, 0x0000ffff, 0xfc0fffff, 0xfffff3de, // 0x88-0x8b 0xfffffeff, 0x7f47afff, 0xffc000fe, 0xff1fffff, // 0x8c-0x8f 0x7ffeffff, 0x80ffffff, 0x7e000000, 0x78000000, // 0x90-0x93 0x8fffffff, 0x0001ffff, 0xffff0fff, 0xf87fffff, // 0x94-0x97 0xffff000f, 0xfff7fe1f, 0xffd70f7f, 0x0001003e, // 0x98-0x9b 0x00007f7f, 0x03ff0000, 0x020c0000, 0x0000ffc0, // 0x9c-0x9f 0x0007ff80, 0x03f10000, 0x0000007e, 0x7f7fffff, // 0xa0-0xa3 0x55555555, 0xaa555555, 0x555554aa, 0x2b555555, // 0xa4-0xa7 0xb1dbced6, 0x11aed295, 0x4aaaadb0, 0x54165555, // 0xa8-0xab 0x00555555, 0xfffed740, 0x00000ffb, 0x541c0000, // 0xac-0xaf 0x00005555, 0x55550001, 0x5555088a, 0x01154555, // 0xb0-0xb3 0x00155555, 0x01555555, 0x3f00ff00, 0xff00ff00, // 0xb4-0xb7 0xaa003f00, 0x0000ff00, 0x1f00ff00, 0x0f001f00, // 0xb8-0xbb 0x1f001f00, 0xffc00000, 0xaaaaaaaa, 0x55aaaaaa, // 0xbc-0xbf 0xaaaaab55, 0xd4aaaaaa, 0x4e243129, 0x2651292a, // 0xc0-0xc3 0xb5555b60, 0xa82daaaa, 0x00aaaaaa, 0xffaffbfb, // 0xc4-0xc7 0x640f7ffc, 0x000001f9, 0xfffff000, 0x00637fff, // 0xc8-0xcb 0x000faaa8, 0xaaaa0002, 0xaaaa1114, 0x022a8aaa, // 0xcc-0xcf 0x07eaaaaa, 0x02aaaaaa, 0x003f00ff, 0x00ff00ff, // 0xd0-0xd3 0x00ff003f, 0x3fff00ff, 0x00df00ff, 0x00cf00dc, // 0xd4-0xd7 0x00dc00ff, 0x00f8007f }; const BYTE abIndex[98][8] = { { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x00 { 0x00, 0x00, 0x01, 0x01, 0x00, 0x00, 0x02, 0x02 }, // 0x01 { 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x04 }, // 0x02 { 0x05, 0x00, 0x06, 0x03, 0x03, 0x07, 0x00, 0x00 }, // 0x03 { 0x00, 0x00, 0x00, 0x00, 0x08, 0x09, 0x0a, 0x0b }, // 0x04 { 0x0c, 0x03, 0x0d, 0x03, 0x0e, 0x03, 0x0f, 0x10 }, // 0x05 { 0x00, 0x11, 0x12, 0x13, 0x14, 0x00, 0x06, 0x15 }, // 0x06 { 0x00, 0x01, 0x16, 0x11, 0x03, 0x17, 0x18, 0x00 }, // 0x07 { 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f, 0x20 }, // 0x08 { 0x21, 0x22, 0x23, 0x00, 0x24, 0x25, 0x00, 0x26 }, // 0x09 { 0x1d, 0x27, 0x1f, 0x1c, 0x28, 0x29, 0x00, 0x00 }, // 0x0a { 0x2a, 0x2b, 0x00, 0x1c, 0x2a, 0x2b, 0x2c, 0x1c }, // 0x0b { 0x2a, 0x2d, 0x00, 0x1c, 0x00, 0x00, 0x00, 0x00 }, // 0x0c { 0x13, 0x2e, 0x2f, 0x00, 0x30, 0x31, 0x32, 0x00 }, // 0x0d { 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0x33, 0x12 }, // 0x0e { 0x03, 0x03, 0x34, 0x03, 0x03, 0x35, 0x03, 0x1a }, // 0x0f { 0x03, 0x03, 0x03, 0x03, 0x36, 0x03, 0x03, 0x1a }, // 0x10 { 0x37, 0x03, 0x38, 0x39, 0x03, 0x3a, 0x3b, 0x3c }, // 0x11 { 0x00, 0x00, 0x00, 0x00, 0x3d, 0x03, 0x03, 0x3e }, // 0x12 { 0x3f, 0x00, 0x13, 0x03, 0x40, 0x13, 0x03, 0x41 }, // 0x13 { 0x19, 0x42, 0x03, 0x03, 0x43, 0x00, 0x00, 0x00 }, // 0x14 { 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x03 }, // 0x15 { 0x03, 0x03, 0x03, 0x03, 0x03, 0x2f, 0x00, 0x00 }, // 0x16 { 0x03, 0x03, 0x03, 0x03, 0x03, 0x44, 0x00, 0x00 }, // 0x17 { 0x03, 0x45, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x18 { 0x46, 0x47, 0x48, 0x03, 0x03, 0x49, 0x4a, 0x4b }, // 0x19 { 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x4c }, // 0x1a { 0x03, 0x39, 0x06, 0x03, 0x4d, 0x03, 0x14, 0x4e }, // 0x1b { 0x00, 0x00, 0x00, 0x00, 0x03, 0x03, 0x03, 0x4f }, // 0x1c { 0x00, 0x01, 0x01, 0x50, 0x03, 0x51, 0x52, 0x00 }, // 0x1d { 0x53, 0x26, 0x00, 0x00, 0x00, 0x26, 0x00, 0x00 }, // 0x1e { 0x54, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x1f { 0x26, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x20 { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x55 }, // 0x21 { 0x00, 0x56, 0x57, 0x58, 0x00, 0x13, 0x59, 0x59 }, // 0x22 { 0x00, 0x00, 0x00, 0x00, 0x5a, 0x00, 0x00, 0x00 }, // 0x23 { 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x5b, 0x3e }, // 0x24 { 0x03, 0x03, 0x2f, 0x5c, 0x5d, 0x00, 0x00, 0x00 }, // 0x25 { 0x00, 0x00, 0x00, 0x00, 0x5e, 0x00, 0x00, 0x00 }, // 0x26 { 0x00, 0x00, 0x5f, 0x00, 0x60, 0x06, 0x44, 0x61 }, // 0x27 { 0x62, 0x00, 0x63, 0x64, 0x00, 0x00, 0x65, 0x45 }, // 0x28 { 0x66, 0x3d, 0x67, 0x68, 0x66, 0x69, 0x6a, 0x6b }, // 0x29 { 0x6c, 0x69, 0x6d, 0x6e, 0x66, 0x3d, 0x6f, 0x00 }, // 0x2a { 0x66, 0x3d, 0x70, 0x71, 0x72, 0x73, 0x74, 0x00 }, // 0x2b { 0x66, 0x73, 0x75, 0x00, 0x72, 0x73, 0x75, 0x00 }, // 0x2c { 0x72, 0x73, 0x76, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x2d { 0x00, 0x77, 0x78, 0x00, 0x00, 0x79, 0x7a, 0x00 }, // 0x2e { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7b }, // 0x2f { 0x00, 0x00, 0x00, 0x00, 0x00, 0x7c, 0x7d, 0x7e }, // 0x30 { 0x03, 0x7f, 0x80, 0x81, 0x82, 0x54, 0x06, 0x1c }, // 0x31 { 0x03, 0x83, 0x4a, 0x03, 0x84, 0x03, 0x03, 0x85 }, // 0x32 { 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x49 }, // 0x33 { 0x4c, 0x03, 0x03, 0x36, 0x00, 0x00, 0x00, 0x00 }, // 0x34 { 0x03, 0x86, 0x85, 0x03, 0x03, 0x03, 0x03, 0x85 }, // 0x35 { 0x03, 0x03, 0x03, 0x03, 0x87, 0x88, 0x03, 0x89 }, // 0x36 { 0x8a, 0x03, 0x03, 0x89, 0x00, 0x00, 0x00, 0x00 }, // 0x37 { 0x8b, 0x8c, 0x8d, 0x8e, 0x8f, 0x90, 0x00, 0x00 }, // 0x38 { 0x13, 0x91, 0x00, 0x00, 0x92, 0x00, 0x00, 0x93 }, // 0x39 { 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00 }, // 0x3a { 0x4f, 0x03, 0x44, 0x94, 0x03, 0x95, 0x96, 0x5b }, // 0x3b { 0x03, 0x03, 0x03, 0x97, 0x03, 0x03, 0x39, 0x5b }, // 0x3c { 0x2c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x3d { 0x00, 0x73, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x3e { 0x00, 0x98, 0x99, 0x9a, 0x03, 0x03, 0x03, 0x4f }, // 0x3f { 0x56, 0x57, 0x58, 0x9b, 0x73, 0x26, 0x00, 0x9c }, // 0x40 { 0x00, 0x9d, 0x00, 0x00, 0x00, 0x9e, 0x00, 0x00 }, // 0x41 { 0x00, 0x00, 0x00, 0x3e, 0x00, 0x00, 0x00, 0x9d }, // 0x42 { 0x00, 0x00, 0x00, 0x9f, 0x00, 0x00, 0x00, 0x9f }, // 0x43 { 0x00, 0x00, 0x00, 0x9f, 0x00, 0x00, 0x00, 0xa0 }, // 0x44 { 0x00, 0x00, 0x00, 0x9f, 0x00, 0x00, 0x00, 0x00 }, // 0x45 { 0x00, 0x00, 0x9d, 0x00, 0x00, 0x00, 0x9d, 0x00 }, // 0x46 { 0x00, 0x00, 0x00, 0xa1, 0x3e, 0x00, 0x00, 0x00 }, // 0x47 { 0x9d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x48 { 0x00, 0x9d, 0xa2, 0xa2, 0x00, 0x00, 0x00, 0x00 }, // 0x49 { 0x9d, 0xa2, 0xa2, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x4a { 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0xa3, 0x00 }, // 0x4b { 0xa4, 0xa5, 0xa6, 0xa7, 0xa8, 0xa9, 0xaa, 0xab }, // 0x4c { 0xac, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x4d { 0x00, 0x00, 0x00, 0x00, 0xad, 0xae, 0xaf, 0xb0 }, // 0x4e { 0x0c, 0x89, 0x00, 0xa4, 0xb1, 0xa4, 0xb2, 0xb3 }, // 0x4f { 0x00, 0x11, 0x12, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x50 { 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0x2f, 0x00 }, // 0x51 { 0xa4, 0xa4, 0xa4, 0xa4, 0xb4, 0xa4, 0xa4, 0xb5 }, // 0x52 { 0xb6, 0xb7, 0xb8, 0xb9, 0xb7, 0xba, 0xbb, 0xbc }, // 0x53 { 0x00, 0x00, 0x00, 0x00, 0x00, 0xbd, 0x89, 0x00 }, // 0x54 { 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x55 { 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x55, 0x02 }, // 0x56 { 0xbe, 0xbf, 0xc0, 0xc1, 0xc2, 0xc3, 0xc4, 0xc5 }, // 0x57 { 0xc6, 0x00, 0x06, 0xc7, 0xc8, 0xc9, 0x00, 0x00 }, // 0x58 { 0x00, 0x00, 0x00, 0x00, 0x71, 0xca, 0xcb, 0xcc }, // 0x59 { 0x00, 0x06, 0x0d, 0xbe, 0xcd, 0xbe, 0xce, 0xcf }, // 0x5a { 0x00, 0x00, 0x00, 0x13, 0x14, 0x00, 0x00, 0x00 }, // 0x5b { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x12 }, // 0x5c { 0xbe, 0xbe, 0xbe, 0xbe, 0xd0, 0xbe, 0xbe, 0xd1 }, // 0x5d { 0xd2, 0xd3, 0xd4, 0xd5, 0xd3, 0xd6, 0xd7, 0xd8 }, // 0x5e { 0x00, 0x00, 0x00, 0x00, 0x3d, 0x87, 0x06, 0x3e }, // 0x5f { 0xd9, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // 0x60 { 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00 } // 0x61 }; const BYTE abType1Alpha[256] = // 154 { 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x00, 0x00, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x00, 0x0e, 0x0f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x11, 0x00, 0x00, 0x00, 0x00, 0x12, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x13, 0x14, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x16, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x15, 0x17, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x15, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d }; BOOL IsCharSpaceW(WCHAR wch) { int nType; switch (wch >> 8) { case 0x00: nType = 0x1e; break; case 0x20: nType = 0x1f; break; case 0x30: nType = 0x20; break; case 0xfe: nType = 0x21; break; default: nType = 0x00; break; } return (adwData[abIndex[nType][(wch >> __INDEX_SHIFT) & __INDEX_MASK]] >> (wch & __BIT_MASK)) & 1; } const BYTE abType1Punct[256] = // 32 { 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x00, 0x00, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x00, 0x2f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x39, 0x3a, 0x3b, 0x3c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3d, 0x00, 0x3e, 0x3f, 0x40 }; const BYTE abType1Digit[256] = // 11 { 0x41, 0x00, 0x00, 0x00, 0x00, 0x00, 0x42, 0x00, 0x00, 0x43, 0x43, 0x44, 0x43, 0x45, 0x46, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x47, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x48 }; BOOL IsCharDigitW(WCHAR wch) { return ISCHARFUNC(Digit, wch); } BOOL IsCharXDigitW(WCHAR wch) { int nType; switch (wch >> 8) { case 0x00: nType = 0x49; break; case 0xff: nType = 0x4a; break; default: nType = 0x00; break; } return (adwData[abIndex[nType][(wch >> __INDEX_SHIFT) & __INDEX_MASK]] >> (wch & __BIT_MASK)) & 1; } const BYTE abType1Upper[256] = // 12 { 0x4b, 0x4c, 0x4d, 0x4e, 0x4f, 0x50, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x51, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x52, 0x53, 0x00, 0x00, 0x00, 0x00, 0x54, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x55 }; const BYTE abType1Lower[256] = // 13 { 0x56, 0x57, 0x58, 0x59, 0x5a, 0x5b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x5c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x5d, 0x5e, 0x00, 0x00, 0x00, 0x00, 0x5f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x60, 0x00, 0x00, 0x00, 0x61 }; BOOL IsCharPunctW(WCHAR wch) { return ISCHARFUNC(Punct, wch); } BOOL IsCharCntrlW(WCHAR wch) { return (unsigned)(wch - 0x0000) <= (0x001f - 0x0000) || (unsigned)(wch - 0x007f) <= (0x009f - 0x007f); } // NB (cthrash) WCH_NBSP is considered blank, for compatibility. BOOL IsCharBlankW(WCHAR wch) { return wch == 0x0009 || wch == 0x0020 || wch == 0x00a0 || wch == 0x3000 || wch == 0xfeff; } BOOL IsCharAlphaWrap(WCHAR wch) { return ISCHARFUNC(Alpha, wch); } BOOL IsCharUpperWrap(WCHAR wch) { return ISCHARFUNC(Upper, wch); } BOOL IsCharLowerWrap(WCHAR wch) { return ISCHARFUNC(Lower, wch); } BOOL IsCharAlphaNumericWrap(WCHAR wch) { return ISCHARFUNC(Alpha, wch) || ISCHARFUNC(Digit, wch); } static const BYTE abType3PageSub[256] = { 0x00, 0x80, 0x81, 0x82, 0x00, 0x83, 0x84, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x85, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x86, 0x00, 0x87, 0x88, 0x89, 0x8a, 0x8b, 0x8c, 0x8d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x8e, 0x8f, 0x90, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x91, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x20, 0x92, 0x00, 0x00, 0x93, 0x94, 0x00 }; static const BYTE abType3Page0[256] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x09, 0x09, 0x09, 0x09, 0x09, 0x09, 0x09, 0x08, 0x09, 0x09, 0x09, 0x09, 0x09, 0x08, 0x09, 0x09, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x09, 0x09, 0x09, 0x09, 0x09, 0x09, 0x09, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x09, 0x09, 0x09, 0x09, 0x09, 0x09, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x09, 0x09, 0x09, 0x09, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x09, 0x09, 0x01, 0x09, 0x09, 0x01, 0x01, 0x01, 0x00, 0x01, 0x09, 0x01, 0x01, 0x09, 0x01, 0x01, 0x00, 0x00, 0x01, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; static const BYTE abType3Page32[256] = { 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x11, 0x11, 0x01, 0x01, 0x11, 0x11, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x00, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x00, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x09, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; static const BYTE abType3Page48[256] = { 0x11, 0x11, 0x11, 0x00, 0x00, 0x20, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x11, 0x11, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x00, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x06, 0x06, 0x06, 0x06, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x04, 0x00, 0x00, 0x00, 0x00, 0x06, 0x06, 0x16, 0x16, 0x04, 0x04, 0x00, 0x00, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x12, 0x02, 0x12, 0x02, 0x12, 0x02, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x02, 0x02, 0x12, 0x02, 0x02, 0x12, 0x02, 0x02, 0x12, 0x02, 0x02, 0x12, 0x02, 0x02, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x02, 0x12, 0x02, 0x02, 0x12, 0x12, 0x02, 0x02, 0x02, 0x02, 0x02, 0x02, 0x02, 0x13, 0x06, 0x02, 0x02, 0x00 }; static const BYTE abType3Page255[256] = { 0x00, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x10, 0x11, 0x11, 0x11, 0x11, 0x11, 0x10, 0x11, 0x11, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x11, 0x11, 0x11, 0x11, 0x00, 0x00, 0x09, 0x09, 0x09, 0x09, 0x09, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0e, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x0a, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x00, 0x00, 0x00, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x00, 0x00, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x00, 0x00, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x00, 0x00, 0x08, 0x08, 0x08, 0x00, 0x00, 0x00, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x00, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; struct tagType3DualValue { DWORD adwBitfield[8]; DWORD adwValue[2]; } const aType3DualValue[21] = { { 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, // Page1 0x00000000, 0x0000000f, 0x00000000, 0x00000000, 0x00000001 }, { 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, // Page2 0x00000000, 0x3f000000, 0x00000000, 0x00000000, 0x00000001 }, { 0x00000000, 0x00000000, 0x00000000, 0x04000000, 0x000000b0, // Page3 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000001 }, { 0x00000000, 0x00000000, 0xf8000000, 0x00000000, 0x00000200, // Page5 0x40000000, 0x00000009, 0x00180000, 0x00000000, 0x00000001 }, { 0x88001000, 0x00000000, 0x00000000, 0x00003c00, 0x00000000, // Page6 0x00000000, 0x00100000, 0x00000200, 0x00000000, 0x00000001 }, { 0x00000000, 0x80008000, 0x0c008040, 0x00000000, 0x00000000, // Page14 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000001 }, { 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, // Page31 0xe0000000, 0xe000e003, 0x6000e000, 0x00000000, 0x00000001 }, { 0x00800000, 0x00000000, 0x00000000, 0x00000000, 0xffff0000, // Page33 0xffffffff, 0xffffffff, 0x000007ff, 0x00000000, 0x00000001 }, { 0x40000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, // Page34 0x00000000, 0x00000000, 0xfffc0000, 0x00000001, 0x00000000 }, { 0x00000002, 0x00000000, 0x00000000, 0xf8000000, 0xffffffff, // Page35 0xffffffff, 0xffffffff, 0xffffffff, 0x00000001, 0x00000000 }, { 0x00000000, 0xffffffe0, 0xfffff800, 0xffffffff, 0xffffffff, // Page36 0xffffffff, 0xffffffff, 0xffffffff, 0x00000001, 0x00000000 }, { 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0xffc00000, // Page37 0x00002000, 0x00000000, 0xffff8000, 0x00000001, 0x00000000 }, { 0x03f00000, 0x00000000, 0x00000000, 0xffff0000, 0xffffffff, // Page38 0xffffffff, 0xffffffff, 0xffffffff, 0x00000001, 0x00000000 }, { 0xfffff3de, 0xfffffeff, 0x7f47afff, 0x000000fe, 0xff100000, // Page39 0x7ffeffff, 0x00000000, 0x00000000, 0x00000000, 0x00000001 }, { 0x00000000, 0xfffe0000, 0xffffffff, 0x0000001f, 0x00000000, // Page49 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000010 }, { 0x00000000, 0x00000000, 0x00000000, 0x80000000, 0x00000000, // Page50 0x00000000, 0x00000fff, 0x00000000, 0x00000000, 0x00000001 }, { 0x00000000, 0x00000000, 0xff000000, 0x0001ffff, 0x00000000, // Page51 0x00000000, 0x00000000, 0x7fffffff, 0x00000000, 0x00000001 }, { 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, // Page159 0xffffffc0, 0xffffffff, 0xffffffff, 0x00000020, 0x00000000 }, { 0x00000000, 0xffffc000, 0xffffffff, 0xffffffff, 0xffffffff, // Page250 0xffffffff, 0xffffffff, 0xffffffff, 0x00000020, 0x00000000 }, { 0x00000000, 0xc0000000, 0x00000000, 0x00000000, 0x00000000, // Page253 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000001 }, { 0x00000000, 0xfff90000, 0xfef7fe1f, 0x00000f77, 0x00000000, // Page254 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000001 } }; // CType 3 Flag Bits. // In the interest of reducing our table complexity, we've here a reduced // bitfield. Only those bits currently used by IE4 are returned by // GetStringType3Ex(). // These are the flags are they are defined in winnls.h // C3_NONSPACING 0x0001 // C3_DIACRITIC 0x0002 // C3_VOWELMARK 0x0004 // C3_SYMBOL 0x0008 // C3_KATAKANA 0x0010 // C3_HIRAGANA 0x0020 // C3_HALFWIDTH 0x0040 // C3_FULLWIDTH 0x0080 // C3_IDEOGRAPH 0x0100 // C3_KASHIDA 0x0200 // C3_LEXICAL 0x0400 // C3_ALPHA 0x8000 // The supported flags are encoded by shifting them to the right 3 bits. // C3_SYMBOL 0x0001 // C3_KATAKANA 0x0002 // C3_HIRAGANA 0x0004 // C3_HALFWIDTH 0x0008 // C3_FULLWIDTH 0x0010 // C3_IDEOGRAPH 0x0020 // GetStringType3Ex returns the correct Win32 flags NOT the compressed flags. BOOL GetStringType3ExW( LPCWSTR lpSrcStr, // string arg int cchSrc, // length (or -1) LPWORD lpCharType) // output buffer { LPCWSTR lpStop = lpSrcStr + ((cchSrc == -1) ? MAXLONG : cchSrc); while (lpSrcStr < lpStop) { WCHAR wch = *lpSrcStr++; WORD wCharType; BYTE bPageSub; if (!wch && cchSrc == -1) break; switch (wch & (unsigned int)0xff00) { case 0x0000: wCharType = abType3Page0[wch]; // Page0: 4 values break; case 0x2000: wCharType = abType3Page32[wch & 0xff]; // Page32: 4 values break; case 0x3000: wCharType = abType3Page48[wch & 0xff]; // Page48: 10 values break; case 0xff00: wCharType = abType3Page255[wch & 0xff]; // Page255: 7 values break; default: bPageSub = abType3PageSub[wch >> 8]; if (bPageSub & 0x80) // 21 pages have 2 values { const struct tagType3DualValue* p = aType3DualValue + (bPageSub & 0x7f); wCharType = (BYTE)p->adwValue[(p->adwBitfield[(wch >> 5) & 7] >> (wch & 0x1f)) & 1]; } else // 231 pages have 1 value { wCharType = bPageSub; } break; } *lpCharType++ = wCharType << 3; } return TRUE; } // Str Functions from SHLWAPI int StrCmpW( IN LPCWSTR pwsz1, IN LPCWSTR pwsz2) { int iRet = -1; // arbitrary on failure ASSERT(IS_VALID_STRING_PTRW(pwsz1, -1)); ASSERT(IS_VALID_STRING_PTRW(pwsz2, -1)); if (pwsz1 && pwsz2) { CStrIn psz1(pwsz1); CStrIn psz2(pwsz2); iRet = lstrcmpA(psz1, psz2); } return iRet; } int StrCmpIW( IN LPCWSTR pwsz1, IN LPCWSTR pwsz2) { int iRet = -1; // arbitrary on failure ASSERT(IS_VALID_STRING_PTRW(pwsz1, -1)); ASSERT(IS_VALID_STRING_PTRW(pwsz2, -1)); if (pwsz1 && pwsz2) { CStrIn psz1(pwsz1); CStrIn psz2(pwsz2); iRet = lstrcmpiA(psz1, psz2); } return iRet; } #if 0 // BUGBUG: We have another StrCpyW in strings.c LPWSTR StrCpyW(LPWSTR psz1, LPCWSTR psz2) { LPWSTR psz = psz1; ASSERT(psz1); ASSERT(psz2); while (*psz1++ = *psz2++) ; return psz; } #endif LPWSTR StrCatW(LPWSTR psz1, LPCWSTR psz2) { LPWSTR psz = psz1; ASSERT(psz1); ASSERT(psz2); while (0 != *psz1) psz1++; while (*psz1++ = *psz2++) ; return psz; } // Implementation of the wrapped functions BOOL AppendMenuWrap( HMENU hMenu, UINT uFlags, UINT uIDnewItem, LPCWSTR lpnewItem) { ASSERT(!(uFlags & MF_BITMAP) && !(uFlags & MF_OWNERDRAW)); CStrIn str(lpnewItem); return AppendMenuA(hMenu, uFlags, uIDnewItem, str); } BOOL CallMsgFilterWrap(LPMSG lpMsg, int nCode) { return CallMsgFilterA(lpMsg, nCode); } LRESULT CallWindowProcWrap( WNDPROC lpPrevWndFunc, HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam) { return CallWindowProcA(lpPrevWndFunc, hWnd, Msg, wParam, lParam); } // function: CharLowerWrap( LPWSTR pch ) // purpose: Converts character to lowercase. Takes either a pointer // to a string, or a character masquerading as a pointer. // In the later case, the HIWORD must be zero. This is // as spec'd for Win32. // returns: Lowercased character or string. In the string case, // the lowercasing is done inplace. LPWSTR CharLowerWrap(LPWSTR pch) { if (!HIWORD64(pch)) { WCHAR ch = (WCHAR)(LONG_PTR)pch; CharLowerBuffWrap(&ch, 1); pch = (LPWSTR)MAKEINTATOM(ch); } else { CharLowerBuffWrap(pch, lstrlenW(pch)); } return pch; } // function: CharLowerBuffWrap( LPWSTR pch, DWORD cch ) // purpose: Converts a string to lowercase. String must be cch // characters in length. // returns: Character count (cch). The lowercasing is done inplace. DWORD CharLowerBuffWrap(LPWSTR pch, DWORD cchLength) { DWORD cch; for (cch = cchLength; cch--; pch++) { WCHAR ch = *pch; if (IsCharUpperWrap(ch)) { if (ch < 0x0100) { *pch += 32; // Get Latin-1 out of the way first } else if (ch < 0x0531) { if (ch < 0x0391) { if (ch < 0x01cd) { if (ch <= 0x178) { if (ch < 0x0178) { *pch += (ch == 0x0130) ? 0 : 1; } else { *pch -= 121; } } else { static const BYTE abLookup[] = { // 0/8 1/9 2/a 3/b 4/c 5/d 6/e 7/f /* 0x0179-0x17f */ 1, 0, 1, 0, 1, 0, 0, /* 0x0180-0x187 */ 0, 210, 1, 0, 1, 0, 206, 1, /* 0x0188-0x18f */ 0, 205, 205, 1, 0, 0, 79, 202, /* 0x0190-0x197 */ 203, 1, 0, 205, 207, 0, 211, 209, /* 0x0198-0x19f */ 1, 0, 0, 0, 211, 213, 0, 214, /* 0x01a0-0x1a7 */ 1, 0, 1, 0, 1, 0, 0, 1, /* 0x01a8-0x1af */ 0, 218, 0, 0, 1, 0, 218, 1, /* 0x01b0-0x1b7 */ 0, 217, 217, 1, 0, 1, 0, 219, /* 0x01b8-0x1bf */ 1, 0, 0, 0, 1, 0, 0, 0, /* 0x01c0-0x1c7 */ 0, 0, 0, 0, 2, 0, 0, 2, /* 0x01c8-0x1cb */ 0, 0, 2, 0 }; *pch += abLookup[ch - 0x0179]; } } else if (ch < 0x0386) { switch (ch) { case 0x01f1: *pch += 2; break; case 0x01f2: break; default: *pch += 1; } } else { static const BYTE abLookup[] = {38, 0, 37, 37, 37, 0, 64, 0, 63, 63}; *pch += abLookup[ch - 0x0386]; } } else { if (ch < 0x0410) { if (ch < 0x0401) { if (ch < 0x03e2) { if (!InRange(ch, 0x03d2, 0x03d4) && !(InRange(ch, 0x3da, 0x03e0) & !(ch & 1))) { *pch += 32; } } else { *pch += 1; } } else { *pch += 80; } } else { if (ch < 0x0460) { *pch += 32; } else { *pch += 1; } } } } else { if (ch < 0x2160) { if (ch < 0x1fba) { if (ch < 0x1f08) { if (ch < 0x1e00) { *pch += 48; } else { *pch += 1; } } else if (!(InRange(ch, 0x1f88, 0x1faf) && (ch & 15) > 7)) { *pch -= 8; } } else { static const BYTE abLookup[] = { // 8 9 a b c d e f 0, 0, 74, 74, 0, 0, 0, 0, 86, 86, 86, 86, 0, 0, 0, 0, 8, 8, 100, 100, 0, 0, 0, 0, 8, 8, 112, 112, 7, 0, 0, 0, 128, 128, 126, 126, 0, 0, 0, 0 }; int i = (ch - 0x1fb0); *pch -= (int)abLookup[((i >> 1) & ~7) | (i & 7)]; } } else { if (ch < 0xff21) { if (ch < 0x24b6) { *pch += 16; } else { *pch += 26; } } else { *pch += 32; } } } } else { // These are Unicode Number Forms. They have lowercase counter- // parts, but are not considered uppercase. Why, I don't know. if (InRange(ch, 0x2160, 0x216f)) { *pch += 16; } } } return cchLength; } // BUGBUG - Do CharNextWrap and CharPrevWrap need to call the // CharNextW, CharPrevW on WinNT? Couldn't these be MACROS? LPWSTR CharNextWrap(LPCWSTR lpszCurrent) { if (*lpszCurrent) { return (LPWSTR)lpszCurrent + 1; } else { return (LPWSTR)lpszCurrent; } } LPWSTR CharPrevWrap(LPCWSTR lpszStart, LPCWSTR lpszCurrent) { if (lpszCurrent == lpszStart) { return (LPWSTR)lpszStart; } else { return (LPWSTR)lpszCurrent - 1; } } BOOL CharToOemWrap(LPCWSTR lpszSrc, LPSTR lpszDst) { CStrIn str(lpszSrc); return CharToOemA(str, lpszDst); } // function: CharUpperWrap( LPWSTR pch ) // purpose: Converts character to uppercase. Takes either a pointer // to a string, or a character masquerading as a pointer. // In the later case, the HIWORD must be zero. This is // as spec'd for Win32. // returns: Uppercased character or string. In the string case, // the uppercasing is done inplace. LPWSTR CharUpperWrap(LPWSTR pch) { if (!HIWORD64(pch)) { WCHAR ch = (WCHAR)(LONG_PTR)pch; CharUpperBuffWrap(&ch, 1); pch = (LPWSTR)MAKEINTATOM(ch); } else { CharUpperBuffWrap(pch, lstrlenW(pch)); } return pch; } // function: CharUpperBuffWrap( LPWSTR pch, DWORD cch ) // purpose: Converts a string to uppercase. String must be cch // characters in length. Note that this function is // is messier that CharLowerBuffWrap, and the reason for // this is many Unicode characters are considered uppercase, // even when they don't have an uppercase counterpart. // returns: Character count (cch). The uppercasing is done inplace. DWORD CharUpperBuffWrap(LPWSTR pch, DWORD cchLength) { DWORD cch; for (cch = cchLength; cch--; pch++) { WCHAR ch = *pch; if (IsCharLowerWrap(ch)) { if (ch < 0x00ff) { *pch -= ((ch != 0xdf) << 5); } else if (ch < 0x03b1) { if (ch < 0x01f5) { if (ch < 0x01ce) { if (ch < 0x017f) { if (ch < 0x0101) { *pch += 121; } else { *pch -= (ch != 0x0131 && ch != 0x0138 && ch != 0x0149); } } else if (ch < 0x01c9) { static const BYTE abMask[] = { // 6543210f edcba987 0xfc, 0xbf, // 11111100 10111111 0xbf, 0x67, // 10111111 01100111 0xff, 0xef, // 11111111 11101111 0xff, 0xf7, // 11111111 11110111 0xbf, 0xfd // 10111111 11111101 }; int i = ch - 0x017f; *pch -= ((abMask[i >> 3] >> (i & 7)) & 1) + (ch == 0x01c6); } else { *pch -= ((ch != 0x01cb) << 1); } } else { if (ch < 0x01df) { if (ch < 0x01dd) { *pch -= 1; } else { *pch -= 79; } } else { *pch -= 1 + (ch == 0x01f3) - InRange(ch, 0x01f0, 0x01f2); } } } else if (ch < 0x0253) { *pch -= (ch < 0x0250); } else if (ch < 0x03ac) { static const BYTE abLookup[] = {// 0/8 1/9 2/a 3/b 4/c 5/d 6/e 7/f /* 0x0253-0x0257 */ 210, 206, 0, 205, 205, /* 0x0258-0x025f */ 0, 202, 0, 203, 0, 0, 0, 0, /* 0x0260-0x0267 */ 205, 0, 0, 207, 0, 0, 0, 0, /* 0x0268-0x026f */ 209, 211, 0, 0, 0, 0, 0, 211, /* 0x0270-0x0277 */ 0, 0, 213, 0, 0, 214, 0, 0, /* 0x0278-0x027f */ 0, 0, 0, 0, 0, 0, 0, 0, /* 0x0280-0x0287 */ 0, 0, 0, 218, 0, 0, 0, 0, /* 0x0288-0x028f */ 218, 0, 217, 217, 0, 0, 0, 0, /* 0x0290-0x0297 */ 0, 0, 219 }; if (ch <= 0x0292) { *pch -= abLookup[ch - 0x0253]; } } else { *pch -= (ch == 0x03b0) ? 0 : (37 + (ch == 0x03ac)); } } else { if (ch < 0x0561) { if (ch < 0x0451) { if (ch < 0x03e3) { if (ch < 0x03cc) { *pch -= (ch != 0x03c2) << 5; } else { int i = (ch < 0x03d0); *pch -= (i << 6) - i + (ch == 0x03cc); } } else if (ch < 0x0430) { *pch -= (ch < 0x03f0); } else { *pch -= 32; } } else if (ch < 0x0461) { *pch -= 80; } else { *pch -= 1; } } else { if (ch < 0x1fb0) { if (ch < 0x1f70) { if (ch < 0x1e01) { int i = ch != 0x0587 && ch != 0x10f6; *pch -= ((i << 5) + (i << 4)); /* 48 */ } else if (ch < 0x1f00) { *pch -= !InRange(ch, 0x1e96, 0x1e9a); } else { int i = !InRange(ch, 0x1f50, 0x1f56) || (ch & 1); *pch += (i << 3); } } else { static const BYTE abLookup[] = {74, 86, 86, 100, 128, 112, 126}; if (ch <= 0x1f7d) { *pch += abLookup[(ch - 0x1f70) >> 1]; } } } else { if (ch < 0x24d0) { if (ch < 0x1fe5) { *pch += (0x0023 & (1 << (ch & 15))) ? 8 : 0; } else if (ch < 0x2170) { *pch += (0x0023 & (1 << (ch & 15))) ? 7 : 0; } else { *pch -= ((ch > 0x24b5) << 4); } } else if (ch < 0xff41) { int i = !InRange(ch, 0xfb00, 0xfb17); *pch -= (i << 4) + (i << 3) + (i << 1); /* 26 */ } else { *pch -= 32; } } } } } else { int i = InRange(ch, 0x2170, 0x217f); *pch -= (i << 4); } } return cchLength; } int CopyAcceleratorTableWrap( HACCEL hAccelSrc, LPACCEL lpAccelDst, int cAccelEntries) { return CopyAcceleratorTableA(hAccelSrc, lpAccelDst, cAccelEntries); } HACCEL CreateAcceleratorTableWrap(LPACCEL lpAccel, int cEntries) { return CreateAcceleratorTableA(lpAccel, cEntries); } typedef HDC(*FnCreateHDCA)(LPCSTR, LPCSTR, LPCSTR, CONST DEVMODEA*); HDC CreateHDCWrap( LPCWSTR lpszDriver, LPCWSTR lpszDevice, LPCWSTR lpszOutput, CONST DEVMODEW* lpInitData, FnCreateHDCA pfn) { DEVMODEA* pdevmode = NULL; CStrIn strDriver(lpszDriver); CStrIn strDevice(lpszDevice); CStrIn strOutput(lpszOutput); HDC hdcReturn = 0; if (lpInitData) { pdevmode = (DEVMODEA*)LocalAlloc(LPTR, lpInitData->dmSize + lpInitData->dmDriverExtra); if (pdevmode) { MbcsFromUnicode((CHAR*)pdevmode->dmDeviceName, CCHDEVICENAME, lpInitData->dmDeviceName); memcpy(&pdevmode->dmSpecVersion, &lpInitData->dmSpecVersion, FIELD_OFFSET(DEVMODEW, dmFormName) - FIELD_OFFSET(DEVMODEW, dmSpecVersion)); MbcsFromUnicode((CHAR*)pdevmode->dmFormName, CCHFORMNAME, lpInitData->dmFormName); memcpy(&pdevmode->dmLogPixels, &lpInitData->dmLogPixels, lpInitData->dmDriverExtra + lpInitData->dmSize - FIELD_OFFSET(DEVMODEW, dmLogPixels)); pdevmode->dmSize -= (sizeof(BCHAR) - sizeof(char)) * (CCHDEVICENAME + CCHFORMNAME); } } hdcReturn = (*pfn)(strDriver, strDevice, strOutput, pdevmode); if (pdevmode) { LocalFree(pdevmode); } return hdcReturn; } HDC CreateDCWrap( LPCWSTR lpszDriver, LPCWSTR lpszDevice, LPCWSTR lpszOutput, CONST DEVMODEW* lpInitData) { return CreateHDCWrap(lpszDriver, lpszDevice, lpszOutput, lpInitData, CreateDCA); } HDC CreateICWrap( LPCWSTR lpszDriver, LPCWSTR lpszDevice, LPCWSTR lpszOutput, CONST DEVMODEW* lpInitData) { return CreateHDCWrap(lpszDriver, lpszDevice, lpszOutput, lpInitData, CreateICA); } BOOL CreateDirectoryWrap( LPCWSTR lpPathName, LPSECURITY_ATTRIBUTES lpSecurityAttributes) { CStrIn str(lpPathName); ASSERT(!lpSecurityAttributes); return CreateDirectoryA(str, lpSecurityAttributes); } HANDLE CreateEventWrap( LPSECURITY_ATTRIBUTES lpEventAttributes, BOOL bManualReset, BOOL bInitialState, LPCWSTR lpName) { return CreateEventA(lpEventAttributes, bManualReset, bInitialState, (LPCSTR)lpName); } HANDLE CreateFileWrap( LPCWSTR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode, LPSECURITY_ATTRIBUTES lpSecurityAttributes, DWORD dwCreationDisposition, DWORD dwFlagsAndAttributes, HANDLE hTemplateFile) { CStrIn str(lpFileName); return CreateFileA( str, dwDesiredAccess, dwShareMode, lpSecurityAttributes, dwCreationDisposition, dwFlagsAndAttributes, hTemplateFile); } HANDLE CreateFileMappingWrap( HANDLE hFile, LPSECURITY_ATTRIBUTES lpFileMappingAttributes, DWORD flProtect, DWORD dwMaxSizeHigh, DWORD dwMaxSizeLow, LPCWSTR lpName) { CStrIn str(lpName); return CreateFileMappingA(hFile, lpFileMappingAttributes, flProtect, dwMaxSizeHigh, dwMaxSizeLow, str); } HFONT CreateFontWrap( int nHeight, int nWidth, int nEscapement, int nOrientation, int fnWeight, DWORD fdwItalic, DWORD fdwUnderline, DWORD fdwStrikeOut, DWORD fdwCharSet, DWORD fdwOutputPrecision, DWORD fdwClipPrecision, DWORD fdwQuality, DWORD fdwPitchAndFamily, LPCWSTR lpszFace) { CStrIn str(lpszFace); return CreateFontA(nHeight, nWidth, nEscapement, nOrientation, fnWeight, fdwItalic, fdwUnderline, fdwStrikeOut, fdwCharSet, fdwOutputPrecision, fdwClipPrecision, fdwQuality, fdwPitchAndFamily, str); } HFONT CreateFontIndirectWrap(CONST LOGFONTW* plfw) { LOGFONTA lfa; HFONT hFont; memcpy(&lfa, plfw, FIELD_OFFSET(LOGFONTA, lfFaceName)); MbcsFromUnicode(lfa.lfFaceName, ARRAYSIZE(lfa.lfFaceName), plfw->lfFaceName); hFont = CreateFontIndirectA(&lfa); return hFont; } HWND CreateWindowExWrap( DWORD dwExStyle, LPCWSTR lpClassName, LPCWSTR lpWindowName, DWORD dwStyle, int X, int Y, int nWidth, int nHeight, HWND hWndParent, HMENU hMenu, HINSTANCE hInstance, LPVOID lpParam) { CStrIn strClass(lpClassName); CStrIn strWindow(lpWindowName); return CreateWindowExA( dwExStyle, strClass, strWindow, dwStyle, X, Y, nWidth, nHeight, hWndParent, hMenu, hInstance, lpParam); } LRESULT DefWindowProcWrap(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) { return DefWindowProcA(hWnd, msg, wParam, lParam); } BOOL DeleteFileWrap(LPCWSTR pwsz) { CStrIn str(pwsz); return DeleteFileA(str); } LRESULT DispatchMessageWrap(CONST MSG* lpMsg) { return DispatchMessageA(lpMsg); } #ifndef FONT_LINK int DrawTextWrap( HDC hDC, LPCWSTR lpString, int nCount, LPRECT lpRect, UINT uFormat) { CStrIn str(lpString, nCount); return DrawTextA(hDC, str, str.strlen(), lpRect, uFormat); } // shlwapi also have this. int DrawTextExPrivWrap( HDC hDC, LPWSTR lpString, int nCount, LPRECT lpRect, UINT uFormat, LPDRAWTEXTPARAMS lpDTParams) { CStrIn str(lpString, nCount); return DrawTextExA(hDC, str, str.strlen(), lpRect, uFormat, lpDTParams); } #endif struct EFFSTAT { LPARAM lParam; FONTENUMPROC lpEnumFontProc; BOOL fFamilySpecified; }; int CALLBACK EnumFontFamiliesCallbackWrap( ENUMLOGFONTA* lpelf, NEWTEXTMETRIC* lpntm, DWORD FontType, LPARAM lParam) { ENUMLOGFONTW elf; // Convert strings from ANSI to Unicode if (((EFFSTAT*)lParam)->fFamilySpecified && (FontType & TRUETYPE_FONTTYPE)) { UnicodeFromMbcs( elf.elfFullName, ARRAYSIZE(elf.elfFullName), (LPCSTR)lpelf->elfFullName); UnicodeFromMbcs( elf.elfStyle, ARRAYSIZE(elf.elfStyle), (LPCSTR)lpelf->elfStyle); } else { elf.elfStyle[0] = L'\0'; elf.elfFullName[0] = L'\0'; } UnicodeFromMbcs( elf.elfLogFont.lfFaceName, ARRAYSIZE(elf.elfLogFont.lfFaceName), (LPCSTR)lpelf->elfLogFont.lfFaceName); // Copy the non-string data memcpy( &elf.elfLogFont, &lpelf->elfLogFont, FIELD_OFFSET(LOGFONTA, lfFaceName)); // Chain to the original callback function return (*((EFFSTAT*)lParam)->lpEnumFontProc)( (const LOGFONTW*)&elf, (const TEXTMETRICW*)lpntm, FontType, ((EFFSTAT*)lParam)->lParam); } int EnumFontFamiliesWrap( HDC hdc, LPCWSTR lpszFamily, FONTENUMPROC lpEnumFontProc, LPARAM lParam) { CStrIn str(lpszFamily); EFFSTAT effstat; effstat.lParam = lParam; effstat.lpEnumFontProc = lpEnumFontProc; effstat.fFamilySpecified = lpszFamily != NULL; return EnumFontFamiliesA( hdc, str, (FONTENUMPROCA)EnumFontFamiliesCallbackWrap, (LPARAM)&effstat); } int EnumFontFamiliesExWrap( HDC hdc, LPLOGFONTW lplfw, FONTENUMPROC lpEnumFontProc, LPARAM lParam, DWORD dwFlags) { LOGFONTA lfa; CStrIn str(lplfw->lfFaceName); EFFSTAT effstat; ASSERT(FIELD_OFFSET(LOGFONTW, lfFaceName) == FIELD_OFFSET(LOGFONTA, lfFaceName)); memcpy(&lfa, lplfw, sizeof(LOGFONTA) - FIELD_OFFSET(LOGFONTA, lfFaceName)); memcpy(lfa.lfFaceName, str, LF_FACESIZE); effstat.lParam = lParam; effstat.lpEnumFontProc = lpEnumFontProc; effstat.fFamilySpecified = lplfw->lfFaceName != NULL; return EnumFontFamiliesExA( hdc, &lfa, (FONTENUMPROCA)EnumFontFamiliesCallbackWrap, (LPARAM)&effstat, dwFlags); } BOOL EnumResourceNamesWrap( HINSTANCE hModule, LPCWSTR lpType, ENUMRESNAMEPROCW lpEnumFunc, LONG lParam) { ASSERT(HIWORD64(lpType) == 0); return EnumResourceNamesA(hModule, (LPCSTR)lpType, (ENUMRESNAMEPROCA)lpEnumFunc, lParam); } #ifndef FONT_LINK // There's an app that patches Win95 GDI and their ExtTextOutW handler // is broken. It always dereferences the lpStr parameter, even if // cb is zero. Consequently, any time we are about to pass NULL as // the lpStr, we have to change our mind and pass a null UNICODE string // instead. // The name of this app: Lotus SmartSuite ScreenCam 97. BOOL ExtTextOutWrap(HDC hdc, int x, int y, UINT fuOptions, CONST RECT* lprc, LPCWSTR lpStr, UINT cch, CONST INT* lpDx) { // Force a thunk to ANSI if running Win95 + ME if (g_fMEEnabled && !g_bRunOnMemphis) { CStrIn str(lpStr, cch); return ExtTextOutA(hdc, x, y, fuOptions, lprc, str, str.strlen(), lpDx); } else { if (lpStr == NULL) // Stupid workaround lpStr = TEXT(""); // for ScreenCam 97 return ExtTextOutW(hdc, x, y, fuOptions, lprc, lpStr, cch, lpDx); } } #endif HANDLE FindFirstFileWrap( LPCWSTR lpFileName, LPWIN32_FIND_DATAW pwszFd) { CStrIn str(lpFileName); WIN32_FIND_DATAA fd; HANDLE ret; memcpy(&fd, pwszFd, sizeof(FILETIME) * 3 + sizeof(DWORD) * 5); ret = FindFirstFileA(str, &fd); memcpy(pwszFd, &fd, sizeof(FILETIME) * 3 + sizeof(DWORD) * 5); UnicodeFromMbcs(pwszFd->cFileName, ARRAYSIZE(pwszFd->cFileName), fd.cFileName); UnicodeFromMbcs(pwszFd->cAlternateFileName, ARRAYSIZE(pwszFd->cAlternateFileName), fd.cAlternateFileName); return ret; } // Although Win95 implements FindResource[Ex]W, its implementation is buggy // if you pass a string parameter, so we must thunk to the ANSI side. // The bug is that FindResource[Ex]W will accidentally // call LocalFree(lpName) and LocalFree(lpType), so if lpName and lpType // point to heap memory, Kernel32 secretly freed your memory and you fault // five minutes later. HRSRC FindResourceExWrap(HINSTANCE hModule, LPCWSTR lpType, LPCWSTR lpName, WORD wLang) { CStrIn strType(lpType); // rlefile.cpp passes TEXT("AVI") CStrIn strName(lpName); return FindResourceExA(hModule, strType, strName, wLang); } HWND FindWindowWrap(LPCWSTR lpClassName, LPCWSTR lpWindowName) { CStrIn strClass(lpClassName); CStrIn strWindow(lpWindowName); return FindWindowA(strClass, strWindow); } DWORD FormatMessageWrap( DWORD dwFlags, LPCVOID lpSource, DWORD dwMessageId, DWORD dwLanguageId, LPWSTR lpBuffer, DWORD nSize, va_list* Arguments) { //This assert is only valid on Windows 95. ASSERT(!(dwFlags & FORMAT_MESSAGE_ALLOCATE_BUFFER)); CStrOut str(lpBuffer, nSize); FormatMessageA(dwFlags, lpSource, dwMessageId, dwLanguageId, str, str.BufSize(), Arguments); return str.ConvertExcludingNul(); } BOOL GetClassInfoWrap(HINSTANCE hModule, LPCWSTR lpClassName, LPWNDCLASSW lpWndClassW) { BOOL ret; CStrIn strClassName(lpClassName); ASSERT(sizeof(WNDCLASSA) == sizeof(WNDCLASSW)); ret = GetClassInfoA(hModule, strClassName, (LPWNDCLASSA)lpWndClassW); lpWndClassW->lpszMenuName = NULL; lpWndClassW->lpszClassName = NULL; return ret; } DWORD GetClassLongWrap(HWND hWnd, int nIndex) { return GetClassLongA(hWnd, nIndex); } int GetClassNameWrap(HWND hWnd, LPWSTR lpClassName, int nMaxCount) { CStrOut strClassName(lpClassName, nMaxCount); GetClassNameA(hWnd, strClassName, strClassName.BufSize()); return strClassName.ConvertIncludingNul(); } int GetClipboardFormatNameWrap(UINT format, LPWSTR lpFormatName, int cchFormatName) { CStrOut strFormatName(lpFormatName, cchFormatName); GetClipboardFormatNameA(format, strFormatName, strFormatName.BufSize()); return strFormatName.ConvertIncludingNul(); } DWORD GetCurrentDirectoryWrap(DWORD nBufferLength, LPWSTR lpBuffer) { CStrOut str(lpBuffer, nBufferLength); GetCurrentDirectoryA(str.BufSize(), str); return str.ConvertExcludingNul(); } int GetDateFormatWrap( LCID Locale, DWORD dwFlags, CONST SYSTEMTIME* lpDate, LPCWSTR lpFormat, LPWSTR lpDateStr, int cchDate) { CStrIn strFormat(lpFormat); CStrOut str(lpDateStr, cchDate); ASSERT(cchDate != 0 || lpDateStr == NULL); int iRc = GetDateFormatA(Locale, dwFlags, lpDate, strFormat, str, str.BufSize()); // If app was merely querying, then return size and stop if (!str) return iRc; return str.ConvertIncludingNul(); } UINT GetDlgItemTextWrap( HWND hWndDlg, int idControl, LPWSTR lpsz, int cchMax) { CStrOut str(lpsz, cchMax); GetDlgItemTextA(hWndDlg, idControl, str, str.BufSize()); return str.ConvertExcludingNul(); } DWORD GetFileAttributesWrap(LPCWSTR lpFileName) { CStrIn str(lpFileName); return GetFileAttributesA(str); } int GetKeyNameTextWrap(LONG lParam, LPWSTR lpsz, int nSize) { CStrOut str(lpsz, nSize); GetKeyNameTextA(lParam, str, str.BufSize()); return str.ConvertExcludingNul(); } int GetLocaleInfoWrap(LCID Locale, LCTYPE LCType, LPWSTR lpsz, int cchData) { CStrOut str(lpsz, cchData); GetLocaleInfoA(Locale, LCType, str, str.BufSize()); return str.ConvertIncludingNul(); } BOOL GetMenuItemInfoWrap( HMENU hMenu, UINT uItem, BOOL fByPosition, LPMENUITEMINFOW lpmiiW) { BOOL fRet; ASSERT(sizeof(MENUITEMINFOW) == sizeof(MENUITEMINFOA) && FIELD_OFFSET(MENUITEMINFOW, dwTypeData) == FIELD_OFFSET(MENUITEMINFOA, dwTypeData)); if ((MIIM_TYPE & lpmiiW->fMask) && 0 == (lpmiiW->fType & (MFT_BITMAP | MFT_SEPARATOR))) { MENUITEMINFOA miiA; CStrOut str(lpmiiW->dwTypeData, lpmiiW->cch); memcpy(&miiA, lpmiiW, sizeof(MENUITEMINFOA)); miiA.dwTypeData = str; miiA.cch = str.BufSize(); fRet = GetMenuItemInfoA(hMenu, uItem, fByPosition, &miiA); memcpy(lpmiiW, &miiA, FIELD_OFFSET(MENUITEMINFOW, dwTypeData)); } else { fRet = GetMenuItemInfoA(hMenu, uItem, fByPosition, (LPMENUITEMINFOA)lpmiiW); } return fRet; } int GetMenuStringWrap( HMENU hMenu, UINT uIDItem, LPWSTR lpString, int nMaxCount, UINT uFlag) { CStrOut str(lpString, nMaxCount); GetMenuStringA(hMenu, uIDItem, str, str.BufSize(), uFlag); return str.ConvertExcludingNul(); } BOOL GetMessageWrap( LPMSG lpMsg, HWND hWnd, UINT wMsgFilterMin, UINT wMsgFilterMax) { return GetMessageA(lpMsg, hWnd, wMsgFilterMin, wMsgFilterMax); } DWORD GetModuleFileNameWrap(HINSTANCE hModule, LPWSTR pwszFilename, DWORD nSize) { CStrOut str(pwszFilename, nSize); GetModuleFileNameA(hModule, str, str.BufSize()); return str.ConvertIncludingNul(); } int GetNumberFormatWrap( LCID Locale, DWORD dwFlags, LPCWSTR lpValue, CONST NUMBERFMTW* lpFormat, LPWSTR lpNumberStr, int cchNumber) { CStrIn strValue(lpValue); CStrOut str(lpNumberStr, cchNumber); ASSERT(cchNumber != 0); NUMBERFMTA nfA; CopyMemory(&nfA, lpFormat, sizeof(nfA)); CStrIn strDec(lpFormat->lpDecimalSep); nfA.lpDecimalSep = strDec; CStrIn strThou(lpFormat->lpThousandSep); nfA.lpThousandSep = strThou; GetNumberFormatA(Locale, dwFlags, strValue, &nfA, str, str.BufSize()); return str.ConvertIncludingNul(); } UINT GetSystemDirectoryWrap(LPWSTR lpBuffer, UINT uSize) { CStrOut str(lpBuffer, uSize); GetSystemDirectoryA(str, str.BufSize()); return str.ConvertExcludingNul(); } DWORD SearchPathWrap( LPCWSTR lpPathName, LPCWSTR lpFileName, LPCWSTR lpExtension, DWORD cchReturnBuffer, LPWSTR lpReturnBuffer, LPWSTR* plpfilePart) { CStrIn strPath(lpPathName); CStrIn strFile(lpFileName); CStrIn strExtension(lpExtension); CStrOut strReturnBuffer(lpReturnBuffer, cchReturnBuffer); DWORD dwLen = SearchPathA( strPath, strFile, strExtension, strReturnBuffer.BufSize(), strReturnBuffer, (LPSTR*)plpfilePart); // Getting the correct value for plpfilePart requires // a strrchr on the converted string. If this value // is needed, just add the code to do it here. *plpfilePart = NULL; if (cchReturnBuffer == 0) dwLen = 2 * dwLen; else dwLen = strReturnBuffer.ConvertExcludingNul(); return dwLen; } HMODULE GetModuleHandleWrap(LPCWSTR lpModuleName) { CStrIn str(lpModuleName); return GetModuleHandleA(str); } int GetObjectWrap(HGDIOBJ hgdiObj, int cbBuffer, LPVOID lpvObj) { int nRet; if (cbBuffer != sizeof(LOGFONTW)) { nRet = GetObjectA(hgdiObj, cbBuffer, lpvObj); } else { LOGFONTA lfa; nRet = GetObjectA(hgdiObj, sizeof(lfa), &lfa); if (nRet > 0) { memcpy(lpvObj, &lfa, FIELD_OFFSET(LOGFONTW, lfFaceName)); UnicodeFromMbcs(((LOGFONTW*)lpvObj)->lfFaceName, ARRAYSIZE(((LOGFONTW*)lpvObj)->lfFaceName), lfa.lfFaceName, -1); nRet = sizeof(LOGFONTW); } } return nRet; } // GetFullPathNameWrap DWORD GetFullPathNameWrap(LPCWSTR lpFileName, DWORD nBufferLength, LPWSTR lpBuffer, LPWSTR* lpFilePart) { CStrIn strIn(lpFileName); CStrOut strOut(lpBuffer, nBufferLength); LPSTR pFile; DWORD dwRet; dwRet = GetFullPathNameA(strIn, nBufferLength, strOut, &pFile); strOut.ConvertIncludingNul(); *lpFilePart = lpBuffer + (pFile - strOut); return dwRet; } BOOL GetStringTypeExWrap(LCID lcid, DWORD dwInfoType, LPCTSTR lpSrcStr, int cchSrc, LPWORD lpCharType) { CStrIn str(lpSrcStr, cchSrc); return GetStringTypeExA(lcid, dwInfoType, str, str.strlen(), lpCharType); } UINT GetPrivateProfileIntWrap( LPCWSTR lpAppName, LPCWSTR lpKeyName, INT nDefault, LPCWSTR lpFileName) { CStrIn strApp(lpAppName); CStrIn strKey(lpKeyName); CStrIn strFile(lpFileName); return GetPrivateProfileIntA(strApp, strKey, nDefault, strFile); } UINT GetProfileIntWrap( LPCWSTR lpAppName, LPCWSTR lpKeyName, INT nDefault) { CStrIn strApp(lpAppName); CStrIn strKey(lpKeyName); return GetProfileIntA(strApp, strKey, nDefault); } DWORD GetProfileStringWrap( LPCWSTR lpAppName, LPCWSTR lpKeyName, LPCWSTR lpDefault, LPWSTR lpBuffer, DWORD dwBuffersize) { CStrIn strApp(lpAppName); CStrIn strKey(lpKeyName); CStrIn strDefault(lpDefault); CStrOut strBuffer(lpBuffer, dwBuffersize); GetProfileStringA(strApp, strKey, strDefault, strBuffer, dwBuffersize); return strBuffer.ConvertIncludingNul(); } HANDLE GetPropWrap(HWND hWnd, LPCWSTR lpString) { CStrIn str(lpString); return GetPropA(hWnd, str); } UINT GetTempFileNameWrap( LPCWSTR lpPathName, LPCWSTR lpPrefixString, UINT uUnique, LPWSTR lpTempFileName) { CStrIn strPath(lpPathName); CStrIn strPrefix(lpPrefixString); CStrOut strFileName(lpTempFileName, MAX_PATH); return GetTempFileNameA(strPath, strPrefix, uUnique, strFileName); } DWORD GetTempPathWrap(DWORD nBufferLength, LPWSTR lpBuffer) { CStrOut str(lpBuffer, nBufferLength); GetTempPathA(str.BufSize(), str); return str.ConvertExcludingNul(); } #ifndef FONT_LINK BOOL GetTextExtentPointWrap( HDC hdc, LPCWSTR pwsz, int cb, LPSIZE pSize) { CStrIn str(pwsz, cb); return GetTextExtentPointA(hdc, str, str.strlen(), pSize); } BOOL GetTextExtentPoint32Wrap( HDC hdc, LPCWSTR pwsz, int cb, LPSIZE pSize) { CStrIn str(pwsz, cb); return GetTextExtentPoint32A(hdc, str, str.strlen(), pSize); } #endif int GetTextFaceWrap( HDC hdc, int cch, LPWSTR lpFaceName) { CStrOut str(lpFaceName, cch); GetTextFaceA(hdc, str.BufSize(), str); return str.ConvertIncludingNul(); } BOOL GetTextMetricsWrap(HDC hdc, LPTEXTMETRICW lptm) { BOOL ret; TEXTMETRICA tm; ret = GetTextMetricsA(hdc, &tm); if (ret) { lptm->tmHeight = tm.tmHeight; lptm->tmAscent = tm.tmAscent; lptm->tmDescent = tm.tmDescent; lptm->tmInternalLeading = tm.tmInternalLeading; lptm->tmExternalLeading = tm.tmExternalLeading; lptm->tmAveCharWidth = tm.tmAveCharWidth; lptm->tmMaxCharWidth = tm.tmMaxCharWidth; lptm->tmWeight = tm.tmWeight; lptm->tmOverhang = tm.tmOverhang; lptm->tmDigitizedAspectX = tm.tmDigitizedAspectX; lptm->tmDigitizedAspectY = tm.tmDigitizedAspectY; lptm->tmItalic = tm.tmItalic; lptm->tmUnderlined = tm.tmUnderlined; lptm->tmStruckOut = tm.tmStruckOut; lptm->tmPitchAndFamily = tm.tmPitchAndFamily; lptm->tmCharSet = tm.tmCharSet; UnicodeFromMbcs(&lptm->tmFirstChar, 1, (LPSTR)&tm.tmFirstChar, 1); UnicodeFromMbcs(&lptm->tmLastChar, 1, (LPSTR)&tm.tmLastChar, 1); UnicodeFromMbcs(&lptm->tmDefaultChar, 1, (LPSTR)&tm.tmDefaultChar, 1); UnicodeFromMbcs(&lptm->tmBreakChar, 1, (LPSTR)&tm.tmBreakChar, 1); } return ret; } int GetTimeFormatWrap( LCID Locale, DWORD dwFlags, CONST SYSTEMTIME* lpTime, LPCWSTR lpFormat, LPWSTR lpTimeStr, int cchTime) { CStrIn strFormat(lpFormat); CStrOut str(lpTimeStr, cchTime); ASSERT(cchTime != 0); GetTimeFormatA(Locale, dwFlags, lpTime, strFormat, str, str.BufSize()); return str.ConvertIncludingNul(); } LONG GetWindowLongWrap(HWND hWnd, int nIndex) { return GetWindowLongA(hWnd, nIndex); } int GetWindowTextWrap(HWND hWnd, LPWSTR lpString, int nMaxCount) { CStrOut str(lpString, nMaxCount); GetWindowTextA(hWnd, str, str.BufSize()); return str.ConvertExcludingNul(); } int GetWindowTextLengthWrap(HWND hWnd) { WCHAR wstr[MAX_PATH]; return GetWindowTextWrap(hWnd, wstr, ARRAYSIZE(wstr)); } UINT GetWindowsDirectoryWrap(LPWSTR lpWinPath, UINT cch) { CStrOut str(lpWinPath, cch); GetWindowsDirectoryA(str, str.BufSize()); return str.ConvertExcludingNul(); } ATOM GlobalAddAtomWrap(LPCWSTR lpString) { CStrIn str(lpString); return GlobalAddAtomA(str); } BOOL GrayStringWrap( HDC hDC, HBRUSH hBrush, GRAYSTRINGPROC lpOutputFunc, LPARAM lpData, int nCount, int x, int y, int nWidth, int nHeight) { CStrIn str((LPWSTR)lpData); return GrayStringA(hDC, hBrush, lpOutputFunc, (LPARAM)(LPCSTR)str, str.strlen(), x, y, nWidth, nHeight); } LONG ImmGetCompositionStringWrap(HIMC hIMC, DWORD dwIndex, LPVOID lpBuf, DWORD dwBufLen) { int cb = 0; if ((dwIndex & GCS_COMPSTR) || (dwIndex & GCS_RESULTSTR)) { if (dwBufLen) { CStrOut str((LPWSTR)lpBuf, dwBufLen / sizeof(WCHAR) + 1); cb = ImmGetCompositionStringA(hIMC, dwIndex, str, str.BufSize()); *(WCHAR*)((LPSTR)str + cb) = L'\0'; return str.ConvertExcludingNul() * sizeof(WCHAR); } else { LPWSTR lpStr; cb = ImmGetCompositionStringA(hIMC, dwIndex, lpBuf, dwBufLen); lpStr = (LPWSTR)LocalAlloc(LPTR, (cb + 1) * sizeof(WCHAR)); if (lpStr) { CStrOut str(lpStr, cb + 1); cb = ImmGetCompositionStringA(hIMC, dwIndex, str, str.BufSize()); *(WCHAR*)((LPSTR)str + cb) = L'\0'; cb = str.ConvertExcludingNul() * sizeof(WCHAR); LocalFree(lpStr); return cb; } } } else if (dwIndex & GCS_COMPATTR) { if (dwBufLen) { LPSTR lpStr, lpAttr; UINT i = 0; lpStr = (LPSTR)LocalAlloc(LPTR, dwBufLen); if (lpStr) { lpAttr = (LPSTR)LocalAlloc(LPTR, dwBufLen); if (lpAttr) { LPSTR lpNext = lpStr; cb = ImmGetCompositionStringA(hIMC, GCS_COMPSTR, lpStr, dwBufLen); ImmGetCompositionStringA(hIMC, GCS_COMPATTR, lpAttr, dwBufLen); for (i = 0; (lpNext - lpStr < cb) && (i < dwBufLen); i++) { ((LPSTR)lpBuf)[i] = lpAttr[lpNext - lpStr]; lpNext = CharNextA(lpNext); } LocalFree(lpAttr); } LocalFree(lpStr); } return i; } } return ImmGetCompositionStringA(hIMC, dwIndex, lpBuf, dwBufLen); } LONG ImmSetCompositionStringWrap(HIMC hIMC, DWORD dwIndex, LPVOID lpComp, DWORD dwCompLen, LPVOID lpRead, DWORD dwReadLen) { if (dwIndex & SCS_SETSTR) { CStrIn str((LPWSTR)lpComp); ASSERT(!lpRead); return ImmSetCompositionStringA(hIMC, dwIndex, str, str.strlen(), lpRead, dwReadLen); } return ImmSetCompositionStringA(hIMC, dwIndex, lpComp, dwCompLen, lpRead, dwReadLen); } BOOL InsertMenuWrap( HMENU hMenu, UINT uPosition, UINT uFlags, UINT uIDNewItem, LPCWSTR lpNewItem) { CStrIn str(lpNewItem); return InsertMenuA(hMenu, uPosition, uFlags, uIDNewItem, str); } BOOL IsDialogMessageWrap(HWND hWndDlg, LPMSG lpMsg) { return IsDialogMessageA(hWndDlg, lpMsg); } HACCEL LoadAcceleratorsWrap(HINSTANCE hInstance, LPCWSTR lpTableName) { CStrIn str(lpTableName); return LoadAcceleratorsA(hInstance, (LPCSTR)str); } HBITMAP LoadBitmapWrap(HINSTANCE hInstance, LPCWSTR lpBitmapName) { CStrIn str(lpBitmapName); return LoadBitmapA(hInstance, str); } HCURSOR LoadCursorWrap(HINSTANCE hInstance, LPCWSTR lpCursorName) { CStrIn str(lpCursorName); return LoadCursorA(hInstance, (LPCSTR)str); } HICON LoadIconWrap(HINSTANCE hInstance, LPCWSTR lpIconName) { CStrIn str(lpIconName); return LoadIconA(hInstance, str); } HANDLE LoadImageWrap( HINSTANCE hInstance, LPCWSTR lpName, UINT uType, int cxDesired, int cyDesired, UINT fuLoad) { CStrIn str(lpName); return LoadImageA( hInstance, str, uType, cxDesired, cyDesired, fuLoad); } HINSTANCE LoadLibraryWrap(LPCWSTR lpLibFileName) { CStrIn str(lpLibFileName); return LoadLibraryA(str); } HINSTANCE LoadLibraryExWrap( LPCWSTR lpLibFileName, HANDLE hFile, DWORD dwFlags) { CStrIn str(lpLibFileName); return LoadLibraryExA(str, hFile, dwFlags); } HMENU LoadMenuWrap(HINSTANCE hInstance, LPCWSTR lpMenuName) { ASSERT(HIWORD64(lpMenuName) == 0); return LoadMenuA(hInstance, (LPCSTR)lpMenuName); } int LoadStringWrap(HINSTANCE hInstance, UINT uID, LPWSTR lpBuffer, int nBufferMax) { // Do it manually. The old code used to call LoadStringA and then // convert it up to unicode, which is stupid since resources are // physically already Unicode! Just suck it out directly. if (nBufferMax <= 0) return 0; // sanity check PWCHAR pwch; /* * String tables are broken up into "bundles" of 16 strings each. */ HRSRC hrsrc; int cwch = 0; hrsrc = FindResourceA(hInstance, (LPSTR)(LONG_PTR)(1 + uID / 16), (LPSTR)RT_STRING); if (hrsrc) { pwch = (PWCHAR)LoadResource(hInstance, hrsrc); if (pwch) { /* * Now skip over the strings in the resource until we * hit the one we want. Each entry is a counted string, * just like Pascal. */ for (uID %= 16; uID; uID--) { pwch += *pwch + 1; } cwch = min(*pwch, nBufferMax - 1); memcpy(lpBuffer, pwch + 1, cwch * sizeof(WCHAR)); /* Copy the goo */ } } lpBuffer[cwch] = L'\0'; /* Terminate the string */ return cwch; } UINT MapVirtualKeyWrap(UINT uCode, UINT uMapType) { return MapVirtualKeyA(uCode, uMapType); } // function: TransformCharNoOp1( WCHAR **ppch ) // purpose: Stand-in for TransformCharWidth. Used by the function // CompareStringString. // returns: Character at *ppch. The value *ppch is incremented. static WCHAR TransformCharNoOp1(LPCWSTR* ppch, int) { WCHAR ch = **ppch; (*ppch)++; return ch; } // function: TransformCharWidth( WCHAR **ppch, cchRemaining ) // purpose: Converts halfwidth characters to fullwidth characters. // Also combines voiced (dakuon) and semi-voiced (handakuon) // characters. *pch is advanced by one, unless there is a // (semi)voiced character, in which case it is advanced by // two characters. // Note that unlike the full widechar version, we do not // combine other characters, notably the combining Hiragana // characters (U+3099 and U+309A.) This is to keep the // tables from getting unnecessarily large. // cchRemaining is passed so as to not include the voiced // marks if it's passed the end of the specified buffer. // returns: Full width character. *pch is incremented. static WCHAR TransformCharWidth(LPCWSTR* ppch, int cchRemaining) { WCHAR ch = **ppch; (*ppch)++; if (ch == 0x0020) { ch = 0x3000; } else if (ch == 0x005c) { // REVERSE SOLIDUS (aka BACKSLASH) maps to itself } else if (InRange(ch, 0x0021, 0x07e)) { ch += 65248; } else if (InRange(ch, 0x00a2, 0x00af)) { static const WCHAR achFull[] = { 0xffe0, 0xffe1, 0x00a4, 0xffe5, 0xffe4, 0x00a7, 0x00a8, // 0xa2-0xa8 0x00a9, 0x00aa, 0x00ab, 0xffe2, 0x00ad, 0x00ae, 0xffe3 // 0xa9-0xaf }; ch = achFull[ch - 0x00a2]; } else if (ch == 0x20a9) // WON SIGN { ch = 0xffe6; } else if (InRange(ch, 0xff61, 0xffdc)) { WCHAR chNext = (cchRemaining > 1) ? **ppch : 0; if (chNext == 0xff9e && InRange(ch, 0xff73, 0xff8e)) { if (cchRemaining != 1) { static const WCHAR achFull[] = { /* 0xff73-0xff79 */ 0xb0f4, 0x30a8, 0x30aa, 0xb0ac, 0xb0ae, 0xb0b0, 0xb0b2, /* 0xff7a-0xff80 */ 0xb0b4, 0xb0b6, 0xb0b8, 0xb0ba, 0xb0bc, 0xb0be, 0xb0c0, /* 0xff81-0xff87 */ 0xb0c2, 0xb0c5, 0xb0c7, 0xb0c9, 0x30ca, 0x30cb, 0x30cc, /* 0xff88-0xff8e */ 0x30cd, 0x30ce, 0xb0d0, 0xb0d3, 0xb0d6, 0xb0d9, 0xb0dc }; // HALFWIDTH KATAKANA VOICED SOUND MARK WCHAR chTemp = achFull[ch - 0xff73]; // Some in the range absorb the sound mark. // These are indicated by the set high-bit. ch = chTemp & 0x7fff; if (chTemp & 0x8000) { (*ppch)++; } } } else if (chNext == 0xff9f && InRange(ch, 0xff8a, 0xff8e)) { // HALFWIDTH KATAKANA SEMI-VOICED SOUND MARK ch = 0x30d1 + (ch - 0xff8a) * 3; (*ppch)++; } else { static const WCHAR achMapFullFFxx[] = { 0x3002, 0x300c, 0x300d, 0x3001, 0x30fb, 0x30f2, 0x30a1, // 0xff61-0xff67 0x30a3, 0x30a5, 0x30a7, 0x30a9, 0x30e3, 0x30e5, 0x30e7, // 0xff68-0xff6e 0x30c3, 0x30fc, 0x30a2, 0x30a4, 0x30a6, 0x30a8, 0x30aa, // 0xff6f-0xff75 0x30ab, 0x30ad, 0x30af, 0x30b1, 0x30b3, 0x30b5, 0x30b7, // 0xff76-0xff7c 0x30b9, 0x30bb, 0x30bd, 0x30bf, 0x30c1, 0x30c4, 0x30c6, // 0xff7d-0xff83 0x30c8, 0x30ca, 0x30cb, 0x30cc, 0x30cd, 0x30ce, 0x30cf, // 0xff84-0xff8a 0x30d2, 0x30d5, 0x30d8, 0x30db, 0x30de, 0x30df, 0x30e0, // 0xff8b-0xff91 0x30e1, 0x30e2, 0x30e4, 0x30e6, 0x30e8, 0x30e9, 0x30ea, // 0xff92-0xff98 0x30eb, 0x30ec, 0x30ed, 0x30ef, 0x30f3, 0x309b, 0x309c, // 0xff99-0xff9f 0x3164, 0x3131, 0x3132, 0x3133, 0x3134, 0x3135, 0x3136, // 0xffa0-0xffa6 0x3137, 0x3138, 0x3139, 0x313a, 0x313b, 0x313c, 0x313d, // 0xffa7-0xffad 0x313e, 0x313f, 0x3140, 0x3141, 0x3142, 0x3143, 0x3144, // 0xffae-0xffb4 0x3145, 0x3146, 0x3147, 0x3148, 0x3149, 0x314a, 0x314b, // 0xffb5-0xffbb 0x314c, 0x314d, 0x314e, 0xffbf, 0xffc0, 0xffc1, 0x314f, // 0xffbc-0xffc2 0x3150, 0x3151, 0x3152, 0x3153, 0x3154, 0xffc8, 0xffc9, // 0xffc3-0xffc9 0x3155, 0x3156, 0x3157, 0x3158, 0x3159, 0x315a, 0xffd0, // 0xffca-0xffd0 0xffd1, 0x315b, 0x315c, 0x315d, 0x315e, 0x315f, 0x3160, // 0xffd1-0xffd7 0xffd8, 0xffd9, 0x3161, 0x3162, 0x3163 // 0xffd8-0xffac }; ch = achMapFullFFxx[ch - 0xff61]; } } return ch; } // function: TransformaCharNoOp2( WCHAR ch ) // purpose: Stand-in for CharLowerBuffWrap. Used by the function // CompareStringString. // returns: Original character static WCHAR TransformCharNoOp2(WCHAR ch) { return ch; } // function: TransformaCharKana( WCHAR ch ) // purpose: Converts Hiragana characters to Katakana characters // returns: Original character if not Hiragana, // Katanaka character if Hiragana static WCHAR TransformCharKana(WCHAR ch) { if (((ch & 0xff00) == 0x3000) && (InRange(ch, 0x3041, 0x3094) || InRange(ch, 0x309d, 0x309e))) { ch += 0x060; } return ch; } // function: TransformCharNoOp3( LPWSTR pch, DWORD cch ) // purpose: Stand-in for CharLowerBuffWrap. Used by the function // CompareStringString. // returns: Character count (cch). static DWORD TransformCharNoOp3(LPWSTR, DWORD cch) { return cch; } // function: TransformaCharFinal( WCHAR ch ) // purpose: Converts "final" forms to regular forms // returns: Original character if not Hiragana, // Katanaka character if Hiragana // BUGBUG (cthrash) We do not fold Presentation Forms (Alphabetic or Arabic) static WCHAR TransformCharFinal(WCHAR ch) { WCHAR chRet = ch; if (ch >= 0x3c2) // short-circuit ASCII + { switch (ch) { case 0x03c2: // GREEK SMALL LETTER FINAL SIGMA case 0x05da: // HEBREW LETTER FINAL KAF case 0x05dd: // HEBREW LETTER FINAL MEM case 0x05df: // HEBREW LETTER FINAL NUN case 0x05e3: // HEBREW LETTER FINAL PE case 0x05e5: // HEBREW LETTER FINAL TSADI case 0xfb26: // HEBREW LETTER WIDE FINAL MEM case 0xfb3a: // HEBREW LETTER FINAL KAF WITH DAGESH case 0xfb43: // HEBREW LETTER FINAL PE WITH DAGESH chRet++; break; } } return ch; } // function: CompareStringString( ... ) // purpose: Helper for CompareStringWrap. // We handle the string comparsion for CompareStringWrap. // We can convert each character to (1) fullwidth, // (2) Katakana, and (3) lowercase, as necessary. // returns: 1 - string A is less in lexical value as string B // 2 - string B is equal in lexical value as string B // 3 - string B is greater in lexical value as string B static int CompareStringString( DWORD dwFlags, LPCWSTR lpA, int cchA, LPCWSTR lpB, int cchB) { int nRet = 0; WCHAR wchIgnoreNulA = cchA == -1 ? 0 : -1; WCHAR wchIgnoreNulB = cchB == -1 ? 0 : -1; WCHAR(*pfnTransformWidth)(LPCWSTR*, int); WCHAR(*pfnTransformKana)(WCHAR); DWORD(*pfnTransformLower)(LPWSTR, DWORD); WCHAR(*pfnTransformFinal)(WCHAR); pfnTransformWidth = (dwFlags & NORM_IGNOREWIDTH) ? TransformCharWidth : TransformCharNoOp1; pfnTransformKana = (dwFlags & NORM_IGNOREKANATYPE) ? TransformCharKana : TransformCharNoOp2; pfnTransformLower = (dwFlags & NORM_IGNORECASE) ? CharLowerBuffWrap : TransformCharNoOp3; pfnTransformFinal = (dwFlags & NORM_IGNORECASE) ? TransformCharFinal : TransformCharNoOp2; while (!nRet && cchA && cchB && (*lpA | wchIgnoreNulA) && (*lpB | wchIgnoreNulB)) { WCHAR chA, chB; LPCWSTR lpAOld = lpA; LPCWSTR lpBOld = lpB; chA = (*pfnTransformWidth)(&lpA, cchA); chA = (*pfnTransformKana)(chA); (*pfnTransformLower)(&chA, 1); chA = (*pfnTransformFinal)(chA); chB = (*pfnTransformWidth)(&lpB, cchB); chB = (*pfnTransformKana)(chB); (*pfnTransformLower)(&chB, 1); chB = (*pfnTransformFinal)(chB); nRet = (int)chA - (int)chB; cchA -= (int)(lpA - lpAOld); cchB -= (int)(lpB - lpBOld); } if (!nRet) { nRet = cchA - cchB; } if (nRet) { nRet = nRet > 0 ? 1 : -1; } return nRet + 2; } // function: CompareStringWord( ... ) // purpose: Helper for CompareStringWrap. // We handle the word comparsion for CompareStringWrap. // returns: 1 - string A is less in lexical value as string B // 2 - string B is equal in lexical value as string B // 3 - string B is greater in lexical value as string B static int CompareStringWord( LCID lcid, DWORD dwFlags, LPCWSTR lpA, int cchA, LPCWSTR lpB, int cchB) { // BUGBUG (cthrash) We won't properly support word compare for the // time being. Do the same old CP_ACP trick, which should cover // enough cases. // fail if either string is NULL, as it causes assert on debug windows if (!lpA || !lpB) return 0; CStrIn strA(lpA, cchA); CStrIn strB(lpB, cchB); cchA = strA.strlen(); cchB = strB.strlen(); return CompareStringA(lcid, dwFlags, strA, cchA, strB, cchB); } // function: CompareStringWrap( ... ) // purpose: Unicode wrapper of CompareString for Win95. // Note not all bits in dwFlags are honored; specifically, // since we don't do a true widechar word compare, we // won't properly handle NORM_IGNORENONSPACE or // NORM_IGNORESYMBOLS for arbitrary widechar strings. // returns: 1 - string A is less in lexical value as string B // 2 - string B is equal in lexical value as string B // 3 - string B is greater in lexical value as string B LWSTDAPI_(int) CompareStringAltW( LCID lcid, DWORD dwFlags, LPCWSTR lpA, int cchA, LPCWSTR lpB, int cchB) { int nRet; if (dwFlags & SORT_STRINGSORT) { nRet = CompareStringString(dwFlags, lpA, cchA, lpB, cchB); } else { nRet = CompareStringWord(lcid, dwFlags, lpA, cchA, lpB, cchB); } return nRet; } int CompareStringWrap( LCID Locale, DWORD dwCmpFlags, LPCWSTR lpString1, int cchCount1, LPCWSTR lpString2, int cchCount2) { // fail if either string is NULL, as it causes assert on debug windows if (!lpString1 || !lpString2) return 0; CStrIn strString1(lpString1, cchCount1); CStrIn strString2(lpString2, cchCount2); cchCount1 = strString1.strlen(); cchCount2 = strString2.strlen(); return CompareStringA(Locale, dwCmpFlags, strString1, cchCount1, strString2, cchCount2); } BOOL MessageBoxIndirectWrap(MSGBOXPARAMS* pmbp) { CStrIn strText(pmbp->lpszText); CStrIn strCaption(pmbp->lpszCaption); MSGBOXPARAMSA mbp; memcpy(&mbp, pmbp, sizeof(mbp)); mbp.lpszText = strText; mbp.lpszCaption = strCaption; ASSERT(HIWORD64(mbp.lpszIcon) == 0); return MessageBoxIndirectA(&mbp); } DWORD GetCharacterPlacementWrap( HDC hdc, // handle to device context LPCTSTR lpString, // pointer to string int nCount, // number of characters in string int nMaxExtent, // maximum extent for displayed string LPGCP_RESULTS lpResults, // pointer to buffer for placement result DWORD dwFlags // placement flags ) { CStrIn strText(lpString); DWORD dwRet; // Leave for someone else. ASSERT(lpResults->lpOutString == NULL); ASSERT(lpResults->lpClass == NULL); dwRet = GetCharacterPlacementA(hdc, strText, nCount, nMaxExtent, (LPGCP_RESULTSA)lpResults, dwFlags); return dwRet; } #ifndef FONT_LINK BOOL GetCharWidthWrap( HDC hdc, UINT iFirstChar, UINT iLastChar, LPINT lpBuffer) { // Note that we expect to do only one character at a time for anything but // ISO Latin 1. if (iFirstChar > 255) { UINT mbChar = 0; WCHAR ch; // Convert string ch = (WCHAR)iFirstChar; WideCharToMultiByte(CP_ACP, 0, &ch, 1, (char*)&mbChar, 2, NULL, NULL); } return (GetCharWidthA(hdc, iFirstChar, iLastChar, lpBuffer)); } #endif BOOL ModifyMenuWrap( HMENU hMenu, UINT uPosition, UINT uFlags, UINT uIDNewItem, LPCWSTR lpNewItem) { ASSERT(!(uFlags & MF_BITMAP) && !(uFlags & MF_OWNERDRAW)); CStrIn str(lpNewItem); return ModifyMenuA(hMenu, uPosition, uFlags, uIDNewItem, str); } BOOL CopyFileWrap(LPCWSTR lpExistingFileName, LPCWSTR lpNewFileName, BOOL bFailIfExists) { CStrIn strOld(lpExistingFileName); CStrIn strNew(lpNewFileName); return CopyFileA(strOld, strNew, bFailIfExists); } BOOL MoveFileWrap(LPCWSTR lpExistingFileName, LPCWSTR lpNewFileName) { CStrIn strOld(lpExistingFileName); CStrIn strNew(lpNewFileName); return MoveFileA(strOld, strNew); } BOOL OemToCharWrap(LPCSTR lpszSrc, LPWSTR lpszDst) { CStrOut strDst(lpszDst, lstrlenA(lpszSrc)); return OemToCharA(lpszSrc, strDst); } VOID OutputDebugStringWrap(LPCWSTR lpOutputString) { CStrIn str(lpOutputString); OutputDebugStringA(str); } BOOL PeekMessageWrap( LPMSG lpMsg, HWND hWnd, UINT wMsgFilterMin, UINT wMsgFilterMax, UINT wRemoveMsg) { return PeekMessageA(lpMsg, hWnd, wMsgFilterMin, wMsgFilterMax, wRemoveMsg); } BOOL PostMessageWrap( HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam) { return PostMessageA(hWnd, Msg, wParam, lParam); } BOOL PostThreadMessageWrap( DWORD idThread, UINT Msg, WPARAM wParam, LPARAM lParam) { return PostThreadMessageA(idThread, Msg, wParam, lParam); } LONG RegCreateKeyWrap(HKEY hKey, LPCWSTR lpSubKey, PHKEY phkResult) { CStrIn str(lpSubKey); return RegCreateKeyA(hKey, str, phkResult); } LONG RegCreateKeyExWrap(HKEY hKey, LPCTSTR lpSubKey, DWORD Reserved, LPTSTR lpClass, DWORD dwOptions, REGSAM samDesired, LPSECURITY_ATTRIBUTES lpSecurityAttributes, PHKEY phkResult, LPDWORD lpdwDisposition) { CStrIn strSubKey(lpSubKey); CStrIn strClass(lpClass); return RegCreateKeyExA(hKey, strSubKey, Reserved, strClass, dwOptions, samDesired, lpSecurityAttributes, phkResult, lpdwDisposition); } LONG RegDeleteKeyWrap(HKEY hKey, LPCWSTR pwszSubKey) { CStrIn str(pwszSubKey); return RegDeleteKeyA(hKey, str); } LONG RegEnumKeyWrap( HKEY hKey, DWORD dwIndex, LPWSTR lpName, DWORD cbName) { CStrOut str(lpName, cbName); return RegEnumKeyA(hKey, dwIndex, str, str.BufSize()); } LONG RegEnumKeyExWrap( HKEY hKey, DWORD dwIndex, LPWSTR lpName, LPDWORD lpcbName, LPDWORD lpReserved, LPWSTR lpClass, LPDWORD lpcbClass, PFILETIME lpftLastWriteTime) { long ret; DWORD dwClass = 0; if (!lpcbClass) { lpcbClass = &dwClass; } CStrOut strName(lpName, *lpcbName); CStrOut strClass(lpClass, *lpcbClass); ret = RegEnumKeyExA( hKey, dwIndex, strName, lpcbName, lpReserved, strClass, lpcbClass, lpftLastWriteTime); *lpcbName = strName.ConvertExcludingNul(); *lpcbClass = strClass.ConvertExcludingNul(); return ret; } LONG RegOpenKeyWrap(HKEY hKey, LPCWSTR pwszSubKey, PHKEY phkResult) { CStrIn str(pwszSubKey); return RegOpenKeyA(hKey, str, phkResult); } LONG RegOpenKeyExWrap(HKEY hKey, LPCWSTR lpSubKey, DWORD ulOptions, REGSAM samDesired, PHKEY phkResult) { CStrIn str(lpSubKey); return RegOpenKeyExA(hKey, str, ulOptions, samDesired, phkResult); } LONG RegQueryInfoKeyWrap( HKEY hKey, LPWSTR lpClass, LPDWORD lpcbClass, LPDWORD lpReserved, LPDWORD lpcSubKeys, LPDWORD lpcbMaxSubKeyLen, LPDWORD lpcbMaxClassLen, LPDWORD lpcValues, LPDWORD lpcbMaxValueNameLen, LPDWORD lpcbMaxValueLen, LPDWORD lpcbSecurityDescriptor, PFILETIME lpftLastWriteTime) { CStrIn str(lpClass); return RegQueryInfoKeyA( hKey, str, lpcbClass, lpReserved, lpcSubKeys, lpcbMaxSubKeyLen, lpcbMaxClassLen, lpcValues, lpcbMaxValueNameLen, lpcbMaxValueLen, lpcbSecurityDescriptor, lpftLastWriteTime); } LONG RegQueryValueWrap( HKEY hKey, LPCWSTR pwszSubKey, LPWSTR pwszValue, PLONG lpcbValue) { long ret; long cb; CStrIn strKey(pwszSubKey); CStrOut strValue(pwszValue, (*lpcbValue) / sizeof(WCHAR)); cb = strValue.BufSize(); ret = RegQueryValueA(hKey, strKey, strValue, &cb); if (ret != ERROR_SUCCESS) goto Cleanup; if (strValue) { cb = strValue.ConvertIncludingNul(); } *lpcbValue = cb * sizeof(WCHAR); Cleanup: return ret; } LONG RegQueryValueExWrap( HKEY hKey, LPCWSTR lpValueName, LPDWORD lpReserved, LPDWORD lpType, LPBYTE lpData, LPDWORD lpcbData) { LONG ret; CStrIn strValueName(lpValueName); DWORD dwTempType; DWORD cb; // Determine the type of buffer needed ret = RegQueryValueExA(hKey, strValueName, lpReserved, &dwTempType, NULL, &cb); if (ret != ERROR_SUCCESS) goto Cleanup; ASSERT(dwTempType != REG_MULTI_SZ); switch (dwTempType) { case REG_EXPAND_SZ: case REG_SZ: { CStrOut strData((LPWSTR)lpData, (*lpcbData) / sizeof(WCHAR)); cb = strData.BufSize(); ret = RegQueryValueExA(hKey, strValueName, lpReserved, lpType, (LPBYTE)(LPSTR)strData, &cb); if (ret != ERROR_SUCCESS) break; if (strData) { cb = strData.ConvertIncludingNul(); } *lpcbData = cb * sizeof(WCHAR); break; } default: { ret = RegQueryValueExA(hKey, strValueName, lpReserved, lpType, lpData, lpcbData); break; } } Cleanup: return ret; } LONG RegSetValueWrap( HKEY hKey, LPCWSTR lpSubKey, DWORD dwType, LPCWSTR lpData, DWORD cbData) { CStrIn strKey(lpSubKey); CStrIn strValue(lpData); return RegSetValueA(hKey, strKey, dwType, strValue, cbData); } LONG RegSetValueExWrap( HKEY hKey, LPCWSTR lpValueName, DWORD Reserved, DWORD dwType, CONST BYTE* lpData, DWORD cbData) { ASSERT(dwType != REG_MULTI_SZ); CStrIn strKey(lpValueName); CStrIn strSZ((dwType == REG_SZ || dwType == REG_EXPAND_SZ) ? (LPCWSTR)lpData : NULL); if (strSZ) { lpData = (LPBYTE)(LPSTR)strSZ; cbData = strSZ.strlen() + 1; } return RegSetValueExA(hKey, strKey, Reserved, dwType, lpData, cbData); } ATOM RegisterClassWrap(CONST WNDCLASSW* lpWndClass) { WNDCLASSA wc; CStrIn strMenuName(lpWndClass->lpszMenuName); CStrIn strClassName(lpWndClass->lpszClassName); ASSERT(sizeof(wc) == sizeof(*lpWndClass)); memcpy(&wc, lpWndClass, sizeof(wc)); wc.lpszMenuName = strMenuName; wc.lpszClassName = strClassName; return RegisterClassA(&wc); } UINT RegisterClipboardFormatWrap(LPCWSTR lpString) { CStrIn str(lpString); return RegisterClipboardFormatA(str); } UINT RegisterWindowMessageWrap(LPCWSTR lpString) { CStrIn str(lpString); return RegisterWindowMessageA(str); } HANDLE RemovePropWrap( HWND hWnd, LPCWSTR lpString) { CStrIn str(lpString); return RemovePropA(hWnd, str); } // NOTE (SumitC) Instead of calling SendDlgItemMessageA below, I'm forwarding to // SendMessageWrap so as not to have to re-do the special-case processing. LRESULT SendDlgItemMessageWrap( HWND hDlg, int nIDDlgItem, UINT Msg, WPARAM wParam, LPARAM lParam) { HWND hWnd; hWnd = GetDlgItem(hDlg, nIDDlgItem); return SendMessageWrap(hWnd, Msg, wParam, lParam); } // AdjustECPosition // Convert mulitbyte position to unicode number of character position in EDIT control // iType: ADJUST_TO_WCHAR_POS or ADJUST_TO_CHAR_POS #define ADJUST_TO_WCHAR_POS 0 #define ADJUST_TO_CHAR_POS 1 int AdjustECPosition(char* psz, int iPos, int iType) { char* pstr = psz; int iNewPos = iPos; if (ADJUST_TO_WCHAR_POS == iType) { iNewPos = 0; while (*pstr && (pstr - psz != iPos)) { pstr = CharNextA(pstr); iNewPos++; } } else if (ADJUST_TO_CHAR_POS == iType) { while (*pstr && iPos--) pstr = CharNextA(pstr); iNewPos = (int)(pstr - psz); } return iNewPos; } // Edit controls can get really huge, so the MAX_PATH buffer in // SendMessageWrap just doesn't cut it when push comes to shove. // Try to use the small buffer, and switch to an allocated buffer // only if the small buffer doesn't work. // Use the handy CConvertStr class as our basis. class CStrA : public CConvertStr { public: CStrA(int cch); inline int bufsize() { return _cchLen; } protected: int _cchLen; }; CStrA::CStrA(int cch) : CConvertStr(CP_ACP) { _cchLen = cch; if (cch <= ARRAYSIZE(_ach)) { // It fits in our small buffer _pstr = _ach; } else { // Need to allocate a big buffer _pstr = new char[cch]; if (!_pstr) { // On failure, use the small buffer after all. _pstr = _ach; _cchLen = ARRAYSIZE(_ach); } } } LRESULT SendEditMessageWrap( HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam) { WORD wStart, wEnd; DWORD dwPos; // EM_SETSEL is special - We can often handle it without having to // get the client text, which is good since some clients are stupid // and return bogus data. // If the start and end positions are both either 0 or -1, then we // don't need to do adjustment since 0 is always 0 and -1 is always -1. if (Msg == EM_SETSEL) { if ((wParam == 0 || (DWORD)wParam == 0xFFFFFFFF) && (lParam == 0 || (DWORD)lParam == 0xFFFFFFFF)) return SendMessageA(hWnd, Msg, wParam, lParam); } // Get the current window text, since we will be studying it CStrA sz(GetWindowTextLengthA(hWnd) + 1); GetWindowTextA(hWnd, sz, sz.bufsize()); switch (Msg) { case EM_GETSEL: { DWORD_PTR dwPos; dwPos = SendMessageA(hWnd, Msg, wParam, lParam); wStart = (WORD)AdjustECPosition(sz, GET_X_LPARAM(dwPos), ADJUST_TO_WCHAR_POS); wEnd = (WORD)AdjustECPosition(sz, GET_Y_LPARAM(dwPos), ADJUST_TO_WCHAR_POS); return MAKELONG(wStart, wEnd); } case EM_SETSEL: wStart = (WORD)AdjustECPosition(sz, wParam, ADJUST_TO_CHAR_POS); wEnd = (WORD)AdjustECPosition(sz, lParam, ADJUST_TO_CHAR_POS); return SendMessageA(hWnd, Msg, wStart, wEnd); case EM_LINEINDEX: dwPos = SendMessageA(hWnd, Msg, wParam, lParam); return AdjustECPosition(sz, dwPos, ADJUST_TO_WCHAR_POS); case EM_LINELENGTH: wStart = (WORD)AdjustECPosition(sz, wParam, ADJUST_TO_CHAR_POS); dwPos = SendMessageA(hWnd, Msg, wStart, lParam); return AdjustECPosition(sz + wStart, dwPos, ADJUST_TO_WCHAR_POS); case EM_LINEFROMCHAR: case EM_POSFROMCHAR: wStart = (WORD)AdjustECPosition(sz, wParam, ADJUST_TO_CHAR_POS); return SendMessageA(hWnd, Msg, wStart, lParam); default: AssertMsg(FALSE, TEXT("error: unknown message leaked into SendEditMessageWrap")); return SendMessageA(hWnd, Msg, wParam, lParam); } } #ifndef UNIX #define SHLWAPI_SENDMESSAGEWRAPW_ORD 136 #else #define SHLWAPI_SENDMESSAGEWRAPW_ORD "SendMessageWrapW" #endif typedef LRESULT(*PFNSENDMESSAGEWRAPW)(HWND, UINT, WPARAM, LPARAM); LRESULT SendMessageWrap( HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam) { // For XCP PlugUI: // 1)If shlwapi is in memory, ask it to do the work, otherwise let it // fall through to original comctl32 wrap implementation. // 2)This implementation only apply to LB for fixing #67837 switch (Msg) { case LB_ADDSTRING: case LB_FINDSTRING: case LB_FINDSTRINGEXACT: case LB_INSERTSTRING: case LB_GETTEXT: case LB_GETTEXTLEN: case LB_SELECTSTRING: { extern HMODULE GetShlwapiHModule(); PFNSENDMESSAGEWRAPW pfnSndMsgWrapW = NULL; HMODULE hShlwapi; hShlwapi = GetShlwapiHModule(); if (hShlwapi) pfnSndMsgWrapW = (PFNSENDMESSAGEWRAPW)GetProcAddress(hShlwapi, (LPCSTR)SHLWAPI_SENDMESSAGEWRAPW_ORD); if (pfnSndMsgWrapW) return pfnSndMsgWrapW(hWnd, Msg, wParam, lParam); else break; // fall through the regular comctl32's wrap } } // original comctl32's wrap implementation CHAR sz[MAX_PATH]; // BUGBUG: It's big enough current comctl32 usage until now ... switch (Msg) { case WM_GETTEXT: { CStrOut str((LPWSTR)lParam, (int)wParam); SendMessageA(hWnd, Msg, (WPARAM)str.BufSize(), (LPARAM)(LPSTR)str); return str.ConvertExcludingNul(); } // The sz[] buffer is not large enough for these guys, so use a // separate helper function. case EM_GETSEL: case EM_SETSEL: case EM_LINEINDEX: case EM_LINELENGTH: case EM_LINEFROMCHAR: case EM_POSFROMCHAR: return SendEditMessageWrap(hWnd, Msg, wParam, lParam); // BUGBUG raymondc - This is wrong. EM_GETLIMITTEXT returns the number // of characters, not bytes. But the only place we use it is in our // IME composition code, and maybe they really meant to divide by two... case EM_GETLIMITTEXT: return SendMessageA(hWnd, Msg, wParam, lParam) / sizeof(WCHAR); case EM_GETLINE: { LRESULT nLen; CStrOut str((LPWSTR)lParam, (*(SHORT*)lParam) + 1); *(SHORT*)(LPSTR)str = *(SHORT*)lParam; nLen = SendMessageA(hWnd, Msg, (WPARAM)wParam, (LPARAM)(LPSTR)str); if (nLen > 0) ((LPSTR)str)[nLen] = '\0'; return nLen; } // BUGBUG: Always assume lParam points structure, not string buffer case CB_INSERTSTRING: { return SendMessageA(hWnd, Msg, wParam, (LPARAM)lParam); } case WM_SETTEXT: case LB_ADDSTRING: case CB_ADDSTRING: case EM_REPLACESEL: ASSERT(wParam == 0 && "wParam should be 0 for these messages"); // fall through case CB_SELECTSTRING: case CB_FINDSTRINGEXACT: case CB_FINDSTRING: case LB_INSERTSTRING: case LB_FINDSTRINGEXACT: { CStrIn str((LPWSTR)lParam); return SendMessageA(hWnd, Msg, wParam, (LPARAM)(LPSTR)str); } case LB_GETTEXTLEN: case CB_GETLBTEXTLEN: ASSERT((LB_GETTEXTLEN - LB_GETTEXT) == (CB_GETLBTEXTLEN - CB_GETLBTEXT)); lParam = (LPARAM)sz; // use temp buffer Msg -= (LB_GETTEXTLEN - LB_GETTEXT); // fall through ... case LB_GETTEXT: case CB_GETLBTEXT: { CStrOut str((LPWSTR)lParam, 255); SendMessageA(hWnd, Msg, wParam, (LPARAM)(LPSTR)str); return str.ConvertExcludingNul(); } case EM_SETPASSWORDCHAR: { WPARAM wp; ASSERT(HIWORD64(wParam) == 0); MbcsFromUnicode((LPSTR)&wp, sizeof(wp), (LPWSTR)&wParam); ASSERT(HIWORD64(wp) == 0); return SendMessageA(hWnd, Msg, wp, lParam); } default: return SendMessageA(hWnd, Msg, wParam, lParam); } } BOOL SendNotifyMessageWrap(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam) { // BUGBUG: Should we use SendMessageWarp like SendDlgItemMessageWrap above? return SendNotifyMessageA(hWnd, Msg, wParam, lParam); } BOOL SetCurrentDirectoryWrap(LPCWSTR lpszCurDir) { CStrIn str(lpszCurDir); return SetCurrentDirectoryA(str); } BOOL SetDlgItemTextWrap(HWND hDlg, int nIDDlgItem, LPCWSTR lpString) { CStrIn str(lpString); return SetDlgItemTextA(hDlg, nIDDlgItem, str); } BOOL SetMenuItemInfoWrap( HMENU hMenu, UINT uItem, BOOL fByPosition, LPCMENUITEMINFOW lpmiiW) { BOOL fRet; ASSERT(sizeof(MENUITEMINFOW) == sizeof(MENUITEMINFOA) && FIELD_OFFSET(MENUITEMINFOW, dwTypeData) == FIELD_OFFSET(MENUITEMINFOA, dwTypeData)); if ((MIIM_TYPE & lpmiiW->fMask) && 0 == (lpmiiW->fType & (MFT_BITMAP | MFT_SEPARATOR))) { MENUITEMINFOA miiA; CStrIn str(lpmiiW->dwTypeData, lpmiiW->cch); memcpy(&miiA, lpmiiW, sizeof(MENUITEMINFOA)); miiA.dwTypeData = str; miiA.cch = str.strlen(); fRet = SetMenuItemInfoA(hMenu, uItem, fByPosition, &miiA); } else { fRet = SetMenuItemInfoA(hMenu, uItem, fByPosition, (LPCMENUITEMINFOA)lpmiiW); } return fRet; } BOOL SetPropWrap( HWND hWnd, LPCWSTR lpString, HANDLE hData) { CStrIn str(lpString); return SetPropA(hWnd, str, hData); } LONG SetWindowLongWrap(HWND hWnd, int nIndex, LONG dwNewLong) { return SetWindowLongA(hWnd, nIndex, dwNewLong); } HHOOK SetWindowsHookExWrap(int idHook, HOOKPROC lpfn, HINSTANCE hmod, DWORD dwThreadId) { return SetWindowsHookExA(idHook, lpfn, hmod, dwThreadId); } BOOL SetWindowTextWrap(HWND hWnd, LPCWSTR lpString) { CStrIn str(lpString); return SetWindowTextA(hWnd, str); } BOOL SystemParametersInfoWrap( UINT uiAction, UINT uiParam, PVOID pvParam, UINT fWinIni) { BOOL ret; char ach[LF_FACESIZE]; if (uiAction == SPI_SETDESKWALLPAPER) { CStrIn str((LPCWSTR)pvParam); ret = SystemParametersInfoA( uiAction, uiParam, str, fWinIni); } else if (uiAction == SPI_GETNONCLIENTMETRICS) { NONCLIENTMETRICSA ncmA; NONCLIENTMETRICS* pncm = (NONCLIENTMETRICS*)pvParam; ASSERT(uiParam == sizeof(NONCLIENTMETRICS) && pncm->cbSize == sizeof(NONCLIENTMETRICS)); ncmA.cbSize = sizeof(ncmA); ret = SystemParametersInfoA( uiAction, sizeof(ncmA), &ncmA, fWinIni); pncm->iBorderWidth = ncmA.iBorderWidth; pncm->iScrollWidth = ncmA.iScrollWidth; pncm->iScrollHeight = ncmA.iScrollHeight; pncm->iCaptionWidth = ncmA.iCaptionWidth; pncm->iCaptionHeight = ncmA.iCaptionHeight; pncm->iSmCaptionWidth = ncmA.iSmCaptionWidth; pncm->iSmCaptionHeight = ncmA.iSmCaptionHeight; pncm->iMenuWidth = ncmA.iMenuWidth; pncm->iMenuHeight = ncmA.iMenuHeight; memcpy(&pncm->lfCaptionFont, &ncmA.lfCaptionFont, FIELD_OFFSET(LOGFONTW, lfFaceName)); UnicodeFromMbcs(pncm->lfCaptionFont.lfFaceName, ARRAYSIZE(pncm->lfCaptionFont.lfFaceName), ncmA.lfCaptionFont.lfFaceName); memcpy(&pncm->lfSmCaptionFont, &ncmA.lfSmCaptionFont, FIELD_OFFSET(LOGFONTW, lfFaceName)); UnicodeFromMbcs(pncm->lfSmCaptionFont.lfFaceName, ARRAYSIZE(pncm->lfSmCaptionFont.lfFaceName), ncmA.lfSmCaptionFont.lfFaceName); memcpy(&pncm->lfMenuFont, &ncmA.lfMenuFont, FIELD_OFFSET(LOGFONTW, lfFaceName)); UnicodeFromMbcs(pncm->lfMenuFont.lfFaceName, ARRAYSIZE(pncm->lfMenuFont.lfFaceName), ncmA.lfMenuFont.lfFaceName); memcpy(&pncm->lfStatusFont, &ncmA.lfStatusFont, FIELD_OFFSET(LOGFONTW, lfFaceName)); UnicodeFromMbcs(pncm->lfStatusFont.lfFaceName, ARRAYSIZE(pncm->lfStatusFont.lfFaceName), ncmA.lfStatusFont.lfFaceName); memcpy(&pncm->lfMessageFont, &ncmA.lfMessageFont, FIELD_OFFSET(LOGFONTW, lfFaceName)); UnicodeFromMbcs(pncm->lfMessageFont.lfFaceName, ARRAYSIZE(pncm->lfMessageFont.lfFaceName), ncmA.lfMessageFont.lfFaceName); } else ret = SystemParametersInfoA( uiAction, uiParam, pvParam, fWinIni); if ((uiAction == SPI_GETICONTITLELOGFONT) && ret) { strcpy(ach, ((LPLOGFONTA)pvParam)->lfFaceName); UnicodeFromMbcs( ((LPLOGFONTW)pvParam)->lfFaceName, ARRAYSIZE(((LPLOGFONTW)pvParam)->lfFaceName), ach); } return ret; } #ifndef FONT_LINK BOOL TextOutWrap(HDC hdc, int x, int y, LPCWSTR lpStr, int cb) { if (g_fMEEnabled && !g_bRunOnMemphis) { CStrIn str(lpStr); return TextOutA(hdc, x, y, str, str.strlen()); } else return TextOutW(hdc, x, y, lpStr, cb); } #endif int TranslateAcceleratorWrap(HWND hWnd, HACCEL hAccTable, LPMSG lpMsg) { return TranslateAcceleratorA(hWnd, hAccTable, lpMsg); } BOOL UnregisterClassWrap(LPCWSTR lpClassName, HINSTANCE hInstance) { CStrIn str(lpClassName); return UnregisterClassA(str, hInstance); } SHORT VkKeyScanWrap(WCHAR ch) { CStrIn str(&ch, 1); return VkKeyScanA(*(char*)str); } BOOL WinHelpWrap(HWND hwnd, LPCWSTR szFile, UINT uCmd, DWORD dwData) { CStrIn str(szFile); return WinHelpA(hwnd, str, uCmd, dwData); } #define DBCS_CHARSIZE (2) // N versions of wsprintf and wvsprintf which take an output buffer size to prevent overflow // bugs. Taken from the NT wsprintf source code. // _MBToWCS and _WCSToMB are actually macros which call ntrtl functions in the NT version. int _MBToWCS(LPCSTR pszIn, int cchIn, LPWSTR* ppwszOut) { int cch = 0; int cbAlloc; if ((0 != cchIn) && (NULL != ppwszOut)) { cchIn++; cbAlloc = cchIn * sizeof(WCHAR); *ppwszOut = (LPWSTR)LocalAlloc(LMEM_FIXED, cbAlloc); if (NULL != *ppwszOut) { cch = MultiByteToWideChar(CP_ACP, 0, pszIn, cchIn, *ppwszOut, cchIn); if (!cch) { LocalFree(*ppwszOut); *ppwszOut = NULL; } else { cch--; // Just return the number of characters } } } return cch; } /** Module Header **\ * Module Name: wsprintf.c * * Copyright (c) 1985-91, Microsoft Corporation * sprintf.c * * Implements Windows friendly versions of sprintf and vsprintf * * History: * 2-15-89 craigc Initial * 11-12-90 MikeHar Ported from windows 3 \**/ /* Max number of characters. Doesn't include termination character */ #define out(c) if (cchLimit) {*lpOut++=(c); cchLimit--;} else goto errorout /* * SP_GetFmtValueW * * reads a width or precision value from the format string * * History: * 11-12-90 MikeHar Ported from windows 3 * 07-27-92 GregoryW Created Unicode version (copied from SP_GetFmtValue) \**/ LPCWSTR SP_GetFmtValueW( LPCWSTR lpch, int* lpw) { int ii = 0; /* It might not work for some locales or digit sets */ while (*lpch >= L'0' && *lpch <= L'9') { ii *= 10; ii += (int)(*lpch - L'0'); lpch++; } *lpw = ii; /* * return the address of the first non-digit character */ return lpch; } /* * SP_PutNumberW * * Takes an unsigned long integer and places it into a buffer, respecting * a buffer limit, a radix, and a case select (upper or lower, for hex). * * * History: * 11-12-90 MikeHar Ported from windows 3 asm --> C * 12-11-90 GregoryW need to increment lpstr after assignment of mod * 02-11-92 GregoryW temporary version until we have C runtime support \**/ int SP_PutNumberW( LPWSTR lpstr, DWORD n, int limit, DWORD radix, int uppercase) { DWORD mod; int count = 0; /* It might not work for some locales or digit sets */ if (uppercase) uppercase = 'A' - '0' - 10; else uppercase = 'a' - '0' - 10; if (count < limit) { do { mod = n % radix; n /= radix; mod += '0'; if (mod > '9') mod += uppercase; *lpstr++ = (WCHAR)mod; count++; } while ((count < limit) && n); } return count; } /* * SP_ReverseW * * reverses a string in place * * History: * 11-12-90 MikeHar Ported from windows 3 asm --> C * 12-11-90 GregoryW fixed boundary conditions; removed count * 02-11-92 GregoryW temporary version until we have C runtime support \**/ void SP_ReverseW( LPWSTR lpFirst, LPWSTR lpLast) { WCHAR ch; while (lpLast > lpFirst) { ch = *lpFirst; *lpFirst++ = *lpLast; *lpLast-- = ch; } } /* * wvsprintfW (API) * * wsprintfW() calls this function. * * History: * 11-Feb-1992 GregoryW copied xwvsprintf * Temporary hack until we have C runtime support * 1-22-97 tnoonan Converted to wvnsprintfW \**/ int wvnsprintfW( LPWSTR lpOut, int cchLimitIn, LPCWSTR lpFmt, va_list arglist) { BOOL fAllocateMem = FALSE; WCHAR prefix, fillch; int left, width, prec, size, sign, radix, upper, hprefix; int cchLimit = --cchLimitIn, cch; LPWSTR lpT, lpTWC; LPCSTR psz; va_list varglist = arglist; union { long l; unsigned long ul; char sz[2]; WCHAR wsz[2]; } val; if (cchLimit < 0) return 0; while (*lpFmt != 0) { if (*lpFmt == L'%') { /* * read the flags. These can be in any order */ left = 0; prefix = 0; while (*++lpFmt) { if (*lpFmt == L'-') left++; else if (*lpFmt == L'#') prefix++; else break; } /* * find fill character */ if (*lpFmt == L'0') { fillch = L'0'; lpFmt++; } else fillch = L' '; /* * read the width specification */ lpFmt = SP_GetFmtValueW(lpFmt, &cch); width = cch; /* * read the precision */ if (*lpFmt == L'.') { lpFmt = SP_GetFmtValueW(++lpFmt, &cch); prec = cch; } else prec = -1; /* * get the operand size * default size: size == 0 * long number: size == 1 * wide chars: size == 2 * It may be a good idea to check the value of size when it * is tested for non-zero below (IanJa) */ hprefix = 0; if ((*lpFmt == L'w') || (*lpFmt == L't')) { size = 2; lpFmt++; } else if (*lpFmt == L'l') { size = 1; lpFmt++; } else { size = 0; if (*lpFmt == L'h') { lpFmt++; hprefix = 1; } } upper = 0; sign = 0; radix = 10; switch (*lpFmt) { case 0: goto errorout; case L'i': case L'd': size = 1; sign++; /** FALL THROUGH to case 'u' **/ case L'u': /* turn off prefix if decimal */ prefix = 0; donumeric: /* special cases to act like MSC v5.10 */ if (left || prec >= 0) fillch = L' '; /* * if size == 1, "%lu" was specified (good) * if size == 2, "%wu" was specified (bad) */ if (size) { val.l = va_arg(varglist, LONG); } else if (sign) { val.l = va_arg(varglist, SHORT); } else { val.ul = va_arg(varglist, unsigned); } if (sign && val.l < 0L) val.l = -val.l; else sign = 0; lpT = lpOut; /* * blast the number backwards into the user buffer */ cch = SP_PutNumberW(lpOut, val.l, cchLimit, radix, upper); if (!(cchLimit -= cch)) goto errorout; lpOut += cch; width -= cch; prec -= cch; if (prec > 0) width -= prec; /* * fill to the field precision */ while (prec-- > 0) out(L'0'); if (width > 0 && !left) { /* * if we're filling with spaces, put sign first */ if (fillch != L'0') { if (sign) { sign = 0; out(L'-'); width--; } if (prefix) { out(prefix); out(L'0'); prefix = 0; } } if (sign) width--; /* * fill to the field width */ while (width-- > 0) out(fillch); /* * still have a sign? */ if (sign) out(L'-'); if (prefix) { out(prefix); out(L'0'); } /* * now reverse the string in place */ SP_ReverseW(lpT, lpOut - 1); } else { /* * add the sign character */ if (sign) { out(L'-'); width--; } if (prefix) { out(prefix); out(L'0'); } /* * reverse the string in place */ SP_ReverseW(lpT, lpOut - 1); /* * pad to the right of the string in case left aligned */ while (width-- > 0) out(fillch); } break; case L'X': upper++; /** FALL THROUGH to case 'x' **/ case L'x': radix = 16; if (prefix) if (upper) prefix = L'X'; else prefix = L'x'; goto donumeric; case L'c': case L'C': if (!size && !hprefix) { size = 1; // force WCHAR } /** FALL THROUGH to case 'C' **/ /* * if size == 0, "%C" or "%hc" was specified (CHAR) * if size == 1, "%c" or "%lc" was specified (WCHAR) * if size == 2, "%wc" or "%tc" was specified (WCHAR) */ cch = 1; /* One character must be copied to the output buffer */ if (size) { val.wsz[0] = va_arg(varglist, WCHAR); val.wsz[1] = 0; lpT = val.wsz; goto putwstring; } else { val.sz[0] = va_arg(varglist, CHAR); val.sz[1] = 0; psz = val.sz; goto putstring; } case L's': case L'S': if (!size && !hprefix) { size = 1; // force LPWSTR } /** FALL THROUGH to case 'S' **/ /* * if size == 0, "%S" or "%hs" was specified (LPSTR) * if size == 1, "%s" or "%ls" was specified (LPWSTR) * if size == 2, "%ws" or "%ts" was specified (LPWSTR) */ if (size) { lpT = va_arg(varglist, LPWSTR); cch = lstrlenW(lpT); // Win95 supports lstrlenW! } else { psz = va_arg(varglist, LPSTR); cch = lstrlenA(psz); putstring: cch = _MBToWCS(psz, cch, &lpTWC); fAllocateMem = (BOOL)cch; lpT = lpTWC; } putwstring: if (prec >= 0 && cch > prec) cch = prec; width -= cch; if (left) { while (cch--) out(*lpT++); while (width-- > 0) out(fillch); } else { while (width-- > 0) out(fillch); while (cch--) out(*lpT++); } if (fAllocateMem) { LocalFree(lpTWC); fAllocateMem = FALSE; } break; default: normalch: out((WCHAR)*lpFmt); break; } /* END OF SWITCH(*lpFmt) */ } /* END OF IF(%) */ else goto normalch; /* character not a '%', just do it */ /* * advance to next format string character */ lpFmt++; } /* END OF OUTER WHILE LOOP */ errorout: *lpOut = 0; if (fAllocateMem) { LocalFree(lpTWC); } return cchLimitIn - cchLimit; } LWSTDAPIV_(int) wnsprintfW( LPWSTR lpOut, int cchLimitIn, LPCWSTR lpFmt, ...) { va_list arglist; int ret; va_start(arglist, lpFmt); ret = wvnsprintfW(lpOut, cchLimitIn, lpFmt, arglist); va_end(arglist); return ret; } LWSTDAPIV_(int) wsprintfW( LPWSTR lpOut, LPCWSTR lpFmt, ...) { // unsafe printf. arbitrary max of 0x10000 length va_list arglist; int ret; va_start(arglist, lpFmt); ret = wvnsprintfW(lpOut, 0x10000, lpFmt, arglist); va_end(arglist); return ret; } // StartDoc int StartDocWrap(HDC hDC, const DOCINFO* lpdi) { CStrIn strDocName(lpdi->lpszDocName); CStrIn strOutput(lpdi->lpszOutput); CStrIn strDatatype(lpdi->lpszDatatype); DOCINFOA dia; dia.cbSize = sizeof(DOCINFO); dia.lpszDocName = strDocName; dia.lpszOutput = strOutput; dia.lpszDatatype = strDatatype; dia.fwType = lpdi->fwType; return StartDocA(hDC, &dia); } #endif // !WINNT // Plug UI support with SHLWAPI typedef HRESULT(*PFNDLLGETVERSION)(DLLVERSIONINFO* pinfo); HMODULE GetShlwapiHModule() { HMODULE hShlwapi = GetModuleHandle(TEXT("SHLWAPI")); if (hShlwapi) { PFNDLLGETVERSION pfnDllGetVersion = (PFNDLLGETVERSION)GetProcAddress(hShlwapi, "DllGetVersion"); if (pfnDllGetVersion) { DLLVERSIONINFO dllinfo; dllinfo.cbSize = sizeof(DLLVERSIONINFO); if (pfnDllGetVersion(&dllinfo) == NOERROR) { if (dllinfo.dwMajorVersion < 5) { // This guy doesn't support ML functions hShlwapi = NULL; } } } } return hShlwapi; } // First, we need access to some helper functions: #ifndef UNIX #define SHLWAPIMLISMLHINSTANCE_ORD 429 #else #define SHLWAPIMLISMLHINSTANCE_ORD "MLIsMLHInstance" #endif typedef BOOL(*PFNMLISMLHINSTANCE)(HINSTANCE); BOOL MLIsMLHInstanceWrap(HINSTANCE hInst) { HMODULE hShlwapi = GetShlwapiHModule(); if (hShlwapi) { PFNMLISMLHINSTANCE pfn; pfn = (PFNMLISMLHINSTANCE)GetProcAddress(hShlwapi, (LPCSTR)SHLWAPIMLISMLHINSTANCE_ORD); if (pfn) return pfn(hInst); } // BUGBUG: What if an app told comctl32 to be PlugUI and we picked // a resource that cannot be displayed on Win9x without // shlwapi's font linking? Seems like we need to foricbly // load shlwapi in that case... // No shlwapi? then this can't be an ML hinstance. return FALSE; } #ifndef UNIX #define SHLWAPIMLSETMLHINSTANCE_ORD 430 #else #define SHLWAPIMLSETMLHINSTANCE_ORD "MLSetMLHInstance" #endif typedef HRESULT(*PFNMLSETMLHINSTANCE)(HINSTANCE, LANGID); HRESULT MLSetMLHInstanceWrap(HINSTANCE hInst, LANGID lidUI) { HMODULE hShlwapi; PFNMLSETMLHINSTANCE pfnMLSet = NULL; hShlwapi = GetShlwapiHModule(); if (hShlwapi) { pfnMLSet = (PFNMLSETMLHINSTANCE)GetProcAddress(hShlwapi, (LPCSTR)SHLWAPIMLSETMLHINSTANCE_ORD); if (pfnMLSet) return pfnMLSet(hInst, lidUI); } return E_FAIL; } #ifndef UNIX #define SHLWAPIMLCLEARMLHINSTANCE_ORD 431 #else #define SHLWAPIMLCLEARMLHINSTANCE_ORD "MLClearMLHInstance" #endif typedef HRESULT(*PFNMLCLEARMLHINSTANCE)(HINSTANCE); HRESULT MLClearMLHinstanceWrap(HINSTANCE hInst) { HMODULE hShlwapi; PFNMLCLEARMLHINSTANCE pfnMLClear = NULL; hShlwapi = GetShlwapiHModule(); if (hShlwapi) { pfnMLClear = (PFNMLCLEARMLHINSTANCE)GetProcAddress(hShlwapi, (LPCSTR)SHLWAPIMLCLEARMLHINSTANCE_ORD); if (pfnMLClear) return pfnMLClear(hInst); } return E_FAIL; } // And now, when shlwapi is around we delegate to it's ML-enabled implementations: // Make sure we get the real USER32 versions. #undef CreateDialogIndirectParamW #ifndef UNIX #define SHLWAPICREATEDIALOGINDIRECTPARAM_ORD 393 #else #define SHLWAPICREATEDIALOGINDIRECTPARAM_ORD "CreateDialogIndirectParamWrapW" #endif typedef HWND(*PFNCREATEDIALOGINDIRECTPARAM)(HINSTANCE, LPCDLGTEMPLATE, HWND, DLGPROC, LPARAM); HWND CreateDialogIndirectParamWrap( HINSTANCE hInstance, LPCDLGTEMPLATE lpTemplate, HWND hWndParent, DLGPROC lpDialogFunc, LPARAM dwInitParam) { HMODULE hShlwapi; PFNCREATEDIALOGINDIRECTPARAM pfnCDIP = NULL; HWND hwndRet; // If shlwapi is in memory, ask it to create the dialog, // as then we get ML dialogs on downlevel platforms (if // the hInstance is from MLLoadLibrary -- otherwise it // thunks to the real A/W api for us). hShlwapi = GetShlwapiHModule(); if (hShlwapi) { pfnCDIP = (PFNCREATEDIALOGINDIRECTPARAM)GetProcAddress(hShlwapi, (LPCSTR)SHLWAPICREATEDIALOGINDIRECTPARAM_ORD); } if (!pfnCDIP) { if (g_bRunOnNT) pfnCDIP = CreateDialogIndirectParamW; else pfnCDIP = CreateDialogIndirectParamA; } // If this is from comctl32, assume it was loaded via the MUI-Language if (HINST_THISDLL == hInstance) MLSetMLHInstanceWrap(hInstance, GetMUILanguage()); hwndRet = pfnCDIP(hInstance, lpTemplate, hWndParent, lpDialogFunc, dwInitParam); if (HINST_THISDLL == hInstance) MLClearMLHinstanceWrap(hInstance); return(hwndRet); } // Make sure we get the real USER32 versions. #undef DialogBoxIndirectParamW #ifndef UNIX #define SHLWAPIDIALOGBOXINDIRECTPARAM_ORD 58 #else #define SHLWAPIDIALOGBOXINDIRECTPARAM_ORD "DialogBoxIndirectParamWrapW" #endif typedef INT_PTR(*PFNDIALOGBOXINDIRECTPARAM)(HINSTANCE, LPCDLGTEMPLATE, HWND, DLGPROC, LPARAM); INT_PTR DialogBoxIndirectParamWrap( HINSTANCE hInstance, LPCDLGTEMPLATEW hDialogTemplate, HWND hWndParent, DLGPROC lpDialogFunc, LPARAM dwInitParam) { HMODULE hShlwapi; INT_PTR iRet; PFNDIALOGBOXINDIRECTPARAM pfnDBIP = NULL; // If shlwapi is in memory, ask it to create the dialog, // as then we get ML dialogs on downlevel platforms (if // the hInstance is from MLLoadLibrary -- otherwise it // thunks to the real A/W api for us). hShlwapi = GetShlwapiHModule(); if (hShlwapi) pfnDBIP = (PFNDIALOGBOXINDIRECTPARAM)GetProcAddress(hShlwapi, (LPCSTR)SHLWAPIDIALOGBOXINDIRECTPARAM_ORD); if (!pfnDBIP) { if (g_bRunOnNT) pfnDBIP = DialogBoxIndirectParamW; else pfnDBIP = DialogBoxIndirectParamA; } // If this is from comctl32, assume it was loaded via the MUI-Language if (HINST_THISDLL == hInstance) MLSetMLHInstanceWrap(hInstance, GetMUILanguage()); iRet = pfnDBIP(hInstance, hDialogTemplate, hWndParent, lpDialogFunc, dwInitParam); if (HINST_THISDLL == hInstance) MLClearMLHinstanceWrap(hInstance); return iRet; } #endif // UNICODE