Click here to Skip to main content
Click here to Skip to main content

Shell Tray Info - Arrange your system tray icons

By , 26 Jun 2005
 

Overview

The Tray Icon Info application lets you enumerate your system tray icons and rearrange their positions, so that you can have your more frequently used icons positioned to the left most side (or right most depending on your personal preference). I wrote this as I got used to having the MSN Messenger icon on the left most side of the tray and found it annoying and inconvenient when newly added icons pushed it to the right. I had to exit and restart MSN Messenger to reposition it where I wanted. This application simplifies things for me.

Supported OS

This application only works on Windows XP. It may run on Windows 2003 too, but since I wasn't sure and since I didn't have the option to test it out, I have a version check and the program exits if it's a non-XP OS. If anyone's interested, they can comment out the version check and run it in on 2003 - but I have no idea as to whether it'll work or not.

Notes

  • For some tray icons, I am unable to retrieve the icon, so I show a red octagon with a white question mark.
  • Using the toolbar or the menu, you can send a left click, right click or a double click message to the tray icon.
  • You can use the << and >> icons to move the icons around the tray.
  • Copy (Ctrl-C) will copy some textual info to the clipboard (includes both the tool-tip text as well as the owner process path).
  • Double clicking an entry in the list view is equivalent to sending a double-click message.
  • The tray has hidden icons - mostly put there by Explorer. These icons won't have tool-tips.
  • And er, if you are wondering why the toolbar icons look so ghastly, guess who designed them!

Technical notes

The trick used here is to enumerate the buttons of the ToolbarWindow32 window that represents the system tray. The following code is used to locate this window (routine FindWindow/FindWindowEx stuff) :-

HWND FindTrayToolbarWindow()
{
    HWND hWnd = ::FindWindow(_T("Shell_TrayWnd"), NULL);
    if(hWnd)
    {
        hWnd = ::FindWindowEx(hWnd,NULL,_T("TrayNotifyWnd"), NULL);
        if(hWnd)
        {
            hWnd = ::FindWindowEx(hWnd,NULL,_T("SysPager"), NULL);
            if(hWnd)
            {                
                hWnd = ::FindWindowEx(hWnd, NULL,_T("ToolbarWindow32"), NULL);
            }
        }
    }
    return hWnd;
}

Now I retrieve the count of tray icons :-

int count = (int)::SendMessage(m_hTrayWnd, TB_BUTTONCOUNT, 0, 0);

The number won't match the number of visible icons because of some hidden icons inserted by Explorer + the Hide Inactive Icons setting may be enabled.

BTW to retrieve toolbar info for each button, I use my CProcessData class. [CProcessData is a template class that makes it easy to use data allocated in a different process, and is useful when making inter-process SendMessage/PostMessage calls]

The dwData member of each TBBUTTON structure of the toolbar points to an undocumented structure. The first few bytes of the structure are as follows (on XP anyway) :-

struct TRAYDATA
{
    HWND hwnd;                
    UINT uID;                
    UINT uCallbackMessage;    
    DWORD Reserved[2];        
    HICON hIcon;                
};

There's more info, but I am not sure what the rest of it means. Reserved[0] has something to do with the visibility state of an icon when the Hide Inactive Icons setting is enabled, but it's behavior was too sporadic for me to give it a proper meaning and since I didn't really want that info, I didn't bother too much. All my Google searches on this undocumented structure resulted in nothing. It's times like this when you wish Windows provided full source code :-(

Anyway here's the code I use to retrieve the rest of the information I require.

CProcessData<TBBUTTON> data(dwTrayPid);
TBBUTTON tb = {0};
TRAYDATA tray = {0};
TrayItemInfo tifo = {0};

