悬浮窗

 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 ControlType指定为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设为ResizingMaximize Box 和 Minimize Box设为True就可以了。话没错,照做运行。最大化没事,最小化呢,哎,我们辛辛苦苦做的悬浮窗怎么也不见了,主窗口还原,悬浮窗又出来了。哦原来是跟主窗口一起最小化了,这可不行,测试一下迅雷,发现主窗口最小化时悬浮窗还在,嗯,朋友又要动手了。先别忙动手,做到现在也可以说是一个阶段了,此时不要急于改程序,而应该先备份一下,在项目中更应该如此,客户可能会要求改来改去,多备份总是不会错的。

清一下垃圾文件,备份,然后给备份文件取一个有意义的名字。好了,备份完毕,开始下一步的思考:主窗口最小化时悬浮窗也会跟着最小化,如果主窗口不最小化就好了。哎,灵光闪现了,悬浮窗就是不能最小化呀,那好就把它当主窗口得了。不过有问题,这样任务栏中会显示悬浮窗,而真正的主窗口反而在任务栏中不显示了。怎么办?看一下《VBShowInTaskbar功能分析以及用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.hb.h 不能相互包含,如a.h中包含了b.h,而类b中又要用a类型,怎么办呢?可以在b.cpp中包含a.h

 

春风剑客 于2010-07-10完成

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值