VC托盘程序详解

(一)  原理
1、最小化的原理:首先要将窗口隐藏,然后 在右下角绘制图标。
2、恢复的原理:将窗口显示,再将托盘中的图片删除。
(二)程序实现
1、自定义消息WM_SHOWTASK: #define WM_SHOWTASK (WM_USER +1)
2、在MFC的::OnSysCommand(UINT nID, LPARAM lParam)函数体中增加一个命令响应                        
            if(nID==SC_MINIMIZE)
               ToTray(); //最小化到托盘的函数
 
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)); 
    strcpy(nid.szTip,"程序名称");    //信息提示条 
    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,"关闭"); 
            //确定弹出式菜单的位置 
            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)); 
  strcpy(nid.szTip,"程序名称");    //信息提示条为“计划任务提醒” 
  Shell_NotifyIcon(NIM_DELETE,&nid);    //在托盘区删除图标 
}
/***该系统需要处理的任务栏状态区的消息***/  
typedef struct _NOTIFYICONDATA {  
    DWORD cbSize;        //结构体的大小,以字节为单位  
    HWND hWnd;           //窗口的句柄  
    UINT uID;            //应用程序定义的任务栏图标的标识符  
    UINT uFlags;         //此成员表明具体哪些其他成员为合法数据  
    UINT uCallbackMessage;   //应用程序定义的消息标示  
    HICON hIcon;         //增加、修改或删除的图标的句柄  
    TCHAR szTip[64];     //指向一个以\0结束的字符串的指针  
    DWORD dwState;       //Version 5.0,图标的状态  
    DWORD dwStateMask;   //Version 5.0. 指明dwState成员的那些位可以被设置或者访问  
    TCHAR szInfo[256];   //指向一个以\0结束的字符串的指针,字符串的内容为气球提示内容  
    union {  
        UINT uTimeout;   //表示气球提示超时的时间,单位为毫秒,此时间后气球提示将消失  
        UINT uVersion;   //用来设置使用Windows 95 还是 Windows 2000风格的图标消息接口  
    };  
    TCHAR szInfoTitle[64];   //指向一个以\0结束的字符串的指针。字符串的内容为气球提示的标题  
    DWORD dwInfoFlags;   //设置此成员用来给气球提示框增加一个图标,增加的图标出现在气球提示标题的左侧  
    GUID guidItem;       //保留  
    HICON hBalloonIcon;  //用于Windows Vista或更高版本的自定义气球图标  
} NOTIFYICONDATA, *PNOTIFYICONDATA;  


