转自:http://hi.baidu.com/kaileychina/item/19da7d21bd4120dea517b604
VC实现透明窗体
SetWindowLong(this->GetSafeHwnd(),GWL_EXSTYLE,
GetWindowLong(this->GetSafeHwnd(),GWL_EXSTYLE)^0x80000);
HINSTANCE hInst = LoadLibrary("User32.DLL");
if(hInst)
{
typedef BOOL (WINAPI *MYFUNC)(HWND,COLORREF,BYTE,DWORD);
MYFUNC fun = NULL;
fun=(MYFUNC)GetProcAddress(hInst, "SetLayeredWindowAttributes");
if(fun)
fun(this->GetSafeHwnd(),0,m_ValueLen,2);
FreeLibrary(hInst);
}
BOOL SetLayeredWindowAttributes(
HWND hwnd, // 应用目标窗口的句柄
COLORREF crKey, // 掩码的颜色,可以用RGB(r,g,b)来指定
BYTE bAlpha, // 掩码颜色部分的Alpha值,0是全透明,255是完全不透明
DWORD dwFlags // 透明方式
);
dwFlags的取值
LWA_COLORKEY(0x0001) //使crKye为透明值
LWA_ALPHA(0X0002) /使bAlpha为窗口的不透明值
要说名的是这个函数只在Windows2000及以上版本才支持。MSDN对要求的描述如下
<Requirements>
Windows NT/2000/XP: Included in Windows 2000 and later.
Windows 95/98/Me: Unsupported.
Header: Declared in Winuser.h; include Windows.h.
Library: Use User32.lib.
还有就是这个函数对于有标题框的窗体支持不好,就是它裁切的只是客户区域,好在我们要制作透明窗体的场合一般用不到标题框下面就说名例程的制作过程。(我旨在说明这种透明窗体的思路及函数的用法,所以代码非常简单,并且没有必要的错误验证机制,希望大家谅解)
建立一张用于在窗体上绘制的背景图片,把要裁切的部分用一种颜色标记出来,我们叫它MaskColor,我的图片如下:
我的MaskColor = 0xFF00,也就是 RGB(0,255,0).
建立一个基于对话框的工程,修改对话框资源的属性,主要修改两个地方。一是指定没有TitleBar,二是指定BorderStyle为None.这样才能保证出来的窗体符合你的要求
把图片加入资源,付ID = IDB_BACKGROUND
下面就开始写代码了,呵呵,看下面的代码这么长。是不是头有点大呀,别急,这些多半都是工程向导自动生成的,我加的都已经注解上了,并用黄色表示没有几行的。要不然我怎么敢向各位吹嘘这个实现方法简单呢。
a.首先我们给窗体添加两个成员变量:
CBitmap * m_oldBitmap; //指向内存DC原来的 Bitmap
CDC m_DC; //用于存放背景图片的内存DC
b.在窗体的OnInitDialog()函数中做一番初始化:
BOOL CTransWindowDlg::OnInitDialog()
{
CDialog::OnInitDialog();
// 设置此对话框的图标。当应用程序主窗口不是对话框时,框架将自动
// 执行此操作
SetIcon(m_hIcon, TRUE); // 设置大图标
SetIcon(m_hIcon, FALSE); // 设置小图标
///////////////////////////////////////
//实现背景图以及窗口透明
//调用背景图片
CBitmap bitmap;
BITMAP bitInfo;
bitmap.LoadBitmap(IDB_BACKGROUND);
//得到图片大小并调整窗口大小适应图片
bitmap.GetBitmap(&bitInfo);
CRect rect;
GetWindowRect(&rect);
rect.right = rect.left + bitInfo.bmWidth;
rect.bottom = rect.top + bitInfo.bmHeight;
MoveWindow(rect);
//创建并保存DC
m_DC.CreateCompatibleDC(GetDC());
m_oldBitmap = m_DC.SelectObject(&bitmap);
//设置窗口掩码颜色和模式
//首先获得掩码颜色
COLORREF maskColor = m_DC.GetPixel(0,0);
#define LWA_COLORKEY 0x00000001
#define WS_EX_LAYERED 0x00080000
typedef BOOL (WINAPI *lpfnSetLayeredWindowAttributes)(HWND hWnd,
COLORREF crKey,
BYTE bAlpha,
DWORD dwFlags);
lpfnSetLayeredWindowAttributes SetLayeredWindowAttributes;
HMODULE hUser32 = GetModuleHandle("user32.dll");
SetLayeredWindowAttributes = (lpfnSetLayeredWindowAttributes)GetProcAddress(hUser32,
"SetLayeredWindowAttributes");
SetWindowLong(GetSafeHwnd(),
GWL_EXSTYLE,
GetWindowLong(GetSafeHwnd(),
GWL_EXSTYLE) | WS_EX_LAYERED);
SetLayeredWindowAttributes(GetSafeHwnd(),
maskColor,
255,
LWA_COLORKEY);
FreeLibrary(hUser32);
////////////////////////////////////////
return TRUE; // 除非设置了控件的焦点,否则返回 TRUE
}
就像注释的那样,我们首先把图片Load进来,然后把m_DC创建一个与窗口DC兼容的DC,并把刚才Load进来的图片绑定到该内存DC上,并用m_oldBitmap 记录下原有Bitmap,用户最后释放。
c.向OnPaint中添加代码,用于把背景图片绘制到窗口上:
void CTransWindowDlg::OnPaint()
{
if (IsIconic())
{
//这里是MFC的框架代码,为了减少篇幅省略…
}
else
{
////////////////////////////////////
//用于绘制背景图片
//CDialog::OnPaint();
CDC * pDC = this->GetDC();
CRect rect;
GetWindowRect(&rect);
pDC->BitBlt(0,0,rect.Width(),rect.Height(),&m_DC,0,0,SRCCOPY);
////////////////////////////////////
}
}
d.到这里我们要的功能就已经能够实现了,但是好的程序员绝对不应该忘记释放资源,你也一样,一定没有忘记在程序结束时释放资源,呵呵,我们可以写到析构函数中,我们也可放到OnClose()函数中,都一样,我采用了后一种:添加函数并添加释放资源的代码
void CTransWindowDlg::OnClose()
{
////////////////////////////////////
//释放资源
CBitmap * bitmap = m_DC.SelectObject(m_oldBitmap);
m_DC.DeleteDC();
bitmap->DeleteObject();
////////////////////////////////////
CDialog::OnClose();
}
e.现在好了,运行一下吧,真爽,就这么几行代码搞定了一个漂亮的窗口。是不是很有成就感?呵呵。又看了几次真的美滋滋的。不好发现问题了,怎么程序运行的时候开始有一下闪烁呢?哦,是清空背景的时候画了一下,没关系,让我们干掉它。添加WM_EraseBkgnd事件的响应函数,把原来的注释掉直接返回True,再运行一下看看吧?怎么样,满意了吗?
BOOL CTransWindowDlg::OnEraseBkgnd(CDC* pDC)
{
////////////////////////////////////
//防止开始绘制的一下闪烁
//return CDialog::OnEraseBkgnd(pDC);
return true;
////////////////////////////////////
}
f.好了,我已经非常满意了,要休息一下了,但我怎么关闭这个窗口呢?糟糕,非要我用Alt+F4不成?算了再多用一下功,写个双击事件好了:
void CTransWindowDlg::OnLButtonDblClk(UINT nFlags, CPoint point)
{
///////////////////////////////////
//双击窗口关闭Windows
this->PostMessage(WM_CLOSE);
///////////////////////////////////
CDialog::OnLButtonDblClk(nFlags, point);
}
vc实现如何平滑地关闭窗口
技术准备:
很简单,就是使用 Windows API函数
SetLayeredWindowAttributes(HWND, COLORREF, BYTE,DWORD)
SetLayeredWindowAttributes函数在USER32.DLL中,你需要装载该DLL并使用它。
第一步:
你必须改变窗口的样式,将窗口变成具有图层样式,使用Windows API函数SetWindowLong。
在对话框窗体OnInitDialog()函数或者在文档/视类型的窗体OnCreate()函数如下使用:
SetWindowLong(m_hWnd,
GWL_EXTYLE,
::GetWindowLong(m_hWnd, GWL_EXSTYLE) | WS_EX_LAYERED);
然后调用函数:
SetTransparent( m_hWnd, 0, 255 , LWA_ALPHA );
让bAlpha的值为255,该函数如下:
// This function sets the transparency layered window
// by calling SetLayeredWindowAttributes API function.
BOOL SetTransparent(HWND hWnd, COLORREF crKey,
BYTE bAlpha, DWORD dwFlags)
{
BOOL bRet = TRUE;
typedef BOOL (WINAPI* lpfnSetTransparent)(HWND hWnd,
COLORREF crKey,
BYTE bAlpha,
DWORD dwFlags);
// Check that "USER32.dll" library has been
// loaded successfully...
if ( m_hUserDll )
{
lpfnSetTransparent pFnSetTransparent = NULL;
pFnSetTransparent =
(lpfnSetTransparent)GetProcAddress(m_hUserDll,
"SetLayeredWindowAttributes");
if (pFnSetTransparent )
bRet = pFnSetTransparent(hWnd, crKey, bAlpha, dwFlags);
else
bRet = FALSE;
} // if( m_hUserDll )
return bRet;
} // End of SetTransparent function
第二步:
在窗体OnClose()中调用CloseSmoothly()函数,该函数如下:
void CloseSmoothly()
{
// Increase transparencyone percent each time...
for(int nPercent=100; nPercent >= 0 ;nPercent--)
SetTransparent( m_hWnd, 0, 255 * nPercent/100, LWA_ALPHA);
}