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

    最近有个朋友做了一个基于对话框的小程序,大家知道,一般具有用户界面的 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("http://www.vckbase.com");
	m_wndLink2.m_link = _T("http://www.vckbase.com");
	m_wndLink3.m_link = _T("http://www.vckbase.com");
	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("http://www.vckbase.com");
	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对隐不隐藏对话框没用太大关系。这方面谁有更好的经验,欢迎来信交流。

最后祝大家身体健康!

你可能感兴趣的:(Windows界面编程,MFC)