mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
6701 lines
225 KiB
6701 lines
225 KiB
//-------------------------------------------------------------------------//
|
|
// NCTheme.cpp
|
|
//-------------------------------------------------------------------------//
|
|
// bug: resizable dialog (themesel) doesn't repaint client when needed
|
|
// (for test case, resize "themesel" using "bussolid" theme.
|
|
//-------------------------------------------------------------------------//
|
|
#include "stdafx.h"
|
|
#include "nctheme.h"
|
|
#include "sethook.h"
|
|
#include "info.h"
|
|
#include "rgn.h" // AddToCompositeRgn()
|
|
#include "scroll.h" // DrawSizeBox, DrawScrollBar, HandleScrollCmd
|
|
#include "resource.h"
|
|
#include "tmreg.h"
|
|
#include "wrapper.h"
|
|
#include "appinfo.h"
|
|
|
|
//-------------------------------------------------------------------------//
|
|
/// local macros, consts, vars
|
|
//-------------------------------------------------------------------------//
|
|
const RECT rcNil = {-1,-1,-1,-1};
|
|
const WINDOWPARTS BOGUS_WINDOWPART = (WINDOWPARTS)0;
|
|
#define VALID_WINDOWPART(part) ((part)!=BOGUS_WINDOWPART)
|
|
#define WS_MINMAX (WS_MINIMIZEBOX | WS_MAXIMIZEBOX)
|
|
#define HAS_CAPTIONBAR( dwStyle ) (WS_CAPTION == ((dwStyle) & WS_CAPTION))
|
|
#define DLGWNDCLASSNAME TEXT("#32770")
|
|
#define DLGWNDCLASSNAMEW L"#32770"
|
|
|
|
#define NUMBTNSTATES 4 /*number of defined states*/
|
|
#define MAKE_BTNSTATE(framestate, state) ((((framestate)-1) * NUMBTNSTATES) + (state))
|
|
#define MDIBTNINDEX(ncrc) ((ncrc)-NCMDIBTNFIRST)
|
|
|
|
#ifdef MAKEPOINT
|
|
#undef MAKEPOINT
|
|
#endif MAKEPOINT
|
|
#define MAKEPOINT(pt,lParam) POINTSTOPOINT(pt, MAKEPOINTS(lParam))
|
|
|
|
#define IsHTFrameButton(htCode) \
|
|
(((htCode) == HTMINBUTTON) || \
|
|
((htCode) == HTMAXBUTTON) || \
|
|
((htCode) == HTCLOSE) || \
|
|
((htCode) == HTHELP))
|
|
|
|
#define IsTopLevelWindow(hwnd) (IsWindow(hwnd) && NULL==GetParent(hwnd))
|
|
|
|
#define IsHTScrollBar(htCode) (((htCode) == HTVSCROLL) || ((htCode) == HTHSCROLL))
|
|
|
|
#define SIG_CTHEMEWND_HEAD "themewnd"
|
|
#define SIG_CTHEMEWND_TAIL "end"
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HWND _hwndFirstTop = NULL; // first themed window in process
|
|
TCHAR _szWindowMetrics[128] = {0}; // WM_SETTINGCHANGE string param.
|
|
//-------------------------------------------------------------------------//
|
|
|
|
// debug painting switch.
|
|
#define DEBUG_NCPAINT
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// internal helper forwards
|
|
//-------------------------------------------------------------------------//
|
|
HDC _GetNonclientDC( IN HWND hwnd, IN OPTIONAL HRGN hrgnUpdate );
|
|
BOOL _ClientRectToScreen( HWND, LPRECT prcClient );
|
|
void _ScreenToParent( HWND, LPRECT prcWnd );
|
|
BOOL _GetWindowMonitorRect( HWND hwnd, LPRECT prcMonitor );
|
|
BOOL _GetMaximizedContainer( IN HWND hwnd, OUT LPRECT prcContainer );
|
|
BOOL _IsFullMaximized( IN OPTIONAL HWND hwnd, IN LPCRECT prcWnd );
|
|
BOOL _IsMessageWindow( HWND );
|
|
void _MDIUpdate( HWND hwndMDIChildOrClient, UINT uSwpFlags );
|
|
BOOL _MDIClientUpdateChildren( HWND hwndMDIClient );
|
|
void _MDIChildUpdateParent( HWND hwndMDIChild, BOOL fSetMenu = FALSE );
|
|
HWND _MDIGetActive( HWND, OUT OPTIONAL BOOL* pfMaximized = NULL );
|
|
HWND _MDIGetParent( HWND hwnd, OUT OPTIONAL CThemeWnd** ppMdiFrame = NULL, OUT OPTIONAL HWND *phwndMDIClient = NULL );
|
|
HRESULT _CreateBackgroundBitmap( IN OPTIONAL HDC, IN HTHEME, IN int iPartId, IN int iStateId, IN OUT LPSIZE, OUT HBITMAP*);
|
|
HRESULT _CreateMdiMenuItemBitmap( IN HDC hdc, IN HTHEME hTheme, IN OUT SIZE* pSize, IN OUT MENUITEMINFO* pmii );
|
|
void _ComputeElementOffset( HTHEME, int iPartId, int iStateId, LPCRECT prcBase, POINT *pptOffset);
|
|
int _GetRawClassicCaptionHeight( DWORD dwStyle, DWORD dwExStyle );
|
|
int _GetSumClassicCaptionHeight( DWORD dwStyle, DWORD dwExStyle );
|
|
void _ComputeNcWindowStatus( IN HWND, IN DWORD dwStatus, IN OUT NCWNDMET* pncwm );
|
|
BOOL _NeedsWindowEdgeStyle(DWORD dwStyle, DWORD dwExStyle );
|
|
BOOL _MNCanClose(HWND);
|
|
int _GetWindowBorders(LONG lStyle, DWORD dwExStyle );
|
|
BOOL _GetWindowMetrics( HWND, IN OPTIONAL HWND hwndMDIActive, OUT NCWNDMET* pncwm );
|
|
BOOL _IsNcPartTransparent( WINDOWPARTS part, const NCTHEMEMET& nctm );
|
|
BOOL _GetNcFrameMetrics( HWND, HTHEME hTheme, const NCTHEMEMET&, IN OUT NCWNDMET& );
|
|
BOOL _GetNcCaptionMargins( HTHEME hTheme, IN const NCTHEMEMET& nctm, IN OUT NCWNDMET& ncwm );
|
|
LPWSTR _AllocWindowText( IN HWND hwnd );
|
|
BOOL _GetNcCaptionTextSize( IN HTHEME hTheme, IN HWND hwnd, IN HFONT hf, OUT SIZE* psizeCaption );
|
|
BOOL _GetNcCaptionTextRect( IN OUT NCWNDMET* pncwm );
|
|
COLORREF _GetNcCaptionTextColor( FRAMESTATES iStateId );
|
|
void _GetNcBtnHitTestRect( IN const NCWNDMET* pncwm, IN UINT uHitcode, BOOL fWindowRelative, OUT LPRECT prcHit );
|
|
void _GetBrushesForPart(HTHEME hTheme, int iPart, HBITMAP* phbm, HBRUSH* phbr);
|
|
BOOL _ShouldAssignFrameRgn( IN const NCWNDMET* pncwm, IN const NCTHEMEMET& nctm );
|
|
BOOL _IsNcPartTransparent( WINDOWPARTS part, const NCTHEMEMET& nctm );
|
|
BOOL _ComputeNcPartTransparency( HTHEME, IN OUT NCTHEMEMET* pnctm );
|
|
HRESULT _LoadNcThemeMetrics( HWND, IN OUT OPTIONAL NCTHEMEMET* pnctm );
|
|
HRESULT _LoadNcThemeSysMetrics( HWND hwnd, IN OUT OPTIONAL NCTHEMEMET* pnctm );
|
|
void _NcSetPreviewMetrics( BOOL fPreview );
|
|
BOOL _NcUsingPreviewMetrics();
|
|
|
|
//-------------------------------------------------------------------------//
|
|
#ifdef THEMED_NCBTNMETRICS
|
|
BOOL _GetClassicNcBtnMetrics( IN OPTIONAL NCWNDMET* pncwm, IN OPTIONAL HICON hAppIcon,
|
|
IN OPTIONAL BOOL fCanClose, IN OPTIONAL BOOL fRefresh = FALSE );
|
|
#else THEMED_NCBTNMETRICS
|
|
BOOL _GetNcBtnMetrics( IN OUT NCWNDMET*, IN const NCTHEMEMET*, IN HICON, IN OPTIONAL BOOL );
|
|
#endif THEMED_NCBTNMETRICS
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Debug painting.
|
|
#if defined(DEBUG)
|
|
ULONG _NcTraceFlags = 0;
|
|
# if defined(DEBUG_NCPAINT)
|
|
# define BEGIN_DEBUG_NCPAINT() int cgbl = 0; if(TESTFLAG(_NcTraceFlags, NCTF_NCPAINT)) {GdiSetBatchLimit(1);}
|
|
# define END_DEBUG_NCPAINT() if(TESTFLAG(_NcTraceFlags, NCTF_NCPAINT)) {GdiSetBatchLimit(cgbl);}
|
|
HRESULT _DebugDrawThemeBackground(HTHEME, HDC, int, int, const RECT*, OPTIONAL const RECT*);
|
|
HRESULT _DebugDrawThemeBackgroundEx(HTHEME, HDC, int, int, const RECT *prc, OPTIONAL const DTBGOPTS*);
|
|
void NcDebugClipRgn( HDC hdc, COLORREF rgbPaint );
|
|
# define NcDrawThemeBackground _DebugDrawThemeBackground
|
|
# define NcDrawThemeBackgroundEx _DebugDrawThemeBackgroundEx
|
|
# else //defined(DEBUG_NCPAINT)
|
|
# define BEGIN_DEBUG_NCPAINT()
|
|
# define END_DEBUG_NCPAINT()
|
|
# define NcDrawThemeBackground DrawThemeBackground
|
|
# define NcDrawThemeBackgroundEx DrawThemeBackgroundEx
|
|
# define NcDebugClipRgn(hdc,rgbPaint)
|
|
# endif //defined(DEBUG_NCPAINT)
|
|
#else
|
|
# define BEGIN_DEBUG_NCPAINT()
|
|
# define END_DEBUG_NCPAINT()
|
|
# define NcDrawThemeBackground DrawThemeBackground
|
|
# define NcDrawThemeBackgroundEx DrawThemeBackgroundEx
|
|
# define NcDebugClipRgn(hdc,rgbPaint)
|
|
#endif //defined(DEBUG)
|
|
#define RGBDEBUGBKGND RGB(0xFF,0x00,0xFF) // debug background indicator fill color
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// process-global metrics
|
|
static NCTHEMEMET _nctmCurrent = {0};
|
|
|
|
CRITICAL_SECTION _csNcSysMet = {0}; // protects access to _incmCurrent
|
|
CRITICAL_SECTION _csThemeMet = {0}; // protects access to _nctmCurrent
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// process NONCLIENTMETRICS cache.
|
|
struct CInternalNonclientMetrics
|
|
//-------------------------------------------------------------------------//
|
|
{
|
|
const NONCLIENTMETRICS& GetNcm()
|
|
{
|
|
Acquire(FALSE);
|
|
return _ncm;
|
|
}
|
|
|
|
HFONT GetFont( BOOL fSmallCaption )
|
|
{
|
|
if( _fSet)
|
|
{
|
|
return fSmallCaption ? _hfSmCaption : _hfCaption;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void operator =( const NONCLIENTMETRICS& ncmSrc )
|
|
{
|
|
_ncm = ncmSrc;
|
|
|
|
SAFE_DELETE_GDIOBJ(_hfCaption);
|
|
_hfCaption = CreateFontIndirect( &_ncm.lfCaptionFont );
|
|
|
|
SAFE_DELETE_GDIOBJ(_hfSmCaption);
|
|
_hfSmCaption = CreateFontIndirect( &_ncm.lfSmCaptionFont );
|
|
|
|
_fSet = TRUE;
|
|
}
|
|
|
|
BOOL Acquire( BOOL fRefresh )
|
|
{
|
|
//---- quick check for outdated metrics ----
|
|
if (!_fPreview)
|
|
{
|
|
int iNewHeight = GetSystemMetrics(SM_CYSIZE);
|
|
|
|
if (iNewHeight != _iCaptionButtonHeight) // out of date
|
|
{
|
|
fRefresh = TRUE; // force the issue
|
|
_iCaptionButtonHeight = iNewHeight;
|
|
}
|
|
}
|
|
|
|
// normal metrics
|
|
if( !_fSet || fRefresh )
|
|
{
|
|
// save logfont checksum
|
|
LOGFONT lfCaption = _ncm.lfCaptionFont;
|
|
LOGFONT lfSmCaption = _ncm.lfSmCaptionFont;
|
|
|
|
Log(LOG_TMLOAD, L"Acquire: calling ClassicSystemParmetersInfo");
|
|
|
|
_ncm.cbSize = sizeof(_ncm);
|
|
_fSet = ClassicSystemParametersInfo( SPI_GETNONCLIENTMETRICS, 0, &_ncm, FALSE );
|
|
|
|
if( _fSet )
|
|
{
|
|
// if old, new logfont checksums don't match, recycle our fonts
|
|
if( CompareLogfont( &lfCaption, &_ncm.lfCaptionFont) )
|
|
{
|
|
SAFE_DELETE_GDIOBJ(_hfCaption);
|
|
_hfCaption = CreateFontIndirect(&_ncm.lfCaptionFont);
|
|
}
|
|
|
|
if( CompareLogfont( &lfSmCaption, &_ncm.lfSmCaptionFont) )
|
|
{
|
|
SAFE_DELETE_GDIOBJ(_hfSmCaption);
|
|
_hfSmCaption = CreateFontIndirect(&_ncm.lfSmCaptionFont);
|
|
}
|
|
}
|
|
}
|
|
return _fSet;
|
|
}
|
|
|
|
void Clear()
|
|
{
|
|
SAFE_DELETE_GDIOBJ(_hfCaption);
|
|
SAFE_DELETE_GDIOBJ(_hfSmCaption);
|
|
ZeroMemory( &_ncm, sizeof(_ncm) );
|
|
_fSet = FALSE;
|
|
}
|
|
|
|
static int CompareLogfont( const LOGFONT* plf1, const LOGFONT* plf2 )
|
|
{
|
|
int n = memcmp( plf1, plf2, sizeof(LOGFONT) - sizeof(plf1->lfFaceName) );
|
|
if( !n )
|
|
{
|
|
n = lstrcmp( plf1->lfFaceName, plf2->lfFaceName );
|
|
}
|
|
return n;
|
|
}
|
|
|
|
NONCLIENTMETRICS _ncm;
|
|
int _iCaptionButtonHeight;
|
|
BOOL _fSet;
|
|
HFONT _hfCaption;
|
|
HFONT _hfSmCaption;
|
|
BOOL _fPreview;
|
|
|
|
} _incmCurrent = {0}, _incmPreview = {0};
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// MDI sys button group abstraction
|
|
class CMdiBtns
|
|
//-------------------------------------------------------------------------//
|
|
{
|
|
public:
|
|
CMdiBtns();
|
|
~CMdiBtns() { Unload(); }
|
|
|
|
BOOL Load( IN HTHEME hTheme, IN OPTIONAL HDC hdc = NULL, IN OPTIONAL UINT uSysCmd = 0 );
|
|
BOOL ThemeItem( HMENU hMenu, int iPos, MENUITEMINFO* pmii, BOOL fTheme );
|
|
void Unload( IN OPTIONAL UINT uSysCmd = 0 );
|
|
BOOL Measure( IN HTHEME hTheme, IN OUT MEASUREITEMSTRUCT* pmis );
|
|
BOOL Draw( IN HTHEME hTheme, IN DRAWITEMSTRUCT* pdis );
|
|
|
|
private:
|
|
|
|
#define MDIBTNCOUNT 3 // 1=min, 2=restore, 3=close
|
|
//------------------------------------//
|
|
// MDI sys button descriptor element
|
|
struct MDIBTN
|
|
{
|
|
UINT wID;
|
|
WINDOWPARTS iPartId;
|
|
SIZINGTYPE sizingType;
|
|
SIZE size;
|
|
UINT fTypePrev;
|
|
HBITMAP hbmPrev;
|
|
HBITMAP hbmTheme;
|
|
|
|
} _rgBtns[MDIBTNCOUNT];
|
|
|
|
private:
|
|
MDIBTN* _FindBtn( IN UINT wID );
|
|
static CLOSEBUTTONSTATES _CalcState( IN ULONG ulodAction, IN ULONG ulodState );
|
|
};
|
|
|
|
//-------------------------------------------------------------------------//
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// utility impl
|
|
//-------------------------------------------------------------------------//
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL _ClientRectToScreen( HWND hwnd, LPRECT prcClient )
|
|
{
|
|
if( prcClient && GetClientRect( hwnd, prcClient ) )
|
|
{
|
|
POINT* pp = (POINT*)prcClient;
|
|
|
|
//---- use MapWindowPoints() to account for mirrored windows ----
|
|
MapWindowPoints(hwnd, HWND_DESKTOP, pp, 2);
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void _ScreenToParent( HWND hwnd, LPRECT prcWnd )
|
|
{
|
|
//if we have a parent, we need to convert to those coords
|
|
HWND hwndParent = GetAncestor(hwnd, GA_PARENT);
|
|
POINT* pp = (POINT*)prcWnd;
|
|
|
|
//---- use MapWindowPoints() to account for mirrored windows ----
|
|
MapWindowPoints(HWND_DESKTOP, hwndParent, pp, 2);
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
inline BOOL _StrictPtInRect( LPCRECT prc, const POINT& pt )
|
|
{
|
|
// Win32 PtInRect will test positive for an empty rectangle...
|
|
return !IsRectEmpty(prc) &&
|
|
PtInRect( prc, pt );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
inline BOOL _RectInRect( LPCRECT prcTest, LPCRECT prc )
|
|
{
|
|
if ( prc->left < prcTest->left &&
|
|
prc->right > prcTest->right &&
|
|
prc->top < prcTest->top &&
|
|
prc->bottom > prcTest->bottom )
|
|
{
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
inline HDC _GetNonclientDC( IN HWND hwnd, IN OPTIONAL HRGN hrgnUpdate )
|
|
{
|
|
// private GetDCEx #defines from user
|
|
#define DCX_USESTYLE 0x00010000L
|
|
#define DCX_NODELETERGN 0x00040000L
|
|
|
|
DWORD dwDCX = DCX_USESTYLE|DCX_WINDOW|DCX_LOCKWINDOWUPDATE;
|
|
|
|
if( hrgnUpdate != NULL )
|
|
dwDCX |= (DCX_INTERSECTRGN|DCX_NODELETERGN);
|
|
|
|
return GetDCEx( hwnd, hrgnUpdate, dwDCX );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HWND _MDIGetActive( HWND hwndMDIClient, OUT OPTIONAL BOOL* pfMaximized )
|
|
{
|
|
BOOL fMaximized = FALSE;
|
|
HWND hwndActive = NULL;
|
|
|
|
if( IsWindow( hwndMDIClient ) )
|
|
hwndActive = (HWND)SendMessage( hwndMDIClient, WM_MDIGETACTIVE, 0, (LPARAM)&fMaximized );
|
|
|
|
if( pfMaximized ) *pfMaximized = fMaximized;
|
|
return hwndActive;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------////
|
|
// computes rectangle of window's default monitor
|
|
BOOL _GetWindowMonitorRect( HWND hwnd, LPRECT prcMonitor )
|
|
{
|
|
if( IsWindow(hwnd) )
|
|
{
|
|
// default to primary monitor
|
|
SetRect( prcMonitor, 0, 0,
|
|
NcGetSystemMetrics(SM_CXSCREEN),
|
|
NcGetSystemMetrics(SM_CYSCREEN));
|
|
|
|
// try determining window's real monitor
|
|
HMONITOR hMon = MonitorFromWindow( hwnd, MONITOR_DEFAULTTONULL );
|
|
if( hMon )
|
|
{
|
|
MONITORINFO mi;
|
|
mi.cbSize = sizeof(mi);
|
|
if( GetMonitorInfo( hMon, &mi ) )
|
|
{
|
|
*prcMonitor = mi.rcWork;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------////
|
|
// determines whether the indicate window is as large or larger than
|
|
// the target monitor
|
|
BOOL _GetMaximizedContainer(
|
|
IN HWND hwnd,
|
|
OUT LPRECT prcContainer )
|
|
{
|
|
ASSERT(IsWindow(hwnd));
|
|
|
|
HWND hwndParent = GetParent(hwnd);
|
|
if( hwndParent )
|
|
{
|
|
return GetWindowRect( hwndParent, prcContainer );
|
|
}
|
|
|
|
// top-level window: container is primary monitor
|
|
return _GetWindowMonitorRect( hwnd, prcContainer );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------////
|
|
// determines whether the indicate window is as large or larger than
|
|
// the target monitor
|
|
BOOL _IsFullMaximized( IN OPTIONAL HWND hwnd, IN LPCRECT prcWnd )
|
|
{
|
|
if( !IsWindow(hwnd) )
|
|
return TRUE; // assume full-screen maximized window
|
|
|
|
if( IsZoomed(hwnd) )
|
|
{
|
|
RECT rcContainer = {0};
|
|
if( !_GetMaximizedContainer( hwnd, &rcContainer ) )
|
|
return TRUE;
|
|
|
|
// determine whether the rect is contained in the screen rect
|
|
return _RectInRect( &rcContainer, prcWnd );
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
//
|
|
// _GetRawClassicCaptionHeight() -
|
|
//
|
|
// Using system metrics, computes the total height of the caption bar
|
|
// including edge and borders
|
|
//
|
|
inline int _GetRawClassicCaptionHeight( DWORD dwStyle, DWORD dwExStyle )
|
|
{
|
|
ASSERT(HAS_CAPTIONBAR(dwStyle)); // shouldn't be here without WS_CAPTION
|
|
return NcGetSystemMetrics(
|
|
TESTFLAG(dwExStyle, WS_EX_TOOLWINDOW ) ? SM_CYSMCAPTION : SM_CYCAPTION );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
//
|
|
// _GetSumClassicCaptionHeight() -
|
|
//
|
|
// Using system metrics, computes the total height of the caption bar
|
|
// including edge and borders
|
|
//
|
|
inline int _GetSumClassicCaptionHeight( DWORD dwStyle, DWORD dwExStyle )
|
|
{
|
|
ASSERT(HAS_CAPTIONBAR(dwStyle)); // shouldn't be here without WS_CAPTION
|
|
// Factor in window border width.
|
|
return _GetWindowBorders( dwStyle, dwExStyle) +
|
|
_GetRawClassicCaptionHeight( dwStyle, dwExStyle );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
//
|
|
// GetWindowBorders() - port from win32k, rtl\winmgr.c
|
|
//
|
|
// Computes window border dimensions based on style bits.
|
|
//
|
|
int _GetWindowBorders(LONG lStyle, DWORD dwExStyle )
|
|
{
|
|
int cBorders = 0;
|
|
|
|
//
|
|
// Is there a 3D border around the window?
|
|
//
|
|
if( TESTFLAG(dwExStyle, WS_EX_WINDOWEDGE) )
|
|
cBorders += 2;
|
|
else if ( TESTFLAG(dwExStyle, WS_EX_STATICEDGE) )
|
|
++cBorders;
|
|
|
|
//
|
|
// Is there a single flat border around the window? This is true for
|
|
// WS_BORDER, WS_DLGFRAME, and WS_EX_DLGMODALFRAME windows.
|
|
//
|
|
if( TESTFLAG(lStyle, WS_CAPTION) || TESTFLAG(dwExStyle, WS_EX_DLGMODALFRAME) )
|
|
++cBorders;
|
|
|
|
//
|
|
// Is there a sizing flat border around the window?
|
|
//
|
|
if( TESTFLAG(lStyle, WS_THICKFRAME) && !TESTFLAG(lStyle, WS_MINIMIZE) )
|
|
{
|
|
NONCLIENTMETRICS ncm;
|
|
cBorders += (NcGetNonclientMetrics( &ncm, FALSE ) ?
|
|
ncm.iBorderWidth : NcGetSystemMetrics( SM_CXBORDER ));
|
|
}
|
|
|
|
return(cBorders);
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Determines whether WS_EX_WINDOWEDGE should be assumed.
|
|
// Ripped from USER sources (rtl\winmgr.c)
|
|
BOOL _NeedsWindowEdgeStyle(DWORD dwStyle, DWORD dwExStyle )
|
|
{
|
|
BOOL fGetsWindowEdge = FALSE;
|
|
|
|
if (dwExStyle & WS_EX_DLGMODALFRAME)
|
|
fGetsWindowEdge = TRUE;
|
|
else if (dwExStyle & WS_EX_STATICEDGE)
|
|
fGetsWindowEdge = FALSE;
|
|
else if (dwStyle & WS_THICKFRAME)
|
|
fGetsWindowEdge = TRUE;
|
|
else switch (dwStyle & WS_CAPTION)
|
|
{
|
|
case WS_DLGFRAME:
|
|
fGetsWindowEdge = TRUE;
|
|
break;
|
|
case WS_CAPTION:
|
|
fGetsWindowEdge = TRUE; // PORTPORT: SHIMSHIM should be: = (RtlGetExpWinVer(hMod) > VER40)
|
|
// we will assume a new app; old apps are denied.
|
|
break;
|
|
}
|
|
|
|
return(fGetsWindowEdge);
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// _MNCanClose
|
|
//
|
|
// returns TRUE only if USER32 determines that the window can be closed
|
|
// (by checking its system menu items and their disabled state)
|
|
//
|
|
BOOL _MNCanClose(HWND hwnd)
|
|
{
|
|
LogEntryNC(L"_MNCanClose");
|
|
|
|
BOOL fRetVal = FALSE;
|
|
|
|
TITLEBARINFO tbi = {sizeof(tbi)};
|
|
|
|
//---- don't use GetSystemMenu() - has user handle leak issues ----
|
|
if (GetTitleBarInfo(hwnd, &tbi))
|
|
{
|
|
//---- mask out the good bits ----
|
|
DWORD dwVal = (tbi.rgstate[5] & (~(STATE_SYSTEM_PRESSED | STATE_SYSTEM_FOCUSABLE)));
|
|
fRetVal = (dwVal == 0); // only if no bad bits are left
|
|
}
|
|
|
|
if ( !fRetVal && TESTFLAG(GetWindowLong(hwnd, GWL_EXSTYLE), WS_EX_MDICHILD) )
|
|
{
|
|
HMENU hMenu = GetSystemMenu(hwnd, FALSE);
|
|
MENUITEMINFO menuInfo;
|
|
|
|
menuInfo.cbSize = sizeof(MENUITEMINFO);
|
|
menuInfo.fMask = MIIM_STATE;
|
|
if ( GetMenuItemInfo(hMenu, SC_CLOSE, FALSE, &menuInfo) )
|
|
{
|
|
fRetVal = !(menuInfo.fState & MFS_GRAYED) ? TRUE : FALSE;
|
|
}
|
|
}
|
|
|
|
LogExitNC(L"_MNCanClose");
|
|
return fRetVal;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::UpdateMDIFrameStuff( HWND hwndMDIClient, BOOL fSetMenu )
|
|
{
|
|
HWND hwndMDIActive = _MDIGetActive( hwndMDIClient, NULL );
|
|
|
|
// cache MDIClient, maximized M window handle
|
|
_hwndMDIClient = IsWindow(hwndMDIActive) ? hwndMDIClient : NULL;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL CALLBACK _FreshenThemeMetricsCB( HWND hwnd, LPARAM lParam )
|
|
{
|
|
CThemeWnd* pwnd = CThemeWnd::FromHwnd( hwnd );
|
|
if( VALID_THEMEWND(pwnd) )
|
|
{
|
|
pwnd->AddRef();
|
|
pwnd->GetNcWindowMetrics( NULL, NULL, NULL, NCWMF_RECOMPUTE );
|
|
pwnd->Release();
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL _IsMessageWindow( HWND hwnd )
|
|
{
|
|
// A window parented by HWND_MESSAGE has no UI and should not be themed.
|
|
static ATOM _atomMsgWnd = 0;
|
|
|
|
HWND hwndParent = (HWND)GetWindowLongPtr( hwnd, GWLP_HWNDPARENT );
|
|
if( hwndParent )
|
|
{
|
|
ATOM atomParent = (ATOM)GetClassLong( hwndParent, GCW_ATOM );
|
|
|
|
// have we seen the message window wndclass before?
|
|
if( _atomMsgWnd )
|
|
return (atomParent == _atomMsgWnd); // compare class atoms
|
|
|
|
// haven't seen a message window come through in this process,
|
|
// so compare class names.
|
|
WCHAR szClass[128];
|
|
if( GetClassNameW( hwndParent, szClass, ARRAYSIZE(szClass) ) )
|
|
{
|
|
if( 0 == AsciiStrCmpI( szClass, L"Message" ) )
|
|
{
|
|
_atomMsgWnd = atomParent;
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Retrieves MDI frame and/or MDICLIENT window for an MDI child window
|
|
HWND _MDIGetParent(
|
|
HWND hwnd, OUT OPTIONAL CThemeWnd** ppMdiFrame, OUT OPTIONAL HWND* phwndMDIClient )
|
|
{
|
|
if( ppMdiFrame ) *ppMdiFrame = NULL;
|
|
if( phwndMDIClient ) *phwndMDIClient = NULL;
|
|
|
|
if( TESTFLAG(GetWindowLong( hwnd, GWL_EXSTYLE ), WS_EX_MDICHILD) )
|
|
{
|
|
HWND hwndMDIClient = GetParent(hwnd);
|
|
if( IsWindow(hwndMDIClient) )
|
|
{
|
|
HWND hwndFrame = GetParent(hwndMDIClient);
|
|
if( IsWindow(hwndFrame) )
|
|
{
|
|
if( phwndMDIClient ) *phwndMDIClient = hwndMDIClient;
|
|
if( ppMdiFrame )
|
|
*ppMdiFrame = CThemeWnd::FromHwnd(hwndFrame);
|
|
|
|
return hwndFrame;
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HWND _FindMDIClient( HWND hwndFrame )
|
|
{
|
|
for( HWND hwndChild = GetWindow(hwndFrame, GW_CHILD); hwndChild != NULL;
|
|
hwndChild = GetNextWindow(hwndChild, GW_HWNDNEXT))
|
|
{
|
|
TCHAR szClass[48];
|
|
if( GetClassName(hwndChild, szClass, ARRAYSIZE(szClass)) )
|
|
{
|
|
if( 0 == lstrcmpi(szClass, TEXT("MDIClient")) )
|
|
{
|
|
return hwndChild;
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Handle MDI relative updating on WM_WINDOWPOSCHANGED
|
|
void _MDIUpdate( HWND hwnd, UINT uSwpFlags)
|
|
{
|
|
// Notify MDI frame if we became maximized, etc.
|
|
BOOL bIsClient = FALSE;
|
|
|
|
// Could be the MDI client, could be a MDI child
|
|
if (!(TESTFLAG(uSwpFlags, SWP_NOMOVE) && TESTFLAG(uSwpFlags, SWP_NOSIZE)))
|
|
{
|
|
bIsClient = _MDIClientUpdateChildren( hwnd );
|
|
}
|
|
if (!bIsClient)
|
|
{
|
|
_MDIChildUpdateParent( hwnd, FALSE );
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Post-WM_WINDOWPOSCHANGED processing for MDI client or children.
|
|
// We need to recompute each child when the MDI client moves.
|
|
BOOL _MDIClientUpdateChildren( HWND hwndMDIChildOrClient )
|
|
{
|
|
// Find if it's the MDI client window
|
|
HWND hWndChild = GetWindow(hwndMDIChildOrClient, GW_CHILD);
|
|
if (IsWindow(hWndChild) && TESTFLAG(GetWindowLong(hWndChild, GWL_EXSTYLE), WS_EX_MDICHILD))
|
|
{
|
|
// Yes it's the MDI client, refresh each MDI child's metrics
|
|
do
|
|
{
|
|
_FreshenThemeMetricsCB(hWndChild, NULL);
|
|
} while (NULL != (hWndChild = GetWindow(hWndChild, GW_HWNDNEXT)));
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Informs MDI frame that a child window may
|
|
void _MDIChildUpdateParent( HWND hwndMDIChild, BOOL fSetMenu )
|
|
{
|
|
CThemeWnd* pwndParent;
|
|
HWND hwndMDIClient;
|
|
|
|
if( _MDIGetParent( hwndMDIChild, &pwndParent, &hwndMDIClient ) &&
|
|
VALID_THEMEWND(pwndParent) )
|
|
{
|
|
pwndParent->UpdateMDIFrameStuff( hwndMDIClient, fSetMenu );
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Creates a replacement menu item bitmap for MDI frame window menubar
|
|
// buttons for maximized MDI child.
|
|
HRESULT _CreateMdiMenuItemBitmap(
|
|
IN HDC hdc,
|
|
IN HTHEME hTheme,
|
|
IN OUT SIZE* pSize,
|
|
IN OUT MENUITEMINFO* pmii )
|
|
{
|
|
WINDOWPARTS iPartId;
|
|
CLOSEBUTTONSTATES iStateId = CBS_NORMAL;
|
|
|
|
switch( (UINT)pmii->wID )
|
|
{
|
|
case SC_CLOSE:
|
|
iPartId = WP_MDICLOSEBUTTON;
|
|
break;
|
|
|
|
case SC_MINIMIZE:
|
|
iPartId = WP_MDIMINBUTTON;
|
|
break;
|
|
|
|
case SC_RESTORE:
|
|
iPartId = WP_MDIRESTOREBUTTON;
|
|
break;
|
|
|
|
default:
|
|
return E_INVALIDARG;
|
|
}
|
|
iStateId = TESTFLAG(pmii->fState, MFS_DISABLED) ? CBS_DISABLED : CBS_NORMAL;
|
|
|
|
if( NULL == pSize )
|
|
{
|
|
SIZE size;
|
|
size.cx = NcGetSystemMetrics(SM_CXMENUSIZE);
|
|
size.cy = NcGetSystemMetrics(SM_CYMENUSIZE);
|
|
pSize = &size;
|
|
}
|
|
|
|
return _CreateBackgroundBitmap( hdc, hTheme, iPartId, iStateId, pSize, &pmii->hbmpItem );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HRESULT _CreateBackgroundBitmap(
|
|
IN OPTIONAL HDC hdcCompatible,
|
|
IN HTHEME hTheme,
|
|
IN int iPartId,
|
|
IN int iStateId,
|
|
IN OUT LPSIZE pSize, // in: if cx <= 0 || cx <=0, assume truesize. out: background size
|
|
OUT HBITMAP* phbmOut )
|
|
{
|
|
ASSERT(hdcCompatible);
|
|
ASSERT(hTheme);
|
|
ASSERT(pSize);
|
|
ASSERT(phbmOut);
|
|
|
|
HRESULT hr = E_FAIL;
|
|
SIZE size;
|
|
|
|
*phbmOut = NULL;
|
|
size = *pSize;
|
|
pSize->cx = pSize->cy = 0;
|
|
|
|
// Create working DC.
|
|
HDC hdcMem = CreateCompatibleDC(hdcCompatible);
|
|
|
|
if( hdcMem != NULL )
|
|
{
|
|
// determine output size;
|
|
hr = (size.cx <= 0 || size.cy <= 0) ?
|
|
GetThemePartSize( hTheme, hdcCompatible, iPartId, iStateId, NULL, TS_TRUE, &size ) : S_OK;
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
HBITMAP hbmOut = CreateCompatibleBitmap( hdcCompatible, size.cx, size.cy );
|
|
if( hbmOut )
|
|
{
|
|
HBITMAP hbm0 = (HBITMAP)SelectObject(hdcMem, hbmOut);
|
|
RECT rcBkgnd;
|
|
SetRect( &rcBkgnd, 0, 0, size.cx, size.cy );
|
|
hr = NcDrawThemeBackground( hTheme, hdcMem, iPartId, iStateId, &rcBkgnd, 0 );
|
|
SelectObject( hdcMem, hbm0 );
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
*phbmOut = hbmOut;
|
|
pSize->cx = size.cx;
|
|
pSize->cy = size.cy;
|
|
}
|
|
else
|
|
{
|
|
SAFE_DELETE_GDIOBJ(hbmOut);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
DeleteDC(hdcMem);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// _ComputeElementOffset() - calculates specified offset for caption or button
|
|
// This could be generalized in the themeapi, yes? [scotthan]
|
|
void _ComputeElementOffset(
|
|
HTHEME hTheme,
|
|
int iPartId,
|
|
int iStateId,
|
|
LPCRECT prcBase,
|
|
POINT *pptOffset)
|
|
{
|
|
OFFSETTYPE eOffsetType;
|
|
if (FAILED(GetThemeEnumValue(hTheme, iPartId, iStateId, TMT_OFFSETTYPE, (int *)&eOffsetType)))
|
|
eOffsetType = OT_TOPLEFT; // default value
|
|
|
|
POINT ptOffset;
|
|
if (FAILED(GetThemePosition(hTheme, iPartId, iStateId, TMT_OFFSET, &ptOffset)))
|
|
{
|
|
ptOffset.x = 0;
|
|
ptOffset.y = 0;
|
|
}
|
|
|
|
RECT rcBase = *prcBase;
|
|
|
|
switch (eOffsetType)
|
|
{
|
|
case OT_TOPLEFT:
|
|
ptOffset.x += rcBase.left;
|
|
ptOffset.y += rcBase.top;
|
|
break;
|
|
|
|
case OT_TOPRIGHT:
|
|
ptOffset.x += rcBase.right;
|
|
ptOffset.y += rcBase.top;
|
|
break;
|
|
|
|
case OT_TOPMIDDLE:
|
|
ptOffset.x += (rcBase.left + rcBase.right)/2;
|
|
ptOffset.y += rcBase.top;
|
|
break;
|
|
|
|
case OT_BOTTOMLEFT:
|
|
ptOffset.x += rcBase.left;
|
|
ptOffset.y += rcBase.bottom;
|
|
break;
|
|
|
|
case OT_BOTTOMRIGHT:
|
|
ptOffset.x += rcBase.right;
|
|
ptOffset.y += rcBase.bottom;
|
|
break;
|
|
|
|
case OT_BOTTOMMIDDLE:
|
|
ptOffset.x += (rcBase.left + rcBase.right)/2;
|
|
ptOffset.y += rcBase.bottom;
|
|
break;
|
|
|
|
// Todo: handle the remaining cases:
|
|
case OT_LEFTOFCAPTION:
|
|
case OT_RIGHTOFCAPTION:
|
|
case OT_LEFTOFLASTBUTTON:
|
|
case OT_RIGHTOFLASTBUTTON:
|
|
case OT_ABOVELASTBUTTON:
|
|
case OT_BELOWLASTBUTTON:
|
|
ASSERT(FALSE);
|
|
break;
|
|
}
|
|
|
|
*pptOffset = ptOffset;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// _ComputeNcWindowStatus
|
|
//
|
|
// Assigns and translates window status bits to/in NCWNDMET block.
|
|
//
|
|
void _ComputeNcWindowStatus( IN HWND hwnd, IN DWORD dwStatus, IN OUT NCWNDMET* pncwm )
|
|
{
|
|
BOOL fActive = TESTFLAG( dwStatus, WS_ACTIVECAPTION );
|
|
|
|
if (fActive || !HAS_CAPTIONBAR(pncwm->dwStyle) )
|
|
{
|
|
pncwm->framestate = FS_ACTIVE;
|
|
}
|
|
else
|
|
{
|
|
pncwm->framestate = FS_INACTIVE;
|
|
}
|
|
|
|
if( HAS_CAPTIONBAR(pncwm->dwStyle) )
|
|
{
|
|
pncwm->rgbCaption = _GetNcCaptionTextColor( pncwm->framestate );
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------////
|
|
BOOL _GetWindowMetrics( HWND hwnd, IN OPTIONAL HWND hwndMDIActive, OUT NCWNDMET* pncwm )
|
|
{
|
|
WINDOWINFO wi;
|
|
wi.cbSize = sizeof(wi);
|
|
if( GetWindowInfo( hwnd, &wi ) )
|
|
{
|
|
pncwm->dwStyle = wi.dwStyle;
|
|
pncwm->dwExStyle = wi.dwExStyle;
|
|
pncwm->rcS0[NCRC_WINDOW] = wi.rcWindow;
|
|
pncwm->rcS0[NCRC_CLIENT] = wi.rcClient;
|
|
|
|
pncwm->fMin = IsIconic(hwnd);
|
|
pncwm->fMaxed = IsZoomed(hwnd);
|
|
pncwm->fFullMaxed = pncwm->fMaxed ? _IsFullMaximized(hwnd, &wi.rcWindow) : FALSE;
|
|
|
|
pncwm->dwWindowStatus = wi.dwWindowStatus;
|
|
|
|
|
|
// if this window is the active MDI child and is owned by the foreground window
|
|
// (which may not be the case if a popup, for example, is foremost), then
|
|
// fix up the status bit.
|
|
if( hwnd == hwndMDIActive )
|
|
{
|
|
HWND hwndFore = GetForegroundWindow();
|
|
if( IsChild(hwndFore, hwndMDIActive) )
|
|
{
|
|
pncwm->dwWindowStatus = WS_ACTIVECAPTION;
|
|
}
|
|
}
|
|
|
|
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL _ShouldAssignFrameRgn(
|
|
IN const NCWNDMET* pncwm,
|
|
IN const NCTHEMEMET& nctm )
|
|
{
|
|
if( TESTFLAG( CThemeWnd::EvaluateStyle(pncwm->dwStyle, pncwm->dwExStyle), TWCF_FRAME|TWCF_TOOLFRAME) )
|
|
{
|
|
// always need window region for maximized windows.
|
|
if( pncwm->fFullMaxed )
|
|
return TRUE;
|
|
|
|
// otherwise, need region only if the background is transparent
|
|
for( int i = 0; i < ARRAYSIZE( pncwm->rgframeparts ); i++ )
|
|
{
|
|
if( _IsNcPartTransparent( pncwm->rgframeparts[i], nctm ) )
|
|
return TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL _IsNcPartTransparent( WINDOWPARTS part, const NCTHEMEMET& nctm )
|
|
{
|
|
#define GET_NCTRANSPARENCY(part,field) \
|
|
case part: return nctm.nct.##field
|
|
|
|
switch(part)
|
|
{
|
|
GET_NCTRANSPARENCY(WP_CAPTION, fCaption);
|
|
GET_NCTRANSPARENCY(WP_SMALLCAPTION, fCaption);
|
|
GET_NCTRANSPARENCY(WP_MINCAPTION, fMinCaption);
|
|
GET_NCTRANSPARENCY(WP_SMALLMINCAPTION, fSmallMinCaption);
|
|
GET_NCTRANSPARENCY(WP_MAXCAPTION, fMaxCaption);
|
|
GET_NCTRANSPARENCY(WP_SMALLMAXCAPTION, fSmallMaxCaption);
|
|
GET_NCTRANSPARENCY(WP_FRAMELEFT, fFrameLeft);
|
|
GET_NCTRANSPARENCY(WP_FRAMERIGHT, fFrameRight);
|
|
GET_NCTRANSPARENCY(WP_FRAMEBOTTOM, fFrameBottom);
|
|
GET_NCTRANSPARENCY(WP_SMALLFRAMELEFT, fSmFrameLeft);
|
|
GET_NCTRANSPARENCY(WP_SMALLFRAMERIGHT, fSmFrameRight);
|
|
GET_NCTRANSPARENCY(WP_SMALLFRAMEBOTTOM, fSmFrameBottom);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL _ComputeNcPartTransparency( HTHEME hTheme, IN OUT NCTHEMEMET* pnctm )
|
|
{
|
|
#define TEST_NCTRANSPARENCY(part) IsThemePartDefined(hTheme,part,0) ? \
|
|
IsThemeBackgroundPartiallyTransparent(hTheme,part,FS_ACTIVE) : FALSE;
|
|
|
|
pnctm->nct.fCaption = TEST_NCTRANSPARENCY(WP_CAPTION);
|
|
pnctm->nct.fSmallCaption = TEST_NCTRANSPARENCY(WP_SMALLCAPTION);
|
|
pnctm->nct.fMinCaption = TEST_NCTRANSPARENCY(WP_MINCAPTION);
|
|
pnctm->nct.fSmallMinCaption = TEST_NCTRANSPARENCY(WP_SMALLMINCAPTION);
|
|
pnctm->nct.fMaxCaption = TEST_NCTRANSPARENCY(WP_MAXCAPTION);
|
|
pnctm->nct.fSmallMaxCaption = TEST_NCTRANSPARENCY(WP_SMALLMAXCAPTION);
|
|
|
|
pnctm->nct.fFrameLeft = TEST_NCTRANSPARENCY(WP_FRAMELEFT);
|
|
pnctm->nct.fFrameRight = TEST_NCTRANSPARENCY(WP_FRAMERIGHT);
|
|
pnctm->nct.fFrameBottom = TEST_NCTRANSPARENCY(WP_FRAMEBOTTOM);
|
|
pnctm->nct.fSmFrameLeft = TEST_NCTRANSPARENCY(WP_SMALLFRAMELEFT);
|
|
pnctm->nct.fSmFrameRight = TEST_NCTRANSPARENCY(WP_SMALLFRAMERIGHT);
|
|
pnctm->nct.fSmFrameBottom = TEST_NCTRANSPARENCY(WP_SMALLFRAMEBOTTOM);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// NCTHEMEMET implementation
|
|
//-------------------------------------------------------------------------//
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL GetCurrentNcThemeMetrics( OUT NCTHEMEMET* pnctm )
|
|
{
|
|
*pnctm = _nctmCurrent;
|
|
return IsValidNcThemeMetrics( pnctm );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HTHEME GetCurrentNcThemeHandle()
|
|
{
|
|
return _nctmCurrent.hTheme;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void InitNcThemeMetrics( NCTHEMEMET* pnctm )
|
|
{
|
|
if( !pnctm )
|
|
pnctm = &_nctmCurrent;
|
|
|
|
ZeroMemory( pnctm, sizeof(*pnctm) );
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
void ClearNcThemeMetrics( NCTHEMEMET* pnctm )
|
|
{
|
|
if( !pnctm )
|
|
pnctm = &_nctmCurrent;
|
|
|
|
//---- minimize THREAD-UNSAFE access to _nctmCurrent by ----
|
|
//---- NULL-ing out the hTheme type members as soon as ----
|
|
//---- they are closed ----
|
|
|
|
if( pnctm->hTheme )
|
|
{
|
|
CloseThemeData( pnctm->hTheme );
|
|
pnctm->hTheme = NULL;
|
|
}
|
|
|
|
if( pnctm->hThemeTab )
|
|
{
|
|
CloseThemeData( pnctm->hThemeTab );
|
|
pnctm->hThemeTab = NULL;
|
|
}
|
|
|
|
SAFE_DELETE_GDIOBJ( pnctm->hbmTabDialog );
|
|
SAFE_DELETE_GDIOBJ( pnctm->hbrTabDialog );
|
|
|
|
InitNcThemeMetrics( pnctm );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Computes process-global, per-theme metrics for the nonclient area theme.
|
|
HRESULT AcquireNcThemeMetrics()
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
EnterCriticalSection( &_csThemeMet );
|
|
|
|
ClearNcThemeMetrics( &_nctmCurrent );
|
|
NcGetNonclientMetrics( NULL, FALSE );
|
|
hr = _LoadNcThemeMetrics(NULL, &_nctmCurrent);
|
|
|
|
LeaveCriticalSection( &_csThemeMet );
|
|
|
|
Log(LOG_TMCHANGE, L"AcquireNcThemeMetrics: got hTheme=0x%x", _nctmCurrent.hTheme);
|
|
|
|
return hr;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Computes and/or loads per-theme (as opposed to per-window)
|
|
// system metrics and resources not managed by the theme manager.
|
|
//
|
|
// Called by _LoadNcThemeMetrics
|
|
HRESULT _LoadNcThemeSysMetrics( HWND hwnd, IN OUT NCTHEMEMET* pnctm )
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
ASSERT(pnctm);
|
|
|
|
// grab system metrics for nonclient area.
|
|
NONCLIENTMETRICS ncm = {0};
|
|
ncm.cbSize = sizeof(ncm);
|
|
if( NcGetNonclientMetrics( &ncm, FALSE ) )
|
|
{
|
|
#ifdef THEMED_NCBTNMETRICS
|
|
_GetClassicNcBtnMetrics( NULL, NULL, FALSE, TRUE );
|
|
#endif THEMED_NCBTNMETRICS
|
|
|
|
hr = S_OK;
|
|
|
|
// Establish minimized window size
|
|
if( 0 >= pnctm->sizeMinimized.cx )
|
|
pnctm->sizeMinimized.cx = NcGetSystemMetrics( SM_CXMINIMIZED );
|
|
if( 0 >= pnctm->sizeMinimized.cy )
|
|
pnctm->sizeMinimized.cy = NcGetSystemMetrics( SM_CYMINIMIZED );
|
|
}
|
|
else
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
if( SUCCEEDED(hr) )
|
|
hr = E_FAIL;
|
|
}
|
|
|
|
// Maximized caption height or width
|
|
pnctm->cyMaxCaption = _GetRawClassicCaptionHeight( WS_CAPTION|WS_OVERLAPPED, 0 );
|
|
|
|
return hr;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Computes and/or loads per-theme (as opposed to per-window)
|
|
// metrics and resources not managed by the theme manager
|
|
HRESULT _LoadNcThemeMetrics( HWND hwnd, NCTHEMEMET* pnctm )
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
|
|
// Initialize incoming NCTHEMEMET:
|
|
if( pnctm )
|
|
{
|
|
InitNcThemeMetrics( pnctm );
|
|
|
|
HTHEME hTheme = ::OpenNcThemeData( hwnd, L"Window" );
|
|
if( hTheme )
|
|
{
|
|
pnctm->hTheme = hTheme;
|
|
|
|
// determine transparency for each frame part.
|
|
_ComputeNcPartTransparency(hTheme, pnctm);
|
|
|
|
// menubar pixels not accounted for by CalcMenuBar or PaintMenuBar
|
|
pnctm->dyMenuBar = NcGetSystemMetrics(SM_CYMENU) - NcGetSystemMetrics(SM_CYMENUSIZE);
|
|
|
|
// normal caption margins
|
|
if( FAILED( GetThemeMargins( hTheme, NULL, WP_CAPTION, CS_ACTIVE, TMT_CAPTIONMARGINS,
|
|
NULL, &pnctm->marCaptionText )) )
|
|
{
|
|
FillMemory( &pnctm->marCaptionText, sizeof(pnctm->marCaptionText), 0 );
|
|
}
|
|
|
|
// maximized caption margins
|
|
if( FAILED( GetThemeMargins( hTheme, NULL, WP_MAXCAPTION, CS_ACTIVE, TMT_CAPTIONMARGINS,
|
|
NULL, &pnctm->marMaxCaptionText )) )
|
|
{
|
|
FillMemory( &pnctm->marMaxCaptionText, sizeof(pnctm->marMaxCaptionText), 0 );
|
|
}
|
|
|
|
// minimized caption margins
|
|
if( FAILED( GetThemeMargins( hTheme, NULL, WP_MINCAPTION, CS_ACTIVE, TMT_CAPTIONMARGINS,
|
|
NULL, &pnctm->marMinCaptionText )) )
|
|
{
|
|
FillMemory( &pnctm->marMinCaptionText, sizeof(pnctm->marMinCaptionText), 0 );
|
|
}
|
|
|
|
|
|
// dynamically resizing small (toolframe) caption margins
|
|
if( FAILED( GetThemeMargins( hTheme, NULL, WP_SMALLCAPTION, CS_ACTIVE, TMT_CAPTIONMARGINS,
|
|
NULL, &pnctm->marSmCaptionText )) )
|
|
{
|
|
FillMemory( &pnctm->marSmCaptionText, sizeof(pnctm->marSmCaptionText), 0 );
|
|
}
|
|
|
|
// caption and frame resizing border hittest template parts
|
|
pnctm->fCapSizingTemplate = IsThemePartDefined( hTheme, WP_CAPTIONSIZINGTEMPLATE, 0);
|
|
pnctm->fLeftSizingTemplate = IsThemePartDefined( hTheme, WP_FRAMELEFTSIZINGTEMPLATE, 0);
|
|
pnctm->fRightSizingTemplate = IsThemePartDefined( hTheme, WP_FRAMERIGHTSIZINGTEMPLATE, 0);
|
|
pnctm->fBottomSizingTemplate = IsThemePartDefined( hTheme, WP_FRAMEBOTTOMSIZINGTEMPLATE, 0);
|
|
|
|
// toolwindow caption and frame resizing border hittest template parts
|
|
pnctm->fSmCapSizingTemplate = IsThemePartDefined( hTheme, WP_SMALLCAPTIONSIZINGTEMPLATE, 0);
|
|
pnctm->fSmLeftSizingTemplate = IsThemePartDefined( hTheme, WP_SMALLFRAMELEFTSIZINGTEMPLATE, 0);
|
|
pnctm->fSmRightSizingTemplate = IsThemePartDefined( hTheme, WP_SMALLFRAMERIGHTSIZINGTEMPLATE, 0);
|
|
pnctm->fSmBottomSizingTemplate = IsThemePartDefined( hTheme, WP_SMALLFRAMEBOTTOMSIZINGTEMPLATE, 0);
|
|
|
|
// Minimized window size.
|
|
// If this is a truesize image, honor its dimensions; otherwise use
|
|
// width, height properties. Fall back on system metrics.
|
|
SIZINGTYPE st = ST_TRUESIZE;
|
|
hr = GetThemeInt( hTheme, WP_MINCAPTION, FS_ACTIVE, TMT_SIZINGTYPE, (int*)&st );
|
|
|
|
if( ST_TRUESIZE == st )
|
|
{
|
|
hr = GetThemePartSize( hTheme, NULL, WP_MINCAPTION, FS_ACTIVE, NULL,
|
|
TS_TRUE, &pnctm->sizeMinimized );
|
|
|
|
if( FAILED(hr) )
|
|
{
|
|
GetThemeMetric( hTheme, NULL, WP_MINCAPTION, FS_ACTIVE, TMT_WIDTH,
|
|
(int*)&pnctm->sizeMinimized.cx );
|
|
GetThemeMetric( hTheme, NULL, WP_MINCAPTION, FS_ACTIVE, TMT_HEIGHT,
|
|
(int*)&pnctm->sizeMinimized.cy );
|
|
}
|
|
}
|
|
|
|
// -- normal nonclient button size.
|
|
int cy = NcGetSystemMetrics( SM_CYSIZE );
|
|
hr = GetThemePartSize( pnctm->hTheme, NULL, WP_CLOSEBUTTON, 0, NULL, TS_TRUE, &pnctm->sizeBtn );
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
pnctm->theme_sysmets.cxBtn = MulDiv( cy, pnctm->sizeBtn.cx, pnctm->sizeBtn.cy );
|
|
}
|
|
else
|
|
{
|
|
pnctm->theme_sysmets.cxBtn =
|
|
pnctm->sizeBtn.cx = NcGetSystemMetrics( SM_CXSIZE );
|
|
|
|
pnctm->sizeBtn.cy = cy;
|
|
}
|
|
|
|
// -- toolframe nonclient button size.
|
|
cy = NcGetSystemMetrics( SM_CYSMSIZE );
|
|
hr = GetThemePartSize( pnctm->hTheme, NULL, WP_SMALLCLOSEBUTTON, 0, NULL, TS_TRUE, &pnctm->sizeSmBtn );
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
pnctm->theme_sysmets.cxSmBtn = MulDiv( cy, pnctm->sizeSmBtn.cx, pnctm->sizeSmBtn.cy );
|
|
}
|
|
else
|
|
{
|
|
pnctm->theme_sysmets.cxSmBtn =
|
|
pnctm->sizeSmBtn.cx = NcGetSystemMetrics( SM_CXSMSIZE );
|
|
|
|
pnctm->sizeSmBtn.cy = cy;
|
|
}
|
|
|
|
// -- validate sysmet hook values
|
|
pnctm->theme_sysmets.fValid = TRUE;
|
|
|
|
// dialog background for dialogs parented by PROPSHEETs or
|
|
// specifically stamped via EnableThemeDialogTexture to match the tab control background.
|
|
//
|
|
// We need to open the tab control's theme so that we can get the background of tab dialogs
|
|
// We can't dynamically load this because of how this cache is set up: It's all or nothing.
|
|
pnctm->hThemeTab = ::OpenThemeData(hwnd, L"Tab");
|
|
_GetBrushesForPart(pnctm->hThemeTab, TABP_BODY, &pnctm->hbmTabDialog, &pnctm->hbrTabDialog);
|
|
|
|
hr = _LoadNcThemeSysMetrics( hwnd, pnctm );
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL IsValidNcThemeMetrics( NCTHEMEMET* pnctm )
|
|
{
|
|
return pnctm->hTheme != NULL;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// THREADWINDOW implementation
|
|
//-------------------------------------------------------------------------//
|
|
//
|
|
// Note: this is a fixed length array of threads-window mappings.
|
|
// We'll use this to keep track of the threads processing a certain message
|
|
//
|
|
// Thread local storage would be better suited to the task, but we
|
|
// learned early on that the unique load/unload situation of uxtheme
|
|
// causes us to miss DLL_THREAD_DETACH in some scenarios, which would mean
|
|
// leaking the TLS.
|
|
//
|
|
|
|
typedef struct _THREADWINDOW
|
|
{
|
|
DWORD dwThread;
|
|
HWND hwnd;
|
|
|
|
} THREADWINDOW;
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCPAINT tracking:
|
|
THREADWINDOW _rgtwNcPaint[16] = {0}; // threads processing NCPAINT in this process
|
|
int _cNcPaintWnd = 0; // count of threads processing NCPAINT in this process
|
|
CRITICAL_SECTION _csNcPaint; // serializes access to _rgtwNcPaint
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void NcPaintWindow_Add( HWND hwnd )
|
|
{
|
|
EnterCriticalSection( &_csNcPaint );
|
|
for( int i = 0; i < ARRAYSIZE(_rgtwNcPaint); i++ )
|
|
{
|
|
if( 0 == _rgtwNcPaint[i].dwThread )
|
|
{
|
|
_rgtwNcPaint[i].dwThread = GetCurrentThreadId();
|
|
_rgtwNcPaint[i].hwnd = hwnd;
|
|
_cNcPaintWnd++;
|
|
}
|
|
}
|
|
LeaveCriticalSection( &_csNcPaint );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void NcPaintWindow_Remove()
|
|
{
|
|
if( _cNcPaintWnd )
|
|
{
|
|
DWORD dwThread = GetCurrentThreadId();
|
|
EnterCriticalSection( &_csNcPaint );
|
|
for( int i = 0; i < ARRAYSIZE(_rgtwNcPaint); i++ )
|
|
{
|
|
if( dwThread == _rgtwNcPaint[i].dwThread )
|
|
{
|
|
_rgtwNcPaint[i].dwThread = 0;
|
|
_rgtwNcPaint[i].hwnd = 0;
|
|
_cNcPaintWnd--;
|
|
break;
|
|
}
|
|
}
|
|
LeaveCriticalSection( &_csNcPaint );
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HWND NcPaintWindow_Find()
|
|
{
|
|
HWND hwnd = NULL;
|
|
|
|
if( _cNcPaintWnd )
|
|
{
|
|
DWORD dwThread = GetCurrentThreadId();
|
|
|
|
EnterCriticalSection( &_csNcPaint );
|
|
for( int i = 0; i < ARRAYSIZE(_rgtwNcPaint); i++ )
|
|
{
|
|
if( dwThread == _rgtwNcPaint[i].dwThread )
|
|
{
|
|
hwnd = _rgtwNcPaint[i].hwnd;
|
|
break;
|
|
}
|
|
}
|
|
LeaveCriticalSection( &_csNcPaint );
|
|
}
|
|
return hwnd;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd implementation
|
|
//-------------------------------------------------------------------------//
|
|
|
|
//-------------------------------------------------------------------------//
|
|
LONG CThemeWnd::_cObj = 0;
|
|
|
|
//-------------------------------------------------------------------------//
|
|
CThemeWnd::CThemeWnd()
|
|
: _hwnd(NULL),
|
|
_hTheme(NULL),
|
|
_dwRenderedNcParts(0),
|
|
_hwndMDIClient(NULL),
|
|
_hAppIcon(NULL),
|
|
_hrgnWnd(NULL),
|
|
_fClassFlags(0),
|
|
_fDirtyFrameRgn(0),
|
|
_fFrameThemed(FALSE),
|
|
_fThemedMDIBtns(FALSE),
|
|
_pMdiBtns(NULL),
|
|
_fAssigningFrameRgn(FALSE),
|
|
_fAssignedFrameRgn(FALSE),
|
|
_fSuppressStyleMsgs(FALSE),
|
|
_fInThemeSettingChange(FALSE),
|
|
_fDetached(FALSE),
|
|
_fCritSectsInit(FALSE),
|
|
_dwRevokeFlags(0),
|
|
_cLockRedraw(0),
|
|
_cNcPaint(0),
|
|
_cNcThemePaint(0),
|
|
_htHot(HTERROR),
|
|
_fProcessedEraseBk(0),
|
|
#ifdef LAME_BUTTON
|
|
_hFontLame(NULL),
|
|
#endif // LAME_BUTTON
|
|
#ifdef DEBUG_THEMEWND_DESTRUCTOR
|
|
_fDestructed(FALSE),
|
|
_fDeleteCritsec(FALSE),
|
|
#endif DEBUG_THEMEWND_DESTRUCTOR
|
|
_cRef(1)
|
|
{
|
|
InterlockedIncrement( &_cObj );
|
|
|
|
// set object validation signature tags
|
|
strcpy(_szHead, SIG_CTHEMEWND_HEAD);
|
|
strcpy(_szTail, SIG_CTHEMEWND_TAIL);
|
|
|
|
// cached subregion arrays
|
|
ZeroMemory( _rghrgnParts, sizeof(_rghrgnParts) );
|
|
ZeroMemory( _rghrgnSizingTemplates, sizeof(_rghrgnSizingTemplates) );
|
|
|
|
// initialize add'l structures.
|
|
InitWindowMetrics();
|
|
FillMemory(&_sizeRgn, sizeof(_sizeRgn), 0xFF);
|
|
|
|
#ifdef DEBUG
|
|
*_szCaption = *_szWndClass = 0;
|
|
#endif DEBUG
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
CThemeWnd::~CThemeWnd()
|
|
{
|
|
_CloseTheme();
|
|
_FreeRegionHandles();
|
|
UnloadMdiBtns();
|
|
ClearLameResources();
|
|
|
|
if( _fCritSectsInit )
|
|
{
|
|
DeleteCriticalSection( &_cswm );
|
|
#ifdef DEBUG_THEMEWND_DESTRUCTOR
|
|
_fDeleteCritsec = TRUE;
|
|
#endif DEBUG_THEMEWND_DESTRUCTOR
|
|
}
|
|
InterlockedDecrement( &_cObj );
|
|
|
|
#ifdef DEBUG_THEMEWND_DESTRUCTOR
|
|
_fDestructed = TRUE; // destructor has been called.
|
|
#endif DEBUG_THEMEWND_DESTRUCTOR
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::_CloseTheme()
|
|
{
|
|
if( _hTheme )
|
|
{
|
|
CloseThemeData(_hTheme);
|
|
_hTheme = NULL;
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
LONG CThemeWnd::AddRef()
|
|
{
|
|
return InterlockedIncrement( &_cRef );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
LONG CThemeWnd::Release()
|
|
{
|
|
LONG cRef = InterlockedDecrement( &_cRef );
|
|
|
|
if( 0 == cRef )
|
|
{
|
|
if (_hwnd)
|
|
{
|
|
//---- check if last window of app ----
|
|
ShutDownCheck(_hwnd);
|
|
}
|
|
|
|
//Log(LOG_RFBUG, L"DELETING CThemeWnd=0x%08x", this);
|
|
delete this;
|
|
}
|
|
return cRef;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
ULONG CThemeWnd::EvaluateWindowStyle( HWND hwnd )
|
|
{
|
|
ULONG dwStyle = GetWindowLong( hwnd, GWL_STYLE );
|
|
ULONG dwExStyle = GetWindowLong( hwnd, GWL_EXSTYLE );
|
|
|
|
return EvaluateStyle( dwStyle, dwExStyle );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::EvaluateStyle() - determines appropriate theming flags for the
|
|
// specified window style bits.
|
|
ULONG CThemeWnd::EvaluateStyle( DWORD dwStyle, DWORD dwExStyle )
|
|
{
|
|
ULONG fClassFlags = 0;
|
|
|
|
//--- frame check ---
|
|
if( HAS_CAPTIONBAR(dwStyle) )
|
|
{
|
|
fClassFlags |=
|
|
(TESTFLAG(dwExStyle, WS_EX_TOOLWINDOW) ? TWCF_TOOLFRAME : TWCF_FRAME );
|
|
}
|
|
|
|
//--- client edge check ---
|
|
if( TESTFLAG(dwExStyle, WS_EX_CLIENTEDGE) )
|
|
fClassFlags |= TWCF_CLIENTEDGE;
|
|
|
|
//--- scrollbar check ---
|
|
if( TESTFLAG(dwStyle, WS_HSCROLL|WS_VSCROLL) )
|
|
fClassFlags |= TWCF_SCROLLBARS;
|
|
|
|
return fClassFlags;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::_EvaluateExclusions() - determines special-case per-window exclusions
|
|
ULONG CThemeWnd::_EvaluateExclusions( HWND hwnd, NCEVALUATE* pnce )
|
|
{
|
|
// Windows parented by HWND_MESSAGE should not be themed..
|
|
if( _IsMessageWindow(hwnd) )
|
|
{
|
|
pnce->fExile = TRUE;
|
|
return 0L;
|
|
}
|
|
|
|
TCHAR szWndClass[128];
|
|
*szWndClass = 0;
|
|
|
|
|
|
if( TESTFLAG(pnce->fClassFlags, (TWCF_FRAME|TWCF_TOOLFRAME)) )
|
|
{
|
|
do
|
|
{
|
|
if( !pnce->fIgnoreWndRgn )
|
|
{
|
|
//--- Complex region check on frame
|
|
RECT rcRgn = {0};
|
|
int nRgn = GetWindowRgnBox( hwnd, &rcRgn );
|
|
if( COMPLEXREGION == nRgn || SIMPLEREGION == nRgn )
|
|
{
|
|
pnce->fClassFlags &= ~TWCF_FRAME;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// SHIMSHIM [scotthan]:
|
|
#ifndef __NO_APPHACKS__
|
|
// Check for excluded window classes.
|
|
static LPCWSTR _rgExcludedClassesW[] =
|
|
{
|
|
L"MsoCommandBar", // Outlook's custom combobox control.
|
|
// (122225) In OnOwpPostCreate we call SetWindowPos which causes
|
|
// a WM_WINDOWPOSCHANGING to be sent to the control. However
|
|
// the controls isn't ready to begin accepting messages and
|
|
// the following error message is display:
|
|
//
|
|
// Runtime Error!
|
|
// Program: Outlook.exe
|
|
// R6025 - pure virtual function call
|
|
|
|
L"Exceed", // 150248: Hummingbird Exceed 6.xx
|
|
// The application's main window class name, a hidden window
|
|
// whose only purpose is to appear in the task bar in order to handle
|
|
// his context menu. The ExceedWndProc AVs when themed due to the
|
|
// additional messages generated in OnOwpPostCreate.
|
|
|
|
//---- winlogoon hidden windows ----
|
|
L"NDDEAgnt", // on private desktop
|
|
L"MM Notify Callback", // on private desktop
|
|
L"SAS window class", // on private desktop
|
|
};
|
|
|
|
if( GetClassNameW( hwnd, szWndClass, ARRAYSIZE(szWndClass) ) &&
|
|
AsciiScanStringList( szWndClass, _rgExcludedClassesW,
|
|
ARRAYSIZE(_rgExcludedClassesW), TRUE ) )
|
|
{
|
|
pnce->fClassFlags &= ~TWCF_FRAME;
|
|
pnce->fExile = TRUE;
|
|
break;
|
|
}
|
|
#endif __NO_APPHACKS__
|
|
|
|
} while(0);
|
|
}
|
|
|
|
// Some applications (MsDev) create scrollbar controls and incorrectly include
|
|
// WS_[V|H]SCROLL style bits causing us to think they are non-client scrolls.
|
|
// See #204191.
|
|
if( TESTFLAG(pnce->fClassFlags, TWCF_SCROLLBARS) )
|
|
{
|
|
if( !*szWndClass && GetClassName( hwnd, szWndClass, ARRAYSIZE(szWndClass) ) )
|
|
{
|
|
if( 0 == AsciiStrCmpI(szWndClass, L"scrollbar") )
|
|
pnce->fClassFlags &= ~TWCF_SCROLLBARS;
|
|
}
|
|
}
|
|
|
|
return pnce->fClassFlags;
|
|
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::_Evaluate() - determines appropriate theming flags for the
|
|
// specified window.
|
|
ULONG CThemeWnd::_Evaluate( HWND hwnd, NCEVALUATE* pnce )
|
|
{
|
|
pnce->fClassFlags = 0;
|
|
pnce->dwStyle = GetWindowLong( hwnd, GWL_STYLE );
|
|
pnce->dwExStyle = GetWindowLong( hwnd, GWL_EXSTYLE );
|
|
|
|
if( GetClassLong( hwnd, GCW_ATOM ) == (DWORD)(DWORD_PTR)WC_DIALOG )
|
|
{
|
|
pnce->fClassFlags |= TWCF_DIALOG;
|
|
}
|
|
|
|
|
|
#ifdef DEBUG
|
|
//--- dialog check ---
|
|
if( TESTFLAG( pnce->fClassFlags, TWCF_DIALOG ) )
|
|
{
|
|
TCHAR szWndClass[96];
|
|
if( !GetClassNameW( hwnd, szWndClass, ARRAYSIZE(szWndClass) ) )
|
|
return 0;
|
|
ASSERT(0 == lstrcmpW(szWndClass, DLGWNDCLASSNAMEW));
|
|
}
|
|
#endif DEBUG
|
|
|
|
pnce->fClassFlags |= EvaluateStyle( pnce->dwStyle, pnce->dwExStyle );
|
|
|
|
if( pnce->fClassFlags )
|
|
{
|
|
pnce->fClassFlags = _EvaluateExclusions( hwnd, pnce );
|
|
}
|
|
|
|
return pnce->fClassFlags;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Retrieves the address of the CThemeWnd object instance from the
|
|
// indicated window.
|
|
CThemeWnd* CThemeWnd::FromHwnd( HWND hwnd )
|
|
{
|
|
CThemeWnd *pwnd = NULL;
|
|
|
|
if( IsWindow(hwnd) )
|
|
{
|
|
if( g_dwProcessId )
|
|
{
|
|
DWORD dwPid = 0;
|
|
GetWindowThreadProcessId( hwnd, &dwPid );
|
|
if( dwPid == g_dwProcessId )
|
|
{
|
|
pwnd = (CThemeWnd*)GetProp( hwnd, MAKEINTATOM(GetThemeAtom(THEMEATOM_NONCLIENT)) );
|
|
|
|
if ( VALID_THEMEWND(pwnd) )
|
|
{
|
|
// verify this is a valid CThemeWnd object pointer
|
|
if ( IsBadReadPtr(pwnd, sizeof(CThemeWnd)) ||
|
|
(memcmp(pwnd->_szHead, SIG_CTHEMEWND_HEAD, ARRAYSIZE(pwnd->_szHead)) != 0) ||
|
|
(memcmp(pwnd->_szTail, SIG_CTHEMEWND_TAIL, ARRAYSIZE(pwnd->_szTail)) != 0) )
|
|
{
|
|
pwnd = THEMEWND_REJECT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return pwnd;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// retrieves CThemeWnd instance from window or ancestors.
|
|
CThemeWnd* CThemeWnd::FromHdc( HDC hdc, int cAncestors )
|
|
{
|
|
HWND hwnd = NULL;
|
|
|
|
for( hwnd = WindowFromDC(hdc);
|
|
cAncestors >=0 && IsWindow(hwnd);
|
|
cAncestors--, hwnd = GetParent(hwnd) )
|
|
{
|
|
CThemeWnd* pwnd = FromHwnd(hwnd);
|
|
if( VALID_THEMEWND(pwnd) )
|
|
{
|
|
return pwnd;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Static wrapper: attaches a CThemeWnd instance to the specified window.
|
|
CThemeWnd* CThemeWnd::Attach( HWND hwnd, IN OUT OPTIONAL NCEVALUATE* pnce )
|
|
{
|
|
LogEntryNC(L"Attach");
|
|
|
|
#ifdef LOGGING
|
|
//---- remember first window (app window) hooked for ShutDownCheck() ----
|
|
//---- this is only for BoundsChecker (tm) runs for finding leaks ----
|
|
if (! g_hwndFirstHooked)
|
|
{
|
|
if ((GetMenu(hwnd)) && (! GetParent(hwnd)))
|
|
g_hwndFirstHooked = hwnd;
|
|
}
|
|
#endif
|
|
|
|
CThemeWnd* pwnd = NULL;
|
|
|
|
// Note: Important not to do anything here that causes
|
|
// a window message to be posted or sent to the window: could
|
|
// mean tying ourselves up in a recursive knot (see _ThemeDefWindowProc).
|
|
|
|
pwnd = FromHwnd( hwnd );
|
|
|
|
if( NULL == pwnd )
|
|
{
|
|
HTHEME hTheme = NULL;
|
|
NCEVALUATE nce;
|
|
|
|
// copy any IN params from NCEVALUATE struct
|
|
if( !pnce )
|
|
{
|
|
ZeroMemory(&nce, sizeof(nce));
|
|
pnce = &nce;
|
|
}
|
|
|
|
ULONG ulTargetFlags = _Evaluate( hwnd, pnce );
|
|
|
|
// Anything worth theming?
|
|
if( TESTFLAG(ulTargetFlags, TWCF_NCTHEMETARGETMASK) )
|
|
{
|
|
hTheme = _AcquireThemeHandle( hwnd, &ulTargetFlags );
|
|
if( NULL == hTheme )
|
|
{
|
|
Fail(hwnd);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// reject windows with untargeted a
|
|
Reject(hwnd, pnce->fExile);
|
|
}
|
|
|
|
if( NULL != hTheme )
|
|
{
|
|
// Yes, create a real nctheme object for the window
|
|
if( (pwnd = new CThemeWnd) != NULL )
|
|
{
|
|
if( !pwnd->_AttachInstance( hwnd, hTheme, ulTargetFlags, pnce->pvWndCompat ) )
|
|
{
|
|
pwnd->Release();
|
|
pwnd = NULL;
|
|
}
|
|
}
|
|
else // cleanup hTheme if CThemeWnd creation failed
|
|
{
|
|
CloseThemeData(hTheme);
|
|
}
|
|
}
|
|
}
|
|
|
|
LogExitNC(L"Attach");
|
|
return pwnd;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Instance method: attaches the CThemeWnd object to the specified window.
|
|
BOOL CThemeWnd::_AttachInstance( HWND hwnd, HTHEME hTheme, ULONG ulTargetFlags, PVOID pvWndCompat )
|
|
{
|
|
if( _fCritSectsInit || NT_SUCCESS(RtlInitializeCriticalSection( &_cswm )) )
|
|
{
|
|
Log(LOG_NCATTACH, L"_AttachInstance: Nonclient attached to hwnd=0x%x", hwnd);
|
|
|
|
_fCritSectsInit = TRUE;
|
|
_hwnd = hwnd;
|
|
_hTheme = hTheme;
|
|
_fClassFlags = ulTargetFlags;
|
|
|
|
_fFrameThemed = TESTFLAG( ulTargetFlags, TWCF_FRAME|TWCF_TOOLFRAME );
|
|
return SetProp( hwnd, MAKEINTATOM(GetThemeAtom(THEMEATOM_NONCLIENT)), this );
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::RemoveWindowProperties(HWND hwnd, BOOL fDestroying)
|
|
{
|
|
//---- remove properties that require theme or hooks ----
|
|
RemoveProp(hwnd, MAKEINTATOM(GetThemeAtom(THEMEATOM_HTHEME)));
|
|
|
|
if (fDestroying)
|
|
{
|
|
// Help apps by cleaning up the dialog texture.
|
|
RemoveProp(hwnd, MAKEINTATOM(GetThemeAtom(THEMEATOM_DLGTEXTURING)));
|
|
|
|
//---- remove all remaining theme properties ----
|
|
ApplyStringProp(hwnd, NULL, GetThemeAtom(THEMEATOM_SUBIDLIST));
|
|
ApplyStringProp(hwnd, NULL, GetThemeAtom(THEMEATOM_SUBAPPNAME));
|
|
|
|
//---- notify appinfo (foreign tracking, preview) ----
|
|
g_pAppInfo->OnWindowDestroyed(hwnd);
|
|
}
|
|
else
|
|
{
|
|
//---- only do this if hwnd is not being destroyed ----
|
|
ClearExStyleBits(hwnd);
|
|
}
|
|
}
|
|
//-------------------------------------------------------------------------//
|
|
// Static wrapper: detaches and destroys the CThemeWnd instance attached to the indicated
|
|
// window
|
|
void CThemeWnd::Detach( HWND hwnd, DWORD dwDisposition )
|
|
{
|
|
LogEntryNC(L"Detach");
|
|
|
|
// DO NOT GENERATE ANY WINDOW MESSAGES FROM THIS FUNCTION!!!
|
|
// (unless cleaning up frame).
|
|
|
|
// Prevent message threads from detaching when unhook thread (DetachAll) is executing...
|
|
if( !UNHOOKING() || TESTFLAG(dwDisposition, HMD_BULKDETACH) )
|
|
{
|
|
CThemeWnd* pwnd = FromHwnd( hwnd );
|
|
|
|
if( pwnd ) // nonclient tagged
|
|
{
|
|
if( VALID_THEMEWND(pwnd) )
|
|
{
|
|
// only one thread flips the _fDetached bit and proceeds through
|
|
// instance detatch and object free. Otherwise, object can be freed
|
|
// simultaneously on two different threads,
|
|
// e.g. (1) message thread and (2) UIAH_UNHOOK thread (ouch! scotthan).
|
|
if( !InterlockedCompareExchange( (LONG*)&pwnd->_fDetached, TRUE, FALSE ) )
|
|
{
|
|
pwnd->_DetachInstance( dwDisposition );
|
|
pwnd->Release();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
RemoveProp( hwnd, MAKEINTATOM(GetThemeAtom(THEMEATOM_NONCLIENT)) );
|
|
}
|
|
}
|
|
|
|
if (hwnd)
|
|
{
|
|
RemoveWindowProperties( hwnd, ((dwDisposition & HMD_WINDOWDESTROY) != 0) );
|
|
}
|
|
}
|
|
|
|
LogExitNC(L"Detach");
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Instance method: detaches the CThemeWnd object from the specified window.
|
|
BOOL CThemeWnd::_DetachInstance( DWORD dwDisposition )
|
|
{
|
|
HWND hwnd = _hwnd;
|
|
|
|
// untheme maxed MDI child sysbuttons.
|
|
ThemeMDIMenuButtons(FALSE, FALSE);
|
|
|
|
// Here's our last chance to ensure frame theme is withdrawn cleanly.
|
|
if( (IsFrameThemed() || IsRevoked(RF_REGION)) && AssignedFrameRgn() &&
|
|
!TESTFLAG(dwDisposition, HMD_PROCESSDETACH|HMD_WINDOWDESTROY))
|
|
{
|
|
RemoveFrameTheme( FTF_REDRAW );
|
|
}
|
|
|
|
//SPEW_THEMEWND( pwnd, 0, TEXT("UxTheme - Detaching and deleting themewnd: %s\n") );
|
|
DetachScrollBars( hwnd );
|
|
|
|
_hwnd =
|
|
_hwndMDIClient = NULL;
|
|
|
|
UnloadMdiBtns();
|
|
|
|
_CloseTheme();
|
|
|
|
Log(LOG_NCATTACH, L"_DetachInstance: Nonclient detached to hwnd=0x%x", hwnd);
|
|
|
|
RemoveProp( hwnd, MAKEINTATOM(GetThemeAtom(THEMEATOM_NONCLIENT)) );
|
|
return TRUE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Ensures that the specified window will not be themed during its lifetime
|
|
BOOL CThemeWnd::Reject( HWND hwnd, BOOL fExile )
|
|
{
|
|
// set a 'nil' tag on the window
|
|
return hwnd ? SetProp( hwnd, MAKEINTATOM(GetThemeAtom(THEMEATOM_NONCLIENT)),
|
|
fExile ? THEMEWND_EXILE : THEMEWND_REJECT ) : FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Ensures that the specified window will not be themed during its lifetime
|
|
BOOL CThemeWnd::Fail( HWND hwnd )
|
|
{
|
|
// set a failure tag on the window
|
|
return hwnd ? SetProp( hwnd, MAKEINTATOM(GetThemeAtom(THEMEATOM_NONCLIENT)),
|
|
THEMEWND_FAILURE ) : FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Revokes theming on a themed window
|
|
BOOL CThemeWnd::Revoke()
|
|
{
|
|
// Warning Will Robinson: After we detach, the CThemeWnd::_hwnd
|
|
// and related members will be reset, so save this on the stack.
|
|
|
|
BOOL fRet = TRUE;
|
|
HWND hwnd = _hwnd;
|
|
|
|
if( !IsRevoked(RF_INREVOKE) )
|
|
{
|
|
EnterRevoke();
|
|
_dwRevokeFlags &= ~RF_DEFER;
|
|
Detach( hwnd, HMD_REVOKE );
|
|
fRet = Reject( hwnd, TRUE );
|
|
LeaveRevoke();
|
|
}
|
|
return fRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// cookie passed to EnumChildWindows callback for CThemeWnd::DetachAll
|
|
typedef struct
|
|
{
|
|
DWORD dwProcessId;
|
|
DWORD dwDisposition;
|
|
}DETACHALL;
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// EnumChildWindows callback for CThemeWnd::DetachAll
|
|
BOOL CThemeWnd::_DetachDesktopWindowsCB( HWND hwnd, LPARAM lParam )
|
|
{
|
|
DETACHALL* pda = (DETACHALL*)lParam;
|
|
|
|
// detach this window
|
|
if( IsWindowProcess( hwnd, pda->dwProcessId ) )
|
|
{
|
|
//---- clear the nonclient theme ----
|
|
CThemeWnd::Detach(hwnd, HMD_THEMEDETACH|pda->dwDisposition);
|
|
|
|
if( !TESTFLAG(pda->dwDisposition, HMD_PROCESSDETACH) )
|
|
{
|
|
//---- clear the client theme now, so that we can invalidate ----
|
|
//---- all old theme handles after this. ----
|
|
SafeSendMessage(hwnd, WM_THEMECHANGED, (WPARAM)-1, 0);
|
|
|
|
Log(LOG_TMHANDLE, L"Did SEND of WM_THEMECHANGED to client hwnd=0x%x", hwnd);
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Detaches all themed windows managed by this process.
|
|
void CThemeWnd::DetachAll( DWORD dwDisposition )
|
|
{
|
|
DETACHALL da;
|
|
da.dwProcessId = GetCurrentProcessId();
|
|
da.dwDisposition = dwDisposition;
|
|
da.dwDisposition |= HMD_BULKDETACH;
|
|
|
|
//---- this will enum all windows for this process (all desktops, all child levels) ----
|
|
EnumProcessWindows( _DetachDesktopWindowsCB, (LPARAM)&da );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HTHEME CThemeWnd::_AcquireThemeHandle( HWND hwnd, ULONG* pfClassFlags )
|
|
{
|
|
HTHEME hTheme = ::OpenNcThemeData( hwnd, L"Window" );
|
|
|
|
if( NULL == hTheme )
|
|
{
|
|
if( pfClassFlags )
|
|
{
|
|
if( TESTFLAG(*pfClassFlags, TWCF_ANY) )
|
|
(*pfClassFlags) &= ~TWCF_ALL;
|
|
else
|
|
*pfClassFlags = 0;
|
|
}
|
|
}
|
|
|
|
//---- Did OpenNcThemeData() discover a new theme ----
|
|
if (g_pAppInfo->HasThemeChanged())
|
|
{
|
|
//---- IMPORTANT: we must refresh our theme metrics now, ----
|
|
//---- BEFORE we do our nonclient layout calcs & build a region window ----
|
|
AcquireNcThemeMetrics();
|
|
}
|
|
|
|
return hTheme;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::SetFrameTheme
|
|
//
|
|
// Initiates theming of the frame.
|
|
void CThemeWnd::SetFrameTheme(
|
|
IN ULONG dwFlags,
|
|
IN OPTIONAL WINDOWINFO* pwi )
|
|
{
|
|
LogEntryNC(L"SetFrameTheme");
|
|
|
|
ASSERT(TestCF( TWCF_FRAME|TWCF_TOOLFRAME ));
|
|
InitLameResources();
|
|
|
|
DWORD fSwp = SWP_NOZORDER|SWP_NOACTIVATE;
|
|
RECT rcWnd = {0};
|
|
BOOL bSwp = FALSE;
|
|
|
|
if( !TESTFLAG( dwFlags, FTF_NOMODIFYPLACEMENT ) )
|
|
{
|
|
GetWindowRect( _hwnd, &rcWnd );
|
|
fSwp |= (SWP_NOSIZE|SWP_NOMOVE/*|SWP_FRAMECHANGED 341700: this flag causes some apps to crash on WINDOWPOSCHANGED*/);
|
|
bSwp = TRUE;
|
|
}
|
|
|
|
// Generate a WM_WINDOWPOSCHANGING message to
|
|
// force a SetWindowRgn + frame repaint.
|
|
if( TESTFLAG(dwFlags, FTF_REDRAW) )
|
|
{
|
|
fSwp |= SWP_DRAWFRAME;
|
|
}
|
|
else
|
|
{
|
|
fSwp |= SWP_NOSENDCHANGING;
|
|
}
|
|
|
|
// theme MDI menubar buttons
|
|
_hwndMDIClient = _FindMDIClient(_hwnd);
|
|
if( _hwndMDIClient )
|
|
{
|
|
ThemeMDIMenuButtons(TRUE, FALSE);
|
|
}
|
|
|
|
// Kick frame region update.
|
|
_fFrameThemed = TRUE; // we invoked SetFrameTheme. Must be set BEFORE SetWindowPos.so we handle NCCALCSIZE properly.
|
|
SetDirtyFrameRgn(TRUE, TRUE); // ensure region assembly on non-resizing windows and dlgs.
|
|
|
|
if( !TESTFLAG( dwFlags, FTF_NOMODIFYPLACEMENT ) && bSwp )
|
|
{
|
|
_ScreenToParent( _hwnd, &rcWnd );
|
|
SetWindowPos( _hwnd, NULL, rcWnd.left, rcWnd.top,
|
|
RECTWIDTH(&rcWnd), RECTHEIGHT(&rcWnd), fSwp );
|
|
}
|
|
|
|
LogExitNC(L"SetFrameTheme");
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::_FreeRegionHandles()
|
|
{
|
|
#ifdef DEBUG
|
|
if( _hrgnWnd )
|
|
{
|
|
SPEW_RGNRECT(NCTF_RGNWND, TEXT("_FreeRegionHandles() - deleting window region"), _hrgnWnd, -1 );
|
|
}
|
|
#endif DEBUG
|
|
|
|
SAFE_DELETE_GDIOBJ(_hrgnWnd);
|
|
|
|
for( int i = 0; i < cFRAMEPARTS; i++ )
|
|
{
|
|
#ifdef DEBUG
|
|
if( _rghrgnParts[i] )
|
|
{
|
|
SPEW_RGNRECT(NCTF_RGNWND, TEXT("_FreeRegionHandles() - deleting component region"), _rghrgnParts[i], _ncwm.rgframeparts[i] );
|
|
}
|
|
|
|
if( _rghrgnSizingTemplates[i] )
|
|
{
|
|
SPEW_RGNRECT(NCTF_RGNWND, TEXT("_FreeRegionHandles() - deleting template region"), _rghrgnSizingTemplates[i], _ncwm.rgframeparts[i] );
|
|
}
|
|
#endif DEBUG
|
|
|
|
SAFE_DELETE_GDIOBJ(_rghrgnParts[i]);
|
|
SAFE_DELETE_GDIOBJ(_rghrgnSizingTemplates[i]);
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::RemoveFrameTheme
|
|
//
|
|
// Initiates theming of the frame. This method will not free the
|
|
// theme handle nor update the theme index.
|
|
void CThemeWnd::RemoveFrameTheme( ULONG dwFlags )
|
|
{
|
|
LogEntryNC(L"RemoveFrameTheme");
|
|
|
|
ASSERT(TestCF( TWCF_FRAME|TWCF_TOOLFRAME ));
|
|
|
|
_fFrameThemed = FALSE; // we're reverting SetFrameTheme
|
|
ClearRenderedNcPart(RNCF_ALL);
|
|
|
|
// Remove region
|
|
if( AssignedFrameRgn() && !TESTFLAG(dwFlags, FTF_NOMODIFYRGN) )
|
|
{
|
|
_fAssignedFrameRgn = FALSE;
|
|
_AssignRgn( NULL, dwFlags );
|
|
_FreeRegionHandles();
|
|
}
|
|
|
|
// Force redraw
|
|
if( TESTFLAG(dwFlags, FTF_REDRAW) )
|
|
InvalidateRect( _hwnd, NULL, TRUE );
|
|
|
|
ClearLameResources();
|
|
|
|
LogExitNC(L"RemoveFrameTheme");
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL CThemeWnd::IsNcThemed()
|
|
{
|
|
if( _hTheme != NULL && (IsRevoked(RF_DEFER) || !IsRevoked(RF_INREVOKE|RF_TYPEMASK)) &&
|
|
TestCF(TWCF_ANY & TWCF_NCTHEMETARGETMASK) )
|
|
{
|
|
if( TestCF(TWCF_FRAME|TWCF_TOOLFRAME) )
|
|
{
|
|
// if we're a frame window, we should be properly initialized
|
|
// w/ SetFrameTheme()
|
|
return _fFrameThemed;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL CThemeWnd::IsFrameThemed()
|
|
{
|
|
return IsNcThemed() && _fFrameThemed &&
|
|
(AssignedFrameRgn() ? TRUE : TestCF( TWCF_FRAME|TWCF_TOOLFRAME ));
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::SetDirtyFrameRgn( BOOL fDirty, BOOL fFrameChanged )
|
|
{
|
|
_fDirtyFrameRgn = fDirty;
|
|
|
|
Log(LOG_NCATTACH, L"SetDirtyFrameRgn: fDirty=%d, fFrameChanged=%d",
|
|
fDirty, fFrameChanged);
|
|
|
|
if( fFrameChanged ) // assure a region update despite no size change.
|
|
{
|
|
_sizeRgn.cx = _sizeRgn.cy = -1;
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::CreateCompositeRgn() - assembles a composite region from
|
|
// non-client segment regions sized to fill the specified window rectangle.
|
|
//
|
|
HRGN CThemeWnd::CreateCompositeRgn(
|
|
IN const NCWNDMET* pncwm,
|
|
OUT HRGN rghrgnParts[],
|
|
OUT HRGN rghrgnTemplates[] )
|
|
{
|
|
ASSERT( pncwm->fFrame == TRUE ); // shouldn't be here unless we're a frame window
|
|
|
|
HRGN hrgnWnd = NULL, hrgnContent = NULL;
|
|
HRGN rghrgn[cFRAMEPARTS] = {0};
|
|
int i;
|
|
|
|
if( pncwm->fFullMaxed )
|
|
{
|
|
// All full-screen maximized windows get a region, which is used to clip
|
|
// the window to the current monitor. The window region for a maximized
|
|
// window consists of the maxcaption region combined with a rect region
|
|
// corresponding to the content area.
|
|
RECT rcFullCaption = pncwm->rcW0[NCRC_CAPTION];
|
|
rcFullCaption.top += pncwm->cnBorders;
|
|
rcFullCaption.left += pncwm->cnBorders;
|
|
rcFullCaption.right -= pncwm->cnBorders;
|
|
|
|
if( SUCCEEDED(GetThemeBackgroundRegion(_hTheme, NULL, pncwm->rgframeparts[iCAPTION], pncwm->framestate,
|
|
&rcFullCaption, &rghrgn[iCAPTION])) )
|
|
{
|
|
SPEW_RGNRECT(NCTF_RGNWND, TEXT("CreateCompositeRgn() maximized caption rgn"), rghrgn[iCAPTION], pncwm->rgframeparts[iCAPTION] );
|
|
AddToCompositeRgn(&hrgnWnd, rghrgn[iCAPTION], 0, 0);
|
|
|
|
if( !IsRectEmpty( &pncwm->rcW0[NCRC_CONTENT] ) )
|
|
{
|
|
// remainder of full-maxed frame region is the content area (client+menubar+scrollbars),
|
|
// and is always rectangular
|
|
hrgnContent = CreateRectRgnIndirect( &pncwm->rcW0[NCRC_CONTENT] );
|
|
SPEW_RGNRECT(NCTF_RGNWND, TEXT("CreateCompositeRgn() maximized frame content rgn"), hrgnContent, 0 );
|
|
|
|
AddToCompositeRgn(&hrgnWnd, hrgnContent, 0, 0);
|
|
SAFE_DELETE_GDIOBJ(hrgnContent);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Normal windows consist of either a stand-alone frame part, or a frame
|
|
// part plus a caption part. In the first case, the window region is
|
|
// the frame region. In the second case, the window region is a composite
|
|
// of the frame and caption rects.
|
|
|
|
for( i = 0; i < ARRAYSIZE(pncwm->rgframeparts); i++ )
|
|
{
|
|
if( (iCAPTION == i || !pncwm->fMin) && !IsRectEmpty( &pncwm->rcW0[NCRC_FRAMEFIRST + i] ) )
|
|
{
|
|
if( _IsNcPartTransparent(pncwm->rgframeparts[i], _nctmCurrent) )
|
|
{
|
|
if( FAILED(GetThemeBackgroundRegion( _hTheme, NULL, pncwm->rgframeparts[i], pncwm->framestate,
|
|
&pncwm->rcW0[NCRC_FRAMEFIRST+i], &rghrgn[i] )) )
|
|
{
|
|
rghrgn[i] = NULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
rghrgn[i] = CreateRectRgnIndirect( &pncwm->rcW0[NCRC_FRAMEFIRST + i] );
|
|
}
|
|
}
|
|
|
|
if( rghrgn[i] != NULL )
|
|
{
|
|
SPEW_RGNRECT(NCTF_RGNWND, TEXT("CreateCompositeRgn() frame subrgn"), rghrgn[i], pncwm->rgframeparts[i] );
|
|
AddToCompositeRgn(&hrgnWnd, rghrgn[i], 0, 0);
|
|
}
|
|
}
|
|
|
|
// don't forget window content area (client+menubar+scrollbars), which is always rectangular
|
|
if( !pncwm->fMin && !IsRectEmpty( &pncwm->rcW0[NCRC_CONTENT] ) )
|
|
{
|
|
hrgnContent = CreateRectRgnIndirect( &pncwm->rcW0[NCRC_CONTENT] );
|
|
SPEW_RGNRECT(NCTF_RGNWND, TEXT("CreateCompositeRgn() normal frame content rgn"), hrgnContent, 0 );
|
|
|
|
AddToCompositeRgn(&hrgnWnd, hrgnContent, 0, 0);
|
|
SAFE_DELETE_GDIOBJ(hrgnContent);
|
|
}
|
|
}
|
|
|
|
// copy subregions back to caller
|
|
CopyMemory( rghrgnParts, rghrgn, sizeof(rghrgn) );
|
|
|
|
// extract frame resizing templates
|
|
ZeroMemory( rghrgn, sizeof(rghrgn) ); // reuse region array
|
|
for( i = 0; i < cFRAMEPARTS; i++ )
|
|
{
|
|
const RECT* prc = &pncwm->rcW0[NCRC_FRAMEFIRST + i];
|
|
|
|
if( VALID_WINDOWPART(pncwm->rgsizehitparts[i]) && !IsRectEmpty( prc ) )
|
|
{
|
|
if( SUCCEEDED(GetThemeBackgroundRegion( _hTheme, NULL, pncwm->rgsizehitparts[i], pncwm->framestate,
|
|
prc, &rghrgn[i])) )
|
|
{
|
|
|
|
SPEW_RGNRECT(NCTF_RGNWND, TEXT("CreateCompositeRgn() sizing template"), rghrgn[i], pncwm->rgframeparts[i] );
|
|
}
|
|
}
|
|
}
|
|
CopyMemory( rghrgnTemplates, rghrgn, sizeof(rghrgn) );
|
|
|
|
return hrgnWnd;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::AssignFrameRgn( BOOL fAssign, DWORD dwFlags )
|
|
{
|
|
if( fAssign )
|
|
{
|
|
NCWNDMET* pncwm = NULL;
|
|
NCTHEMEMET nctm = {0};
|
|
if( GetNcWindowMetrics( NULL, &pncwm, &nctm, 0 ) )
|
|
{
|
|
// should we set up a window region on this frame?
|
|
if( pncwm->fFrame )
|
|
{
|
|
if( _ShouldAssignFrameRgn( pncwm, nctm ) )
|
|
{
|
|
if( (_sizeRgn.cx != RECTWIDTH(&pncwm->rcW0[NCRC_WINDOW]) ||
|
|
_sizeRgn.cy != RECTHEIGHT(&pncwm->rcW0[NCRC_WINDOW])) )
|
|
{
|
|
HRGN hrgnWnd = NULL;
|
|
HRGN rghrgnParts[cFRAMEPARTS] = {0};
|
|
HRGN rghrgnTemplates[cFRAMEPARTS] = {0};
|
|
|
|
if( (hrgnWnd = CreateCompositeRgn( pncwm, rghrgnParts, rghrgnTemplates )) != NULL )
|
|
{
|
|
_sizeRgn.cx = RECTWIDTH(&pncwm->rcW0[NCRC_WINDOW]);
|
|
_sizeRgn.cy = RECTHEIGHT(&pncwm->rcW0[NCRC_WINDOW]);
|
|
|
|
// cache all of our regions for fast hit-testing.
|
|
_FreeRegionHandles();
|
|
_hrgnWnd = _DupRgn( hrgnWnd ); // dup this one cuz after _AssignRgn, we don't own it.
|
|
CopyMemory( _rghrgnParts, rghrgnParts, sizeof(_rghrgnParts) );
|
|
CopyMemory( _rghrgnSizingTemplates, rghrgnTemplates, sizeof(_rghrgnSizingTemplates) );
|
|
|
|
// assign the region
|
|
_AssignRgn( hrgnWnd, dwFlags );
|
|
}
|
|
}
|
|
}
|
|
// otherwise, if we've assigned a region, make sure we remove it.
|
|
else if( AssignedFrameRgn() )
|
|
{
|
|
fAssign = FALSE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if( !fAssign )
|
|
{
|
|
_AssignRgn( NULL, dwFlags );
|
|
FillMemory(&_sizeRgn, sizeof(_sizeRgn), 0xFF);
|
|
_FreeRegionHandles();
|
|
}
|
|
SetDirtyFrameRgn(FALSE); // make sure we reset this in case we didn't hit _AssignRgn.
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::_AssignRgn() - assigns the specified region
|
|
// to the window, prevents recursion (SetWindowRgn w/ bRedraw == TRUE
|
|
// generates WM_WINDOWPOSCHANGING, WM_NCCALCSIZE, && WM_NCPAINT).
|
|
//
|
|
void CThemeWnd::_AssignRgn( HRGN hrgn, DWORD dwFlags )
|
|
{
|
|
if( TESTFLAG(dwFlags, FTF_NOMODIFYRGN) )
|
|
{
|
|
_fAssignedFrameRgn = FALSE;
|
|
}
|
|
else if( !IsWindowInDestroy(_hwnd) )
|
|
{
|
|
// Assign the new region.
|
|
_fAssigningFrameRgn = TRUE;
|
|
SPEW_RGNRECT(NCTF_RGNWND, TEXT("_AssignRgn() rect"), hrgn, -1 );
|
|
_fAssignedFrameRgn = SetWindowRgn( _hwnd, hrgn, TESTFLAG(dwFlags, FTF_REDRAW) ) != 0;
|
|
_fAssigningFrameRgn = FALSE;
|
|
|
|
}
|
|
SetDirtyFrameRgn(FALSE);
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::GetNcWindowMetrics
|
|
//
|
|
// Computes internal per-window theme metrics.
|
|
//
|
|
BOOL CThemeWnd::GetNcWindowMetrics(
|
|
IN OPTIONAL LPCRECT prcWnd,
|
|
OUT OPTIONAL NCWNDMET** ppncwm,
|
|
OUT OPTIONAL NCTHEMEMET* pnctm,
|
|
IN DWORD dwOptions )
|
|
{
|
|
LogEntryNC(L"GetNcWindowMetrics");
|
|
|
|
NCTHEMEMET nctm;
|
|
BOOL bRet = FALSE;
|
|
BOOL fMenuBar = _ncwm.cyMenu != 0;
|
|
WINDOWPARTS rgframeparts[cFRAMEPARTS];
|
|
|
|
CopyMemory( rgframeparts, _ncwm.rgframeparts, sizeof(rgframeparts) );
|
|
|
|
// fetch per-theme metrics; we're going to need theme throughout
|
|
if (TESTFLAG(dwOptions, NCWMF_PREVIEW))
|
|
{
|
|
_LoadNcThemeMetrics(_hwnd, &nctm);
|
|
}
|
|
else if( !GetCurrentNcThemeMetrics( &nctm ) )
|
|
{
|
|
goto exit;
|
|
}
|
|
|
|
if( pnctm ) *pnctm = nctm;
|
|
|
|
if( !_ncwm.fValid || prcWnd != NULL )
|
|
dwOptions |= NCWMF_RECOMPUTE;
|
|
|
|
if( TESTFLAG(dwOptions, NCWMF_RECOMPUTE) )
|
|
{
|
|
// get caption text size before entering critsec (sends WM_GETTEXTLENGTH, WM_GETTEXT).
|
|
SIZE sizeCaptionText = {0};
|
|
HFONT hfCaption = NULL;
|
|
HWND hwndMDIActive = NULL;
|
|
|
|
// Do rough determination of whether or not we're a frame window and need to compute text metrics.
|
|
// We'll finalize this later
|
|
BOOL fFrame, fSmallFrame;
|
|
|
|
if( _ncwm.fValid )
|
|
{
|
|
fFrame = TESTFLAG( CThemeWnd::EvaluateStyle(_ncwm.dwStyle, _ncwm.dwExStyle), TWCF_FRAME|TWCF_TOOLFRAME );
|
|
fSmallFrame = TESTFLAG( CThemeWnd::EvaluateStyle(_ncwm.dwStyle, _ncwm.dwExStyle), TWCF_TOOLFRAME );
|
|
}
|
|
else
|
|
{
|
|
fFrame = TestCF(TWCF_FRAME|TWCF_TOOLFRAME);
|
|
fSmallFrame = TestCF(TWCF_TOOLFRAME);
|
|
}
|
|
|
|
// Compute text metrics outside of critical section (sends WM_GETTEXT);
|
|
if( fFrame && _fFrameThemed )
|
|
{
|
|
hfCaption = NcGetCaptionFont( fSmallFrame );
|
|
_GetNcCaptionTextSize( _hTheme, _hwnd, hfCaption, &sizeCaptionText );
|
|
}
|
|
|
|
// Retrieve active MDI sibling outside of critical section (sends WM_MDIGETACTIVE);
|
|
if( TESTFLAG(GetWindowLong(_hwnd, GWL_EXSTYLE), WS_EX_MDICHILD) )
|
|
{
|
|
hwndMDIActive = _MDIGetActive( GetParent(_hwnd) );
|
|
}
|
|
|
|
ASSERT(_fCritSectsInit);
|
|
EnterCriticalSection( &_cswm );
|
|
|
|
ZeroMemory( &_ncwm, sizeof(_ncwm) );
|
|
|
|
if( (bRet = _GetWindowMetrics( _hwnd, hwndMDIActive, &_ncwm )) != FALSE )
|
|
{
|
|
_ComputeNcWindowStatus( _hwnd, _ncwm.dwWindowStatus, &_ncwm );
|
|
|
|
// if window RECT is provided by the caller, stuff it now.
|
|
if( prcWnd )
|
|
{
|
|
_ncwm.rcS0[NCRC_WINDOW] = *prcWnd;
|
|
SetRectEmpty( &_ncwm.rcS0[NCRC_CLIENT] );
|
|
}
|
|
|
|
// stuff caption text size
|
|
_ncwm.sizeCaptionText = sizeCaptionText;
|
|
_ncwm.hfCaption = hfCaption;
|
|
|
|
// retrieve frame metrics.
|
|
if( _GetNcFrameMetrics( _hwnd, _hTheme, nctm, _ncwm ) )
|
|
{
|
|
if( _ncwm.fFrame )
|
|
{
|
|
// user32!SetMenu has been called, or the caption or frame part has changed
|
|
// So ensure frame region update.
|
|
if( (_ncwm.cyMenu == 0 && fMenuBar) || (_ncwm.cyMenu > 0 && !fMenuBar) ||
|
|
memcmp( rgframeparts, _ncwm.rgframeparts, sizeof(rgframeparts) ) )
|
|
{
|
|
SetDirtyFrameRgn(TRUE, TRUE);
|
|
}
|
|
|
|
// Compute NC button placement
|
|
AcquireFrameIcon(_ncwm.dwStyle, _ncwm.dwExStyle, FALSE);
|
|
|
|
#ifdef THEMED_NCBTNMETRICS
|
|
_GetClassicNcBtnMetrics( &_ncwm, _hAppIcon, _MNCanClose(_hwnd), FALSE );
|
|
#else THEMED_NCBTNMETRICS
|
|
_GetNcBtnMetrics( &_ncwm, &nctm, _hAppIcon, _MNCanClose(_hwnd) );
|
|
#endif THEMED_NCBTNMETRICS
|
|
|
|
// Determine the caption margin for lame button metrics.
|
|
_GetNcCaptionMargins( _hTheme, nctm, _ncwm );
|
|
_GetNcCaptionTextRect( &_ncwm );
|
|
|
|
if( _ncwm.fFrame )
|
|
{
|
|
GetLameButtonMetrics( &_ncwm, &sizeCaptionText );
|
|
}
|
|
}
|
|
|
|
// Compute window-relative metrics
|
|
//
|
|
// If passed a window rect, base offsets on current window rect.
|
|
// This is done to ensure preview window's (_hwnd) fake child windows are rendered correctly.
|
|
RECT rcWnd = _ncwm.rcS0[NCRC_WINDOW];
|
|
|
|
if( prcWnd )
|
|
{
|
|
if( _hwnd )
|
|
GetWindowRect( _hwnd, &rcWnd );
|
|
|
|
// for an incoming window rect, assign the computed client rect.
|
|
_ncwm.rcS0[NCRC_CLIENT] = _ncwm.rcS0[NCRC_UXCLIENT];
|
|
|
|
}
|
|
|
|
for( int i = NCRC_FIRST; i < NCRC_COUNT; i++ )
|
|
{
|
|
_ncwm.rcW0[i] = _ncwm.rcS0[i];
|
|
OffsetRect( &_ncwm.rcW0[i], -rcWnd.left, -rcWnd.top );
|
|
}
|
|
|
|
// All base computations are done; mark valid.
|
|
_ncwm.fValid = TRUE;
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection( &_cswm );
|
|
}
|
|
|
|
if( ppncwm )
|
|
{
|
|
*ppncwm = &_ncwm;
|
|
}
|
|
|
|
bRet = TRUE;
|
|
|
|
exit:
|
|
LogExitNC(L"GetNcWindowMetrics");
|
|
return bRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::ReleaseNcWindowMetrics( IN NCWNDMET* pncwm )
|
|
{
|
|
LeaveCriticalSection( &_cswm );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
inline COLORREF _GetNcCaptionTextColor( FRAMESTATES iStateId )
|
|
{
|
|
return GetSysColor( FS_ACTIVE == iStateId ?
|
|
COLOR_CAPTIONTEXT : COLOR_INACTIVECAPTIONTEXT );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Get CTLCOLOR brush for solid fills
|
|
void _GetBrushesForPart(HTHEME hTheme, int iPart, HBITMAP* phbm, HBRUSH* phbr)
|
|
{
|
|
int nBgType;
|
|
|
|
*phbm = NULL;
|
|
*phbr = NULL;
|
|
|
|
// Get CTLCOLOR brush for solid fills
|
|
HRESULT hr = GetThemeEnumValue( hTheme, iPart, 0, TMT_BGTYPE, &nBgType );
|
|
if( SUCCEEDED( hr ))
|
|
{
|
|
if (BT_BORDERFILL == nBgType)
|
|
{
|
|
int nFillType;
|
|
hr = GetThemeEnumValue( hTheme, iPart, 0, TMT_FILLTYPE, &nFillType );
|
|
if (SUCCEEDED( hr ) &&
|
|
FT_SOLID == nFillType)
|
|
{
|
|
COLORREF cr;
|
|
hr = GetThemeColor( hTheme, iPart, 0, TMT_FILLCOLOR, &cr);
|
|
|
|
*phbr = CreateSolidBrush(cr);
|
|
}
|
|
else
|
|
{
|
|
ASSERTMSG(FALSE, "Themes: The theme file specified an invalid fill type for dialog boxes");
|
|
}
|
|
}
|
|
else if (BT_IMAGEFILE == nBgType)
|
|
{
|
|
HDC hdc = GetWindowDC(NULL);
|
|
if ( hdc )
|
|
{
|
|
hr = GetThemeBitmap(hTheme, hdc, iPart, 0, NULL, phbm);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
*phbr = CreatePatternBrush(*phbm);
|
|
}
|
|
ReleaseDC(NULL, hdc);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
//
|
|
// Chooses appropriate hit testing parts for the various Nc area
|
|
//
|
|
void _GetNcSizingTemplates(
|
|
IN const NCTHEMEMET& nctm,
|
|
IN OUT NCWNDMET& ncwm ) // window metric block. dwStyle, dwExStyle, rcS0[NCRC_WINDOW] members are required.
|
|
{
|
|
FillMemory( ncwm.rgsizehitparts, sizeof(ncwm.rgsizehitparts), BOGUS_WINDOWPART );
|
|
|
|
// No need on windows without frames
|
|
if( !ncwm.fFrame )
|
|
return;
|
|
|
|
// minimized or full-screen maximized window
|
|
if( ncwm.fMin || ncwm.fFullMaxed )
|
|
return;
|
|
|
|
// No need on windows that aren't sizable
|
|
if( !TESTFLAG(ncwm.dwStyle, WS_THICKFRAME) )
|
|
return;
|
|
|
|
if( ncwm.fSmallFrame)
|
|
{
|
|
if (nctm.fSmCapSizingTemplate)
|
|
ncwm.rgsizehitparts[iCAPTION] = WP_SMALLCAPTIONSIZINGTEMPLATE;
|
|
|
|
if (nctm.fSmLeftSizingTemplate)
|
|
ncwm.rgsizehitparts[iFRAMELEFT] = WP_SMALLFRAMELEFTSIZINGTEMPLATE;
|
|
|
|
if (nctm.fSmRightSizingTemplate)
|
|
ncwm.rgsizehitparts[iFRAMERIGHT] = WP_SMALLFRAMERIGHTSIZINGTEMPLATE;
|
|
|
|
if (nctm.fSmBottomSizingTemplate)
|
|
ncwm.rgsizehitparts[iFRAMEBOTTOM] = WP_SMALLFRAMEBOTTOMSIZINGTEMPLATE;
|
|
}
|
|
else
|
|
{
|
|
if (nctm.fCapSizingTemplate)
|
|
ncwm.rgsizehitparts[iCAPTION] = WP_CAPTIONSIZINGTEMPLATE;
|
|
|
|
if (nctm.fLeftSizingTemplate)
|
|
ncwm.rgsizehitparts[iFRAMELEFT] = WP_FRAMELEFTSIZINGTEMPLATE;
|
|
|
|
if (nctm.fRightSizingTemplate)
|
|
ncwm.rgsizehitparts[iFRAMERIGHT] = WP_FRAMERIGHTSIZINGTEMPLATE;
|
|
|
|
if (nctm.fBottomSizingTemplate)
|
|
ncwm.rgsizehitparts[iFRAMEBOTTOM] = WP_FRAMEBOTTOMSIZINGTEMPLATE;
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
//
|
|
// Computes theme metrics for frame window.
|
|
//
|
|
BOOL _GetNcFrameMetrics(
|
|
IN OPTIONAL HWND hwnd, // window handle (required for multiline menubar calcs).
|
|
IN HTHEME hTheme, // theme handle (required)
|
|
IN const NCTHEMEMET& nctm, // theme metric block
|
|
IN OUT NCWNDMET& ncwm ) // window metric block. dwStyle, dwExStyle, rcS0[NCRC_WINDOW] members are required.
|
|
{
|
|
LogEntryNC(L"_GetNcFrameMetrics");
|
|
ASSERT(hTheme);
|
|
|
|
// recompute style class
|
|
ncwm.dwStyleClass = CThemeWnd::EvaluateStyle( ncwm.dwStyle, ncwm.dwExStyle );
|
|
ncwm.cnBorders = _GetWindowBorders( ncwm.dwStyle, ncwm.dwExStyle );
|
|
|
|
// compute frame attributes, state
|
|
ncwm.fFrame = TESTFLAG( ncwm.dwStyleClass, (TWCF_FRAME|TWCF_TOOLFRAME) );
|
|
ncwm.fSmallFrame = TESTFLAG( ncwm.dwStyleClass, TWCF_TOOLFRAME );
|
|
|
|
// compute frame and caption parts
|
|
if( ncwm.fFrame )
|
|
{
|
|
ncwm.rgframeparts[iFRAMEBOTTOM] =
|
|
ncwm.rgframeparts[iFRAMELEFT] =
|
|
ncwm.rgframeparts[iFRAMERIGHT] =
|
|
ncwm.rgframeparts[iCAPTION] = BOGUS_WINDOWPART;
|
|
|
|
if( ncwm.fMin ) // minimized window
|
|
{
|
|
ncwm.rgframeparts[iCAPTION] = WP_MINCAPTION;
|
|
}
|
|
else if( ncwm.fFullMaxed ) // full-screen maximized window
|
|
{
|
|
ncwm.rgframeparts[iCAPTION] = WP_MAXCAPTION;
|
|
}
|
|
else // normal or partial-screen maximized window with thick border
|
|
{
|
|
if( ncwm.fSmallFrame )
|
|
{
|
|
ncwm.rgframeparts[iCAPTION] = WP_SMALLCAPTION;
|
|
ncwm.rgframeparts[iFRAMELEFT] = WP_SMALLFRAMELEFT;
|
|
ncwm.rgframeparts[iFRAMERIGHT] = WP_SMALLFRAMERIGHT;
|
|
ncwm.rgframeparts[iFRAMEBOTTOM] = WP_SMALLFRAMEBOTTOM;
|
|
}
|
|
else
|
|
{
|
|
ncwm.rgframeparts[iCAPTION] = ncwm.fMaxed ? WP_MAXCAPTION : WP_CAPTION;
|
|
ncwm.rgframeparts[iFRAMELEFT] = WP_FRAMELEFT;
|
|
ncwm.rgframeparts[iFRAMERIGHT] = WP_FRAMERIGHT;
|
|
ncwm.rgframeparts[iFRAMEBOTTOM] = WP_FRAMEBOTTOM;
|
|
}
|
|
}
|
|
|
|
// stash caption text color.
|
|
ncwm.rgbCaption = _GetNcCaptionTextColor( ncwm.framestate );
|
|
|
|
// retrieve sizing templates.
|
|
_GetNcSizingTemplates( nctm, ncwm );
|
|
}
|
|
|
|
//-----------------------------------------------------------//
|
|
// Frame metrics
|
|
//
|
|
// Frame area includes 'skin' boundaries,
|
|
// menu, integrated caption and client edge.
|
|
//
|
|
// Independent of the frame is the separate caption seg,
|
|
// scrollbars, and sizebox
|
|
//-----------------------------------------------------------//
|
|
if( ncwm.fFrame ) // frame windows only
|
|
{
|
|
// Initialize positions of main frame components...
|
|
|
|
// Content rect: area bounded by frame theme.
|
|
// Client rect: area contained in content rect that excludes all nonclient
|
|
// elements (viz, scrollbars, menubar, inside edges).
|
|
// Caption rect: pertains to minimized and maximized windows,
|
|
// and normal windows if the theme defines a caption part
|
|
ncwm.rcS0[NCRC_CAPTION] =
|
|
ncwm.rcS0[NCRC_CONTENT] = ncwm.rcS0[NCRC_WINDOW];
|
|
SetRectEmpty( &ncwm.rcS0[NCRC_UXCLIENT] );
|
|
|
|
if( ncwm.fMin ) /* minimized frame */
|
|
{
|
|
// zero out content, client rectangles.
|
|
ncwm.rcS0[NCRC_CONTENT].right = ncwm.rcS0[NCRC_CONTENT].left;
|
|
ncwm.rcS0[NCRC_CONTENT].bottom = ncwm.rcS0[NCRC_CONTENT].top;
|
|
|
|
ncwm.rcS0[NCRC_CLIENT] =
|
|
ncwm.rcS0[NCRC_UXCLIENT] = ncwm.rcS0[NCRC_CONTENT];
|
|
}
|
|
else
|
|
{
|
|
NONCLIENTMETRICS ncm;
|
|
if( NcGetNonclientMetrics( &ncm, FALSE ) )
|
|
{
|
|
ncwm.rcS0[NCRC_FRAMEBOTTOM] =
|
|
ncwm.rcS0[NCRC_FRAMELEFT] =
|
|
ncwm.rcS0[NCRC_FRAMERIGHT] = ncwm.rcS0[NCRC_WINDOW];
|
|
|
|
// themed caption rect spans left, top, right bordersS
|
|
// and 1 pixel edge below caption
|
|
ncwm.rcS0[NCRC_CAPTION].bottom =
|
|
ncwm.rcS0[NCRC_CAPTION].top + ncwm.cnBorders +
|
|
(ncwm.fSmallFrame ? ncm.iSmCaptionHeight : ncm.iCaptionHeight) +
|
|
1 /* 1 pixel below caption */;
|
|
|
|
// update the content and rects while we're here:
|
|
InflateRect( &ncwm.rcS0[NCRC_CONTENT], -ncwm.cnBorders, -ncwm.cnBorders );
|
|
ncwm.rcS0[NCRC_CONTENT].top = ncwm.rcS0[NCRC_CAPTION].bottom;
|
|
if( ncwm.rcS0[NCRC_CONTENT].bottom < ncwm.rcS0[NCRC_CONTENT].top )
|
|
ncwm.rcS0[NCRC_CONTENT].bottom = ncwm.rcS0[NCRC_CONTENT].top;
|
|
|
|
// at this point the client rect is identical to the content rect (haven't computed menubar, scrollbars).
|
|
ncwm.rcS0[NCRC_UXCLIENT] = ncwm.rcS0[NCRC_CONTENT];
|
|
|
|
// bottom border segment.
|
|
ncwm.rcS0[NCRC_FRAMEBOTTOM].top = ncwm.rcS0[NCRC_FRAMEBOTTOM].bottom - ncwm.cnBorders;
|
|
|
|
// side border segments
|
|
ncwm.rcS0[NCRC_FRAMELEFT].top =
|
|
ncwm.rcS0[NCRC_FRAMERIGHT].top = ncwm.rcS0[NCRC_CAPTION].bottom;
|
|
|
|
ncwm.rcS0[NCRC_FRAMELEFT].bottom =
|
|
ncwm.rcS0[NCRC_FRAMERIGHT].bottom = ncwm.rcS0[NCRC_FRAMEBOTTOM].top;
|
|
|
|
ncwm.rcS0[NCRC_FRAMELEFT].right = ncwm.rcS0[NCRC_FRAMELEFT].left + ncwm.cnBorders;
|
|
ncwm.rcS0[NCRC_FRAMERIGHT].left = ncwm.rcS0[NCRC_FRAMERIGHT].right - ncwm.cnBorders;
|
|
}
|
|
}
|
|
}
|
|
else // frameless windows with scrollbars and/or client-edge:
|
|
{
|
|
// Non-frame windows
|
|
ncwm.rcS0[NCRC_UXCLIENT] = ncwm.rcS0[NCRC_WINDOW];
|
|
InflateRect( &ncwm.rcS0[NCRC_UXCLIENT], -ncwm.cnBorders, -ncwm.cnBorders );
|
|
ncwm.rcS0[NCRC_CONTENT] = ncwm.rcS0[NCRC_UXCLIENT];
|
|
}
|
|
|
|
// Menubar
|
|
if( !(ncwm.fMin || TESTFLAG( ncwm.dwStyle, WS_CHILD )) ) // child windows don't have menubars
|
|
{
|
|
// Menubar offsets (for painting)
|
|
ncwm.cnMenuOffsetTop = ncwm.rcS0[NCRC_CONTENT].top - ncwm.rcS0[NCRC_WINDOW].top;
|
|
ncwm.cnMenuOffsetLeft = ncwm.rcS0[NCRC_CONTENT].left - ncwm.rcS0[NCRC_WINDOW].left;
|
|
ncwm.cnMenuOffsetRight = ncwm.rcS0[NCRC_WINDOW].right - ncwm.rcS0[NCRC_CONTENT].right;
|
|
|
|
if( hwnd )
|
|
{
|
|
// calc menubar does the right thing for multiline menubars
|
|
ncwm.cyMenu = CalcMenuBar( hwnd, ncwm.cnMenuOffsetLeft,
|
|
ncwm.cnMenuOffsetRight,
|
|
ncwm.cnMenuOffsetTop,
|
|
&ncwm.rcS0[NCRC_WINDOW] );
|
|
}
|
|
else
|
|
{
|
|
// no window (e.g. preview) == no menu, meaning don't call CalcMenuBar.
|
|
// we emulate computations best we can:
|
|
ncwm.cyMenu = NcGetSystemMetrics( SM_CYMENUSIZE );
|
|
}
|
|
|
|
// CalcMenuBar and SM_CYMENUSIZE are 1 pixel short of reality.
|
|
if( ncwm.cyMenu )
|
|
ncwm.cyMenu += nctm.dyMenuBar;
|
|
|
|
// Menubar rect (for hit-testing and clipping)
|
|
SetRect( &ncwm.rcS0[NCRC_MENUBAR],
|
|
ncwm.rcS0[NCRC_CONTENT].left,
|
|
ncwm.rcS0[NCRC_CONTENT].top,
|
|
ncwm.rcS0[NCRC_CONTENT].right,
|
|
min(ncwm.rcS0[NCRC_CONTENT].bottom, ncwm.rcS0[NCRC_CONTENT].top + ncwm.cyMenu) );
|
|
|
|
ncwm.rcS0[NCRC_UXCLIENT].top = ncwm.rcS0[NCRC_MENUBAR].bottom;
|
|
}
|
|
|
|
// Client Edge.
|
|
if( !ncwm.fMin && TESTFLAG(ncwm.dwExStyle, WS_EX_CLIENTEDGE) )
|
|
{
|
|
CopyRect( &ncwm.rcS0[NCRC_CLIENTEDGE], &ncwm.rcS0[NCRC_UXCLIENT] );
|
|
InflateRect( &ncwm.rcS0[NCRC_UXCLIENT],
|
|
-NcGetSystemMetrics( SM_CXEDGE ),
|
|
-NcGetSystemMetrics( SM_CYEDGE ));
|
|
}
|
|
|
|
//-----------------------------------------------------------//
|
|
// Scrollbars and sizebox/gripper
|
|
//-----------------------------------------------------------//
|
|
|
|
if( !ncwm.fMin )
|
|
{
|
|
// horizontal scroll bar.
|
|
if( TESTFLAG(ncwm.dwStyle, WS_HSCROLL) )
|
|
{
|
|
ncwm.rcS0[NCRC_HSCROLL] = ncwm.rcS0[NCRC_UXCLIENT];
|
|
ncwm.rcS0[NCRC_HSCROLL].top = ncwm.rcS0[NCRC_UXCLIENT].bottom =
|
|
ncwm.rcS0[NCRC_HSCROLL].bottom - NcGetSystemMetrics( SM_CYHSCROLL );
|
|
|
|
if( IsRectEmpty( &ncwm.rcS0[NCRC_CLIENT] ) /* this happens in preview */ )
|
|
{
|
|
ncwm.rcS0[NCRC_HSCROLL].left = ncwm.rcS0[NCRC_UXCLIENT].left;
|
|
ncwm.rcS0[NCRC_HSCROLL].right = ncwm.rcS0[NCRC_UXCLIENT].right;
|
|
}
|
|
else
|
|
{
|
|
ncwm.rcS0[NCRC_HSCROLL].left = ncwm.rcS0[NCRC_CLIENT].left;
|
|
ncwm.rcS0[NCRC_HSCROLL].right = ncwm.rcS0[NCRC_CLIENT].right;
|
|
}
|
|
}
|
|
|
|
// vertical scroll bar
|
|
if( TESTFLAG(ncwm.dwStyle, WS_VSCROLL) )
|
|
{
|
|
ncwm.rcS0[NCRC_VSCROLL] = ncwm.rcS0[NCRC_UXCLIENT];
|
|
|
|
if( TESTFLAG(ncwm.dwExStyle, WS_EX_LAYOUTRTL) ^ TESTFLAG(ncwm.dwExStyle, WS_EX_LEFTSCROLLBAR) )
|
|
{
|
|
ncwm.rcS0[NCRC_VSCROLL].right = ncwm.rcS0[NCRC_UXCLIENT].left =
|
|
ncwm.rcS0[NCRC_VSCROLL].left + NcGetSystemMetrics( SM_CXVSCROLL );
|
|
|
|
// Adjust for horz scroll, gripper
|
|
if( TESTFLAG(ncwm.dwStyle, WS_HSCROLL) )
|
|
{
|
|
ncwm.rcS0[NCRC_SIZEBOX]= ncwm.rcS0[NCRC_HSCROLL];
|
|
ncwm.rcS0[NCRC_SIZEBOX].right = ncwm.rcS0[NCRC_HSCROLL].left =
|
|
ncwm.rcS0[NCRC_UXCLIENT].left;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ncwm.rcS0[NCRC_VSCROLL].left = ncwm.rcS0[NCRC_UXCLIENT].right =
|
|
ncwm.rcS0[NCRC_VSCROLL].right - NcGetSystemMetrics( SM_CXVSCROLL );
|
|
|
|
// Adjust for horz scroll, gripper
|
|
if( TESTFLAG(ncwm.dwStyle, WS_HSCROLL) )
|
|
{
|
|
ncwm.rcS0[NCRC_SIZEBOX]= ncwm.rcS0[NCRC_HSCROLL];
|
|
ncwm.rcS0[NCRC_SIZEBOX].left = ncwm.rcS0[NCRC_HSCROLL].right =
|
|
ncwm.rcS0[NCRC_UXCLIENT].right;
|
|
}
|
|
}
|
|
|
|
if( IsRectEmpty( &ncwm.rcS0[NCRC_CLIENT] ) /* this happens in preview */ )
|
|
{
|
|
ncwm.rcS0[NCRC_VSCROLL].top = ncwm.rcS0[NCRC_UXCLIENT].top;
|
|
ncwm.rcS0[NCRC_VSCROLL].bottom = ncwm.rcS0[NCRC_UXCLIENT].bottom;
|
|
}
|
|
else
|
|
{
|
|
ncwm.rcS0[NCRC_VSCROLL].top = ncwm.rcS0[NCRC_CLIENT].top;
|
|
ncwm.rcS0[NCRC_VSCROLL].bottom = ncwm.rcS0[NCRC_CLIENT].bottom;
|
|
}
|
|
}
|
|
}
|
|
|
|
LogExitNC(L"_GetNcFrameMetrics");
|
|
return TRUE;
|
|
}
|
|
|
|
#define EXT_TRACK_VERT 0x01
|
|
#define EXT_TRACK_HORZ 0x02
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void _GetNcBtnHitTestRect(
|
|
IN const NCWNDMET* pncwm,
|
|
IN UINT uHitcode,
|
|
BOOL fWindowRelative,
|
|
OUT LPRECT prcHit )
|
|
{
|
|
const RECT* prcBtn = NULL;
|
|
int dxLeft = 0; // button's left side delta
|
|
int dxRight = 0; // button's right side delta
|
|
|
|
// adjust hitrect to classic-look caption bar strip:
|
|
RECT rcHit = fWindowRelative ? pncwm->rcW0[NCRC_CAPTION] : pncwm->rcS0[NCRC_CAPTION];
|
|
rcHit.top += pncwm->cnBorders;
|
|
rcHit.left += pncwm->cnBorders;
|
|
rcHit.right -= pncwm->cnBorders;
|
|
rcHit.bottom -= 1;
|
|
|
|
// determine which button we're working with, how to extend the left, right sides.
|
|
switch( uHitcode )
|
|
{
|
|
case HTMINBUTTON:
|
|
prcBtn = fWindowRelative ? &pncwm->rcW0[NCRC_MINBTN] : &pncwm->rcS0[NCRC_MINBTN];
|
|
dxLeft = -1;
|
|
break;
|
|
|
|
case HTMAXBUTTON:
|
|
prcBtn = fWindowRelative ? &pncwm->rcW0[NCRC_MAXBTN] : &pncwm->rcS0[NCRC_MAXBTN];
|
|
dxRight = 1;
|
|
break;
|
|
|
|
case HTHELP:
|
|
prcBtn = fWindowRelative ? &pncwm->rcW0[NCRC_HELPBTN] : &pncwm->rcS0[NCRC_HELPBTN];
|
|
dxLeft = -1;
|
|
dxRight = 1;
|
|
break;
|
|
|
|
case HTCLOSE:
|
|
prcBtn = fWindowRelative ? &pncwm->rcW0[NCRC_CLOSEBTN] : &pncwm->rcS0[NCRC_CLOSEBTN];
|
|
dxLeft = -1;
|
|
dxRight = rcHit.right - prcBtn->right;
|
|
break;
|
|
|
|
case HTSYSMENU:
|
|
prcBtn = fWindowRelative ? &pncwm->rcW0[NCRC_SYSBTN] : &pncwm->rcS0[NCRC_SYSBTN];
|
|
dxLeft = rcHit.left - prcBtn->left;
|
|
dxRight = 1;
|
|
break;
|
|
}
|
|
|
|
if( prcBtn )
|
|
{
|
|
*prcHit = *prcBtn;
|
|
if( !IsRectEmpty( prcBtn ) )
|
|
{
|
|
rcHit.left = prcBtn->left + dxLeft;
|
|
rcHit.right = prcBtn->right + dxRight;
|
|
*prcHit = rcHit;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SetRectEmpty( prcHit );
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// wraps alloc, retrieval of window text
|
|
LPWSTR _AllocWindowText( IN HWND hwnd )
|
|
{
|
|
LPWSTR pszRet = NULL;
|
|
|
|
if (hwnd && IsWindow(hwnd))
|
|
{
|
|
if( (pszRet = new WCHAR[MAX_PATH]) != NULL )
|
|
{
|
|
int cch;
|
|
if( (cch = InternalGetWindowText(hwnd, pszRet, MAX_PATH)) <= 0 )
|
|
{
|
|
__try // some wndprocs can't handle an early WM_GETTEXT (eg.310700).
|
|
{
|
|
cch = GetWindowText(hwnd, pszRet, MAX_PATH);
|
|
}
|
|
__except(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
cch = 0;
|
|
}
|
|
}
|
|
|
|
if( !cch )
|
|
{
|
|
SAFE_DELETE_ARRAY(pszRet); // delete and zero pointer
|
|
}
|
|
}
|
|
}
|
|
|
|
return pszRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// _GetNcCaptionMargins() - computes a margin from the window ULC based on the
|
|
// offsets in the theme and the location of enabled caption buttons. The left
|
|
// margin is to the right of the last left-aligned button, and the right margin
|
|
// is to the left of the first right-aligned button.
|
|
//
|
|
BOOL _GetNcCaptionMargins( IN HTHEME hTheme, IN const NCTHEMEMET& nctm, IN OUT NCWNDMET& ncwm )
|
|
{
|
|
ZeroMemory( &ncwm.CaptionMargins, sizeof(ncwm.CaptionMargins) );
|
|
|
|
if( ncwm.fFrame )
|
|
{
|
|
// assign per-window CaptinMargins, hfCaption values
|
|
if( ncwm.fSmallFrame )
|
|
{
|
|
ncwm.CaptionMargins = nctm.marSmCaptionText;
|
|
}
|
|
else
|
|
{
|
|
if( ncwm.fMaxed )
|
|
{
|
|
ncwm.CaptionMargins = nctm.marMaxCaptionText;
|
|
}
|
|
else if( ncwm.fMin )
|
|
{
|
|
ncwm.CaptionMargins = nctm.marMinCaptionText;
|
|
}
|
|
else
|
|
{
|
|
ncwm.CaptionMargins = nctm.marCaptionText;
|
|
}
|
|
}
|
|
ncwm.hfCaption = NcGetCaptionFont(ncwm.fSmallFrame);
|
|
|
|
|
|
RECT rcContainer = ncwm.rcS0[NCRC_CAPTION];
|
|
RECT *prcBtn = &ncwm.rcS0[NCBTNFIRST];
|
|
rcContainer.left += ncwm.cnBorders;
|
|
rcContainer.right -= ncwm.cnBorders;
|
|
|
|
// sysmenu icon, if present, is the leftmost limit
|
|
if( !IsRectEmpty( &ncwm.rcS0[NCRC_SYSBTN] ) )
|
|
{
|
|
rcContainer.left = ncwm.rcS0[NCRC_SYSBTN].right;
|
|
}
|
|
|
|
// Compute our rightmost limit
|
|
for( UINT cRects = NCBTNRECTS; cRects; --cRects, ++prcBtn )
|
|
{
|
|
if (!IsRectEmpty(prcBtn))
|
|
{
|
|
if( prcBtn->left < rcContainer.right )
|
|
{
|
|
rcContainer.right = prcBtn->left;
|
|
}
|
|
}
|
|
}
|
|
|
|
if( rcContainer.right < rcContainer.left )
|
|
{
|
|
rcContainer.right = rcContainer.left;
|
|
}
|
|
|
|
// final captions margins are adjusted to accommodate buttons.
|
|
ncwm.CaptionMargins.cxLeftWidth += (rcContainer.left - ncwm.rcS0[NCRC_CAPTION].left);
|
|
ncwm.CaptionMargins.cxRightWidth += (ncwm.rcS0[NCRC_CAPTION].right - rcContainer.right);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL _GetNcCaptionTextSize( HTHEME hTheme, HWND hwnd, HFONT hf, OUT SIZE* psizeCaption )
|
|
{
|
|
BOOL fRet = FALSE;
|
|
LPWSTR pszCaption = _AllocWindowText( hwnd );
|
|
|
|
psizeCaption->cx = psizeCaption->cy = 0;
|
|
|
|
if( pszCaption )
|
|
{
|
|
HDC hdc = GetWindowDC(hwnd);
|
|
if( hdc )
|
|
{
|
|
//---- select font ----
|
|
HFONT hf0 = (HFONT)SelectObject(hdc, hf);
|
|
|
|
//---- let theme mgr do the calculation ----
|
|
RECT rcExtent;
|
|
HRESULT hr = GetThemeTextExtent( hTheme, hdc, WP_CAPTION, 0,
|
|
pszCaption, lstrlen(pszCaption), 0, NULL, &rcExtent );
|
|
|
|
//---- store result in "psizeCaption ----
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
psizeCaption->cx = WIDTH(rcExtent);
|
|
psizeCaption->cy = HEIGHT(rcExtent);
|
|
}
|
|
|
|
//---- clean up ----
|
|
SelectObject(hdc, hf0);
|
|
ReleaseDC(hwnd, hdc);
|
|
}
|
|
|
|
SAFE_DELETE_ARRAY(pszCaption);
|
|
}
|
|
return fRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Retrieves position of available area for caption text, in window-relative
|
|
// coordinates
|
|
BOOL _GetNcCaptionTextRect( IN OUT NCWNDMET* pncwm )
|
|
{
|
|
pncwm->rcS0[NCRC_CAPTIONTEXT] = pncwm->rcS0[NCRC_CAPTION];
|
|
|
|
// accommodate classic top sizing border:
|
|
pncwm->rcS0[NCRC_CAPTIONTEXT].top += pncwm->cnBorders;
|
|
|
|
// Assign left, right based on resp. caption margins
|
|
pncwm->rcS0[NCRC_CAPTIONTEXT].left += pncwm->CaptionMargins.cxLeftWidth;
|
|
pncwm->rcS0[NCRC_CAPTIONTEXT].right -= pncwm->CaptionMargins.cxRightWidth;
|
|
|
|
// vertically center the text between margins
|
|
int cyPadding = (RECTHEIGHT(&pncwm->rcS0[NCRC_CAPTIONTEXT]) - pncwm->sizeCaptionText.cy)/2;
|
|
pncwm->rcS0[NCRC_CAPTIONTEXT].top += cyPadding;
|
|
pncwm->rcS0[NCRC_CAPTIONTEXT].bottom -= cyPadding;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// retrieve the window icon
|
|
HICON CThemeWnd::AcquireFrameIcon(
|
|
DWORD dwStyle, DWORD dwExStyle, BOOL fWinIniChange )
|
|
{
|
|
if( _hAppIcon != NULL )
|
|
{
|
|
if( fWinIniChange )
|
|
{
|
|
_hAppIcon = NULL;
|
|
}
|
|
}
|
|
|
|
if( !TESTFLAG(dwStyle, WS_SYSMENU) || TESTFLAG(dwExStyle, WS_EX_TOOLWINDOW) )
|
|
{
|
|
// return nil value without throwing away cached icon handle;
|
|
// this may be a transient style change.
|
|
return NULL;
|
|
}
|
|
|
|
NONCLIENTMETRICS ncm = {0};
|
|
NcGetNonclientMetrics( &ncm, FALSE );
|
|
BOOL fPerferLargeIcon = ((30 < ncm.iCaptionHeight) ? TRUE : FALSE);
|
|
if( NULL == _hAppIcon && NULL == (_hAppIcon = _GetWindowIcon(_hwnd, fPerferLargeIcon)) )
|
|
{
|
|
if ( HAS_CAPTIONBAR(dwStyle) &&
|
|
((dwStyle & (WS_BORDER|WS_DLGFRAME)) != WS_DLGFRAME) &&
|
|
!TESTFLAG(dwExStyle, WS_EX_DLGMODALFRAME) )
|
|
{
|
|
// If we still can't get an icon and the window has
|
|
// SYSMENU set, then they get the default winlogo icon
|
|
_hAppIcon = LoadIcon(NULL, IDI_WINLOGO);
|
|
}
|
|
}
|
|
|
|
return _hAppIcon;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void _CopyInflateRect( LPRECT prcDst, LPCRECT prcSrc, int cx, int cy)
|
|
{
|
|
prcDst->left = prcSrc->left - cx;
|
|
prcDst->right = prcSrc->right + cx;
|
|
prcDst->top = prcSrc->top - cy;
|
|
prcDst->bottom = prcSrc->bottom + cy;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::ScreenToWindow() - transforms points from screen coords to
|
|
// window coords.
|
|
//
|
|
void CThemeWnd::ScreenToWindow( LPPOINT prgPts, UINT cPts )
|
|
{
|
|
RECT rcWnd;
|
|
if( GetWindowRect( _hwnd, &rcWnd ) )
|
|
{
|
|
for( UINT i = 0; i < cPts; i++ )
|
|
{
|
|
prgPts[i].x -= rcWnd.left;
|
|
prgPts[i].y -= rcWnd.top;
|
|
}
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::ScreenToWindow() - transforms non-empty rectangles from
|
|
// screen coords to window coords.
|
|
//
|
|
void CThemeWnd::ScreenToWindowRect( LPRECT prc )
|
|
{
|
|
if( !IsRectEmpty(prc) )
|
|
ScreenToWindow( (LPPOINT)prc, 2 );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::InitWindowMetrics()
|
|
//
|
|
// initializes theme resources
|
|
void CThemeWnd::InitWindowMetrics()
|
|
{
|
|
ZeroMemory( &_ncwm, sizeof(_ncwm) );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL _fClassicNcBtnMetricsReset = TRUE;
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// computes classic button position
|
|
BOOL _GetClassicNcBtnMetrics(
|
|
IN OPTIONAL NCWNDMET* pncwm,
|
|
IN HICON hAppIcon,
|
|
IN OPTIONAL BOOL fCanClose,
|
|
BOOL fRefresh )
|
|
{
|
|
static int cxEdge, cyEdge;
|
|
static int cxBtn, cyBtn, cxSmBtn, cySmBtn;
|
|
static RECT rcClose, rcMin, rcMax, rcHelp, rcSys;
|
|
static RECT rcSmClose;
|
|
static BOOL fInit = FALSE;
|
|
|
|
if( _fClassicNcBtnMetricsReset || fRefresh )
|
|
{
|
|
cxBtn = NcGetSystemMetrics( SM_CXSIZE );
|
|
cyBtn = NcGetSystemMetrics( SM_CYSIZE );
|
|
cxSmBtn = NcGetSystemMetrics( SM_CXSMSIZE );
|
|
cySmBtn = NcGetSystemMetrics( SM_CYSMSIZE );
|
|
cxEdge = NcGetSystemMetrics( SM_CXEDGE );
|
|
cyEdge = NcGetSystemMetrics( SM_CYEDGE );
|
|
|
|
// common top, w/ zero v-offset
|
|
rcClose.top = rcMin.top = rcMax.top = rcHelp.top = rcSys.top = rcSmClose.top = 0;
|
|
|
|
// common bottom, w/ zero v-offset
|
|
rcClose.bottom = rcMin.bottom = rcMax.bottom = rcHelp.bottom = rcClose.top + (cyBtn - (cyEdge * 2));
|
|
rcSmClose.bottom= (cySmBtn - (cyEdge * 2));
|
|
|
|
// sysmenu icon bottom
|
|
rcSys.bottom = rcSys.top + NcGetSystemMetrics(SM_CYSMICON);
|
|
|
|
// close, min, max left, right (as offsets from container's right boundary)
|
|
rcClose.right = -cxEdge;
|
|
rcClose.left = rcClose.right - (cxBtn - cxEdge);
|
|
|
|
rcMax.right = rcClose.left - cxEdge;
|
|
rcMax.left = rcMax.right - (cxBtn - cxEdge);
|
|
rcHelp = rcMax;
|
|
|
|
rcMin.right = rcMax.left;
|
|
rcMin.left = rcMin.right - (cxBtn - cxEdge);
|
|
|
|
// appicon left, right (as offsets from container's left boundary)
|
|
rcSys.left = cxEdge;
|
|
rcSys.right = rcSys.left + NcGetSystemMetrics(SM_CXSMICON);
|
|
|
|
// toolwindow close, left, right
|
|
rcSmClose.right = -cxEdge;
|
|
rcSmClose.left = rcSmClose.right - (cxSmBtn - cxEdge);
|
|
|
|
_fClassicNcBtnMetricsReset = FALSE;
|
|
}
|
|
|
|
if( !_fClassicNcBtnMetricsReset &&
|
|
pncwm && pncwm->fFrame && TESTFLAG(pncwm->dwStyle, WS_SYSMENU) )
|
|
{
|
|
NONCLIENTMETRICS ncm;
|
|
|
|
if( NcGetNonclientMetrics( &ncm, FALSE ) )
|
|
{
|
|
const RECT* prcBox = &pncwm->rcS0[NCRC_CAPTION];
|
|
int cnLOffset = prcBox->left + pncwm->cnBorders;
|
|
int cnROffset = prcBox->right - pncwm->cnBorders;
|
|
int cnCtrOffset = pncwm->cnBorders + prcBox->top +
|
|
(pncwm->fSmallFrame ? (ncm.iCaptionHeight - RECTHEIGHT(&rcClose))/2 :
|
|
(ncm.iSmCaptionHeight - RECTHEIGHT(&rcSmClose))/2);
|
|
|
|
// assign outbound rectangles.
|
|
// vertically center w/ respect to classic caption area,
|
|
// horizontally position w/ respect to respective container boundary.
|
|
|
|
// close button
|
|
pncwm->rcS0[NCRC_CLOSEBTN] = pncwm->fSmallFrame ? rcSmClose : rcClose;
|
|
OffsetRect( &pncwm->rcS0[NCRC_CLOSEBTN], cnROffset, cnCtrOffset );
|
|
|
|
pncwm->rawCloseBtnState = fCanClose ? CBS_NORMAL : CBS_DISABLED;
|
|
|
|
// (1) min/max/help/appicons not displayed for toolwindows
|
|
// (2) min/max btns mutually exclusive w/ context help btn
|
|
if( !TESTFLAG(pncwm->dwExStyle, WS_EX_TOOLWINDOW) )
|
|
{
|
|
// min, max
|
|
if( TESTFLAG(pncwm->dwStyle, WS_MINIMIZEBOX|WS_MAXIMIZEBOX) )
|
|
{
|
|
pncwm->rcS0[NCRC_MINBTN] = rcMin;
|
|
OffsetRect( &pncwm->rcS0[NCRC_MINBTN], cnROffset, cnCtrOffset );
|
|
|
|
pncwm->rcS0[NCRC_MAXBTN] = rcMax;
|
|
OffsetRect( &pncwm->rcS0[NCRC_MAXBTN], cnROffset, cnCtrOffset );
|
|
|
|
pncwm->iMaxButtonPart = pncwm->fMaxed ? WP_RESTOREBUTTON : WP_MAXBUTTON;
|
|
pncwm->iMinButtonPart = pncwm->fMin ? WP_RESTOREBUTTON : WP_MINBUTTON;
|
|
|
|
pncwm->rawMaxBtnState = TESTFLAG(pncwm->dwStyle, WS_MAXIMIZEBOX) ? CBS_NORMAL : CBS_DISABLED;
|
|
pncwm->rawMinBtnState = TESTFLAG(pncwm->dwStyle, WS_MINIMIZEBOX) ? CBS_NORMAL : CBS_DISABLED;
|
|
}
|
|
// help btn
|
|
else if( TESTFLAG(pncwm->dwExStyle, WS_EX_CONTEXTHELP) )
|
|
{
|
|
pncwm->rcS0[NCRC_HELPBTN] = rcHelp;
|
|
OffsetRect( &pncwm->rcS0[NCRC_HELPBTN], cnROffset, cnCtrOffset );
|
|
}
|
|
|
|
if( hAppIcon )
|
|
{
|
|
// sysmenu icon
|
|
pncwm->rcS0[NCRC_SYSBTN] = rcSys;
|
|
OffsetRect( &pncwm->rcS0[NCRC_SYSBTN], cnLOffset,
|
|
pncwm->cnBorders + prcBox->top + (ncm.iCaptionHeight - RECTHEIGHT(&rcSys))/2 );
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
return fInit;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// computes classic button position
|
|
BOOL _GetNcBtnMetrics(
|
|
IN OUT NCWNDMET* pncwm,
|
|
IN const NCTHEMEMET* pnctm,
|
|
IN HICON hAppIcon,
|
|
IN OPTIONAL BOOL fCanClose )
|
|
{
|
|
BOOL fRet = TRUE;
|
|
|
|
if( pncwm && pncwm->fFrame && TESTFLAG(pncwm->dwStyle, WS_SYSMENU) )
|
|
{
|
|
NONCLIENTMETRICS ncm;
|
|
fRet = NcGetNonclientMetrics( &ncm, FALSE );
|
|
if( fRet )
|
|
{
|
|
// (1) compute baseline rectangles
|
|
int cxEdge = NcGetSystemMetrics( SM_CXEDGE );
|
|
int cyEdge = NcGetSystemMetrics( SM_CYEDGE );
|
|
|
|
int cyBtn = NcGetSystemMetrics( SM_CYSIZE );
|
|
int cxBtn = MulDiv( cyBtn, pnctm->sizeBtn.cx, pnctm->sizeBtn.cy );
|
|
|
|
int cySmBtn = NcGetSystemMetrics( SM_CYSMSIZE );
|
|
int cxSmBtn = MulDiv( cySmBtn, pnctm->sizeSmBtn.cx, pnctm->sizeSmBtn.cy );
|
|
|
|
// remove padding from x,y
|
|
cyBtn -= (cyEdge * 2);
|
|
cxBtn -= (cyEdge * 2);
|
|
cySmBtn -= (cyEdge * 2);
|
|
cxSmBtn -= (cyEdge * 2);
|
|
|
|
RECT rcClose, rcMin, rcMax, rcHelp, rcSys, rcSmClose;
|
|
|
|
// common top, w/ zero v-offset
|
|
rcClose.top = rcMin.top = rcMax.top = rcHelp.top = rcSys.top = rcSmClose.top = 0;
|
|
|
|
// common bottom, w/ zero v-offset
|
|
rcClose.bottom = rcMin.bottom = rcMax.bottom = rcHelp.bottom =
|
|
max( rcClose.top, rcClose.top + cyBtn );
|
|
|
|
rcSmClose.bottom =
|
|
max( rcSmClose.top, cySmBtn );
|
|
|
|
// sysmenu icon bottom
|
|
rcSys.bottom = rcSys.top + NcGetSystemMetrics(SM_CYSMICON);
|
|
|
|
// close, min, max left, right (as offsets from container's right boundary)
|
|
rcClose.right = -cxEdge;
|
|
rcClose.left = rcClose.right - cxBtn;
|
|
|
|
rcMax.right = rcClose.left - cxEdge;
|
|
rcMax.left = rcMax.right - cxBtn;
|
|
rcHelp = rcMax;
|
|
|
|
rcMin.right = rcMax.left - cxEdge;
|
|
rcMin.left = rcMin.right - cxBtn;
|
|
|
|
// appicon left, right (as offsets from container's left boundary)
|
|
rcSys.left = cxEdge;
|
|
rcSys.right = rcSys.left + NcGetSystemMetrics(SM_CXSMICON);
|
|
|
|
// toolwindow close, left, right
|
|
rcSmClose.right = -cxEdge;
|
|
rcSmClose.left = rcSmClose.right - cxSmBtn;
|
|
|
|
const RECT* prcBox = &pncwm->rcS0[NCRC_CAPTION];
|
|
int cnLOffset = prcBox->left + pncwm->cnBorders;
|
|
int cnROffset = prcBox->right - pncwm->cnBorders;
|
|
int cnCtrOffset = pncwm->cnBorders + prcBox->top +
|
|
(pncwm->fSmallFrame ? (ncm.iCaptionHeight - RECTHEIGHT(&rcClose))/2 :
|
|
(ncm.iSmCaptionHeight - RECTHEIGHT(&rcSmClose))/2);
|
|
|
|
// (2) assign outbound rectangles.
|
|
// vertically center w/ respect to classic caption area,
|
|
// horizontally position w/ respect to respective container boundary.
|
|
|
|
// close button
|
|
pncwm->rcS0[NCRC_CLOSEBTN] = pncwm->fSmallFrame ? rcSmClose : rcClose;
|
|
OffsetRect( &pncwm->rcS0[NCRC_CLOSEBTN], cnROffset, cnCtrOffset );
|
|
|
|
pncwm->rawCloseBtnState = fCanClose ? CBS_NORMAL : CBS_DISABLED;
|
|
|
|
// (1) min/max/help/appicons not displayed for toolwindows
|
|
// (2) min/max btns mutually exclusive w/ context help btn
|
|
if( !TESTFLAG(pncwm->dwExStyle, WS_EX_TOOLWINDOW) )
|
|
{
|
|
// min, max
|
|
if( TESTFLAG(pncwm->dwStyle, WS_MINIMIZEBOX|WS_MAXIMIZEBOX) )
|
|
{
|
|
pncwm->rcS0[NCRC_MINBTN] = rcMin;
|
|
OffsetRect( &pncwm->rcS0[NCRC_MINBTN], cnROffset, cnCtrOffset );
|
|
|
|
pncwm->rcS0[NCRC_MAXBTN] = rcMax;
|
|
OffsetRect( &pncwm->rcS0[NCRC_MAXBTN], cnROffset, cnCtrOffset );
|
|
|
|
pncwm->iMaxButtonPart = pncwm->fMaxed ? WP_RESTOREBUTTON : WP_MAXBUTTON;
|
|
pncwm->iMinButtonPart = pncwm->fMin ? WP_RESTOREBUTTON : WP_MINBUTTON;
|
|
|
|
pncwm->rawMaxBtnState = TESTFLAG(pncwm->dwStyle, WS_MAXIMIZEBOX) ? CBS_NORMAL : CBS_DISABLED;
|
|
pncwm->rawMinBtnState = TESTFLAG(pncwm->dwStyle, WS_MINIMIZEBOX) ? CBS_NORMAL : CBS_DISABLED;
|
|
}
|
|
// help btn
|
|
else if( TESTFLAG(pncwm->dwExStyle, WS_EX_CONTEXTHELP) )
|
|
{
|
|
pncwm->rcS0[NCRC_HELPBTN] = rcHelp;
|
|
OffsetRect( &pncwm->rcS0[NCRC_HELPBTN], cnROffset, cnCtrOffset );
|
|
}
|
|
|
|
if( hAppIcon )
|
|
{
|
|
// sysmenu icon
|
|
pncwm->rcS0[NCRC_SYSBTN] = rcSys;
|
|
OffsetRect( &pncwm->rcS0[NCRC_SYSBTN], cnLOffset,
|
|
pncwm->cnBorders + prcBox->top + (ncm.iCaptionHeight - RECTHEIGHT(&rcSys))/2 );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return fRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::NcBackgroundHitTest() - hit test the Caption or Frame part
|
|
//
|
|
WORD CThemeWnd::NcBackgroundHitTest(
|
|
POINT ptHit, LPCRECT prcWnd,
|
|
DWORD dwStyle, DWORD dwExStyle,
|
|
FRAMESTATES fs,
|
|
const WINDOWPARTS rgiParts[],
|
|
const WINDOWPARTS rgiTemplates[],
|
|
const RECT rgrcParts[] )
|
|
{
|
|
WORD hitcode = HTNOWHERE;
|
|
HRESULT hr = E_FAIL;
|
|
eFRAMEPARTS iPartHit = (eFRAMEPARTS)-1;
|
|
|
|
// do a standard rect hit test:
|
|
for( int i = 0; i < cFRAMEPARTS; i++ )
|
|
{
|
|
if( _StrictPtInRect(&rgrcParts[i], ptHit) )
|
|
{
|
|
iPartHit = (eFRAMEPARTS)i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if( iPartHit >= 0 )
|
|
{
|
|
BOOL fResizing = TESTFLAG(dwStyle, WS_THICKFRAME);
|
|
DWORD dwHTFlags = fResizing ? HTTB_RESIZINGBORDER : HTTB_FIXEDBORDER;
|
|
|
|
RECT rcHit = rgrcParts[iPartHit];
|
|
|
|
switch( iPartHit )
|
|
{
|
|
case iCAPTION:
|
|
// Ensure caption rect and test point are zero-relative to
|
|
// the correct origin (if we have a window region,
|
|
// this would be window origin, otherwise, it's the part origin.)
|
|
if( _hrgnWnd != NULL )
|
|
rcHit = *prcWnd;
|
|
if( fResizing )
|
|
dwHTFlags &= ~HTTB_RESIZINGBORDER_BOTTOM;
|
|
break;
|
|
|
|
case iFRAMEBOTTOM:
|
|
if( fResizing )
|
|
dwHTFlags &= ~HTTB_RESIZINGBORDER_TOP;
|
|
break;
|
|
|
|
case iFRAMELEFT:
|
|
if( fResizing )
|
|
dwHTFlags = HTTB_RESIZINGBORDER_LEFT;
|
|
break;
|
|
|
|
case iFRAMERIGHT:
|
|
if( fResizing )
|
|
dwHTFlags = HTTB_RESIZINGBORDER_RIGHT;
|
|
break;
|
|
}
|
|
|
|
ptHit.x -= prcWnd->left;
|
|
ptHit.y -= prcWnd->top;
|
|
OffsetRect( &rcHit, -prcWnd->left, -prcWnd->top );
|
|
|
|
|
|
// Here our assumption is that the hit testing for the template
|
|
// is "as good" or "better" than the rectangles checking applied
|
|
// to the caption part. So we do one or the other. There are
|
|
// situations where you would need to do both (if the template
|
|
// were outside the window region and you were able to get USER to
|
|
// send you NcHitTest messages for it). For those situations
|
|
// you would need to call both so that you can distinguish between
|
|
// a mouse hit over the caption "client" area vs. over the
|
|
// outside-transparent area.
|
|
if( VALID_WINDOWPART(rgiTemplates[iPartHit]) )
|
|
{
|
|
hr = HitTestThemeBackground( _hTheme, NULL, rgiTemplates[iPartHit], fs,
|
|
dwHTFlags | (fResizing ? HTTB_SIZINGTEMPLATE : 0),
|
|
&rcHit, _rghrgnSizingTemplates[iPartHit], ptHit, &hitcode );
|
|
}
|
|
else
|
|
{
|
|
hr = HitTestThemeBackground( _hTheme, NULL, rgiParts[iPartHit], fs,
|
|
dwHTFlags | (fResizing ? HTTB_SYSTEMSIZINGMARGINS : 0),
|
|
&rcHit, _hrgnWnd, ptHit, &hitcode );
|
|
}
|
|
|
|
if( SUCCEEDED(hr) )
|
|
{
|
|
if( iCAPTION == iPartHit && (HTCLIENT == hitcode || HTBORDER == hitcode) )
|
|
hitcode = HTCAPTION;
|
|
}
|
|
}
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
hitcode = HTNOWHERE;
|
|
}
|
|
|
|
return hitcode;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::TrackFrameButton() - track the mouse over the caption buttons,
|
|
// pressing/releasing as appropriate. Return back SC_* command to report or 0
|
|
// if the mouse was released off of the button.
|
|
//
|
|
BOOL CThemeWnd::TrackFrameButton(
|
|
HWND hwnd,
|
|
INT iHitCode,
|
|
OUT OPTIONAL WPARAM* puSysCmd,
|
|
BOOL fHottrack )
|
|
{
|
|
int iStateId, iNewStateId;
|
|
int iPartId = -1;
|
|
UINT cmd = 0;
|
|
MSG msg = {0};
|
|
LPRECT prcBtnPaint = NULL;
|
|
RECT rcBtnTrack;
|
|
HDC hdc;
|
|
|
|
if (puSysCmd)
|
|
{
|
|
*puSysCmd = 0;
|
|
}
|
|
|
|
// map iHitCode to the correct part number
|
|
switch (iHitCode)
|
|
{
|
|
case HTHELP:
|
|
cmd = SC_CONTEXTHELP;
|
|
iPartId = WP_HELPBUTTON;
|
|
prcBtnPaint = &_ncwm.rcW0[NCRC_HELPBTN];
|
|
break;
|
|
|
|
case HTCLOSE:
|
|
cmd = SC_CLOSE;
|
|
iPartId = _ncwm.fSmallFrame ? WP_SMALLCLOSEBUTTON : WP_CLOSEBUTTON;
|
|
prcBtnPaint = &_ncwm.rcW0[NCRC_CLOSEBTN];
|
|
break;
|
|
|
|
case HTMINBUTTON:
|
|
cmd = _ncwm.fMin ? SC_RESTORE : SC_MINIMIZE;
|
|
iPartId = _ncwm.iMinButtonPart;
|
|
prcBtnPaint = &_ncwm.rcW0[NCRC_MINBTN];
|
|
break;
|
|
|
|
case HTMAXBUTTON:
|
|
cmd = _ncwm.fMaxed ? SC_RESTORE : SC_MAXIMIZE;
|
|
iPartId = _ncwm.iMaxButtonPart;
|
|
prcBtnPaint = &_ncwm.rcW0[NCRC_MAXBTN];
|
|
break;
|
|
|
|
case HTSYSMENU:
|
|
if (puSysCmd)
|
|
{
|
|
*puSysCmd = SC_MOUSEMENU | iHitCode;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
// If we didn't recognize the hit code there's nothing to track
|
|
if (iPartId >= 0 )
|
|
{
|
|
// Get the window DC, in window coords
|
|
hdc = _GetNonclientDC(_hwnd, NULL);
|
|
if ( hdc )
|
|
{
|
|
// Don't paint in the window's content area, clip to the content area
|
|
ExcludeClipRect( hdc, _ncwm.rcW0[NCRC_CONTENT].left,
|
|
_ncwm.rcW0[NCRC_CONTENT].top,
|
|
_ncwm.rcW0[NCRC_CONTENT].right,
|
|
_ncwm.rcW0[NCRC_CONTENT].bottom );
|
|
|
|
// Calculate the tracking rect. We track a larger button rect when maximized
|
|
// but paint into the normal sized rect.
|
|
rcBtnTrack = *prcBtnPaint;
|
|
_GetNcBtnHitTestRect( &_ncwm, iHitCode, TRUE, &rcBtnTrack );
|
|
|
|
// when tracking MDI child window frame buttons, clip to their
|
|
// parent rect.
|
|
if ( TESTFLAG(GetWindowLong(hwnd, GWL_EXSTYLE), WS_EX_MDICHILD) )
|
|
{
|
|
RECT rcMDIClient;
|
|
|
|
GetWindowRect(GetParent(hwnd), &rcMDIClient);
|
|
ScreenToWindowRect(&rcMDIClient);
|
|
InflateRect(&rcMDIClient, -NcGetSystemMetrics(SM_CXEDGE), -NcGetSystemMetrics(SM_CYEDGE));
|
|
IntersectClipRect(hdc, rcMDIClient.left, rcMDIClient.top, rcMDIClient.right, rcMDIClient.bottom);
|
|
}
|
|
|
|
if (fHottrack)
|
|
{
|
|
// draw the button hot if the mouse is over it
|
|
iStateId = (iHitCode == _htHot) ? SBS_HOT : CBS_NORMAL;
|
|
}
|
|
else
|
|
{
|
|
// draw the button depressed
|
|
iStateId = SBS_PUSHED;
|
|
}
|
|
|
|
iStateId = MAKE_BTNSTATE(_ncwm.framestate, iStateId);
|
|
NcDrawThemeBackground(_hTheme, hdc, iPartId, iStateId, prcBtnPaint, 0);
|
|
|
|
// TODO NotifyWinEvent(EVENT_OBJECT_STATECHANGE, pwnd, OBJID_TITLEBAR, iButton, 0);
|
|
|
|
|
|
if ( !fHottrack )
|
|
{
|
|
BOOL fTrack, fMouseUp = FALSE;
|
|
SetCapture(hwnd); // take mouse capture
|
|
|
|
do // mouse button tracking loop
|
|
{
|
|
fTrack = FALSE;
|
|
|
|
// Let's go to sleep, to be awakened only on a mouse message placed in our
|
|
// thread's queue.
|
|
|
|
switch (MsgWaitForMultipleObjectsEx(0, NULL, INFINITE /*why consume CPU processing a timeout when we don't have to?*/,
|
|
QS_MOUSE, MWMO_INPUTAVAILABLE))
|
|
{
|
|
case WAIT_OBJECT_0: // a mouse message or important system event has been queued
|
|
|
|
|
|
if (PeekMessage(&msg, NULL, WM_MOUSEFIRST, WM_MOUSELAST, PM_REMOVE))
|
|
{
|
|
|
|
// PeekMessage returns a point in screen relative coords. Mirror the
|
|
// point it this is a RTL window. Translate the point to window coords.
|
|
if ( TESTFLAG(_ncwm.dwExStyle, WS_EX_LAYOUTRTL) )
|
|
{
|
|
// mirror the point to hittest correctly
|
|
MIRROR_POINT(_ncwm.rcS0[NCRC_WINDOW], msg.pt);
|
|
}
|
|
ScreenToWindow( &msg.pt, 1 );
|
|
|
|
if (msg.message == WM_LBUTTONUP)
|
|
{
|
|
ReleaseCapture();
|
|
fMouseUp = TRUE;
|
|
}
|
|
else if ((msg.message == WM_MOUSEMOVE) && cmd)
|
|
{
|
|
iNewStateId = MAKE_BTNSTATE(_ncwm.framestate, PtInRect(&rcBtnTrack, msg.pt) ? SBS_PUSHED : SBS_NORMAL);
|
|
|
|
if (iStateId != iNewStateId)
|
|
{
|
|
iStateId = iNewStateId;
|
|
NcDrawThemeBackground(_hTheme, hdc, iPartId, iStateId, prcBtnPaint, 0);
|
|
// TODO NotifyWinEvent(EVENT_OBJECT_STATECHANGE, pwnd, OBJID_TITLEBAR, iButton, 0);
|
|
}
|
|
fTrack = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Check loss of capture. This can happen if we loose activation
|
|
// via alt-tab and may not have received a WM_CAPTURECHANGED message
|
|
|
|
if (GetCapture() != hwnd)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
// Dequeue CAPTURECHANGED
|
|
if (PeekMessage(&msg, NULL, WM_CAPTURECHANGED, WM_CAPTURECHANGED, PM_REMOVE) ||
|
|
fMouseUp)
|
|
{
|
|
break;
|
|
}
|
|
|
|
|
|
fTrack = TRUE; // go back to sleep until the next mouse event
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
|
|
}
|
|
} while (fTrack);
|
|
|
|
// draw button in normal state if it is not in that state already
|
|
iNewStateId = MAKE_BTNSTATE(_ncwm.framestate, CBS_NORMAL);
|
|
if (iStateId != iNewStateId)
|
|
{
|
|
NcDrawThemeBackground(_hTheme, hdc, iPartId, iNewStateId, prcBtnPaint, 0);
|
|
}
|
|
|
|
// if we did not end up on a button return 0
|
|
if( puSysCmd && (*puSysCmd = cmd) != 0 )
|
|
{
|
|
// TODO NotifyWinEvent(EVENT_OBJECT_STATECHANGE, pwnd, OBJID_TITLEBAR, iButton, 0);
|
|
|
|
// If mouse wasn't released over the button, cancel the command.
|
|
if( !(fMouseUp && PtInRect(&rcBtnTrack, msg.pt)) )
|
|
*puSysCmd = 0;
|
|
}
|
|
|
|
}
|
|
// Done with DC now
|
|
ReleaseDC(_hwnd, hdc);
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
DWORD GetTextAlignFlags(HTHEME hTheme, IN NCWNDMET* pncwm, BOOL fReverse)
|
|
{
|
|
CONTENTALIGNMENT contentAlignment = CA_LEFT;
|
|
DWORD dwAlignFlags = 0;
|
|
|
|
//---- compute text alignment ----
|
|
GetThemeInt(hTheme, pncwm->rgframeparts[iCAPTION], pncwm->framestate, TMT_CONTENTALIGNMENT,
|
|
(int *)&contentAlignment);
|
|
|
|
if (fReverse)
|
|
{
|
|
//---- reverse alignment ----
|
|
switch(contentAlignment)
|
|
{
|
|
default:
|
|
case CA_LEFT: dwAlignFlags |= DT_RIGHT; break;
|
|
case CA_CENTER: dwAlignFlags |= DT_CENTER; break;
|
|
case CA_RIGHT: dwAlignFlags |= DT_LEFT; break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//---- normal alignment ----
|
|
switch(contentAlignment)
|
|
{
|
|
default:
|
|
case CA_LEFT: dwAlignFlags |= DT_LEFT; break;
|
|
case CA_CENTER: dwAlignFlags |= DT_CENTER; break;
|
|
case CA_RIGHT: dwAlignFlags |= DT_RIGHT; break;
|
|
}
|
|
}
|
|
|
|
return dwAlignFlags;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void _BorderRect( HDC hdc, COLORREF rgb, LPCRECT prc, int cxBorder, int cyBorder )
|
|
{
|
|
COLORREF rgbSave = SetBkColor( hdc, rgb );
|
|
RECT rc = *prc;
|
|
|
|
// bottom border
|
|
rc = *prc; rc.top = prc->bottom - cyBorder;
|
|
ExtTextOut( hdc, rc.left, rc.top, ETO_OPAQUE, &rc, NULL, 0, NULL );
|
|
|
|
// right border
|
|
rc = *prc; rc.left = prc->right - cxBorder;
|
|
ExtTextOut( hdc, rc.left, rc.top, ETO_OPAQUE, &rc, NULL, 0, NULL );
|
|
|
|
// left border
|
|
rc = *prc; rc.right = prc->left + cxBorder;
|
|
ExtTextOut( hdc, rc.left, rc.top, ETO_OPAQUE, &rc, NULL, 0, NULL );
|
|
|
|
// top border
|
|
rc = *prc; rc.bottom = prc->top + cyBorder;
|
|
ExtTextOut( hdc, rc.left, rc.top, ETO_OPAQUE, &rc, NULL, 0, NULL );
|
|
|
|
SetBkColor( hdc, rgbSave );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void _DrawWindowEdges( HDC hdc, NCWNDMET* pncwm, BOOL fIsFrame )
|
|
{
|
|
// non-frame window edge & border
|
|
if( !fIsFrame )
|
|
{
|
|
RECT rcWnd = pncwm->rcW0[NCRC_WINDOW];
|
|
|
|
int cxBorder = NcGetSystemMetrics(SM_CXBORDER),
|
|
cyBorder = NcGetSystemMetrics(SM_CYBORDER);
|
|
|
|
// static, window edge
|
|
if( TESTFLAG(pncwm->dwExStyle, WS_EX_WINDOWEDGE) )
|
|
{
|
|
RECT rcClip = rcWnd;
|
|
|
|
InflateRect( &rcClip, -pncwm->cnBorders, -pncwm->cnBorders );
|
|
ExcludeClipRect( hdc, rcClip.left, rcClip.top, rcClip.right, rcClip.bottom );
|
|
DrawEdge( hdc, &rcWnd, EDGE_RAISED, BF_RECT | BF_ADJUST | BF_MIDDLE);
|
|
SelectClipRgn( hdc, NULL );
|
|
}
|
|
else if( TESTFLAG(pncwm->dwExStyle, WS_EX_STATICEDGE) )
|
|
{
|
|
DrawEdge( hdc, &rcWnd, BDR_SUNKENOUTER, BF_RECT | BF_ADJUST );
|
|
}
|
|
// Normal border
|
|
else if( TESTFLAG(pncwm->dwStyle, WS_BORDER) )
|
|
{
|
|
_BorderRect( hdc, GetSysColor( COLOR_WINDOWFRAME),
|
|
&rcWnd, cxBorder, cyBorder );
|
|
}
|
|
}
|
|
|
|
// client edge
|
|
if( TESTFLAG(pncwm->dwExStyle, WS_EX_CLIENTEDGE) )
|
|
{
|
|
#ifdef _TEST_CLIENTEDGE_
|
|
|
|
HBRUSH hbr = CreateSolidBrush( RGB(255,0,255) );
|
|
FillRect(hdc, &ncwm.rcW0[NCRC_CLIENTEDGE], hbr);
|
|
DeleteObject(hbr);
|
|
|
|
#else _TEST_CLIENTEDGE_
|
|
|
|
DrawEdge( hdc, &pncwm->rcW0[NCRC_CLIENTEDGE], EDGE_SUNKEN, BF_RECT | BF_ADJUST);
|
|
|
|
#endif _TEST_CLIENTEDGE_
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::NcPaintCaption(
|
|
IN HDC hdcOut,
|
|
IN NCWNDMET* pncwm,
|
|
IN BOOL fBuffered,
|
|
IN DWORD dwCaptionFlags, // draw caption flags (DC_xxx, winuser.h)
|
|
IN DTBGOPTS* pdtbopts )
|
|
{
|
|
ASSERT(hdcOut);
|
|
ASSERT(pncwm);
|
|
ASSERT(pncwm->fFrame);
|
|
ASSERT(HAS_CAPTIONBAR(pncwm->dwStyle));
|
|
|
|
DWORD dwOldAlign = 0;
|
|
|
|
// caption text implies caption background
|
|
if( TESTFLAG( dwCaptionFlags, DC_TEXT|DC_ICON ) || 0 == dwCaptionFlags )
|
|
{
|
|
dwCaptionFlags = DC_ENTIRECAPTION;
|
|
}
|
|
|
|
if( dwCaptionFlags != DC_ENTIRECAPTION
|
|
#if defined(DEBUG) && defined(DEBUG_NCPAINT)
|
|
|| TESTFLAG( _NcTraceFlags, NCTF_NCPAINT )
|
|
#endif DEBUG
|
|
)
|
|
{
|
|
fBuffered = FALSE;
|
|
}
|
|
|
|
// create caption double buffer
|
|
HBITMAP hbmBuf = fBuffered ? CreateCompatibleBitmap(hdcOut, RECTWIDTH(&pncwm->rcW0[NCRC_CAPTION]),
|
|
RECTHEIGHT(&pncwm->rcW0[NCRC_CAPTION])) :
|
|
NULL;
|
|
|
|
if( !fBuffered || hbmBuf )
|
|
{
|
|
HDC hdc = fBuffered ? CreateCompatibleDC(hdcOut) : hdcOut;
|
|
if( hdc )
|
|
{
|
|
//--- DO NOT EXIT FROM WITHIN THIS CONDITIONAL ---//
|
|
EnterNcThemePaint();
|
|
|
|
HBITMAP hbm0 = fBuffered ? (HBITMAP)SelectObject(hdc, hbmBuf) : NULL;
|
|
|
|
if( TESTFLAG( dwCaptionFlags, DC_BACKGROUND ) )
|
|
{
|
|
// Draw caption background
|
|
|
|
RECT rcBkgnd = pncwm->rcW0[NCRC_CAPTION];
|
|
if( pncwm->fFullMaxed )
|
|
{
|
|
rcBkgnd.top += pncwm->cnBorders;
|
|
rcBkgnd.left += pncwm->cnBorders;
|
|
rcBkgnd.right -= pncwm->cnBorders;
|
|
}
|
|
NcDrawThemeBackgroundEx( _hTheme, hdc, pncwm->rgframeparts[iCAPTION], pncwm->framestate,
|
|
&rcBkgnd, pdtbopts );
|
|
}
|
|
|
|
if( TESTFLAG( dwCaptionFlags, DC_BUTTONS ) )
|
|
{
|
|
// Draw standard caption buttons
|
|
if (!IsRectEmpty(&pncwm->rcW0[NCRC_CLOSEBTN]))
|
|
{
|
|
NcDrawThemeBackground( _hTheme, hdc, pncwm->fSmallFrame ? WP_SMALLCLOSEBUTTON : WP_CLOSEBUTTON,
|
|
MAKE_BTNSTATE(pncwm->framestate, pncwm->rawCloseBtnState),
|
|
&pncwm->rcW0[NCRC_CLOSEBTN], 0);
|
|
}
|
|
|
|
if (!IsRectEmpty(&pncwm->rcW0[NCRC_MAXBTN]))
|
|
{
|
|
|
|
NcDrawThemeBackground(_hTheme, hdc, pncwm->iMaxButtonPart,
|
|
MAKE_BTNSTATE(pncwm->framestate, pncwm->rawMaxBtnState),
|
|
&pncwm->rcW0[NCRC_MAXBTN], 0);
|
|
}
|
|
|
|
if (!IsRectEmpty(&pncwm->rcW0[NCRC_MINBTN]))
|
|
{
|
|
NcDrawThemeBackground( _hTheme, hdc, pncwm->iMinButtonPart,
|
|
MAKE_BTNSTATE(pncwm->framestate, pncwm->rawMinBtnState),
|
|
&pncwm->rcW0[NCRC_MINBTN], 0);
|
|
}
|
|
|
|
if (!IsRectEmpty(&pncwm->rcW0[NCRC_HELPBTN]))
|
|
NcDrawThemeBackground(_hTheme, hdc, WP_HELPBUTTON, MAKE_BTNSTATE(pncwm->framestate, CBS_NORMAL),
|
|
&pncwm->rcW0[NCRC_HELPBTN], 0);
|
|
}
|
|
|
|
// Draw sysmenu icon
|
|
if( TESTFLAG( dwCaptionFlags, DC_ICON ) )
|
|
{
|
|
if (!IsRectEmpty(&pncwm->rcW0[NCRC_SYSBTN]) && _hAppIcon)
|
|
{
|
|
#define MAX_APPICON_RETRIES 1
|
|
int cRetries = 0;
|
|
|
|
DWORD dwLayout = GetLayout(hdc);
|
|
if( GDI_ERROR != dwLayout && TESTFLAG(dwLayout, LAYOUT_RTL) )
|
|
{
|
|
SetLayout(hdc, dwLayout|LAYOUT_BITMAPORIENTATIONPRESERVED);
|
|
}
|
|
|
|
do
|
|
{
|
|
// note: we don't draw sysmenu icon mirrored
|
|
if( DrawIconEx(hdc, pncwm->rcW0[NCRC_SYSBTN].left, pncwm->rcW0[NCRC_SYSBTN].top, _hAppIcon,
|
|
RECTWIDTH(&pncwm->rcW0[NCRC_SYSBTN]), RECTHEIGHT(&pncwm->rcW0[NCRC_SYSBTN]),
|
|
0, NULL, DI_NORMAL))
|
|
{
|
|
break; // success; done
|
|
}
|
|
|
|
// failure; try recycling the handle
|
|
if( _hAppIcon && GetLastError() == ERROR_INVALID_CURSOR_HANDLE )
|
|
{
|
|
_hAppIcon = NULL;
|
|
AcquireFrameIcon( pncwm->dwStyle, pncwm->dwExStyle, FALSE );
|
|
|
|
if( (++cRetries) > MAX_APPICON_RETRIES )
|
|
{
|
|
_hAppIcon = NULL; // failed to retrieve a new icon handle; bail for good.
|
|
}
|
|
}
|
|
|
|
} while( _hAppIcon && cRetries <= MAX_APPICON_RETRIES );
|
|
|
|
if( GDI_ERROR != dwLayout )
|
|
{
|
|
SetLayout(hdc, dwLayout);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
if( TESTFLAG( dwCaptionFlags, DC_TEXT ) )
|
|
{
|
|
// Draw caption text
|
|
HFONT hf0 = NULL;
|
|
DWORD dwDTFlags = DT_SINGLELINE | DT_NOPREFIX | DT_END_ELLIPSIS;
|
|
BOOL fSelFont = FALSE;
|
|
LPWSTR pszText = _AllocWindowText(_hwnd);
|
|
|
|
if( pszText && *pszText )
|
|
{
|
|
// Compute frame text rect
|
|
if( pncwm->hfCaption )
|
|
{
|
|
hf0 = (HFONT)SelectObject( hdc, pncwm->hfCaption );
|
|
fSelFont = TRUE;
|
|
}
|
|
|
|
//---- compute text alignment ----
|
|
BOOL fReverse = TESTFLAG(_ncwm.dwExStyle, WS_EX_RIGHT);
|
|
|
|
dwDTFlags |= GetTextAlignFlags(_hTheme, pncwm, fReverse);
|
|
}
|
|
|
|
//---- adjust text align for WS_EX_RTLREADING ----
|
|
if (TESTFLAG(_ncwm.dwExStyle, WS_EX_RTLREADING))
|
|
dwOldAlign = SetTextAlign(hdc, TA_RTLREADING | GetTextAlign(hdc));
|
|
|
|
if( pszText && *pszText )
|
|
{
|
|
//---- set options for DrawThemeText() ----
|
|
DTTOPTS DttOpts = {sizeof(DttOpts)};
|
|
DttOpts.dwFlags = DTT_TEXTCOLOR;
|
|
DttOpts.crText = pncwm->rgbCaption;
|
|
|
|
Log(LOG_RFBUG, L"Drawing Caption Text: left=%d, state=%d, text=%s",
|
|
pncwm->rcW0[NCRC_CAPTIONTEXT].left, pncwm->framestate, pszText);
|
|
|
|
//---- draw the caption text ----
|
|
DrawThemeTextEx(_hTheme, hdc, pncwm->rgframeparts[iCAPTION], pncwm->framestate,
|
|
pszText, -1, dwDTFlags, &pncwm->rcW0[NCRC_CAPTIONTEXT], &DttOpts);
|
|
}
|
|
|
|
//---- free the text, if temp. allocated ----
|
|
SAFE_DELETE_ARRAY(pszText)
|
|
|
|
//---- draw the "Comments?" text ----
|
|
SetBkMode( hdc, TRANSPARENT );
|
|
SetTextColor( hdc, pncwm->rgbCaption );
|
|
DrawLameButton(hdc, pncwm);
|
|
|
|
//---- restore the text align ----
|
|
if (TESTFLAG(_ncwm.dwExStyle, WS_EX_RTLREADING))
|
|
SetTextAlign(hdc, dwOldAlign);
|
|
|
|
if( fSelFont )
|
|
{
|
|
SelectObject(hdc, hf0);
|
|
}
|
|
}
|
|
|
|
if( hdc != hdcOut )
|
|
{
|
|
// Slap the bits on the output DC.
|
|
BitBlt( hdcOut, pncwm->rcW0[NCRC_CAPTION].left, pncwm->rcW0[NCRC_CAPTION].top,
|
|
WIDTH(pncwm->rcW0[NCRC_CAPTION]), HEIGHT(pncwm->rcW0[NCRC_CAPTION]),
|
|
hdc, 0, 0, SRCCOPY );
|
|
SelectObject(hdc, hbm0);
|
|
DeleteDC(hdc);
|
|
}
|
|
|
|
LeaveNcThemePaint();
|
|
}
|
|
DeleteObject( hbmBuf );
|
|
}
|
|
|
|
if( IsWindowVisible(_hwnd) )
|
|
{
|
|
SetRenderedNcPart(RNCF_CAPTION);
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CThemeWnd::NcPaint() - NC painting worker
|
|
//
|
|
void CThemeWnd::NcPaint(
|
|
IN OPTIONAL HDC hdcIn,
|
|
IN ULONG dwFlags,
|
|
IN OPTIONAL HRGN hrgnUpdate,
|
|
IN OPTIONAL NCPAINTOVERIDE* pncpo)
|
|
{
|
|
NCTHEMEMET nctm;
|
|
NCWNDMET* pncwm = NULL;
|
|
HDC hdc = NULL;
|
|
|
|
if( _cLockRedraw > 0 )
|
|
return;
|
|
|
|
// Compute all metrics before painting:
|
|
if (pncpo) // preview override
|
|
{
|
|
ASSERT(hdcIn);
|
|
hdc = hdcIn;
|
|
pncwm = pncpo->pncwm;
|
|
nctm = pncpo->nctm;
|
|
}
|
|
else // live window
|
|
{
|
|
if( !GetNcWindowMetrics( NULL, &pncwm, &nctm, NCWMF_RECOMPUTE ) )
|
|
return;
|
|
|
|
// Ensure status bits reflect caller's intention for frame state
|
|
if( dwFlags != NCPF_DEFAULT )
|
|
{
|
|
_ComputeNcWindowStatus( _hwnd, TESTFLAG(dwFlags, NCPF_ACTIVEFRAME) ? WS_ACTIVECAPTION : 0, pncwm );
|
|
}
|
|
|
|
hdc = hdcIn ? hdcIn : _GetNonclientDC( _hwnd, hrgnUpdate );
|
|
|
|
if (! hdc)
|
|
{
|
|
//---- don't assert here since stress (out of memory) could cause a legit failure ----
|
|
Log(LOG_ALWAYS, L"call to GetDCEx() for nonclient painting failed");
|
|
}
|
|
}
|
|
|
|
if( hdc != NULL )
|
|
{
|
|
//--- DO NOT EXIT FROM WITHIN THIS CONDITIONAL ---//
|
|
|
|
BEGIN_DEBUG_NCPAINT();
|
|
EnterNcThemePaint();
|
|
|
|
// Clip to content rect (alleviates flicker in menubar and scrollbars as we paint background)
|
|
RECT rcClip;
|
|
rcClip = pncwm->rcW0[NCRC_CONTENT];
|
|
if( TESTFLAG(pncwm->dwExStyle, WS_EX_LAYOUTRTL) )
|
|
{
|
|
// mirror the clip rect relative to the window rect
|
|
// and apply that as the clipping region for the dc
|
|
MIRROR_RECT(pncwm->rcW0[NCRC_WINDOW], rcClip);
|
|
}
|
|
|
|
ExcludeClipRect( hdc, rcClip.left, rcClip.top, rcClip.right, rcClip.bottom );
|
|
|
|
if( pncwm->fFrame )
|
|
{
|
|
//---- DrawThemeBackgroundEx() options ----
|
|
DTBGOPTS dtbopts = {sizeof(dtbopts)};
|
|
DTBGOPTS *pdtbopts = NULL;
|
|
|
|
// if not drawing preview, set "draw solid" option
|
|
if(!pncpo)
|
|
{
|
|
// Because of the interleaving of NCPAINT and SetWindowRgn, drawing solid results
|
|
// in some flicker and transparency bleed. Commenting this out for now [scotthan]
|
|
//dtbopts.dwFlags |= DTBG_DRAWSOLID;
|
|
pdtbopts = &dtbopts;
|
|
}
|
|
|
|
// Frame Background
|
|
if( pncwm->fMin )
|
|
{
|
|
NcDrawThemeBackgroundEx( _hTheme, hdc, WP_MINCAPTION, pncwm->framestate,
|
|
&pncwm->rcW0[NCRC_CAPTION], pdtbopts ) ;
|
|
}
|
|
else if( !pncwm->fFullMaxed )
|
|
{
|
|
NcDrawThemeBackgroundEx( _hTheme, hdc, pncwm->rgframeparts[iFRAMELEFT], pncwm->framestate,
|
|
&pncwm->rcW0[NCRC_FRAMELEFT], pdtbopts );
|
|
NcDrawThemeBackgroundEx( _hTheme, hdc, pncwm->rgframeparts[iFRAMERIGHT], pncwm->framestate,
|
|
&pncwm->rcW0[NCRC_FRAMERIGHT], pdtbopts );
|
|
NcDrawThemeBackgroundEx( _hTheme, hdc, pncwm->rgframeparts[iFRAMEBOTTOM], pncwm->framestate,
|
|
&pncwm->rcW0[NCRC_FRAMEBOTTOM], pdtbopts );
|
|
}
|
|
|
|
SetRenderedNcPart(RNCF_FRAME);
|
|
|
|
// Caption
|
|
NcPaintCaption( hdc, pncwm, !(pncwm->fMin || pncwm->fFullMaxed || pncpo),
|
|
DC_ENTIRECAPTION, pdtbopts );
|
|
}
|
|
|
|
// Clip to client rect
|
|
SelectClipRgn( hdc, NULL );
|
|
|
|
// Menubar
|
|
if( !(pncwm->fMin || TESTFLAG(pncwm->dwStyle, WS_CHILD))
|
|
&& !IsRectEmpty(&pncwm->rcW0[NCRC_MENUBAR]) )
|
|
{
|
|
RECT rcMenuBar = pncwm->rcW0[NCRC_MENUBAR];
|
|
BOOL fClip = RECTHEIGHT(&rcMenuBar) < pncwm->cyMenu;
|
|
|
|
if( fClip )
|
|
{
|
|
IntersectClipRect( hdc, rcMenuBar.left, rcMenuBar.top,
|
|
rcMenuBar.right, rcMenuBar.bottom );
|
|
}
|
|
|
|
PaintMenuBar( _hwnd, hdc, pncwm->cnMenuOffsetLeft,
|
|
pncwm->cnMenuOffsetRight, pncwm->cnMenuOffsetTop,
|
|
TESTFLAG(pncwm->framestate, FS_ACTIVE) ? PMB_ACTIVE : 0 );
|
|
|
|
// deal with unpainted menubar pixels:
|
|
if( nctm.dyMenuBar > 0 && RECTHEIGHT(&pncwm->rcW0[NCRC_MENUBAR]) >= pncwm->cyMenu )
|
|
{
|
|
rcMenuBar.top = rcMenuBar.bottom - nctm.dyMenuBar;
|
|
COLORREF rgbBk = SetBkColor( hdc, GetSysColor(COLOR_MENU) );
|
|
ExtTextOut(hdc, rcMenuBar.left, rcMenuBar.top, ETO_OPAQUE, &rcMenuBar, NULL, 0, NULL );
|
|
SetBkColor( hdc, rgbBk );
|
|
}
|
|
|
|
if( fClip )
|
|
SelectClipRgn( hdc, NULL );
|
|
}
|
|
|
|
// Scrollbars
|
|
if( !pncwm->fMin )
|
|
{
|
|
// Draw static, window, client edges.
|
|
_DrawWindowEdges( hdc, pncwm, pncwm->fFrame );
|
|
|
|
RECT rcVScroll = pncwm->rcW0[NCRC_VSCROLL];
|
|
if ( TESTFLAG(pncwm->dwExStyle, WS_EX_LAYOUTRTL) )
|
|
{
|
|
MIRROR_RECT(pncwm->rcW0[NCRC_WINDOW], rcVScroll);
|
|
}
|
|
|
|
if( TESTFLAG(pncwm->dwStyle, WS_VSCROLL) &&
|
|
( HasRenderedNcPart(RNCF_SCROLLBAR) || RectVisible(hdc, &rcVScroll)) )
|
|
{
|
|
if( TESTFLAG(pncwm->dwStyle, WS_HSCROLL) )
|
|
{
|
|
|
|
// Draw sizebox.
|
|
RECT rcSizeBox = pncwm->rcW0[NCRC_SIZEBOX];
|
|
|
|
if ( TESTFLAG(pncwm->dwExStyle, WS_EX_LAYOUTRTL) )
|
|
{
|
|
MIRROR_RECT(pncwm->rcW0[NCRC_WINDOW], rcSizeBox);
|
|
}
|
|
|
|
DrawSizeBox( _hwnd, hdc, rcSizeBox.left, rcSizeBox.top );
|
|
}
|
|
|
|
DrawScrollBar( _hwnd, hdc, pncpo ? &pncwm->rcW0[NCRC_VSCROLL]: NULL, TRUE /*vertical*/ );
|
|
SetRenderedNcPart( RNCF_SCROLLBAR );
|
|
}
|
|
|
|
if( TESTFLAG(pncwm->dwStyle, WS_HSCROLL) &&
|
|
( HasRenderedNcPart(RNCF_SCROLLBAR) || RectVisible(hdc, &pncwm->rcW0[NCRC_HSCROLL])) )
|
|
{
|
|
DrawScrollBar( _hwnd, hdc, pncpo ? &pncwm->rcW0[NCRC_HSCROLL] : NULL, FALSE /*vertical*/ );
|
|
SetRenderedNcPart( RNCF_SCROLLBAR );
|
|
}
|
|
}
|
|
|
|
if (pncpo || hdcIn)
|
|
{
|
|
SelectClipRgn( hdc, NULL );
|
|
}
|
|
else
|
|
{
|
|
ReleaseDC( _hwnd, hdc );
|
|
}
|
|
|
|
LeaveNcThemePaint();
|
|
END_DEBUG_NCPAINT();
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_STYLECHANGED themewnd instance handler
|
|
void CThemeWnd::StyleChanged( UINT iGWL, DWORD dwOld, DWORD dwNew )
|
|
{
|
|
DWORD dwStyleOld, dwStyleNew, dwExStyleOld, dwExStyleNew;
|
|
|
|
switch( iGWL )
|
|
{
|
|
case GWL_STYLE:
|
|
dwStyleOld = dwOld;
|
|
dwStyleNew = dwNew;
|
|
dwExStyleOld = dwExStyleNew = GetWindowLong(_hwnd, GWL_EXSTYLE);
|
|
break;
|
|
|
|
case GWL_EXSTYLE:
|
|
dwExStyleOld = dwOld;
|
|
dwExStyleNew = dwNew;
|
|
dwStyleOld = dwStyleNew = GetWindowLong(_hwnd, GWL_STYLE);
|
|
break;
|
|
|
|
default:
|
|
return;
|
|
}
|
|
|
|
DWORD fClassFlagsOld = CThemeWnd::EvaluateStyle( dwStyleOld, dwExStyleOld);
|
|
DWORD fClassFlagsNew = CThemeWnd::EvaluateStyle( dwStyleNew, dwExStyleNew);
|
|
|
|
// Update theme class flags.
|
|
// Always keep the scrollbar class flag if the window had it initially. User
|
|
// flips scroll styles on and off without corresponding style change notification.
|
|
_fClassFlags = fClassFlagsNew | (_fClassFlags & TWCF_SCROLLBARS);
|
|
_fFrameThemed = TESTFLAG( _fClassFlags, TWCF_FRAME|TWCF_TOOLFRAME );
|
|
|
|
// Are we losing the frame?
|
|
if( TESTFLAG( fClassFlagsOld, TWCF_FRAME|TWCF_TOOLFRAME ) &&
|
|
!TESTFLAG( fClassFlagsNew, TWCF_FRAME|TWCF_TOOLFRAME ) )
|
|
{
|
|
ThemeMDIMenuButtons(FALSE, FALSE);
|
|
|
|
if( AssignedFrameRgn() )
|
|
{
|
|
AssignFrameRgn(FALSE /*strip off frame rgn*/, FTF_REDRAW);
|
|
}
|
|
}
|
|
// Are we gaining a frame?
|
|
else if( TESTFLAG( fClassFlagsNew, TWCF_FRAME|TWCF_TOOLFRAME ) &&
|
|
!TESTFLAG( fClassFlagsOld, TWCF_FRAME|TWCF_TOOLFRAME ) )
|
|
{
|
|
SetFrameTheme(0, NULL);
|
|
}
|
|
|
|
// Freshen window metrics
|
|
GetNcWindowMetrics( NULL, NULL, NULL, NCWMF_RECOMPUTE );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// ThemeDefWindowProc message handlers
|
|
//-------------------------------------------------------------------------//
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_THEMECHANGED post-wndproc msg handler
|
|
LRESULT CALLBACK OnOwpPostThemeChanged( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if (IS_THEME_CHANGE_TARGET(ptm->lParam))
|
|
{
|
|
//---- avoid redundant retheming (except for SetWindowTheme() calls)
|
|
if ((HTHEME(*pwnd) == _nctmCurrent.hTheme) && (! (ptm->lParam & WTC_CUSTOMTHEME)))
|
|
{
|
|
Log(LOG_NCATTACH, L"OnOwpPostThemeChanged, just kicking the frame");
|
|
|
|
//---- window got correct theme thru _XXXWindowProc() from sethook ----
|
|
//---- we just need to redraw the frame for all to be right ----
|
|
if (pwnd->IsFrameThemed())
|
|
{
|
|
//---- attach the region to the window now ----
|
|
pwnd->SetFrameTheme(FTF_REDRAW, NULL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Log(LOG_NCATTACH, L"OnOwpPostThemeChanged, calling Full ::ChangeTheme()");
|
|
|
|
//---- its a real, app/system theme change ----
|
|
pwnd->ChangeTheme( ptm );
|
|
}
|
|
}
|
|
|
|
MsgHandled( ptm );
|
|
return 1L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::ChangeTheme( THEME_MSG* ptm )
|
|
{
|
|
if( _hTheme ) // hwnd attached for previous theme
|
|
{
|
|
// do a lightweight detach from current theme
|
|
_DetachInstance( HMD_CHANGETHEME );
|
|
}
|
|
|
|
if( IsAppThemed() ) // new theme is active
|
|
{
|
|
// retrieve window client rect, style bits.
|
|
WINDOWINFO wi;
|
|
wi.cbSize = sizeof(wi);
|
|
GetWindowInfo( ptm->hwnd, &wi );
|
|
ULONG ulTargetFlags = EvaluateStyle( wi.dwStyle, wi.dwExStyle );
|
|
|
|
// If the window is themable
|
|
if( TESTFLAG(ulTargetFlags, TWCF_NCTHEMETARGETMASK) )
|
|
{
|
|
// Open the new theme
|
|
HTHEME hTheme = ::OpenNcThemeData( ptm->hwnd, L"Window" );
|
|
|
|
if( hTheme )
|
|
{
|
|
// do a lightweight attach
|
|
if( _AttachInstance( ptm->hwnd, hTheme, ulTargetFlags, NULL ) )
|
|
{
|
|
// reattach scrollbars
|
|
if( TESTFLAG( ulTargetFlags, TWCF_SCROLLBARS ) )
|
|
{
|
|
AttachScrollBars(ptm->hwnd);
|
|
}
|
|
|
|
if (IsFrameThemed())
|
|
{
|
|
//---- attach the region to the window now ----
|
|
SetFrameTheme(FTF_REDRAW, NULL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CloseThemeData( hTheme );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (! _hTheme) // if an hwnd is no longer attached
|
|
{
|
|
// Left without a theme handle: This means either we failed to open a new theme handle or
|
|
// failed to evaulate as a target, no new theme, etc.
|
|
RemoveWindowProperties(ptm->hwnd, FALSE);
|
|
|
|
//---- release our CThemeWnd obj so it doesn't leak (addref-protected by caller) ----
|
|
Release();
|
|
}
|
|
|
|
}
|
|
//-------------------------------------------------------------------------//
|
|
BOOL IsPropertySheetChild(HWND hDlg)
|
|
{
|
|
while(hDlg)
|
|
{
|
|
ULONG ulFlags = HandleToUlong(GetProp(hDlg, MAKEINTATOM(GetThemeAtom(THEMEATOM_DLGTEXTURING))));
|
|
|
|
if( ETDT_ENABLETAB == (ulFlags & ETDT_ENABLETAB) /* all bits in this mask are required */ )
|
|
{
|
|
return TRUE;
|
|
}
|
|
hDlg = GetAncestor(hDlg, GA_PARENT);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
//---------------------------------------------------------------------------
|
|
void PrintClientNotHandled(HWND hwnd)
|
|
{
|
|
ATOM aIsPrinting = GetThemeAtom(THEMEATOM_PRINTING);
|
|
DWORD dw = PtrToUlong(GetProp(hwnd, (PCTSTR)aIsPrinting));
|
|
if (dw == PRINTING_ASKING)
|
|
SetProp(hwnd, (PCTSTR)aIsPrinting, (HANDLE)PRINTING_WINDOWDIDNOTHANDLE);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
HBRUSH GetDialogColor(HWND hwnd, NCTHEMEMET &nctm)
|
|
{
|
|
HBRUSH hbr = NULL;
|
|
|
|
// if this is a PROPSHEET child or the app called
|
|
// EnableThemeDialogTexture() on this hwnd, we'll use the tab background.
|
|
if (IsPropertySheetChild(hwnd))
|
|
{
|
|
hbr = nctm.hbrTabDialog;
|
|
}
|
|
|
|
if( NULL == hbr )
|
|
{
|
|
hbr = GetSysColorBrush(COLOR_3DFACE);
|
|
}
|
|
|
|
return hbr;
|
|
}
|
|
//---------------------------------------------------------------------------
|
|
LRESULT CALLBACK OnDdpPrint(CThemeWnd* pwnd, THEME_MSG* ptm)
|
|
{
|
|
LRESULT lRet = 0L;
|
|
if (!ptm->lRet)
|
|
{
|
|
if (pwnd->HasProcessedEraseBk())
|
|
{
|
|
RECT rc;
|
|
HDC hdc = (HDC)ptm->wParam;
|
|
NCTHEMEMET nctm;
|
|
if( GetCurrentNcThemeMetrics( &nctm ))
|
|
{
|
|
HBRUSH hbr = GetDialogColor(*pwnd, nctm);
|
|
|
|
if (hbr)
|
|
{
|
|
POINT pt;
|
|
|
|
if (GetClipBox(hdc, &rc) == NULLREGION)
|
|
GetClientRect(*pwnd, &rc);
|
|
|
|
SetBrushOrgEx(hdc, -rc.left, -rc.top, &pt);
|
|
FillRect(hdc, &rc, hbr);
|
|
SetBrushOrgEx(hdc, pt.x, pt.y, NULL);
|
|
|
|
lRet = (LRESULT)1;
|
|
MsgHandled( ptm );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
PrintClientNotHandled(*pwnd);
|
|
}
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
LRESULT CALLBACK OnDdpCtlColor(CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = 0L;
|
|
if (!ptm->lRet && pwnd->HasProcessedEraseBk())
|
|
{
|
|
NCTHEMEMET nctm;
|
|
if( GetCurrentNcThemeMetrics( &nctm ))
|
|
{
|
|
HBRUSH hbr = GetDialogColor(*pwnd, nctm);
|
|
if (hbr)
|
|
{
|
|
RECT rc;
|
|
HDC hdc = (HDC)ptm->wParam;
|
|
|
|
GetWindowRect(((HWND)ptm->lParam), &rc);
|
|
MapWindowPoints(NULL, *pwnd, (POINT*)&rc, 2);
|
|
SetBkMode(hdc, TRANSPARENT);
|
|
SetBrushOrgEx(hdc, -rc.left, -rc.top, NULL);
|
|
|
|
// the hdc's default background color needs to be set
|
|
// for for those controls that insist on using OPAQUE
|
|
SetBkColor(hdc, GetSysColor(COLOR_BTNFACE));
|
|
|
|
lRet = (LRESULT)hbr;
|
|
MsgHandled( ptm );
|
|
}
|
|
}
|
|
}
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_CTLCOLORxxx defwindowproc override handler
|
|
LRESULT CALLBACK OnDdpPostCtlColor( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = 0L;
|
|
if (!ptm->lRet)
|
|
{
|
|
// This is sent to the parent in the case of WM_CTLCOLORMSGBOX, but to the
|
|
// dialog itself in the case of a WM_CTLCOLORDLG. This gets both.
|
|
CThemeWnd* pwndDlg = CThemeWnd::FromHwnd((HWND)ptm->lParam);
|
|
|
|
|
|
// WM_CTLCOLORMSGBOX is sent for Both the dialog AND the static
|
|
// control inside. So we need to sniff: Are we talking to a dialog or a
|
|
// control. the pwnd is associated with the dialog, but not the control
|
|
if (pwndDlg && VALID_THEMEWND(pwndDlg))
|
|
{
|
|
if (IsPropertySheetChild(*pwnd))
|
|
{
|
|
NCTHEMEMET nctm;
|
|
if( GetCurrentNcThemeMetrics( &nctm ))
|
|
{
|
|
HBRUSH hbr = GetDialogColor(*pwndDlg, nctm);
|
|
if (hbr)
|
|
{
|
|
lRet = (LRESULT) hbr;
|
|
pwndDlg->ProcessedEraseBk(TRUE);
|
|
MsgHandled(ptm);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// If we're talking to a control, forward to the control handler
|
|
// because we have to offset the brush
|
|
lRet = OnDdpCtlColor(pwnd, ptm );
|
|
|
|
}
|
|
}
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
LRESULT CALLBACK OnDwpPrintClient( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
PrintClientNotHandled(*pwnd);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
//---- Non-Client ----
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCPAINT pre-wndmproc msg handler
|
|
LRESULT CALLBACK OnOwpPreNcPaint( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
NcPaintWindow_Add(*pwnd);
|
|
|
|
if( !pwnd->InNcPaint() )
|
|
{
|
|
pwnd->ClearRenderedNcPart(RNCF_ALL);
|
|
}
|
|
pwnd->EnterNcPaint();
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCPAINT DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpNcPaint( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = 0L;
|
|
if( !pwnd->IsNcThemed() )
|
|
return lRet;
|
|
|
|
if( IsWindowVisible(*pwnd) )
|
|
{
|
|
pwnd->NcPaint( NULL, NCPF_DEFAULT, 1 == ptm->wParam ? NULL : (HRGN)ptm->wParam, NULL );
|
|
}
|
|
|
|
MsgHandled( ptm );
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCPAINT post-wndmproc msg handler
|
|
LRESULT CALLBACK OnOwpPostNcPaint( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
pwnd->LeaveNcPaint();
|
|
NcPaintWindow_Remove();
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_PRINT DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpPrint( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = DoMsgDefault(ptm);
|
|
if( !pwnd->IsNcThemed() )
|
|
return lRet;
|
|
|
|
if( ptm->lParam & PRF_NONCLIENT )
|
|
{
|
|
int iLayoutSave = GDI_ERROR;
|
|
HDC hdc = (HDC)ptm->wParam;
|
|
|
|
if (TESTFLAG(GetWindowLong(*pwnd, GWL_EXSTYLE), WS_EX_LAYOUTRTL))
|
|
{
|
|
// AnimateWindow sends WM_PRINT with an unmirrored memory hdc
|
|
iLayoutSave = SetLayout(hdc, LAYOUT_RTL);
|
|
}
|
|
|
|
pwnd->NcPaint( (HDC)ptm->wParam, NCPF_DEFAULT, NULL, NULL );
|
|
|
|
if (iLayoutSave != GDI_ERROR)
|
|
{
|
|
SetLayout(hdc, iLayoutSave);
|
|
}
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCUAHDRAWCAPTION DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpNcThemeDrawCaption( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = 0L;
|
|
if( !pwnd->IsNcThemed() || !pwnd->HasRenderedNcPart(RNCF_CAPTION) )
|
|
return lRet;
|
|
|
|
NCWNDMET* pncwm;
|
|
if( pwnd->GetNcWindowMetrics( NULL, &pncwm, NULL, NCWMF_RECOMPUTE ) )
|
|
{
|
|
HDC hdc = _GetNonclientDC( *pwnd, NULL );
|
|
if( hdc )
|
|
{
|
|
DTBGOPTS dtbo;
|
|
dtbo.dwSize = sizeof(dtbo);
|
|
dtbo.dwFlags = DTBG_DRAWSOLID;
|
|
|
|
pwnd->NcPaintCaption( hdc, pncwm, TRUE, (DWORD)ptm->wParam, &dtbo );
|
|
ReleaseDC( *pwnd, hdc );
|
|
MsgHandled( ptm );
|
|
}
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCUAHDRAWFRAME DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpNcThemeDrawFrame( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = 0L;
|
|
if( !pwnd->IsNcThemed() || !pwnd->HasRenderedNcPart(RNCF_FRAME) )
|
|
return lRet;
|
|
|
|
pwnd->NcPaint( (HDC)ptm->wParam, ptm->lParam & DF_ACTIVE ? NCPF_ACTIVEFRAME : NCPF_INACTIVEFRAME, NULL, NULL );
|
|
|
|
MsgHandled( ptm );
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
CMdiBtns* CThemeWnd::LoadMdiBtns( IN OPTIONAL HDC hdc, IN OPTIONAL UINT uSysCmd )
|
|
{
|
|
if( NULL == _pMdiBtns && NULL == (_pMdiBtns = new CMdiBtns) )
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
return _pMdiBtns->Load( _hTheme, hdc, uSysCmd ) ? _pMdiBtns : NULL;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::UnloadMdiBtns( IN OPTIONAL UINT uSysCmd )
|
|
{
|
|
SAFE_DELETE(_pMdiBtns);
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_MEASUREITEM pre-wndproc msg handler
|
|
LRESULT CALLBACK OnOwpPreMeasureItem( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if( pwnd->IsNcThemed() && IsWindow(pwnd->GetMDIClient()) )
|
|
{
|
|
MEASUREITEMSTRUCT* pmis = (MEASUREITEMSTRUCT*)ptm->lParam;
|
|
|
|
CMdiBtns* pBtns = pwnd->LoadMdiBtns( NULL, pmis->itemID );
|
|
if( pBtns )
|
|
{
|
|
if( pBtns->Measure( *pwnd, pmis ) )
|
|
{
|
|
MsgHandled(ptm);
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_DRAWITEM pre-wndproc msg handler
|
|
LRESULT CALLBACK OnOwpPreDrawItem( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if( pwnd->IsNcThemed() && IsWindow(pwnd->GetMDIClient()) )
|
|
{
|
|
DRAWITEMSTRUCT* pdis = (DRAWITEMSTRUCT*)ptm->lParam;
|
|
|
|
CMdiBtns* pBtns = pwnd->LoadMdiBtns( NULL, pdis->itemID );
|
|
if( pBtns )
|
|
{
|
|
if( pBtns->Draw( *pwnd, pdis ) )
|
|
{
|
|
MsgHandled(ptm);
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_MENUCHAR pre-wndproc msg handler
|
|
LRESULT CALLBACK OnOwpPreMenuChar( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
// Route MENUCHAR messages relating to themed MDI buttons to
|
|
// DefWindowProc (some apps assume all owner-drawn menuitems
|
|
// belong to themselves).
|
|
HWND hwndMDIClient = pwnd->GetMDIClient();
|
|
|
|
if( pwnd->IsNcThemed() && IsWindow(hwndMDIClient))
|
|
{
|
|
if( LOWORD(ptm->wParam) == TEXT('-') )
|
|
{
|
|
BOOL fMaxedChild;
|
|
HWND hwndActive = _MDIGetActive(hwndMDIClient, &fMaxedChild );
|
|
if( hwndActive && fMaxedChild )
|
|
{
|
|
MsgHandled(ptm);
|
|
return DefFrameProc(ptm->hwnd, hwndMDIClient, ptm->uMsg,
|
|
ptm->wParam, ptm->lParam);
|
|
}
|
|
}
|
|
}
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCHITTEST DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpNcHitTest( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if( !pwnd->IsNcThemed() )
|
|
return DoMsgDefault( ptm );
|
|
|
|
NCTHEMEMET nctm;
|
|
NCWNDMET* pncwm;
|
|
POINT pt;
|
|
MAKEPOINT( pt, ptm->lParam );
|
|
MsgHandled( ptm );
|
|
|
|
if( pwnd->GetNcWindowMetrics( NULL, &pncwm, &nctm, 0 ) )
|
|
{
|
|
if( _StrictPtInRect( &pncwm->rcS0[NCRC_CLIENT], pt ) )
|
|
return HTCLIENT;
|
|
|
|
if( _StrictPtInRect( &pncwm->rcS0[NCRC_HSCROLL], pt ) )
|
|
return HTHSCROLL;
|
|
|
|
if( _StrictPtInRect( &pncwm->rcS0[NCRC_SIZEBOX], pt ) )
|
|
{
|
|
if (SizeBoxHwnd(*pwnd) && !TESTFLAG(pncwm->dwExStyle, WS_EX_LEFTSCROLLBAR))
|
|
|
|
{
|
|
return TESTFLAG(pncwm->dwExStyle, WS_EX_LAYOUTRTL) ? HTBOTTOMLEFT : HTBOTTOMRIGHT;
|
|
}
|
|
else
|
|
{
|
|
return HTGROWBOX;
|
|
}
|
|
}
|
|
|
|
if ( TESTFLAG(pncwm->dwExStyle, WS_EX_LAYOUTRTL) )
|
|
{
|
|
// mirror the point to hittest correctly
|
|
MIRROR_POINT(pncwm->rcS0[NCRC_WINDOW], pt);
|
|
}
|
|
|
|
if( _StrictPtInRect( &pncwm->rcS0[NCRC_VSCROLL], pt ) )
|
|
return HTVSCROLL;
|
|
|
|
if( _StrictPtInRect( &pncwm->rcS0[NCRC_MENUBAR], pt ) )
|
|
return HTMENU;
|
|
|
|
if( pncwm->fFrame )
|
|
{
|
|
RECT rcButton;
|
|
|
|
// ---- close button ----
|
|
_GetNcBtnHitTestRect( pncwm, HTCLOSE, FALSE, &rcButton );
|
|
|
|
if ( _StrictPtInRect( &rcButton, pt ) )
|
|
{
|
|
return HTCLOSE;
|
|
}
|
|
|
|
// ---- minimize button ----
|
|
_GetNcBtnHitTestRect( pncwm, HTMINBUTTON, FALSE, &rcButton );
|
|
|
|
if ( _StrictPtInRect( &rcButton, pt ) )
|
|
{
|
|
return HTMINBUTTON;
|
|
}
|
|
|
|
// ---- maximize button ----
|
|
_GetNcBtnHitTestRect( pncwm, HTMAXBUTTON, FALSE, &rcButton );
|
|
|
|
if ( _StrictPtInRect( &rcButton, pt ) )
|
|
{
|
|
return HTMAXBUTTON;
|
|
}
|
|
|
|
// ---- sys menu ----
|
|
_GetNcBtnHitTestRect( pncwm, HTSYSMENU, FALSE, &rcButton );
|
|
|
|
if ( _StrictPtInRect( &rcButton, pt ) )
|
|
{
|
|
return HTSYSMENU;
|
|
}
|
|
|
|
// ---- help button ----
|
|
_GetNcBtnHitTestRect( pncwm, HTHELP, FALSE, &rcButton );
|
|
|
|
if ( _StrictPtInRect( &rcButton, pt ) )
|
|
{
|
|
return HTHELP;
|
|
}
|
|
|
|
#ifdef LAME_BUTTON
|
|
if ( TESTFLAG(pncwm->dwExStyle, WS_EX_LAMEBUTTON) )
|
|
{
|
|
if ( _StrictPtInRect( &pncwm->rcS0[NCRC_LAMEBTN], pt ) )
|
|
return HTLAMEBUTTON;
|
|
}
|
|
#endif // LAME_BUTTON
|
|
|
|
// don't need a mirrored point for the remaining hittests
|
|
MAKEPOINT( pt, ptm->lParam );
|
|
|
|
if( !_StrictPtInRect( &pncwm->rcS0[NCRC_CONTENT], pt ) )
|
|
{
|
|
if( pncwm->fMin || pncwm->fMaxed )
|
|
{
|
|
if( _StrictPtInRect( &pncwm->rcS0[NCRC_CAPTION], pt ) )
|
|
return HTCAPTION;
|
|
}
|
|
|
|
//---- combined caption/frame case ----
|
|
return pwnd->NcBackgroundHitTest( pt, &pncwm->rcS0[NCRC_WINDOW], pncwm->dwStyle, pncwm->dwExStyle,
|
|
pncwm->framestate, pncwm->rgframeparts, pncwm->rgsizehitparts,
|
|
pncwm->rcS0 + NCRC_FRAMEFIRST );
|
|
}
|
|
}
|
|
}
|
|
|
|
return DoMsgDefault( ptm );
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_WINDOWPOSCHANGING pre-wndproc override handler
|
|
LRESULT CALLBACK OnOwpPreWindowPosChanging( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if( pwnd->IsFrameThemed() )
|
|
{
|
|
// Suppress WM_WINDOWPOSCHANGING from being sent to wndproc if it
|
|
// was generated by us calling SetWindowRgn.
|
|
|
|
// Many apps (e.g. Adobe Acrobat Reader, Photoshop dialogs, etc) that handle
|
|
// WM_NCCALCSIZE, WM_WINDOWPOSCHANGING and/or WM_WINDOWPOSCHANGED are not
|
|
// reentrant on their handlers for these messages, and therefore botch the
|
|
// recursion induced by our SetWindowRgn call when we post-process
|
|
// WM_WINDOWPOSCHANGED.
|
|
|
|
// There is no reason that a theme-complient wndproc should ever know that
|
|
// it's window(s) host a region managed by the system.
|
|
if( pwnd->AssigningFrameRgn() )
|
|
{
|
|
MsgHandled(ptm);
|
|
return DefWindowProc(ptm->hwnd, ptm->uMsg, ptm->wParam, ptm->lParam);
|
|
}
|
|
}
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_WINDOWPOSCHANGED pre-wndproc override handler
|
|
LRESULT CALLBACK OnOwpPreWindowPosChanged( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if( pwnd->IsFrameThemed() )
|
|
{
|
|
// Suppress WM_WINDOWPOSCHANGING from being sent to wndproc if it
|
|
// was generated by us calling SetWindowRgn.
|
|
|
|
// Many apps (e.g. Adobe Acrobat Reader, Photoshop dialogs, etc) that handle
|
|
// WM_NCCALCSIZE, WM_WINDOWPOSCHANGING and/or WM_WINDOWPOSCHANGED are not
|
|
// reentrant on their handlers for these messages, and therefore botch the
|
|
// recursion induced by our SetWindowRgn call when we post-process
|
|
// WM_WINDOWPOSCHANGED.
|
|
|
|
// There is no reason that a theme-complient wndproc should ever know that
|
|
// it's window(s) host a region managed by the system.
|
|
|
|
if( pwnd->AssigningFrameRgn() )
|
|
{
|
|
MsgHandled(ptm);
|
|
return DefWindowProc(ptm->hwnd, ptm->uMsg, ptm->wParam, ptm->lParam);
|
|
}
|
|
}
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_WINDOWPOSCHANGED message handler
|
|
inline LRESULT WindowPosChangedWorker( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if( pwnd->IsRevoked(RF_DEFER) )
|
|
{
|
|
if( !pwnd->IsRevoked(RF_INREVOKE) )
|
|
{
|
|
pwnd->Revoke(); // don't touch PWND after this!
|
|
}
|
|
}
|
|
else if( pwnd->IsNcThemed() && !IsWindowInDestroy(*pwnd) )
|
|
{
|
|
// If were not resizing, update the window region.
|
|
if( pwnd->IsFrameThemed() )
|
|
{
|
|
NCWNDMET* pncwm = NULL;
|
|
NCTHEMEMET nctm = {0};
|
|
|
|
// Freshen per-window metrics
|
|
if( !pwnd->AssigningFrameRgn() )
|
|
{
|
|
WINDOWPOS *pWndPos = (WINDOWPOS*) ptm->lParam;
|
|
|
|
// Freshen this window's per-window metrics
|
|
pwnd->GetNcWindowMetrics( NULL, &pncwm, &nctm, NCWMF_RECOMPUTE );
|
|
|
|
// Freshen window metrics for nc-themed children (e.g., MDI child frames)
|
|
EnumChildWindows( *pwnd, _FreshenThemeMetricsCB, NULL );
|
|
|
|
if( !TESTFLAG(pWndPos->flags, SWP_NOSIZE) || pwnd->DirtyFrameRgn() ||
|
|
TESTFLAG(pWndPos->flags, SWP_FRAMECHANGED) )
|
|
{
|
|
if( pWndPos->cx > 0 && pWndPos->cy > 0 )
|
|
{
|
|
pwnd->AssignFrameRgn( TRUE, FTF_REDRAW );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
_MDIUpdate( *pwnd, ((WINDOWPOS*) ptm->lParam)->flags);
|
|
}
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_WINDOWPOSCHANGED post-wndproc override handler
|
|
//
|
|
// Note: we'll handle this post-wndproc for normal, client-side wndprocs
|
|
LRESULT CALLBACK OnOwpPostWindowPosChanged( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if( !IsServerSideWindow(ptm->hwnd) )
|
|
{
|
|
return WindowPosChangedWorker( pwnd, ptm );
|
|
}
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_WINDOWPOSCHANGED DefWindowProc override handler.
|
|
//
|
|
// Note: we'll handle this in DefWindowProc only for windows with win32k-based
|
|
// wndprocs, which are deprived of OWP callbacks.
|
|
LRESULT CALLBACK OnDwpWindowPosChanged( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if( IsServerSideWindow(ptm->hwnd) )
|
|
{
|
|
WindowPosChangedWorker( pwnd, ptm );
|
|
}
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NACTIVATE DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpNcActivate( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = 1L;
|
|
|
|
if( pwnd->IsNcThemed() )
|
|
{
|
|
// We need to forward on. The DWP remembers the state
|
|
// and MFC apps (for one) need this as well
|
|
// but we don't want to actually paint, so lock the window
|
|
ptm->lParam = (LPARAM)-1;
|
|
lRet = DoMsgDefault(ptm);
|
|
|
|
pwnd->NcPaint( NULL, ptm->wParam ? NCPF_ACTIVEFRAME : NCPF_INACTIVEFRAME, NULL, NULL );
|
|
MsgHandled(ptm);
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL CThemeWnd::ShouldTrackFrameButton( UINT uHitcode )
|
|
{
|
|
switch(uHitcode)
|
|
{
|
|
case HTHELP:
|
|
return TESTFLAG(_ncwm.dwExStyle, WS_EX_CONTEXTHELP);
|
|
|
|
case HTMAXBUTTON:
|
|
if( !TESTFLAG(_ncwm.dwStyle, WS_MAXIMIZEBOX) ||
|
|
(CBS_DISABLED == _ncwm.rawMaxBtnState && FS_ACTIVE == _ncwm.framestate) )
|
|
{
|
|
break;
|
|
}
|
|
return TRUE;
|
|
|
|
case HTMINBUTTON:
|
|
if( !TESTFLAG(_ncwm.dwStyle, WS_MINIMIZEBOX) ||
|
|
(CBS_DISABLED == _ncwm.rawMinBtnState && FS_ACTIVE == _ncwm.framestate) )
|
|
{
|
|
break;
|
|
}
|
|
return TRUE;
|
|
|
|
case HTCLOSE:
|
|
if( !_MNCanClose(_hwnd) ||
|
|
(CBS_DISABLED == _ncwm.rawCloseBtnState && FS_ACTIVE == _ncwm.framestate) )
|
|
{
|
|
break;
|
|
}
|
|
return TRUE;
|
|
|
|
case HTSYSMENU:
|
|
return TESTFLAG(_ncwm.dwStyle, WS_SYSMENU);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCLBUTTONDOWN DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpNcLButtonDown( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
WPARAM uSysCmd = 0;
|
|
MsgHandled( ptm );
|
|
|
|
switch( ptm->wParam /* hittest code */ )
|
|
{
|
|
case HTHELP:
|
|
case HTMAXBUTTON:
|
|
case HTMINBUTTON:
|
|
case HTCLOSE:
|
|
case HTSYSMENU:
|
|
if( pwnd->ShouldTrackFrameButton(ptm->wParam) )
|
|
{
|
|
if( pwnd->HasRenderedNcPart(RNCF_CAPTION) )
|
|
{
|
|
POINT pt;
|
|
MAKEPOINT( pt, ptm->lParam );
|
|
if( !pwnd->TrackFrameButton( *pwnd, (int)ptm->wParam, &uSysCmd ) )
|
|
{
|
|
return DoMsgDefault( ptm );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return DoMsgDefault( ptm );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case HTHSCROLL:
|
|
case HTVSCROLL:
|
|
if( pwnd->HasRenderedNcPart(RNCF_SCROLLBAR) )
|
|
{
|
|
uSysCmd = ptm->wParam | ((ptm->wParam == HTVSCROLL) ? SC_HSCROLL:SC_VSCROLL);
|
|
|
|
break;
|
|
}
|
|
|
|
// fall thru
|
|
|
|
default:
|
|
return DoMsgDefault( ptm );
|
|
}
|
|
|
|
// TODO USER will ignore system command if it is disabled on system menu here,
|
|
// don't know why. Imitating the code caused standard min/max/close buttons to
|
|
// render so be careful.
|
|
|
|
if( uSysCmd != 0 )
|
|
{
|
|
SendMessage( *pwnd, WM_SYSCOMMAND, uSysCmd, ptm->lParam );
|
|
}
|
|
|
|
return 0L;
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCMOUSEMOVE DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpNcMouseMove(CThemeWnd* pwnd, THEME_MSG *ptm)
|
|
{
|
|
LRESULT lRet = DoMsgDefault(ptm);
|
|
|
|
int htHotLast = pwnd->GetNcHotItem();
|
|
int htHot;
|
|
|
|
//
|
|
// If the mouse has just entered the NC area, request
|
|
// that we be notified when it leaves.
|
|
//
|
|
if (htHotLast == HTERROR)
|
|
{
|
|
TRACKMOUSEEVENT tme;
|
|
|
|
tme.cbSize = sizeof(tme);
|
|
tme.dwFlags = TME_LEAVE | TME_NONCLIENT;
|
|
tme.hwndTrack = *pwnd;
|
|
tme.dwHoverTime = 0;
|
|
|
|
TrackMouseEvent(&tme);
|
|
}
|
|
|
|
//
|
|
// Filter out the NC elements we don't care about hottracking. And only
|
|
// track the element if we've previously rendered it. Some apps handle
|
|
// painting non-client elements by handling ncpaint. They may not expect
|
|
// that we now hottrack.
|
|
//
|
|
if ( (IsHTFrameButton(ptm->wParam) && pwnd->HasRenderedNcPart(RNCF_CAPTION) &&
|
|
pwnd->ShouldTrackFrameButton(ptm->wParam)) ||
|
|
|
|
(IsHTScrollBar(ptm->wParam) && pwnd->HasRenderedNcPart(RNCF_SCROLLBAR)) )
|
|
{
|
|
htHot = (int)ptm->wParam;
|
|
}
|
|
else
|
|
{
|
|
htHot = HTNOWHERE;
|
|
}
|
|
|
|
//
|
|
// anything to do?
|
|
//
|
|
if ((htHot != htHotLast) || IsHTScrollBar(htHot) || IsHTScrollBar(htHotLast))
|
|
{
|
|
POINT pt;
|
|
|
|
MAKEPOINT( pt, ptm->lParam );
|
|
|
|
//
|
|
// save the hittest code of the NC element the mouse is
|
|
// currently over
|
|
//
|
|
pwnd->SetNcHotItem(htHot);
|
|
|
|
//
|
|
// Determine what should be repainted because the mouse
|
|
// is no longer over it
|
|
//
|
|
if ( IsHTFrameButton(htHotLast) && pwnd->HasRenderedNcPart(RNCF_CAPTION) )
|
|
{
|
|
pwnd->TrackFrameButton(*pwnd, htHotLast, NULL, TRUE);
|
|
}
|
|
else if ( IsHTScrollBar(htHotLast) && pwnd->HasRenderedNcPart(RNCF_SCROLLBAR) )
|
|
{
|
|
ScrollBar_MouseMove(*pwnd, (htHot == htHotLast) ? &pt : NULL, (htHotLast == HTVSCROLL) ? TRUE : FALSE);
|
|
}
|
|
|
|
//
|
|
// Determine what should be repainted because the mouse
|
|
// is now over it
|
|
//
|
|
if ( IsHTFrameButton(htHot) && pwnd->HasRenderedNcPart(RNCF_CAPTION) )
|
|
{
|
|
pwnd->TrackFrameButton(*pwnd, htHot, NULL, TRUE);
|
|
}
|
|
else if ( IsHTScrollBar(htHot) && pwnd->HasRenderedNcPart(RNCF_SCROLLBAR) )
|
|
{
|
|
ScrollBar_MouseMove(*pwnd, &pt, (htHot == HTVSCROLL) ? TRUE : FALSE);
|
|
}
|
|
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_NCMOUSELEAVE DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpNcMouseLeave(CThemeWnd* pwnd, THEME_MSG *ptm)
|
|
{
|
|
LRESULT lRet = DoMsgDefault(ptm);
|
|
|
|
int htHot = pwnd->GetNcHotItem();
|
|
|
|
//
|
|
// the mouse has left NC area, nothing should be drawn in the
|
|
// hot state anymore
|
|
//
|
|
pwnd->SetNcHotItem(HTERROR);
|
|
|
|
if ( IsHTFrameButton(htHot) && pwnd->ShouldTrackFrameButton(htHot) &&
|
|
pwnd->HasRenderedNcPart(RNCF_CAPTION) )
|
|
{
|
|
pwnd->TrackFrameButton(*pwnd, htHot, NULL, TRUE);
|
|
}
|
|
else if ( IsHTScrollBar(htHot) && pwnd->HasRenderedNcPart(RNCF_SCROLLBAR) )
|
|
{
|
|
ScrollBar_MouseMove(*pwnd, NULL, (htHot == HTVSCROLL) ? TRUE : FALSE);
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_CONTEXTMENU DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpContextMenu(CThemeWnd* pwnd, THEME_MSG *ptm)
|
|
{
|
|
NCWNDMET* pncwm;
|
|
POINT pt;
|
|
MAKEPOINT( pt, ptm->lParam );
|
|
MsgHandled( ptm );
|
|
|
|
if( pwnd->GetNcWindowMetrics( NULL, &pncwm, NULL, 0 ) )
|
|
{
|
|
if ( TESTFLAG(pncwm->dwExStyle, WS_EX_LAYOUTRTL) )
|
|
{
|
|
// mirror the point to hittest correctly
|
|
MIRROR_POINT(pncwm->rcS0[NCRC_WINDOW], pt);
|
|
}
|
|
|
|
if( _StrictPtInRect( &pncwm->rcS0[NCRC_HSCROLL], pt ) )
|
|
{
|
|
ScrollBar_Menu(*pwnd, *pwnd, ptm->lParam, FALSE);
|
|
return 0;
|
|
}
|
|
|
|
if( _StrictPtInRect( &pncwm->rcS0[NCRC_VSCROLL], pt ) )
|
|
{
|
|
ScrollBar_Menu(*pwnd, *pwnd, ptm->lParam, TRUE);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return DoMsgDefault( ptm );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_SYSCOMMAND DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpSysCommand( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = 0L;
|
|
|
|
switch( ptm->wParam & ~0x0F )
|
|
{
|
|
// Handle scroll commands
|
|
case SC_VSCROLL:
|
|
case SC_HSCROLL:
|
|
HandleScrollCmd( *pwnd, ptm->wParam, ptm->lParam );
|
|
MsgHandled( ptm );
|
|
return lRet;
|
|
}
|
|
return DoMsgDefault( ptm );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// MDI menubar button theme/untheme wrapper
|
|
void CThemeWnd::ThemeMDIMenuButtons( BOOL fTheme, BOOL fRedraw )
|
|
{
|
|
// Verify we're an MDI frame with a maximized mdi child
|
|
if( _hwndMDIClient && !IsWindowInDestroy(_hwndMDIClient) )
|
|
{
|
|
BOOL fMaxed = FALSE;
|
|
HWND hwndActive = _MDIGetActive( _hwndMDIClient, &fMaxed );
|
|
|
|
if( hwndActive && fMaxed )
|
|
{
|
|
ModifyMDIMenubar(fTheme, fRedraw );
|
|
}
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// MDI menubar button theme/untheme worker
|
|
void CThemeWnd::ModifyMDIMenubar( BOOL fTheme, BOOL fRedraw )
|
|
{
|
|
_fThemedMDIBtns = FALSE;
|
|
|
|
if( IsFrameThemed() || !fTheme )
|
|
{
|
|
MENUBARINFO mbi;
|
|
mbi.cbSize = sizeof(mbi);
|
|
|
|
if( GetMenuBarInfo( _hwnd, OBJID_MENU, 0, &mbi ) )
|
|
{
|
|
_NcTraceMsg( NCTF_MDIBUTTONS, TEXT("ModifyMDIMenubar: GetMenuBarInfo() returns hMenu: %08lX, hwndMenu: %08lX"), mbi.hMenu, mbi.hwndMenu );
|
|
|
|
int cItems = GetMenuItemCount( mbi.hMenu );
|
|
int cThemedItems = 0;
|
|
int cRedraw = 0;
|
|
|
|
_NcTraceMsg( NCTF_MDIBUTTONS, TEXT("ModifyMDIMenubar: on entry, GetMenuItemCount(hMenu = %08lX) returns %d"), mbi.hMenu, cItems );
|
|
|
|
if( cItems > 0 )
|
|
{
|
|
for( int i = cItems - 1; i >= 0 && cThemedItems < MDIBTNCOUNT; i-- )
|
|
{
|
|
MENUITEMINFO mii;
|
|
mii.cbSize = sizeof(mii);
|
|
mii.fMask = MIIM_ID|MIIM_STATE|MIIM_FTYPE|MIIM_BITMAP;
|
|
|
|
if( GetMenuItemInfo( mbi.hMenu, i, TRUE, &mii ) )
|
|
{
|
|
_NcTraceMsg( NCTF_MDIBUTTONS, TEXT("GetMenuItemInfo by pos (%d) returns ID %04lX"), i, mii.wID );
|
|
|
|
switch( mii.wID )
|
|
{
|
|
case SC_RESTORE:
|
|
case SC_MINIMIZE:
|
|
case SC_CLOSE:
|
|
{
|
|
BOOL fThemed = TESTFLAG(mii.fType, MFT_OWNERDRAW);
|
|
if( (fThemed && fTheme) || (fThemed == fTheme) )
|
|
{
|
|
cThemedItems = MDIBTNCOUNT; // one item is already done, assume all to be.
|
|
}
|
|
else
|
|
{
|
|
CMdiBtns* pBtns = LoadMdiBtns( NULL, mii.wID );
|
|
if( pBtns )
|
|
{
|
|
if( pBtns->ThemeItem( mbi.hMenu, i, &mii, fTheme ) )
|
|
{
|
|
cThemedItems++;
|
|
cRedraw++;
|
|
_NcTraceMsg( NCTF_MDIBUTTONS, TEXT("ModifyMDIMenubar: on entry, GetMenuItemCount(hMenu = %08lX) returns %d"), mbi.hMenu, GetMenuItemCount(mbi.hMenu) );
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if( cThemedItems )
|
|
{
|
|
_fThemedMDIBtns = fTheme;
|
|
|
|
if( fRedraw && cRedraw )
|
|
{
|
|
DrawMenuBar( _hwnd );
|
|
}
|
|
}
|
|
|
|
_NcTraceMsg( NCTF_MDIBUTTONS, TEXT("ModifyMDIMenubar: Modified %d menu items, exiting"), cThemedItems );
|
|
}
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL CThemeWnd::_PreDefWindowProc(
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam,
|
|
LRESULT *plRet )
|
|
{
|
|
if (uMsg == WM_PRINTCLIENT)
|
|
{
|
|
PrintClientNotHandled(hwnd);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL CThemeWnd::_PostDlgProc(
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam,
|
|
LRESULT *plRet )
|
|
{
|
|
switch( uMsg )
|
|
{
|
|
case WM_PRINTCLIENT:
|
|
{
|
|
PrintClientNotHandled(hwnd);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Handles Defwindowproc post-processing for unthemed windows.
|
|
BOOL CThemeWnd::_PostWndProc(
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam,
|
|
LRESULT *plRet )
|
|
{
|
|
switch( uMsg )
|
|
{
|
|
// Special-case WM_SYSCOMMAND for MDI frame window updates
|
|
case WM_WINDOWPOSCHANGED:
|
|
if( lParam /* don't trust this */)
|
|
{
|
|
_MDIUpdate( hwnd, ((WINDOWPOS*) lParam)->flags);
|
|
}
|
|
break;
|
|
|
|
case WM_MDISETMENU:
|
|
{
|
|
HWND hwndActive = _MDIGetActive(hwnd);
|
|
if( hwndActive )
|
|
_MDIChildUpdateParent( hwndActive, TRUE );
|
|
break;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_CREATE post-wndproc msg handler
|
|
LRESULT CALLBACK OnOwpPostCreate( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
if( -1 != ptm->lRet )
|
|
{
|
|
if( pwnd->TestCF( TWCF_FRAME|TWCF_TOOLFRAME ))
|
|
{
|
|
DWORD dwFTFlags = FTF_CREATE;
|
|
CREATESTRUCT* pcs = (CREATESTRUCT*)ptm->lParam;
|
|
|
|
if( pcs )
|
|
{
|
|
// don't resize dialogs until post-WM_INITDIALOG
|
|
if( pwnd->TestCF(TWCF_DIALOG) )
|
|
{
|
|
dwFTFlags |= FTF_NOMODIFYPLACEMENT;
|
|
}
|
|
|
|
pwnd->SetFrameTheme( dwFTFlags, NULL );
|
|
MsgHandled(ptm);
|
|
}
|
|
}
|
|
}
|
|
return 0L;
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
// WM_INITDIALOG post defdialogproc handler
|
|
LRESULT CALLBACK OnDdpPostInitDialog(CThemeWnd* pwnd, THEME_MSG* ptm)
|
|
{
|
|
LRESULT lRet = ptm->lRet;
|
|
|
|
// Do this sequence for dialogs only
|
|
if( pwnd->TestCF( TWCF_DIALOG ) && pwnd->TestCF( TWCF_FRAME|TWCF_TOOLFRAME ) )
|
|
{
|
|
DWORD dwFTFlags = FTF_CREATE;
|
|
pwnd->SetFrameTheme( dwFTFlags, NULL );
|
|
MsgHandled(ptm);
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_STYLECHANGING/WM_SYTLECHANGED Pre DefWindowProc msg handler
|
|
LRESULT CALLBACK OnOwpPreStyleChange( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
// Allow this message to arrive at detination WndProc?
|
|
if ( pwnd->SuppressingStyleMsgs() )
|
|
{
|
|
MsgHandled(ptm);
|
|
return DefWindowProc(ptm->hwnd, ptm->uMsg, ptm->wParam, ptm->lParam);
|
|
}
|
|
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_SYTLECHANGED DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpStyleChanged( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
pwnd->StyleChanged((UINT)ptm->wParam, ((STYLESTRUCT*)ptm->lParam)->styleOld,
|
|
((STYLESTRUCT*)ptm->lParam)->styleNew );
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_SETTINGCHANGE post-wndproc handler
|
|
LRESULT CALLBACK OnOwpPostSettingChange( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
/*ignore theme setting change process refresh*/
|
|
|
|
if( SPI_SETNONCLIENTMETRICS == ptm->wParam && !pwnd->InThemeSettingChange() )
|
|
{
|
|
// recompute per-theme metrics.
|
|
EnterCriticalSection( &_csThemeMet );
|
|
|
|
// force refresh of NONCLIENTMETRICS cache.
|
|
NcGetNonclientMetrics( NULL, TRUE );
|
|
|
|
LeaveCriticalSection( &_csThemeMet );
|
|
|
|
pwnd->UnloadMdiBtns();
|
|
|
|
// recycle frame icon handle; current one is no longer valid.
|
|
pwnd->AcquireFrameIcon( GetWindowLong(*pwnd, GWL_STYLE),
|
|
GetWindowLong(*pwnd, GWL_EXSTYLE), TRUE );
|
|
|
|
// frame windows should be invalidated.
|
|
if( pwnd->IsFrameThemed() )
|
|
{
|
|
SetWindowPos( *pwnd, NULL, 0,0,0,0, SWP_DRAWFRAME|
|
|
SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOACTIVATE );
|
|
}
|
|
}
|
|
|
|
return 0L;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_SETTEXT DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpSetText( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = 0L;
|
|
if( pwnd->IsFrameThemed() )
|
|
{
|
|
// prevent ourselves from painting as we call on RealDefWindowProc()
|
|
// to cache the new window text.
|
|
pwnd->LockRedraw( TRUE );
|
|
lRet = DoMsgDefault(ptm);
|
|
pwnd->LockRedraw( FALSE );
|
|
}
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// WM_SETICON DefWindowProc msg handler
|
|
LRESULT CALLBACK OnDwpSetIcon( CThemeWnd* pwnd, THEME_MSG *ptm )
|
|
{
|
|
LRESULT lRet = 0L;
|
|
|
|
// invalidate our app icon handle, force re-acquire.
|
|
pwnd->SetFrameIcon(NULL);
|
|
|
|
// call on RealDefWindowProc to cache the icon
|
|
lRet = DoMsgDefault( ptm );
|
|
|
|
// RealDefWindowProc won't call send a WM_NCUAHDRAWCAPTION for large icons
|
|
if( ICON_BIG == ptm->wParam && pwnd->IsFrameThemed() )
|
|
{
|
|
NCWNDMET* pncwm;
|
|
if( pwnd->GetNcWindowMetrics( NULL, &pncwm, NULL, NCWMF_RECOMPUTE ) )
|
|
{
|
|
HDC hdc = _GetNonclientDC( *pwnd, NULL );
|
|
if( hdc )
|
|
{
|
|
DTBGOPTS dtbo;
|
|
dtbo.dwSize = sizeof(dtbo);
|
|
dtbo.dwFlags = DTBG_DRAWSOLID;
|
|
|
|
pwnd->NcPaintCaption( hdc, pncwm, TRUE, (DWORD)DC_ICON, &dtbo );
|
|
ReleaseDC( *pwnd, hdc );
|
|
}
|
|
}
|
|
}
|
|
return lRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
#define NCPREV_CLASS TEXT("NCPreviewFakeWindow")
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL _fPreviewSysMetrics = FALSE;
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void _NcSetPreviewMetrics( BOOL fPreview )
|
|
{
|
|
BOOL fPrev = _fPreviewSysMetrics;
|
|
_fPreviewSysMetrics = fPreview;
|
|
|
|
if( fPreview != fPrev )
|
|
{
|
|
// make sure we reset button metrics if something has changed
|
|
_fClassicNcBtnMetricsReset = TRUE;
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
inline BOOL _NcUsingPreviewMetrics()
|
|
{
|
|
return _fPreviewSysMetrics;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL NcGetNonclientMetrics( OUT OPTIONAL NONCLIENTMETRICS* pncm, BOOL fRefresh )
|
|
{
|
|
BOOL fRet = FALSE;
|
|
CInternalNonclientMetrics *pincm = NULL;
|
|
|
|
EnterCriticalSection( &_csNcSysMet );
|
|
|
|
// Feed off a static instance of NONCLIENTMETRICS to reduce call overhead.
|
|
if( _NcUsingPreviewMetrics() )
|
|
{
|
|
// hand off preview metrics and get out.
|
|
pincm = &_incmPreview;
|
|
}
|
|
else
|
|
{
|
|
if( _incmCurrent.Acquire( fRefresh ) )
|
|
{
|
|
pincm = &_incmCurrent;
|
|
}
|
|
}
|
|
|
|
if( pincm )
|
|
{
|
|
if( pncm )
|
|
{
|
|
*pncm = pincm->GetNcm();
|
|
}
|
|
fRet = TRUE;
|
|
}
|
|
|
|
LeaveCriticalSection( &_csNcSysMet );
|
|
|
|
return fRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HFONT NcGetCaptionFont( BOOL fSmallCaption )
|
|
{
|
|
EnterCriticalSection( &_csNcSysMet );
|
|
|
|
HFONT hf = _NcUsingPreviewMetrics() ? _incmPreview.GetFont( fSmallCaption ) :
|
|
_incmCurrent.GetFont( fSmallCaption );
|
|
|
|
LeaveCriticalSection( &_csNcSysMet );
|
|
return hf;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void NcClearNonclientMetrics()
|
|
{
|
|
_incmCurrent.Clear();
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
int NcGetSystemMetrics(int nIndex)
|
|
{
|
|
if( _NcUsingPreviewMetrics() )
|
|
{
|
|
int iValue;
|
|
const NONCLIENTMETRICS& ncmPreview = _incmPreview.GetNcm();
|
|
|
|
switch (nIndex)
|
|
{
|
|
case SM_CXHSCROLL: // fall through
|
|
case SM_CXVSCROLL: iValue = ncmPreview.iScrollWidth; break;
|
|
case SM_CYHSCROLL: // fall through
|
|
case SM_CYVSCROLL: iValue = ncmPreview.iScrollHeight; break;
|
|
|
|
case SM_CXSIZE: iValue = ncmPreview.iCaptionWidth; break;
|
|
case SM_CYSIZE: iValue = ncmPreview.iCaptionHeight; break;
|
|
case SM_CYCAPTION: iValue = ncmPreview.iCaptionHeight + 1; break;
|
|
case SM_CXSMSIZE: iValue = ncmPreview.iSmCaptionWidth; break;
|
|
case SM_CYSMSIZE: iValue = ncmPreview.iSmCaptionHeight; break;
|
|
case SM_CXMENUSIZE: iValue = ncmPreview.iMenuWidth; break;
|
|
case SM_CYMENUSIZE: iValue = ncmPreview.iMenuHeight; break;
|
|
|
|
default: iValue = ClassicGetSystemMetrics(nIndex); break;
|
|
}
|
|
return iValue;
|
|
}
|
|
else
|
|
{
|
|
return ClassicGetSystemMetrics(nIndex);
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// _InternalGetSystemMetrics() - Themed implementation of GetSystemMetrics().
|
|
//
|
|
int _InternalGetSystemMetrics( int iMetric, BOOL& fHandled )
|
|
{
|
|
int iRet = 0;
|
|
int* plSysMet = NULL;
|
|
NCTHEMEMET nctm;
|
|
|
|
switch( iMetric )
|
|
{
|
|
case SM_CXSIZE:
|
|
plSysMet = &nctm.theme_sysmets.cxBtn; break;
|
|
|
|
case SM_CXSMSIZE:
|
|
plSysMet = &nctm.theme_sysmets.cxSmBtn; break;
|
|
}
|
|
|
|
if( plSysMet &&
|
|
GetCurrentNcThemeMetrics( &nctm ) && nctm.hTheme != NULL &&
|
|
nctm.theme_sysmets.fValid )
|
|
{
|
|
iRet = *plSysMet;
|
|
fHandled = TRUE; /*was missing (doh! - 408190)*/
|
|
}
|
|
|
|
return iRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// _InternalSystemParametersInfo() - Themed implementation of SystemParametersInfo().
|
|
//
|
|
// return value of FALSE interpreted by caller as not handled.
|
|
BOOL _InternalSystemParametersInfo(
|
|
IN UINT uiAction,
|
|
IN UINT uiParam,
|
|
IN OUT PVOID pvParam,
|
|
IN UINT fWinIni,
|
|
IN BOOL fUnicode,
|
|
BOOL& fHandled )
|
|
{
|
|
SYSTEMPARAMETERSINFO pfnDefault =
|
|
fUnicode ? ClassicSystemParametersInfoW : ClassicSystemParametersInfoA;\
|
|
|
|
BOOL fRet = pfnDefault( uiAction, uiParam, pvParam, fWinIni );
|
|
fHandled = TRUE;
|
|
|
|
if( SPI_GETNONCLIENTMETRICS == uiAction && fRet )
|
|
{
|
|
NCTHEMEMET nctm;
|
|
if( GetCurrentNcThemeMetrics( &nctm ) && nctm.hTheme != NULL && nctm.theme_sysmets.fValid )
|
|
{
|
|
NONCLIENTMETRICS* pncm = (NONCLIENTMETRICS*)pvParam;
|
|
pncm->iCaptionWidth = nctm.theme_sysmets.cxBtn;
|
|
}
|
|
}
|
|
return fRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
THEMEAPI DrawNCWindow(CThemeWnd* pThemeWnd, HWND hwndFake, HDC hdc, DWORD dwFlags, LPRECT prc, NONCLIENTMETRICS* pncm, COLORREF* prgb)
|
|
{
|
|
// Build up Overide structure
|
|
NCPAINTOVERIDE ncpo;
|
|
pThemeWnd->GetNcWindowMetrics( prc, &ncpo.pncwm, &ncpo.nctm, NCWMF_RECOMPUTE|NCWMF_PREVIEW );
|
|
|
|
// Force window to look active
|
|
if (dwFlags & NCPREV_ACTIVEWINDOW)
|
|
{
|
|
ncpo.pncwm->framestate = FS_ACTIVE;
|
|
|
|
ncpo.pncwm->rawCloseBtnState =
|
|
ncpo.pncwm->rawMaxBtnState =
|
|
ncpo.pncwm->rawMinBtnState = CBS_NORMAL;
|
|
}
|
|
ncpo.pncwm->rgbCaption = prgb[FS_ACTIVE == ncpo.pncwm->framestate ? COLOR_CAPTIONTEXT : COLOR_INACTIVECAPTIONTEXT];
|
|
ncpo.pncwm->dwStyle &= ~WS_SIZEBOX;
|
|
// Paint the beautiful visual styled window
|
|
pThemeWnd->NcPaint(hdc, NCPF_DEFAULT, NULL, &ncpo);
|
|
|
|
COLORREF rgbBk = prgb[(dwFlags & NCPREV_MESSAGEBOX) ? COLOR_3DFACE : COLOR_WINDOW];
|
|
HBRUSH hbrBack = CreateSolidBrush(rgbBk);
|
|
FillRect(hdc, &ncpo.pncwm->rcW0[NCRC_CLIENT], hbrBack);
|
|
DeleteObject(hbrBack);
|
|
|
|
WCHAR szText[MAX_PATH];
|
|
// Draw client area
|
|
|
|
HFONT hfont = CreateFont(-MulDiv(8, GetDeviceCaps(hdc, LOGPIXELSY), 72), 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, DEFAULT_CHARSET, OUT_CHARACTER_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH, L"MS Shell Dlg");
|
|
if (hfont)
|
|
{
|
|
if (dwFlags & NCPREV_MESSAGEBOX)
|
|
{
|
|
HTHEME htheme = OpenThemeData( hwndFake, L"Button" );
|
|
int offsetX = ((ncpo.pncwm->rcW0[NCRC_CLIENT].right + ncpo.pncwm->rcW0[NCRC_CLIENT].left) / 2) - 40;
|
|
int offsetY = ((ncpo.pncwm->rcW0[NCRC_CLIENT].bottom + ncpo.pncwm->rcW0[NCRC_CLIENT].top) / 2) - 15;
|
|
RECT rcButton = { offsetX, offsetY, offsetX + 80, offsetY + 30 };
|
|
NcDrawThemeBackground(htheme, hdc, BP_PUSHBUTTON, PBS_DEFAULTED, &rcButton, 0);
|
|
RECT rcContent;
|
|
GetThemeBackgroundContentRect(htheme, hdc, BP_PUSHBUTTON, PBS_DEFAULTED, &rcButton, &rcContent);
|
|
LoadString(g_hInst, IDS_OKBUTTON, szText, ARRAYSIZE(szText));
|
|
if (szText[0])
|
|
{
|
|
HFONT hfontOld = (HFONT)SelectObject(hdc, hfont);
|
|
DrawThemeText(htheme, hdc, BP_PUSHBUTTON, PBS_DEFAULTED, szText, lstrlen(szText), DT_CENTER | DT_VCENTER | DT_SINGLELINE, 0, &rcContent);
|
|
SelectObject(hdc, hfontOld);
|
|
}
|
|
CloseThemeData(htheme);
|
|
}
|
|
else if (dwFlags & NCPREV_ACTIVEWINDOW)
|
|
{
|
|
HTHEME htheme = OpenThemeData( hwndFake, L"Button" );
|
|
RECT rcButton = ncpo.pncwm->rcW0[NCRC_CLIENT];
|
|
LoadString(g_hInst, IDS_WINDOWTEXT, szText, ARRAYSIZE(szText));
|
|
if (szText[0])
|
|
{
|
|
HFONT hfontOld = (HFONT)SelectObject(hdc, hfont);
|
|
DTTOPTS DttOpts = {sizeof(DttOpts)};
|
|
DttOpts.dwFlags = DTT_TEXTCOLOR;
|
|
DttOpts.crText = prgb[COLOR_WINDOWTEXT];
|
|
|
|
DrawThemeTextEx(htheme, hdc, BP_PUSHBUTTON, PBS_DEFAULTED, szText, lstrlen(szText), DT_SINGLELINE, &rcButton, &DttOpts);
|
|
SelectObject(hdc, hfontOld);
|
|
}
|
|
CloseThemeData(htheme);
|
|
}
|
|
}
|
|
DeleteObject(hfont);
|
|
|
|
|
|
ClearNcThemeMetrics(&ncpo.nctm);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
THEMEAPI DrawNCPreview(HDC hdc, DWORD dwFlags, LPRECT prc, LPCWSTR pszVSPath, LPCWSTR pszVSColor, LPCWSTR pszVSSize, NONCLIENTMETRICS* pncm, COLORREF* prgb)
|
|
{
|
|
WNDCLASS wc;
|
|
|
|
// Create a fake Window and attach NC themeing to it
|
|
if (!GetClassInfo(g_hInst, NCPREV_CLASS, &wc)) {
|
|
wc.style = 0;
|
|
wc.lpfnWndProc = DefWindowProc;
|
|
wc.cbClsExtra = 0;
|
|
wc.cbWndExtra = 0;
|
|
wc.hInstance = g_hInst;
|
|
wc.hIcon = NULL;
|
|
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
|
|
wc.hbrBackground = (HBRUSH)(COLOR_3DFACE+1);
|
|
wc.lpszMenuName = NULL;
|
|
wc.lpszClassName = NCPREV_CLASS;
|
|
|
|
if (!RegisterClass(&wc))
|
|
return FALSE;
|
|
}
|
|
|
|
_incmPreview = *pncm;
|
|
_incmPreview._fPreview = TRUE;
|
|
_NcSetPreviewMetrics( TRUE );
|
|
|
|
DWORD dwExStyle = WS_EX_DLGMODALFRAME | ((dwFlags & NCPREV_RTL) ? WS_EX_RTLREADING : 0);
|
|
HWND hwndFake = CreateWindowEx(dwExStyle, NCPREV_CLASS, L"", 0, 0, 0, RECTWIDTH(prc), RECTHEIGHT(prc), NULL, NULL, g_hInst, NULL);
|
|
|
|
if (hwndFake)
|
|
{
|
|
HTHEMEFILE htFile = NULL;
|
|
|
|
WCHAR szCurVSPath[MAX_PATH];
|
|
WCHAR szCurVSColor[MAX_PATH];
|
|
WCHAR szCurVSSize[MAX_PATH];
|
|
|
|
GetCurrentThemeName(szCurVSPath, ARRAYSIZE(szCurVSPath), szCurVSColor, ARRAYSIZE(szCurVSColor), szCurVSSize, ARRAYSIZE(szCurVSSize));
|
|
|
|
if ((lstrcmp(szCurVSPath, pszVSPath) != 0) ||
|
|
(lstrcmp(szCurVSColor, pszVSColor) != 0) ||
|
|
(lstrcmp(szCurVSSize, pszVSSize) != 0))
|
|
{
|
|
HRESULT hr = OpenThemeFile(pszVSPath, pszVSColor, pszVSSize, &htFile, FALSE);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
//---- first, detach from the normal theme ----
|
|
CThemeWnd::Detach(hwndFake, FALSE);
|
|
|
|
//---- apply the preview theme ----
|
|
hr = ApplyTheme(htFile, 0, hwndFake);
|
|
}
|
|
}
|
|
|
|
//---- attach to the preview theme ----
|
|
CThemeWnd* pThemeWnd = CThemeWnd::Attach(hwndFake);
|
|
|
|
if (VALID_THEMEWND(pThemeWnd))
|
|
{
|
|
struct {
|
|
DWORD dwNcPrev;
|
|
UINT uIDStr;
|
|
DWORD dwFlags;
|
|
RECT rc;
|
|
} fakeWindow[]= { {NCPREV_INACTIVEWINDOW, IDS_INACTIVEWINDOW, 0, { prc->left, prc->top, prc->right - 17, prc->bottom - 20 }},
|
|
{NCPREV_ACTIVEWINDOW, IDS_ACTIVEWINDOW, NCPREV_ACTIVEWINDOW, { prc->left + 10, prc->top + 22, prc->right, prc->bottom }},
|
|
{NCPREV_MESSAGEBOX, IDS_MESSAGEBOX, NCPREV_ACTIVEWINDOW | NCPREV_MESSAGEBOX, { prc->left + (RECTWIDTH(prc)/2) - 75, prc->top + (RECTHEIGHT(prc)/2) - 50 + 22,
|
|
prc->left + (RECTWIDTH(prc)/2) + 75, prc->top + (RECTHEIGHT(prc)/2) + 50 + 22}}};
|
|
|
|
WCHAR szWindowName[MAX_PATH];
|
|
for (int i = 0; i < ARRAYSIZE(fakeWindow); i++)
|
|
{
|
|
if (dwFlags & fakeWindow[i].dwNcPrev)
|
|
{
|
|
LoadString(g_hInst, fakeWindow[i].uIDStr, szWindowName, ARRAYSIZE(szWindowName));
|
|
SetWindowText(hwndFake, szWindowName);
|
|
|
|
if (fakeWindow[i].dwNcPrev & NCPREV_MESSAGEBOX)
|
|
{
|
|
SetWindowLongPtr(hwndFake, GWL_STYLE, WS_TILED | WS_CAPTION | WS_SYSMENU);
|
|
}
|
|
else
|
|
{
|
|
SetWindowLongPtr(hwndFake, GWL_STYLE, WS_TILEDWINDOW | WS_VSCROLL);
|
|
}
|
|
|
|
DrawNCWindow(pThemeWnd, hwndFake, hdc, fakeWindow[i].dwFlags, &fakeWindow[i].rc, pncm, prgb);
|
|
}
|
|
}
|
|
|
|
// Clean Up
|
|
CThemeWnd::Detach(hwndFake, 0);
|
|
}
|
|
|
|
if (htFile)
|
|
{
|
|
CloseThemeFile(htFile);
|
|
|
|
//---- clear the preview hold on the theme file ----
|
|
ApplyTheme(NULL, 0, hwndFake);
|
|
}
|
|
|
|
DestroyWindow(hwndFake);
|
|
}
|
|
|
|
_NcSetPreviewMetrics( FALSE );
|
|
_incmPreview.Clear();
|
|
return S_OK;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// CMdiBtns impl
|
|
//-------------------------------------------------------------------------//
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// ctor
|
|
CMdiBtns::CMdiBtns()
|
|
{
|
|
ZeroMemory( _rgBtns, sizeof(_rgBtns) );
|
|
_rgBtns[0].wID = SC_CLOSE;
|
|
_rgBtns[1].wID = SC_RESTORE;
|
|
_rgBtns[2].wID = SC_MINIMIZE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Helper: button lookup based on syscmd ID.
|
|
CMdiBtns::MDIBTN* CMdiBtns::_FindBtn( UINT wID )
|
|
{
|
|
for( int i = 0; i < ARRAYSIZE(_rgBtns); i++ )
|
|
{
|
|
if( wID == _rgBtns[i].wID )
|
|
{
|
|
return (_rgBtns + i);
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Acquires MDI button resources,.computes metrics
|
|
BOOL CMdiBtns::Load( HTHEME hTheme, IN OPTIONAL HDC hdc, UINT uSysCmd )
|
|
{
|
|
// if caller wants all buttons loaded, call recursively.
|
|
if( 0 == uSysCmd )
|
|
{
|
|
return Load( hTheme, hdc, SC_CLOSE ) &&
|
|
Load( hTheme, hdc, SC_RESTORE ) &&
|
|
Load( hTheme, hdc, SC_MINIMIZE );
|
|
}
|
|
|
|
MDIBTN* pBtn = _FindBtn( uSysCmd );
|
|
|
|
if( pBtn && !VALID_WINDOWPART(pBtn->iPartId) /*only if necessary*/ )
|
|
{
|
|
// select appropriate window part
|
|
WINDOWPARTS iPartId = BOGUS_WINDOWPART;
|
|
switch( uSysCmd )
|
|
{
|
|
case SC_CLOSE: iPartId = WP_MDICLOSEBUTTON; break;
|
|
case SC_RESTORE: iPartId = WP_MDIRESTOREBUTTON; break;
|
|
case SC_MINIMIZE: iPartId = WP_MDIMINBUTTON; break;
|
|
}
|
|
|
|
if( VALID_WINDOWPART(iPartId) )
|
|
{
|
|
if( IsThemePartDefined( hTheme, iPartId, 0) )
|
|
{
|
|
// Retrieve sizing type, defaulting to 'stretch'.
|
|
if( FAILED( GetThemeInt( hTheme, iPartId, 0, TMT_SIZINGTYPE, (int*)&pBtn->sizingType ) ) )
|
|
{
|
|
pBtn->sizingType = ST_STRETCH;
|
|
}
|
|
|
|
// if 'truesize', retrieve the size.
|
|
if( ST_TRUESIZE == pBtn->sizingType )
|
|
{
|
|
// If no DC provided, base size on screen DC of default monitor.
|
|
HDC hdcSize = hdc;
|
|
if( NULL == hdcSize )
|
|
{
|
|
hdcSize = GetDC(NULL);
|
|
}
|
|
|
|
if( FAILED( GetThemePartSize( hTheme, hdc, iPartId, 0, NULL, TS_TRUE, &pBtn->size ) ) )
|
|
{
|
|
pBtn->sizingType = ST_STRETCH;
|
|
}
|
|
|
|
if( hdcSize != hdc )
|
|
{
|
|
ReleaseDC(NULL, hdcSize);
|
|
}
|
|
}
|
|
|
|
// not 'truesize'; use system metrics for MDI buttons
|
|
if( pBtn->sizingType != ST_TRUESIZE )
|
|
{
|
|
pBtn->size.cx = NcGetSystemMetrics( SM_CXMENUSIZE );
|
|
pBtn->size.cy = NcGetSystemMetrics( SM_CYMENUSIZE );
|
|
}
|
|
|
|
// assign button attributes
|
|
pBtn->iPartId = iPartId;
|
|
}
|
|
}
|
|
}
|
|
return pBtn != NULL && VALID_WINDOWPART(pBtn->iPartId);
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Releases MDI button resources,.resets metrics
|
|
void CMdiBtns::Unload( IN OPTIONAL UINT uSysCmd )
|
|
{
|
|
// if caller wants all buttons unloaded, call recursively.
|
|
if( 0 == uSysCmd )
|
|
{
|
|
Unload( SC_CLOSE );
|
|
Unload( SC_RESTORE );
|
|
Unload( SC_MINIMIZE );
|
|
return;
|
|
}
|
|
|
|
MDIBTN* pBtn = _FindBtn( uSysCmd );
|
|
|
|
if( pBtn )
|
|
{
|
|
SAFE_DELETE_GDIOBJ(pBtn->hbmTheme);
|
|
ZeroMemory(pBtn, sizeof(*pBtn));
|
|
|
|
// restore our zeroed syscommand value
|
|
pBtn->wID = uSysCmd;
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Theme/untheme MDI frame menubar's Minimize, Restore, Close menu items.
|
|
BOOL CMdiBtns::ThemeItem( HMENU hMenu, int iPos, MENUITEMINFO* pmii, BOOL fTheme )
|
|
{
|
|
// To theme, we simply make the item owner draw. To untheme,
|
|
// we restore it to system-drawn.
|
|
BOOL fRet = FALSE;
|
|
MDIBTN* pBtn = _FindBtn( pmii->wID );
|
|
|
|
if( pBtn && pmii && hMenu )
|
|
{
|
|
if( fTheme )
|
|
{
|
|
// save off previous menuitem type, bitmap
|
|
pBtn->fTypePrev = pmii->fType;
|
|
pBtn->hbmPrev = pmii->hbmpItem;
|
|
|
|
pmii->fType &= ~MFT_BITMAP;
|
|
pmii->fType |= MFT_OWNERDRAW|MFT_RIGHTJUSTIFY;
|
|
pmii->hbmpItem = NULL;
|
|
}
|
|
else
|
|
{
|
|
// restore menuitem type, bitmap
|
|
pmii->fType = pBtn->fTypePrev|MFT_RIGHTJUSTIFY /*409042 - force right-justify on the way out*/;
|
|
pmii->hbmpItem = pBtn->hbmPrev;
|
|
}
|
|
|
|
pmii->fMask = MIIM_FTYPE;
|
|
|
|
fRet = SetMenuItemInfo( hMenu, iPos, TRUE, pmii );
|
|
|
|
if( !fRet || !fTheme )
|
|
{
|
|
pBtn->fTypePrev = 0;
|
|
pBtn->hbmPrev = NULL;
|
|
}
|
|
}
|
|
return fRet;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// Computes button state identifier from Win32 owner draw state
|
|
CLOSEBUTTONSTATES CMdiBtns::_CalcState( ULONG ulOwnerDrawAction, ULONG ulOwnerDrawState )
|
|
{
|
|
CLOSEBUTTONSTATES iStateId = CBS_NORMAL;
|
|
|
|
if( TESTFLAG(ulOwnerDrawState, ODS_DISABLED|ODS_GRAYED|ODS_INACTIVE) )
|
|
{
|
|
iStateId = CBS_DISABLED;
|
|
}
|
|
else if( TESTFLAG(ulOwnerDrawState, ODS_SELECTED) )
|
|
{
|
|
iStateId = CBS_PUSHED;
|
|
}
|
|
else if( TESTFLAG(ulOwnerDrawState, ODS_HOTLIGHT) )
|
|
{
|
|
iStateId = CBS_HOT;
|
|
}
|
|
return iStateId;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// MDI sys button WM_DRAWITEM handler
|
|
BOOL CMdiBtns::Measure( HTHEME hTheme, MEASUREITEMSTRUCT* pmis )
|
|
{
|
|
MDIBTN* pBtn = _FindBtn( pmis->itemID );
|
|
|
|
if( pBtn && VALID_WINDOWPART(pBtn->iPartId) )
|
|
{
|
|
pmis->itemWidth = pBtn->size.cx;
|
|
pmis->itemHeight = pBtn->size.cy;
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// MDI sys button WM_DRAWITEM handler
|
|
BOOL CMdiBtns::Draw( HTHEME hTheme, DRAWITEMSTRUCT* pdis )
|
|
{
|
|
MDIBTN* pBtn = _FindBtn( pdis->itemID );
|
|
|
|
if( pBtn && VALID_WINDOWPART(pBtn->iPartId) )
|
|
{
|
|
return SUCCEEDED(NcDrawThemeBackground(
|
|
hTheme, pdis->hDC, pBtn->iPartId, _CalcState( pdis->itemAction, pdis->itemState ), &pdis->rcItem, 0 ));
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------////
|
|
// "Comments?" link in caption bar, known as the PHellyar (lame) button
|
|
//-------------------------------------------------------------------------//
|
|
#ifdef LAME_BUTTON
|
|
|
|
//-------------------------------------------------------------------------//
|
|
WCHAR g_szLameText[50] = {0};
|
|
|
|
//-------------------------------------------------------------------------//
|
|
#define SZ_LAMETEXT_SUBKEY TEXT("Control Panel\\Desktop")
|
|
#define SZ_LAMETEXT_VALUE TEXT("LameButtonText")
|
|
#define SZ_LAMETEXT_DEFAULT TEXT("Comments?")
|
|
#define CLR_LAMETEXT RGB(91, 171, 245)
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void InitLameText()
|
|
{
|
|
CCurrentUser hkeyCurrentUser(KEY_READ);
|
|
HKEY hkLame;
|
|
HRESULT hr = E_FAIL;
|
|
|
|
|
|
if ( RegOpenKeyEx(hkeyCurrentUser, SZ_LAMETEXT_SUBKEY, 0, KEY_QUERY_VALUE, &hkLame) == ERROR_SUCCESS )
|
|
{
|
|
hr = RegistryStrRead(hkLame, SZ_LAMETEXT_VALUE, g_szLameText, ARRAYSIZE(g_szLameText));
|
|
RegCloseKey(hkLame);
|
|
}
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
lstrcpy(g_szLameText, SZ_LAMETEXT_DEFAULT);
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
VOID CThemeWnd::InitLameResources()
|
|
{
|
|
//
|
|
// Using GetWindowInfo here bc GetWindowLong masks
|
|
// out the WS_EX_LAMEBUTTON bit.
|
|
//
|
|
SAFE_DELETE_GDIOBJ(_hFontLame);
|
|
|
|
WINDOWINFO wi = {0};
|
|
|
|
wi.cbSize = sizeof(wi);
|
|
if ( GetWindowInfo(_hwnd, &wi) && TESTFLAG(wi.dwExStyle, WS_EX_LAMEBUTTON) )
|
|
{
|
|
SIZE sizeLame;
|
|
HFONT hfCaption = NcGetCaptionFont(TESTFLAG(wi.dwExStyle, WS_EX_TOOLWINDOW));
|
|
|
|
if( hfCaption != NULL )
|
|
{
|
|
LOGFONT lfLame;
|
|
if( GetObject( hfCaption, sizeof(lfLame), &lfLame ) )
|
|
{
|
|
lfLame.lfHeight -= (lfLame.lfHeight > 0) ? 2 : -2;
|
|
lfLame.lfUnderline = TRUE;
|
|
lfLame.lfWeight = FW_THIN;
|
|
|
|
HFONT hFontLame = CreateFontIndirect(&lfLame);
|
|
if ( hFontLame != NULL )
|
|
{
|
|
HDC hdc = GetWindowDC(_hwnd);
|
|
|
|
if ( hdc != NULL )
|
|
{
|
|
SelectObject(hdc, hFontLame);
|
|
|
|
if (GetTextExtentPoint32(hdc, g_szLameText, lstrlen(g_szLameText), &sizeLame))
|
|
{
|
|
_hFontLame = hFontLame;
|
|
hFontLame = NULL; // don't free at end of this function
|
|
_sizeLame = sizeLame;
|
|
}
|
|
|
|
ReleaseDC(_hwnd, hdc);
|
|
}
|
|
}
|
|
|
|
if (hFontLame) // didn't assign this font
|
|
DeleteObject(hFontLame);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
VOID CThemeWnd::ClearLameResources()
|
|
{
|
|
SAFE_DELETE_GDIOBJ(_hFontLame);
|
|
ZeroMemory( &_sizeLame, sizeof(_sizeLame) );
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
inline VOID CThemeWnd::DrawLameButton(HDC hdc, IN const NCWNDMET* pncwm)
|
|
{
|
|
if ( TESTFLAG(pncwm->dwExStyle, WS_EX_LAMEBUTTON) && _hFontLame )
|
|
{
|
|
Log(LOG_RFBUG, L"DrawLameButton; _hFontLame=0x%x", _hFontLame);
|
|
|
|
HFONT hFontSave = (HFONT)SelectObject(hdc, _hFontLame);
|
|
COLORREF clrSave = SetTextColor(hdc, CLR_LAMETEXT);
|
|
|
|
DrawText(hdc, g_szLameText, lstrlen(g_szLameText), (LPRECT)&pncwm->rcW0[NCRC_LAMEBTN],
|
|
DT_LEFT | DT_SINGLELINE);
|
|
|
|
SetTextColor(hdc, clrSave);
|
|
SelectObject(hdc, hFontSave);
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
VOID CThemeWnd::GetLameButtonMetrics( NCWNDMET* pncwm, const SIZE* psizeCaption )
|
|
{
|
|
if( TESTFLAG(pncwm->dwExStyle, WS_EX_LAMEBUTTON) && _hFontLame )
|
|
{
|
|
BOOL fLameOn;
|
|
RECT rcCaptionText = pncwm->rcS0[NCRC_CAPTIONTEXT];
|
|
RECT* prcButton = &pncwm->rcS0[NCRC_LAMEBTN];
|
|
int cxPad = NcGetSystemMetrics(SM_CXEDGE) * 2;
|
|
|
|
// Enough room to draw the lame button link?
|
|
fLameOn = RECTWIDTH(&rcCaptionText) >
|
|
psizeCaption->cx +
|
|
cxPad + // between caption, lame text
|
|
_sizeLame.cx +
|
|
cxPad; // between lame text, nearest button;
|
|
|
|
//---- compute lame button alignment ----
|
|
BOOL fReverse = TRUE; // normally, lame goes on right side
|
|
|
|
//---- WS_EX_RIGHT wants the opposite ----
|
|
if (TESTFLAG(_ncwm.dwExStyle, WS_EX_RIGHT))
|
|
fReverse = FALSE;
|
|
|
|
DWORD dwFlags = GetTextAlignFlags(_hTheme, &_ncwm, fReverse);
|
|
|
|
//---- turn off lame button for center captions ----
|
|
if (dwFlags & DT_CENTER)
|
|
fLameOn = FALSE;
|
|
|
|
if ( fLameOn )
|
|
{
|
|
CopyRect(prcButton, &rcCaptionText);
|
|
|
|
//---- note: pMargins already includes the theme specified ----
|
|
//---- CaptionMargins (which scale with DPI) and the ----
|
|
//---- icon and buttons widths ----
|
|
|
|
if(dwFlags & DT_RIGHT) // put lame on right
|
|
{
|
|
prcButton->left = (prcButton->right - _sizeLame.cx) - cxPad ;
|
|
|
|
//---- adjust margins to remove lame area ----
|
|
pncwm->CaptionMargins.cxRightWidth -= _sizeLame.cx;
|
|
}
|
|
else // put lame on left
|
|
{
|
|
prcButton->right = (prcButton->left + _sizeLame.cx) + cxPad;
|
|
|
|
//---- adjust margins to remove lame area ----
|
|
pncwm->CaptionMargins.cxLeftWidth += _sizeLame.cx;
|
|
}
|
|
|
|
// vertically center the text between margins
|
|
prcButton->top += (RECTHEIGHT(&rcCaptionText) - _sizeLame.cy)/2;
|
|
prcButton->bottom = prcButton->top + _sizeLame.cy;
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif // LAME_BUTTON
|
|
|
|
|
|
#ifdef DEBUG
|
|
//-------------------------------------------------------------------------//
|
|
void CDECL _NcTraceMsg( ULONG uFlags, LPCTSTR pszFmt, ...)
|
|
{
|
|
if( TESTFLAG(_NcTraceFlags, uFlags) || NCTF_ALWAYS == uFlags )
|
|
{
|
|
va_list args;
|
|
va_start(args, pszFmt);
|
|
|
|
TCHAR szSpew[2048];
|
|
wvsprintf(szSpew, pszFmt, args);
|
|
OutputDebugString(szSpew);
|
|
OutputDebugString(TEXT("\n"));
|
|
|
|
va_end(args);
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void INIT_THEMEWND_DBG( CThemeWnd* pwnd )
|
|
{
|
|
if( IsWindow( *pwnd ) )
|
|
{
|
|
GetWindowText( *pwnd, pwnd->_szCaption, ARRAYSIZE(pwnd->_szCaption) );
|
|
GetClassName( *pwnd, pwnd->_szWndClass, ARRAYSIZE(pwnd->_szWndClass) );
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::Spew( DWORD dwSpewFlags, LPCTSTR pszFmt, LPCTSTR pszClassList )
|
|
{
|
|
if( pszClassList && *pszClassList )
|
|
{
|
|
if( !_tcsstr( pszClassList, _szWndClass ) )
|
|
return;
|
|
}
|
|
|
|
TCHAR szInfo[MAX_PATH*2];
|
|
TCHAR szMsg[MAX_PATH*2];
|
|
|
|
wsprintf( szInfo, TEXT("%08lX -'%s' ('%s') cf: %08lX"), _hwnd, _szCaption, _szWndClass, _fClassFlags );
|
|
wsprintf( szMsg, pszFmt, szInfo );
|
|
Log(LOG_NCATTACH, szMsg );
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
DWORD dwProcessId;
|
|
DWORD dwThreadId;
|
|
DWORD dwSpewFlags;
|
|
LPCTSTR pszFmt;
|
|
LPCTSTR pszClassList;
|
|
} SPEW_ALL;
|
|
|
|
//-------------------------------------------------------------------------//
|
|
BOOL _SpewAllEnumCB( HWND hwnd, LPARAM lParam )
|
|
{
|
|
SPEW_ALL* psa = (SPEW_ALL*)lParam;
|
|
|
|
if( IsWindowProcess( hwnd, psa->dwProcessId ) )
|
|
{
|
|
CThemeWnd* pwnd = CThemeWnd::FromHwnd( hwnd );
|
|
if( VALID_THEMEWND(pwnd) )
|
|
pwnd->Spew( psa->dwSpewFlags, psa->pszFmt );
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::SpewAll( DWORD dwSpewFlags, LPCTSTR pszFmt, LPCTSTR pszClassList )
|
|
{
|
|
SPEW_ALL sa;
|
|
sa.dwThreadId = GetCurrentThreadId();
|
|
sa.dwProcessId = GetCurrentProcessId();
|
|
sa.dwSpewFlags = dwSpewFlags;
|
|
sa.pszFmt = pszFmt;
|
|
sa.pszClassList = pszClassList;
|
|
|
|
//---- this will enum all windows for this process (all desktops, all child levels) ----
|
|
EnumProcessWindows( _SpewAllEnumCB, (LPARAM)&sa );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void CThemeWnd::SpewLeaks()
|
|
{
|
|
if( _cObj > 0 )
|
|
{
|
|
Log(LOG_NCATTACH, L"LEAK WARNING: %d CThemeWnd instances outstanding.", _cObj );
|
|
}
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void SPEW_RECT( ULONG ulTrace, LPCTSTR pszMsg, LPCRECT prc )
|
|
{
|
|
LPCTSTR pszFmt = TEXT("%s: {L:%d,T:%d,R:%d,B:%d}, (%d x %d)");
|
|
WCHAR szMsg[1024];
|
|
|
|
wsprintfW( szMsg, pszFmt, pszMsg,
|
|
prc->left, prc->top, prc->right, prc->bottom,
|
|
RECTWIDTH(prc), RECTHEIGHT(prc) );
|
|
_NcTraceMsg(ulTrace, szMsg);
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void SPEW_MARGINS( ULONG ulTrace, LPCTSTR pszMsg,
|
|
LPCRECT prcParent, LPCRECT prcChild )
|
|
{
|
|
LPCTSTR pszFmt = TEXT("%s: {L:%d,T:%d,R:%d,B:%d}");
|
|
WCHAR szMsg[1024];
|
|
|
|
wsprintfW( szMsg, pszFmt, pszMsg,
|
|
prcChild->left - prcParent->left,
|
|
prcChild->top - prcParent->top,
|
|
prcParent->right - prcChild->right,
|
|
prcParent->bottom - prcChild->bottom );
|
|
_NcTraceMsg(ulTrace, szMsg);
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void SPEW_RGNRECT( ULONG ulTrace, LPCTSTR pszMsg, HRGN hrgn, int iPartID )
|
|
{
|
|
RECT rc;
|
|
if( NULLREGION == GetRgnBox( hrgn, &rc ) )
|
|
FillMemory( &rc, sizeof(&rc), static_cast<UCHAR>(-1) );
|
|
|
|
_NcTraceMsg( ulTrace, TEXT("Region %08lX for partID[%d]:\n\t"), hrgn, iPartID );
|
|
SPEW_RECT( ulTrace, pszMsg, &rc );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void SPEW_WINDOWINFO( ULONG ulTrace, WINDOWINFO* pwi )
|
|
{
|
|
SPEW_RECT(ulTrace, TEXT("->wi.rcWindow"), &pwi->rcWindow );
|
|
SPEW_RECT(ulTrace, TEXT("->wi.rcClient"), &pwi->rcClient );
|
|
_NcTraceMsg(ulTrace, TEXT("->wi.dwStyle: %08lX"), pwi->dwStyle );
|
|
_NcTraceMsg(ulTrace, TEXT("->wi.dwExStyle: %08lX"), pwi->dwExStyle );
|
|
_NcTraceMsg(ulTrace, TEXT("->wi.dwWindowStatus: %08lX"), pwi->dwWindowStatus );
|
|
_NcTraceMsg(ulTrace, TEXT("->wi.cxWindowBorders: %d"), pwi->cxWindowBorders );
|
|
_NcTraceMsg(ulTrace, TEXT("->wi.cyWindowBorders: %d"), pwi->cyWindowBorders );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void SPEW_NCWNDMET( ULONG ulTrace, LPCTSTR pszMsg, NCWNDMET* pncwm )
|
|
{
|
|
_NcTraceMsg(ulTrace, TEXT("\n%s - Spewing NCWNDMET @ %08lx..."), pszMsg, pncwm );
|
|
|
|
_NcTraceMsg(ulTrace, TEXT("->fValid: %d"), pncwm->fValid );
|
|
_NcTraceMsg(ulTrace, TEXT("->dwStyle: %08lX"), pncwm->dwStyle );
|
|
_NcTraceMsg(ulTrace, TEXT("->dwExStyle: %08lX"), pncwm->dwExStyle );
|
|
_NcTraceMsg(ulTrace, TEXT("->dwWindowStatus: %08lX"), pncwm->dwWindowStatus );
|
|
_NcTraceMsg(ulTrace, TEXT("->fFrame: %d"), pncwm->fFrame );
|
|
_NcTraceMsg(ulTrace, TEXT("->fSmallFrame: %d"), pncwm->fSmallFrame );
|
|
_NcTraceMsg(ulTrace, TEXT("->iFRAMEBOTTOM %d"), pncwm->rgframeparts[iFRAMEBOTTOM] );
|
|
_NcTraceMsg(ulTrace, TEXT("->iFRAMELEFT: %d"), pncwm->rgframeparts[iFRAMELEFT] );
|
|
_NcTraceMsg(ulTrace, TEXT("->iFRAMERIGHT: %d"), pncwm->rgframeparts[iFRAMERIGHT] );
|
|
_NcTraceMsg(ulTrace, TEXT("->framestate: %d"), pncwm->framestate );
|
|
_NcTraceMsg(ulTrace, TEXT("->iMinButtonPart: %d"), pncwm->iMinButtonPart);
|
|
_NcTraceMsg(ulTrace, TEXT("->iMaxButtonPart: %d"), pncwm->iMaxButtonPart);
|
|
_NcTraceMsg(ulTrace, TEXT("->rawCloseBtnState: %d"), pncwm->rawCloseBtnState);
|
|
_NcTraceMsg(ulTrace, TEXT("->rawMinBtnState: %d"), pncwm->rawMinBtnState);
|
|
_NcTraceMsg(ulTrace, TEXT("->rawMaxBtnState: %d"), pncwm->rawMaxBtnState);
|
|
_NcTraceMsg(ulTrace, TEXT("->cyMenu: %d"), pncwm->cyMenu );
|
|
_NcTraceMsg(ulTrace, TEXT("->cnMenuOffsetLeft: %d"), pncwm->cnMenuOffsetLeft );
|
|
_NcTraceMsg(ulTrace, TEXT("->cnMenuOffsetRight: %d"), pncwm->cnMenuOffsetRight );
|
|
_NcTraceMsg(ulTrace, TEXT("->cnMenuOffsetTop: %d"), pncwm->cnMenuOffsetTop );
|
|
_NcTraceMsg(ulTrace, TEXT("->cnBorders: %d"), pncwm->cnBorders );
|
|
_NcTraceMsg(ulTrace, TEXT("->CaptionMargins: (%d,%d,%d,%d)"), pncwm->CaptionMargins );
|
|
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_WINDOW] "), &pncwm->rcS0[NCRC_WINDOW] );
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_CLIENT] "), &pncwm->rcS0[NCRC_CLIENT] );
|
|
SPEW_MARGINS(ulTrace, TEXT("Window-Client margins"), &pncwm->rcS0[NCRC_WINDOW], &pncwm->rcS0[NCRC_CLIENT] );
|
|
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_CONTENT] "), &pncwm->rcS0[NCRC_CONTENT] );
|
|
SPEW_MARGINS(ulTrace, TEXT("Window-Content margins"), &pncwm->rcS0[NCRC_WINDOW], &pncwm->rcS0[NCRC_CONTENT]);
|
|
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_MENUBAR] "), &pncwm->rcS0[NCRC_MENUBAR] );
|
|
SPEW_MARGINS(ulTrace, TEXT("Window-Menubar margins"), &pncwm->rcS0[NCRC_WINDOW], &pncwm->rcS0[NCRC_MENUBAR]);
|
|
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_CAPTION] "), &pncwm->rcS0[NCRC_CAPTION] );
|
|
SPEW_MARGINS(ulTrace, TEXT("Window-Caption margins"), &pncwm->rcS0[NCRC_WINDOW], &pncwm->rcS0[NCRC_CAPTION]);
|
|
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_FRAMELEFT] "), &pncwm->rcS0[NCRC_FRAMELEFT] );
|
|
SPEW_MARGINS(ulTrace, TEXT("Window-FrameLeft margins"), &pncwm->rcS0[NCRC_WINDOW], &pncwm->rcS0[NCRC_FRAMELEFT]);
|
|
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_FRAMERIGHT]"), &pncwm->rcS0[NCRC_FRAMERIGHT] );
|
|
SPEW_MARGINS(ulTrace, TEXT("Window-FrameRight margins"), &pncwm->rcS0[NCRC_WINDOW], &pncwm->rcS0[NCRC_FRAMERIGHT]);
|
|
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_FRAMEBOTTOM]"), &pncwm->rcS0[NCRC_FRAMEBOTTOM] );
|
|
SPEW_MARGINS(ulTrace, TEXT("Window-FrameBottom margins"), &pncwm->rcS0[NCRC_WINDOW], &pncwm->rcS0[NCRC_FRAMEBOTTOM]);
|
|
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_CLIENTEDGE]"), &pncwm->rcS0[NCRC_CLIENTEDGE] );
|
|
SPEW_MARGINS(ulTrace, TEXT("Window-ClientEdge margins"), &pncwm->rcS0[NCRC_WINDOW], &pncwm->rcS0[NCRC_CLIENTEDGE]);
|
|
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_HSCROLL] "), &pncwm->rcS0[NCRC_HSCROLL] );
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_VSCROLL] "), &pncwm->rcS0[NCRC_VSCROLL] );
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_SIZEBOX] "), &pncwm->rcS0[NCRC_SIZEBOX] );
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_CLOSEBTN] "), &pncwm->rcS0[NCRC_CLOSEBTN] );
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_MINBTN] "), &pncwm->rcS0[NCRC_MINBTN] );
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_MAXBTN] "), &pncwm->rcS0[NCRC_MAXBTN] );
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_SYSBTN] "), &pncwm->rcS0[NCRC_SYSBTN] );
|
|
SPEW_RECT(ulTrace, TEXT("->rcS0[NCRC_HELPBTN] "), &pncwm->rcS0[NCRC_HELPBTN] );
|
|
#ifdef LAME_BUTTON
|
|
SPEW_RECT(ulTrace, TEXT("rcLame"), &pncwm->rcS0[NCRC_LAMEBTN] );
|
|
#endif // LAME_BUTTON
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void SPEW_THEMEMSG( ULONG ulTrace, LPCTSTR pszMsg, THEME_MSG* ptm )
|
|
{
|
|
_NcTraceMsg(ulTrace, TEXT("%s hwnd: %08lX, uMsg: %04lX, handled?: %d"),
|
|
pszMsg, (ptm)->hwnd, (ptm)->uMsg, (ptm)->fHandled );
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void SPEW_SCROLLINFO( LPCTSTR pszMsg, HWND hwnd, LPCSCROLLINFO psi )
|
|
{
|
|
#ifdef _ENABLE_SCROLL_SPEW_
|
|
_NcTraceMsg(ulTrace, L"%s for HWND %08lX...\ncbSize: %d\nfMask: %08lX\nnMin: %d\nnMax: %d\nnPage: %d\nnPos: %d",
|
|
pszMsg, hwnd, psi->cbSize, psi->fMask, psi->nMin, psi->nMax, psi->nPage, psi->nPos );
|
|
#endif _ENABLE_SCROLL_SPEW_
|
|
}
|
|
|
|
#if defined(DEBUG_NCPAINT)
|
|
|
|
static int _cPaintSleep = 10;
|
|
|
|
void _DebugBackground(
|
|
HDC hdc,
|
|
COLORREF rgb,
|
|
const RECT *prc )
|
|
{
|
|
// paint some indicator stuff
|
|
COLORREF rgb0 = SetBkColor( hdc, rgb );
|
|
SPEW_RECT( NCTF_ALWAYS, TEXT("\tprc"), prc );
|
|
ExtTextOut( hdc, prc->left, prc->top, ETO_OPAQUE, prc, NULL, 0, NULL );
|
|
Sleep(_cPaintSleep);
|
|
SetBkColor( hdc, rgb0 );
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HRESULT _DebugDrawThemeBackground(
|
|
HTHEME hTheme,
|
|
HDC hdc,
|
|
int iPartId,
|
|
int iStateId,
|
|
const RECT *prc,
|
|
OPTIONAL const RECT* prcClip )
|
|
{
|
|
if( TESTFLAG( _NcTraceFlags, NCTF_NCPAINT ) )
|
|
{
|
|
_NcTraceMsg( NCTF_ALWAYS, TEXT("DrawThemeBackground( hTheme = %08lX, hdc = %08lX, iPartId = %d, iStateId = %d"),
|
|
hTheme, hdc, iPartId, iStateId );
|
|
_DebugBackground( hdc, RGBDEBUGBKGND, prc );
|
|
}
|
|
|
|
// paint the real background.
|
|
HRESULT hr = DrawThemeBackground( hTheme, hdc, iPartId, iStateId, prc, prcClip );
|
|
|
|
if( TESTFLAG( _NcTraceFlags, NCTF_NCPAINT ) )
|
|
{
|
|
Sleep(_cPaintSleep);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------//
|
|
HRESULT _DebugDrawThemeBackgroundEx(
|
|
HTHEME hTheme,
|
|
HDC hdc,
|
|
int iPartId,
|
|
int iStateId,
|
|
const RECT *prc,
|
|
OPTIONAL const DTBGOPTS *pOptions )
|
|
{
|
|
if( TESTFLAG( _NcTraceFlags, NCTF_NCPAINT ) )
|
|
{
|
|
_NcTraceMsg( NCTF_ALWAYS, TEXT("DrawThemeBackground( hTheme = %08lX, hdc = %08lX, iPartId = %d, iStateId = %d"),
|
|
hTheme, hdc, iPartId, iStateId );
|
|
_DebugBackground( hdc, RGBDEBUGBKGND, prc );
|
|
}
|
|
|
|
// paint the real background.
|
|
HRESULT hr = DrawThemeBackgroundEx( hTheme, hdc, iPartId, iStateId, prc, pOptions );
|
|
|
|
if( TESTFLAG( _NcTraceFlags, NCTF_NCPAINT ) )
|
|
{
|
|
Sleep(_cPaintSleep);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
void NcDebugClipRgn( HDC hdc, COLORREF rgbPaint )
|
|
{
|
|
if( TESTFLAG( _NcTraceFlags, NCTF_NCPAINT ) )
|
|
{
|
|
HRGN hrgn = CreateRectRgn(0,0,1,1);
|
|
|
|
if( hrgn )
|
|
{
|
|
if( GetClipRgn( hdc, hrgn ) > 0 )
|
|
{
|
|
HBRUSH hbr = CreateSolidBrush(rgbPaint);
|
|
FillRgn( hdc, hrgn, hbr );
|
|
DeleteObject(hbr);
|
|
Sleep(_cPaintSleep);
|
|
}
|
|
DeleteObject(hrgn);
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif //defined(DEBUG_NCPAINT)
|
|
|
|
|
|
#endif DEBUG
|