让Windows 2000/XP中的任意窗口透明起来
简介
已经有很多的文章展示了如何通过使用新的系统函数在Windows 2000或Windows XP中建立透明窗口的应用程序,本文在此基础上为您展现了一种可以让任意应用程序窗口透明起来的方法,哪怕您根本没有那个应用程序的源程序。
使用本文介绍的“WinTrans”程序,您只需把程序中的“魔棒”(程序左上角的那个图标)拖曳到另一个正在运行的程序的标题栏上就可以使它的窗口变得透明。您还可以通过拖到滑动杆来调节透明度。“WinTrans”的界面很像SPY程序的界面,它演示了如何使用Win32 API函数来捕获一个位于鼠标光标下的窗口并获取它的窗口类、窗口标题等信息。
当您需要在一个最大化的窗口中工作而同时又需要查看另外一个在后台运行的程序的状态时,您会发现“WinTrans”程序是一个很实用的程序。
背景
在Windows 2000和Windows XP中,User32.dll中添加了一个新的函数,名字是SetLayeredWindowAttributes。如果要在应用程序中使用这个函数,需要在创建窗口时为窗口的window style设置WS_EX_LAYERED(0x00080000)位,也可以在创建窗口后用SetWindowLong函数添加这个位。一旦这个标志位被设定了,我们就可以通过调用SetLayeredWindowsAttributes函数,并把窗口的句柄传给它来使得窗口或窗口上特定的颜色变得透明。这个函数的参数如下:
HWND hWnd: 窗口的名柄
COLORREF col: 希望变透明的颜色
BYTE bAlpha: 如果这个值设为0,窗口会变得完全透明。如果设为255,窗口会变得完全不透明。
DWORD dwFlags: 如果这个标志设为1,只有col指定的颜色会变得透明。如果这个标志设为2,则整个窗口会按bAlpha指定的程度变得透明。
代码解释
首先,在WinTransDlg.h中的主对话框类中加下如下的成员变量。
bool m_bTracking; // 当鼠标被捕捉时会置为true
HWND m_hCurrWnd; // 鼠标最后一次指向的窗口的句柄
HCURSOR m_hCursor; // 魔棒光标
我们还定义了一个指向SetLayeredWindowAttributes函数的指针。这个函数位于User32.dll中。
// 全局定义
typedef BOOL (WINAPI *lpfn) (HWND hWnd, COLORREF cr,
BYTE bAlpha, DWORD dwFlags);
lpfn g_pSetLayeredWindowAttributes;
在OnInitDialog消息响应函数中,我们获取了SetLayeredWindowAttributes函数的地址并把它保存在g_pSetLayeredWindowAttributes变量中。同时,我们还加载了魔棒光标并把它的句柄放在m_hCursor中
BOOL CWinTransDlg::OnInitDialog()
{
....
// 获取User32.dll中SetLayeredWindowAttributes 函数的地址
HMODULE hUser32 = GetModuleHandle(_T("USER32.DLL"));
g_pSetLayeredWindowAttributes = (lpfn)GetProcAddress(hUser32,
"SetLayeredWindowAttributes");
if (g_pSetLayeredWindowAttributes == NULL)
AfxMessageBox (
"Layering is not supported in this version of Windows",
MB_ICONEXCLAMATION);
// 加载魔棒光标
HINSTANCE hInstResource = AfxFindResourceHandle(
MAKEINTRESOURCE(IDC_WAND), RT_GROUP_CURSOR);
m_hCursor = ::LoadCursor( hInstResource, MAKEINTRESOURCE(IDC_WAND) );
...
}
下面定义WM_LBUTTONDOWN, WM_LBUTTONUP和WM_MOUSEMOVE消息的响应函数。WM_LBUTTONDOWN的响应函数如下:
void CWinTransDlg::OnLButtonDown(UINT nFlags, CPoint point)
{
...
SetCapture(); // 使鼠标消息发回本窗口
m_hCurrWnd = NULL; // 当前没有窗口被设成透明的
m_bTracking = true; // 设置鼠标被捕获的标志
::SetCursor(m_hCursor); // 把鼠标光标变成魔棒光标
...
}
对于处理鼠标移动消息的响应代码如下
void CWinTransDlg::OnMouseMove(UINT nFlags, CPoint point)
{
...
if (m_bTracking)
{
...
// 把鼠标坐标转换为屏幕坐标
ClientToScreen(&point);
...
// 获取鼠标位置的光标
m_hCurrWnd = ::WindowFromPoint(point);
...
// 显示窗口的详细情况,包括窗口类、标题等
...
}
...
}
这样就使得只要在主窗口中按下鼠标按钮不松开就可以使鼠标光标变为魔棒光标,并且光标下的窗口的一些信息就会显示在WinTrans程序的对话框中。
当鼠标按钮松开时,WM_LBUTTONUP消息的响应函数就会被调用。
void CWinTransDlg::OnLButtonUp(UINT nFlags, CPoint point)
{
...
// 停止捕获鼠标
ReleaseCapture();
m_bTracking = false;
// 如果鼠标光标所指的窗口不是本应用程序本身
// 就设置它的Layer样式位并按滑动杆的设置值来设置窗口的Alpha值
if (g_pSetLayeredWindowAttributes && m_hCurrWnd != m_hWnd)
{
::SetWindowLong(m_hCurrWnd, GWL_EXSTYLE,
GetWindowLong(m_hCurrWnd,
GWL_EXSTYLE) ^ WS_EX_LAYERED);
g_pSetLayeredWindowAttributes(m_hCurrWnd, 0,
(BYTE)m_slider.GetPos(), LWA_ALPHA);
::RedrawWindow(m_hCurrWnd, NULL, NULL,
RDW_ERASE | RDW_INVALIDATE |
RDW_FRAME | RDW_ALLCHILDREN);
}
...
}
其它值得关注的地方
当前这个程序只有在魔棒指向窗口的标题栏或一个基于对话框的程序窗口体上才能有效的工作。举例来说:如果你把魔棒指向记事本的窗口体是没有用的。
如果要去掉窗口透明的效果,您只需再次把魔棒拖曳到相应的窗口上即可。因为在OnLButtonUp中,我们是切换WS_EX_LAYERED标志位的设置状态的,透明效果也会进行相应的切换。
TransWand程序不能在命令提示行窗口上正确的工作。
本文相关源程序下载:
原始下载地址(12K)(要在该网站注册后方可下载)
国内镜象(12K)