Chinaunix首页 | 论坛 | 博客
  • 博客访问: 588681
  • 博文数量: 752
  • 博客积分: 40000
  • 博客等级: 大将
  • 技术积分: 5005
  • 用 户 组: 普通用户
  • 注册时间: 2008-10-13 14:47
文章分类

全部博文(752)

文章存档

2011年(1)

2008年(751)

我的朋友

分类:

2008-10-13 16:51:42

如何隐藏显示在任务栏中的对话框程序


作者:




    最近有个朋友做了一个基于对话框的小程序,大家知道,一般具有用户界面的 Windows 程序运行起来后,通常都会在任务栏里体现出来。我的这个朋友不想让她做的对话框程序运行的时候显示在任务栏里。问我如何隐藏?我参考了 MSDN 后告诉她说使用 WS_EX_TOOLWINDOW 扩展窗口式样。她按照我说的方法试了一下,结果没有成功。后来我琢磨了半天,发现这件事情并不像文档中说的那么简单。MSDN 里对 WS_EX_APPWINDOW 的描述是这样的:

      用 WS_EX_TOOLWINDOW 可以创建一个工具窗口,被作为浮动工具栏使用。工具窗口的标
      题栏比常规标题栏短,并且使用的窗口字体更小。工具窗口不会出现在任务栏里;当用户
      按下 ALT+TAB 健后,也不会出现在任务表中......
显然,按照上面的文档所讲的方法无法实现对话框的隐藏。那么答案在哪里?下面就让我将诀窍和技巧告诉你吧:

第一、创建对话框时必须将它作为某个不可见框架窗口的子窗口;
第二、这个不可见窗口的扩展式样必须设置 WS_EX_TOOLWINDOW;
第三、保证对话框的扩展式样没有设置 WS_EX_APPWINDOW;

下面是例子代码的实现细节说明,这个例子程序(HideDlg)很简单,头文件和实现文件都在同一个文件中:
////////////////////////////////////////////////////////////////
// HideDlg.cpp 声明部分
// 
////////////////////////////////////////////////////////////////

#include "stdafx.h"
#include "resource.h"
#include "statlink.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif

class CMainFrame : public CFrameWnd {
protected:
	CString m_sClassName;
	virtual BOOL PreCreateWindow(CREATESTRUCT& cs);
public:
	CMainFrame() { }
	~CMainFrame() { }
};

class CMyDlg : public CDialog {
public:
	CMyDlg(CWnd* pParent = NULL); // 标准构造函数
protected:
	HICON m_hIcon;
	CStaticLink	m_wndLink;

	virtual BOOL OnInitDialog();
	afx_msg void OnSysCommand(UINT nID, LPARAM lParam);
	afx_msg void OnPaint();
	afx_msg HCURSOR OnQueryDragIcon();
	DECLARE_MESSAGE_MAP()
};

class CMyApp : public CWinApp {
public:
	CMyApp();
	virtual BOOL InitInstance();
	DECLARE_MESSAGE_MAP()
};


//////////////////////////////////////////////////////
// HideDlg.cpp 实现部分
//
//////////////////////////////////////////////////////

// 创建不可见框架窗口:设置 WS_EX_TOOLWINDOW 式样
BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs)
{
	/*
	// 设置 WS_EX_TOOLWINDOW 扩展式样
	if (CFrameWnd::PreCreateWindow(cs)) {
		cs.dwExStyle |= WS_EX_TOOLWINDOW;
		return TRUE;
	}
	return FALSE;
*/
	// 不设置 WS_EX_TOOLWINDOW 扩展式样
	return CFrameWnd::PreCreateWindow(cs);
}

BEGIN_MESSAGE_MAP(CMyApp, CWinApp)
	ON_COMMAND(ID_HELP, CWinApp::OnHelp)
END_MESSAGE_MAP()

CMyApp::CMyApp()
{
}

CMyApp theApp;

////////////////////////////////////////////////////////////////////////
// InitInstance: 创建对话框时,把它作为不可见主框架窗口的子窗口对待
//
////////////////////////////////////////////////////////////////////////
BOOL CMyApp::InitInstance()
{
	CMainFrame* pFrame = new CMainFrame;
	m_pMainWnd = pFrame;
	pFrame->LoadFrame(IDR_MAINFRAME, WS_OVERLAPPED, NULL, NULL);
	CMyDlg dlg(pFrame);
	int nResponse = dlg.DoModal();
	if (nResponse == IDOK) {
	} else if (nResponse == IDCANCEL) {
	}
	return FALSE;
}

