Windows编程入门(5)MFC程序最小托盘化

2024-03-27 12:32

本文主要是介绍Windows编程入门(5)MFC程序最小托盘化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 转载自http://blog.csdn.net/csf111/article/details/6980508

  为了使MFC程序(SDI,MDI,DLG)均能最小化到系统托盘,这里运用消息机制来实现系统托盘效果。

          那么什么是托盘呢?所谓的“托盘”,在Windows系统界面中,指的就是下面任务条右侧,有系统时间等等的标志的那一部分。在程序最小化或挂起时,但有不希望占据任务栏的时候,就可以把程序放到托盘区。

          怎么实现呢?这里需要运用到的Windows API函数:

BOOL Shell_NotifyIcon(DWORD dwMessage, PNOTIFYICONDATA lpdata
);
          函数了里面的参数对于实现不同的效果尤其重要,包括托盘图标、托盘菜单等等。那么我们来了解一下两个参数吧:

dwMessage可以取以下值:

NIM_ADD 向托盘中加入一个图标
NIM_MODIFY 修改托盘中的图标
NIM_DELETE 从托盘中删除一个图标

参数pnid是NOTIFYICONDATA结构的一个引用。该结构的原型如下:

[cpp] view plaincopyprint?
  1. typedef struct _NOTIFYICONDATA {   
  2.     DWORD cbSize; // 结构的大小,必须在程序中给出  
  3.     HWND hWnd;    // 程序中将要接收托盘消息的窗口句柄  
  4.     UINT uID;     // 应用程序中定义的托盘图标ID,此参数用作标识  
  5.     UINT uFlags;  //设置属性 标记下边3个参数是否有效  
  6.     UINT uCallbackMessage;// 自定义的消息ID值  
  7.     HICON hIcon;//显示在系统托盘上的Icon的句柄  
  8.     #if (_WIN32_IE < 0x0500)   
  9.         TCHAR szTip[64;// 用于图标显示的提示字符串  
  10.     #else   
  11.         TCHAR szTip[128];  
  12.     #endif   
  13.     #if (_WIN32_IE >= 0x0500)  
  14.         DWORD dwState;   
  15.         DWORD dwStateMask;   
  16.         TCHAR szInfo[256];   
  17.         union {  
  18.             UINT  uTimeout;   
  19.             UINT  uVersion;   
  20.         } DUMMYUNIONNAME;  
  21.         TCHAR szInfoTitle[64];   
  22.         DWORD dwInfoFlags;   
  23.     #endif   
  24.     #if (_WIN32_IE >= 0x600)   
  25.         GUID guidItem;  
  26.     #endif   
  27. } NOTIFYICONDATA, *PNOTIFYICONDATA;   
[cpp] view plain copy
 print?
  1. typedef struct _NOTIFYICONDATA {   
  2.     DWORD cbSize; // 结构的大小,必须在程序中给出  
  3.     HWND hWnd;    // 程序中将要接收托盘消息的窗口句柄  
  4.     UINT uID;     // 应用程序中定义的托盘图标ID,此参数用作标识  
  5.     UINT uFlags;  //设置属性 标记下边3个参数是否有效  
  6.     UINT uCallbackMessage;// 自定义的消息ID值  
  7.     HICON hIcon;//显示在系统托盘上的Icon的句柄  
  8.     #if (_WIN32_IE < 0x0500)  
  9.         TCHAR szTip[64;// 用于图标显示的提示字符串  
  10.     #else  
  11.         TCHAR szTip[128];  
  12.     #endif  
  13.     #if (_WIN32_IE >= 0x0500)  
  14.         DWORD dwState;   
  15.         DWORD dwStateMask;   
  16.         TCHAR szInfo[256];   
  17.         union {  
  18.             UINT  uTimeout;   
  19.             UINT  uVersion;   
  20.         } DUMMYUNIONNAME;  
  21.         TCHAR szInfoTitle[64];   
  22.         DWORD dwInfoFlags;   
  23.     #endif  
  24.     #if (_WIN32_IE >= 0x600)  
  25.         GUID guidItem;  
  26.     #endif  
  27. } NOTIFYICONDATA, *PNOTIFYICONDATA;   
           具体步骤如下:

1.在程序中增加一个私有的NOTIFYICONDAT对象

[cpp] view plaincopyprint?
  1. private:  
  2.         NOTIFYICONDATA m_tnid;  
[cpp] view plain copy
 print?
  1. private:  
  2.         NOTIFYICONDATA m_tnid;  
2.在初始化窗口时对NOTIFYICONDAT对象初始化 用Shell_NotifyIcon函数进行注册
[cpp] view plaincopyprint?
  1. m_notify.cbSize=sizeof NOTIFYICONDATA;  
  2. m_notify.hWnd=this->m_hWnd;    
  3. m_notify.uID=IDR_MAINFRAME;  
  4. m_notify.hIcon=LoadIcon(AfxGetInstanceHandle(),MAKEINTRESOURCE(IDR_MAINFRAME));  
  5. strcpy(m_notify.szTip,"Michael_Chen is a good man");  
  6. m_notify.uCallbackMessage=WM_USER_NOTIFYICON;  
  7. m_notify.uFlags=NIF_ICON|NIF_MESSAGE|NIF_TIP; //OK,下面就是托盘产生了.   
  8. Shell_NotifyIcon(NIM_ADD,&m_notify);  
[cpp] view plain copy
 print?
  1. m_notify.cbSize=sizeof NOTIFYICONDATA;  
  2. m_notify.hWnd=this->m_hWnd;    
  3. m_notify.uID=IDR_MAINFRAME;  
  4. m_notify.hIcon=LoadIcon(AfxGetInstanceHandle(),MAKEINTRESOURCE(IDR_MAINFRAME));  
  5. strcpy(m_notify.szTip,"Michael_Chen is a good man");  
  6. m_notify.uCallbackMessage=WM_USER_NOTIFYICON;  
  7. m_notify.uFlags=NIF_ICON|NIF_MESSAGE|NIF_TIP; //OK,下面就是托盘产生了.   
  8. Shell_NotifyIcon(NIM_ADD,&m_notify);  
3.增加自定义的消息处理函数或者重载WindowProc函数,在消息处理函数或WindowProc函数中对自定义的消息进行处理

1)运用自定义消息

i,WM_USER_NOTIFYICON是自定义的消息 在头文件中进行声明;

[cpp] view plaincopyprint?
  1. #define WM_USER_NOTIFYICON WM_USER+1  
[cpp] view plain copy
 print?
  1. #define WM_USER_NOTIFYICON WM_USER+1  
ii,在.h文件中声明消息函数原型
[cpp] view plaincopyprint?
  1. afx_msg LRESULT OnNotifyMsg(WPARAM wparam,LPARAM lparam);  
[cpp] view plain copy
 print?
  1. afx_msg LRESULT OnNotifyMsg(WPARAM wparam,LPARAM lparam);  
iii,在.cpp文件中进行消息映射
[cpp] view plaincopyprint?
  1. BEGIN_MESSAGE_MAP(CNotifyiconDlg, CDialog)  
  2.     //{{AFX_MSG_MAP(CNotifyiconDlg)   
  3.                  ...  
  4.     ON_MESSAGE(WM_USER_NOTIFYICON,OnNotifyMsg)  
  5.     //}}AFX_MSG_MAP   
  6. END_MESSAGE_MAP()  
[cpp] view plain copy
 print?
  1. BEGIN_MESSAGE_MAP(CNotifyiconDlg, CDialog)  
  2.     //{{AFX_MSG_MAP(CNotifyiconDlg)  
  3.                  ...  
  4.     ON_MESSAGE(WM_USER_NOTIFYICON,OnNotifyMsg)  
  5.     //}}AFX_MSG_MAP  
  6. END_MESSAGE_MAP()  
iiii,在.cpp文件中添加自定义的消息响应代码,左键双击弹出应用程序主窗口,右键单击弹出菜单。
[cpp] view plaincopyprint?
  1. LRESULT  CNotifyiconDlg::OnNotifyMsg(WPARAM wparam,LPARAM lparam)  
  2. //wParam接收的是图标的ID,而lParam接收的是鼠标的行为      
  3. {  
  4.     if(wparam!=IDR_MAINFRAME)     
  5.         return    1;     
  6.     switch(lparam)     
  7.     {     
  8.     case  WM_RBUTTONUP://右键起来时弹出快捷菜单,这里只有一个“关闭”     
  9.         {       
  10.             LPPOINT    lpoint=new    tagPOINT;     
  11.             ::GetCursorPos(lpoint);//得到鼠标位置     
  12.             CMenu    menu;     
  13.             menu.CreatePopupMenu();//声明一个弹出式菜单     
  14.             //增加菜单项“关闭”,点击则发送消息WM_DESTROY给主窗口(已     
  15.             //隐藏),将程序结束。      
  16.             menu.AppendMenu(MF_STRING,WM_DESTROY,"关闭");       
  17.             //确定弹出式菜单的位置      
  18.             menu.TrackPopupMenu(TPM_LEFTALIGN,lpoint->x,lpoint->y,this);     
  19.             //资源回收      
  20.             HMENU    hmenu=menu.Detach();     
  21.             menu.DestroyMenu();     
  22.             delete    lpoint;     
  23.         }     
  24.         break;     
  25.     case    WM_LBUTTONDBLCLK://双击左键的处理     
  26.         {     
  27.             this->ShowWindow(SW_SHOW);//简单的显示主窗口完事儿     
  28.         }     
  29.         break;     
  30.     }      
  31.     return 0;  
  32. }  
[cpp] view plain copy
 print?
  1. LRESULT  CNotifyiconDlg::OnNotifyMsg(WPARAM wparam,LPARAM lparam)  
  2. //wParam接收的是图标的ID,而lParam接收的是鼠标的行为     
  3. {  
  4.     if(wparam!=IDR_MAINFRAME)     
  5.         return    1;     
  6.     switch(lparam)     
  7.     {     
  8.     case  WM_RBUTTONUP://右键起来时弹出快捷菜单,这里只有一个“关闭”     
  9.         {       
  10.             LPPOINT    lpoint=new    tagPOINT;     
  11.             ::GetCursorPos(lpoint);//得到鼠标位置     
  12.             CMenu    menu;     
  13.             menu.CreatePopupMenu();//声明一个弹出式菜单     
  14.             //增加菜单项“关闭”,点击则发送消息WM_DESTROY给主窗口(已     
  15.             //隐藏),将程序结束。     
  16.             menu.AppendMenu(MF_STRING,WM_DESTROY,"关闭");       
  17.             //确定弹出式菜单的位置     
  18.             menu.TrackPopupMenu(TPM_LEFTALIGN,lpoint->x,lpoint->y,this);     
  19.             //资源回收     
  20.             HMENU    hmenu=menu.Detach();     
  21.             menu.DestroyMenu();     
  22.             delete    lpoint;     
  23.         }     
  24.         break;     
  25.     case    WM_LBUTTONDBLCLK://双击左键的处理     
  26.         {     
  27.             this->ShowWindow(SW_SHOW);//简单的显示主窗口完事儿     
  28.         }     
  29.         break;     
  30.     }      
  31.     return 0;  
  32. }  

2)重载WindowProc函数

