| OLD | NEW |
| (Empty) |
| 1 // Windows Template Library - WTL version 8.0 | |
| 2 // Copyright (C) Microsoft Corporation. All rights reserved. | |
| 3 // | |
| 4 // This file is a part of the Windows Template Library. | |
| 5 // The use and distribution terms for this software are covered by the | |
| 6 // Microsoft Permissive License (Ms-PL) which can be found in the file | |
| 7 // Ms-PL.txt at the root of this distribution. | |
| 8 | |
| 9 #ifndef __ATLTHEME_H__ | |
| 10 #define __ATLTHEME_H__ | |
| 11 | |
| 12 #pragma once | |
| 13 | |
| 14 #ifndef __cplusplus | |
| 15 #error ATL requires C++ compilation (use a .cpp suffix) | |
| 16 #endif | |
| 17 | |
| 18 #ifdef _WIN32_WCE | |
| 19 #error atltheme.h is not supported on Windows CE | |
| 20 #endif | |
| 21 | |
| 22 #ifndef __ATLAPP_H__ | |
| 23 #error atltheme.h requires atlapp.h to be included first | |
| 24 #endif | |
| 25 | |
| 26 #ifndef __ATLWIN_H__ | |
| 27 #error atltheme.h requires atlwin.h to be included first | |
| 28 #endif | |
| 29 | |
| 30 #if (_WIN32_WINNT < 0x0501) | |
| 31 #error atltheme.h requires _WIN32_WINNT >= 0x0501 | |
| 32 #endif // (_WIN32_WINNT < 0x0501) | |
| 33 | |
| 34 #if defined(_WTL_USE_VSSYM32) || (defined(NTDDI_VERSION) && (NTDDI_VERSION >= NT
DDI_LONGHORN)) | |
| 35 #include <vssym32.h> | |
| 36 #else | |
| 37 #include <tmschema.h> | |
| 38 #endif | |
| 39 | |
| 40 #include <uxtheme.h> | |
| 41 #pragma comment(lib, "uxtheme.lib") | |
| 42 | |
| 43 // Note: To create an application that also runs on older versions of Windows, | |
| 44 // use delay load of uxtheme.dll and ensure that no calls to the Theme API are | |
| 45 // made if theming is not supported. It is enough to check if m_hTheme is NULL. | |
| 46 // Example: | |
| 47 // if(m_hTheme != NULL) | |
| 48 // { | |
| 49 // DrawThemeBackground(dc, BP_PUSHBUTTON, PBS_NORMAL, &rect, NULL); | |
| 50 // DrawThemeText(dc, BP_PUSHBUTTON, PBS_NORMAL, L"Button", -1, DT_S
INGLELINE | DT_CENTER | DT_VCENTER, 0, &rect); | |
| 51 // } | |
| 52 // else | |
| 53 // { | |
| 54 // dc.DrawFrameControl(&rect, DFC_BUTTON, DFCS_BUTTONPUSH); | |
| 55 // dc.DrawText(_T("Button"), -1, &rect, DT_SINGLELINE | DT_CENTER |
DT_VCENTER); | |
| 56 // } | |
| 57 // | |
| 58 // Delay load is NOT AUTOMATIC for VC++ 7, you have to link to delayimp.lib, | |
| 59 // and add uxtheme.dll in the Linker.Input.Delay Loaded DLLs section of the | |
| 60 // project properties. | |
| 61 #if (_MSC_VER < 1300) && !defined(_WTL_NO_THEME_DELAYLOAD) | |
| 62 #pragma comment(lib, "delayimp.lib") | |
| 63 #pragma comment(linker, "/delayload:uxtheme.dll") | |
| 64 #endif // (_MSC_VER < 1300) && !defined(_WTL_NO_THEME_DELAYLOAD) | |
| 65 | |
| 66 // Hack: Signatures in uxtheme.h changed - the only way to check which variant o
f uxtheme.h | |
| 67 // is included is to check for presence of new defines MAX_THEMECOLOR and MAX_TH
EMESIZE | |
| 68 #ifndef _WTL_NEW_UXTHEME | |
| 69 #if defined(MAX_THEMECOLOR) && defined(MAX_THEMESIZE) | |
| 70 #define _WTL_NEW_UXTHEME | |
| 71 #endif // defined(MAX_THEMECOLOR) && defined(MAX_THEMESIZE) | |
| 72 #endif // _WTL_NEW_UXTHEME | |
| 73 | |
| 74 | |
| 75 /////////////////////////////////////////////////////////////////////////////// | |
| 76 // Classes in this file: | |
| 77 // | |
| 78 // CTheme | |
| 79 // CThemeImpl<T, TBase> | |
| 80 // | |
| 81 // CBufferedPaint | |
| 82 // CBufferedPaintImpl<T> | |
| 83 // CBufferedPaintWindowImpl<T, TBase, TWinTraits> | |
| 84 // CBufferedAnimation | |
| 85 // CBufferedAnimationImpl<T, TState> | |
| 86 // CBufferedAnimationWindowImpl<T, TState, TBase, TWinTraits> | |
| 87 // | |
| 88 // Global functions: | |
| 89 // AtlDrawThemeClientEdge() | |
| 90 | |
| 91 | |
| 92 namespace WTL | |
| 93 { | |
| 94 | |
| 95 /////////////////////////////////////////////////////////////////////////////// | |
| 96 // CTheme - wrapper for theme handle | |
| 97 | |
| 98 class CTheme | |
| 99 { | |
| 100 public: | |
| 101 // Data members | |
| 102 HTHEME m_hTheme; | |
| 103 static int m_nIsThemingSupported; | |
| 104 | |
| 105 // Constructor | |
| 106 CTheme(HTHEME hTheme = NULL) : m_hTheme(hTheme) | |
| 107 { | |
| 108 IsThemingSupported(); | |
| 109 } | |
| 110 | |
| 111 // Operators and helpers | |
| 112 bool IsThemeNull() const | |
| 113 { | |
| 114 return (m_hTheme == NULL); | |
| 115 } | |
| 116 | |
| 117 CTheme& operator =(HTHEME hTheme) | |
| 118 { | |
| 119 m_hTheme = hTheme; | |
| 120 return *this; | |
| 121 } | |
| 122 | |
| 123 operator HTHEME() const | |
| 124 { | |
| 125 return m_hTheme; | |
| 126 } | |
| 127 | |
| 128 void Attach(HTHEME hTheme) | |
| 129 { | |
| 130 m_hTheme = hTheme; | |
| 131 } | |
| 132 | |
| 133 HTHEME Detach() | |
| 134 { | |
| 135 HTHEME hTheme = m_hTheme; | |
| 136 m_hTheme = NULL; | |
| 137 return hTheme; | |
| 138 } | |
| 139 | |
| 140 // Theme support helper | |
| 141 static bool IsThemingSupported() | |
| 142 { | |
| 143 if(m_nIsThemingSupported == -1) | |
| 144 { | |
| 145 CStaticDataInitCriticalSectionLock lock; | |
| 146 if(FAILED(lock.Lock())) | |
| 147 { | |
| 148 ATLTRACE2(atlTraceUI, 0, _T("ERROR : Unable to l
ock critical section in CTheme::IsThemingSupported.\n")); | |
| 149 ATLASSERT(FALSE); | |
| 150 return false; | |
| 151 } | |
| 152 | |
| 153 if(m_nIsThemingSupported == -1) | |
| 154 { | |
| 155 HMODULE hThemeDLL = ::LoadLibrary(_T("uxtheme.dl
l")); | |
| 156 m_nIsThemingSupported = (hThemeDLL != NULL) ? 1
: 0; | |
| 157 if(hThemeDLL != NULL) | |
| 158 ::FreeLibrary(hThemeDLL); | |
| 159 } | |
| 160 | |
| 161 lock.Unlock(); | |
| 162 } | |
| 163 | |
| 164 ATLASSERT(m_nIsThemingSupported != -1); | |
| 165 return (m_nIsThemingSupported == 1); | |
| 166 } | |
| 167 | |
| 168 // Operations and theme properties | |
| 169 HTHEME OpenThemeData(HWND hWnd, LPCWSTR pszClassList) | |
| 170 { | |
| 171 if(!IsThemingSupported()) | |
| 172 return NULL; | |
| 173 | |
| 174 ATLASSERT(m_hTheme == NULL); | |
| 175 m_hTheme = ::OpenThemeData(hWnd, pszClassList); | |
| 176 return m_hTheme; | |
| 177 } | |
| 178 | |
| 179 HRESULT CloseThemeData() | |
| 180 { | |
| 181 HRESULT hRet = S_FALSE; | |
| 182 if(m_hTheme != NULL) | |
| 183 { | |
| 184 hRet = ::CloseThemeData(m_hTheme); | |
| 185 if(SUCCEEDED(hRet)) | |
| 186 m_hTheme = NULL; | |
| 187 } | |
| 188 return hRet; | |
| 189 } | |
| 190 | |
| 191 HRESULT DrawThemeBackground(HDC hDC, int nPartID, int nStateID, LPCRECT
pRect, LPCRECT pClipRect = NULL) | |
| 192 { | |
| 193 ATLASSERT(m_hTheme != NULL); | |
| 194 return ::DrawThemeBackground(m_hTheme, hDC, nPartID, nStateID, p
Rect, pClipRect); | |
| 195 } | |
| 196 | |
| 197 HRESULT DrawThemeBackgroundEx(HDC hDC, int nPartID, int nStateID, LPCREC
T pRect, const DTBGOPTS* pOptions = NULL) | |
| 198 { | |
| 199 ATLASSERT(m_hTheme != NULL); | |
| 200 return ::DrawThemeBackgroundEx(m_hTheme, hDC, nPartID, nStateID,
pRect, pOptions); | |
| 201 } | |
| 202 | |
| 203 HRESULT DrawThemeText(HDC hDC, int nPartID, int nStateID, LPCWSTR pszTex
t, int nCharCount, DWORD dwTextFlags, DWORD dwTextFlags2, LPCRECT pRect) | |
| 204 { | |
| 205 ATLASSERT(m_hTheme != NULL); | |
| 206 return ::DrawThemeText(m_hTheme, hDC, nPartID, nStateID, pszText
, nCharCount, dwTextFlags, dwTextFlags2, pRect); | |
| 207 } | |
| 208 | |
| 209 HRESULT GetThemeBackgroundContentRect(HDC hDC, int nPartID, int nStateID
, LPCRECT pBoundingRect, LPRECT pContentRect) const | |
| 210 { | |
| 211 ATLASSERT(m_hTheme != NULL); | |
| 212 return ::GetThemeBackgroundContentRect(m_hTheme, hDC, nPartID, n
StateID, pBoundingRect, pContentRect); | |
| 213 } | |
| 214 | |
| 215 HRESULT GetThemeBackgroundExtent(HDC hDC, int nPartID, int nStateID, LPC
RECT pContentRect, LPRECT pExtentRect) const | |
| 216 { | |
| 217 ATLASSERT(m_hTheme != NULL); | |
| 218 return ::GetThemeBackgroundExtent(m_hTheme, hDC, nPartID, nState
ID, pContentRect, pExtentRect); | |
| 219 } | |
| 220 | |
| 221 HRESULT GetThemePartSize(HDC hDC, int nPartID, int nStateID, LPRECT pRec
t, enum THEMESIZE eSize, LPSIZE pSize) const | |
| 222 { | |
| 223 ATLASSERT(m_hTheme != NULL); | |
| 224 return ::GetThemePartSize(m_hTheme, hDC, nPartID, nStateID, pRec
t, eSize, pSize); | |
| 225 } | |
| 226 | |
| 227 HRESULT GetThemeTextExtent(HDC hDC, int nPartID, int nStateID, LPCWSTR p
szText, int nCharCount, DWORD dwTextFlags, LPCRECT pBoundingRect, LPRECT pExten
tRect) const | |
| 228 { | |
| 229 ATLASSERT(m_hTheme != NULL); | |
| 230 return ::GetThemeTextExtent(m_hTheme, hDC, nPartID, nStateID, ps
zText, nCharCount, dwTextFlags, pBoundingRect, pExtentRect); | |
| 231 } | |
| 232 | |
| 233 HRESULT GetThemeTextMetrics(HDC hDC, int nPartID, int nStateID, PTEXTMET
RICW pTextMetric) const | |
| 234 { | |
| 235 ATLASSERT(m_hTheme != NULL); | |
| 236 #ifdef _WTL_NEW_UXTHEME | |
| 237 return ::GetThemeTextMetrics(m_hTheme, hDC, nPartID, nStateID, p
TextMetric); | |
| 238 #else // !_WTL_NEW_UXTHEME | |
| 239 // Note: The cast to PTEXTMETRIC is because uxtheme.h incorrectl
y uses it instead of PTEXTMETRICW | |
| 240 return ::GetThemeTextMetrics(m_hTheme, hDC, nPartID, nStateID, (
PTEXTMETRIC)pTextMetric); | |
| 241 #endif // !_WTL_NEW_UXTHEME | |
| 242 } | |
| 243 | |
| 244 HRESULT GetThemeBackgroundRegion(HDC hDC, int nPartID, int nStateID, LPC
RECT pRect, HRGN* pRegion) const | |
| 245 { | |
| 246 ATLASSERT(m_hTheme != NULL); | |
| 247 return ::GetThemeBackgroundRegion(m_hTheme, hDC, nPartID, nState
ID, pRect, pRegion); | |
| 248 } | |
| 249 | |
| 250 HRESULT HitTestThemeBackground(HDC hDC, int nPartID, int nStateID, DWORD
dwOptions, LPCRECT pRect, HRGN hrgn, POINT ptTest, WORD* pwHitTestCode) const | |
| 251 { | |
| 252 ATLASSERT(m_hTheme != NULL); | |
| 253 return ::HitTestThemeBackground(m_hTheme, hDC, nPartID, nStateID
, dwOptions, pRect, hrgn, ptTest, pwHitTestCode); | |
| 254 } | |
| 255 | |
| 256 HRESULT DrawThemeEdge(HDC hDC, int nPartID, int nStateID, LPCRECT pDestR
ect, UINT uEdge, UINT uFlags, LPRECT pContentRect = NULL) | |
| 257 { | |
| 258 ATLASSERT(m_hTheme != NULL); | |
| 259 return ::DrawThemeEdge(m_hTheme, hDC, nPartID, nStateID, pDestRe
ct, uEdge, uFlags, pContentRect); | |
| 260 } | |
| 261 | |
| 262 HRESULT DrawThemeIcon(HDC hDC, int nPartID, int nStateID, LPCRECT pRect,
HIMAGELIST himl, int nImageIndex) | |
| 263 { | |
| 264 ATLASSERT(m_hTheme != NULL); | |
| 265 return ::DrawThemeIcon(m_hTheme, hDC, nPartID, nStateID, pRect,
himl, nImageIndex); | |
| 266 } | |
| 267 | |
| 268 BOOL IsThemePartDefined(int nPartID, int nStateID) const | |
| 269 { | |
| 270 ATLASSERT(m_hTheme != NULL); | |
| 271 return ::IsThemePartDefined(m_hTheme, nPartID, nStateID); | |
| 272 } | |
| 273 | |
| 274 BOOL IsThemeBackgroundPartiallyTransparent(int nPartID, int nStateID) co
nst | |
| 275 { | |
| 276 ATLASSERT(m_hTheme != NULL); | |
| 277 return ::IsThemeBackgroundPartiallyTransparent(m_hTheme, nPartID
, nStateID); | |
| 278 } | |
| 279 | |
| 280 HRESULT GetThemeColor(int nPartID, int nStateID, int nPropID, COLORREF*
pColor) const | |
| 281 { | |
| 282 ATLASSERT(m_hTheme != NULL); | |
| 283 return ::GetThemeColor(m_hTheme, nPartID, nStateID, nPropID, pCo
lor); | |
| 284 } | |
| 285 | |
| 286 HRESULT GetThemeMetric(HDC hDC, int nPartID, int nStateID, int nPropID,
int* pnVal) const | |
| 287 { | |
| 288 ATLASSERT(m_hTheme != NULL); | |
| 289 return ::GetThemeMetric(m_hTheme, hDC, nPartID, nStateID, nPropI
D, pnVal); | |
| 290 } | |
| 291 | |
| 292 HRESULT GetThemeString(int nPartID, int nStateID, int nPropID, LPWSTR ps
zBuff, int cchMaxBuffChars) const | |
| 293 { | |
| 294 ATLASSERT(m_hTheme != NULL); | |
| 295 return ::GetThemeString(m_hTheme, nPartID, nStateID, nPropID, ps
zBuff, cchMaxBuffChars); | |
| 296 } | |
| 297 | |
| 298 HRESULT GetThemeBool(int nPartID, int nStateID, int nPropID, BOOL* pfVal
) const | |
| 299 { | |
| 300 ATLASSERT(m_hTheme != NULL); | |
| 301 return ::GetThemeBool(m_hTheme, nPartID, nStateID, nPropID, pfVa
l); | |
| 302 } | |
| 303 | |
| 304 HRESULT GetThemeInt(int nPartID, int nStateID, int nPropID, int* pnVal)
const | |
| 305 { | |
| 306 ATLASSERT(m_hTheme != NULL); | |
| 307 return ::GetThemeInt(m_hTheme, nPartID, nStateID, nPropID, pnVal
); | |
| 308 } | |
| 309 | |
| 310 HRESULT GetThemeEnumValue(int nPartID, int nStateID, int nPropID, int* p
nVal) const | |
| 311 { | |
| 312 ATLASSERT(m_hTheme != NULL); | |
| 313 return ::GetThemeEnumValue(m_hTheme, nPartID, nStateID, nPropID,
pnVal); | |
| 314 } | |
| 315 | |
| 316 HRESULT GetThemePosition(int nPartID, int nStateID, int nPropID, LPPOINT
pPoint) const | |
| 317 { | |
| 318 ATLASSERT(m_hTheme != NULL); | |
| 319 return ::GetThemePosition(m_hTheme, nPartID, nStateID, nPropID,
pPoint); | |
| 320 } | |
| 321 | |
| 322 // deprecated | |
| 323 HRESULT GetThemeFont(int nPartID, HDC hDC, int nStateID, int nPropID, LO
GFONTW* pFont) const | |
| 324 { | |
| 325 ATLASSERT(m_hTheme != NULL); | |
| 326 #ifdef _WTL_NEW_UXTHEME | |
| 327 return ::GetThemeFont(m_hTheme, hDC, nPartID, nStateID, nPropID,
pFont); | |
| 328 #else // !_WTL_NEW_UXTHEME | |
| 329 // Note: The cast to LOGFONT* is because uxtheme.h incorrectly u
ses it instead of LOGFONTW* | |
| 330 return ::GetThemeFont(m_hTheme, hDC, nPartID, nStateID, nPropID,
(LOGFONT*)pFont); | |
| 331 #endif // !_WTL_NEW_UXTHEME | |
| 332 } | |
| 333 | |
| 334 HRESULT GetThemeFont(HDC hDC, int nPartID, int nStateID, int nPropID, LO
GFONTW* pFont) const | |
| 335 { | |
| 336 ATLASSERT(m_hTheme != NULL); | |
| 337 #ifdef _WTL_NEW_UXTHEME | |
| 338 return ::GetThemeFont(m_hTheme, hDC, nPartID, nStateID, nPropID,
pFont); | |
| 339 #else // !_WTL_NEW_UXTHEME | |
| 340 // Note: The cast to LOGFONT* is because uxtheme.h incorrectly u
ses it instead of LOGFONTW* | |
| 341 return ::GetThemeFont(m_hTheme, hDC, nPartID, nStateID, nPropID,
(LOGFONT*)pFont); | |
| 342 #endif // !_WTL_NEW_UXTHEME | |
| 343 } | |
| 344 | |
| 345 HRESULT GetThemeRect(int nPartID, int nStateID, int nPropID, LPRECT pRec
t) const | |
| 346 { | |
| 347 ATLASSERT(m_hTheme != NULL); | |
| 348 return ::GetThemeRect(m_hTheme, nPartID, nStateID, nPropID, pRec
t); | |
| 349 } | |
| 350 | |
| 351 HRESULT GetThemeMargins(HDC hDC, int nPartID, int nStateID, int nPropID,
LPRECT pRect, PMARGINS pMargins) const | |
| 352 { | |
| 353 ATLASSERT(m_hTheme != NULL); | |
| 354 return ::GetThemeMargins(m_hTheme, hDC, nPartID, nStateID, nProp
ID, pRect, pMargins); | |
| 355 } | |
| 356 | |
| 357 HRESULT GetThemeIntList(int nPartID, int nStateID, int nPropID, INTLIST*
pIntList) const | |
| 358 { | |
| 359 ATLASSERT(m_hTheme != NULL); | |
| 360 return ::GetThemeIntList(m_hTheme, nPartID, nStateID, nPropID, p
IntList); | |
| 361 } | |
| 362 | |
| 363 HRESULT GetThemePropertyOrigin(int nPartID, int nStateID, int nPropID, e
num PROPERTYORIGIN* pOrigin) const | |
| 364 { | |
| 365 ATLASSERT(m_hTheme != NULL); | |
| 366 return ::GetThemePropertyOrigin(m_hTheme, nPartID, nStateID, nPr
opID, pOrigin); | |
| 367 } | |
| 368 | |
| 369 HRESULT GetThemeFilename(int nPartID, int nStateID, int nPropID, LPWSTR
pszThemeFileName, int cchMaxBuffChars) const | |
| 370 { | |
| 371 ATLASSERT(m_hTheme != NULL); | |
| 372 return ::GetThemeFilename(m_hTheme, nPartID, nStateID, nPropID,
pszThemeFileName, cchMaxBuffChars); | |
| 373 } | |
| 374 | |
| 375 COLORREF GetThemeSysColor(int nColorID) const | |
| 376 { | |
| 377 ATLASSERT(m_hTheme != NULL); | |
| 378 return ::GetThemeSysColor(m_hTheme, nColorID); | |
| 379 } | |
| 380 | |
| 381 HBRUSH GetThemeSysColorBrush(int nColorID) const | |
| 382 { | |
| 383 ATLASSERT(m_hTheme != NULL); | |
| 384 return ::GetThemeSysColorBrush(m_hTheme, nColorID); | |
| 385 } | |
| 386 | |
| 387 int GetThemeSysSize(int nSizeID) const | |
| 388 { | |
| 389 ATLASSERT(m_hTheme != NULL); | |
| 390 return ::GetThemeSysSize(m_hTheme, nSizeID); | |
| 391 } | |
| 392 | |
| 393 BOOL GetThemeSysBool(int nBoolID) const | |
| 394 { | |
| 395 ATLASSERT(m_hTheme != NULL); | |
| 396 return ::GetThemeSysBool(m_hTheme, nBoolID); | |
| 397 } | |
| 398 | |
| 399 HRESULT GetThemeSysFont(int nFontID, LOGFONTW* plf) const | |
| 400 { | |
| 401 ATLASSERT(m_hTheme != NULL); | |
| 402 #ifdef _WTL_NEW_UXTHEME | |
| 403 return ::GetThemeSysFont(m_hTheme, nFontID, plf); | |
| 404 #else // !_WTL_NEW_UXTHEME | |
| 405 // Note: The cast to LOGFONT* is because uxtheme.h incorrectly u
ses it instead of LOGFONTW* | |
| 406 return ::GetThemeSysFont(m_hTheme, nFontID, (LOGFONT*)plf); | |
| 407 #endif // !_WTL_NEW_UXTHEME | |
| 408 } | |
| 409 | |
| 410 HRESULT GetThemeSysString(int nStringID, LPWSTR pszStringBuff, int cchMa
xStringChars) const | |
| 411 { | |
| 412 ATLASSERT(m_hTheme != NULL); | |
| 413 return ::GetThemeSysString(m_hTheme, nStringID, pszStringBuff, c
chMaxStringChars); | |
| 414 } | |
| 415 | |
| 416 HRESULT GetThemeSysInt(int nIntID, int* pnValue) const | |
| 417 { | |
| 418 ATLASSERT(m_hTheme != NULL); | |
| 419 return ::GetThemeSysInt(m_hTheme, nIntID, pnValue); | |
| 420 } | |
| 421 | |
| 422 #ifdef _WTL_NEW_UXTHEME | |
| 423 HTHEME OpenThemeDataEx(HWND hWnd, LPCWSTR pszClassList, DWORD dwFlags) | |
| 424 { | |
| 425 if(!IsThemingSupported()) | |
| 426 return NULL; | |
| 427 | |
| 428 ATLASSERT(m_hTheme == NULL); | |
| 429 m_hTheme = ::OpenThemeDataEx(hWnd, pszClassList, dwFlags); | |
| 430 return m_hTheme; | |
| 431 } | |
| 432 | |
| 433 HRESULT DrawThemeTextEx(HDC hDC, int nPartID, int nStateID, LPCWSTR pszT
ext, int cchText, DWORD dwTextFlags, LPRECT lpRect, const DTTOPTS* pOptions) | |
| 434 { | |
| 435 ATLASSERT(m_hTheme != NULL); | |
| 436 return ::DrawThemeTextEx(m_hTheme, hDC, nPartID, nStateID, pszTe
xt, cchText, dwTextFlags, lpRect, pOptions); | |
| 437 } | |
| 438 | |
| 439 HRESULT GetThemeTransitionDuration(int nPartID, int nFromStateID, int nT
oStateID, int nPropID, DWORD& dwDuration) | |
| 440 { | |
| 441 ATLASSERT(m_hTheme != NULL); | |
| 442 return ::GetThemeTransitionDuration(m_hTheme, nPartID, nFromStat
eID, nToStateID, nPropID, &dwDuration); | |
| 443 } | |
| 444 #endif // _WTL_NEW_UXTHEME | |
| 445 | |
| 446 #if (_WIN32_WINNT >= 0x0600) | |
| 447 HRESULT GetThemeBitmap(int nPartID, int nStateID, int nPropID, ULONG uFl
ags, HBITMAP& hBitmap) | |
| 448 { | |
| 449 ATLASSERT(m_hTheme != NULL); | |
| 450 return ::GetThemeBitmap(m_hTheme, nPartID, nStateID, nPropID, uF
lags, &hBitmap); | |
| 451 } | |
| 452 | |
| 453 HRESULT GetThemeStream(int nPartID, int nStateID, int nPropID, VOID** pp
vStream, DWORD* pcbStream, HINSTANCE hInstance) | |
| 454 { | |
| 455 ATLASSERT(m_hTheme != NULL); | |
| 456 return ::GetThemeStream(m_hTheme, nPartID, nStateID, nPropID, pp
vStream, pcbStream, hInstance); | |
| 457 } | |
| 458 #endif // (_WIN32_WINNT >= 0x0600) | |
| 459 }; | |
| 460 | |
| 461 __declspec(selectany) int CTheme::m_nIsThemingSupported = -1; | |
| 462 | |
| 463 | |
| 464 /////////////////////////////////////////////////////////////////////////////// | |
| 465 // CThemeImpl - theme support implementation | |
| 466 | |
| 467 // Derive from this class to implement window with theme support. | |
| 468 // Example: | |
| 469 // class CMyThemeWindow : public CWindowImpl<CMyThemeWindow>, public CTheme
Impl<CMyThemeWindow> | |
| 470 // { | |
| 471 // ... | |
| 472 // BEGIN_MSG_MAP(CMyThemeWindow) | |
| 473 // CHAIN_MSG_MAP(CThemeImpl<CMyThemeWindow>) | |
| 474 // ... | |
| 475 // END_MSG_MAP() | |
| 476 // ... | |
| 477 // }; | |
| 478 // | |
| 479 // If you set theme class list, the class will automaticaly open/close/reopen th
eme data. | |
| 480 | |
| 481 | |
| 482 // Helper for drawing theme client edge | |
| 483 inline bool AtlDrawThemeClientEdge(HTHEME hTheme, HWND hWnd, HRGN hRgnUpdate = N
ULL, HBRUSH hBrush = NULL, int nPartID = 0, int nStateID = 0) | |
| 484 { | |
| 485 ATLASSERT(hTheme != NULL); | |
| 486 ATLASSERT(::IsWindow(hWnd)); | |
| 487 | |
| 488 CWindowDC dc(hWnd); | |
| 489 if(dc.IsNull()) | |
| 490 return false; | |
| 491 | |
| 492 // Get border size | |
| 493 int cxBorder = GetSystemMetrics(SM_CXBORDER); | |
| 494 int cyBorder = GetSystemMetrics(SM_CYBORDER); | |
| 495 if(SUCCEEDED(::GetThemeInt(hTheme, nPartID, nStateID, TMT_SIZINGBORDERWI
DTH, &cxBorder))) | |
| 496 cyBorder = cxBorder; | |
| 497 | |
| 498 RECT rect; | |
| 499 ::GetWindowRect(hWnd, &rect); | |
| 500 | |
| 501 // Remove the client edge from the update region | |
| 502 int cxEdge = GetSystemMetrics(SM_CXEDGE); | |
| 503 int cyEdge = GetSystemMetrics(SM_CYEDGE); | |
| 504 ::InflateRect(&rect, -cxEdge, -cyEdge); | |
| 505 CRgn rgn; | |
| 506 rgn.CreateRectRgnIndirect(&rect); | |
| 507 if(rgn.IsNull()) | |
| 508 return false; | |
| 509 | |
| 510 if(hRgnUpdate != NULL) | |
| 511 rgn.CombineRgn(hRgnUpdate, rgn, RGN_AND); | |
| 512 | |
| 513 ::OffsetRect(&rect, -rect.left, -rect.top); | |
| 514 | |
| 515 ::OffsetRect(&rect, cxEdge, cyEdge); | |
| 516 dc.ExcludeClipRect(&rect); | |
| 517 ::InflateRect(&rect, cxEdge, cyEdge); | |
| 518 | |
| 519 ::DrawThemeBackground(hTheme, dc, nPartID, nStateID, &rect, NULL); | |
| 520 | |
| 521 // Use background brush too, since theme border might not cover everythi
ng | |
| 522 if(cxBorder < cxEdge && cyBorder < cyEdge) | |
| 523 { | |
| 524 if(hBrush == NULL) | |
| 525 // need conditional code because types don't match in winuser.h | |
| 526 #ifdef _WIN64 | |
| 527 hBrush = (HBRUSH)::GetClassLongPtr(hWnd, GCLP_HBRBACKGRO
UND); | |
| 528 #else | |
| 529 hBrush = (HBRUSH)UlongToPtr(::GetClassLongPtr(hWnd, GCLP
_HBRBACKGROUND)); | |
| 530 #endif | |
| 531 | |
| 532 ::InflateRect(&rect, cxBorder - cxEdge, cyBorder - cyEdge); | |
| 533 dc.FillRect(&rect, hBrush); | |
| 534 } | |
| 535 | |
| 536 ::DefWindowProc(hWnd, WM_NCPAINT, (WPARAM)rgn.m_hRgn, 0L); | |
| 537 | |
| 538 return true; | |
| 539 } | |
| 540 | |
| 541 | |
| 542 // Theme extended styles | |
| 543 #define THEME_EX_3DCLIENTEDGE 0x00000001 | |
| 544 #define THEME_EX_THEMECLIENTEDGE 0x00000002 | |
| 545 | |
| 546 template <class T, class TBase = CTheme> | |
| 547 class CThemeImpl : public TBase | |
| 548 { | |
| 549 public: | |
| 550 // Data members | |
| 551 LPWSTR m_lpstrThemeClassList; | |
| 552 DWORD m_dwExtendedStyle; // theme specific extended styles | |
| 553 | |
| 554 // Constructor & destructor | |
| 555 CThemeImpl() : m_lpstrThemeClassList(NULL), m_dwExtendedStyle(0) | |
| 556 { } | |
| 557 | |
| 558 ~CThemeImpl() | |
| 559 { | |
| 560 delete [] m_lpstrThemeClassList; | |
| 561 } | |
| 562 | |
| 563 // Attributes | |
| 564 bool SetThemeClassList(LPCWSTR lpstrThemeClassList) | |
| 565 { | |
| 566 if(m_lpstrThemeClassList != NULL) | |
| 567 { | |
| 568 delete [] m_lpstrThemeClassList; | |
| 569 m_lpstrThemeClassList = NULL; | |
| 570 } | |
| 571 | |
| 572 if(lpstrThemeClassList == NULL) | |
| 573 return true; | |
| 574 | |
| 575 int cchLen = lstrlenW(lpstrThemeClassList) + 1; | |
| 576 ATLTRY(m_lpstrThemeClassList = new WCHAR[cchLen]); | |
| 577 if(m_lpstrThemeClassList == NULL) | |
| 578 return false; | |
| 579 | |
| 580 SecureHelper::strcpyW_x(m_lpstrThemeClassList, cchLen, lpstrThem
eClassList); | |
| 581 | |
| 582 return true; | |
| 583 } | |
| 584 | |
| 585 bool GetThemeClassList(LPWSTR lpstrThemeClassList, int cchListBuffer) co
nst | |
| 586 { | |
| 587 int cchLen = lstrlenW(m_lpstrThemeClassList) + 1; | |
| 588 if(cchListBuffer < cchLen) | |
| 589 return false; | |
| 590 | |
| 591 SecureHelper::strcpyW_x(lpstrThemeClassList, cchListBuffer, m_lp
strThemeClassList); | |
| 592 | |
| 593 return true; | |
| 594 } | |
| 595 | |
| 596 LPCWSTR GetThemeClassList() const | |
| 597 { | |
| 598 return m_lpstrThemeClassList; | |
| 599 } | |
| 600 | |
| 601 DWORD SetThemeExtendedStyle(DWORD dwExtendedStyle, DWORD dwMask = 0) | |
| 602 { | |
| 603 DWORD dwPrevStyle = m_dwExtendedStyle; | |
| 604 if(dwMask == 0) | |
| 605 m_dwExtendedStyle = dwExtendedStyle; | |
| 606 else | |
| 607 m_dwExtendedStyle = (m_dwExtendedStyle & ~dwMask) | (dwE
xtendedStyle & dwMask); | |
| 608 return dwPrevStyle; | |
| 609 } | |
| 610 | |
| 611 DWORD GetThemeExtendedStyle() const | |
| 612 { | |
| 613 return m_dwExtendedStyle; | |
| 614 } | |
| 615 | |
| 616 // Operations | |
| 617 HTHEME OpenThemeData() | |
| 618 { | |
| 619 T* pT = static_cast<T*>(this); | |
| 620 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 621 ATLASSERT(m_lpstrThemeClassList != NULL); | |
| 622 if(m_lpstrThemeClassList == NULL) | |
| 623 return NULL; | |
| 624 CloseThemeData(); | |
| 625 return TBase::OpenThemeData(pT->m_hWnd, m_lpstrThemeClassList); | |
| 626 } | |
| 627 | |
| 628 HTHEME OpenThemeData(LPCWSTR pszClassList) | |
| 629 { | |
| 630 if(!SetThemeClassList(pszClassList)) | |
| 631 return NULL; | |
| 632 return OpenThemeData(); | |
| 633 } | |
| 634 | |
| 635 HRESULT SetWindowTheme(LPCWSTR pszSubAppName, LPCWSTR pszSubIDList) | |
| 636 { | |
| 637 if(!IsThemingSupported()) | |
| 638 return S_FALSE; | |
| 639 | |
| 640 T* pT = static_cast<T*>(this); | |
| 641 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 642 return ::SetWindowTheme(pT->m_hWnd, pszSubAppName, pszSubIDList)
; | |
| 643 } | |
| 644 | |
| 645 HTHEME GetWindowTheme() const | |
| 646 { | |
| 647 if(!IsThemingSupported()) | |
| 648 return NULL; | |
| 649 | |
| 650 const T* pT = static_cast<const T*>(this); | |
| 651 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 652 return ::GetWindowTheme(pT->m_hWnd); | |
| 653 } | |
| 654 | |
| 655 HRESULT EnableThemeDialogTexture(DWORD dwFlags) | |
| 656 { | |
| 657 if(!IsThemingSupported()) | |
| 658 return S_FALSE; | |
| 659 | |
| 660 T* pT = static_cast<T*>(this); | |
| 661 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 662 return ::EnableThemeDialogTexture(pT->m_hWnd, dwFlags); | |
| 663 } | |
| 664 | |
| 665 BOOL IsThemeDialogTextureEnabled() const | |
| 666 { | |
| 667 if(!IsThemingSupported()) | |
| 668 return FALSE; | |
| 669 | |
| 670 const T* pT = static_cast<const T*>(this); | |
| 671 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 672 return ::IsThemeDialogTextureEnabled(pT->m_hWnd); | |
| 673 } | |
| 674 | |
| 675 HRESULT DrawThemeParentBackground(HDC hDC, const RECT* pRect = NULL) | |
| 676 { | |
| 677 if(!IsThemingSupported()) | |
| 678 return S_FALSE; | |
| 679 | |
| 680 T* pT = static_cast<T*>(this); | |
| 681 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 682 #ifdef _WTL_NEW_UXTHEME | |
| 683 return ::DrawThemeParentBackground(pT->m_hWnd, hDC, pRect); | |
| 684 #else | |
| 685 return ::DrawThemeParentBackground(pT->m_hWnd, hDC, (RECT*)pRect
); | |
| 686 #endif | |
| 687 } | |
| 688 | |
| 689 #ifdef _WTL_NEW_UXTHEME | |
| 690 HRESULT SetWindowThemeAttribute(WINDOWTHEMEATTRIBUTETYPE type, PVOID pvA
ttribute, DWORD cbAttribute) | |
| 691 { | |
| 692 if(!IsThemingSupported()) | |
| 693 return S_FALSE; | |
| 694 | |
| 695 T* pT = static_cast<T*>(this); | |
| 696 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 697 return ::SetWindowThemeAttribute(pT->m_hWnd, type, pvAttribute,
cbAttribute); | |
| 698 } | |
| 699 | |
| 700 HRESULT SetWindowThemeNonClientAttributes(DWORD dwAttributes, DWORD dwMa
sk) | |
| 701 { | |
| 702 if(!IsThemingSupported()) | |
| 703 return S_FALSE; | |
| 704 | |
| 705 T* pT = static_cast<T*>(this); | |
| 706 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 707 WTA_OPTIONS opt = { dwAttributes, dwMask }; | |
| 708 return ::SetWindowThemeAttribute(pT->m_hWnd, WTA_NONCLIENT, (PVO
ID)&opt, sizeof(opt)); | |
| 709 } | |
| 710 | |
| 711 HRESULT DrawThemeParentBackgroundEx(HDC hDC, DWORD dwFlags, const RECT*
lpRect = NULL) | |
| 712 { | |
| 713 if(!IsThemingSupported()) | |
| 714 return S_FALSE; | |
| 715 | |
| 716 T* pT = static_cast<T*>(this); | |
| 717 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 718 return ::DrawThemeParentBackgroundEx(pT->m_hWnd, hDC, dwFlags, l
pRect); | |
| 719 } | |
| 720 #endif // _WTL_NEW_UXTHEME | |
| 721 | |
| 722 // Message map and handlers | |
| 723 // Note: If you handle any of these messages in your derived class, | |
| 724 // it is better to put CHAIN_MSG_MAP at the start of your message map. | |
| 725 BEGIN_MSG_MAP(CThemeImpl) | |
| 726 MESSAGE_HANDLER(WM_CREATE, OnCreate) | |
| 727 MESSAGE_HANDLER(WM_DESTROY, OnDestroy) | |
| 728 MESSAGE_HANDLER(WM_THEMECHANGED, OnThemeChanged) | |
| 729 MESSAGE_HANDLER(WM_NCPAINT, OnNcPaint) | |
| 730 END_MSG_MAP() | |
| 731 | |
| 732 LRESULT OnCreate(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BO
OL& bHandled) | |
| 733 { | |
| 734 if(m_lpstrThemeClassList != NULL) | |
| 735 OpenThemeData(); | |
| 736 bHandled = FALSE; | |
| 737 return 1; | |
| 738 } | |
| 739 | |
| 740 LRESULT OnDestroy(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, B
OOL& bHandled) | |
| 741 { | |
| 742 CloseThemeData(); | |
| 743 bHandled = FALSE; | |
| 744 return 1; | |
| 745 } | |
| 746 | |
| 747 LRESULT OnThemeChanged(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam
*/, BOOL& bHandled) | |
| 748 { | |
| 749 CloseThemeData(); | |
| 750 if(m_lpstrThemeClassList != NULL) | |
| 751 OpenThemeData(); | |
| 752 bHandled = FALSE; | |
| 753 return 1; | |
| 754 } | |
| 755 | |
| 756 LRESULT OnNcPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandle
d) | |
| 757 { | |
| 758 T* pT = static_cast<T*>(this); | |
| 759 ATLASSERT(::IsWindow(pT->m_hWnd)); | |
| 760 LRESULT lRet = 0; | |
| 761 bHandled = FALSE; | |
| 762 if(IsThemingSupported() && ((pT->GetExStyle() & WS_EX_CLIENTEDGE
) != 0)) | |
| 763 { | |
| 764 if((m_dwExtendedStyle & THEME_EX_3DCLIENTEDGE) != 0) | |
| 765 { | |
| 766 lRet = ::DefWindowProc(pT->m_hWnd, uMsg, wParam,
lParam); | |
| 767 bHandled = TRUE; | |
| 768 } | |
| 769 else if((m_hTheme != NULL) && ((m_dwExtendedStyle & THEM
E_EX_THEMECLIENTEDGE) != 0)) | |
| 770 { | |
| 771 HRGN hRgn = (wParam != 1) ? (HRGN)wParam : NULL; | |
| 772 if(pT->DrawThemeClientEdge(hRgn)) | |
| 773 bHandled = TRUE; | |
| 774 } | |
| 775 } | |
| 776 return lRet; | |
| 777 } | |
| 778 | |
| 779 // Drawing helper | |
| 780 bool DrawThemeClientEdge(HRGN hRgnUpdate) | |
| 781 { | |
| 782 T* pT = static_cast<T*>(this); | |
| 783 return AtlDrawThemeClientEdge(m_hTheme, pT->m_hWnd, hRgnUpdate,
NULL, 0, 0); | |
| 784 } | |
| 785 }; | |
| 786 | |
| 787 /////////////////////////////////////////////////////////////////////////////// | |
| 788 // Buffered Paint and Animation | |
| 789 | |
| 790 #ifdef _WTL_NEW_UXTHEME | |
| 791 | |
| 792 /////////////////////////////////////////////////////////////////////////////// | |
| 793 // CBufferedPaintBase - Buffered Paint support for othe classes | |
| 794 | |
| 795 class CBufferedPaintBase | |
| 796 { | |
| 797 public: | |
| 798 static int m_nIsBufferedPaintSupported; | |
| 799 | |
| 800 CBufferedPaintBase() | |
| 801 { | |
| 802 if(IsBufferedPaintSupported()) | |
| 803 ATLVERIFY(SUCCEEDED(::BufferedPaintInit())); | |
| 804 } | |
| 805 | |
| 806 ~CBufferedPaintBase() | |
| 807 { | |
| 808 if(IsBufferedPaintSupported()) | |
| 809 ATLVERIFY(SUCCEEDED(::BufferedPaintUnInit())); | |
| 810 } | |
| 811 | |
| 812 static bool IsBufferedPaintSupported() | |
| 813 { | |
| 814 if(m_nIsBufferedPaintSupported == -1) | |
| 815 { | |
| 816 CStaticDataInitCriticalSectionLock lock; | |
| 817 if(FAILED(lock.Lock())) | |
| 818 { | |
| 819 ATLTRACE2(atlTraceUI, 0, _T("ERROR : Unable to l
ock critical section in CBufferedPaintBase::IsBufferedPaintSupported.\n")); | |
| 820 ATLASSERT(FALSE); | |
| 821 return false; | |
| 822 } | |
| 823 | |
| 824 if(m_nIsBufferedPaintSupported == -1) | |
| 825 m_nIsBufferedPaintSupported = RunTimeHelper::IsV
ista() ? 1 : 0; | |
| 826 | |
| 827 lock.Unlock(); | |
| 828 } | |
| 829 | |
| 830 ATLASSERT(m_nIsBufferedPaintSupported != -1); | |
| 831 return (m_nIsBufferedPaintSupported == 1); | |
| 832 } | |
| 833 }; | |
| 834 | |
| 835 __declspec(selectany) int CBufferedPaintBase::m_nIsBufferedPaintSupported = -1; | |
| 836 | |
| 837 | |
| 838 /////////////////////////////////////////////////////////////////////////////// | |
| 839 // CBufferedPaint - support for buffered paint functions | |
| 840 | |
| 841 class CBufferedPaint | |
| 842 { | |
| 843 public: | |
| 844 HPAINTBUFFER m_hPaintBuffer; | |
| 845 | |
| 846 CBufferedPaint() : m_hPaintBuffer(NULL) | |
| 847 { } | |
| 848 | |
| 849 ~CBufferedPaint() | |
| 850 { | |
| 851 ATLVERIFY(SUCCEEDED(End())); | |
| 852 } | |
| 853 | |
| 854 bool IsNull() const | |
| 855 { | |
| 856 return (m_hPaintBuffer == NULL); | |
| 857 } | |
| 858 | |
| 859 HPAINTBUFFER Begin(HDC hdcTarget, const RECT* prcTarget, BP_BUFFERFORMAT
dwFormat, BP_PAINTPARAMS* pPaintParams, HDC* phdcPaint) | |
| 860 { | |
| 861 ATLASSERT(m_hPaintBuffer == NULL); | |
| 862 m_hPaintBuffer = ::BeginBufferedPaint(hdcTarget, prcTarget, dwFo
rmat, pPaintParams, phdcPaint); | |
| 863 return m_hPaintBuffer; | |
| 864 } | |
| 865 | |
| 866 HRESULT End(BOOL bUpdate = TRUE) | |
| 867 { | |
| 868 HRESULT hRet = S_FALSE; | |
| 869 if(m_hPaintBuffer != NULL) | |
| 870 { | |
| 871 hRet = ::EndBufferedPaint(m_hPaintBuffer, bUpdate); | |
| 872 m_hPaintBuffer = NULL; | |
| 873 } | |
| 874 return hRet; | |
| 875 } | |
| 876 | |
| 877 HRESULT GetTargetRect(LPRECT pRect) const | |
| 878 { | |
| 879 ATLASSERT(m_hPaintBuffer != NULL); | |
| 880 return ::GetBufferedPaintTargetRect(m_hPaintBuffer, pRect); | |
| 881 } | |
| 882 | |
| 883 HDC GetTargetDC() const | |
| 884 { | |
| 885 ATLASSERT(m_hPaintBuffer != NULL); | |
| 886 return ::GetBufferedPaintTargetDC(m_hPaintBuffer); | |
| 887 } | |
| 888 | |
| 889 HDC GetPaintDC() const | |
| 890 { | |
| 891 ATLASSERT(m_hPaintBuffer != NULL); | |
| 892 return ::GetBufferedPaintDC(m_hPaintBuffer); | |
| 893 } | |
| 894 | |
| 895 HRESULT GetBits(RGBQUAD** ppbBuffer, int* pcxRow) const | |
| 896 { | |
| 897 ATLASSERT(m_hPaintBuffer != NULL); | |
| 898 return ::GetBufferedPaintBits(m_hPaintBuffer, ppbBuffer, pcxRow)
; | |
| 899 } | |
| 900 | |
| 901 HRESULT Clear(const RECT* pRect = NULL) | |
| 902 { | |
| 903 ATLASSERT(m_hPaintBuffer != NULL); | |
| 904 return ::BufferedPaintClear(m_hPaintBuffer, pRect); | |
| 905 } | |
| 906 | |
| 907 HRESULT SetAlpha(BYTE alpha, const RECT* pRect = NULL) | |
| 908 { | |
| 909 ATLASSERT(m_hPaintBuffer != NULL); | |
| 910 return ::BufferedPaintSetAlpha(m_hPaintBuffer, pRect, alpha); | |
| 911 } | |
| 912 | |
| 913 HRESULT MakeOpaque(const RECT* pRect = NULL) | |
| 914 { | |
| 915 ATLASSERT(m_hPaintBuffer != NULL); | |
| 916 return ::BufferedPaintSetAlpha(m_hPaintBuffer, pRect, 255); | |
| 917 } | |
| 918 }; | |
| 919 | |
| 920 | |
| 921 /////////////////////////////////////////////////////////////////////////////// | |
| 922 // CBufferedPaintImpl - provides buffered paint for any window | |
| 923 | |
| 924 template <class T> | |
| 925 class ATL_NO_VTABLE CBufferedPaintImpl : public CBufferedPaintBase | |
| 926 { | |
| 927 public: | |
| 928 CBufferedPaint m_BufferedPaint; | |
| 929 BP_BUFFERFORMAT m_dwFormat; | |
| 930 BP_PAINTPARAMS m_PaintParams; | |
| 931 | |
| 932 CBufferedPaintImpl() : m_dwFormat(BPBF_TOPDOWNDIB) | |
| 933 { | |
| 934 memset(&m_PaintParams, 0, sizeof(BP_PAINTPARAMS)); | |
| 935 m_PaintParams.cbSize = sizeof(BP_PAINTPARAMS); | |
| 936 } | |
| 937 | |
| 938 // Message map and handlers | |
| 939 BEGIN_MSG_MAP(CBufferedPaintImpl) | |
| 940 MESSAGE_HANDLER(WM_ERASEBKGND, OnEraseBackground) | |
| 941 MESSAGE_HANDLER(WM_PAINT, OnPaint) | |
| 942 MESSAGE_HANDLER(WM_PRINTCLIENT, OnPaint) | |
| 943 END_MSG_MAP() | |
| 944 | |
| 945 LRESULT OnEraseBackground(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lPa
ram*/, BOOL& /*bHandled*/) | |
| 946 { | |
| 947 return 1; // no background needed | |
| 948 } | |
| 949 | |
| 950 LRESULT OnPaint(UINT /*uMsg*/, WPARAM wParam, LPARAM /*lParam*/, BOOL& /
*bHandled*/) | |
| 951 { | |
| 952 T* pT = static_cast<T*>(this); | |
| 953 if(wParam != NULL) | |
| 954 { | |
| 955 RECT rect = { 0 }; | |
| 956 pT->GetClientRect(&rect); | |
| 957 pT->DoPaint((HDC)wParam, rect); | |
| 958 } | |
| 959 else | |
| 960 { | |
| 961 CPaintDC dc(pT->m_hWnd); | |
| 962 pT->DoBufferedPaint(dc.m_hDC, dc.m_ps.rcPaint); | |
| 963 } | |
| 964 | |
| 965 return 0; | |
| 966 } | |
| 967 | |
| 968 // Overrideables | |
| 969 void DoBufferedPaint(CDCHandle dc, RECT& rect) | |
| 970 { | |
| 971 HDC hDCPaint = NULL; | |
| 972 if(IsBufferedPaintSupported()) | |
| 973 m_BufferedPaint.Begin(dc, &rect, m_dwFormat, &m_PaintPar
ams, &hDCPaint); | |
| 974 | |
| 975 T* pT = static_cast<T*>(this); | |
| 976 if(hDCPaint != NULL) | |
| 977 pT->DoPaint(hDCPaint, rect); | |
| 978 else | |
| 979 pT->DoPaint(dc.m_hDC, rect); | |
| 980 | |
| 981 if(IsBufferedPaintSupported()) | |
| 982 m_BufferedPaint.End(); | |
| 983 } | |
| 984 | |
| 985 void DoPaint(CDCHandle /*dc*/, RECT& /*rect*/) | |
| 986 { | |
| 987 // must be implemented in a derived class | |
| 988 ATLASSERT(FALSE); | |
| 989 } | |
| 990 }; | |
| 991 | |
| 992 | |
| 993 /////////////////////////////////////////////////////////////////////////////// | |
| 994 // CBufferedPaintWindowImpl - implements a window that uses buffered paint | |
| 995 | |
| 996 template <class T, class TBase = ATL::CWindow, class TWinTraits = ATL::CControlW
inTraits> | |
| 997 class ATL_NO_VTABLE CBufferedPaintWindowImpl : | |
| 998 public ATL::CWindowImpl<T, TBase, TWinTraits>, | |
| 999 public CBufferedPaintImpl< T > | |
| 1000 { | |
| 1001 public: | |
| 1002 BEGIN_MSG_MAP(CBufferedPaintWindowImpl) | |
| 1003 CHAIN_MSG_MAP(CBufferedPaintImpl< T >) | |
| 1004 END_MSG_MAP() | |
| 1005 }; | |
| 1006 | |
| 1007 | |
| 1008 /////////////////////////////////////////////////////////////////////////////// | |
| 1009 // CBufferedAnimation - support for buffered animation | |
| 1010 | |
| 1011 class CBufferedAnimation | |
| 1012 { | |
| 1013 public: | |
| 1014 HANIMATIONBUFFER m_hAnimationBuffer; | |
| 1015 | |
| 1016 CBufferedAnimation() : m_hAnimationBuffer(NULL) | |
| 1017 { } | |
| 1018 | |
| 1019 ~CBufferedAnimation() | |
| 1020 { | |
| 1021 ATLVERIFY(SUCCEEDED(End())); | |
| 1022 } | |
| 1023 | |
| 1024 bool IsNull() const | |
| 1025 { | |
| 1026 return (m_hAnimationBuffer == NULL); | |
| 1027 } | |
| 1028 | |
| 1029 HANIMATIONBUFFER Begin(HWND hWnd, HDC hDCTarget, const RECT* pRectTarget
, BP_BUFFERFORMAT dwFormat, BP_PAINTPARAMS* pPaintParams, BP_ANIMATIONPARAMS* pA
nimationParams, HDC* phdcFrom, HDC* phdcTo) | |
| 1030 { | |
| 1031 ATLASSERT(m_hAnimationBuffer == NULL); | |
| 1032 m_hAnimationBuffer = ::BeginBufferedAnimation(hWnd, hDCTarget, p
RectTarget, dwFormat, pPaintParams, pAnimationParams, phdcFrom, phdcTo); | |
| 1033 return m_hAnimationBuffer; | |
| 1034 } | |
| 1035 | |
| 1036 HRESULT End(BOOL bUpdate = TRUE) | |
| 1037 { | |
| 1038 HRESULT hRet = S_FALSE; | |
| 1039 if(m_hAnimationBuffer != NULL) | |
| 1040 { | |
| 1041 hRet = ::EndBufferedAnimation(m_hAnimationBuffer, bUpdat
e); | |
| 1042 m_hAnimationBuffer = NULL; | |
| 1043 } | |
| 1044 return hRet; | |
| 1045 } | |
| 1046 | |
| 1047 static bool IsRendering(HWND hWnd, HDC hDC) | |
| 1048 { | |
| 1049 return (::BufferedPaintRenderAnimation(hWnd, hDC) != FALSE); | |
| 1050 } | |
| 1051 }; | |
| 1052 | |
| 1053 | |
| 1054 /////////////////////////////////////////////////////////////////////////////// | |
| 1055 // CBufferedAnimationImpl - provides buffered animation support for any window | |
| 1056 | |
| 1057 // Note: You can either use m_State and m_NewState to store the state informatio
n | |
| 1058 // for the animation change, or map your state to those data members. DoPaint() | |
| 1059 // should only rely on the state information that is passed to it. | |
| 1060 | |
| 1061 template <class T, class TState = DWORD_PTR> | |
| 1062 class ATL_NO_VTABLE CBufferedAnimationImpl : public CBufferedPaintBase | |
| 1063 { | |
| 1064 public: | |
| 1065 BP_BUFFERFORMAT m_dwFormat; | |
| 1066 BP_PAINTPARAMS m_PaintParams; | |
| 1067 BP_ANIMATIONPARAMS m_AnimationParams; | |
| 1068 | |
| 1069 TState m_State; | |
| 1070 TState m_NewState; | |
| 1071 | |
| 1072 CBufferedAnimationImpl(TState InitialState) : m_dwFormat(BPBF_TOPDOWNDIB
) | |
| 1073 { | |
| 1074 memset(&m_PaintParams, 0, sizeof(BP_PAINTPARAMS)); | |
| 1075 m_PaintParams.cbSize = sizeof(BP_PAINTPARAMS); | |
| 1076 | |
| 1077 memset(&m_AnimationParams, 0, sizeof(BP_ANIMATIONPARAMS)); | |
| 1078 m_AnimationParams.cbSize = sizeof(BP_ANIMATIONPARAMS); | |
| 1079 m_AnimationParams.style = BPAS_LINEAR; | |
| 1080 m_AnimationParams.dwDuration = 500; | |
| 1081 | |
| 1082 T* pT = static_cast<T*>(this); | |
| 1083 pT->SetState(InitialState); | |
| 1084 pT->SetNewState(InitialState); | |
| 1085 } | |
| 1086 | |
| 1087 DWORD GetDuration() const | |
| 1088 { | |
| 1089 return m_AnimationParams.dwDuration; | |
| 1090 } | |
| 1091 | |
| 1092 void SetDuration(DWORD dwDuration) | |
| 1093 { | |
| 1094 m_AnimationParams.dwDuration = dwDuration; | |
| 1095 } | |
| 1096 | |
| 1097 void DoAnimation(TState NewState, const RECT* pRect = NULL) | |
| 1098 { | |
| 1099 T* pT = static_cast<T*>(this); | |
| 1100 pT->SetNewState(NewState); | |
| 1101 | |
| 1102 pT->InvalidateRect(pRect, FALSE); | |
| 1103 pT->UpdateWindow(); | |
| 1104 | |
| 1105 pT->SetState(NewState); | |
| 1106 } | |
| 1107 | |
| 1108 // Message map and handlers | |
| 1109 BEGIN_MSG_MAP(CBufferedAnimationImpl) | |
| 1110 MESSAGE_HANDLER(WM_ERASEBKGND, OnEraseBackground) | |
| 1111 MESSAGE_HANDLER(WM_PAINT, OnPaint) | |
| 1112 MESSAGE_HANDLER(WM_PRINTCLIENT, OnPaint) | |
| 1113 END_MSG_MAP() | |
| 1114 | |
| 1115 LRESULT OnEraseBackground(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lPa
ram*/, BOOL& /*bHandled*/) | |
| 1116 { | |
| 1117 return 1; // no background needed | |
| 1118 } | |
| 1119 | |
| 1120 LRESULT OnPaint(UINT /*uMsg*/, WPARAM wParam, LPARAM /*lParam*/, BOOL& /
*bHandled*/) | |
| 1121 { | |
| 1122 T* pT = static_cast<T*>(this); | |
| 1123 if(wParam != NULL) | |
| 1124 { | |
| 1125 RECT rect = { 0 }; | |
| 1126 pT->GetClientRect(&rect); | |
| 1127 pT->DoPaint((HDC)wParam, rect, m_NewState); | |
| 1128 } | |
| 1129 else | |
| 1130 { | |
| 1131 CPaintDC dc(pT->m_hWnd); | |
| 1132 pT->DoAnimationPaint(dc.m_hDC, dc.m_ps.rcPaint); | |
| 1133 } | |
| 1134 | |
| 1135 return 0; | |
| 1136 } | |
| 1137 | |
| 1138 // Overrideables | |
| 1139 void SetState(TState State) | |
| 1140 { | |
| 1141 m_State = State; | |
| 1142 } | |
| 1143 | |
| 1144 void SetNewState(TState State) | |
| 1145 { | |
| 1146 m_NewState = State; | |
| 1147 } | |
| 1148 | |
| 1149 bool AreStatesEqual() const | |
| 1150 { | |
| 1151 return (m_State == m_NewState); | |
| 1152 } | |
| 1153 | |
| 1154 void DoAnimationPaint(CDCHandle dc, RECT& rect) | |
| 1155 { | |
| 1156 T* pT = static_cast<T*>(this); | |
| 1157 if(IsBufferedPaintSupported() && CBufferedAnimation::IsRendering
(pT->m_hWnd, dc)) | |
| 1158 return; | |
| 1159 | |
| 1160 DWORD dwDurationSave = m_AnimationParams.dwDuration; | |
| 1161 if(pT->AreStatesEqual()) | |
| 1162 m_AnimationParams.dwDuration = 0; | |
| 1163 | |
| 1164 HDC hdcFrom = NULL, hdcTo = NULL; | |
| 1165 CBufferedAnimation ba; | |
| 1166 if(IsBufferedPaintSupported()) | |
| 1167 ba.Begin(pT->m_hWnd, dc, &rect, m_dwFormat, &m_PaintPara
ms, &m_AnimationParams, &hdcFrom, &hdcTo); | |
| 1168 | |
| 1169 if(!ba.IsNull()) | |
| 1170 { | |
| 1171 if(hdcFrom != NULL) | |
| 1172 pT->DoPaint(hdcFrom, rect, m_State); | |
| 1173 | |
| 1174 if (hdcTo != NULL) | |
| 1175 pT->DoPaint(hdcTo, rect, m_NewState); | |
| 1176 } | |
| 1177 else | |
| 1178 { | |
| 1179 pT->DoPaint(dc.m_hDC, rect, m_NewState); | |
| 1180 } | |
| 1181 | |
| 1182 m_AnimationParams.dwDuration = dwDurationSave; | |
| 1183 } | |
| 1184 | |
| 1185 void DoPaint(CDCHandle /*dc*/, RECT& /*rect*/, TState /*State*/) | |
| 1186 { | |
| 1187 // must be implemented in a derived class | |
| 1188 ATLASSERT(FALSE); | |
| 1189 } | |
| 1190 }; | |
| 1191 | |
| 1192 | |
| 1193 /////////////////////////////////////////////////////////////////////////////// | |
| 1194 // CBufferedAnimationWindowImpl - implements a window that uses buffered animati
on | |
| 1195 | |
| 1196 template <class T, class TState = DWORD_PTR, class TBase = ATL::CWindow, class T
WinTraits = ATL::CControlWinTraits> | |
| 1197 class ATL_NO_VTABLE CBufferedAnimationWindowImpl : | |
| 1198 public ATL::CWindowImpl<T, TBase, TWinTraits>, | |
| 1199 public CBufferedAnimationImpl< T, TState > | |
| 1200 { | |
| 1201 public: | |
| 1202 CBufferedAnimationWindowImpl(TState InitialState) : CBufferedAnimationIm
pl< T, TState >(InitialState) | |
| 1203 { } | |
| 1204 | |
| 1205 typedef CBufferedAnimationImpl< T, TState > _baseBufferedAnimation; | |
| 1206 BEGIN_MSG_MAP(CBufferedAnimationWindowImpl) | |
| 1207 CHAIN_MSG_MAP(_baseBufferedAnimation) | |
| 1208 END_MSG_MAP() | |
| 1209 }; | |
| 1210 | |
| 1211 #endif // _WTL_NEW_UXTHEME | |
| 1212 | |
| 1213 }; // namespace WTL | |
| 1214 | |
| 1215 #endif // __ATLTHEME_H__ | |
| OLD | NEW |