最近需要做一个主窗体常态隐藏的程序,类似360卫士那样,只有托盘图标常显示。本以为隐藏主窗体很简单,但遇到了意想不到的情况。
无效的做法
最初的想法是设置主对话框资源的 Visiable 属性为 false, 并在OnInitDialog函数里调用 ShowWindow(SW_HIDE) ,发现这些操作根本没有作用,对话框还是好好的显示在那里。开始还以为是改错了项目,或者当前启动项目设置错了,检查了一遍,确认没错,只好开始百度。
最初发现的解决方法是在OnInitDialog函数中执行以下代码:
SetWindowPos(&wndNoTopMost,,,,,SWP_HIDEWINDOW);
ModifyStyleEx(WS_EX_APPWINDOW,WS_EX_TOOLWINDOW);//移除任务栏图标显示
ShowWindow(SW_HIDE);
结果是窗体的大部分隐藏了,还剩了点没隐藏,就是下图这个要死不活的样子
既然还剩了一部分,那我就把对话框资源的 Caption 属性为空,SystemMenu属性设置为false,结果文字和关闭按钮确实都不见了,但还是剩下了一些东西,就是下图中的小长条
只能继续研究。
有效的做法
1 将界面像素置为0,移动界面至屏幕角落
int nFullWidth = GetSystemMetrics(SM_CXSCREEN);
int nFullHeight = GetSystemMetrics(SM_CYSCREEN);
SetWindowPos(NULL, nFullWidth, nFullHeight, , , SWP_NOZORDER); //设置0像素,移到最角落 或者:MoveWindow(0,0,0,0);
ShowWindow(SW_HIDE);
ModifyStyleEx(WS_EX_APPWINDOW, WS_EX_TOOLWINDOW); //移除任务栏图标显示
这种做法只是将上述的小长条移到了不易看到的地方,用户一旦操作界面的任一部位,窗口都会因为失活而不再显示,达到了隐藏界面的目的。但是这种做法明显属于凑合,而且一旦需要显示主界面,还需要将对话框的类型改回来,不能为工具窗口,否则任务栏不显示。
2 采用定时器,在窗口初始化完成后马上隐藏
在OnInitDialog中创建一个定时器SetTimer(1,1,NULL);
然后在OnTimer函数中调用ShowWindow(SW_HIDE)
这种做法的缺点就是窗口会闪烁一次,而且这种手法太野路子,不优雅。
3 响应WM_NCPAINT消息
在第一次处理WM_NCPAINT消息是,调用ShowWindow(SW_HIDE)方法,之所以只调用一次,是为了之后可以将主窗口再显示出来。
这种方法比较优雅,推荐采用
void CMFCApplication1Dlg::OnNcPaint()
{
// TODO: 在此处添加消息处理程序代码
static bool bNotPaint = true;
if (bNotPaint)
{
ShowWindow(SW_HIDE);
bNotPaint = false;
}
else
{
CDialogEx::OnNcPaint();
}
}
4 改变主窗体的创建方式
将 C***App::InitInstance() 函数中的代码
CMFCApplication1Dlg dlg;
m_pMainWnd = &dlg;
INT_PTR nResponse = dlg.DoModal();
替换为
CMFCApplication1Dlg dlg;
m_pMainWnd = &dlg;
//INT_PTR nResponse = dlg.DoModal();
INT_PTR nResponse = dlg.Create(CMFCApplication1Dlg::IDD);
dlg.ShowWindow(SW_HIDE);
dlg.RunModalLoop();
同样比较优雅,推荐使用