MFC系统托盘的实现
通常電腦里邊的軟件,當你打開后會在電腦最右下角的任務欄上生成一個系統托盤,當你點擊最小化或者點擊關閉按鈕后,若想恢復窗口,可以左鍵雙擊或者單機這個系統圖標,同時鼠標右鍵點擊,又會有其他的菜單彈出,比如退出。
我看著感覺還是挺酷炫的,所以打算也在我的程序里邊添加一個這樣的功能。下邊我把我自己探索的過程記錄一下:
首先,我們需要用到NOTIFYICONDATA類,它是我們實現系統托盤的核心。關于這個類,百度百科上是這么說的:NOTIFYICONDATA是一個函數公式,主要含義和作用是以此函數用來向任務欄托盤區域發送消息。好了,其他的就不用管了,接下來進行實際的操作。
1、聲明一個NOTIFYICONDATA類對象,一般可以放在父類里邊作為成員變量或者作為全局的變量。
private:NOTIFYICONDATA NotifyIcon; //系統托盤類
2、自定義一個消息
#define WM_SYSTEMTRAY WM_USER+5
為什么是 WM_USER+5?關于消息WM_USER,為了防止用戶定義的消息ID與系統的消息ID沖突,MS(Microsoft)定義了一個宏WM_USER,小于WM_USER的ID被系統使用,大于WM_USER的ID被用戶使用。加5是我自己隨便定義的,當然你可以自己指定具體的數值。
3、接下來這步,可以自己手動添加,也可以通過類向導來操作。我采用手動添加方式。聲明一個響應函數,用來響應鼠標的操作。
protected:afx_msg LRESULT OnSystemtray(WPARAM wParam, LPARAM lParam);
4、注冊剛才自定義的消息。在BEGIN_MESSAGE_MAP(CMyPlayerDlg, CDialogEx)和END_MESSAGE_MAP()之間添加如下代碼:
ON_MESSAGE(WM_SYSTEMTRAY, &CMyPlayerDlg::OnSystemtray)
5、然后這一步就是開始產生作用的操作。一般添加在Oninitdialog()中,但是也可以是其他的函數中,比如只有在你點擊關閉按鈕后才添加系統拖盤圖標,那么就不是在此處添加,具體在哪里添加,后面我會講到。
BOOL CMyPlayerDlg::OnInitDialog()
{//設置系統托盤NotifyIcon.cbSize=sizeof(NOTIFYICONDATA);//NotifyIcon.hIcon=AfxGetApp()->LoadIcon(IDR_MAINFRAME);NotifyIcon.hIcon=m_hIcon; //上面那句也可以NotifyIcon.hWnd=m_hWnd;lstrcpy(NotifyIcon.szTip,_T("爆瘋牛逼一代"));NotifyIcon.uCallbackMessage=WM_SYSTEMTRAY;NotifyIcon.uFlags=NIF_ICON | NIF_MESSAGE | NIF_TIP;Shell_NotifyIcon(NIM_ADD,&NotifyIcon); //添加系統托盤//...}
其中比較重要的函數是Shell_NotifyIcon(NIM_ADD,&NotifyIcon),關于這個函數的具體解釋,可以看這里:http://www.cnblogs.com/duzouzhe/archive/2010/04/08/1707050.html
NIM_ADD參數表示添加一個圖標到系統托盤區。
6、上面我們定義了響應函數afx_msg LRESULT OnSystemtray(WPARAM wParam, LPARAM lParam),那么具體如何響應鼠標左鍵和右鍵的操作呢?
直接上代碼:
afx_msg LRESULT CMyPlayerDlg::OnSystemtray(WPARAM wParam, LPARAM lParam)
{//wParam接收的是圖標的ID,而lParam接收的是鼠標的行為
// if(wParam!=IDR_MAINFRAME)
// return 1; switch(lParam) { case WM_RBUTTONDOWN://右鍵起來時彈出快捷菜單{ CMenu menuexit;//menu.LoadMenuW(IDR_MENU1);//加載菜單資源menuexit.LoadMenuA(IDR_MENUexit);CMenu *pPopup=menuexit.GetSubMenu(0);CPoint mypoint;GetCursorPos(&mypoint);//ClientToScreen(&mypoint);//將客戶區坐標轉換為屏幕坐標SetForegroundWindow(); PostMessage(WM_NULL,0,0);//顯示右鍵菜單,由視類窗口擁有。pPopup->TrackPopupMenu(TPM_LEFTALIGN,mypoint.x,mypoint.y,this); } break; case WM_LBUTTONDOWN://左鍵單擊的處理 { ModifyStyleEx(0,WS_EX_TOPMOST); //可以改變窗口的顯示風格ShowWindow(SW_SHOWNORMAL); } break; } return 0;
}
代碼中我只實現了兩個消息的響應,即鼠標左鍵單擊和右鍵單擊的消息。當鼠標左鍵單擊后,恢復原窗口的大小和位置;當鼠標右鍵單擊后,彈出菜單。我自己的菜單是如下設置的:
case WM_LBUTTONDOWN:里邊的代碼沒有什么好講的,關于case WM_RBUTTONDOWN:里邊的代碼,其實也沒什么好講的,也就是當鼠標右鍵單擊時,裝載一個右鍵菜單而已,其他的沒有什么。具體可以看我之前的博客: 右鍵彈出菜單和快捷鍵的設置 。
然后菜單里的每一個子菜單的響應函數記得要編寫,但先預留退出子菜單我下面講。
7、其實上面的幾部就已經可以實現想要的效果了。但是如果想要點擊關閉按鈕后程序并不關閉,而只是以系統托盤的形式出現在托盤區,直到你鼠標右鍵點擊圖標,執行退出操作,才真正退出,另一方面當你點擊最小化后,程序最小化,但任務欄上任然具有程序的圖標。如果想要實現這樣的效果,其實也很簡單,我們需要重寫關閉“X”的響應函數。
具體重寫的操作我就不寫了,之前的博客寫過這樣的過程。重寫代碼如下:
void CMyPlayerDlg::OnCancel() //點擊X 按鈕,最小化到系統托盤
{// TODO: 在此添加專用代碼和/或調用基類this->ShowWindow(HIDE_WINDOW);//CDialogEx::OnCancel();
}
注意,注意,上面代碼中的CDialogEx::OnCancel();一定要注釋掉。因為我們只是想改寫關閉操作,但并不真的執行關閉操作,也就是說我們只是想重寫關閉按鈕操作的響應函數。
但是,如果這樣操作以后,你運行會發現,程序關不了了,因為關閉按鈕只是執行隱藏窗口功能,你的右鍵菜單的退出操作也還沒寫,那怎么辦,先殺了這個進程,接著編寫如下的步驟。
(關于第5步提到的如果只是想在點擊關閉按鈕的時候才添加系統托盤,那也應該再OnCancel函數中書寫第5步的代碼,同時書寫this->ShowWindow(HIDE_WINDOW);)
8、關于窗口的關閉過程,我前面的博客里邊也記錄過,此處就不多說了 。窗口關閉的過程中會執行DestroyWindow()函數,那么我們也將重寫這個函數。
BOOL CMyPlayerDlg::DestroyWindow()
{// TODO: 在此添加專用代碼和/或調用基類Shell_NotifyIcon(NIM_DELETE, &NotifyIcon);//消除托盤圖標return CDialogEx::DestroyWindow();
}
其中Shell_NotifyIcon(NIM_DELETE, &NotifyIcon);//消除托盤圖標很關鍵,正好和之前的Shell_NotifyIcon(NIM_ADD,&NotifyIcon)對應起來了。一個是添加,一個是刪除。如果不寫Shell_NotifyIcon(NIM_DELETE, &NotifyIcon),當你程序退出后,在系統托盤區的圖標還會存在,因為進程殘留,需要你鼠標移上去才會消失。
那么我們還需要在子菜單退出的響應函數中調用DestroyWindow()函數。
void CMyPlayerDlg::Onexit()
{// TODO: 在此添加命令處理程序代碼DestroyWindow();
}
到此,我們就實現了一個比較滿意的系統托盤方案,當然我自己探索的時候,還是碰到了 一些問題的,比如如何執行關閉操作,如何調用退出函數等等。但是,都一一克服了,仔細想想,其實也蠻簡單的。
ok ,關于這個就到這里。
拙見,小記!
總結
以上是生活随笔為你收集整理的MFC系统托盘的实现的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: MFC静态文本控件设置超链接
- 下一篇: 磁盘文件目录罗列和list控件的使用