for(int i=0; i<count; i++)
{        
    ::SendMessage(m_hTrayWnd, TB_GETBUTTON, i, (LPARAM)data.GetData());        
    data.ReadData(&tb);            
    data.ReadData<TRAYDATA>(&tray,(LPCVOID)tb.dwData);

    DWORD dwProcessId = 0;
    GetWindowThreadProcessId(tray.hwnd,&dwProcessId);

    tifo.sProcessPath = GetFilenameFromPid(dwProcessId);        

    wchar_t TipChar;
    wchar_t sTip[1024] = {0};
    wchar_t* pTip = (wchar_t*)tb.iString;        

    if(!(tb.fsState&TBSTATE_HIDDEN))
    {            
        int x = 0;
        do 
        {    
            if(x == 1023)
            {
                wcscpy(sTip,L"[ToolTip was either too long or not set]");    
                break;
            }
            data.ReadData<wchar_t>(&TipChar, (LPCVOID)pTip++); 
        }while(sTip[x++] = TipChar);
    }
    else
        wcscpy(sTip,L"[Hidden Icon]");                

    USES_CONVERSION;
    tifo.sTip = W2T(sTip);

    tifo.hwnd = tray.hwnd;
    tifo.uCallbackMessage = tray.uCallbackMessage;
    tifo.uID = tray.uID;

    tifo.bVisible = !(tb.fsState & TBSTATE_HIDDEN);

    int iconindex = 0;
    ICONINFO  iinfo;
    if(GetIconInfo(tray.hIcon,&iinfo) != 0)
    {            
        iconindex = m_Image16List.Add(tray.hIcon);
    }

For the rest of the code, see the included source code zip.

Thanks

History

  • June 21, 2005 : Began work on the app.
  • June 27, 2005 : Published on The Code Project.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

Nish Sivakumar
United States United States
Member
Nish is a real nice guy who has been writing code since 1990 when he first got his hands on an 8088 with 640 KB RAM. Originally from sunny Trivandrum in India, he has been living in various places over the past few years and often thinks it’s time he settled down somewhere.
 
Nish has been a Microsoft Visual C++ MVP since October, 2002 - awfully nice of Microsoft, he thinks. He maintains an MVP tips and tricks web site - www.voidnish.com where you can find a consolidated list of his articles, writings and ideas on VC++, MFC, .NET and C++/CLI. Oh, and you might want to check out his blog on C++/CLI, MFC, .NET and a lot of other stuff - blog.voidnish.com.
 
Nish loves reading Science Fiction, P G Wodehouse and Agatha Christie, and also fancies himself to be a decent writer of sorts. He has authored a romantic comedy Summer Love and Some more Cricket as well as a programming book – Extending MFC applications with the .NET Framework.
 
Nish's latest book C++/CLI in Action published by Manning Publications is now available for purchase. You can read more about the book on his blog.
 
Despite his wife's attempts to get him into cooking, his best effort so far has been a badly done omelette. Some day, he hopes to be a good cook, and to cook a tasty dinner for his wife.

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.
Search this forum  
    Spacing  Noise  Layout  Per page   
QuestionC # version?memberMember 914268823 Nov '12 - 5:06 
Hello, first of all congratulations by this excellent article and my vote 5, I am in a small project with c # where I need this functionality but my knowledge of c++ is poor, reason why it costs to me to translate it, would be possibility of a version c #.
Thanks in advance
SuggestionMaking it work on WoW64 [modified]membermklencke12 Aug '11 - 4:04 
Here's some code to get the process IDs of icons from your 32-bit application running on a 64-bit OS. You can actually read memory of a 64-bit process from a 32-bit process, as long as the addresses are below the 4GB barrier.
 
typedef BOOL (WINAPI *LPFN_ISWOW64PROCESS) (HANDLE, PBOOL);
 
BOOL IsWow64()
{
	static bool isset = false;
    static BOOL bIsWow64 = FALSE;
 
	if (isset) {
		return bIsWow64;
	}
 
    //IsWow64Process is not available on all supported versions of Windows.
    //Use GetModuleHandle to get a handle to the DLL that contains the function
    //and GetProcAddress to get a pointer to the function if available.

    LPFN_ISWOW64PROCESS fnIsWow64Process = (LPFN_ISWOW64PROCESS) GetProcAddress(
        GetModuleHandle(TEXT("kernel32")),"IsWow64Process");
 
    if(NULL != fnIsWow64Process)
    {
        if (!fnIsWow64Process(GetCurrentProcess(),&bIsWow64))
        {
            //TODO handle error?
			return FALSE;
        }
    }
 
	isset = true;
    return bIsWow64;
}
 
typedef struct _TBBUTTON64 {
    int iBitmap;
    int idCommand;
    BYTE fsState;
    BYTE fsStyle;
    BYTE bReserved[6];
	DWORD64 dwData;
    DWORD64 iString;
} TBBUTTON64, NEAR* PTBBUTTON64, *LPTBBUTTON64;
typedef const TBBUTTON64 *LPCTBBUTTON64;
 
bool EnumSystemTray() {	
	bool bFound = false;
	
	// find system tray window
	HWND trayWnd = FindWindow(_T("Shell_TrayWnd"), NULL);
	if (trayWnd) {
		trayWnd = FindWindowEx(trayWnd, NULL,_T("TrayNotifyWnd"), NULL);
		if (trayWnd) {
			trayWnd = FindWindowEx(trayWnd, NULL,_T("SysPager"), NULL);
			if (trayWnd) {				
				trayWnd = FindWindowEx(trayWnd, NULL,_T("ToolbarWindow32"), NULL);
				bFound = true;
			}
		}
	}
 
	ASSERT(bFound);
 
	DWORD dwTrayPid;
	GetWindowThreadProcessId(trayWnd, &dwTrayPid);
 
	int count = (int) SendMessage(trayWnd, TB_BUTTONCOUNT, 0, 0);
 
	HANDLE hProcess = OpenProcess(PROCESS_ALL_ACCESS, FALSE, dwTrayPid);
	if (!hProcess) {
		return true;
	}
 
	BOOL bIsWow64 = IsWow64();
 
	SIZE_T dwSize = bIsWow64 ? sizeof(TBBUTTON64) : sizeof(TBBUTTON);
	LPVOID lpData = VirtualAllocEx(hProcess, NULL, dwSize, MEM_COMMIT, PAGE_READWRITE);
	if (!lpData) {
		return true;
	}
 
	// Loop through all systray icons
	for (int i = 0; i < count; i++) {
		HWND hwnd32;
 
		SendMessage(trayWnd, TB_GETBUTTON, i, (LPARAM)lpData);
		if ( bIsWow64 ) {
			// Try to read memory from 64-bit Explorer process. Hope the address of the traybar data is below 4GB

			TBBUTTON64 tbb;
			if (!ReadProcessMemory(hProcess, lpData, (LPVOID)&tbb, sizeof(TBBUTTON64), NULL)) {
				continue;
			}
 
			DWORD64 hwnd;
 
			// First member of TRAYDATA structure is HWND, so we can just use the address of the struct to read the member
			if (!ReadProcessMemory(hProcess, (LPCVOID)tbb.dwData, (LPVOID)&hwnd, sizeof(DWORD64), NULL)) {
				continue;
			}
 
			// Hope this does not get truncated, but we shouldn't have that many windows
			hwnd32 = (HWND)hwnd;
		} else {
			TBBUTTON tbb;
			if (!ReadProcessMemory(hProcess, lpData, (LPVOID)&tbb, sizeof(TBBUTTON), NULL)) {
				continue;
			}
 
			DWORD32 hwnd;
 
			// First member of TRAYDATA structure is HWND, so we can just use the address of the struct to read the member
			if (!ReadProcessMemory(hProcess, (LPCVOID)tbb.dwData, (LPVOID)&hwnd, sizeof(DWORD32), NULL)) {
				continue;
			}
 
			hwnd32 = (HWND)hwnd;
		}
 
		DWORD dwProcessId = 0;
		GetWindowThreadProcessId(hwnd32, &dwProcessId);
 
		// XXX - DO SOMETHING WITH dwProcessId
	}
 
	VirtualFreeEx(hProcess, lpData, NULL, MEM_RELEASE);
 
	return true;
}

modified on Monday, August 15, 2011 7:32 AM

GeneralMy vote of 5memberswdev.bali16 Feb '11 - 17:31 
I just need the same code!
GeneralMy vote of 5memberzhangfancc19 Jul '10 - 20:02 
good
GeneralNice demo app..memberxsoftwerx15 Feb '10 - 16:40 
Just for s&g I changed it allow Windows 7 -- still works... somewhat.
It enumerates the visible icons but, any on the overflow window are not shown.
GeneralRe: Nice demo app..memberfangyanxiang26 Apr '11 - 16:32 
What to do the code can work in Win7 or vista?
GeneralWrong code and methodmemberkilt14 Sep '09 - 0:18 
The right method is to use internal Shell Structures and code injection in Explorer address space.
Internal Shell Structures had been published on Advanced Win32 newsgroup (news://comp.os.ms-windows.programmer.win32 or http://tinyurl.com/cmhb5g[^])
GeneralVista 32 and Vista 64memberChris Havelick10 Apr '09 - 3:25 
I have created a similar program using VB.Net. I enumerate the system tray icons using the same method. I do this to automatically activate one of the programs associated with a system tray icon.
 
It works in both 32 bit and 64 bits Vista, so I expect that this program will also.
QuestionWhy right click menu are sometime sticky?memberlerognon21 Feb '09 - 9:26 
Sometimes when I use the right click button, the menu corresponding to the tray icon appears but also remains sticky even if I click elsewhere. Anyone having an idea why it is so or how to avoid this behavior?
AnswerRe: Why right click menu are sometime sticky?memberAndreone19 Aug '09 - 3:49 
Before showing the menu, call SetForegroundWindow on the owner window of the icon so the menu can be dismissed.
Maybe there's a better way, but this works in my applications.
QuestionCan we hide a particular icon in the system tray? [modified]memberKishore_Vuppala4 Feb '09 - 17:54 
I need to hide or unhide certain bluetooth applications' icons that are there in system tray based on whether bluetooth is on/off. Can we hide/unhide the system tray icons? It's an urgent requirement!!
 
modified on Thursday, February 5, 2009 1:12 AM

Newsabout "GetIconInfo"memberoneg6615 Jan '09 - 5:26 
why D'Oh! | :doh: the "GetIconInfo" return ZERO in some system's Trayicons that they is exist?
Generalnotesmemberjo0ls16 Dec '08 - 1:50 
Looks like TRAYDATA is NOTIFYICONDATA without the cbsize field.
GETMODULEFILENAMEEX is an easier way to get the normal path from the device path.
Still works in Vista.
I'm doing it with managed code, but the TBBUTTON size varies between x64 and x86 - which is a pain as I can't see a way to lay it out that works with both.
Questionclassmembergq_the_fallen_angel16 Jun '08 - 6:49 
Hello,
 
I'd like to use this code in vb.net but have now idea what to get with it. Unfortunately i have no experience with C++ so if anyone could create a dll from this to be able to use these functions (to get the icons and manipulate them) from vb.net would be really appreciated.
 
Anyways, just needed to delete the windows version checking and works on Vista, too - i tried it.
 
Thanks;
GQ
Questionhide a single tray icon from system traymemberJayapal Chandran15 Mar '08 - 0:27 
hi, i need to hide a particular tray icon from the system tray instead of using the reg key NoTrayItemsDisplay. i am with win32 api C style...
 
Today's Beautiful Moments are
Tomorrow's Beautiful Memories

GeneralSome icons' handle of tray buttions are invalidmembergshine61012 Aug '07 - 20:06 
Some icons' handle of tray buttions are invalid ?
 
When I ran the binary file(ShellTrayInfo.exe), I found that some tray buttons' icon were incorrect, such as MSN(Windows Live Messenger),Kaspersky Antivirus, etc.
 
After I debuged those codes, I found that some icons' handle of tray buttions are invalid.
In those cases, we will fail to call the GetIconInfo API and the error code got by GetLastError is 1402.
 
Is there any difference in those applications?
How can I solve this problem?
 
Hope to receive your reply soon.
 
Thanks in advance!
 

 
Kevin.
GeneralWorks on VistamemberThomassen16 Jun '07 - 0:31 
This app seem to work on Vista. I had to put it in compatibility mode to bypass the version check, but I was able to reposition icons etc.
 
Maybe make the version check forward compatible? Only disallowing older OS's which is known not to work?
QuestionAnyone know of a similar program that just lists the systray icons?memberbadbob00111 May '07 - 6:36 
I'm looking for a commandline utility to just list the icons and tooltips in the system tray and spit it out to a file or stdout so I can parse it. Before I try to tackle modifying this app's code to do what I want, is there already something like what I'm looking for?
 
Thanks!
AnswerRe: Anyone know of a similar program that just lists the systray icons?memberlerognon21 Feb '09 - 9:23 
Try http://exodusdev.com/products/windows-system-tray-scan-utility[^]
GeneralJust what I needed!mvpHans Dietrich9 Apr '07 - 1:14 
After last month's disastrous HD crash, I wanted to start keeping tabs on the temperatures for both of my HD drives. I use HDD Thermometer, a great free tool that shows the temps (one for each HDD) in the tray. Unfortunately, at startup the temps sometimes do not appear next to each other. Your utility is the perfect answer!
 
I agree with 5h17h34d - having Shell Tray Info run at startup would be very nice.
 
Thanks!
 
Best wishes,
Hans
GeneralRe: Just what I needed!mvpNishant Sivakumar9 Apr '07 - 1:54 
I've been meaning to update the source to VC++ 2005, and I guess when I do that I'll also add an installer that'll add this to the startup. Though I could avoid the installer and add an option within the app that'll allow people to specify if they want it to run on startup - it's always good to avoid installers.
 
Regards,
Nish
Nish’s thoughts on MFC, C++/CLI and .NET (my blog)
C++/CLI in Action (*E-Book is out, Print version April 6th*)
Fly on your way like an eagle
Fly as high as the sun
On your wings like an eagle
Fly and touch the sun

GeneralRe: Just what I needed!mvpHans Dietrich9 Apr '07 - 2:00 
Nishant Sivakumar wrote:
I could avoid the installer and add an option within the app that'll allow people to specify if they want it to run on startup

 
I agree - an option within the app would be better.

QuestionHowto make ShellTrayInfo work automatically at boot?member5h17h34d13 Feb '07 - 17:19 
Like this little app except the fact that I must redo it after
every boot.
 
Perhaps I am missing something obvious?
 
Thank you for this little gem of a program for a utility
junky like me!
 
SH
AnswerRe: Howto make ShellTrayInfo work automatically at boot?memberS.H.Bouwhuis18 Jun '07 - 13:14 
My thoughts exactly!
Actually, a simple 'auto alphabetic sort' every 5 minutes or so would be enough.
 
Since the source code is supplied, it should be an easy thing.
I'm currently too busy/lazy to do this, but if there are people out there who REALLY want this, I'll consider it (PM me with request).
GeneralCompiling Error Helpmemberswarup9 Dec '06 - 0:28 
hi guys i am getting some errors while Compiling, can anyone help me out,
 
the errors are
 
Cerror C2552: 'tifo' : non-aggregates cannot be initialized with initializer list
Cerror C2275: 'TRAYDATA' : illegal use of this type as an expression
error C2275: 'wchar_t' : illegal use of this type as an expression
 
these 3 can be solved by using atlbase.h but what about the rest
 
error C2065: 'USES_CONVERSION' : undeclared identifier
error C2065: 'W2T' : undeclared identifier
error C2593: 'operator =' is ambiguous
 
ya one more thing
GetProcessImageFileName is it in psapi, then which is the correct version and can somepne post the 3 files, psapi.h psapi.lib n psapi.dll
because i m getting the error if i comment all the pervious errors
unresolved external symbol _GetProcessImageFileNameW@12
 
Thanks a lot
Swarup

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

Permalink | Advertise | Privacy | Mobile
Web02 | 2.6.130516.1 | Last Updated 27 Jun 2005
Article Copyright 2005 by Nish Sivakumar
Everything else Copyright © CodeProject, 1999-2013
Terms of Use
Layout: fixed | fluid