2069 lines
37 KiB
C++
2069 lines
37 KiB
C++
//+---------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1993 - 1993.
|
|
//
|
|
// File: widewrap.cxx
|
|
//
|
|
// Contents: Unicode wrapper API, used only on Chicago
|
|
//
|
|
// Functions: About fifty Win32 function wrappers
|
|
//
|
|
// Notes: 'sz' is used instead of the "correct" hungarian 'psz'
|
|
// throughout to enhance readability.
|
|
//
|
|
// Not all of every Win32 function is wrapped here. Some
|
|
// obscurely-documented features may not be handled correctly
|
|
// in these wrappers. Caller beware.
|
|
//
|
|
// These are privately exported for use by the Shell.
|
|
//
|
|
// History: 28-Dec-93 ErikGav Created
|
|
// 06-14-94 KentCe Various Chicago build fixes.
|
|
// 21-Dec-94 BruceMa Use olewcstombs + other fixes
|
|
// 21-Feb-95 BruceMa Add support for AreFileApisANSI
|
|
// 29-Feb-96 JeffE Add lots of wide character rtns
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
#include <ole2int.h>
|
|
#include <winbase.h>
|
|
#include "memapi.hxx"
|
|
|
|
#ifdef _CHICAGO_
|
|
|
|
#define HFINDFILE HANDLE
|
|
#define ERR ((char*) -1)
|
|
|
|
|
|
//
|
|
// BUGBUG: 9869
|
|
//
|
|
// The length of a Unicode string (in chars) and a DBCS string are not
|
|
// always equal. We need to review all WideChar to MultiByte conversions
|
|
// logic to verify that the proper result buffer size is used.
|
|
//
|
|
// Make the below Win95 only change to get the Win95 FE build out.
|
|
//
|
|
|
|
int UnicodeToAnsi(LPSTR sz, LPCWSTR pwsz, LONG cb)
|
|
{
|
|
int ret;
|
|
|
|
ret = WideCharToMultiByte(CP_ACP, WC_COMPOSITECHECK, pwsz, -1, sz, cb, NULL, NULL);
|
|
|
|
#if DBG==1
|
|
CairoleAssert(ret != 0 && "Lost characters in Unicode->Ansi conversion");
|
|
if (ret == 0)
|
|
{
|
|
DebugBreak();
|
|
}
|
|
#endif
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
int UnicodeToAnsiOem(LPSTR sz, LPCWSTR pwsz, LONG cb)
|
|
{
|
|
int ret;
|
|
|
|
if (AreFileApisANSI())
|
|
{
|
|
ret = WideCharToMultiByte(CP_ACP, WC_COMPOSITECHECK, pwsz, -1, sz,
|
|
cb, NULL, NULL);
|
|
}
|
|
else
|
|
{
|
|
ret = WideCharToMultiByte(CP_OEMCP, WC_COMPOSITECHECK, pwsz, -1, sz,
|
|
cb, NULL, NULL);
|
|
}
|
|
|
|
#if DBG==1
|
|
CairoleAssert(ret != 0 && "Lost characters in Unicode->Ansi conversion");
|
|
if (ret == 0)
|
|
{
|
|
DebugBreak();
|
|
}
|
|
#endif
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
|
|
#if DBG==1
|
|
int AnsiToUnicode(LPWSTR pwsz, LPCSTR sz, LONG cb)
|
|
{
|
|
int ret;
|
|
DWORD WindowsError;
|
|
|
|
ret = MultiByteToWideChar (CP_ACP, MB_PRECOMPOSED, sz, -1, pwsz, cb);
|
|
if (ret == 0) {
|
|
WindowsError = GetLastError ();
|
|
}
|
|
CairoleAssert(ret != 0 && "Lost characters in Ansi->Unicode conversion");
|
|
if (ret == 0)
|
|
{
|
|
DebugBreak();
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
#else
|
|
#define AnsiToUnicode(pwsz,sz,cb) MultiByteToWideChar (CP_ACP, MB_PRECOMPOSED, sz, -1, pwsz, cb)
|
|
#endif
|
|
|
|
|
|
|
|
int AnsiToUnicodeOem(LPWSTR pwsz, LPCSTR sz, LONG cb)
|
|
{
|
|
int ret;
|
|
|
|
if (AreFileApisANSI())
|
|
{
|
|
ret = MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, sz, cb, pwsz, cb);
|
|
}
|
|
else
|
|
{
|
|
ret = MultiByteToWideChar(CP_OEMCP, MB_PRECOMPOSED, sz, cb, pwsz, cb);
|
|
}
|
|
|
|
#if DBG==1
|
|
CairoleAssert(ret != 0 && "Lost characters in Ansi->Unicode conversion");
|
|
if (ret == 0)
|
|
{
|
|
DebugBreak();
|
|
}
|
|
#endif
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
|
|
|
|
LPSTR Convert(LPCWSTR pwsz)
|
|
{
|
|
LONG len;
|
|
LPSTR sz = NULL;
|
|
|
|
if (pwsz == NULL)
|
|
goto Exit;
|
|
|
|
#if DBG==1
|
|
// some Win32 API accept atoms in their string parameters
|
|
CairoleAssert(HIWORD(pwsz)!=0 && "ATOM passed to Convert -- GP fault coming");
|
|
#endif
|
|
|
|
len = (lstrlenW(pwsz) + 1) * 2;
|
|
|
|
sz = new CHAR[len];
|
|
if (sz==NULL)
|
|
{
|
|
sz = ERR;
|
|
goto Exit;
|
|
}
|
|
|
|
__try
|
|
{
|
|
UnicodeToAnsi(sz, pwsz, len);
|
|
}
|
|
__except(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
#if DBG==1
|
|
MessageBoxA(NULL, "GP fault in unicode conversion -- caught",
|
|
NULL, MB_OK);
|
|
#endif
|
|
if (sz)
|
|
delete sz;
|
|
sz = ERR;
|
|
}
|
|
|
|
Exit:
|
|
return sz;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
LPSTR ConvertOem(LPCWSTR pwsz)
|
|
{
|
|
LONG len;
|
|
LPSTR sz = NULL;
|
|
|
|
if (pwsz == NULL)
|
|
goto Exit;
|
|
|
|
#if DBG==1
|
|
// some Win32 API accept atoms in their string parameters
|
|
CairoleAssert(HIWORD(pwsz)!=0 && "ATOM passed to Convert -- GP fault coming");
|
|
#endif
|
|
|
|
len = (lstrlenW(pwsz) + 1) * 2;
|
|
|
|
sz = new CHAR[len];
|
|
if (sz==NULL)
|
|
{
|
|
sz = ERR;
|
|
goto Exit;
|
|
}
|
|
|
|
__try
|
|
{
|
|
UnicodeToAnsiOem(sz, pwsz, len);
|
|
}
|
|
__except(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
#if DBG==1
|
|
MessageBoxA(NULL, "GP fault in unicode conversion -- caught",
|
|
NULL, MB_OK);
|
|
#endif
|
|
if (sz)
|
|
delete sz;
|
|
sz = ERR;
|
|
}
|
|
|
|
Exit:
|
|
return sz;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HANDLE WINAPI CreateFileX(LPCWSTR pwsz, DWORD fdwAccess, DWORD fdwShareMask,
|
|
LPSECURITY_ATTRIBUTES lpsa, DWORD fdwCreate, DWORD fdwAttrsAndFlags,
|
|
HANDLE hTemplateFile)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("CreateFile\n");
|
|
#endif
|
|
|
|
CHAR sz[MAX_PATH * 2];
|
|
UnicodeToAnsiOem(sz, pwsz, sizeof(sz));
|
|
|
|
return CreateFileA(sz, fdwAccess, fdwShareMask, lpsa, fdwCreate,
|
|
fdwAttrsAndFlags, hTemplateFile);
|
|
}
|
|
|
|
BOOL WINAPI DeleteFileX(LPCWSTR pwsz)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("DeleteFile\n");
|
|
#endif
|
|
|
|
CHAR sz[MAX_PATH * 2];
|
|
UnicodeToAnsi(sz, pwsz, sizeof(sz));
|
|
|
|
return DeleteFileA(sz);
|
|
}
|
|
|
|
UINT WINAPI RegisterClipboardFormatX(LPCWSTR pwszFormat)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegisterClipboardFormat\n");
|
|
#endif
|
|
|
|
UINT ret;
|
|
#if 0
|
|
LPSTR sz;
|
|
|
|
sz = Convert(pwszFormat);
|
|
if (sz == ERR)
|
|
{
|
|
return 0;
|
|
}
|
|
#else
|
|
// BUGBUG: CairOLE calls this from libmain -- have to use static buffer
|
|
|
|
CHAR sz[200];
|
|
|
|
UnicodeToAnsi(sz, pwszFormat, sizeof(sz));
|
|
#endif
|
|
|
|
ret = SSRegisterClipboardFormatA(sz);
|
|
#if 0
|
|
delete sz;
|
|
#endif
|
|
return ret;
|
|
}
|
|
|
|
int WINAPI GetClipboardFormatNameX(UINT format, LPWSTR pwsz,
|
|
int cchMaxCount)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetClipboardFormatName\n");
|
|
#endif
|
|
|
|
LPSTR sz;
|
|
int i;
|
|
|
|
sz = new CHAR[cchMaxCount];
|
|
if (sz == NULL)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
i = SSGetClipboardFormatNameA(format, sz, cchMaxCount);
|
|
|
|
if (i)
|
|
{
|
|
AnsiToUnicode(pwsz, sz, lstrlenA(sz) + 1);
|
|
}
|
|
if (sz)
|
|
delete sz;
|
|
return i;
|
|
}
|
|
|
|
LONG APIENTRY RegOpenKeyX(HKEY hKey, LPCWSTR pwszSubKey, PHKEY phkResult)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegOpenKey\n");
|
|
#endif
|
|
|
|
LONG ret;
|
|
LPSTR sz;
|
|
|
|
sz = Convert(pwszSubKey);
|
|
|
|
if (sz == ERR)
|
|
{
|
|
return ERROR_OUTOFMEMORY;
|
|
}
|
|
|
|
ret = RegOpenKeyA(hKey, sz, phkResult);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
LONG APIENTRY RegQueryValueX(HKEY hKey, LPCWSTR pwszSubKey, LPWSTR pwszValue,
|
|
PLONG lpcbValue)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegQueryValue\n");
|
|
#endif
|
|
|
|
LONG cb, ret;
|
|
LPSTR szValue = NULL;
|
|
LPSTR sz;
|
|
|
|
sz = Convert(pwszSubKey);
|
|
if (sz == ERR)
|
|
{
|
|
return ERROR_OUTOFMEMORY;
|
|
}
|
|
|
|
ret = RegQueryValueA(hKey, sz, NULL, &cb);
|
|
|
|
// If the caller was just asking for the size of the value, jump out
|
|
// now, without actually retrieving and converting the value.
|
|
|
|
if (pwszValue == NULL)
|
|
{
|
|
// Adjust size of buffer to report, to account for CHAR -> WCHAR
|
|
*lpcbValue = cb * sizeof(WCHAR);
|
|
goto Exit;
|
|
}
|
|
|
|
|
|
if (ret == ERROR_SUCCESS)
|
|
{
|
|
// If the caller was asking for the value, but allocated too small
|
|
// of a buffer, set the buffer size and jump out.
|
|
|
|
if (*lpcbValue < (LONG) (cb * sizeof(WCHAR)))
|
|
{
|
|
// Adjust size of buffer to report, to account for CHAR -> WCHAR
|
|
*lpcbValue = cb * sizeof(WCHAR);
|
|
ret = ERROR_MORE_DATA;
|
|
goto Exit;
|
|
}
|
|
|
|
// Otherwise, retrieve and convert the value.
|
|
|
|
szValue = new CHAR[cb];
|
|
if (szValue == NULL)
|
|
{
|
|
ret = ERROR_OUTOFMEMORY;
|
|
goto Exit;
|
|
}
|
|
|
|
ret = RegQueryValueA(hKey, sz, szValue, &cb);
|
|
|
|
if (ret == ERROR_SUCCESS)
|
|
{
|
|
AnsiToUnicode(pwszValue, szValue, cb);
|
|
|
|
// Adjust size of buffer to report, to account for CHAR -> WCHAR
|
|
*lpcbValue = cb * sizeof(WCHAR);
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
if (szValue)
|
|
delete szValue;
|
|
if (sz)
|
|
delete sz;
|
|
|
|
return ret;
|
|
}
|
|
|
|
LONG APIENTRY RegSetValueX(HKEY hKey, LPCWSTR lpSubKey, DWORD dwType,
|
|
LPCWSTR lpData, DWORD cbData)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegSetValue\n");
|
|
#endif
|
|
|
|
LPSTR szKey = NULL;
|
|
LPSTR szValue = NULL;
|
|
LONG ret = ERROR_OUTOFMEMORY;
|
|
|
|
szKey = Convert(lpSubKey);
|
|
if (szKey == ERR)
|
|
{
|
|
szKey = NULL;
|
|
goto Exit;
|
|
}
|
|
|
|
szValue = Convert(lpData);
|
|
if (szValue == ERR)
|
|
{
|
|
szValue = NULL;
|
|
goto Exit;
|
|
}
|
|
|
|
ret = RegSetValueA(hKey, szKey, dwType, szValue, cbData);
|
|
|
|
Exit:
|
|
if (szKey)
|
|
delete szKey;
|
|
if (szValue)
|
|
delete szValue;
|
|
return ret;
|
|
}
|
|
|
|
UINT WINAPI RegisterWindowMessageX(LPCWSTR lpString)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegisterWindowMessage\n");
|
|
#endif
|
|
|
|
UINT ret;
|
|
#if 0
|
|
LPSTR sz;
|
|
|
|
sz = Convert(lpString);
|
|
if (sz == ERR)
|
|
{
|
|
return 0;
|
|
}
|
|
#else
|
|
// BUGBUG: CairOLE calls this from libmain -- have to use static buffer
|
|
|
|
CHAR sz[200];
|
|
UnicodeToAnsi(sz, lpString, sizeof(sz));
|
|
#endif
|
|
|
|
ret = RegisterWindowMessageA(sz);
|
|
#if 0
|
|
delete sz;
|
|
#endif
|
|
return ret;
|
|
}
|
|
|
|
LONG
|
|
APIENTRY
|
|
RegOpenKeyExX (
|
|
HKEY hKey,
|
|
LPCWSTR lpSubKey,
|
|
DWORD ulOptions,
|
|
REGSAM samDesired,
|
|
PHKEY phkResult
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegOpenKeyEx\n");
|
|
#endif
|
|
|
|
LONG ret;
|
|
LPSTR sz;
|
|
|
|
sz = Convert(lpSubKey);
|
|
if (sz == ERR)
|
|
{
|
|
return ERROR_OUTOFMEMORY;
|
|
}
|
|
|
|
ret = RegOpenKeyExA(hKey, sz, ulOptions, samDesired, phkResult);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
LONG
|
|
APIENTRY
|
|
RegQueryValueExX(
|
|
HKEY hKey,
|
|
LPWSTR lpValueName,
|
|
LPDWORD lpReserved,
|
|
LPDWORD lpType,
|
|
LPBYTE lpData,
|
|
LPDWORD lpcbData
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegQueryValueEx\n");
|
|
#endif
|
|
|
|
LPBYTE lpTempBuffer;
|
|
DWORD dwTempType;
|
|
DWORD cb, cbRequired;
|
|
LONG ret;
|
|
LPSTR sz;
|
|
LPWSTR pwszTempWide;
|
|
LPSTR pszTempNarrow;
|
|
ULONG ulStringLength;
|
|
|
|
sz = Convert(lpValueName);
|
|
if (sz == ERR)
|
|
{
|
|
return ERROR_OUTOFMEMORY;
|
|
}
|
|
|
|
ret = RegQueryValueExA(hKey, sz, lpReserved, &dwTempType, NULL, &cb);
|
|
|
|
// If the caller was just asking for the size of the value, jump out
|
|
// now, without actually retrieving and converting the value.
|
|
|
|
if (lpData == NULL)
|
|
{
|
|
switch (dwTempType)
|
|
{
|
|
case REG_EXPAND_SZ:
|
|
case REG_MULTI_SZ:
|
|
case REG_SZ:
|
|
|
|
// Adjust size of buffer to report, to account for CHAR -> WCHAR
|
|
|
|
if (lpcbData != NULL)
|
|
*lpcbData = cb * sizeof(WCHAR);
|
|
break;
|
|
|
|
default:
|
|
|
|
if (lpcbData != NULL)
|
|
*lpcbData = cb;
|
|
break;
|
|
}
|
|
|
|
// Set the type, if required.
|
|
if (lpType != NULL)
|
|
{
|
|
*lpType = dwTempType;
|
|
}
|
|
|
|
goto Exit;
|
|
}
|
|
|
|
|
|
if (ret == ERROR_SUCCESS)
|
|
{
|
|
//
|
|
// Determine the size of buffer needed
|
|
//
|
|
|
|
switch (dwTempType)
|
|
{
|
|
case REG_EXPAND_SZ:
|
|
case REG_MULTI_SZ:
|
|
case REG_SZ:
|
|
|
|
cbRequired = cb * sizeof(WCHAR);
|
|
break;
|
|
|
|
default:
|
|
|
|
cbRequired = cb;
|
|
break;
|
|
}
|
|
|
|
// If the caller was asking for the value, but allocated too small
|
|
// of a buffer, set the buffer size and jump out.
|
|
|
|
if (lpcbData != NULL && *lpcbData < cbRequired)
|
|
{
|
|
// Adjust size of buffer to report, to account for CHAR -> WCHAR
|
|
*lpcbData = cbRequired;
|
|
|
|
// Set the type, if required.
|
|
if (lpType != NULL)
|
|
{
|
|
*lpType = dwTempType;
|
|
}
|
|
|
|
ret = ERROR_MORE_DATA;
|
|
goto Exit;
|
|
}
|
|
|
|
// Otherwise, retrieve and convert the value.
|
|
|
|
switch (dwTempType)
|
|
{
|
|
case REG_EXPAND_SZ:
|
|
case REG_MULTI_SZ:
|
|
case REG_SZ:
|
|
|
|
lpTempBuffer = new BYTE[cbRequired];
|
|
if (lpTempBuffer == NULL)
|
|
{
|
|
return ERROR_OUTOFMEMORY;
|
|
}
|
|
|
|
ret = RegQueryValueExA(hKey,
|
|
sz,
|
|
lpReserved,
|
|
&dwTempType,
|
|
lpTempBuffer,
|
|
&cb);
|
|
|
|
if (ret == ERROR_SUCCESS)
|
|
{
|
|
switch (dwTempType)
|
|
{
|
|
case REG_EXPAND_SZ:
|
|
case REG_SZ:
|
|
|
|
AnsiToUnicode((LPWSTR) lpData, (LPSTR) lpTempBuffer, cb);
|
|
|
|
// Adjust size of buffer to report, to account for CHAR -> WCHAR
|
|
*lpcbData = cbRequired;
|
|
|
|
// Set the type, if required.
|
|
if (lpType != NULL)
|
|
{
|
|
*lpType = dwTempType;
|
|
}
|
|
break;
|
|
|
|
case REG_MULTI_SZ:
|
|
|
|
pszTempNarrow = (LPSTR) lpTempBuffer;
|
|
pwszTempWide = (LPWSTR) lpData;
|
|
|
|
while (*pszTempNarrow != NULL)
|
|
{
|
|
ulStringLength = strlen(pszTempNarrow) + 1;
|
|
AnsiToUnicode(pwszTempWide,
|
|
pszTempNarrow,
|
|
ulStringLength);
|
|
|
|
// Compiler will scale appropriately here
|
|
pszTempNarrow += ulStringLength;
|
|
pwszTempWide += ulStringLength;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (lpTempBuffer)
|
|
delete lpTempBuffer;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
//
|
|
// No conversion of out parameters needed. Just call narrow
|
|
// version with args passed in, and return directly.
|
|
//
|
|
|
|
ret = RegQueryValueExA(hKey,
|
|
sz,
|
|
lpReserved,
|
|
lpType,
|
|
lpData,
|
|
lpcbData);
|
|
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
HWND
|
|
WINAPI
|
|
CreateWindowExX( DWORD dwExStyle,
|
|
LPCWSTR lpClassName,
|
|
LPCWSTR lpWindowName,
|
|
DWORD dwStyle,
|
|
int X,
|
|
int Y,
|
|
int nWidth,
|
|
int nHeight,
|
|
HWND hWndParent ,
|
|
HMENU hMenu,
|
|
HINSTANCE hInstance,
|
|
LPVOID lpParam )
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("CreateWindowEx\n");
|
|
#endif
|
|
|
|
HWND ret = NULL;
|
|
LPSTR szClass;
|
|
LPSTR szWindow;
|
|
BOOL fAtom = FALSE;
|
|
|
|
if (HIWORD(lpClassName) == 0)
|
|
{
|
|
// is it an atom?
|
|
|
|
szClass = (LPSTR) lpClassName;
|
|
fAtom = TRUE;
|
|
}
|
|
else
|
|
{
|
|
// otherwise convert the string
|
|
|
|
szClass = Convert(lpClassName);
|
|
if (szClass == ERR)
|
|
{
|
|
szClass = NULL;
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
szWindow = Convert(lpWindowName);
|
|
if (szWindow == ERR)
|
|
{
|
|
szWindow = NULL;
|
|
goto Exit;
|
|
}
|
|
|
|
ret = SSCreateWindowExA (dwExStyle, szClass, szWindow, dwStyle, X, Y,
|
|
nWidth, nHeight, hWndParent, hMenu, hInstance,
|
|
lpParam);
|
|
|
|
Exit:
|
|
if (!fAtom)
|
|
if (szClass)
|
|
delete szClass;
|
|
if (szWindow)
|
|
delete szWindow;
|
|
return ret;
|
|
}
|
|
|
|
|
|
HWND
|
|
WINAPI
|
|
CreateWindowX( LPCWSTR lpClassName,
|
|
LPCWSTR lpWindowName,
|
|
DWORD dwStyle,
|
|
int X,
|
|
int Y,
|
|
int nWidth,
|
|
int nHeight,
|
|
HWND hWndParent ,
|
|
HMENU hMenu,
|
|
HINSTANCE hInstance,
|
|
LPVOID lpParam )
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("CreateWindow\n");
|
|
#endif
|
|
|
|
return CreateWindowExX(0, lpClassName, lpWindowName, dwStyle, X, Y,
|
|
nWidth, nHeight, hWndParent, hMenu, hInstance, lpParam);
|
|
}
|
|
|
|
|
|
ATOM
|
|
WINAPI
|
|
RegisterClassX(
|
|
CONST WNDCLASSW *lpWndClass)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegisterClass\n");
|
|
#endif
|
|
|
|
WNDCLASSA wc;
|
|
ATOM ret;
|
|
BOOL fAtom = FALSE;
|
|
|
|
Win4Assert(sizeof(WNDCLASSA) == sizeof(WNDCLASSW));
|
|
|
|
memcpy(&wc, lpWndClass, sizeof(WNDCLASS));
|
|
|
|
wc.lpszMenuName = Convert(lpWndClass->lpszMenuName);
|
|
if (wc.lpszMenuName==ERR)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
if (HIWORD(lpWndClass->lpszClassName) == 0)
|
|
{
|
|
wc.lpszClassName = (LPSTR) lpWndClass->lpszClassName;
|
|
fAtom = TRUE;
|
|
}
|
|
else
|
|
{
|
|
wc.lpszClassName = Convert(lpWndClass->lpszClassName);
|
|
if (wc.lpszClassName==ERR)
|
|
{
|
|
if ((LPSTR) wc.lpszMenuName)
|
|
delete (LPSTR) wc.lpszMenuName;
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
ret = RegisterClassA(&wc);
|
|
if ((LPSTR) wc.lpszMenuName)
|
|
delete (LPSTR) wc.lpszMenuName;
|
|
if (!fAtom) delete (LPSTR) wc.lpszClassName;
|
|
return ret;
|
|
}
|
|
|
|
BOOL
|
|
WINAPI
|
|
UnregisterClassX(
|
|
LPCWSTR lpClassName,
|
|
HINSTANCE hInstance)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("UnregisterClass\n");
|
|
#endif
|
|
|
|
LPSTR sz;
|
|
BOOL ret;
|
|
BOOL fAtom = FALSE;
|
|
|
|
if (HIWORD(lpClassName) == 0)
|
|
{
|
|
sz = (LPSTR) lpClassName;
|
|
fAtom = TRUE;
|
|
}
|
|
else
|
|
{
|
|
sz = Convert(lpClassName);
|
|
if (sz == ERR)
|
|
return FALSE;
|
|
}
|
|
|
|
ret = UnregisterClassA(sz, hInstance);
|
|
if (!fAtom) delete sz;
|
|
return ret;
|
|
}
|
|
|
|
HANDLE
|
|
WINAPI
|
|
GetPropX(
|
|
HWND hWnd,
|
|
LPCWSTR lpString)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetProp\n");
|
|
#endif
|
|
|
|
HANDLE ret;
|
|
LPSTR sz;
|
|
BOOL fAtom = FALSE;
|
|
|
|
if (HIWORD(lpString)==0)
|
|
{
|
|
fAtom = TRUE;
|
|
sz = (LPSTR) lpString;
|
|
}
|
|
else
|
|
{
|
|
sz = Convert(lpString);
|
|
if (sz == ERR)
|
|
return NULL;
|
|
}
|
|
|
|
ret = GetPropA(hWnd, sz);
|
|
if (!fAtom) delete sz;
|
|
return ret;
|
|
}
|
|
|
|
|
|
BOOL
|
|
WINAPI
|
|
SetPropX(
|
|
HWND hWnd,
|
|
LPCWSTR lpString,
|
|
HANDLE hData)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("SetProp\n");
|
|
#endif
|
|
|
|
BOOL ret;
|
|
LPSTR sz;
|
|
BOOL fAtom = FALSE;
|
|
|
|
if (HIWORD(lpString)==0)
|
|
{
|
|
sz = (LPSTR) lpString;
|
|
fAtom = TRUE;
|
|
}
|
|
else
|
|
{
|
|
sz = Convert(lpString);
|
|
if (sz == ERR)
|
|
return NULL;
|
|
}
|
|
|
|
ret = SetPropA(hWnd, sz, hData);
|
|
if (!fAtom) delete sz;
|
|
return ret;
|
|
}
|
|
|
|
|
|
HANDLE
|
|
WINAPI
|
|
RemovePropX(
|
|
HWND hWnd,
|
|
LPCWSTR lpString)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RemoveProp\n");
|
|
#endif
|
|
|
|
HANDLE ret;
|
|
LPSTR sz;
|
|
BOOL fAtom = FALSE;
|
|
|
|
if (HIWORD(lpString)==0)
|
|
{
|
|
sz = (LPSTR) lpString;
|
|
fAtom = TRUE;
|
|
}
|
|
else
|
|
{
|
|
sz = Convert(lpString);
|
|
if (sz == ERR)
|
|
return NULL;
|
|
}
|
|
|
|
ret = RemovePropA(hWnd, sz);
|
|
if (!fAtom) delete sz;
|
|
return ret;
|
|
}
|
|
|
|
|
|
UINT
|
|
WINAPI
|
|
GetProfileIntX(
|
|
LPCWSTR lpAppName,
|
|
LPCWSTR lpKeyName,
|
|
INT nDefault
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetProfileInt\n");
|
|
#endif
|
|
|
|
LPSTR szApp;
|
|
LPSTR szKey;
|
|
UINT ret;
|
|
|
|
szApp = Convert(lpAppName);
|
|
if (szApp==ERR)
|
|
{
|
|
return nDefault;
|
|
}
|
|
|
|
szKey = Convert(lpKeyName);
|
|
if (szApp==ERR)
|
|
{
|
|
if (szApp)
|
|
delete szApp;
|
|
return nDefault;
|
|
}
|
|
|
|
ret = GetProfileIntA(szApp, szKey, nDefault);
|
|
if (szApp)
|
|
delete szApp;
|
|
if (szKey)
|
|
delete szKey;
|
|
return ret;
|
|
}
|
|
|
|
ATOM
|
|
WINAPI
|
|
GlobalAddAtomX(
|
|
LPCWSTR lpString
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GlobalAddAtom\n");
|
|
#endif
|
|
|
|
ATOM ret;
|
|
LPSTR sz;
|
|
|
|
sz = Convert(lpString);
|
|
if (sz==ERR)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
ret = GlobalAddAtomA(sz);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
UINT
|
|
WINAPI
|
|
GlobalGetAtomNameX(
|
|
ATOM nAtom,
|
|
LPWSTR pwszBuffer,
|
|
int nSize
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GlobalGetAtomName\n");
|
|
#endif
|
|
|
|
LPSTR sz;
|
|
UINT ret;
|
|
|
|
sz = new CHAR[nSize];
|
|
if (sz == NULL)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
ret = GlobalGetAtomNameA(nAtom, sz, nSize);
|
|
if (ret)
|
|
{
|
|
AnsiToUnicode(pwszBuffer, sz, lstrlenA(sz) + 1);
|
|
}
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
|
|
DWORD
|
|
WINAPI
|
|
GetModuleFileNameX(
|
|
HINSTANCE hModule,
|
|
LPWSTR pwszFilename,
|
|
DWORD nSize
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetModuleFileName\n");
|
|
#endif
|
|
|
|
LPSTR sz;
|
|
DWORD ret;
|
|
|
|
sz = new CHAR[nSize];
|
|
if (sz == NULL)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
ret = GetModuleFileNameA(hModule, sz, nSize);
|
|
if (ret)
|
|
{
|
|
AnsiToUnicode(pwszFilename, sz, lstrlenA(sz) + 1);
|
|
}
|
|
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
|
|
LPWSTR
|
|
WINAPI
|
|
CharPrevX(
|
|
LPCWSTR lpszStart,
|
|
LPCWSTR lpszCurrent)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("CharPrev\n");
|
|
#endif
|
|
|
|
if (lpszCurrent == lpszStart)
|
|
{
|
|
return (LPWSTR) lpszStart;
|
|
}
|
|
else
|
|
{
|
|
return (LPWSTR) lpszCurrent - 1;
|
|
}
|
|
}
|
|
|
|
HFONT WINAPI CreateFontX(int a, int b, int c, int d, int e, DWORD f,
|
|
DWORD g, DWORD h, DWORD i, DWORD j, DWORD k,
|
|
DWORD l, DWORD m, LPCWSTR pwsz)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("CreateFont\n");
|
|
#endif
|
|
|
|
LPSTR sz;
|
|
HFONT ret;
|
|
|
|
sz = Convert(pwsz);
|
|
if (sz == ERR)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
ret = CreateFontA(a,b,c,d,e,f,g,h,i,j,k,l,m,sz);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
|
|
HINSTANCE
|
|
WINAPI
|
|
LoadLibraryX(
|
|
LPCWSTR pwszFileName
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("LoadLibrary\n");
|
|
#endif
|
|
|
|
HINSTANCE ret;
|
|
LPSTR sz;
|
|
|
|
sz = Convert(pwszFileName);
|
|
if (sz == ERR)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
ret = LoadLibraryA(sz);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
|
|
HMODULE
|
|
WINAPI
|
|
LoadLibraryExX(
|
|
LPCWSTR lpLibFileName,
|
|
HANDLE hFile,
|
|
DWORD dwFlags
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("LoadLibrary\n");
|
|
#endif
|
|
|
|
HINSTANCE ret;
|
|
LPSTR sz;
|
|
|
|
sz = ConvertOem(lpLibFileName);
|
|
if (sz == ERR)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
ret = LoadLibraryExA(sz, hFile, dwFlags);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
|
|
|
|
LONG
|
|
APIENTRY
|
|
RegDeleteKeyX(
|
|
HKEY hKey,
|
|
LPCWSTR pwszSubKey
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegDeleteKey\n");
|
|
#endif
|
|
|
|
LONG ret;
|
|
LPSTR sz;
|
|
|
|
sz = Convert(pwszSubKey);
|
|
if (sz == ERR)
|
|
{
|
|
return ERROR_OUTOFMEMORY;
|
|
}
|
|
|
|
ret = RegDeleteKeyA(hKey, sz);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
BOOL
|
|
APIENTRY
|
|
CreateProcessX(
|
|
LPCWSTR lpApplicationName,
|
|
LPWSTR lpCommandLine,
|
|
LPSECURITY_ATTRIBUTES lpProcessAttributes,
|
|
LPSECURITY_ATTRIBUTES lpThreadAttributes,
|
|
BOOL bInheritHandles,
|
|
DWORD dwCreationFlags,
|
|
LPVOID lpEnvironment,
|
|
LPCWSTR lpCurrentDirectory,
|
|
LPSTARTUPINFOW lpStartupInfo,
|
|
LPPROCESS_INFORMATION lpProcessInformation
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("CreateProcess\n");
|
|
#endif
|
|
|
|
STARTUPINFOA si;
|
|
BOOL ret = FALSE;
|
|
LPSTR szApp = NULL;
|
|
LPSTR szCommand = NULL;
|
|
LPSTR szDir = NULL;
|
|
|
|
memcpy(&si, lpStartupInfo, sizeof(STARTUPINFO));
|
|
|
|
si.lpTitle = NULL;
|
|
|
|
si.lpDesktop = Convert(lpStartupInfo->lpDesktop);
|
|
if (si.lpDesktop == ERR)
|
|
{
|
|
si.lpDesktop = NULL;
|
|
goto Error;
|
|
}
|
|
si.lpTitle = Convert(lpStartupInfo->lpTitle);
|
|
if (si.lpTitle == ERR)
|
|
{
|
|
si.lpTitle = NULL;
|
|
goto Error;
|
|
}
|
|
|
|
szApp = Convert(lpApplicationName);
|
|
if (szApp == ERR)
|
|
{
|
|
szApp = NULL;
|
|
goto Error;
|
|
}
|
|
szCommand = ConvertOem(lpCommandLine);
|
|
if (szCommand == ERR)
|
|
{
|
|
szCommand = NULL;
|
|
goto Error;
|
|
}
|
|
szDir = Convert(lpCurrentDirectory);
|
|
if (szDir == ERR)
|
|
{
|
|
szDir = NULL;
|
|
goto Error;
|
|
}
|
|
|
|
ret = CreateProcessA(szApp, szCommand, lpProcessAttributes,
|
|
lpThreadAttributes, bInheritHandles, dwCreationFlags,
|
|
lpEnvironment, szDir, &si, lpProcessInformation);
|
|
|
|
Error:
|
|
if (si.lpDesktop)
|
|
delete si.lpDesktop;
|
|
if (si.lpTitle)
|
|
delete si.lpTitle;
|
|
|
|
if (szApp)
|
|
delete szApp;
|
|
if (szCommand)
|
|
delete szCommand;
|
|
if (szDir)
|
|
delete szDir;
|
|
|
|
return ret;
|
|
}
|
|
|
|
LONG
|
|
APIENTRY
|
|
RegEnumKeyExX(
|
|
HKEY hKey,
|
|
DWORD dwIndex,
|
|
LPWSTR lpName,
|
|
LPDWORD lpcbName,
|
|
LPDWORD lpReserved,
|
|
LPWSTR lpClass,
|
|
LPDWORD lpcbClass,
|
|
PFILETIME lpftLastWriteTime
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegEnumKeyEx\n");
|
|
#endif
|
|
|
|
LPSTR szName;
|
|
LPSTR szClass = NULL;
|
|
LONG ret = ERROR_OUTOFMEMORY;
|
|
|
|
szName = new CHAR[*lpcbName];
|
|
if (szName == NULL)
|
|
goto Exit;
|
|
|
|
if (lpClass != NULL)
|
|
{
|
|
szClass = new CHAR[*lpcbClass + 1];
|
|
if (szName == NULL)
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Return lengths do not include zero char.
|
|
//
|
|
ret = RegEnumKeyExA(hKey, dwIndex, szName, lpcbName, lpReserved,
|
|
szClass, lpcbClass, lpftLastWriteTime);
|
|
|
|
if (ret == ERROR_SUCCESS)
|
|
{
|
|
AnsiToUnicode(lpName, szName, *lpcbName + 1);
|
|
|
|
if (szClass)
|
|
{
|
|
AnsiToUnicode(lpClass, szClass, *lpcbClass + 1);
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
return ret;
|
|
}
|
|
|
|
BOOL
|
|
WINAPI
|
|
AppendMenuX(
|
|
HMENU hMenu,
|
|
UINT uFlags,
|
|
UINT uIDnewItem,
|
|
LPCWSTR lpnewItem
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("AppendMenu\n");
|
|
#endif
|
|
|
|
BOOL ret;
|
|
LPSTR sz;
|
|
|
|
if (uFlags == MF_STRING)
|
|
{
|
|
sz = Convert(lpnewItem);
|
|
if (sz==ERR)
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
sz = (LPSTR) lpnewItem;
|
|
}
|
|
|
|
ret = AppendMenuA(hMenu, uFlags, uIDnewItem, sz);
|
|
|
|
if (uFlags == MF_STRING)
|
|
{
|
|
if (sz)
|
|
delete sz;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
HANDLE
|
|
WINAPI
|
|
OpenEventX(
|
|
DWORD dwDesiredAccess,
|
|
BOOL bInheritHandle,
|
|
LPCWSTR lpName
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("OpenEvent\n");
|
|
#endif
|
|
|
|
LPSTR sz;
|
|
HANDLE ret;
|
|
|
|
sz = Convert(lpName);
|
|
if (sz == ERR)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
ret = OpenEventA(dwDesiredAccess, bInheritHandle, sz);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
HANDLE
|
|
WINAPI
|
|
CreateEventX(
|
|
LPSECURITY_ATTRIBUTES lpEventAttributes,
|
|
BOOL bManualReset,
|
|
BOOL bInitialState,
|
|
LPCWSTR lpName
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("CreateEvent\n");
|
|
#endif
|
|
|
|
LPSTR sz;
|
|
HANDLE ret;
|
|
|
|
sz = Convert(lpName);
|
|
if (sz == ERR)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
ret = CreateEventA(lpEventAttributes, bManualReset, bInitialState, sz);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
UINT
|
|
WINAPI
|
|
GetDriveTypeX(
|
|
LPCWSTR lpRootPathName
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetDriveType\n");
|
|
#endif
|
|
|
|
LPSTR sz;
|
|
UINT ret;
|
|
|
|
sz = Convert(lpRootPathName);
|
|
if (sz == ERR)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
ret = GetDriveTypeA(sz);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
DWORD
|
|
WINAPI
|
|
GetFileAttributesX(
|
|
LPCWSTR lpFileName
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetFileAttributes\n");
|
|
#endif
|
|
|
|
LPSTR sz;
|
|
DWORD ret;
|
|
|
|
sz = ConvertOem(lpFileName);
|
|
if (sz == ERR)
|
|
return 0xFFFFFFFF;
|
|
|
|
ret = GetFileAttributesA(sz);
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
LONG
|
|
APIENTRY
|
|
RegEnumKeyX(
|
|
HKEY hKey,
|
|
DWORD dwIndex,
|
|
LPWSTR lpName,
|
|
DWORD cbName
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("RegEnumKey\n");
|
|
#endif
|
|
|
|
CHAR sz[MAX_PATH+1];
|
|
LONG ret;
|
|
|
|
//
|
|
// Return lengths do not include zero char.
|
|
//
|
|
ret = RegEnumKeyA(hKey, dwIndex, sz, cbName);
|
|
if (ret == ERROR_SUCCESS)
|
|
{
|
|
AnsiToUnicode(lpName, sz, lstrlenA(sz) + 1);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
HFINDFILE
|
|
WINAPI
|
|
FindFirstFileX(
|
|
LPCWSTR lpFileName,
|
|
LPWIN32_FIND_DATAW pwszFd
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("FindFirstFile\n");
|
|
#endif
|
|
|
|
WIN32_FIND_DATAA fd;
|
|
CHAR sz[MAX_PATH * 2];
|
|
HFINDFILE ret;
|
|
int len = lstrlenW(lpFileName) + 1;
|
|
|
|
UnicodeToAnsiOem(sz, lpFileName, sizeof(sz));
|
|
ret = FindFirstFileA(sz, &fd);
|
|
if (ret != INVALID_HANDLE_VALUE)
|
|
{
|
|
memcpy(pwszFd, &fd, sizeof(FILETIME)*3 + sizeof(DWORD)*5);
|
|
AnsiToUnicodeOem(pwszFd->cFileName, fd.cFileName,
|
|
lstrlenA(fd.cFileName) + 1);
|
|
AnsiToUnicodeOem(pwszFd->cAlternateFileName, fd.cAlternateFileName,
|
|
14);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: wsprintfX
|
|
//
|
|
// Synopsis: Nightmare string function
|
|
//
|
|
// Arguments: [pwszOut] --
|
|
// [pwszFormat] --
|
|
// [...] --
|
|
//
|
|
// Returns:
|
|
//
|
|
// History: 1-06-94 ErikGav Created
|
|
//
|
|
// Notes: If you're reading this, you're probably having a problem with
|
|
// this function. Make sure that your "%s" in the format string
|
|
// says "%ws" if you are passing wide strings.
|
|
//
|
|
// %s on NT means "wide string"
|
|
// %s on Chicago means "ANSI string"
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
int WINAPIV wsprintfX(LPWSTR pwszOut, LPCWSTR pwszFormat, ...)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("wsprintf\n");
|
|
#endif
|
|
|
|
LPSTR szFormat;
|
|
LPWSTR pwszTemp = NULL;
|
|
int i = 0;
|
|
|
|
// Convert the format string over
|
|
|
|
szFormat = Convert(pwszFormat);
|
|
if (szFormat == ERR)
|
|
{
|
|
szFormat = NULL;
|
|
goto Exit;
|
|
}
|
|
|
|
// magic voodoo follows:
|
|
//
|
|
// 1. Call wvsprintf passing the varargs
|
|
// 2. Use the pwszOut as a temp buffer to hold the ANSI output
|
|
// 3. Save the returned characters
|
|
|
|
i = wvsprintfA((LPSTR) pwszOut, szFormat,
|
|
(LPSTR) ((BYTE*)&pwszFormat) + sizeof(pwszFormat));
|
|
|
|
// allocate a buffer for the Ansi to Unicode conversion
|
|
|
|
pwszTemp = new WCHAR[i+1];
|
|
|
|
// convert the string
|
|
|
|
AnsiToUnicode(pwszTemp, (LPSTR) pwszOut, i+1);
|
|
|
|
// copy it to the out buffer
|
|
|
|
lstrcpynW(pwszOut, pwszTemp, i+1);
|
|
|
|
Exit:
|
|
if (pwszTemp)
|
|
delete pwszTemp;
|
|
if (szFormat)
|
|
delete szFormat;
|
|
return i;
|
|
}
|
|
|
|
BOOL
|
|
WINAPI
|
|
GetComputerNameX(
|
|
LPWSTR pwszName,
|
|
LPDWORD lpcchBuffer
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetComputerName\n");
|
|
#endif
|
|
|
|
BOOL ret;
|
|
LPSTR sz;
|
|
DWORD OldSize = *lpcchBuffer;
|
|
|
|
sz = new CHAR[*lpcchBuffer];
|
|
ret = GetComputerNameA(sz, &OldSize);
|
|
|
|
if (ret)
|
|
{
|
|
DWORD NewSize = AnsiToUnicode(pwszName, sz, *lpcchBuffer);
|
|
if (NewSize) {
|
|
*lpcchBuffer = NewSize - 1;
|
|
} else {
|
|
*lpcchBuffer = 0;
|
|
}
|
|
}
|
|
|
|
if (sz)
|
|
delete sz;
|
|
return ret;
|
|
}
|
|
|
|
DWORD
|
|
WINAPI
|
|
GetFullPathNameX(
|
|
LPCWSTR lpFileName,
|
|
DWORD cchBuffer,
|
|
LPWSTR lpPathBuffer,
|
|
LPWSTR *lppFilePart
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetFullPathName\n");
|
|
#endif
|
|
|
|
LPSTR szFileName;
|
|
CHAR szPathBuffer[MAX_PATH];
|
|
LPSTR szFilePart;
|
|
DWORD ret;
|
|
|
|
|
|
szFileName = ConvertOem(lpFileName);
|
|
if (szFileName == ERR)
|
|
return 0;
|
|
|
|
ret = GetFullPathNameA(szFileName, cchBuffer, szPathBuffer, &szFilePart);
|
|
|
|
AnsiToUnicode(lpPathBuffer, szPathBuffer, cchBuffer);
|
|
|
|
*lppFilePart = lpPathBuffer + (szFilePart - szPathBuffer);
|
|
|
|
if (szFileName)
|
|
delete szFileName;
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
DWORD
|
|
WINAPI
|
|
GetShortPathNameX(
|
|
LPCWSTR lpszFullPath,
|
|
LPWSTR lpszShortPath,
|
|
DWORD cchBuffer
|
|
)
|
|
{
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetShortPathName\n");
|
|
#endif
|
|
|
|
LPSTR szFullPath;
|
|
CHAR szShortBuffer[MAX_PATH];
|
|
DWORD ret;
|
|
|
|
|
|
szFullPath = Convert(lpszFullPath);
|
|
if (szFullPath == ERR)
|
|
return 0;
|
|
|
|
if (lpszShortPath == NULL)
|
|
{
|
|
ret = GetShortPathNameA(szFullPath, NULL, cchBuffer);
|
|
}
|
|
else
|
|
{
|
|
ret = GetShortPathNameA(szFullPath, szShortBuffer, sizeof(szShortBuffer));
|
|
|
|
//
|
|
// Only convert the actual data, not the whole buffer.
|
|
//
|
|
if (cchBuffer > ret + 1)
|
|
cchBuffer = ret + 1;
|
|
|
|
AnsiToUnicode(lpszShortPath, szShortBuffer, cchBuffer);
|
|
}
|
|
|
|
delete szFullPath;
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
DWORD
|
|
WINAPI
|
|
SearchPathX(
|
|
LPCWSTR lpPath,
|
|
LPCWSTR lpFileName,
|
|
LPCWSTR lpExtension,
|
|
DWORD nBufferLength,
|
|
LPWSTR lpBuffer,
|
|
LPWSTR *lpFilePart
|
|
)
|
|
{
|
|
LPSTR lpszFileName;
|
|
CHAR szBuffer[MAX_PATH];
|
|
DWORD ret;
|
|
|
|
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("SearchPath\n");
|
|
#endif
|
|
|
|
lpszFileName = Convert(lpFileName);
|
|
if (lpszFileName == ERR)
|
|
return 0;
|
|
|
|
ret = SearchPathA(NULL, lpszFileName, NULL, sizeof(szBuffer), szBuffer, NULL);
|
|
|
|
AnsiToUnicode(lpBuffer, szBuffer, lstrlenA(szBuffer) + 1);
|
|
|
|
delete lpszFileName;
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
ATOM
|
|
WINAPI
|
|
GlobalFindAtomX(
|
|
LPCWSTR lpString
|
|
)
|
|
{
|
|
LPSTR lpszString;
|
|
ATOM retAtom;
|
|
|
|
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GlobalFindAtom\n");
|
|
#endif
|
|
|
|
lpszString = Convert(lpString);
|
|
if (lpszString == ERR)
|
|
return 0;
|
|
|
|
retAtom = GlobalFindAtomA(lpszString);
|
|
|
|
delete lpszString;
|
|
|
|
return retAtom;
|
|
}
|
|
|
|
|
|
int
|
|
WINAPI
|
|
GetClassNameX(
|
|
HWND hWnd,
|
|
LPWSTR lpClassName,
|
|
int nMaxCount)
|
|
{
|
|
LPSTR lpszClassName;
|
|
int ret;
|
|
|
|
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("GetClassName\n");
|
|
#endif
|
|
|
|
lpszClassName = Convert(lpClassName);
|
|
if (lpszClassName == ERR)
|
|
return 0;
|
|
|
|
ret = GetClassNameA(hWnd, lpszClassName, nMaxCount);
|
|
|
|
delete lpszClassName;
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
LPWSTR
|
|
WINAPI
|
|
CharLowerX(
|
|
LPWSTR lpsz)
|
|
{
|
|
if (((DWORD)lpsz & 0xffff0000) == 0)
|
|
{
|
|
return (LPWSTR)towlower ((wchar_t)lpsz);
|
|
} else {
|
|
return _wcslwr (lpsz);
|
|
}
|
|
}
|
|
|
|
LPWSTR
|
|
WINAPI
|
|
CharUpperX(
|
|
LPWSTR lpsz)
|
|
{
|
|
if (((DWORD)lpsz & 0xffff0000) == 0)
|
|
{
|
|
return (LPWSTR)towupper ((wchar_t)lpsz);
|
|
} else {
|
|
return _wcsupr (lpsz);
|
|
}
|
|
}
|
|
|
|
BOOL
|
|
WINAPI
|
|
GetStringTypeX(
|
|
DWORD dwInfoType,
|
|
LPCWSTR lpSrcStr,
|
|
int cchSrc,
|
|
LPWORD lpCharType)
|
|
{
|
|
// Convert the source string to MBS. If we don't get the same number
|
|
// of characters, this algorithm doesn't work.
|
|
|
|
int OriginalLength = cchSrc == -1 ? lstrlenW (lpSrcStr) + 1 : cchSrc;
|
|
LPSTR lpConvertedString = new CHAR[OriginalLength+1];
|
|
|
|
if (lpConvertedString == NULL)
|
|
{
|
|
SetLastError (ERROR_NOT_ENOUGH_MEMORY);
|
|
return FALSE;
|
|
}
|
|
|
|
if (WideCharToMultiByte (CP_ACP,
|
|
WC_COMPOSITECHECK,
|
|
lpSrcStr,
|
|
cchSrc,
|
|
lpConvertedString,
|
|
OriginalLength,
|
|
NULL,
|
|
NULL) != OriginalLength)
|
|
{
|
|
delete lpConvertedString;
|
|
SetLastError (ERROR_NO_UNICODE_TRANSLATION);
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL Result;
|
|
|
|
Result = GetStringTypeA (GetThreadLocale (),
|
|
dwInfoType,
|
|
lpConvertedString,
|
|
OriginalLength,
|
|
lpCharType);
|
|
delete lpConvertedString;
|
|
return Result;
|
|
}
|
|
|
|
BOOL
|
|
WINAPI
|
|
IsCharAlphaX(
|
|
WCHAR ch)
|
|
{
|
|
return iswctype (ch, _UPPER | _LOWER);
|
|
}
|
|
|
|
BOOL
|
|
WINAPI
|
|
IsCharAlphaNumericX(
|
|
WCHAR ch)
|
|
{
|
|
return iswctype (ch, _UPPER | _LOWER | _DIGIT);
|
|
}
|
|
|
|
|
|
LPWSTR
|
|
WINAPI
|
|
lstrcatX(
|
|
LPWSTR lpString1,
|
|
LPCWSTR lpString2
|
|
)
|
|
{
|
|
LPWSTR lpDest = lpString1;
|
|
|
|
while (*lpDest) {
|
|
lpDest++;
|
|
}
|
|
|
|
while (*lpDest++ = *lpString2++) ;
|
|
|
|
return lpString1;
|
|
}
|
|
|
|
int
|
|
WINAPI
|
|
lstrcmpX(
|
|
LPCWSTR lpString1,
|
|
LPCWSTR lpString2
|
|
)
|
|
{
|
|
return wcscmp(lpString1, lpString2);
|
|
}
|
|
|
|
LPWSTR
|
|
WINAPI
|
|
lstrcpyX(
|
|
LPWSTR lpString1,
|
|
LPCWSTR lpString2
|
|
)
|
|
{
|
|
LPWSTR lpDest = lpString1;
|
|
|
|
while( *lpDest++ = *lpString2++ )
|
|
;
|
|
|
|
return lpString1;
|
|
}
|
|
|
|
LPWSTR
|
|
WINAPI
|
|
lstrcpynX(
|
|
LPWSTR lpString1,
|
|
LPCWSTR lpString2,
|
|
int iMaxLength
|
|
)
|
|
{
|
|
LPWSTR dst;
|
|
|
|
if (iMaxLength)
|
|
{
|
|
dst = lpString1;
|
|
|
|
while (iMaxLength && *lpString2)
|
|
{
|
|
*dst++ = *lpString2++;
|
|
iMaxLength--;
|
|
}
|
|
if (iMaxLength)
|
|
{
|
|
*dst = L'\0';
|
|
}
|
|
else
|
|
{
|
|
dst[-1] = L'\0';
|
|
}
|
|
}
|
|
|
|
return lpString1;
|
|
}
|
|
|
|
int
|
|
WINAPI
|
|
lstrcmpiX(
|
|
LPCWSTR lpString1,
|
|
LPCWSTR lpString2
|
|
)
|
|
{
|
|
return _wcsicmp(lpString1, lpString2);
|
|
}
|
|
|
|
HANDLE
|
|
WINAPI
|
|
CreateFileMappingX(
|
|
HANDLE hFile,
|
|
LPSECURITY_ATTRIBUTES lpFileMappingAttributes,
|
|
DWORD flProtect,
|
|
DWORD dwMaximumSizeHigh,
|
|
DWORD dwMaximumSizeLow,
|
|
LPCWSTR lpName
|
|
)
|
|
{
|
|
LPSTR lpszAName;
|
|
HANDLE ret;
|
|
|
|
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("CreateFileMapping\n");
|
|
#endif
|
|
|
|
lpszAName = Convert(lpName);
|
|
|
|
if (lpszAName == ERR)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
ret = CreateFileMappingA(
|
|
hFile,
|
|
lpFileMappingAttributes,
|
|
flProtect,
|
|
dwMaximumSizeHigh,
|
|
dwMaximumSizeLow,
|
|
lpszAName);
|
|
|
|
delete lpszAName;
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
HANDLE
|
|
WINAPI
|
|
OpenFileMappingX(
|
|
DWORD dwDesiredAccess,
|
|
BOOL bInheritHandle,
|
|
LPCWSTR lpName
|
|
)
|
|
{
|
|
LPSTR lpszAName;
|
|
HANDLE ret;
|
|
|
|
|
|
#ifdef DEBUG_OUTPUT
|
|
OutputDebugString("CreateFileMapping\n");
|
|
#endif
|
|
|
|
lpszAName = Convert(lpName);
|
|
|
|
if (lpszAName == ERR)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
ret = OpenFileMappingA(
|
|
dwDesiredAccess,
|
|
bInheritHandle,
|
|
lpszAName);
|
|
|
|
delete lpszAName;
|
|
|
|
return ret;
|
|
}
|
|
|
|
#endif // CHICAGO
|