850 lines
23 KiB
C++
850 lines
23 KiB
C++
|
//____________________________________________________________________________
|
||
|
//
|
||
|
// Microsoft Windows
|
||
|
// Copyright (C) Microsoft Corporation, 1995 - 1999
|
||
|
//
|
||
|
// File: util.cpp
|
||
|
//
|
||
|
// Contents:
|
||
|
//
|
||
|
// Classes:
|
||
|
//
|
||
|
// Functions:
|
||
|
//
|
||
|
// History: 7/8/1996 RaviR Created
|
||
|
//
|
||
|
//____________________________________________________________________________
|
||
|
|
||
|
#include <objbase.h>
|
||
|
#include <basetyps.h>
|
||
|
#include "dbg.h"
|
||
|
#include "cstr.h"
|
||
|
#include <Atlbase.h>
|
||
|
#include <winnls.h>
|
||
|
#include "tstring.h"
|
||
|
#include "strings.h"
|
||
|
|
||
|
//############################################################################
|
||
|
//############################################################################
|
||
|
//
|
||
|
// The safer string handling routines
|
||
|
//
|
||
|
//############################################################################
|
||
|
//############################################################################
|
||
|
#include <strsafe.h>
|
||
|
|
||
|
/* define these ourselves until they're defined properly in commctrl.h */
|
||
|
#ifndef ILP_DOWNLEVEL
|
||
|
#define ILP_NORMAL 0 // Writes or reads the stream using new sematics for this version of comctl32
|
||
|
#define ILP_DOWNLEVEL 1 // Write or reads the stream using downlevel sematics.
|
||
|
|
||
|
WINCOMMCTRLAPI HRESULT WINAPI ImageList_ReadEx(DWORD dwFlags, LPSTREAM pstm, REFIID riid, PVOID* ppv);
|
||
|
WINCOMMCTRLAPI HRESULT WINAPI ImageList_WriteEx(HIMAGELIST himl, DWORD dwFlags, LPSTREAM pstm);
|
||
|
#endif
|
||
|
|
||
|
|
||
|
ULONG _ttoul(LPTSTR psz)
|
||
|
{
|
||
|
ULONG ul;
|
||
|
|
||
|
for (ul = 0; *psz != TEXT('\0'); ++psz)
|
||
|
{
|
||
|
ul = ul * 10 + (*psz - TEXT('0'));
|
||
|
}
|
||
|
|
||
|
return ul;
|
||
|
}
|
||
|
|
||
|
|
||
|
WORD I_SplitModuleAndResourceID(LPCTSTR szBuf)
|
||
|
{
|
||
|
WORD wID = (WORD)-1;
|
||
|
|
||
|
// String must be in the form "module, res_id"
|
||
|
|
||
|
for (TCHAR *ptc = (TCHAR *)szBuf;
|
||
|
*ptc != TEXT('\0') && *ptc != TEXT(',');
|
||
|
ptc++);
|
||
|
|
||
|
// If no comma - return
|
||
|
if (*ptc != TEXT(','))
|
||
|
return wID;
|
||
|
|
||
|
*ptc = TEXT('\0');
|
||
|
|
||
|
++ptc;
|
||
|
|
||
|
while (*ptc == TEXT(' ') && *ptc != TEXT('\0'))
|
||
|
{
|
||
|
++ptc;
|
||
|
}
|
||
|
|
||
|
// If it does not have a res_id break.
|
||
|
if (*ptc == TEXT('\0'))
|
||
|
return wID;
|
||
|
|
||
|
// Get the res-id
|
||
|
wID = (WORD)_ttoul(ptc);
|
||
|
|
||
|
return wID;
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL
|
||
|
I_GetStrFromModule(
|
||
|
LPCTSTR pszModule,
|
||
|
ULONG ulMsgNo,
|
||
|
CStr &strBuf)
|
||
|
{
|
||
|
TCHAR szBuf[512];
|
||
|
ULONG cchBuf = 512;
|
||
|
|
||
|
HINSTANCE hinst = LoadLibraryEx(pszModule, NULL,
|
||
|
LOAD_LIBRARY_AS_DATAFILE);
|
||
|
if (hinst)
|
||
|
{
|
||
|
LANGID lidUser = LANGIDFROMLCID(GetUserDefaultLCID());
|
||
|
|
||
|
DWORD cChars = ::FormatMessage(
|
||
|
FORMAT_MESSAGE_FROM_HMODULE |
|
||
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
||
|
(HMODULE)hinst,
|
||
|
ulMsgNo,
|
||
|
lidUser,
|
||
|
szBuf,
|
||
|
cchBuf,
|
||
|
NULL);
|
||
|
|
||
|
FreeLibrary(hinst);
|
||
|
|
||
|
if (cChars != 0)
|
||
|
{
|
||
|
strBuf = szBuf;
|
||
|
return TRUE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//Dbg(DEB_USER1, _T("I_GetStringFromModule failed<%dL>\n"), GetLastError());
|
||
|
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
HICON I_GetHicon(LPCTSTR pszModule, ULONG ulId)
|
||
|
{
|
||
|
HICON hIcon = NULL;
|
||
|
|
||
|
HINSTANCE hinst = LoadLibraryEx(pszModule, NULL,
|
||
|
LOAD_LIBRARY_AS_DATAFILE);
|
||
|
if (hinst)
|
||
|
{
|
||
|
hIcon = LoadIcon(hinst, MAKEINTRESOURCE(ulId));
|
||
|
|
||
|
FreeLibrary(hinst);
|
||
|
}
|
||
|
|
||
|
return hIcon;
|
||
|
}
|
||
|
|
||
|
|
||
|
//+---------------------------------------------------------------------------
|
||
|
//
|
||
|
// Function: NewDupString
|
||
|
//
|
||
|
// Synopsis: Allocates memory & duplicates a given string.
|
||
|
//
|
||
|
// Arguments: [lpszIn] -- IN the string to duplicate.
|
||
|
//
|
||
|
// Returns: The duplicated string. Throws exception if out of memory.
|
||
|
//
|
||
|
//+---------------------------------------------------------------------------
|
||
|
|
||
|
LPTSTR NewDupString(LPCTSTR lpszIn)
|
||
|
{
|
||
|
if(NULL == lpszIn)
|
||
|
{
|
||
|
ASSERT(FALSE);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
register ULONG len = lstrlen(lpszIn) + 1;
|
||
|
|
||
|
TCHAR * lpszOut = new TCHAR[len];
|
||
|
if (lpszOut == NULL)
|
||
|
return NULL;
|
||
|
|
||
|
if(FAILED(StringCchCopy(lpszOut, len, lpszIn)))
|
||
|
{
|
||
|
ASSERT(FALSE);
|
||
|
lpszOut[0] = '\0';
|
||
|
}
|
||
|
return lpszOut;
|
||
|
}
|
||
|
|
||
|
|
||
|
//+---------------------------------------------------------------------------
|
||
|
//
|
||
|
// Function: CoTaskDupString
|
||
|
//
|
||
|
// Synopsis: Allocates memory & duplicates a given string.
|
||
|
//
|
||
|
// Arguments: [lpszIn] -- IN the string to duplicate.
|
||
|
//
|
||
|
// Returns: The duplicated string. Throws exception if out of memory.
|
||
|
//
|
||
|
//+---------------------------------------------------------------------------
|
||
|
// Tony
|
||
|
LPSTR CoTaskDupString(LPCSTR lpszIn)
|
||
|
{
|
||
|
if (lpszIn == NULL)
|
||
|
return NULL;
|
||
|
|
||
|
ULONG cchTemp = (strlen(lpszIn) + 1);
|
||
|
LPSTR lpszOut = (LPSTR) CoTaskMemAlloc(cchTemp * sizeof(CHAR));
|
||
|
|
||
|
if (lpszOut != NULL)
|
||
|
{
|
||
|
if(FAILED(StringCchCopyA(lpszOut, cchTemp, lpszIn)))
|
||
|
{
|
||
|
ASSERT(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return (lpszOut);
|
||
|
}
|
||
|
|
||
|
LPWSTR CoTaskDupString(LPCWSTR lpszIn)
|
||
|
{
|
||
|
if (lpszIn == NULL)
|
||
|
return NULL;
|
||
|
|
||
|
ULONG cchTemp = (wcslen(lpszIn) + 1);
|
||
|
LPWSTR lpszOut = (LPWSTR) CoTaskMemAlloc(cchTemp * sizeof(WCHAR));
|
||
|
|
||
|
if (lpszOut != NULL)
|
||
|
{
|
||
|
if(FAILED(StringCchCopyW(lpszOut, cchTemp, lpszIn)))
|
||
|
{
|
||
|
ASSERT(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return (lpszOut);
|
||
|
}
|
||
|
|
||
|
//+---------------------------------------------------------------------------
|
||
|
//
|
||
|
// Function: GUIDToString
|
||
|
// GUIDFromString
|
||
|
//
|
||
|
// Synopsis: Converts between GUID& and CStr
|
||
|
//
|
||
|
// Returns: FALSE for invalid string, or CMemoryException
|
||
|
//
|
||
|
//+---------------------------------------------------------------------------
|
||
|
|
||
|
HRESULT GUIDToCStr(CStr& str, const GUID& guid)
|
||
|
{
|
||
|
LPOLESTR lpolestr = NULL;
|
||
|
HRESULT hr = StringFromIID( guid, &lpolestr );
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
//TRACE("GUIDToString error %ld\n", hr);
|
||
|
return hr;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
str = lpolestr;
|
||
|
CoTaskMemFree(lpolestr);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT GUIDFromCStr(const CStr& str, GUID* pguid)
|
||
|
{
|
||
|
USES_CONVERSION;
|
||
|
|
||
|
HRESULT hr = IIDFromString( T2OLE( const_cast<LPTSTR>((LPCTSTR)str) ), pguid );
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
//TRACE("GUIDFromString error %ld\n", hr);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
//+---------------------------------------------------------------------------
|
||
|
//
|
||
|
// Function: DoesFileExist
|
||
|
//
|
||
|
// Synopsis: Determines if the specified file exists. The file path may
|
||
|
// include environment variables.
|
||
|
//
|
||
|
// Returns: TRUE/FALSE
|
||
|
//
|
||
|
//+---------------------------------------------------------------------------
|
||
|
|
||
|
BOOL DoesFileExist(LPCTSTR pszFilePath)
|
||
|
{
|
||
|
TCHAR szExpandedPath[MAX_PATH];
|
||
|
|
||
|
DWORD dwCnt = ExpandEnvironmentStrings(pszFilePath, szExpandedPath, MAX_PATH);
|
||
|
if (dwCnt == 0 || dwCnt > MAX_PATH)
|
||
|
return FALSE;
|
||
|
|
||
|
return (::GetFileAttributes(szExpandedPath) != 0xffffffff);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*+-------------------------------------------------------------------------*
|
||
|
*
|
||
|
* GetHelpFile
|
||
|
*
|
||
|
* PURPOSE: Returns a path to the help file
|
||
|
*
|
||
|
* RETURNS:
|
||
|
* static LPCTSTR
|
||
|
*
|
||
|
*+-------------------------------------------------------------------------*/
|
||
|
LPCTSTR GetHelpFile()
|
||
|
{
|
||
|
static const TCHAR NEW_HELP_FILE_STR[] = _T("%windir%\\Help\\MMC_DLG.HLP");
|
||
|
static const TCHAR OLD_HELP_FILE_STR[] = _T("%windir%\\Help\\MMC.HLP");
|
||
|
|
||
|
static LPCTSTR pszHelpFile = NULL;
|
||
|
|
||
|
// See if help file is present. Check new name first, then old name.
|
||
|
// This is done because the old help file may be overwritten by
|
||
|
// an MMC 1.0 installation (see NT bug 299590)
|
||
|
|
||
|
if (pszHelpFile == NULL)
|
||
|
{
|
||
|
if (DoesFileExist(NEW_HELP_FILE_STR))
|
||
|
{
|
||
|
pszHelpFile = NEW_HELP_FILE_STR;
|
||
|
}
|
||
|
else if (DoesFileExist(OLD_HELP_FILE_STR))
|
||
|
{
|
||
|
pszHelpFile = OLD_HELP_FILE_STR;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// if neither file is present, then use the new file name.
|
||
|
// This will let WinHelp display an error message indicating
|
||
|
// that the file is missing and needs to be installed.
|
||
|
pszHelpFile = NEW_HELP_FILE_STR;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return pszHelpFile;
|
||
|
}
|
||
|
|
||
|
//+---------------------------------------------------------------------------
|
||
|
//
|
||
|
// Function: HelpWmHelp
|
||
|
//
|
||
|
// Synopsis: Calls WinHelp with the ID passed to display help
|
||
|
//
|
||
|
// Returns: none
|
||
|
//
|
||
|
//+---------------------------------------------------------------------------
|
||
|
|
||
|
void HelpWmHelp(LPHELPINFO pHelpInfo, const DWORD* pHelpIDs)
|
||
|
{
|
||
|
if (pHelpInfo != NULL)
|
||
|
{
|
||
|
if (pHelpInfo->iContextType == HELPINFO_WINDOW) // must be for a control
|
||
|
{
|
||
|
ASSERT(pHelpIDs != NULL);
|
||
|
if (pHelpIDs)
|
||
|
{
|
||
|
::WinHelp((HWND)pHelpInfo->hItemHandle, GetHelpFile(),
|
||
|
HELP_WM_HELP, (ULONG_PTR)(LPVOID)pHelpIDs);
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*+-------------------------------------------------------------------------*
|
||
|
*
|
||
|
* HelpContextMenuHelp
|
||
|
*
|
||
|
* PURPOSE: Handle context menu help. Invoked when the user right-clicks
|
||
|
* on a dialog item and selects "What's this?"
|
||
|
*
|
||
|
* PARAMETERS:
|
||
|
* HWND hWnd :
|
||
|
* ULONG_PTR p :
|
||
|
*
|
||
|
* RETURNS:
|
||
|
* void
|
||
|
*
|
||
|
*+-------------------------------------------------------------------------*/
|
||
|
void HelpContextMenuHelp(HWND hWnd, ULONG_PTR p)
|
||
|
{
|
||
|
::WinHelp (hWnd, GetHelpFile(), HELP_CONTEXTMENU, p);
|
||
|
}
|
||
|
|
||
|
/*+-------------------------------------------------------------------------*
|
||
|
* InflateFont
|
||
|
*
|
||
|
* Inflates a LOGFONT by the a given number of points
|
||
|
*--------------------------------------------------------------------------*/
|
||
|
|
||
|
bool InflateFont (LOGFONT* plf, int nPointsToGrowBy)
|
||
|
{
|
||
|
if (nPointsToGrowBy != 0)
|
||
|
{
|
||
|
HDC hdc = GetWindowDC (NULL);
|
||
|
|
||
|
if (hdc == NULL)
|
||
|
return (FALSE);
|
||
|
|
||
|
int nLogPixelsY = GetDeviceCaps (hdc, LOGPIXELSY);
|
||
|
int nPoints = -MulDiv (plf->lfHeight, 72, nLogPixelsY);
|
||
|
nPoints += nPointsToGrowBy;
|
||
|
plf->lfHeight = -MulDiv (nPoints, nLogPixelsY, 72);
|
||
|
|
||
|
ReleaseDC (NULL, hdc);
|
||
|
}
|
||
|
|
||
|
return (true);
|
||
|
}
|
||
|
|
||
|
//+-------------------------------------------------------------------
|
||
|
//
|
||
|
// Member: GetTBBtnTextAndStatus
|
||
|
//
|
||
|
// Synopsis: Helper routine to get one/two part button text resource.
|
||
|
//
|
||
|
// Arguments: [hInst] - Instance handle.
|
||
|
// [nID] - String resource id.
|
||
|
// [ppszButton] - Button text.
|
||
|
// [ppszToolTip] - Button status text.
|
||
|
//
|
||
|
// Note: Uses MFC CString.
|
||
|
//
|
||
|
// Returns: bool
|
||
|
//
|
||
|
//--------------------------------------------------------------------
|
||
|
bool GetTBBtnTextAndStatus(HINSTANCE hInst, int nID, std::wstring& szButton, std::wstring& szToolTip)
|
||
|
{
|
||
|
USES_CONVERSION;
|
||
|
|
||
|
CStr str;
|
||
|
str.LoadString(hInst, nID);
|
||
|
ASSERT(!str.IsEmpty());
|
||
|
|
||
|
if (str.IsEmpty())
|
||
|
return false;
|
||
|
|
||
|
int iPos = str.Find(_T('\n'));
|
||
|
if (-1 != iPos)
|
||
|
{
|
||
|
// Two strings. First from 0 to iPos-1
|
||
|
// and second from iPos+1 to end.
|
||
|
szButton = T2CW((LPCTSTR)str.Left(iPos));
|
||
|
szToolTip = T2CW((LPCTSTR)str.Right(str.GetLength() - iPos - 1));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
szButton = (LPCWSTR) NULL;
|
||
|
LPCTSTR pszStr = (LPCTSTR)str;
|
||
|
|
||
|
if (NULL != pszStr)
|
||
|
{
|
||
|
szButton = T2CW(pszStr);
|
||
|
}
|
||
|
|
||
|
szToolTip = szButton;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
|
||
|
#ifdef DBG
|
||
|
|
||
|
/*+-------------------------------------------------------------------------*
|
||
|
* DrawOnDesktop
|
||
|
*
|
||
|
* Draws a bitmap, icon, or imagelist to a specific location on the desktop.
|
||
|
*--------------------------------------------------------------------------*/
|
||
|
|
||
|
void DrawOnDesktop (HBITMAP hbm, int x, int y)
|
||
|
{
|
||
|
HDC hdcDesktop = GetWindowDC (NULL);
|
||
|
HDC hdcMem = CreateCompatibleDC (NULL);
|
||
|
|
||
|
BITMAP bm;
|
||
|
GetObject ((HGDIOBJ) hbm, sizeof(bm), &bm);
|
||
|
HGDIOBJ hbmOld = SelectObject (hdcMem, (HGDIOBJ) hbm);
|
||
|
BitBlt (hdcDesktop, x, y, bm.bmWidth, bm.bmHeight, hdcMem, 0, 0, SRCCOPY);
|
||
|
SelectObject (hdcMem, hbmOld);
|
||
|
|
||
|
DeleteDC (hdcMem);
|
||
|
ReleaseDC (NULL, hdcDesktop);
|
||
|
}
|
||
|
|
||
|
|
||
|
void DrawOnDesktop (HICON hIcon, int x, int y)
|
||
|
{
|
||
|
HDC hdcDesktop = GetWindowDC (NULL);
|
||
|
DrawIconEx (hdcDesktop, x, y, hIcon, 0, 0, 0, NULL, DI_NORMAL);
|
||
|
ReleaseDC (NULL, hdcDesktop);
|
||
|
}
|
||
|
|
||
|
|
||
|
void DrawOnDesktop (HIMAGELIST himl, int x, int y, int iImage /*=-1*/)
|
||
|
{
|
||
|
HDC hdcDesktop = GetWindowDC (NULL);
|
||
|
|
||
|
/*
|
||
|
* draw all images?
|
||
|
*/
|
||
|
if (iImage == -1)
|
||
|
{
|
||
|
int cImages = ImageList_GetImageCount (himl);
|
||
|
int cxImage, cyImage;
|
||
|
ImageList_GetIconSize (himl, &cxImage, &cyImage);
|
||
|
|
||
|
for (int i = 0; i < cImages; i++, x += cxImage)
|
||
|
{
|
||
|
ImageList_Draw (himl, i, hdcDesktop, x, y, ILD_NORMAL);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/*
|
||
|
* draw a specific image
|
||
|
*/
|
||
|
ImageList_Draw (himl, iImage, hdcDesktop, x, y, ILD_NORMAL);
|
||
|
}
|
||
|
|
||
|
ReleaseDC (NULL, hdcDesktop);
|
||
|
}
|
||
|
|
||
|
#endif // DBG
|
||
|
|
||
|
|
||
|
/*+-------------------------------------------------------------------------*
|
||
|
* StripTrailingWhitespace
|
||
|
*
|
||
|
* Removes the whitespace at the end of the input string. Returns a pointer
|
||
|
* the the beginning of the string.
|
||
|
*--------------------------------------------------------------------------*/
|
||
|
|
||
|
LPTSTR StripTrailingWhitespace (LPTSTR pszStart)
|
||
|
{
|
||
|
for (LPTSTR pch = pszStart + _tcslen(pszStart) - 1; pch > pszStart; pch--)
|
||
|
{
|
||
|
/*
|
||
|
* if this isn't a whitespace character, terminate just after this position
|
||
|
*/
|
||
|
if (!_istspace (*pch))
|
||
|
{
|
||
|
*++pch = 0;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return (pszStart);
|
||
|
}
|
||
|
|
||
|
/***************************************************************************\
|
||
|
*
|
||
|
* METHOD: PrivateSetLayout
|
||
|
*
|
||
|
* PURPOSE: Wrapper to invoke GDI function when it is available,
|
||
|
* but not to depend on its availability
|
||
|
*
|
||
|
* PARAMETERS:
|
||
|
* HDC hdc
|
||
|
* DWORD dwLayout
|
||
|
*
|
||
|
* RETURNS:
|
||
|
* DWORD - previous layout, GDI_ERROR on error
|
||
|
*
|
||
|
\***************************************************************************/
|
||
|
DWORD PrivateSetLayout( HDC hdc, DWORD dwLayout )
|
||
|
{
|
||
|
// static pointer to function
|
||
|
static BOOL (WINAPI* pfnSetLayout)(HDC, DWORD) = NULL;
|
||
|
static bool bTriedToGetFunction = false;
|
||
|
|
||
|
if ( !bTriedToGetFunction )
|
||
|
{
|
||
|
bTriedToGetFunction = true;
|
||
|
HINSTANCE hmodGdi = GetModuleHandle (_T("Gdi32.dll"));
|
||
|
|
||
|
if (hmodGdi != NULL)
|
||
|
(FARPROC&)pfnSetLayout = GetProcAddress (hmodGdi, "SetLayout");
|
||
|
}
|
||
|
|
||
|
if (pfnSetLayout == NULL)
|
||
|
return GDI_ERROR;
|
||
|
|
||
|
return (*pfnSetLayout)(hdc, dwLayout);
|
||
|
}
|
||
|
|
||
|
/***************************************************************************\
|
||
|
*
|
||
|
* METHOD: PrivateGetLayout
|
||
|
*
|
||
|
* PURPOSE: Wrapper to invoke GDI function when it is available,
|
||
|
* but not to depend on its availability
|
||
|
*
|
||
|
* PARAMETERS:
|
||
|
* HDC hdc
|
||
|
*
|
||
|
* RETURNS:
|
||
|
* DWORD - layout, 0 if function not found
|
||
|
*
|
||
|
\***************************************************************************/
|
||
|
DWORD PrivateGetLayout( HDC hdc )
|
||
|
{
|
||
|
// static pointer to function
|
||
|
static BOOL (WINAPI* pfnGetLayout)(HDC) = NULL;
|
||
|
static bool bTriedToGetFunction = false;
|
||
|
|
||
|
if ( !bTriedToGetFunction )
|
||
|
{
|
||
|
bTriedToGetFunction = true;
|
||
|
HINSTANCE hmodGdi = GetModuleHandle (_T("Gdi32.dll"));
|
||
|
|
||
|
if (hmodGdi != NULL)
|
||
|
(FARPROC&)pfnGetLayout = GetProcAddress (hmodGdi, "GetLayout");
|
||
|
}
|
||
|
|
||
|
if (pfnGetLayout == NULL)
|
||
|
return 0; // at least not LAYOUT_RTL
|
||
|
|
||
|
return (*pfnGetLayout)(hdc);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*+-------------------------------------------------------------------------*
|
||
|
* IsWhistler
|
||
|
*
|
||
|
* Returns true if we're running on Whistler or higher, false otherwise.
|
||
|
*--------------------------------------------------------------------------*/
|
||
|
bool IsWhistler ()
|
||
|
{
|
||
|
static bool fFirstTime = true;
|
||
|
static bool fWhistler = false;
|
||
|
|
||
|
if (fFirstTime)
|
||
|
{
|
||
|
fFirstTime = false;
|
||
|
|
||
|
OSVERSIONINFO vi;
|
||
|
vi.dwOSVersionInfoSize = sizeof(vi);
|
||
|
GetVersionEx (&vi);
|
||
|
|
||
|
fWhistler = (vi.dwPlatformId == VER_PLATFORM_WIN32_NT) &&
|
||
|
((vi.dwMajorVersion > 5) ||
|
||
|
(vi.dwMajorVersion == 5) && (vi.dwMinorVersion >= 1));
|
||
|
}
|
||
|
|
||
|
return (fWhistler);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*+-------------------------------------------------------------------------*
|
||
|
* WriteCompatibleImageList
|
||
|
*
|
||
|
* Writes an imagelist to a stream in a format that's guaranteed to be
|
||
|
* compatible with comctl32 version 5 imagelists.
|
||
|
*--------------------------------------------------------------------------*/
|
||
|
HRESULT WriteCompatibleImageList (HIMAGELIST himl, IStream* pstm)
|
||
|
{
|
||
|
/*
|
||
|
* If we're running on Whistler, we might be trying to write a v6
|
||
|
* imagelist. Try to write it in a v5-compatible format with
|
||
|
* ImageList_WriteEx.
|
||
|
*/
|
||
|
if (IsWhistler())
|
||
|
{
|
||
|
/*
|
||
|
* ImageList_WriteEx will return E_NOINTERFACE if we're actually
|
||
|
* writing a v5 imagelist, in which case we want to write with
|
||
|
* ImageList_Write. In any other case (success or failure), we
|
||
|
* just want to return.
|
||
|
*/
|
||
|
HRESULT hr = ImageList_WriteEx (himl, ILP_DOWNLEVEL, pstm);
|
||
|
if (hr != E_NOINTERFACE)
|
||
|
return (hr);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* if we get here, we have a v5 imagelist -- just write it
|
||
|
*/
|
||
|
return (ImageList_Write (himl, pstm));
|
||
|
}
|
||
|
|
||
|
|
||
|
/*+-------------------------------------------------------------------------*
|
||
|
* ReadCompatibleImageList
|
||
|
*
|
||
|
* Reads an imagelist from a stream that's in version 5 format.
|
||
|
*--------------------------------------------------------------------------*/
|
||
|
HRESULT ReadCompatibleImageList (IStream* pstm, HIMAGELIST& himl)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
/*
|
||
|
* init the out parameter
|
||
|
*/
|
||
|
himl = NULL;
|
||
|
|
||
|
/*
|
||
|
* If we're running on Whistler, we're trying to create a v6
|
||
|
* imagelist from the stream. Do it in a v5-compatible manner
|
||
|
* with ImageList_ReadEx.
|
||
|
*/
|
||
|
if (IsWhistler())
|
||
|
{
|
||
|
/*
|
||
|
* HACK: We have to query ImageList_ReadEx for IID_IImageList -- the
|
||
|
* one defined by the shell, not the one defined by MMC. If we
|
||
|
* just refer to "IID_IImageList" in the code here, we'll get MMC's
|
||
|
* version, not the shell's. The right way to fix it is to rename
|
||
|
* the shell's IImageList interface (since MMC's interface was defined
|
||
|
* and published first), but that's not going to happen.
|
||
|
*
|
||
|
* We'll hardcode the IID's value in a string here and convert it
|
||
|
* to an IID on the fly. Ugh.
|
||
|
*/
|
||
|
IID iidShellImageList = {0};
|
||
|
hr = CLSIDFromString (L"{46eb5926-582e-4017-9fdf-e8998daa0950}", &iidShellImageList);
|
||
|
if (FAILED (hr))
|
||
|
return (hr);
|
||
|
|
||
|
/*
|
||
|
* ImageList_ReadEx will return E_NOINTERFACE if we're actually
|
||
|
* writing a v5 imagelist, in which case we want to write with
|
||
|
* ImageList_Write. In any other case (success or failure), we
|
||
|
* just want to return.
|
||
|
*/
|
||
|
IUnknownPtr spUnk;
|
||
|
hr = ImageList_ReadEx (ILP_DOWNLEVEL, pstm, iidShellImageList, (void**) &spUnk);
|
||
|
if (FAILED (hr))
|
||
|
return (hr);
|
||
|
|
||
|
/*
|
||
|
* The IUnknown *is* the HIMAGELIST. Don't release it here,
|
||
|
* ImageList_Destroy will take care of it.
|
||
|
*/
|
||
|
himl = reinterpret_cast<HIMAGELIST>(spUnk.Detach());
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/*
|
||
|
* non-Whistler, just read it normally
|
||
|
*/
|
||
|
himl = ImageList_Read (pstm);
|
||
|
|
||
|
/*
|
||
|
* If the read failed, get the last error. Just in case ImageList_Read
|
||
|
* didn't set the last error, make sure we return a failure code.
|
||
|
*/
|
||
|
if (himl == NULL)
|
||
|
{
|
||
|
hr = HRESULT_FROM_WIN32 (GetLastError());
|
||
|
if (!FAILED (hr))
|
||
|
hr = E_FAIL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return (hr);
|
||
|
}
|
||
|
|
||
|
//+-------------------------------------------------------------------
|
||
|
//
|
||
|
// Member: MmcDownlevelActivateActCtx
|
||
|
//
|
||
|
// Synopsis: Calls ActivateActCtx to set the activation context to V5
|
||
|
// common controls. This is needed before calling into snapins
|
||
|
// so that snapin created windows are not themed accidentally.
|
||
|
//
|
||
|
// The snapin can theme its windows by calling appropriate
|
||
|
// fusion apis while calling create-window.
|
||
|
//
|
||
|
// Description:
|
||
|
// When MMC calls into the snapin if the last winproc which
|
||
|
// received a window message is themed and will result in a
|
||
|
// call to snapin then we will call the snapin in themed
|
||
|
// context. If snapin creates & displays any UI then it will
|
||
|
// be themed. This function is to de-activate the theming
|
||
|
// before calling the snapin.
|
||
|
//
|
||
|
// Arguments:
|
||
|
// [hActCtx] - See ActivateActCtx API details
|
||
|
// [pulCookie] - See ActivateActCtx API details
|
||
|
//
|
||
|
// Returns: BOOL, TRUE if we could de-activate V6 context and switch to V5 context
|
||
|
// or if we are in V5 context (W2K, Win95, Win98...)
|
||
|
// FALSE if ActivateActCtx returns failure.
|
||
|
//
|
||
|
//--------------------------------------------------------------------
|
||
|
BOOL WINAPI MmcDownlevelActivateActCtx(HANDLE hActCtx, ULONG_PTR* pulCookie)
|
||
|
{
|
||
|
typedef BOOL (WINAPI* PFN)(HANDLE hActCtx, ULONG_PTR* pulCookie);
|
||
|
static PFN s_pfn;
|
||
|
static DWORD s_dwError;
|
||
|
|
||
|
if (s_pfn == NULL && s_dwError == 0)
|
||
|
if ((s_pfn = (PFN)GetProcAddress(GetModuleHandleA("Kernel32.dll"), "ActivateActCtx")) == NULL)
|
||
|
s_dwError = (GetLastError() == NO_ERROR) ? ERROR_INTERNAL_ERROR : GetLastError();
|
||
|
|
||
|
if (s_pfn != NULL)
|
||
|
return s_pfn(hActCtx, pulCookie);
|
||
|
|
||
|
SetLastError(s_dwError);
|
||
|
|
||
|
if (s_dwError == ERROR_PROC_NOT_FOUND)
|
||
|
return TRUE;
|
||
|
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
|
||
|
//+-------------------------------------------------------------------
|
||
|
//
|
||
|
// Member: MmcDownlevelDeactivateActCtx
|
||
|
//
|
||
|
// Synopsis: Calls DeactivateActCtx to restore the activation context.
|
||
|
// This is needed after calling into snapins, so that
|
||
|
// if we called from themed context then it is restored.
|
||
|
//
|
||
|
// Description:
|
||
|
// When MMC calls into the snapin if the last winproc which
|
||
|
// received a window message is themed and will result in a
|
||
|
// call to snapin then we will call the snapin in themed
|
||
|
// context. If snapin creates & displays any UI then it will
|
||
|
// be themed. This function is to de-activate the theming
|
||
|
// before calling the snapin.
|
||
|
//
|
||
|
// Arguments:
|
||
|
// [dwFlags] - See DeactivateActCtx API details
|
||
|
// [ulCookie] - See DeactivateActCtx API details
|
||
|
//
|
||
|
// Returns: None
|
||
|
//
|
||
|
//--------------------------------------------------------------------
|
||
|
VOID WINAPI MmcDownlevelDeactivateActCtx(DWORD dwFlags, ULONG_PTR ulCookie)
|
||
|
{
|
||
|
typedef VOID (WINAPI* PFN)(DWORD dwFlags, ULONG_PTR ulCookie);
|
||
|
static PFN s_pfn;
|
||
|
static BOOL s_fInited;
|
||
|
|
||
|
if (!s_fInited)
|
||
|
s_pfn = (PFN)GetProcAddress(GetModuleHandleA("Kernel32.dll"), "DeactivateActCtx");
|
||
|
|
||
|
s_fInited = TRUE;
|
||
|
|
||
|
if (s_pfn != NULL)
|
||
|
s_pfn(dwFlags, ulCookie);
|
||
|
}
|
||
|
|
||
|
|