i,重载WindowProc函数(方法略)

ii,在WindowProc中增加的消息相应代码

[cpp] view plaincopyprint?
  1. //WindowProc中增加的代码   
  2. LRESULT CNotifyiconDlg::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)   
  3. {  
  4.     // TODO: Add your specialized code here and/or call the base class  
  5.     switch(message) //判断消息类型  
  6.     {   
  7.     case WM_USER_NOTIFYICON:   
  8.         //如果是用户定义的消息    
  9.         if(lParam==WM_LBUTTONDBLCLK)  
  10.               
  11.         {   
  12.             //鼠标双击时主窗口出现    
  13.             if(AfxGetApp()->m_pMainWnd->IsWindowVisible()) //判断窗口当前状态  
  14.             {  
  15.                 AfxGetApp()->m_pMainWnd->ShowWindow(SW_HIDE); //隐藏窗口  
  16.             }  
  17.             else  
  18.             {  
  19.                 AfxGetApp()->m_pMainWnd->ShowWindow(SW_SHOW); //显示窗口  
  20.             }  
  21.               
  22.         }   
  23.         else if(lParam==WM_RBUTTONDOWN)  
  24.         { //鼠标右键单击弹出选单    
  25.             CMenu menu;   
  26.             menu.LoadMenu(IDR_MENU1); //载入事先定义的选单   
  27.             CMenu *pMenu=menu.GetSubMenu(0);   
  28.             CPoint pos;   
  29.             GetCursorPos(&pos);   
  30.             pMenu->TrackPopupMenu(TPM_LEFTALIGN|TPM_RIGHTBUTTON,pos.x,pos.y,AfxGetMainWnd());   
  31.         }   
  32.         break;   
  33.     case WM_SYSCOMMAND:   
  34.         //如果是系统消息    
  35.         if(wParam==SC_MINIMIZE)  
  36.         {   
  37.             //接收到最小化消息时主窗口隐藏    
  38.             AfxGetApp()->m_pMainWnd->ShowWindow(SW_HIDE);   
  39.             return 0;   
  40.         }   
  41.         if(wParam==SC_CLOSE)  
  42.         {  
  43.             ::Shell_NotifyIcon(NIM_DELETE,&m_notify); //关闭时删除系统托盘图标  
  44.         }  
  45.         break;  
  46.     }  
  47.     return CDialog::WindowProc(message, wParam, lParam);  
  48. }  
