Click here to Skip to main content
Licence 
First Posted 28 Mar 2002
Views 334,431
Bookmarked 80 times

Displaying a Notify Icon's Balloon Tool Tip

By | 29 Mar 2002 | Article
Displaying a balloon tool tip for a notification icon.

Sample Image - NotifyBalloon.png

Introduction

This article shows one approach to displaying a balloon tool tip for a notify icon created using the FCL's NotifyIcon class. This relatively new feature of notification icons is not supported by the NotifyIcon class and adding this feature to my own code was not immediately obvious without some creative coding. This is why I am presenting it here. The other reason is that I am hoping somebody can tell me that I'm doing it all wrong and that there is a much more 'correct' solution.

This article also serves as an example of the use of the platform invoke facility. It shows how to create a data structure required by a Win32 API function and how to pass that structure to the API function.

The way I have accomplished the task involves the following two steps:

  1. Using the Win32 API get the handle to the hidden FCL created notify message window.
  2. 2. Use the Win32 API to send the 'balloon tip' message to the notify icon.

A Little Background

Notification icons communicate with their parent applications by sending windows messages to a window supplied when the notify icon is created. The FCL NotifyIcon class does not supply any information about this window (one of the disadvantages of working at a higher level of abstraction). The FCL creates a hidden window for a NotifyIcon and converts the windows messages sent to it into .NET consumable events.

In order to use the Win32 API to communicate with the notification icon you must obtain the Win32 window handle of the window receiving the messages, and also the numeric ID of the icon.

The Solution

Using Spy++ I determined the class name of the hidden window created by the FCL. I then used the Win32 API to find the handle of that window (looking only at the windows owned by my thread). The ID was determined by trial and error! I found that if you have one notify icon created by your app the ID will be 1 (not 0). Once I have the window handle I then call Shell_NotifyIcon() to send it the 'balloon' message. The only trick to this part is defining the data structure required to be sent to this API function.

The Caveats

This solution is a basically a hack for a few reasons. Here are the assumptions I made that may not always be true.

  1. I assumed that calling the Win32 API function GetCurrentThreadId() would return me the correct thread ID.
  2. I assume the name of the window class created by the FCL is fixed. Of course this may not be true in the future.
  3. I assume that the ID of the icon is 1 for the first created icon.
  4. Part of the reason I am posting this code is to maybe get some feedback on how I can reduce the number of assumptions that are made. So let me know if you come up with anything.

Here is the code for the NotifyIcon class that shows the balloon.

public class NotifyIcon
{
    [StructLayout(LayoutKind.Sequential)]
        public struct NotifyIconData
    {
        public System.UInt32 cbSize; // DWORD
        public System.IntPtr hWnd; // HWND
        public System.UInt32 uID; // UINT
        public NotifyFlags uFlags; // UINT
        public System.UInt32 uCallbackMessage; // UINT
        public System.IntPtr hIcon; // HICON
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst=128)]
        public System.String szTip; // char[128]
        public System.UInt32 dwState; // DWORD
        public System.UInt32 dwStateMask; // DWORD
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst=256)]
        public System.String szInfo; // char[256]
        public System.UInt32 uTimeoutOrVersion; // UINT
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst=64)]
        public System.String szInfoTitle; // char[64]
        public System.UInt32 dwInfoFlags; // DWORD
        //GUID guidItem; > IE 6
    }

    public enum NotifyCommand {Add = 0, Modify = 1, Delete = 2, SetFocus = 3, 
                               SetVersion = 4}
    public enum NotifyFlags {Message = 1, Icon = 2, Tip = 4, State = 8, Info = 16, 
                             Guid = 32}

    [DllImport("shell32.Dll")]
    public static extern System.Int32 Shell_NotifyIcon(NotifyCommand cmd, 
                                                       ref NotifyIconData data);

    [DllImport("Kernel32.Dll")]
    public static extern System.UInt32 GetCurrentThreadId();

    public delegate System.Int32 EnumThreadWndProc(System.IntPtr hWnd, 
                                                   System.UInt32 lParam);

    [DllImport("user32.Dll")]
    public static extern System.Int32 EnumThreadWindows(System.UInt32 threadId, 
                                        EnumThreadWndProc callback, 
                                        System.UInt32 param);

    [DllImport("user32.Dll")]
    public static extern System.Int32 GetClassName(System.IntPtr hWnd, 
                                                  System.Text.StringBuilder className,
                                                  System.Int32 maxCount);

    private System.IntPtr m_notifyWindow;
    private bool m_foundNotifyWindow;

    // Win32 Callback Function
    private System.Int32 FindNotifyWindowCallback(System.IntPtr hWnd, 
                                                  System.UInt32 lParam)
    {
        System.Text.StringBuilder buffer = new System.Text.StringBuilder(256);
        GetClassName(hWnd, buffer, buffer.Capacity);

		// but what if this changes?  - anybody got a better idea?
        if(buffer.ToString() == "WindowsForms10.Window.0.app1") 
        {
            m_notifyWindow = hWnd;
            m_foundNotifyWindow = true;
            return 0; // stop searching
        }
        return 1;
    }

    public void ShowBalloon(uint iconId, string title, string text, uint timeout)
    {
        // find notify window
        uint threadId = GetCurrentThreadId();
        EnumThreadWndProc cb = new EnumThreadWndProc(FindNotifyWindowCallback);
        m_foundNotifyWindow = false;
        EnumThreadWindows(threadId, cb, 0);
        if(m_foundNotifyWindow)
        {
            // show the balloon
            NotifyIconData data = new NotifyIconData();
            data.cbSize = (System.UInt32)
                          System.Runtime.InteropServices.Marshal.SizeOf(
                                                                typeof(NotifyIconData));
            data.hWnd = m_notifyWindow;
            data.uID = iconId;
            data.uFlags = NotifyFlags.Info;
            data.uTimeoutOrVersion = 15000;
            data.szInfo = text;
            data.szInfoTitle = title;
            Shell_NotifyIcon(NotifyCommand.Modify, ref data);
        }
    }
}

