一个托盘程序演示 -闹钟 Alert

From: http://www.vckbase.com/document/viewdoc/?id=996

下载源代码

关键字:托盘 Tray

  想必大家都看见过那些在系统托盘(Tray)中的程序吧,本文就演示了如何创建一个这样的托盘程序Alert。Alert是一个运行在系统托盘中的小闹钟,它在设定的时间后触发,运行屏幕保护程序并播放一首歌曲。这个程序是很久以前写的了,而我现在没有安装VC++了,所以本文的截图都是朋友Leoyin帮我做出来后传给我的(好麻烦,呵呵),在此也对他表示感谢!
  建立一个托盘程序的关键在于Windows API-Shell_NotifyIcon()的使用,这个API的声明包含在ShellAPI.H文件中。请在StdAfx.h中使用“#include <ShellAPI.H>”将其引入工程,别象我当时那么傻-在工程管理器里引入了该文件,导致现在你们在类视图中看见了许多不希望看到的东西。我现在也没有VC++ IDE来修改它了,就这样吧。
  首先,给大家介绍一下相关的函数和结构:Shell_NotifyIcon()的原型和说明(最新版)如下:

BOOL Shell_NotifyIcon(DWORD dwMessage,PNOTIFYICONDATA lpdata);
参数及返回值说明:
dwMessage :[输入参数] 说明要执行的动作。动作的可选值如下:
  • NIM_ADD 增加一个图标到托盘区
  • NIM_DELETE 从托盘区删除一个图标
  • NIM_MODIFY 修改图标
  • NIM_SETFOCUS 将焦点(Focus)返回托盘区。这个消息通常在托盘区图标完成了用户界面下的操作后发出。比如一个托盘图标显示了一个快捷菜单,然后用户按下ESC键了操作,这时使用NIM_SETFOCUS将焦点继续保留在托盘区。该项仅在系统外壳与常用控制DLL( Shlwapi.dll与Comctl32.dll)5.0以上版本才可用
  • NIM_SETVERSION 指定使用特定版本的系统外壳与常用控制DLL。缺省值为0,表示使用Win95方式。该项仅在系统外壳与常用控制DLL 5.0以上版本才可用
lpdata: [输入参数] 一个指向NOTIFYICONDATA结构的指针。
返回值: 成功时函数返回TRUE,否则FALSE。

NOTIFYICONDATA  结构的说明(最新版)如下:
typedef struct _NOTIFYICONDATA {DWORD cbSize;HWND hWnd;UINT uID;UINT uFlags;UINT uCallbackMessage;HICON hIcon;TCHAR szTip[64];DWORD dwState;DWORD dwStateMask;TCHAR szInfo[256];union {UINT uTimeout;UINT uVersion;};TCHAR szInfoTitle[64];DWORD dwInfoFlags;GUID guidItem;
} NOTIFYICONDATA, *PNOTIFYICONDATA;
cbSize以字节计的结构大小,以适应不同版本。
hWnd接收Windows消息的窗口句柄。
uID托盘图标的ID。
uFlags指示结构中的哪些成员包含有效数据,可选值:NIF_ICON, NIF_MESSAGE,NIF_TIP,NIF_STATE,NIF_INFO,NIF_GUID。
uCallbackMessage回调消息ID,由用户自定义。与一个自定义的消息处理函数关联。
hIcon托盘图标的句柄。
szTip托盘图标的提示字符串。
注意:以下数据成员仅限系统外壳与常用控制DLL 5.0及以上版本才有效!
dwState图标的状态:NIS_HIDDEN-隐藏,或NIS_SHAREDICON-可视。
dwStateMask图标状态掩码,用以设置dwState
szInfo气球型提示(Balloon ToolTip)的字符串。
uTimeout以毫秒计的提示显示时间
uVersion确定所依赖的版本。0-Win95,NOTIFYICON_VERSION-Win2000
szInfoTitle气球型提示的标题
dwInfoFlags

设置气球型提示所用的图标(类似MessageBox中所使用的图标):

  • NIIF_ERROR   错误
  • NIIF_INFO    信息
  • NIIF_NONE    没有图标
  • NIIF_WARNING  警告
  • NIIF_ICON_MASK 6.0版本保留
  • NIIF_NOSOUND  限6.0版本,不播放对应的声音
guidItem6.0版本保留

接下来,让我们开始实现这个Alert吧。Alert是一个基于对话框的Win32应用程序,利用AppWizard 创建它吧。然后设计一下这个对话框如图:

 