[cpp] view plain copy
 print?
  1. //WindowProc中增加的代码  
  2. LRESULT CNotifyiconDlg::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)   
  3. {  
  4.     // TODO: Add your specialized code here and/or call the base class  
  5.     switch(message) //判断消息类型  
  6.     {   
  7.     case WM_USER_NOTIFYICON:   
  8.         //如果是用户定义的消息   
  9.         if(lParam==WM_LBUTTONDBLCLK)  
  10.               
  11.         {   
  12.             //鼠标双击时主窗口出现   
  13.             if(AfxGetApp()->m_pMainWnd->IsWindowVisible()) //判断窗口当前状态  
  14.             {  
  15.                 AfxGetApp()->m_pMainWnd->ShowWindow(SW_HIDE); //隐藏窗口  
  16.             }  
  17.             else  
  18.             {  
  19.                 AfxGetApp()->m_pMainWnd->ShowWindow(SW_SHOW); //显示窗口  
  20.             }  
  21.               
  22.         }   
  23.         else if(lParam==WM_RBUTTONDOWN)  
  24.         { //鼠标右键单击弹出选单   
  25.             CMenu menu;   
  26.             menu.LoadMenu(IDR_MENU1); //载入事先定义的选单   
  27.             CMenu *pMenu=menu.GetSubMenu(0);   
  28.             CPoint pos;   
  29.             GetCursorPos(&pos);   
  30.             pMenu->TrackPopupMenu(TPM_LEFTALIGN|TPM_RIGHTBUTTON,pos.x,pos.y,AfxGetMainWnd());   
  31.         }   
  32.         break;   
  33.     case WM_SYSCOMMAND:   
  34.         //如果是系统消息   
  35.         if(wParam==SC_MINIMIZE)  
  36.         {   
  37.             //接收到最小化消息时主窗口隐藏   
  38.             AfxGetApp()->m_pMainWnd->ShowWindow(SW_HIDE);   
  39.             return 0;   
  40.         }   
  41.         if(wParam==SC_CLOSE)  
  42.         {  
  43.             ::Shell_NotifyIcon(NIM_DELETE,&m_notify); //关闭时删除系统托盘图标  
  44.         }  
  45.         break;  
  46.     }  
  47.     return CDialog::WindowProc(message, wParam, lParam);  
  48. }  


        经过以上的步骤,就编写出了一个简单的可以最小化到托盘的程序。该程序在一启动时,在托盘创立应用程序的图标,最小化程序时隐藏主窗口,双击托盘区的图标时,显示主窗口,右键点击托盘区窗口时,弹出菜单。