Here is the code that shows the usage of the class.

private void OnShowBalloon(object sender, System.EventArgs e)
{
    NotifyBalloonDemo.NotifyIcon notifyIcon = new NotifyBalloonDemo.NotifyIcon();
    notifyIcon.ShowBalloon(1, "My Title", "My Text", 15000);
}

So come on people, show me how to do this correctly.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here

About the Author

Joel Matthias

Web Developer

United States United States

Member

Joel is married to MFC/C++ but enjoys a little C# on the side because it's young and sexy.

Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
You must Sign In to use this message board. (secure sign-in)
 
Search this forum  
 FAQ
    Noise  Layout  Per page   
  Refresh
GeneralGood Pinmemberdxlee4:48 23 Oct '09  
GeneralFYI - balloon tips are now built into .net Pinmembermasonmccuskey10:24 5 Jan '09  
GeneralNotify balloon background color PinmemberDaanab1:24 25 Oct '08  
GeneralRe: Notify balloon background color Pinmemberceekays3:25 24 Oct '11  
GeneralThanks. Working. Pinmembertmwinn21:01 1 Oct '08  
QuestionNotify Icon in Windows application in C#, How to make balloon tip show until, I click on ballontip close[X]. Pinmemberanilpkumar16:18 7 Aug '08  
GeneralAdd Title Icon Pinmemberrbrooks4217:52 24 Oct '05  
GeneralOnClick Event PinmemberSteve253:40 14 Sep '05  
QuestionIt works....but Pinmemberlamlai20:25 4 Sep '05  
GeneralGood code, but u can safely remove ur assumptions Pinmembernaveedahmedsiddiqui3:03 2 Sep '05  
GeneralRe: Good code, but u can safely remove ur assumptions PinmemberSGarratt8:59 13 Sep '05  
nice work both of you. as per other post this did not work on XP as posted. Failed to find handle in the enumproc. Tried naveedahmedsiddiqui's reflection method and it worked great. kudos to all. Sorry, CP's messaging doesnt like tabs, so everythings unindented.
 
sg.
 

using System;
using System.Runtime.InteropServices;
using System.Windows.Forms;
using System.Reflection;
 
