1. 程式人生 > >實現最小化到托盤

實現最小化到托盤

轉自http://www.rupeng.com/forum/thread-1474-1-1.html

所謂的“托盤”,在Windows系統介面中,指的就是下面任務條右側,有系統時間等等的標誌的那一部分。在程式最小化或掛起時,但有不希望佔據工作列的時候,就可以把程式放到托盤區。


一、托盤程式設計相關函式  
    把程式放到托盤上的本質就是先在托盤區繪製一個圖示,然後把程式隱藏不見,再對托盤的圖示進行訊息處理,就可以了。   
    繪製圖標以及確定圖示所傳送訊息的函式只有一個:
     WINSHELLAPI BOOL WINAPI Shell_NotifyIcon(DWORD dwMessage, PNOTIFYICONDATA pnid);     

    這個函式負責向系統傳遞訊息,以新增、修改或刪除托盤區的圖示。返回值是個布林型別的。就是說,如果返回0,那就是成仁啦,非0才成功。引數dwMessage   是表示這個函式的應用功能是哪一方面,是新增、刪除,還是修改圖示。如果是新增,則它的值為NIM_ADD;刪除則是NIM_DELETE;而修改是NIM_MODIFY。引數pnid就是具體的和程式在托盤區的圖示有關係的結構了。它的定義如下:      
          typedef   struct   _NOTIFYICONDATA   {    
          DWORD   cbSize;    
          HWND   hWnd;    

          UINT   uID;    
          UINT   uFlags;    
          UINT   uCallbackMessage;    
          HICON   hIcon;    
          char   szTip[64];    
          }   NOTIFYICONDATA,   *PNOTIFYICONDATA;        
          下面就對該結構各個引數進行刨析:       
          cbSize   :   結構的長度,用“位”來做單位。一般在程式中,我們用(DWORD)sizeof(NOTIFYICONDATA)   給它賦值。      

          HWnd   :   一個控制代碼,如果對托盤中的圖示進行操作,相應的訊息就傳給這個控制代碼所代表的視窗。自然了,大多數情況下是this->m_hWnd嘍。    
          uID   :   在工程中定義的圖示ID   
          uFlags   :   這個成員標誌著其他哪些成員的資料是有效的,分別為NIF_ICON,   NIF_MESSAGE,   NIF_TIP,分別代表著資料有效的成員是hIcon,   uCallbackMessage,   szTip。當然,三個值可以用“|”聯絡到一起。下面分別對涉及到的成員進行闡述   
          hIcon   :   要增加,刪除或修改的圖示控制代碼。如果只知道個uID,   一般可能會用函式LoadIcon來得到控制代碼。例如LoadIcon   (   AfxGetInstanceHandle()   ,MAKEINTRESOURCE   (IDR_MAINFRAME)   )。   
          uCallbackMessage   :   這在對托盤區的操作中,是比較重要的資料成員。這是個訊息標誌,當用滑鼠對托盤區相應圖示進行操作的時候,就會傳遞訊息給Hwnd所代表的視窗。所以說,在uFlags中,一般都得標誌它有效。這裡一般都是自定義的訊息。   
          szTip   :   滑鼠移動到托盤圖示上時的提示文字。

二、原理
1、最小化的原理:首先要將視窗隱藏,然後 在右下角繪製圖標。
2、恢復的原理:將視窗顯示,再將托盤中的圖片刪除。

三、程式實現
1、自定義訊息WM_SHOWTASK: #define WM_SHOWTASK (WM_USER +1)
2、響應最小化按鈕

新增訊息對映ON_MESSAGE(WM_SYSCOMMAND,OnSysCommand)

void CMyDlg::OnSysCommand(UINT uID,LPARAM lParam)

{

switch(nID & 0xFFF0)
{
case SC_MINIMIZE:
{
ToTray();
break;
}
default:
CDialog::OnSysCommand(nID,lParam);
}

}

3、在訊息對映中新增 ON_MESSAGE(WM_SHOWTASK,OnShowTask),其中WM_SHOWTASK是訊息名,
OnShowTask是自己定義的訊息響應函式,後面有說明。

四、具體函式內容
1、最小化到托盤函式
void CMyDlg::ToTray(){
    NOTIFYICONDATA nid;
     nid.cbSize=(DWORD)sizeof(NOTIFYICONDATA);
     nid.hWnd=this->m_hWnd;
     nid.uID=IDR_MAINFRAME;
     nid.uFlags=NIF_ICON|NIF_MESSAGE|NIF_TIP ;
     nid.uCallbackMessage=WM_SHOWTASK;//自定義的訊息名稱
     nid.hIcon=LoadIcon(AfxGetInstanceHandle(),MAKEINTRESOURCE(IDR_MAINFRAME));
     wcscpy(nid.szTip,L"程式名稱");    //資訊提示條
     Shell_NotifyIcon(NIM_ADD,&nid);    //在托盤區新增圖示
     ShowWindow(SW_HIDE);    //隱藏主視窗
}
2、恢復介面函式
    在標頭檔案中定義訊息響應函式afx_msg LRESULT OnShowTask(WPARAM wParam,LPARAM lParam) ;//wParam接收的是圖示的ID,而lParam接收的是滑鼠的行為
LRESULT CMyDlg::OnShowTask(WPARAM wParam,LPARAM lParam)
{
     if(wParam!=IDR_MAINFRAME)
         return 1;
     switch(lParam)
     {
            case WM_RBUTTONUP://右鍵起來時彈出快捷選單,這裡只有一個“關閉”
                LPPOINT lpoint=new tagPOINT;
                ::GetCursorPos(lpoint);//得到滑鼠位置
                CMenu menu;
                menu.CreatePopupMenu();//宣告一個彈出式選單
       //增加選單項“關閉”,點選則傳送訊息WM_DESTROY給主視窗(已隱藏),將程式結束。
                menu.AppendMenu(MF_STRING,WM_DESTROY,L"關閉");
       //確定彈出式選單的位置
                menu.TrackPopupMenu(TPM_LEFTALIGN,lpoint->x,lpoint->y,this);
                HMENU hmenu=menu.Detach();//資源回收
                menu.DestroyMenu();
                delete lpoint;
                break;
           case WM_LBUTTONDBLCLK://雙擊左鍵的處理
                 this->ShowWindow(SW_SHOW);//簡單的顯示主視窗完事兒
                 DeleteTray();
                 break; 
           default: 
                 break;
     }
     return 0;

3、刪除托盤圖示函式
void CMyDlg::DeleteTray()

    NOTIFYICONDATA nid;
    nid.cbSize=(DWORD)sizeof(NOTIFYICONDATA);
    nid.hWnd=this->m_hWnd;
    nid.uID=IDR_MAINFRAME;
    nid.uFlags=NIF_ICON|NIF_MESSAGE|NIF_TIP ;
    nid.uCallbackMessage=WM_SHOWTASK;//自定義的訊息名稱
    nid.hIcon=LoadIcon(AfxGetInstanceHandle(),MAKEINTRESOURCE(IDR_MAINFRAME));
    wcscpy(nid.szTip,L"程式名稱");    //資訊提示條
    Shell_NotifyIcon(NIM_DELETE,&nid);    //在托盤區刪除圖示
}