想讓對話框程序隱藏啟動,然后需要時又可以顯示。
剛開始我也想到在OnInitDialog()中加上ShowWindow(SW_HIDE),其實這樣是不行的,不信自己去試驗
找了網上的方法,都沒有滿足我的要求,自己總結了一個比較滿意的方法。
定義一個bool b_show;//決定是否隱藏
b_show=false;//默認隱藏
可以在對話框中重載虛函數DefWindowProc()。
LRESULT CTest2Dlg::DefWindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{
if(message==133 )
{
if(b_show==false)
ShowWindow(SW_HIDE);
else
ShowWindow(SW_SHOW);
}
return CDialog::DefWindowProc(message, wParam, lParam);
}
LRESULT CTest2Dlg::DefWindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{
if(message==133 )
{
if(b_show==false)
ShowWindow(SW_HIDE);
else
ShowWindow(SW_SHOW);
}
return CDialog::DefWindowProc(message, wParam, lParam);
}
程序中想還原顯示的時候可以
b_show=true;
this->ShowWindow(SW_SHOW);
this->ShowWindow(SW_SHOW);
//-------------------------------------網上的相關資料-----------------------------------------------------------------------//
對於這類問題,大家最容易想到的可能就是在PreCreateWindow中添加cs.style &=~WS_VISIBLE;這是不可行的。程序仍可使用ShowWindow()將窗體顯示出來.
1.基於對話框的程序
我在論壇上看到有人說在OnInitDialog()中加上ShowWindow(SW_HIDE)對話框便不出現了,其實是不可行的。至於原因,我認為是系統是在OnInitDialog()后調用ShowWindow(SW_SHOW)讓對話框顯示的.可以添加下面代碼:
CXXDlg::OnInitDialog()
{
...
Sleep(5000);
return TRUE;
}
可以發現5秒后對話框才顯示出來.至於在何時調用的我也不清楚,但是我們可以在OnPaint()中加上ShowWindow(SW_HIDE),來達到隱藏的目的.不過使用的這種方法,會有一點閃爍.另外一種方法就是在OnInitDialog()中使用SetWindowPlacement()
我在論壇上看到有人說在OnInitDialog()中加上ShowWindow(SW_HIDE)對話框便不出現了,其實是不可行的。至於原因,我認為是系統是在OnInitDialog()后調用ShowWindow(SW_SHOW)讓對話框顯示的.可以添加下面代碼:
CXXDlg::OnInitDialog()
{
...
Sleep(5000);
return TRUE;
}
可以發現5秒后對話框才顯示出來.至於在何時調用的我也不清楚,但是我們可以在OnPaint()中加上ShowWindow(SW_HIDE),來達到隱藏的目的.不過使用的這種方法,會有一點閃爍.另外一種方法就是在OnInitDialog()中使用SetWindowPlacement()
GetWindowPlacement(&m_wp); //恢復時用
ModifyStyleEx(WS_EX_APPWINDOW,WS_EX_TOOLWINDOW);//從任務欄中去掉.
ModifyStyleEx(WS_EX_APPWINDOW,WS_EX_TOOLWINDOW);//從任務欄中去掉.
WINDOWPLACEMENT wp;
wp.length=sizeof(WINDOWPLACEMENT);
wp.flags=WPF_RESTORETOMAXIMIZED;
wp.showCmd=SW_HIDE;
SetWindowPlacement(&wp);
wp.length=sizeof(WINDOWPLACEMENT);
wp.flags=WPF_RESTORETOMAXIMIZED;
wp.showCmd=SW_HIDE;
SetWindowPlacement(&wp);
還有一種更簡單的方法:在OnInitDialog()中調用下面代碼.
SetWindowPos(&wndTop,0,0,0,0,NULL);
SetWindowPos(&wndTop,0,0,0,0,NULL);
2.基於單文檔的程序
我們一般采用的方法就是將InitInstance()中的:
CXXApp::InitInstance()
{
//m_pMainWnd->ShowWindow(SW_SHOW);
}
但是這樣窗體還會有閃爍。
因為MFC還要在ActiveFrame顯示框架,所以我們還需要添加下面代碼:
void CMainFrame::ActivateFrame(int nCmdShow)
{
nCmdShow=SW_HIDE;
CFrameWnd::ActivateFrame(nCmdShow);
}
或者:
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
AfxGetApp()->m_nCmdShow=SW_HIDE;
}
順便說一下,上面通過ModifyStyleEx(WS_EX_APPWINDOW,WS_EX_TOOLWINDOW)的方法來實現從任務欄去掉按鈕,這樣當顯示時還要切換顯示的模式,其實還可以通過調用TaskbarList組件直接刪除和添加:
ITaskbarList的定義在shobjidl.h(vs.net)中。
也可以手動定義:
DECLARE_INTERFACE_(ITaskbarList,IUnknown)
{
STDMETHOD(QueryInterface)(THIS_ REFIID riid,LPVOID* ppvObj) PURE;
STDMETHOD_(ULONG,AddRef)(THIS) PURE;
STDMETHOD_(ULONG,Release)(THIS) PURE;
STDMETHOD(ActivateTab)(HWND) PURE;
STDMETHOD(AddTab)(HWND) PURE;
STDMETHOD(DeleteTab)(HWND) PURE;
STDMETHOD(HrInit)(void) PURE;
};
BOOL CMy2App::InitInstance()
{
CoInitialize(0);
ITaskbarList *pObj;
CoCreateInstance(CLSID_TaskbarList,NULL,CLSCTX_INPROC_SERVER,IID_ITaskbarList,(void**)&pObj);
pObj->DeleteTab(m_pMainWnd->m_hWnd); //從任務欄上刪除
//pObj->AddTab(m_pMainWnd->m_hWnd); //添加
CXXApp::InitInstance()
{
//m_pMainWnd->ShowWindow(SW_SHOW);
}
但是這樣窗體還會有閃爍。
因為MFC還要在ActiveFrame顯示框架,所以我們還需要添加下面代碼:
void CMainFrame::ActivateFrame(int nCmdShow)
{
nCmdShow=SW_HIDE;
CFrameWnd::ActivateFrame(nCmdShow);
}
或者:
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
AfxGetApp()->m_nCmdShow=SW_HIDE;
}
順便說一下,上面通過ModifyStyleEx(WS_EX_APPWINDOW,WS_EX_TOOLWINDOW)的方法來實現從任務欄去掉按鈕,這樣當顯示時還要切換顯示的模式,其實還可以通過調用TaskbarList組件直接刪除和添加:
ITaskbarList的定義在shobjidl.h(vs.net)中。
也可以手動定義:
DECLARE_INTERFACE_(ITaskbarList,IUnknown)
{
STDMETHOD(QueryInterface)(THIS_ REFIID riid,LPVOID* ppvObj) PURE;
STDMETHOD_(ULONG,AddRef)(THIS) PURE;
STDMETHOD_(ULONG,Release)(THIS) PURE;
STDMETHOD(ActivateTab)(HWND) PURE;
STDMETHOD(AddTab)(HWND) PURE;
STDMETHOD(DeleteTab)(HWND) PURE;
STDMETHOD(HrInit)(void) PURE;
};
BOOL CMy2App::InitInstance()
{
CoInitialize(0);
ITaskbarList *pObj;
CoCreateInstance(CLSID_TaskbarList,NULL,CLSCTX_INPROC_SERVER,IID_ITaskbarList,(void**)&pObj);
pObj->DeleteTab(m_pMainWnd->m_hWnd); //從任務欄上刪除
//pObj->AddTab(m_pMainWnd->m_hWnd); //添加
pObj->Release();
CoUninitialize();
}
CoUninitialize();
}
所以我們還可以用將窗體最小化,並從任備欄上刪除按鈕的方法來實現隱藏.
有很多應用程序要求一起動就隱藏起來,這些程序多作為后台程序運行,希望不影響其他窗口,往往只在托盤區顯示一個圖標。這些程序通常都是對話框程序,而對話框在初始化的過程上與SDI、MDI的初始化是不同的,對話框只需要DoModule或者是CreateDialog等等對話框函數調用一次便可,SDI、MDI則要好幾步才行。這樣看來,對話框在使用方法上面是隱藏了不少細節的,其中就沒有SDI、MDI所要求的ShowWindow(nCmdShow)這一步。因此對話框要想一運行就隱藏,並不是很直接的。有一些方法可以做到這一點,下面我們就來看看幾種方案。
1.定時器
最直觀,又是最無奈的一個方法就是使用定時器。既然我們在對話框開始顯示之前不能用ShowWindow(SW_HIDE)將其隱藏,那就給一個時間讓它顯示,完了我們在隱藏它。
方法:
1.在OnInitDialog()函數里設置定時器:(WINDOWS API里面響應消息WM_INITDIALOG)
SetTimer(1, 1, NULL);
2.添加處理WM_TIMER的消息處理函數OnTimer,添加代碼:
if(nIDEvent == 1)
{
DeleteTimer(1);
ShowWindow(SW_HIDE);
}
這種方法的缺點是顯而易見的,使用定時器,使得程序的穩定性似乎打一個折扣;窗口是要先顯示出來的,那么效果就是窗口閃了一下消失。
2.改變對話框顯示狀況
在對話框初始化時改變其顯示屬性可以讓它隱藏起來。方法是調用SetWindowPlacement函數:
BOOL CDialogExDlg::OnInitDialog()
{
CDialog::OnInitDialog();
//DO something
WINDOWPLACEMENT wp;
wp.length=sizeof(WINDOWPLACEMENT);
wp.flags=WPF_RESTORETOMAXIMIZED;
wp.showCmd=SW_HIDE;
SetWindowPlacement(&wp);
return TRUE;
}
在需要顯示時(通常是響應熱鍵或者托盤圖標的鼠標消息):
WINDOWPLACEMENT wp;
wp.length=sizeof(WINDOWPLACEMENT);
wp.flags=WPF_RESTORETOMAXIMIZED;
wp.showCmd=SW_SHOW;
SetWindowPlacement(&wp);
這樣的效果很不理想:窗口顯示在屏幕的左上角,並且是只有標題欄,要正常顯示,還需加上如下代碼:
定義一個成員變量CRect rect;
在OnInitDialog()里面:
GetWindowRect(&rect);
在需要顯示的地方:
SetWindowPos(&wndNoTopMost, wndRc.left, wndRc.top, wndRc.right, wndRc.bottom, SWP_SHOWWINDOW);
CenterWindow();
即使這樣,效果還是很差。
這種方法還有一個弊端是當程序開始運行並且隱藏起來后,原來激活的窗口變成了非激活狀態了,而當對話框顯示出來后,對話框自身也是非激活狀態的。
3.不繪制窗口
當對話框顯示時將要響應消息WM_PAINT繪制客戶區,相應消息WM_NCPAINT繪制窗口邊框。我們在窗口第一次自繪自身時隱藏窗口,可以收到比較良好的效果。由於窗口是先畫窗口邊框,所以我們僅需處理WM_NCPAINT即可。代碼如下:
添加WM_NCPAINT處理函數。
void CMyDialog::OnNcPaint()
{
static int i = 2;
if(i > 0)
{
i --;
ShowWindow(SW_HIDE);
}
else
CDialog::OnNcPaint();
}
這里有個問題:為什么要定義靜態變量i而且設其值為2呢?
我們只要窗口隱藏第一次,所以定義這個變量可以判斷是否時首次顯示窗口。當程序開始運行時,系統發送(SendMessage)WM_NCPAINT消息,此時程序的窗口邊框應該被顯示,但是此時我們沒有作任何顯示的操作,而是將窗口隱藏,ShowWindow(SW_HIDE)將把窗口的WS_VISIBLE屬性去掉,繼續執行,程序將檢查WS_VISIBLE屬性,如果沒有則顯示窗口,所以又發送了一個WM_NCPAINT消息。所以我們要處理兩次WM_NCPAINT消息。
在需要窗口顯示時,調用ShowWindow(SW_SHOW)即可。
程序執行的結果是,原來處於激活狀態的窗口可能會閃動兩下,然后仍然處於激活狀態。這種處理方式比上面的方式要優越得多。
4.將對話框作為子窗口
這種方法是采用SDI框架,主窗口始終隱藏,對話框作為主窗口的成員變量,在CMainFrame::OnCreate()里面加入下代碼:
if(!dlg.Create(IDD_MYDIALOG, this))
{
return –1;
}
dlg.ShowWindow(SW_HIDE);
在要顯示對話框的地方用dlg.ShowWindow(SW_SHOW);即可。注意,主窗口一定要隱藏,否則對話框可能會閃現一下。
隱藏狀態欄窗口
上面介紹了幾種檢查對話框的方法,大家如果試過的話可能已經注意到系統狀態欄里在程序啟動時會有程序的圖標閃過,在隱藏對話框的時候這個也是要隱藏的,方法很簡單:
在OnInitDialog()函數里面加上ModifyStyleEx(WS_EX_APPWINDOW, WS_EX_TOOLWINDOW);即可。在要顯示窗口的地方加上代碼ModifyStyleEx(WS_EX_TOOLWINDOW, WS_EX_APPWINDOW);即將窗口的擴展樣式改回來。
1.定時器
最直觀,又是最無奈的一個方法就是使用定時器。既然我們在對話框開始顯示之前不能用ShowWindow(SW_HIDE)將其隱藏,那就給一個時間讓它顯示,完了我們在隱藏它。
方法:
1.在OnInitDialog()函數里設置定時器:(WINDOWS API里面響應消息WM_INITDIALOG)
SetTimer(1, 1, NULL);
2.添加處理WM_TIMER的消息處理函數OnTimer,添加代碼:
if(nIDEvent == 1)
{
DeleteTimer(1);
ShowWindow(SW_HIDE);
}
這種方法的缺點是顯而易見的,使用定時器,使得程序的穩定性似乎打一個折扣;窗口是要先顯示出來的,那么效果就是窗口閃了一下消失。
2.改變對話框顯示狀況
在對話框初始化時改變其顯示屬性可以讓它隱藏起來。方法是調用SetWindowPlacement函數:
BOOL CDialogExDlg::OnInitDialog()
{
CDialog::OnInitDialog();
//DO something
WINDOWPLACEMENT wp;
wp.length=sizeof(WINDOWPLACEMENT);
wp.flags=WPF_RESTORETOMAXIMIZED;
wp.showCmd=SW_HIDE;
SetWindowPlacement(&wp);
return TRUE;
}
在需要顯示時(通常是響應熱鍵或者托盤圖標的鼠標消息):
WINDOWPLACEMENT wp;
wp.length=sizeof(WINDOWPLACEMENT);
wp.flags=WPF_RESTORETOMAXIMIZED;
wp.showCmd=SW_SHOW;
SetWindowPlacement(&wp);
這樣的效果很不理想:窗口顯示在屏幕的左上角,並且是只有標題欄,要正常顯示,還需加上如下代碼:
定義一個成員變量CRect rect;
在OnInitDialog()里面:
GetWindowRect(&rect);
在需要顯示的地方:
SetWindowPos(&wndNoTopMost, wndRc.left, wndRc.top, wndRc.right, wndRc.bottom, SWP_SHOWWINDOW);
CenterWindow();
即使這樣,效果還是很差。
這種方法還有一個弊端是當程序開始運行並且隱藏起來后,原來激活的窗口變成了非激活狀態了,而當對話框顯示出來后,對話框自身也是非激活狀態的。
3.不繪制窗口
當對話框顯示時將要響應消息WM_PAINT繪制客戶區,相應消息WM_NCPAINT繪制窗口邊框。我們在窗口第一次自繪自身時隱藏窗口,可以收到比較良好的效果。由於窗口是先畫窗口邊框,所以我們僅需處理WM_NCPAINT即可。代碼如下:
添加WM_NCPAINT處理函數。
void CMyDialog::OnNcPaint()
{
static int i = 2;
if(i > 0)
{
i --;
ShowWindow(SW_HIDE);
}
else
CDialog::OnNcPaint();
}
這里有個問題:為什么要定義靜態變量i而且設其值為2呢?
我們只要窗口隱藏第一次,所以定義這個變量可以判斷是否時首次顯示窗口。當程序開始運行時,系統發送(SendMessage)WM_NCPAINT消息,此時程序的窗口邊框應該被顯示,但是此時我們沒有作任何顯示的操作,而是將窗口隱藏,ShowWindow(SW_HIDE)將把窗口的WS_VISIBLE屬性去掉,繼續執行,程序將檢查WS_VISIBLE屬性,如果沒有則顯示窗口,所以又發送了一個WM_NCPAINT消息。所以我們要處理兩次WM_NCPAINT消息。
在需要窗口顯示時,調用ShowWindow(SW_SHOW)即可。
程序執行的結果是,原來處於激活狀態的窗口可能會閃動兩下,然后仍然處於激活狀態。這種處理方式比上面的方式要優越得多。
4.將對話框作為子窗口
這種方法是采用SDI框架,主窗口始終隱藏,對話框作為主窗口的成員變量,在CMainFrame::OnCreate()里面加入下代碼:
if(!dlg.Create(IDD_MYDIALOG, this))
{
return –1;
}
dlg.ShowWindow(SW_HIDE);
在要顯示對話框的地方用dlg.ShowWindow(SW_SHOW);即可。注意,主窗口一定要隱藏,否則對話框可能會閃現一下。
隱藏狀態欄窗口
上面介紹了幾種檢查對話框的方法,大家如果試過的話可能已經注意到系統狀態欄里在程序啟動時會有程序的圖標閃過,在隱藏對話框的時候這個也是要隱藏的,方法很簡單:
在OnInitDialog()函數里面加上ModifyStyleEx(WS_EX_APPWINDOW, WS_EX_TOOLWINDOW);即可。在要顯示窗口的地方加上代碼ModifyStyleEx(WS_EX_TOOLWINDOW, WS_EX_APPWINDOW);即將窗口的擴展樣式改回來。
