在 Windows 應(yīng)用程序中使用 Windows XP 的外觀風(fēng)格與 PrintWindow(轉(zhuǎn))
發(fā)表時間:2023-08-01 來源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]注意:此示例僅適用于運(yùn)行 Windows XP 的計算機(jī)。打開示例并打開兩個或多個應(yīng)用程序窗口之后,請使用 AltTab 功能鍵來運(yùn)行 TaskSwitcher 應(yīng)用程序。摘要:本文介紹了一種增強(qiáng)的...
注意:此示例僅適用于運(yùn)行 Windows XP 的計算機(jī)。打開示例并打開兩個或多個應(yīng)用程序窗口之后,請使用 AltTab 功能鍵來運(yùn)行 TaskSwitcher 應(yīng)用程序。
摘要:本文介紹了一種增強(qiáng)的 AltTab 應(yīng)用程序 TaskSwitcher,并以此為框架演示了在 Windows 應(yīng)用程序中使用 Windows XP 新的外觀風(fēng)格和 PrintWindow 的方法。
目錄
簡介
TaskSwitcher 應(yīng)用程序
截取鍵盤輸入
枚舉頂層應(yīng)用程序窗口
顯示頂層應(yīng)用程序窗口
使用 Comctl32.dll 版本 6
總結(jié)
簡介
Microsoft® Windows® XP 引入了一種新的外觀風(fēng)格,它使用方便,并且用戶界面也更加豐富。例如,圓角窗口、更具質(zhì)感的任務(wù)欄以及將鼠標(biāo)懸停在 UI 元素上時,可實現(xiàn) UI 元素的熱跟蹤。
圖 1:新外觀風(fēng)格中的 Calculator(計算器)和 Display Properties(顯示屬性)對話框
Windows XP 還引入了新的打印 API:PrintWindow(英文)。該 API 允許調(diào)用者制作窗口的快照并將其插入設(shè)備環(huán)境。
有關(guān)外觀風(fēng)格以及將其應(yīng)用于應(yīng)用程序的介紹,請參閱 MSDN Library 中的技術(shù)文章“使用 Windows XP 的外觀風(fēng)格”。該文章提供了相關(guān)的概述和介紹信息,而本文的主要目的是提供一個使用外觀風(fēng)格 API 和 PrintWindow API 的實例。本文還為使用某些以前的 Win32 API 提供了一個刷新程序。
本文將特別闡述 TaskSwitcher 應(yīng)用程序,它與目前 Windows 中已有的 AltTab 機(jī)制具有相同的功能。除了顯示圖標(biāo)列表外,該應(yīng)用程序還將顯示將要切換到的應(yīng)用程序的縮略圖預(yù)覽。顯示應(yīng)用程序圖標(biāo)和預(yù)覽的容器窗口將通過外觀風(fēng)格 API 顯示出來,使應(yīng)用程序的外觀符合最終用戶當(dāng)前選擇的外觀風(fēng)格。
TaskSwitcher 應(yīng)用程序
TaskSwitcher 是為代替 Windows XP 的現(xiàn)有 AltTab 應(yīng)用程序切換機(jī)制而設(shè)計的。AltTab 是內(nèi)置的 Windows 超級用戶功能,它使最終用戶能夠在頂層應(yīng)用程序窗口之間進(jìn)行快速切換。當(dāng)按下熱鍵組合 Alt+Tab 時,Windows 會生成最終用戶正在使用的已打開窗口的列表。已打開窗口的列表將以一組圖標(biāo)的形式顯示,其中一個圖標(biāo)帶有矩形的選擇邊框。當(dāng)最終用戶繼續(xù)按住 Alt 鍵并按下 Tab 鍵時,矩形選擇框?qū)⒁浦料乱粋圖標(biāo)。釋放 Alt 鍵后,Windows 將把選定的圖標(biāo)所代表的應(yīng)用程序置于前臺。
圖 2:Windows XP AltTab 容器窗口
此功能在邏輯上可以分成三個部分:首先,應(yīng)用程序必須偵聽組合鍵 Alt+Tab;接收到該組合鍵時,應(yīng)用程序需要枚舉桌面上的頂層應(yīng)用程序窗口;最后,應(yīng)用程序需要在某種 UI 容器中顯示這些窗口,使用戶可以選擇要切換到的應(yīng)用程序的圖標(biāo)。
截取鍵盤輸入
使用 Win32 API,您可以通過幾種方法之一創(chuàng)建偵聽特定擊鍵的應(yīng)用程序。最簡單的方法是使用 API RegisterHotKey(英文)。該 API 包含一個 hwnd、一個 ID、一個虛擬鍵和一個組合鍵。如果此調(diào)用成功,則無論何時按下虛擬鍵和組合鍵,hwnd 的 WndProc 都會收到一個 WM_HOTKEY 消息,該消息的 wParam 等于 ID。無論偵聽?wèi)?yīng)用程序窗口是否處于活動狀態(tài),都是如此。無論何時按下 AltTab,下面的調(diào)用都會使 hwndApp 收到一條 WM_HOTKEY 消息:
RegisterHotKey(hwndApp, IDH_ALTTAB, MOD_ALT, VK_TAB)
在 Windows XP 之前,無法將 AltTab 注冊為熱鍵。在 Windows XP 中,您不僅可以成功地將 AltTab 注冊為熱鍵,而且 Windows XP 還使您可以自己處理該事件,而不用啟動其自身內(nèi)置的 AltTab 熱鍵處理程序。
// 創(chuàng)建一個偵聽熱鍵的虛擬窗口
HWND hwndApp = CreateWindow(WC_APP, NULL, 0, 0, 0, 0, 0, HWND_MESSAGE,
NULL, THIS_EXE, NULL);
if (hwnd)
{
// 注冊 Alt+Tab
RegisterHotKey(hwndApp, IDH_NEXT, MOD_ALT, VK_TAB);
RegisterHotKey(hwndApp, IDH_PREV, MOD_ALT MOD_SHIFT, VK_TAB);
MSG msg;
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
LRESULT WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
case WM_HOTKEY:
{
switch (wParam)
{
// 如果未顯示容器窗口,則枚舉
// 頂層窗口,提取圖標(biāo)和文本,
// 并將其顯示在容器窗口中
case IDH_NEXT:
{
// 在窗口層次結(jié)構(gòu)中選擇
// 下一個頂層窗口的圖標(biāo)
break;
}
case IDH_PREV:
{
// 在窗口層次結(jié)構(gòu)中選擇
// 上一個頂層窗口的圖標(biāo)
}
}
}
}
}
第二種實現(xiàn)鍵盤偵聽的更高級的方法是同時使用 API SetWindowsHookEx(英文)和 WH_KEYBOARD_LL。該方法在當(dāng)前桌面的全局范圍內(nèi)創(chuàng)建一個低級別的鍵盤掛鉤層。在調(diào)用 SetWindowsHookEx 時指定的 LowLevelKeyboardProc 回調(diào)函數(shù)將接收所有的鍵盤輸入。處理完鍵盤輸入后,LowLevelKeyboardProc 應(yīng)調(diào)用 CallNextHookEx 以使下一個掛鉤鏈(很可能是目標(biāo)應(yīng)用程序)能夠接收輸入。由于 LowLevelKeyboardProc 接收了所有的鍵盤事件,因此可以很容易地將其用作一個狀態(tài)機(jī),用于偵聽同時按下的 Alt 和 Tab 組合鍵。如果該應(yīng)用程序?qū)崿F(xiàn)它自己的 AltTab 機(jī)制,則此時將執(zhí)行窗口枚舉算法,并從 LowLevelKeyboardHook 中返回而不把最后的 AltTab 鍵事件轉(zhuǎn)給其他應(yīng)用程序。
hhook = SetWindowsHookEx(WH_KEYBOARD_LL, LowLevelKeyboardProc, hinst, 0);
LRESULT LowLevelKeyboardProc(INT nCode, WPARAM wParam, LPARAM lParam)
{
static BOOL fShiftPressed = FALSE;
BOOL fHandled = FALSE;
if (nCode == HC_ACTION)
{
KBDLLHOOKSTRUCT *pkbdllhook = (KBDLLHOOKSTRUCT *)lParam;
switch (wParam)
{
case WM_SYSKEYDOWN:
switch (pkbdllhook->vkCode)
{
case VK_LSHIFT:
case VK_RSHIFT:
{
// 用戶按下 Shift 鍵
fShiftPressed = TRUE;
break;
}
case VK_TAB:
{
if (pkbdllhook->flags & LLKHF_ALTDOWN)
{
// 用戶按下 Alt+Tab,執(zhí)行 AltTab 熱鍵處理程序
fHandled = TRUE;
}
break;
}
case VK_ESCAPE:
{
if (pkbdllhook->flags & LLKHF_ALTDOWN)
{
// 用戶按下 Esc 鍵,關(guān)閉 AltTab 容器窗口
// 并且不切換到選定窗口
fHandled = TRUE;
}
break;
}
}
break;
case WM_KEYUP:
case WM_SYSKEYUP:
switch (pkbdllhook->vkCode)
{
case VK_LMENU:
case VK_RMENU:
{
// 用戶釋放 Alt 鍵,關(guān)閉 AltTab 容器窗口
// 并切換到選定窗口
break;
}
case VK_LSHIFT:
case VK_RSHIFT:
{
// 用戶釋放 Shift 鍵
fShiftPressed = FALSE;
break;
}
}
break;
}
}
return (fHandled ? TRUE : CallNextHookEx(hhook, nCode, wParam, lParam));
}
枚舉頂層應(yīng)用程序窗口
使用 Win32 API EnumWindows(英文)可以直接枚舉頂層應(yīng)用程序窗口。這個 API 接受 EnumFunc 回調(diào)函數(shù)作為參數(shù)。對于桌面上的每個頂層窗口,系統(tǒng)將用頂層窗口的窗口句柄作為參數(shù),來調(diào)用 EnumFunc 函數(shù)。并不是所有的頂層窗口都應(yīng)顯示在 AltTab 列表中。需要查詢窗口的一些屬性,并且必須滿足幾個條件:窗口是應(yīng)用程序窗口嗎?窗口能被激活嗎?窗口可視嗎?窗口是 ToolWindow 嗎?
接收到 AltTab 事件之后,TaskSwitcher 便開始使用 EnumWindows 枚舉桌面上的頂層窗口。系統(tǒng)為每個頂層窗口調(diào)用回調(diào)函數(shù)。滿足條件的窗口將被添加到窗口列表中,并顯示在 AltTab 列表中。
顯示頂層應(yīng)用程序窗口
在 AltTab 列表的 UI 顯示中,TaskSwitcher 使用了很多 Windows XP 的新編程功能。它使用新的 API DrawShadowText 來顯示選定的應(yīng)用程序的文本,使用新的 API PrintWindow 來生成窗口預(yù)覽。最后,而且也許是應(yīng)用程序開發(fā)人員最感興趣的,TaskSwitcher 使用了 Windows XP 的新外觀風(fēng)格。
收集窗口信息
生成要在 AltTab 列表中顯示的窗口列表后,會檢索列表中每個窗口的各種屬性并將其顯示在預(yù)覽容器中。通過向該窗口發(fā)送一個 WM_GETICON(英文)窗口消息,在列表中顯示每個窗口的圖標(biāo)。當(dāng)用戶按 Tab 鍵在列表中移動時,列表中選定的應(yīng)用程序圖標(biāo)的圖標(biāo)和文本將顯示在預(yù)覽容器的頂部。通過使用 API GetWindowText(英文)來檢索每個窗口的標(biāo)題文本。有趣的是,它使用了新的 API comctl32 v6 API DrawShadowText(英文)來顯示應(yīng)用程序文本。該 API 是 Windows XP 的新增功能,采用了 API DrawText 的所有相同參數(shù),還有兩個表示文本顏色和陰影顏色的 COLORREF 參數(shù),以及陰影的 x 偏移量和 y 偏移量。
繪制窗口預(yù)覽
TaskSwitcher 還可以顯示選定窗口的縮略圖預(yù)覽。(除非最小化所預(yù)覽的窗口,因為此時將只顯示該窗口的標(biāo)題欄。)繪制縮略圖預(yù)覽時,TaskSwitcher 采用了一些高級的 Win32 繪圖技術(shù),例如雙重緩沖和半色調(diào)縮放等。然而,獲取窗口預(yù)覽的核心技術(shù)是新的 Windows XP user32 API PrintWindow。PrintWindow 帶有一個窗口句柄、一個 hdc 和一個保留標(biāo)志。該 API 使用窗口重定向,將窗口的快照繪制到 hdc 中。
// 制作窗口 hwnd 的快照,該窗口存儲在內(nèi)存設(shè)備環(huán)境 hdcMem 中
HDC hdc = GetWindowDC(hwnd);
if (hdc)
{
HDC hdcMem = CreateCompatibleDC(hdc);
if (hdcMem)
{
RECT rc;
GetWindowRect(hwnd, &rc);
HBITMAP hbitmap = CreateCompatibleBitmap(hdc, RECTWIDTH(rc), RECTHEIGHT(rc));
if (hbitmap)
{
SelectObject(hdcMem, hbitmap);
PrintWindow(hwnd, hdcMem, 0);
DeleteObject(hbitmap);
}
DeleteObject(hdcMem);
}
ReleaseDC(hwnd, hdc);
}
使用外觀風(fēng)格 API 來顯示容器
所有這些都繪制在容器窗口上。容器窗口的背景體現(xiàn)了 Windows XP 的新外觀風(fēng)格。也就是說,它和 Windows XP 的其余部分具有相同的外觀,包括圓角窗口以及與標(biāo)題欄類似的更具質(zhì)感的圖案背景。顯示容器背景時,TaskSwitcher 使用了 uxtheme.h 中的很多新的主題 API,例如 OpenThemeData(英文)、CloseThemeData(英文)、GetThemeBackgroundRegion(英文)和 DrawThemeBackground(英文)。在本例中,我們把用于開始面板頂部的外觀風(fēng)格用作容器窗口的背景。
#include <uxtheme.h>
#include <tmschema.h>
// AltTab 列表容器窗口的對話框程序
INT_PTR CALLBACK DlgProc(HWND hwnd, UINT uMsg, WPARAM, LPARAM lParam)
{
static HTHEME htheme = NULL;
switch (uMsg)
{
case WM_INITDIALOG:
{
htheme = OpenThemeData(hwnd, L"StartPanel");
if (htheme)
{
// 獲取要用于繪制容器窗口的
// 背景區(qū)域部分并將其應(yīng)用于
// 對話框。
HRGN hrgn = NULL;
GetWindowRect(hwnd, &rc);
OffsetRect(&rc, -rc.left, -rc.top);
if (SUCCEEDED(GetThemeBackgroundRegion(htheme, NULL,
SPP_USERPANE, 0, &rc, &hrgn)))
{
SetWindowRgn(hwnd, hrgn, FALSE);
}
}
break;
}
case WM_PAINT:
{
PAINTSTRUCT ps;
HDC hdc = BeginPaint(hwnd, &ps);
if (hdc)
{
if (htheme)
{
// 外觀風(fēng)格處于活動狀態(tài),使用外觀
// 風(fēng)格 API 進(jìn)行繪制。
RECT rc;
GetWindowRect(hwnd, &rc);
OffsetRect(&rc, -rc.left, -rc.top);
DrawThemeBackground(htheme, hdc, SPP_USERPANE, 0, &rc, NULL);
}
else
{
// 外觀風(fēng)格不處于活動狀態(tài),按傳統(tǒng)
// 窗口樣式進(jìn)行繪制。
}
}
EndPaint(hwnd, &ps);
break;
}
case WM_THEMECHANGED:
{
// 外觀風(fēng)格已更改,關(guān)閉現(xiàn)有的 htheme 并嘗試
// 打開一個新的 htheme。
if (htheme)
{
CloseThemeData(htheme);
}
htheme = OpenThemeData(hwnd, L"StartPanel");
break;
}
}
}
圖 3:TaskSwitcher AltTab 容器窗口
使用 Comctl32.dll 版本 6
Taskswitcher 利用了 comctl32.dll 版本 6 中的一些新功能。例如,圖標(biāo)列表使用 ListView 控件制作;容器的背景使用了匹配的背景水印,從而達(dá)到與窗口其余部分的自然融合。此外,API DrawShadowText 也是在 comctl32 v6 中找到的。
Comctl32 版本 6 是一個并行 DLL,即 comctl32.dll 版本 5 和版本 6 是同時安裝在系統(tǒng)上的。默認(rèn)情況下,當(dāng)應(yīng)用程序與 comctl32.lib 靜態(tài)鏈接時,將使用版本 5。為了使應(yīng)用程序能夠使用版本 6,必須提供一個如下的應(yīng)用程序聲明文件:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0">
<assemblyIdentity
version="1.0.0.0"
processorArchitecture="X86"
name="Microsoft.Shell.TaskSwitch "
type="win32"
/>
<description>TaskSwitcher:AltTab 替代程序。</description>
<dependency>
<dependentAssembly>
<assemblyIdentity
type="win32"
name="Microsoft.Windows.Common-Controls"
version="6.0.0.0"
processorArchitecture="X86"
publicKeyToken="6595b64144ccf1df"
language="*"
/>
</dependentAssembly>
</dependency>
</assembly>
然后通過在 .rc 文件中指定以下行,將該聲明文件編入應(yīng)用程序的資源部分。
CREATEPROCESS_MANIFEST_RESOURCE_ID RT_MANIFEST "TaskSwitch.exe.manifest"
總結(jié)
Windows XP 提供了一個全新的用戶界面,包括新的外觀風(fēng)格以及能夠直觀捕獲窗口內(nèi)容的能力。使用本文介紹的技術(shù),開發(fā)人員可以利用外觀風(fēng)格 API 為其應(yīng)用程序設(shè)計一個可以與 Windows XP 其余部分的外觀相匹配的獨(dú)特外觀。使用 PrintWindow,開發(fā)人員可以制作指定窗口的快照并將其插入設(shè)備環(huán)境。