在Windows操作系統中,任務欄的右邊(托盤)會常駐幾個圖標,如輸入法切換圖標、音量控制圖標等,此外我們還經常遇到具有托盤圖標的軟件,如金山詞霸、實時監測功能的殺毒軟件等。這些軟件在後台運行,通常不占用太多的屏幕資源,只在通知欄上放一個小小的標志,必要時我們可以通過用鼠標點擊圖標對其進行選單操作或激活其主窗口。有時我們自己編寫的程序也希望有類似的效果,本文將簡單地介紹用VC設計托盤圖標程序的方法。
一、 NOTIFYICONDATA結構
NOTIFYICONDATA結構包含了系統用來處理托盤圖標的信息,它包括選擇的圖標、回調消息、提示消息和圖標對應的窗口等內容。其定義為:
typedef struct—NOTIFYICONDATA {
DWord cbSize; //以字節為單位的這個結構的大小
HWND hWnd; //接收托盤圖標通知消息的窗口句柄
UINT uID; //應用程序定義的該圖標的ID號
UINT uFlags; //設置該圖標的屬性
UINT uCallbackMessage; //應用程序定義的消息ID號,此消息傳遞給hWnd
HICON hIcon; //圖標的句柄
char szTip[64]; //鼠標停留在圖標上顯示的提示信息
} NOTIFYICONDATA, ?PNOTIFYICONDATA;
該結構中,成員uFlags可以是下列的組合或其中之一:
NIF_ICON:設置成員hIcon有效
NIF_MESSAGE:設置成員uCallbackMessage有效
NIF_TIP:設置成員szTip有效
二、 Shell_NotifyIcon函數
全局函數Shell_NotifyIcon()用於在托盤上增加、刪除或修改圖標。其原型為:
WINSHELLAPI BOOL WINAPI Shell_NotifyIcon( DWord dwMessage,PNOTIFYICONDATA pnid);
Pnid是上面的NOTIFYICONDATA結構的指針。
dwMessage是被傳遞的消息,可以是以下消息之一:
NIM_ADD:增加圖標
NIM_DELETE:刪除圖標
NIM_MODIFY:修改圖標
三、 托盤圖標程序設計示例
首先我們用AppWizard創建一個不基於文檔和視圖結構的應用程序Tray。我們並不想在應用程序啟動時顯示主窗口,所以需要刪除應用程序類CTrayApp中成員函數InitInstance()的以下兩句代碼:
pFrame-〉ActivateFrame();
pFrame-〉ShowWindow(SW_SHOW);
在CMainFrame類中加入NOTIFYICONDATA結構的保護成員變量m_tnid,並在其OnCreate函數中return語句前加入生成托盤圖標的代碼:
m_tnid.cbSize=sizeof(NOTIFYICONDATA);
m_tnid.hWnd=this-〉m_hWnd;
m_tnid.uFlags=NIF_MESSAGE|NIF_ICON|NIF_TIP;
m_tnid.uCallbackMessage=MYWM_NOTIFYICON;
//用戶定義的回調消息
CString szToolTip;
szToolTip=_T("托盤圖標實例");
_tcscpy(m_tnid.szTip, szToolTip);
m_tnid.uID=IDR_MAINFRAME;
HICON hIcon;
hIcon=AfxGetApp()-〉LoadIcon(IDR_MAINFRAME);
m_tnid.hIcon=hIcon;
::Shell_NotifyIcon(NIM_ADD,&&m_tnid);
if(hIcon)::DestroyIcon(hIcon);
返回消息的ID應在主框架類的頭函數中定義:
#define MYWM_NOTIFYICON WM_USER+1
為了處理圖標返回消息,如鼠標左鍵雙擊、鼠標右鍵單擊消息,我們重載WindowProc()函數。此外,我們還希望在主框架窗口最小化時圖標不在任務欄的空白區出現,在此函數中同時作相應處理。
LRESULT CMainFrame::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{
switch(message){
case MYWM_NOTIFYICON:
//如果是用戶定義的消息
if(lParam==WM_LBUTTONDBLCLK){
//鼠標雙擊時主窗口出現
AfxGetApp()-〉m_pMainWnd-〉ShowWindow(SW_SHOW);
}
else if(lParam==WM_RBUTTONDOWN){ //鼠標右鍵單擊彈出選單
CMenu menu;
menu.LoadMenu(IDR_RIGHT_MENU); //載入事先定義的選單
CMenu?pMenu=menu.GetSubMenu(0);
CPoint pos;
GetCursorPos(&&pos);
pMenu-〉TrackPopupMenu(TPM_LEFTALIGN|TPM_RIGHTBUTTON,pos.x,pos.y,AfxGetMainWnd());
}
break;
case WM_SYSCOMMAND:
//如果是系統消息
if(wParam= =SC_MINIMIZE){
//接收到最小化消息時主窗口隱藏
AfxGetApp()-〉m_pMainWnd-〉ShowWindow(SW_HIDE);
[1] [2] 下一頁