class CAboutDlg : public CDialog {
public:
	CAboutDlg();
	enum { IDD = IDD_ABOUTBOX };
protected:
	CStaticLink	m_wndLink1;
	CStaticLink	m_wndLink2;
	CStaticLink	m_wndLink3;

// Implementation
protected:
	//{{AFX_MSG(CAboutDlg)
	virtual BOOL OnInitDialog();
	//}}AFX_MSG
	DECLARE_MESSAGE_MAP()
};

CAboutDlg::CAboutDlg() : CDialog(CAboutDlg::IDD)
{
}

BEGIN_MESSAGE_MAP(CAboutDlg, CDialog)
END_MESSAGE_MAP()

BOOL CAboutDlg::OnInitDialog() 
{
	CDialog::OnInitDialog();
	
	m_wndLink1.m_link = _T("");
	m_wndLink2.m_link = _T("");
	m_wndLink3.m_link = _T("");
	m_wndLink1.SubclassDlgItem(IDC_STATIC_ICON, this);
	m_wndLink2.SubclassDlgItem(IDC_VCKBASE, this);
	m_wndLink3.SubclassDlgItem(IDB_STATIC_IMG, this);
	
	return TRUE;  // return TRUE unless you set the focus to a control
	              // EXCEPTION: OCX Property Pages should return FALSE
}

CMyDlg::CMyDlg(CWnd* pParent /*=NULL*/)
	: CDialog(IDD_HIDEDLG, pParent)
{
	m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

BEGIN_MESSAGE_MAP(CMyDlg, CDialog)
	ON_WM_SYSCOMMAND()
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
END_MESSAGE_MAP()

BOOL CMyDlg::OnInitDialog()
{
	CDialog::OnInitDialog();

	m_wndLink.m_link = _T("");
	m_wndLink.SubclassDlgItem(IDC_VCKBASE, this);

//  去掉注释设置对话框的 WS_EX_APPWINDOW 扩展式样
//	ModifyStyleEx(0,WS_EX_APPWINDOW);

	// IDM_ABOUTBOX must be in the system command range.
	ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
	ASSERT(IDM_ABOUTBOX < 0xF000);

	CMenu* pSysMenu = GetSystemMenu(FALSE);
	if (pSysMenu != NULL) {
		CString strAboutMenu;
		strAboutMenu.LoadString(IDS_ABOUTBOX);
		if (!strAboutMenu.IsEmpty()) {
			pSysMenu->AppendMenu(MF_SEPARATOR);
			pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
		}
	}

	// 设置对话框图标。
	// 当应用程序的主窗口不是对话框时,框架会自动设置图标。
	SetIcon(m_hIcon, TRUE);       // 设置大图标
	SetIcon(m_hIcon, FALSE);      // 设置小图标
	
	return TRUE;  // return TRUE  unless you set the focus to a control
}

void CMyDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
	if ((nID & 0xFFF0) == IDM_ABOUTBOX) {
		CAboutDlg dlgAbout;
		dlgAbout.DoModal();
	} else {
		CDialog::OnSysCommand(nID, lParam);
	}
}

// 如果在对话框上添加最小化按钮,必须用CMyDlg::OnPaint() 绘制图标,
// MFC 用文档/视图模型,由框架自动处理。 
void CMyDlg::OnPaint() 
{
	if (IsIconic()) {
		CPaintDC dc(this); // device context for painting

		SendMessage(WM_ICONERASEBKGND, (WPARAM) dc.GetSafeHdc(), 0);

		// Center icon in client rectangle
		int cxIcon = GetSystemMetrics(SM_CXICON);
		int cyIcon = GetSystemMetrics(SM_CYICON);
		CRect rect;
		GetClientRect(&rect);
		int x = (rect.Width() - cxIcon + 1) / 2;
		int y = (rect.Height() - cyIcon + 1) / 2;

		// Draw the icon
		dc.DrawIcon(x, y, m_hIcon);
	} else {
		CDialog::OnPaint();
	}
}

