最近有個朋友做了一個基於對話框的小程序,大家知道,一般具有用戶界面的 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對隱不隱藏對話框沒用太大關系。這方面誰有更好的經驗,歡迎來信交流。
最後祝大家身體健康!
本文配套源碼