ClassXP.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727
  1. ////////////////////////////////////////////////////////////////////////////////////////////////////
  2. // 说明: ClassXP.c 文件
  3. // 更新: 2003-3-10
  4. ////////////////////////////////////////////////////////////////////////////////////////////////////
  5. ////////////////////////////////////////////////////////////////////////////////////////////////////
  6. // 编译预处理
  7. #if _WIN32_WINNT < 0x0400
  8. #define _WIN32_WINNT 0x0400
  9. #endif
  10. #include "ClassXP.h"
  11. #include <CommCtrl.H>
  12. #pragma warning(disable: 4311)
  13. #pragma warning(disable: 4312)
  14. #pragma comment(lib, "Msimg32.lib")
  15. // 导出函数
  16. #ifdef CXP_DLLMODE
  17. #pragma comment(linker, "/EXPORT:ClassXP=_ClassXP@8")
  18. #endif // CXP_DLLMODE
  19. // 强制使用 C 语言方式编译
  20. #ifdef __cplusplus
  21. extern "C"
  22. {
  23. #endif // __cplusplus
  24. ////////////////////////////////////////////////////////////////////////////////////////////////////
  25. // 全局变量
  26. HHOOK g_hPrevHookXP = NULL; // 窗口消息 HOOK 句柄
  27. PCLASSXP g_pClassXP = NULL; // 窗口的 CLASSXP 结构指针
  28. COLORREF g_crDialogbkColor = 0x00FFFFFF; // 窗口背景颜色
  29. #ifdef CXP_DLLMODE
  30. HINSTANCE g_hModuleXP = NULL; // 动态连接库模块句柄
  31. #endif // CXP_DLLMODE
  32. ////////////////////////////////////////////////////////////////////////////////////////////////////
  33. #ifdef CXP_DLLMODE
  34. ////////////////////////////////////////////////////////////////////////////////////////////////////
  35. // 动态连接库主函数
  36. BOOL WINAPI DllMain(HMODULE hModule, DWORD dwReason, LPVOID pvReserved)
  37. {
  38. if (dwReason == DLL_PROCESS_ATTACH)
  39. {
  40. g_hModuleXP = hModule;
  41. DisableThreadLibraryCalls(hModule);
  42. }
  43. return TRUE;
  44. }
  45. ////////////////////////////////////////////////////////////////////////////////////////////////////
  46. #endif // CXP_DLLMODE
  47. ////////////////////////////////////////////////////////////////////////////////////////////////////
  48. // 设置或取消窗口的 ClassXP 风格
  49. BOOL WINAPI ClassXP(HWND hWnd, BOOL bEnable)
  50. {
  51. BOOL bReturn;
  52. bReturn = FALSE;
  53. // 如果是影响当前进程中的所有窗口
  54. if (hWnd == NULL)
  55. {
  56. // 如果是取消当前进程中的所有窗口
  57. if ((bEnable == FALSE) && (g_hPrevHookXP != NULL))
  58. {
  59. // 枚举当前线程的窗口并取消 ClassXP 风格
  60. EnumThreadWindows(GetCurrentThreadId(), EnumWndProcXP, FALSE);
  61. // 取消窗口消息 HOOK
  62. bReturn = UnhookWindowsHookEx(g_hPrevHookXP);
  63. g_hPrevHookXP = NULL;
  64. }
  65. // 如果是设置当前进程中的所有窗口
  66. else if ((bEnable == TRUE) && (g_hPrevHookXP == NULL))
  67. {
  68. // 枚举当前线程中已存在的窗口并设置为 ClassXP 风格
  69. EnumThreadWindows(GetCurrentThreadId(), EnumWndProcXP, TRUE);
  70. // 安装窗口消息 HOOK
  71. g_hPrevHookXP = SetWindowsHookEx(WH_CALLWNDPROC, HookProcXP, 0, GetCurrentThreadId());
  72. bReturn = (BOOL) g_hPrevHookXP;
  73. }
  74. }
  75. else
  76. {
  77. // 如果是取消指定窗口的 ClassXP 风格
  78. if (bEnable == FALSE)
  79. bReturn = (BOOL) DeleteClassXP(hWnd);
  80. // 如果是设置指定窗口的 ClassXP 风格
  81. else
  82. bReturn = (BOOL) CreateClassXP(hWnd);
  83. }
  84. return bReturn;
  85. }
  86. ////////////////////////////////////////////////////////////////////////////////////////////////////
  87. ////////////////////////////////////////////////////////////////////////////////////////////////////
  88. // 创建并初始化 CLASSXP 数据结构;子类化窗口
  89. // 如果返回 NULL,表示没有创建;否则返回新创建节点的指针,同时 g_pClassXP 指向新创建的节点
  90. PCLASSXP WINAPI CreateClassXP(HWND hWnd)
  91. {
  92. LONG lStyle;
  93. DWORD dwType;
  94. PCLASSXP pCxp, pEditCxp;
  95. DWORD pID = NULL;
  96. HICON hIcon;
  97. HINSTANCE hInst;
  98. // 是否已经是 ClassXP 风格
  99. if (GetClassXP(hWnd) == NULL)
  100. {
  101. // 获取窗口类型,如果并判断是否能设置为 ClassXP 风格
  102. dwType = GetWindowTypeXP(hWnd);
  103. // if (dwType >= CXPT_PUSHBUTTON) && (dwType <= CXPT_DIALOG))
  104. if (dwType != CXPT_UNKNOWN)
  105. {
  106. lStyle = GetWindowLong(hWnd, GWL_STYLE);
  107. // 分配存储空间,增加一个节点
  108. pCxp = (PCLASSXP) HeapAlloc(GetProcessHeap(), 0, sizeof(CLASSXP));
  109. pCxp->pNext = g_pClassXP;
  110. g_pClassXP = pCxp;
  111. // 子类化窗口并初始化 CLASSXP 数据结构
  112. pCxp->hWnd = hWnd;
  113. pCxp->dwType = dwType;
  114. pCxp->dwState = (lStyle & WS_DISABLED) ? CXPS_DISABLED : 0;
  115. if (hWnd == GetFocus())
  116. pCxp->dwState |= CXPS_FOCUS;
  117. pCxp->wpPrev = (WNDPROC) SetWindowLong(hWnd, GWL_WNDPROC, (LONG) WindowProcXP);
  118. // 按窗口类型分别 CLASSXP 数据结构
  119. switch (dwType)
  120. {
  121. //case CXPT_STATEBOX:
  122. case CXPT_PUSHBUTTON:
  123. case CXPT_CHECKBOX:
  124. case CXPT_RADIOBOX:
  125. //ase CXPT_GROUPBOX:
  126. if ((lStyle & SS_TYPEMASK) == BS_DEFPUSHBUTTON)
  127. pCxp->dwState |= CXPS_DEFAULT;
  128. lStyle = (LONG) SendMessage(hWnd, BM_GETCHECK, 0, 0);
  129. if (lStyle == BST_CHECKED)
  130. pCxp->dwState |= CXPS_CHECKED;
  131. else if (lStyle == BST_INDETERMINATE)
  132. pCxp->dwState |= CXPS_INDETERMINATE;
  133. if (dwType == CXPT_PUSHBUTTON)//装入与按钮ID号相同的图标ID
  134. {
  135. pID =GetDlgCtrlID(pCxp->hWnd);
  136. hInst = (HINSTANCE)GetWindowLong(pCxp->hWnd, GWL_HINSTANCE);
  137. hIcon = (HICON)LoadIcon(hInst, MAKEINTRESOURCE(pID));
  138. if (!hIcon) break;
  139. SendMessage(pCxp->hWnd, WM_SETICON, (WPARAM)ICON_SMALL, (LPARAM)hIcon);
  140. }
  141. break;
  142. // case CXPT_LISTBOX:
  143. //pCxp->pData = (void *) HeapAlloc(GetProcessHeap(), 0, sizeof(SCROLLXP));
  144. // break;
  145. case CXPT_EDITBOX:
  146. if (lStyle & ES_READONLY)
  147. pCxp->dwState |= CXPS_READONLY;
  148. break;
  149. case CXPT_DIALOG: //生成DIALOG扩展数据
  150. pCxp->pData = (void *) HeapAlloc(GetProcessHeap(), 0, sizeof(DIALOGXP));
  151. pCxp->hRgn = NULL;
  152. break;
  153. case CXPT_SCROLLBOX: //生成SCROLL扩展数据
  154. pCxp->pData = (void *) HeapAlloc(GetProcessHeap(), 0, sizeof(SCROLLXP));
  155. break;
  156. case CXPT_SPINBOX:
  157. if (lStyle & UDS_HORZ)
  158. pCxp->dwState |= CXPS_HORIZON;
  159. if (GetWindowTypeXP((HWND) SendMessage(hWnd, UDM_GETBUDDY, 0, 0)) == CXPT_EDITBOX)
  160. {
  161. pEditCxp = GetClassXP((HWND) SendMessage(hWnd, UDM_GETBUDDY, 0, 0));
  162. if (pEditCxp)
  163. {
  164. if (lStyle & UDS_ALIGNLEFT)
  165. {
  166. pCxp->dwState |= CXPU_ALIGNLEFT;
  167. pEditCxp->dwState |= CXPE_LEFTSPIN;
  168. }
  169. else if (lStyle & UDS_ALIGNRIGHT)
  170. {
  171. pCxp->dwState |= CXPU_ALIGNRIGHT;
  172. pEditCxp->dwState |= CXPE_RIGHTSPIN;
  173. }
  174. }
  175. }
  176. break;
  177. }
  178. // 重画窗口
  179. RedrawWindow(hWnd, NULL, NULL,
  180. RDW_UPDATENOW|RDW_ERASE | RDW_FRAME | RDW_INVALIDATE | RDW_ERASENOW);
  181. return pCxp;
  182. }
  183. }
  184. return NULL;
  185. }
  186. ////////////////////////////////////////////////////////////////////////////////////////////////////
  187. ////////////////////////////////////////////////////////////////////////////////////////////////////
  188. // 取消窗口子类化;销毁窗口的 CLASSXP 数据结构
  189. // 如果返回值不为 NULL 表示成功删除,返回值为指向上一个节点指针;
  190. // 如果返回 NULL 且 g_pClassXP 为 NULL,表全部节点被删除;
  191. // 否则表示没有找到该节点。
  192. // 致谢: 感谢 dREAMtHEATER 改进此函数!
  193. PCLASSXP WINAPI DeleteClassXP(HWND hWnd)
  194. {
  195. PCLASSXP pDel;
  196. PCLASSXP pCxp;
  197. // 获取待删除的节点指针
  198. pDel = GetClassXP(hWnd);
  199. if (pDel != NULL)
  200. {
  201. // 如果待删除的节点就是 g_pClassXP 节点
  202. if (pDel == g_pClassXP)
  203. pCxp = g_pClassXP = pDel->pNext;
  204. else
  205. {
  206. // 循环查找待删除节点的上一个节点
  207. for (pCxp = g_pClassXP; pCxp != NULL; pCxp = pCxp->pNext)
  208. {
  209. // 如果找到
  210. if (pCxp->pNext == pDel)
  211. {
  212. // 使链表跳过待删除的节点
  213. pCxp->pNext = pDel->pNext;
  214. break;
  215. }
  216. }
  217. }
  218. // 取消窗口子类化并重画窗口
  219. SetWindowLong(hWnd, GWL_WNDPROC, (LONG) pDel->wpPrev);
  220. //副加数据
  221. if (pDel->pData)
  222. HeapFree(GetProcessHeap(), 0, pDel->pData);
  223. //窗口区域
  224. if (pDel->hRgn) DeleteObject(pDel->hRgn);
  225. // 删除堆内存
  226. HeapFree(GetProcessHeap(), 0, pDel);
  227. // 重画窗口
  228. RedrawWindow(hWnd, NULL, NULL,
  229. RDW_ERASE | RDW_FRAME | RDW_INVALIDATE | RDW_ERASENOW | RDW_UPDATENOW);
  230. return pCxp;
  231. }
  232. return NULL;
  233. }
  234. ////////////////////////////////////////////////////////////////////////////////////////////////////
  235. ////////////////////////////////////////////////////////////////////////////////////////////////////
  236. // 获取窗口的 CLASSXP 数据结构
  237. // 如果返回 NULL,表示没有找到;否则返回节点的指针
  238. PCLASSXP WINAPI GetClassXP(HWND hWnd)
  239. {
  240. PCLASSXP pCxp;
  241. for (pCxp = g_pClassXP; pCxp != NULL; pCxp = pCxp->pNext)
  242. {
  243. if (pCxp->hWnd == hWnd)
  244. return pCxp;
  245. }
  246. return FALSE;
  247. }
  248. ////////////////////////////////////////////////////////////////////////////////////////////////////
  249. ////////////////////////////////////////////////////////////////////////////////////////////////////
  250. // 获取窗口类型
  251. DWORD WINAPI GetWindowTypeXP(HWND hWnd)
  252. {
  253. DWORD lReturn, dwStyle;
  254. char szTemp[MAX_PATH];
  255. static char s_szClass[][32] =
  256. {
  257. "Button", // 按钮类
  258. "Edit", // 编辑框类
  259. "ComboBox", // 组合框类
  260. "ListBox", //NEW:列表框
  261. "ScrollBar",
  262. "Static",
  263. "#32770",
  264. "msctls_hotkey32", // 热键
  265. "SysIPAddress32", // IP 地址
  266. "SysListView32", // 列表查看
  267. "SysTreeView32", // 树形查看
  268. "SysDateTimePick32", // 日期/时间选择
  269. "msctls_updown32", // 旋转
  270. "SysMonthCal32", // 月历
  271. "SysTabControl32", // TAB 控件
  272. "msctls_progress32", // 进度条
  273. "msctls_trackbar32", // 追踪条
  274. };
  275. // 查找判断匹配的类名称
  276. GetClassName(hWnd, szTemp, sizeof(szTemp));
  277. for (lReturn = 0; lReturn < (sizeof(s_szClass) / sizeof(s_szClass[0])); lReturn++)
  278. if (lstrcmpi(szTemp, s_szClass[lReturn]) == 0)
  279. break;
  280. /*
  281. if ((GetWindowLong(hWnd, GWL_STYLE) && SS_TYPEMASK) == BS_OWNERDRAW)
  282. return CXPT_UNKNOWN; //自绘按钮
  283. */
  284. dwStyle = GetWindowLong(hWnd, GWL_STYLE);
  285. switch (lReturn)
  286. {
  287. case 0:
  288. //lReturn = GetWindowLong(hWnd, GWL_STYLE);
  289. switch (dwStyle & SS_TYPEMASK)
  290. {
  291. case BS_OWNERDRAW: //自绘按钮
  292. lReturn = CXPT_UNKNOWN;
  293. break;
  294. case BS_DEFPUSHBUTTON: // 默认按钮
  295. case BS_PUSHBUTTON: // 普通按钮
  296. lReturn = CXPT_PUSHBUTTON;
  297. break;
  298. case BS_CHECKBOX: // 复选框
  299. case BS_AUTOCHECKBOX: // 自动复选框
  300. case BS_3STATE: // 三状态复选框
  301. case BS_AUTO3STATE: // 自动三状态复选框
  302. if (dwStyle & BS_PUSHLIKE)
  303. lReturn = CXPT_PUSHBUTTON;
  304. else
  305. lReturn = CXPT_CHECKBOX;
  306. break;
  307. case BS_RADIOBUTTON: // 单选框
  308. case BS_AUTORADIOBUTTON: // 自动单选框
  309. if (dwStyle & BS_PUSHLIKE)
  310. lReturn = CXPT_PUSHBUTTON;
  311. else
  312. lReturn = CXPT_RADIOBOX;
  313. break;
  314. case BS_GROUPBOX: // 分组
  315. lReturn = CXPT_GROUPBOX;
  316. break;
  317. default: // 未知类型
  318. lReturn = CXPT_UNKNOWN;
  319. }
  320. break;
  321. case 2: // 组合框
  322. case 11:
  323. case 13:
  324. if ((dwStyle & 0x00000003) == CBS_SIMPLE)
  325. lReturn = CXPT_UNKNOWN;
  326. else
  327. lReturn = CXPT_COMBOBOX;
  328. break;
  329. case 3: //列表框
  330. lReturn = CXPT_LISTBOX;
  331. break;
  332. case 4:
  333. lReturn = CXPT_SCROLLBOX;
  334. // lReturn = CXPT_UNKNOWN;
  335. break;
  336. case 5:
  337. // if((lReturn & SS_BITMAP) && !(lReturn & SS_ICON)) //BMP图象不处理
  338. if((dwStyle & SS_BITMAP)) //BMP图象不处理
  339. return CXPT_UNKNOWN;
  340. lReturn = CXPT_STATEBOX;
  341. break;
  342. case 6:
  343. lReturn = CXPT_DIALOG;
  344. break;
  345. case 12:
  346. lReturn = CXPT_SPINBOX;
  347. break;
  348. case 1: // 编辑框
  349. case 7:
  350. case 8:
  351. case 9:
  352. case 10:
  353. if (dwStyle & WS_BORDER)
  354. lReturn = CXPT_EDITBOX;
  355. else
  356. {
  357. dwStyle = GetWindowLong(hWnd, GWL_EXSTYLE);
  358. if ((dwStyle & WS_EX_CLIENTEDGE) || (dwStyle & WS_EX_STATICEDGE))
  359. lReturn = CXPT_EDITBOX;
  360. else
  361. lReturn = CXPT_UNKNOWN;
  362. }
  363. break;
  364. /*
  365. #ifdef CXP_DLLMODE
  366. // VB 和 VCL 的控件,只有在动态连接库方式下才有可能出现这种情况
  367. case 7:
  368. case 8:
  369. case 9:
  370. lReturn = CXPT_PUSHBUTTON;
  371. break;
  372. case 10:
  373. case 11:
  374. case 12:
  375. lReturn = CXPT_CHECKBOX;
  376. break;
  377. case 13:
  378. case 14:
  379. case 15:
  380. case 16:
  381. lReturn = CXPT_EDITBOX;
  382. break;
  383. case 17:
  384. case 18:
  385. case 19:
  386. lReturn = CXPT_COMBOBOX;
  387. break;
  388. #endif // CXP_DLLMODE
  389. */
  390. default: // 未知类型
  391. lReturn = CXPT_UNKNOWN;
  392. }
  393. return lReturn;
  394. }
  395. ////////////////////////////////////////////////////////////////////////////////////////////////////
  396. ////////////////////////////////////////////////////////////////////////////////////////////////////
  397. // 获取窗口的内存兼容设备场景
  398. HDC WINAPI GetMemDCXP(LPMEMDCXP pMdcxp)
  399. {
  400. RECT Rect;
  401. GetWindowRect(pMdcxp->hWnd, &Rect);
  402. // 创建兼容内存兼容设备场景并设置为透明模式
  403. pMdcxp->hDC = GetWindowDC(pMdcxp->hWnd);
  404. pMdcxp->hMemDC = CreateCompatibleDC(pMdcxp->hDC);;
  405. SetBkMode(pMdcxp->hMemDC, TRANSPARENT);
  406. // 是否已经指定了位图句柄
  407. if (pMdcxp->hBitmap)
  408. {
  409. // 选择位图对象
  410. SelectObject(pMdcxp->hMemDC, pMdcxp->hBitmap);
  411. }
  412. else
  413. {
  414. // 创建并选择位图对象
  415. pMdcxp->hBitmap = (HBITMAP) SelectObject(pMdcxp->hMemDC,
  416. CreateCompatibleBitmap(pMdcxp->hDC, Rect.right - Rect.left, Rect.bottom - Rect.top));
  417. }
  418. // 如果要传送数据
  419. if (pMdcxp->bTransfer == TRUE)
  420. {
  421. BitBlt(pMdcxp->hMemDC, 0 , 0,
  422. Rect.right - Rect.left, Rect.bottom - Rect.top, pMdcxp->hDC, 0 , 0, SRCCOPY);
  423. }
  424. return pMdcxp->hMemDC;
  425. }
  426. ////////////////////////////////////////////////////////////////////////////////////////////////////
  427. ////////////////////////////////////////////////////////////////////////////////////////////////////
  428. // 获取窗口的内存兼容设备场景
  429. VOID WINAPI ReleaseMemDCXP(LPMEMDCXP pMdcxp)
  430. {
  431. RECT Rect;
  432. // 如果要传送数据
  433. if (pMdcxp->bTransfer == TRUE)
  434. {
  435. GetWindowRect(pMdcxp->hWnd, &Rect);
  436. BitBlt(pMdcxp->hDC, 0 , 0,
  437. Rect.right - Rect.left, Rect.bottom - Rect.top, pMdcxp->hMemDC, 0 ,0, SRCCOPY);
  438. }
  439. if (pMdcxp->hBitmap)
  440. DeleteObject(SelectObject(pMdcxp->hMemDC, pMdcxp->hBitmap));
  441. DeleteDC(pMdcxp->hMemDC);
  442. ReleaseDC(pMdcxp->hWnd, pMdcxp->hDC);
  443. }
  444. ////////////////////////////////////////////////////////////////////////////////////////////////////
  445. //平滑绘图
  446. //void CXPDialog::GradientFill(HDC hdc, const CRect &rt, COLORREF crColor[], int fillType)
  447. VOID WINAPI GradientRectXP2(HDC hdc, const RECT &rt,COLORREF crColor[], int fillType)
  448. {
  449. TRIVERTEX vert[4] ;
  450. GRADIENT_RECT gRect;
  451. vert[0].x = rt.left;
  452. vert[0].y = rt.top;
  453. vert[0].Red = (GetRValue(crColor[0]))<<8;
  454. vert[0].Green = (GetGValue(crColor[0]))<<8;
  455. vert[0].Blue = (GetBValue(crColor[0]))<<8;
  456. vert[0].Alpha = 0x0000;
  457. vert[1].Red = (GetRValue(crColor[1]))<<8;
  458. vert[1].Green = (GetGValue(crColor[1]))<<8;
  459. vert[1].Blue = (GetBValue(crColor[1]))<<8;
  460. vert[1].Alpha = 0x0000;
  461. if ( fillType != GRADIENT_FILL_TRIANGLE )
  462. {
  463. vert[1].x = rt.right;
  464. vert[1].y = rt.bottom;
  465. gRect.UpperLeft = 0;
  466. gRect.LowerRight = 1;
  467. ::GradientFill(hdc,vert,2,&gRect,1,fillType);
  468. return ;
  469. }
  470. vert[1].x = rt.right;
  471. vert[1].y = rt.top;
  472. vert[2].x = rt.left;
  473. vert[2].y = rt.bottom;
  474. vert[2].Red = (GetRValue(crColor[2]))<<8;
  475. vert[2].Green = (GetGValue(crColor[2]))<<8;
  476. vert[2].Blue = (GetBValue(crColor[2]))<<8;
  477. vert[2].Alpha = 0x0000;
  478. vert[3].x = rt.right;
  479. vert[3].y = rt.bottom;
  480. vert[3].Red = (GetRValue(crColor[3]))<<8;
  481. vert[3].Green = (GetGValue(crColor[3]))<<8;
  482. vert[3].Blue = (GetBValue(crColor[3]))<<8;
  483. vert[3].Alpha = 0x0000;
  484. GRADIENT_TRIANGLE GTrg;
  485. GTrg.Vertex1=0;
  486. GTrg.Vertex2=1;
  487. GTrg.Vertex3=2;
  488. ::GradientFill(hdc,vert,4,&GTrg,1,fillType);
  489. GTrg.Vertex1=1;
  490. GTrg.Vertex2=2;
  491. GTrg.Vertex3=3;
  492. ::GradientFill(hdc,vert,4,&GTrg,1,fillType);
  493. }
  494. ////////////////////////////////////////////////////////////////////////////////////////////////////
  495. // 绘制渐变矩形
  496. VOID WINAPI GradientRectXP(HDC hDC, LPRECT pRect,COLORREF crColor[4])
  497. {
  498. int i;
  499. TRIVERTEX Tve[4];
  500. GRADIENT_RECT GRect;
  501. GRADIENT_TRIANGLE GTrg;
  502. for (i = 0; i < 4; i++)
  503. {
  504. Tve[i].Red = ((COLOR16) GetRValue(crColor[i])) << 8;
  505. Tve[i].Green = ((COLOR16) GetGValue(crColor[i])) << 8;
  506. Tve[i].Blue = ((COLOR16) GetBValue(crColor[i])) << 8;
  507. Tve[i].Alpha = ((COLOR16) (crColor[i] >> 24)) << 8;
  508. }
  509. Tve[0].x = pRect->left;
  510. Tve[0].y = pRect->top;
  511. Tve[1].x = pRect->right;
  512. Tve[1].y = pRect->top;
  513. Tve[2].x = pRect->left;
  514. Tve[2].y = pRect->bottom;
  515. Tve[3].x = pRect->right;
  516. Tve[3].y = pRect->bottom;
  517. if ((crColor[0] == crColor[2]) &&
  518. (crColor[1] == crColor[3]))
  519. i = GRADIENT_FILL_RECT_H;
  520. if ((crColor[0] == crColor[1]) &&
  521. (crColor[2] == crColor[3]))
  522. i = GRADIENT_FILL_RECT_V;
  523. else
  524. i = GRADIENT_FILL_TRIANGLE;
  525. if (i == GRADIENT_FILL_TRIANGLE)
  526. {
  527. GTrg.Vertex1 = 0;
  528. GTrg.Vertex2 = 1;
  529. GTrg.Vertex3 = 2;
  530. }
  531. else
  532. {
  533. GRect.UpperLeft = 0;
  534. GRect.LowerRight = 3;
  535. }
  536. GradientFill(hDC, Tve, 4,
  537. ((i == GRADIENT_FILL_TRIANGLE) ? ((PVOID) &GTrg) : ((PVOID) &GRect)), 1, i);
  538. if (i == GRADIENT_FILL_TRIANGLE)
  539. {
  540. GTrg.Vertex1 = 3;
  541. GradientFill(hDC,Tve, 4, &GTrg, 1, GRADIENT_FILL_TRIANGLE);
  542. }
  543. }
  544. ////////////////////////////////////////////////////////////////////////////////////////////////////
  545. ////////////////////////////////////////////////////////////////////////////////////////////////////
  546. // 绘制渐变矩形
  547. /*
  548. VOID WINAPI DrawDropGripXP(HDC hDC, LPRECT pRect, COLORREF crColor[4], DWORD crBack)
  549. {
  550. HANDLE hHandle;
  551. RECT Rect;
  552. Rect.left = pRect->left ;
  553. Rect.top = pRect->top;
  554. Rect.right = pRect->right ;
  555. Rect.bottom= pRect->bottom;
  556. static COLORREF s_crGradientXP[][4] =
  557. {
  558. {0x00F7F7F7, 0x00EFF3F7, 0x00EFF3F7, 0x00EFF3F7},
  559. {0x00DEC3BD, 0x00DEB6AD, 0x00FFE3DE, 0x00F7E3DE},
  560. {0x00EFDBCE, 0x00EFCFC6, 0x00E7CFC6, 0x00E7CBBD},
  561. {0x00FFEFE7, 0x00FFE7DE, 0x00FFE3DE, 0x00F7E3DE},
  562. };
  563. // 绘制外框
  564. hHandle = (HANDLE) CreateSolidBrush(0x00FFFFFF);
  565. FrameRect(hDC, &Rect, (HBRUSH) hHandle);
  566. // DeleteObject((HGDIOBJ) hHandle);
  567. // 绘制内框
  568. // InflateRect(&Rect, -1, -1);
  569. // Rect.left = Rect.right - GetSystemMetrics(SM_CYVTHUMB);
  570. FrameRect(hDC, &Rect, (HBRUSH) hHandle);
  571. // 绘制下拉外框
  572. GradientRectXP(hDC, pRect, crColor);
  573. // 绘制下拉外框拐角像素
  574. SetPixel(hDC, Rect.left, Rect.top, s_crGradientXP[0][0]);
  575. SetPixel(hDC, Rect.right - 1, Rect.top, s_crGradientXP[0][1]);
  576. SetPixel(hDC, Rect.left, Rect.bottom - 1, s_crGradientXP[0][2]);
  577. SetPixel(hDC, Rect.right - 1, Rect.bottom - 1, s_crGradientXP[0][3]);
  578. // 绘制下拉内框
  579. // InflateRect(&Rect, -1, -1);
  580. // GradientRectXP(Mdcxp.hMemDC, &Rect, s_crGradientXP[i + 8]);
  581. // DeleteObject(SelectObject(hDC, (HGDIOBJ) hHandle));
  582. }
  583. */
  584. ////////////////////////////////////////////////////////////////////////////////////////////////////
  585. ////////////////////////////////////////////////////////////////////////////////////////////////////
  586. // 枚举窗口回调函数
  587. BOOL CALLBACK EnumWndProcXP(HWND hWnd, LPARAM lParam)
  588. {
  589. // 如果是取消指定窗口的 ClassXP 风格
  590. if (lParam == FALSE)
  591. DeleteClassXP(hWnd);
  592. // 如果是设置指定窗口的 ClassXP 风格
  593. else
  594. CreateClassXP(hWnd);
  595. // 枚举子窗体
  596. EnumChildWindows(hWnd, EnumWndProcXP, lParam);
  597. return TRUE;
  598. }
  599. ////////////////////////////////////////////////////////////////////////////////////////////////////
  600. ////////////////////////////////////////////////////////////////////////////////////////////////////
  601. // 窗口消息 HOOK 回调函数
  602. LRESULT CALLBACK HookProcXP(int iCode, WPARAM wParam, LPARAM lParam)
  603. {
  604. // 设置新建的窗口为 ClassXP 风格
  605. if ((((CWPSTRUCT *) lParam)->message == WM_CREATE) && (iCode >= 0))
  606. CreateClassXP(((CWPSTRUCT *) lParam)->hwnd);
  607. return CallNextHookEx(g_hPrevHookXP, iCode, wParam, lParam);
  608. }
  609. ////////////////////////////////////////////////////////////////////////////////////////////////////
  610. ////////////////////////////////////////////////////////////////////////////////////////////////////
  611. // 窗口子类化回调函数
  612. LRESULT CALLBACK WindowProcXP(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
  613. {
  614. LONG lReturn;
  615. PCLASSXP pCxp;
  616. pCxp = GetClassXP(hWnd);
  617. switch (pCxp->dwType)
  618. {
  619. case CXPT_PUSHBUTTON: // 按钮
  620. case CXPT_CHECKBOX: // 复选框
  621. case CXPT_RADIOBOX: // 单选框
  622. // case CXPT_GROUPBOX: // 组合框
  623. // case CXPT_STATEBOX: // 说明条
  624. return ButtonWindowProc(pCxp, uMsg,wParam, lParam);
  625. case CXPT_EDITBOX: // 编辑框
  626. return EditWindowProc(pCxp, uMsg,wParam, lParam);
  627. case CXPT_COMBOBOX: // 组合框
  628. return ComboWindowProc(pCxp, uMsg,wParam, lParam);
  629. /* case CXPT_LISTBOX: // 列表框
  630. return ListWindowProc(pCxp, uMsg,wParam, lParam);
  631. case CXPT_SCROLLBOX: // 滚动条
  632. return ScrollWindowProc(pCxp, uMsg,wParam, lParam);
  633. case CXPT_DIALOG: // 弹出窗口
  634. return DlgWindowProc(pCxp, uMsg,wParam, lParam);
  635. case CXPT_SPINBOX: // 旋转条
  636. return SpinWindowProc(pCxp, uMsg,wParam, lParam);*/
  637. }
  638. // 调用原来的回调函数
  639. lReturn = (LONG) CallWindowProc(pCxp->wpPrev, hWnd, uMsg, wParam, lParam);
  640. return lReturn;
  641. }
  642. ////////////////////////////////////////////////////////////////////////////////////////////////////
  643. #ifdef __cplusplus
  644. }
  645. #endif // __cplusplus
  646. ////////////////////////////////////////////////////////////////////////////////////////////////////