注:在单文档中,在框架类中添加OnSysCommd消息响应
单文档的托盘程序
一、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);   
              return   0;   
  }   
  break;   
  }   
  return   CFrameWnd::WindowProc   
  (message,   wParam,   lParam);   
                    }   
    
  ----   为使应用程序退出时图标消失,映射WM_DESTROY消息,在OnDestroy()函数中加入:     
  ::Shell_NotifyIcon(NIM_DELETE,&m_tnid);   
    
  ----   至此,托盘图标程序的常规功能我们均已实现。我们还可以通过Shell_NotifyIcon()函数的调用实现不同状态下图标的改变,就象金山词霸在主窗口打开与暂停取词时图标有所改变一样。     
  ----   本程序在VC++6.0,Windows98/2000   Professional下调试通过。
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
制作系统托盘程序 VC 点击数:695 发布日期:2006-9-24 12:32:00 【收藏】 【评论】 【打印】 【编程爱好者论坛】 【关闭】 其实,在任务条上添加托盘比较容易实现,调用VC中的函数就可以解决,只是注意协调。接下来,我将给大家提供一个类,可以很容易的达到我们的目的,希望它能给大家一点帮助,如果你发现BUG也欢迎你和我联系。运行程序,左键双击或者右键单击任务条上的托盘,我们可以看到效果。   1. 有关类CsystemTray的说明:(文章的最后有本类的具体实现代码)   CSystemTray是CObject的扩展类,实现以下功能:    1、在任务条显示托盘图标。    2、设置提示Tip    3、设置图标的形状   主要函数说明: Create(CWnd* pWnd, UINT uCallbackMessage, LPCTSTR szTip, HICON icon, UINT uID);   功能:    生成一个图标。   参数说明:    pWnd:程序的主窗口,    uCallbackMessage:对应的消息映射,    szTip:鼠标停留时的提示文字,    icon:显示的图标,    uID:与之对应的菜单ID   · BOOL CSystemTray::SetIcon(HICON hIcon)     BOOL CSystemTray::SetIcon(LPCTSTR lpszIconName)     BOOL CSystemTray::SetIcon(UINT nIDResource)     BOOL CSystemTray::SetStandardIcon(LPCTSTR lpIconName)     BOOL CSystemTray::SetStandardIcon(UINT nIDResource)    功能:更改托盘上的图标。   void CSystemTray::ShowIcon()    功能:显示图标。   · BOOL CSystemTray::SetTooltipText(LPCTSTR pszTip)     BOOL CSystemTray::SetTooltipText(UINT nID)    功能:提示显示文字。   2. 使用本类的步骤如下:   第一步:在VC编程环境下,建立一个工程,基于对话框或者是单文档(或者是多文档)你随便,所有的选项都取默认值即可。   第二步:在mainfrm.h中定义变量CSystemTray m_TrayIcon;并添加函数声明:    afx_msg LRESULT OnTrayNotification(WPARAM wParam, LPARAM lParam);    afx_msg void OnSysCommand(UINT nID, LPARAM lParam);   第三步:在StaAFX.h中定义消息:    #define WM_ICON_NOTIFY WM_USER + 1   第四步:自定义菜单IDR_POPUPMENU,其中最少包括一项:ID:ID_VIEW_MAIN_WINDOW,Caption为"显示主窗口",并为该项在类CmainFrame中添加消息映射函数,COMMAND和UPDATE_COMMAND_UI。   第五步:在Mainframe.cpp中添加以下内容: 1. BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd) ON_COMMAND(ID_VIEW_MAIN_WINDOW, OnViewMainWindow) ON_UPDATE_COMMAND_UI(ID_VIEW_MAIN_WINDOW, OnUpdateViewMainWindow) ON_MESSAGE(WM_ICON_NOTIFY, OnTrayNotification) ON_WM_SYSCOMMAND() //}}AFX_MSG_MAP END_MESSAGE_MAP() 2. void CMainFrame::OnViewMainWindow() { if(IsWindowVisible()) { ShowWindow(SW_SHOWMINIMIZED); ShowWindow(SW_HIDE); m_TrayIcon.SetIcon(IDI_ICON1); } else { ShowWindow(SW_SHOW); ShowWindow(SW_RESTORE); m_TrayIcon.SetIcon(IDI_ICON2); } } void CMainFrame::OnUpdateViewMainWindow(CCmdUI* pCmdUI) { pCmdUI->SetCheck(IsWindowVisible()); } 3. int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct) { if (CFrameWnd::OnCreate(lpCreateStruct) == -1) return -1; //创建托盘图标 if (!m_TrayIcon.Create(this, WM_ICON_NOTIFY,"大屏实时显示程序", NULL, IDR_POPUPMENU)) return -1; m_TrayIcon.SetIcon(IDI_ICON1); SetMenu(NULL); return 0; } 4. LRESULT CMainFrame::OnTrayNotification(WPARAM wParam, LPARAM lParam) { if (wParam != IDR_POPUPMENU) return 0L; CMenu menu, *pSubMenu; if (LOWORD(lParam) == WM_RBUTTONUP) { CPoint pos; GetCursorPos(&pos;); if (!menu.LoadMenu(IDR_POPUPMENU)) return 0; if (!(pSubMenu=menu.GetSubMenu(0))) return 0; ::SetMenuDefaultItem(pSubMenu->m_hMenu, 3, TRUE); SetForegroundWindow(); pSubMenu->TrackPopupMenu(TPM_RIGHTALIGN | TPM_BOTTOMALIGN | TPM_RIGHTBUTTON, pos.x, pos.y, this); menu.DestroyMenu(); } else if (LOWORD(lParam) == WM_LBUTTONDBLCLK) { if (!menu.LoadMenu(IDR_POPUPMENU)) return 0; if (!(pSubMenu = menu.GetSubMenu(0))) return 0; SetForegroundWindow(); //激活第2个菜单项 SendMessage(WM_COMMAND, pSubMenu->GetMenuItemID(1), 0); menu.DestroyMenu(); } return 0; } void CMainFrame::OnSysCommand(UINT nID, LPARAM lParam) { if(nID==SC_MINIMIZE) { ShowWindow(SW_SHOWMINIMIZED); ShowWindow(SW_HIDE); } else CFrameWnd::OnSysCommand(nID, lParam); } 3. 附录:类的实现代码 CsystemTray的头文件 #ifndef _INCLUDED_SYSTEMTRAY_H_ #define _INCLUDED_SYSTEMTRAY_H_ ///////////////////////////////////////////////////////////////////////////// // CSystemTray window class CSystemTray : public CObject { // Construction/destruction public: CSystemTray(); CSystemTray(CWnd* pWnd, UINT uCallbackMessage, LPCTSTR szTip, HICON icon, UINT uID); virtual ~CSystemTray(); // Operations public: CFrameWnd * m_pFrame; BOOL Enabled() { return m_bEnabled; } BOOL Visible() { return !m_bHidden; } //Create the tray icon Create(CWnd* pWnd, UINT uCallbackMessage, LPCTSTR szTip, HICON icon, UINT uID); //Change or retrieve the Tooltip text BOOL SetTooltipText(LPCTSTR pszTooltipText); BOOL SetTooltipText(UINT nID); CString GetTooltipText() const; //Change or retrieve the icon displayed BOOL SetIcon(HICON hIcon); BOOL SetIcon(LPCTSTR lpIconName); BOOL SetIcon(UINT nIDResource); BOOL SetStandardIcon(LPCTSTR lpIconName); BOOL SetStandardIcon(UINT nIDResource); HICON GetIcon() const; void HideIcon(); void ShowIcon(); void RemoveIcon(); void MoveToRight(); //Change or retrieve the window to send notification messages to BOOL SetNotificationWnd(CWnd* pNotifyWnd); CWnd* GetNotificationWnd() const; //Default handler for tray notification message // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CSystemTray) //}}AFX_VIRTUAL // Implementation protected: BOOL m_bEnabled; // does O/S support tray icon? BOOL m_bHidden; // Has the icon been hidden? NOTIFYICONDATA m_tnd; DECLARE_DYNAMIC(CSystemTray) }; #endif /////////////////////////////////////////////////// CsystemTray的实现文件 #include "stdafx.h" #include "SystemTray.h" #ifdef _DEBUG #define new DEBUG_NEW #undef THIS_FILE static char THIS_FILE[] = __FILE__; #endif IMPLEMENT_DYNAMIC(CSystemTray, CObject) ///////////////////////////////////////////////// // CSystemTray construction/creation/destruction CSystemTray::CSystemTray() { memset(&m_tnd, 0, sizeof(m_tnd)); m_bEnabled = FALSE; m_bHidden = FALSE; } CSystemTray::CSystemTray(CWnd* pWnd, UINT uCallbackMessage, LPCTSTR szToolTip, HICON icon, UINT uID) { Create(pWnd, uCallbackMessage, szToolTip, icon, uID); m_bHidden = FALSE; } BOOL CSystemTray::Create(CWnd* pWnd, UINT uCallbackMessage, LPCTSTR szToolTip, HICON icon, UINT uID) { // this is only for Windows 95 (or higher) VERIFY(m_bEnabled = ( GetVersion() & 0xff ) >= 4); if (!m_bEnabled) return FALSE; //Make sure Notification window is valid VERIFY(m_bEnabled = (pWnd && ::IsWindow(pWnd->GetSafeHwnd()))); if (!m_bEnabled) return FALSE; //Make sure we avoid conflict with other messages ASSERT(uCallbackMessage >= WM_USER); //Tray only supports tooltip text up to 64 characters ASSERT(_tcslen(szToolTip) <= 64); // load up the NOTIFYICONDATA structure m_tnd.cbSize = sizeof(NOTIFYICONDATA); m_tnd.hWnd = pWnd->GetSafeHwnd(); m_tnd.uID = uID; m_tnd.hIcon = icon; m_tnd.uFlags = NIF_MESSAGE | NIF_ICON | NIF_TIP; m_tnd.uCallbackMessage = uCallbackMessage; strcpy (m_tnd.szTip, szToolTip); // Set the tray icon m_pFrame = (CFrameWnd*)pWnd; VERIFY(m_bEnabled = Shell_NotifyIcon(NIM_ADD, &m_tnd)); return m_bEnabled; } CSystemTray::~CSystemTray() { RemoveIcon(); } ///////////////////////////////////////////// // CSystemTray icon manipulation void CSystemTray::MoveToRight() { HideIcon(); ShowIcon(); } void CSystemTray::RemoveIcon() { if (!m_bEnabled) return; m_tnd.uFlags = 0; Shell_NotifyIcon(NIM_DELETE, &m_tnd); m_bEnabled = FALSE; } void CSystemTray::HideIcon() { if (m_bEnabled && !m_bHidden) { m_tnd.uFlags = NIF_ICON; Shell_NotifyIcon (NIM_DELETE, &m_tnd); m_bHidden = TRUE; } } void CSystemTray::ShowIcon() { if (m_bEnabled && m_bHidden) { m_tnd.uFlags = NIF_MESSAGE | NIF_ICON | NIF_TIP; Shell_NotifyIcon(NIM_ADD, &m_tnd); m_bHidden = FALSE; } } BOOL CSystemTray::SetIcon(HICON hIcon) { if (!m_bEnabled) return FALSE; m_tnd.uFlags = NIF_ICON; m_tnd.hIcon = hIcon; return Shell_NotifyIcon(NIM_MODIFY, &m_tnd); } BOOL CSystemTray::SetIcon(LPCTSTR lpszIconName) { HICON hIcon = AfxGetApp()->LoadIcon(lpszIconName); return SetIcon(hIcon); } BOOL CSystemTray::SetIcon(UINT nIDResource) { HICON hIcon = AfxGetApp()->LoadIcon(nIDResource); return SetIcon(hIcon); } BOOL CSystemTray::SetStandardIcon(LPCTSTR lpIconName) { HICON hIcon = LoadIcon(NULL, lpIconName); return SetIcon(hIcon); } BOOL CSystemTray::SetStandardIcon(UINT nIDResource) { HICON hIcon = ::LoadIcon(AfxGetInstanceHandle(), MAKEINTRESOURCE(nIDResource)); return SetIcon(hIcon); } HICON CSystemTray::GetIcon() const { HICON hIcon = NULL; if (m_bEnabled) hIcon = m_tnd.hIcon; return hIcon; } ////////////////////////////////////////////////// // CSystemTray tooltip text manipulation BOOL CSystemTray::SetTooltipText(LPCTSTR pszTip) { if (!m_bEnabled) return FALSE; m_tnd.uFlags = NIF_TIP; _tcscpy(m_tnd.szTip, pszTip); return Shell_NotifyIcon(NIM_MODIFY, &m_tnd); } BOOL CSystemTray::SetTooltipText(UINT nID) { CString strText; VERIFY(strText.LoadString(nID)); return SetTooltipText(strText); } CString CSystemTray::GetTooltipText() const { CString strText; if (m_bEnabled) strText = m_tnd.szTip; return strText; } //////////////////////////////////////////////// // CSystemTray notification window stuff BOOL CSystemTray::SetNotificationWnd(CWnd* pWnd) { if (!m_bEnabled) return FALSE; //Make sure Notification window is valid ASSERT(pWnd && ::IsWindow(pWnd->GetSafeHwnd())); m_tnd.hWnd = pWnd->GetSafeHwnd(); m_tnd.uFlags = 0; return Shell_NotifyIcon(NIM_MODIFY, &m_tnd); } CWnd* CSystemTray::GetNotificationWnd() const { return CWnd::FromHandle(m_tnd.hWnd); } 引用地址:http://blog.programfan.com/trackback.asp?id=18739
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值