namespace NotifyBalloonDemo
{
public class NotifyIconBalloon
{
[StructLayout(LayoutKind.Sequential)]
public struct NotifyIconData
{
public System.UInt32 cbSize; // DWORD
public System.IntPtr hWnd; // HWND
public System.UInt32 uID; // UINT
public NotifyFlags uFlags; // UINT
public System.UInt32 uCallbackMessage; // UINT
public System.IntPtr hIcon; // HICON
[MarshalAs(UnmanagedType.ByValTStr, SizeConst=128)]
public System.String szTip; // char[128]
public System.UInt32 dwState; // DWORD
public System.UInt32 dwStateMask; // DWORD
[MarshalAs(UnmanagedType.ByValTStr, SizeConst=256)]
public System.String szInfo; // char[256]
public System.UInt32 uTimeoutOrVersion; // UINT
[MarshalAs(UnmanagedType.ByValTStr, SizeConst=64)]
public System.String szInfoTitle; // char[64]
public System.UInt32 dwInfoFlags; // DWORD
//GUID guidItem; > IE 6
}
 
public enum NotifyCommand {Add = 0, Modify = 1, Delete = 2, SetFocus = 3, SetVersion = 4}
public enum NotifyFlags {Message = 1, Icon = 2, Tip = 4, State = 8, Info = 16, Guid = 32}
 
[DllImport("shell32.Dll")]
public static extern System.Int32 Shell_NotifyIcon(NotifyCommand cmd, ref NotifyIconData data);
 
[DllImport("Kernel32.Dll")]
public static extern System.UInt32 GetCurrentThreadId();
 
public delegate System.Int32 EnumThreadWndProc(System.IntPtr hWnd, System.UInt32 lParam);
 
[DllImport("user32.Dll")]
public static extern System.Int32 EnumThreadWindows(System.UInt32 threadId, EnumThreadWndProc callback, System.UInt32 param);
 
[DllImport("user32.Dll")]
public static extern System.Int32 GetClassName(System.IntPtr hWnd, System.Text.StringBuilder className, System.Int32 maxCount);
 
public void ShowBalloon(NotifyIcon ni, uint iconId, string title, string text, uint timeout)
{
IntPtr ni_handle = GetNotifyIconHandle(ni);
 
if(ni_handle != IntPtr.Zero)
{
// show the balloon
NotifyIconData data = new NotifyIconData();
data.cbSize = (System.UInt32)System.Runtime.InteropServices.Marshal.SizeOf(typeof(NotifyIconData));
data.hWnd = ni_handle;
data.uID = iconId;
data.uFlags = NotifyFlags.Info;
data.uTimeoutOrVersion = 15000;
data.szInfo = text;
data.szInfoTitle = title;
Shell_NotifyIcon(NotifyCommand.Modify, ref data);
}
}
 
public static object GetField(object o, string fieldName, Type type)
{
MemberInfo[] mi = type.GetMember(fieldName,BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.GetField);
if(mi.Length == 0)
return -1;
FieldInfo fi = (FieldInfo)mi[0];
return fi.GetValue(o);
}
public static object GetField(object o, string fieldName)
{
return GetField(o,fieldName,o.GetType());
}
public static int GetNotifyIconID(NotifyIcon ni)
{
const string ID = "id";
return (int)GetField(ni,ID);
}
public static IntPtr GetNotifyIconHandle(NotifyIcon ni)
{
const string WINDOW = "window";
const string HANDLE = "handle";
object obj = GetField(ni, WINDOW);
if(obj == null)
return IntPtr.Zero;
return (IntPtr)GetField(obj,HANDLE,obj.GetType().BaseType);
}
 
}
}

 
SGarratt
Generaloops, forgot to add GetNotifyIconID() PinmemberSGarratt9:10 13 Sep '05  
GeneralDidn't work in XP Pinmemberrpires14:21 23 Aug '05  
GeneralRe: Didn't work in XP Pinmembertayspen9:55 21 Sep '05  
GeneralRe: Didn't work in XP Pinmembermmansf7:34 25 Nov '05  
GeneralRe: Didn't work in XP Pinmembercalvinchang23:37 11 Jan '07  
GeneralRe: Didn't work in XP Pinmembersawo.12:20 9 Oct '07  
GeneralGood Drop-in NotifyIcon Replacement with Balloon Text PinmemberGoalstate19:01 17 Aug '05  
Questionwhat if i want to make it a library PinmemberSecrets11:33 6 Aug '05  
QuestionProblems with API calls ? PinmemberBrett Swift7:43 18 Jul '05  
GeneralDoing the same thing in VB.Net PinmemberEek1:54 16 Sep '04  
GeneralRe: Doing the same thing in VB.Net Pinmemberjosephxxv10:59 7 Dec '04  
GeneralRe: Doing the same thing in VB.Net Pinmemberkajus6:44 23 Dec '04  
GeneralRe: Doing the same thing in VB.Net Pinmemberjosephxxv10:49 23 Dec '04  
GeneralRe: Doing the same thing in VB.Net Pinmemberkajus0:25 27 Dec '04  

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.

Permalink | Advertise | Privacy | Mobile
Web04 | 2.5.120529.1 | Last Updated 30 Mar 2002
Article Copyright 2002 by Joel Matthias
Everything else Copyright © CodeProject, 1999-2012
Terms of Use
Layout: fixed | fluid