// 当用户要求最小化窗口时,系统调用此函数获取要显示的图标。 
HCURSOR CMyDlg::OnQueryDragIcon()
{
	return (HCURSOR) m_hIcon;
}
以上是主要的源代码清单, 下面对关键部分作一说明:

    在常规的 MFC 应用中,CMyApp::CInitInstance 的作用是加载框架窗口,但这里创建的窗口时不可见的。创建了主框架之后,我用它作为对话框的父窗口,即便它是不可见的,也应该在资源文件中给它一个菜单,否则 MFC 会很不爽。
    CMainFrame::PreCreateWindow 是在创建窗口之前由框架调用的函数,因此可以在这里设置窗口的扩展属性 WS_EX_TOOLWINDOW,这样就将它丛任务栏隐藏了,因为框架是不可见的,所以不用关心标题栏的大小问题。
此外,对话框的扩展式样必须关闭。可以设置这个扩展式样看看效果:

     ModifyStyleEx(0,WS_EX_APPWINDOW);
如果在代码中加上这行,那么对话框无法隐藏。这一点是一般想不到的诀窍,在默认的情况下,VC++ 的 IDE 在工程资源文件中会有这样一行式样描述:
      EXSTYLE WS_EX_APPWINDOW 
直接编辑.rc文件删除这行。
    从理论上讲,之所以要创建隐藏的父窗口(WS_EX_TOOLWINDOW),是因为必须让对话框具备常规标题的缘故;如果你愿意要小标题,完全可以不设置 WS_EX_TOOLWINDOW,但 WS_ EX_APPWINDOW 一定要关闭。我试了一下没问题。这说明 WS_EX_TOOLWINDOW对隐不隐藏对话框没用太大关系。这方面谁有更好的经验,欢迎来信交流。

最后祝大家身体健康!
--------------------next---------------------

我不知道你为什么不用ShowWindow来隐藏主窗口 ( BenjaminZ 发表于 2005-6-6 9:33:00)
 
-------------------------------------------------
站在各位的肩膀上,这是最简单的解决方法:
CDialog::OnInitDialog()
{
   ...
   SetParent(FindWindow("ProgMan",NULL));
   ShowWindow(SW_MINIMIZE);
   ...

}

说明:把当前应用的主窗口作为另一个隐藏窗口的子窗口,从而实现隐藏。
-------------------------------------------------

-------------------------------------------------
( proad 发表于 2004-10-21 11:04:00)
 
这是本文作者的误会,并非 MSDN 说错,只是所得不全而已。

选中 WS_EX_TOOLWINDOW 扩展窗口式样后,还要在 OnInitDlg() 中添加: 
    ShowWindow(SW_HIDE);
    ::SetWindowLong(GetSafeHwnd(), GWL_EXSTYLE, WS_EX_TOOLWINDOW); 
    ShowWindow(SW_SHOWNORMAL); ( WindsonZhL 发表于 2004-4-19 11:00:00)
 
( _foo 发表于 2004-4-14 8:24:00)
 
TOMG:在 CDialog::OnInitDialog() 里,
改成这样
ModifyStyleEx(WS_EX_APPWINDOW,WS_EX_TOOLWINDOW, SWP_DRAWFRAME);

标题栏就不会偏了. ( yy2better 发表于 2004-2-27 17:19:00)
 
在 CDialog::OnInitDialog() 里

只要加入:
CWnd *pwnd=FindWindow("ProgMan",NULL);
this->SetParent(pwnd);
就可以了,且不会改变程序的界面。
( 13917020359 发表于 2003-8-9 21:07:00)
 
第一、创建对话框时必须将它作为某个不可见框架窗口的子窗口;

直接按2,3步修改对话框即可实现。或如一楼所言。
不用这一步。 ( dexo 发表于 2003-6-13 23:19:00)
 
用App生成一个单文档工程,然后
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{

if (CFrameWnd::OnCreate(lpCreateStruct) == -1)
return -1;
////////////////////
//add by wz
ModifyStyleEx(0,WS_EX_TOOLWINDOW );
    。。。。。
可以隐藏,但是没有最小化和最大化按钮!why? ( wzcaca 发表于 2003-5-8 10:08:00)
 
TOMG 的方法倒是挺简单,如果能将标题栏的显示再做一些调整就更好了。除此之外,还要解决一个问题,那就是在任务管理器的“应用程序”栏里不要将程序名显示出来。 ( NorthTibet 发表于 2003-4-30 22:41:00)
 
在 CDialog::OnInitDialog() 里

只需加一句:
ModifyStyleEx(WS_EX_APPWINDOW,WS_EX_TOOLWINDOW);

就完事了! (标题栏会扁一点)
( TOMG 发表于 2003-4-30 15:14:00)
 
.......................................................

--------------------next---------------------

阅读(342) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~