VC对话框隐藏运行 (四)悬浮窗
Windows的菜单一层一层的,有时操作起来不方便,就想自己写个工具直接调用。对迅雷的悬浮窗垂涎好久了,哈,正好趁这个机会搞一搞。在悬浮窗上用快捷菜单不是很方便吗?
在VC知识库中找到了一篇介绍悬浮窗实现的例子,很好,拿来就用,又做了些增强功能,呵呵,特此Show一下。知识库文章地址:点这里。
要实现悬浮窗,就得弄明白悬浮窗是什么,有什么行为和作用。悬浮窗其实是一个无边框可以拖动的窗口,再有就是它上面铺满一张图,当然也可以是空白,可以设置透明度,双击可实现主窗口的显示和隐藏,右键有菜单,可以实现快捷操作;还有一点就是它在任务栏中不显示。好了,明白了原理就开始找资料动手了。
我的习惯是先攻克所有的技术点,然后再动手写程序。在资源文件中插入一个Dialog,给它添加一个类CFloatWnd。首先是无边框,这个容易,将窗口的Border设为None就可以了。无边框窗口的拖动也很容易,重写OnNcHitTest就可以了。原理就是当用户点击了窗口客户区后,把它当成标题栏来传给操作系统,这样就实现了拖动。代码如下:
LRESULT CFloatWnd::OnNcHitTest(CPoint point)
{
// TODO: 在此添加消息处理程序代码和/或调用默认值
LRESULT nHitTest = CDialog::OnNcHitTest(point);
if (nHitTest == HTCLIENT && GetAsyncKeyState(MK_LBUTTON) < 0 )//如果是客户区//如果鼠标左键按下,GetAsyncKeyState函数的返回值小于0
nHitTest = HTCAPTION;//则把它当成标题栏
return nHitTest;
//return CDialog::OnNcHitTest(point);
}
不要认为GetAsyncKeyState没有用,这样做的目的是点击左键拖动,而点击右键则不能拖动且要弹出右键菜单。
好了,现在窗口能拖动了,但是是空白窗口,不好看。那好,加个位图,首先添加资源文件Logo.bmp,资源ID设为IDB_BITMAP_Logo,然后在窗口上放个Picture Control,Type指定为Bitmap,image设为IDB_BITMAP_Logo就可以了。至于图片位置和窗口大小就写程序控制好了。给图片添加变量m_Logo。在OnInitDialog中添加如下代码:
// 让窗体和Logo相符
//mfc里的Attach和Detach函数主要用于对mfc类对象跟sdk句柄
//之间关联的处理,attach将mfc类对象与句柄关联起来
//对该对象的操作都会施加在相应的句柄身上
//detach则消除这种关联关系
CBitmap m_Bitmap;
HBITMAP hBitmap = m_Logo.GetBitmap();
ASSERT(hBitmap);
m_Bitmap.Attach(hBitmap);//关联
BITMAP bmp;
m_Bitmap.GetBitmap(&bmp);
int nX = bmp.bmWidth;
int nY = bmp.bmHeight;
MoveWindow(0,0,nX,nY);//窗口移到左上角并设置宽高
m_Logo.MoveWindow(0,0,nX,nY);//图片移到窗口的左上角
CenterWindow();//窗口居中
::SetWindowPos(m_hWnd,HWND_TOPMOST,0,0,0,0,SWP_NOMOVE | SWP_NOSIZE);//窗口置顶
m_Bitmap.Detach();//释放关联
做完这些以后在主窗口CMainAppDlg中添加成员变量CFloatWnd m_FloatWnd; 然后在CMainAppDlg的OnInitDialog中添加如下代码,来显示我们的悬浮窗。
//显示
m_FloatWnd.Create(CFloatWnd::IDD,this);//注意父窗口为this
m_FloatWnd.ShowWindow(SW_SHOW);
还有,别忘记在MainAppDlg.h 中添加这个:#include "FloatWnd.h"
好了,现在运行。感觉上挺不错了,但是双击不能显示主窗口,也没有右键菜单。嗯,继续做吧。添加成员函数afx_msg void OnShowHide();//显示隐藏 具体代码如下:
//显示的时候隐藏,隐藏的时候显示
void CFloatWnd::OnShowHide()
{
CWnd *pParent = GetParent();//得到父窗口,即主窗口
ASSERT(pParent);
if(pParent->IsWindowVisible())//如果显示
pParent->ShowWindow(SW_HIDE);//则隐藏
else//否则显示
pParent->ShowWindow(SW_SHOW);
}
//在窗口双击事件中调用
void CFloatWnd::OnNcLButtonDblClk(UINT nHitTest, CPoint point)
{
// TODO: 在此添加消息处理程序代码和/或调用默认值
OnShowHide();
CDialog::OnNcLButtonDblClk(nHitTest, point);
}
运行一下,双击能够显示和隐藏,效果还是不错的。下一步添加右键菜单。在资源文件里添加个新菜单,ID设为IDR_MENU_FloatWnd,然后添加两个子菜单项:1、Caption为显示/隐藏主窗口(&S) ,ID为ID_ShowHide;2、Caption为退出(&E) ,ID为ID_Exit。给CFloatWnd添加菜单成员变量:CMenu m_Right;//右键菜单 在OnInitDialog中加载:
//加载右键菜单
m_Right.LoadMenu(IDR_MENU_FloatWnd);
//加载好了就要在OnRButtonUp中显示了:
//鼠标右键显示弹出菜单
void CFloatWnd::OnRButtonUp(UINT nFlags, CPoint point)
{
// TODO: 在此添加消息处理程序代码和/或调用默认值
CMenu *pSub = m_Right.GetSubMenu(0);//取第一个子菜单
ClientToScreen(&point);//转换坐标为窗口坐标
pSub->TrackPopupMenu(TPM_LEFTALIGN,point.x,point.y,this,NULL);//显示
CDialog::OnRButtonUp(nFlags, point);
}
//显示是显示了,可没有功能,嗯,那就做消息映射写代码喽:
ON_COMMAND(ID_Exit, OnExit)//右键退出
ON_COMMAND(ID_ShowHide, OnShowHide)//右键显示/隐藏
//OnShowHide的代码有了,下面是OnExit:
//退出
void CFloatWnd::OnExit()
{
CWnd *pParent = GetParent();
this->DestroyWindow();//销毁自己
pParent->DestroyWindow();//销毁主窗口
}
运行一下,效果很棒。但还没有透明度设置。
在主窗口上放一个Slider Control,给它添加变量m_Slider,最大值设为255,最小值设为0。然后重写窗口的OnHScroll事件,在事件中根据滑动条的值来设置窗口的透明度。关于窗口的透明度,在VC2005中可以直接用SetLayeredWindowAttributes来设置,在VC6中要用LoadLibrary("User32.DLL")来实现。好了,完整代码如下:
//在CMainAppDlg::OnInitDialog()中
//设定滑动条最大值和最小值
m_Slider.SetRangeMax(255,TRUE);
m_Slider.SetRangeMin(1,TRUE);
//鼠标单击时一次滚动个单位
m_Slider.SetPageSize(10);
//设置透明度函数
void CFloatWnd::OnUpdateTransparent(int iTransparent)
{
SetLayeredWindowAttributes(0,iTransparent,2);
//在VC6中请用如下代码
//HINSTANCE hInst = LoadLibrary("User32.DLL");
//if(hInst)
//{
//typedef BOOL (WINAPI *SLWA)(HWND,COLORREF,BYTE,DWORD);
//SLWA pFun = NULL;
////取得SetLayeredWindowAttributes函数指针
//pFun = (SLWA)GetProcAddress(hInst,"SetLayeredWindowAttributes");
//if(pFun)
//{
//pFun(m_hWnd,0,iTransparent,2);
//}
//FreeLibrary(hInst);
//}
}
//在CFloatWnd::OnInitDialog()中给窗体加入扩展属性,否则无法设置透明度
//加入WS_EX_LAYERED扩展属性
SetWindowLong(m_hWnd,GWL_EXSTYLE,GetWindowLong(this->GetSafeHwnd(),GWL_EXSTYLE)^0x80000);
//默认不透明
OnUpdateTransparent(255);
好了,运行一下效果不错,以上代码基本上全是VC知识库中文章的,呵呵,在此谢谢作者。不过能让悬浮窗记住透明度和位置就更好了。说干就干。这里先介绍几个函数。GetWindowRect能得到窗口所在位置和宽高。SetRegistryKey让自己的程序在注册表中占一席之地。GetProfileString和WriteProfileString可以对子项进行读和写了。嗯,现在思路也比较清晰。窗口加载的时候读配置,窗口关闭的时候保存配置。不过无模式对话框的关闭事件比较多,没找到统一的关闭事件,所以在WindowProc中进行监听多个事件。完整代码如下:
//在CMainAppApp::InitInstance()中
SetRegistryKey(_T("MainApp"));
//在CMainAppApp::InitInstance()中
//从配置文件读上次窗体位置
//StrTemp是全局变量CString
GetProfileString("MainApp","FWndLeft","-1",StrTemp.GetBuffer(50),50);
int left=atoi(StrTemp.GetBuffer());//CString转整数多字节中用atoi,宽字节中用_wtoi
GetProfileString("MainApp","FWndTop","-1",StrTemp.GetBuffer(50),50);
int top=atoi(StrTemp.GetBuffer());
//没有配置文件则默认右上方,屏幕黄金分割位置
if ( left==-1 || top==-1 )
{
left=GetSystemMetrics(SM_CXSCREEN)*0.618;
top=GetSystemMetrics(SM_CYSCREEN)*0.382;
}
MoveWindow(left,top,nX,nY);
//读取透明度
GetProfileString("MainApp","FWndTransP","255",StrTemp.GetBuffer(50),50);
int transP=atoi(StrTemp.GetBuffer());
OnUpdateTransparent(transP);
//保存窗口位置
void CFloatWnd::SaveWinPosition()
{
RECT rcWin;
this->GetWindowRect(&rcWin);
StrTemp.Format("%d",rcWin.left);
WriteProfileString("MainApp","FWndLeft",StrTemp);
StrTemp.Format("%d",rcWin.top);
WriteProfileString("MainApp","FWndTop",StrTemp);
}
//程序退出时记忆位置和透明度
BOOL CFloatWnd::OnWndMsg(UINT message, WPARAM wParam, LPARAM lParam, LRESULT* pResult)
{
// TODO: 在此添加专用代码和/或调用基类
switch(message)
{
case WM_CLOSE://窗口关闭
case WM_DESTROY://窗口销毁
case WM_ENDSESSION://程序退出
SaveWinPosition();
break;
}
return CDialog::OnWndMsg(message, wParam, lParam, pResult);
}
//在CMainAppDlg::OnHScroll中添加,滑块滚动时记忆透明度
//记忆透明度
StrTemp.Format("%d",m_Slider.GetPos());WriteProfileString("MainApp","FWndTransP",StrTemp);
//别忘了在主窗口初始化中给滑条赋值
//读取透明度
GetProfileString("MainApp","FWndTransP","255",StrTemp.GetBuffer(50),50);
int transP=atoi(StrTemp.GetBuffer());
m_Slider.SetPos(transP);
好了到此一切顺利,似乎很完美了。配置能记忆能读出。有朋友曾经说过:软件接近完美的时候,也是麻烦显现的时候。什么,你不相信,受打击了?没关系,咱们继续走。还能走,不是做完了吗?
现在的主窗口是对话框,不可以最大化和最小化。那么朋友说了,这个简单,窗口Border设为Resizing,Maximize Box 和 Minimize Box设为True就可以了。话没错,照做运行。最大化没事,最小化呢,哎,我们辛辛苦苦做的悬浮窗怎么也不见了,主窗口还原,悬浮窗又出来了。哦原来是跟主窗口一起最小化了,这可不行,测试一下迅雷,发现主窗口最小化时悬浮窗还在,嗯,朋友又要动手了。先别忙动手,做到现在也可以说是一个阶段了,此时不要急于改程序,而应该先备份一下,在项目中更应该如此,客户可能会要求改来改去,多备份总是不会错的。
清一下垃圾文件,备份,然后给备份文件取一个有意义的名字。好了,备份完毕,开始下一步的思考:主窗口最小化时悬浮窗也会跟着最小化,如果主窗口不最小化就好了。哎,灵光闪现了,悬浮窗就是不能最小化呀,那好就把它当主窗口得了。不过有问题,这样任务栏中会显示悬浮窗,而真正的主窗口反而在任务栏中不显示了。怎么办?看一下《VB的ShowInTaskbar功能分析以及用VC的实现》。哈,能解决了。顶级窗口,但加了WS_EX_TOOLWINDOW,并去掉WS_EX_APPWINDOW风格,那么Windows不会为它在任务栏上创建一个按钮。非顶级窗口,但有WS_EX_APPWINDOW风格,那么Windows将为它在任务栏上创建一个按钮,否则不会有相应的任务栏按钮。WS_EX_TOOLWINDOW风格就是工具窗口风格,像VS中的查找窗口一样,但我们的悬浮窗是无边框的,所以不用考虑。还有一点,在《自启动+只运行一次》中我为自启动加了个 “Hide”参数,现在就是揭开谜底。那就是迅雷在自启动的时候并不显示主窗口,而在手动启动的时候却显示主窗口,为了实现这个效果就在自启动那里加了个参数,以便区分是自启动还是手动,然后决定主窗口的显示与否。 那好万事俱备只欠动手,开始动手。
给CFloatWnd添加成员变量bool isShowMain;//启动时是否显示主窗口,然后给构造函数加参数CFloatWnd(bool isShowMain,CWnd* pParent = NULL); // 标准构造函数,将主窗口做个成员变量CMainAppDlg dlg;//主窗口。主窗口中的悬浮窗成员去掉,初始化中去掉显示悬浮窗的部分,对悬浮窗透明度设置的部分,操作对象改为父窗口。相反的,在悬浮窗中对主窗口的操作都成了对成员变量的操作。代码如下:
//重载悬浮窗构造函数
CFloatWnd::CFloatWnd(bool ShowMain,CWnd* pParent /*=NULL*/)
: CDialog(CFloatWnd::IDD, pParent)
{
isShowMain=ShowMain;
}
//在CMainAppApp::InitInstance()中
bool isShow=true;
int argc=0;
LPWSTR *argv=CommandLineToArgvW(GetCommandLineW(),&argc);
//利用wcsicmp就比较好, 它将字符串转换成小写字符串进行比较,这样就忽略了大小写的情况.
if( wcsicmp(argv[1],L"hide")==0)//判断是否显示主窗口
isShow=false;
CFloatWnd dlg(isShow);
m_pMainWnd = &dlg;
//主窗口为悬浮窗时
dlg.Create(CFloatWnd::IDD);//创建窗口
dlg.ShowWindow(SW_SHOW);//显示窗口
dlg.RunModalLoop();//模拟DoModal
//在CFloatWnd::OnInitDialog()中
//在任务栏中隐藏
ModifyStyleEx(WS_EX_APPWINDOW,WS_EX_TOOLWINDOW,SWP_FRAMECHANGED);
dlg.Create(CMainAppDlg::IDD,this); //创建主窗口
if(isShowMain)//根据参数来选择是否显示主窗口
dlg.ShowWindow(SW_SHOW);
//显示的时候隐藏,隐藏的时候显示
void CFloatWnd::OnShowHide()
{
if(dlg.IsWindowVisible())//主窗口如果显示
dlg.ShowWindow(SW_HIDE);//则隐藏
else//否则显示
dlg.ShowWindow(SW_SHOW);
}
//退出
void CFloatWnd::OnExit()
{
dlg.DestroyWindow();//销毁主窗口
this->DestroyWindow();//销毁自己
}
//点击主窗口取消按钮时隐藏主窗口,而非关闭
void CMainAppDlg::OnBnClickedCancel()
{
// TODO: 在此添加控件通知处理程序代码
//用第二种方法要启用下面代码
//PostMessage(WM_QUIT,0,0);
this->ShowWindow(SW_HIDE);
//OnCancel();
}
开发环境:Visual Studio 2005+win2003。
《VC对话框隐藏运行》系列文章包括:(一) 隐藏运行的两种方法 (二)热键呼出 (三)自启动+只运行一次 (四)悬浮窗 。欢迎关注!
本站域名: http://www.our-code.com
本文地址: http://www.our-code.com/news/2010710/n376523.html
饮水思源,希望读者学习的同时多多支持本站!
本站原创,转载请注明出处。
附小技巧:
有四个文件 a.h a.cpp b.h b.cpp,其中a.h和b.h 不能相互包含,如a.h中包含了b.h,而类b中又要用a类型,怎么办呢?可以在b.cpp中包含a.h。
春风剑客 于2010-07-10完成