这篇关于Windows编程入门(5)MFC程序最小托盘化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/852107

相关文章

Spring WebClient从入门到精通

《SpringWebClient从入门到精通》本文详解SpringWebClient非阻塞响应式特性及优势,涵盖核心API、实战应用与性能优化,对比RestTemplate,为微服务通信提供高效解决... 目录一、WebClient 概述1.1 为什么选择 WebClient?1.2 WebClient 与

Spring Boot 与微服务入门实战详细总结

《SpringBoot与微服务入门实战详细总结》本文讲解SpringBoot框架的核心特性如快速构建、自动配置、零XML与微服务架构的定义、演进及优缺点,涵盖开发环境准备和HelloWorld实战... 目录一、Spring Boot 核心概述二、微服务架构详解1. 微服务的定义与演进2. 微服务的优缺点三

从入门到精通详解LangChain加载HTML内容的全攻略

《从入门到精通详解LangChain加载HTML内容的全攻略》这篇文章主要为大家详细介绍了如何用LangChain优雅地处理HTML内容,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录引言:当大语言模型遇见html一、HTML加载器为什么需要专门的HTML加载器核心加载器对比表二

从入门到进阶讲解Python自动化Playwright实战指南

《从入门到进阶讲解Python自动化Playwright实战指南》Playwright是针对Python语言的纯自动化工具,它可以通过单个API自动执行Chromium,Firefox和WebKit... 目录Playwright 简介核心优势安装步骤观点与案例结合Playwright 核心功能从零开始学习