在AlertDlg.H中加入:
#include < mmsystem.h >          //这个头文件是干什么的记不得了,呵呵#define WM_NOTIFYICON	WM_USER+5  //自定义消息的ID
#define IDI_ICON		0x0005     //图标ID
#define IDT_APPLY		WM_USER+6  //我要使用的定时器Timer的自定义消息ID

  然后为类CAlertDlg添加数据成员m_Interval和自定义成员函数的声明(怎么添加不用我教了吧?)。最后形成CAlertDlg类的声明(已去掉了一些注释)如下:

class CAlertDlg : public CDialog
{
public:~CAlertDlg();CAlertDlg(CWnd* pParent = NULL);enum { IDD = IDD_ALERT_DIALOG };int	m_Interval;		//定时间隔,以分钟为单位protected:virtual void DoDataExchange(CDataExchange* pDX);protected:void ShowMessage(void);  //自定义的定时触发时运行的函数HICON m_hIcon;  //图标句柄virtual BOOL OnInitDialog();afx_msg void OnPaint();afx_msg HCURSOR OnQueryDragIcon();afx_msg void OnClickApply();  //"应用"按钮的点击消息处理函数virtual void OnCancel();  //"退出"按钮的点击消息处理函数afx_msg void OnTimer(UINT nIDEvent);  //定时器消息处理函数afx_msg void OnClickHide();  //"隐藏"按钮的点击消息处理函数//自定义WM_NOTIFYICON消息的处理函数afx_msg void OnNotifyIcon(WPARAM wParam, LPARAM lParam);  afx_msg void OnSysCommand(UINT nID, LPARAM lParam);  //系统消息处理函数afx_msg BOOL OnQueryEndSession();  //Windows关闭消息处理函数DECLARE_MESSAGE_MAP()
};


以下是AlertDlg.CPP中相应函数的实现(还是设置点锚点吧):  

BOOL CAlertDlg::OnInitDialog()
{SetIcon(m_hIcon, TRUE);SetIcon(m_hIcon, FALSE);// 初始化时间间隔CSpinButtonCtrl * pSpin;pSpin = (CSpinButtonCtrl *) GetDlgItem(IDC_SPIN);pSpin->SetRange(5,150);m_Interval = 30;CDialog::OnInitDialog();// 设置定时器,每秒一次心跳SetTimer(1, 1000, NULL);// 将图标放入系统托盘NOTIFYICONDATA nd;nd.cbSize	= sizeof (NOTIFYICONDATA);nd.hWnd	= m_hWnd;nd.uID	= IDI_ICON;nd.uFlags	= NIF_ICON|NIF_MESSAGE|NIF_TIP;nd.uCallbackMessage= WM_NOTIFYICON;nd.hIcon	= m_hIcon;strcpy(nd.szTip, "我的闹钟");Shell_NotifyIcon(NIM_ADD, &nd);return TRUE;
}void CAlertDlg::OnClickApply() 
{// 重置时间间隔UpdateData();OnTimer(IDT_APPLY);
}void CAlertDlg::OnCancel() 
{// 释放定时器KillTimer(1);// 将图标从系统托盘中删除NOTIFYICONDATA nd;nd.cbSize	= sizeof (NOTIFYICONDATA);nd.hWnd	= m_hWnd;Shell_NotifyIcon(NIM_DELETE, &nd);CDialog::OnCancel();
}void CAlertDlg::OnTimer(UINT nIDEvent) 
{static CString	strTemp;static int		Count = 0;COleDateTime	dtTime;// 刷新显示的时间//常在论坛里遇到人问怎么获得当前时间,这相知道了吧?dtTime = COleDateTime::GetCurrentTime();  strTemp.Format("现在时间 %02i:%02i:%02i ",dtTime.GetHour(),dtTime.GetMinute(),dtTime.GetSecond());GetDlgItem(IDC_CURTIME)->SetWindowText(strTemp);// 刷新剩余时间值if (nIDEvent == IDT_APPLY) Count = 0;  //若用户重新定义了时间间隔,则重新开始计时Count++;int RestTime;RestTime = m_Interval - Count/60;if (RestTime <= 0) {Count = 0;ShowMessage();}strTemp.Format("离下次提醒还差 %i 分钟", RestTime);GetDlgItem(IDC_RESTTIME)->SetWindowText(strTemp);	CDialog::OnTimer(nIDEvent);
}//这个ShowMessasge()函数,你们需要自己修改一下,我的代码太死板了
void CAlertDlg::ShowMessage()
{//调用C:\InfoCD\WinaMP\下的WinaMP3播放Music.m3u清单::WinExec("C:\\INFOCD\\WINAMP\\WINAMP.EXE music.m3u",SW_HIDE);MessageBox("您该休息一会儿了......", "休息", MB_SYSTEMMODAL|MB_OK|MB_ICONEXCLAMATION|MB_ICONWARNING);//运行C:\Windows\System下的太空屏保ShellExecute(m_hWnd,"open","C:\\WINDOWS\\SYSTEM\\太空.scr", NULL,NULL,SW_SHOWNORMAL);
}afx_msg void CAlertDlg::OnNotifyIcon(WPARAM wParam, LPARAM lParam)
{// 响应在托盘图标上的单击//wParam中是响应消息的图标ID,lParam中则是Windows的消息if ((wParam == IDI_ICON)&&(lParam == WM_LBUTTONDOWN))ShowWindow(SW_SHOWNORMAL);
}void CAlertDlg::OnSysCommand(UINT nID, LPARAM lParam)
{// 屏蔽最大化(MFC Bug?),将最小化重定向至隐藏窗口if (nID == SC_MAXIMIZE)return;if (nID == SC_MINIMIZE)ShowWindow(SW_HIDE);elseCWnd::OnSysCommand(nID, lParam);
}BOOL CAlertDlg::OnQueryEndSession()
{// 在用户退出Windows时自动退出应用程序CAlertDlg::OnCancel();return TRUE;
}void CAlertDlg::OnClickHide() 
{OnSysCommand(SC_MINIMIZE, 0x0000);return;
}

