来个更BT的NotifyIcon支持BalloonTip

简介:
  前两天通过Reflect FX1.1的NotifyIcon实现了一个可以支持BalloonTip的NotifyIconEx,后来发现其实如果不需要截获Balloon Tip上的事件,比如balloon show、click和close,的话。可以使用更加简便且BT的方法来获得Balloon Tip,而更本不用修改FX1.1中的NotifyIcon,使用Reflector代码必须要自己注册一大堆Native方法,忒麻烦。

    使用BT方法也就是访问FX1.1 NotifyIcon的私有成员,获取窗口的Handle,这个参数是控制NotifyIcon的关键。另一个关键参数是NotifyIcon.Icon.Handle,不过这个时pulice的,可以随意取到。看看关于NotifyIcon的前两篇文中,知道NotifyIcon控件所使用的窗口Handle是通过一个内部类来获得的。如果我们能访问到这个内部类,岂不是就能获得最关键的参数Handle了吗?所以我们通过Reflection来获取了NotifyIcon实例中的内部类的窗口Handle,参考代码如下:
private IntPtr GetWindowHandle(NotifyIcon notifyIcon)
{
     if ( notifyIcon ==  null )
    {
         return IntPtr.Zero;
    }

    Type type = notifyIcon.GetType();
    BindingFlags bf = BindingFlags.Instance | BindingFlags.NonPublic;
    FieldInfo fiWindow = type.GetField("window", bf);
     object objWindow = fiWindow.GetValue( this.m_NotifyIcon);

    type = objWindow.GetType().BaseType;
    FieldInfo fiHandle = type.GetField("handle", bf);
    IntPtr handle = (IntPtr)fiHandle.GetValue(objWindow);
     return handle;
}

    这样一来,我们只需要使用5.0版本以上的NOTIFYICONDATA,并使用已有的NotifyIcon实例,就可以让其支持Balloon Tip了 。ShowBalloonTip方法参考如下:
public  void ShowBubbleTip(InfoIcon infoIcon,  string title,  string message,  uint timeout)
{
     if (  this.m_NotifyIcon !=  null )
    {
        nid.hWnd = GetWindowHandle( this.m_NotifyIcon);
        nid.hIcon =  this.m_NotifyIcon.Icon.Handle;
    }

    nid.uTimeoutOrVersion = timeout;
    nid.szInfoTitle = title;
    nid.szInfo = message;
    nid.dwInfoFlags = ( int)infoIcon;

    Shell_NotifyIcon(0x01, nid);
}

    整个NotifyIconDummy加上NOTIFYICONDATA结构及Shell_NotifyIcon方法导入,才130来行代码。而且由于不用重新编译FX1.1的NotifyIcon,似乎少了一些潜在出错的危险。不过这个Dummy有个问题,就是没有办法listen到Balloon Tip上的事件,它们是:Show、Click和Close。看来有得必有失呀

    使用NotifyIconDummy非常简单,只用拖一个到窗体上,把需要显示Balloon Tip的NotifyIcon实例赋值给它就行了:
     NotifyIconDummy.gif

    Demo project and full code download from here, enjoy it.


本文转自博客园鸟食轩的博客,原文链接:http://www.cnblogs.com/birdshome/,如需转载请自行联系原博主。

目录
相关文章
|
4月前
|
存储 机器学习/深度学习 芯片
8086 汇编笔记(十二):int 指令 & 端口 & 直接定址表
8086 汇编笔记(十二):int 指令 & 端口 & 直接定址表
|
6月前
|
前端开发
前端知识笔记(五)———break和continue的区别是什么?(详细)
前端知识笔记(五)———break和continue的区别是什么?(详细)
54 0
|
缓存 网络协议 Linux
【修正版】动图图解!代码执行send成功后,数据就发出去了吗?
【修正版】动图图解!代码执行send成功后,数据就发出去了吗?
116 0
|
前端开发
#yyds干货盘点 【React工作记录九】switch对按钮进行判断操作
#yyds干货盘点 【React工作记录九】switch对按钮进行判断操作
103 0
|
Arthas 前端开发 测试技术
tt 命令的案例演示 | 学习笔记
快速学习 tt 命令的案例演示
tt 命令的案例演示 | 学习笔记
|
域名解析 安全 搜索推荐
DHL
|
算法 Java 程序员
程序员 PK 刘畊宏 | 我用代码实现刘畊宏龙拳
程序员来挑战刘畊宏健身操了,我用代码实现了刘畊宏的龙拳和周大侠,对比原版,来看看代码版本健身操怎么样。
DHL
197 0
程序员 PK 刘畊宏 | 我用代码实现刘畊宏龙拳
|
C++
多玩YY语音的面试题:C++中如何在main()函数之前执行操作?
第一反应main()函数是所有函数执行的开始。但是问题是main()函数执行之前如何执行呢? 联想到MFC里面的 C**App类的theApp对象,其执行顺序就在main函数之前。道理相通,顺理推下,能够想到:如果在main函数之前声明一个类的全局的对象。那么其执行顺序,根据全局对象的生存期和作用域,肯定先于main函数。
183 0