Windows环境下解决Matplotlib中文字体显示问题的详细教程

《Windows环境下解决Matplotlib中文字体显示问题的详细教程》本文详细介绍了在Windows下解决Matplotlib中文显示问题的方法,包括安装字体、更新缓存、配置文件设置及编码調整,并... 目录引言问题分析解决方案详解1. 检查系统已安装字体2. 手动添加中文字体(以SimHei为例)步骤

golang程序打包成脚本部署到Linux系统方式

《golang程序打包成脚本部署到Linux系统方式》Golang程序通过本地编译(设置GOOS为linux生成无后缀二进制文件),上传至Linux服务器后赋权执行,使用nohup命令实现后台运行,完... 目录本地编译golang程序上传Golang二进制文件到linux服务器总结本地编译Golang程序

使用Docker构建Python Flask程序的详细教程

《使用Docker构建PythonFlask程序的详细教程》在当今的软件开发领域,容器化技术正变得越来越流行,而Docker无疑是其中的佼佼者,本文我们就来聊聊如何使用Docker构建一个简单的Py... 目录引言一、准备工作二、创建 Flask 应用程序三、创建 dockerfile四、构建 Docker

从入门到精通MySQL联合查询

《从入门到精通MySQL联合查询》:本文主要介绍从入门到精通MySQL联合查询,本文通过实例代码给大家介绍的非常详细,需要的朋友可以参考下... 目录摘要1. 多表联合查询时mysql内部原理2. 内连接3. 外连接4. 自连接5. 子查询6. 合并查询7. 插入查询结果摘要前面我们学习了数据库设计时要满

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