好了,通过阅读上面这些源代码,我想大家对托盘程序的创建有了一定的认识了吧,这也是本文的根本目的。 只是这个程序有个Bug:在程序退出后图标仍在托盘区显示,要把鼠标在图标上面过一下才会消失。这个Bug记得有人提出过解决办法,好象是要释放图标句柄什么的。可惜我记不得了,这个程序也没有再修改。 


衷心希望大家给我回复,谢谢!

参考文章:
1、MSDN中的相关链接: The Taskbar TrayNot Sample
2、VCKBase里也有不少优秀文章(比如NorthTibet编译的“ 系统托盘编程完全指南 ”)。

在硬盘版的MSDN还会有相应的专题文章,自己搜搜吧。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/406494.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

adam算法效果差原因_冷库制冷效果差原因

冷库常见的现象有冷库温度降不下来和下降缓慢的时候&#xff0c;现在对库温下降缓慢的原因简单分析&#xff1a;1、冷库工程由于隔热或密封性能差&#xff0c;导致冷量损耗大隔热性能差是由于管道、库房隔热墙等的保温层厚度不够&#xff0c;隔热和保温效果不良&#xff0c;它主…

解决hive交互模式退格键乱码

在hive的交互模式下&#xff0c;输入退格、方向键等&#xff0c;出现乱码&#xff0c;可以通过如下方法解决&#xff1a; 1、修改bashrc文件&#xff1a; vi ~/.bashrc 在文件最后添加一行&#xff1a; stty erase ^H。 2、使修改生效&#xff1a; source ~/.bashrc 这样&#…

VS2010删除所有断点时不弹出提示窗口

用VS.NET平台程序时&#xff0c;如何才能在删除所有断点时不弹出提示窗口呢&#xff1f;请看下图&#xff0c;马上明白&#xff1a;

关于Eclipse基本设置(字体大小、项目导入、简单)

1、如果事先你用其他软件写好了一个项目文件夹&#xff0c;导入到Eclipse继续编辑&#xff0c;可以&#xff1a; 打开File→Import→Genral→File System→ Next &#xff0c;然后选择文件夹&#xff08;建议在导入之前&#xff0c;先建立一个工程object&#xff09; 2、字体大…

多通道_人机交互界面的多通道RGB LED驱动设计

现在的产品变得非常智能且彼此相互连接。扬声器、电视、冰箱、机顶盒和烟雾探测器等器件不再只是处在某一位置的工具 - 用户可远程或通过语音唤醒控制它们。这些器件比以前更加智能&#xff0c;这意味着它们还需要更加智慧的人机交互界面。  人机交互的界面可以多种多样&…

转,rsa算法

[编辑] 公钥和私钥的产生 假设Alice想要通过一个不可靠的媒体接收Bob的一条私人訊息。她可以用以下的方式来产生一个公钥和一个私钥&#xff1a;随意选择两个大的质数p和q&#xff0c;p不等于q&#xff0c;计算Npq。 根据歐拉函數&#xff0c;不大于N且与N互质的整数个数为(p-1…

宿舍管理系统项目管理师_2020下半年信息系统项目管理师真题——案例分析(带解析)...

2020下半年信息系统项目管理师真题——案例分析(带解析~强哥版)案例分析这次继续是计算题分析题的分布&#xff0c;但这个题由于理解上有歧义&#xff0c;所以也是大家经常争论的一个题&#xff0c;但无论第一列还是第二列对考分影响不会很大&#xff0c;所以大家还是要耐心等成…

python 计算小于某个数_python 列表寻找满足某个条件的开始索引和结束索引、区间范围...

在使用python列表的时候&#xff0c;我们经常需要找到满足某个条件的数的开始索引和结束索引&#xff0c;即满足某个条件的数的区间范围&#xff0c;本文以寻找绝对值大于等于0且小于等于3的数值区间为例&#xff0c;代码如下所示&#xff1a;这是我在做项目写python代码的时候…

Robotium编写测试用例如何模拟Junit4的BeforeClass和AfterClass方法1 - 条件判断法

Robotium的测试类ActivityInstrumentationTestCase2是继承于Junit3的TestCase类&#xff0c;所以并没有提供Junit4的特性.如网上总结说的 不能通过annotate的方式来识别子类的新特征&#xff0c;如不能实现beforeclass&#xff0c;afterclass等特征。只能通过写setup和teardown…

保姆级!!前端必会Taro入门级教学!!

我是歌谣 放弃很容易 但是坚持一定很酷 微信公众号关注前端小歌谣获取前端知识 本文将介绍 Taro 是如何从一个为了解决小程序开发问题的框架演变成一个多端统一开发框架的。 使用 React 语法来写小程序 谈一谈小程序开发 微信小程序为我们的业务提供了一种新的展现形态&…

js获取用户当前所在城市(ip)

被安利到一个是搜狐家查询ip地址的接口&#xff1a; http://pv.sohu.com/cityjson?ieutf-8 直接打开效果如下&#xff1a; 依照惯例&#xff0c;例子用法给大家放上来&#xff1a; <script src"http://pv.sohu.com/cityjson?ieutf-8"></script> <…

Vue、Element-ui项目中如何使用Iconfont(阿里图标库)

我们使用element-ui、vue开发网站的时候&#xff0c;往往图标是起着很重要的作用。 1.傻瓜式引用&#xff1a;由于这种方法及其简单 省去了很多繁琐的步骤&#xff0c; a.引入项目生成的&#xff0c;图标地址。 b.使用 推荐使用这种&#xff0c;详情请看文档&#xff01;&…

Unity与Android交互

转载自 http://www.narkii.com/club/thread-320335-1.html 这段时间在研究嵌入sdk&#xff0c;途中遇到不少问题和坑&#xff0c;在这就不一一说了。嵌入sdk是比较常见的吧&#xff0c;如接入广告啊&#xff0c; 新浪的分享...... 这些都需要接入sdk的。&#xff08;这些我在这…

深入分析Flex [Bindable] (总结)

[Bindable]大概又是Flex用得最多的元数据了。刚开始用用确实好简单&#xff0c;效率真是没的说。不过这几天用着却碰到了些问题&#xff0c;我自己搜集了些资料&#xff0c;想着有必要在blog里总结一下吧。啥是元数据&#xff08;metadata&#xff09;知道就跳过吧。今天不晓得…

MFC让窗口最前端显示

基于对话框的MFC应用程序&#xff0c;环境&#xff1a;Win XP VC2010。程序界面&#xff1a; 接着为IDC_CHECK1绑定一个BOOL类型的变量m_bShowTopmost&#xff0c;并添加BN_CLICKED事件响应函数&#xff1a; void CWorkHelperDlg::OnBnClickedCheck1() {UpdateData();if(m_bSh…

HTML DOM console.table() 方法

HTML DOM console.table() 方法

百度之后,电子银行还安全么?

用户&#xff1a;百度在哪里啊&#xff1f;百度在哪里&#xff1f;***&#xff1a;百度就在小朋友的心里&#xff01;用户&#xff1a;银行在哪里啊&#xff1f;银行在哪里&#xff1f;***&#xff1a;银行就在我家里&#xff0c;这里没有防盗门啦&#xff01;这里没有监控啦。…