|

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.
| You must Sign In to use this message board. |
|
| | Msgs 1 to 25 of 32 (Total in Forum: 32) (Refresh) | FirstPrevNext |
|
|
 |
|
|
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
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
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
|
| Sign In·View Thread·PermaLink | 3.00/5 (2 votes) |
|
|
|
 |
|
|
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.
|
| Sign In·View Thread·PermaLink | 5.00/5 (1 vote) |
|
|
|
 |
|
|
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?
|
| Sign In·View Thread·PermaLink | 2.00/5 (1 vote) |
|
|
|
 |
|
|
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!
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
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
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
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 Fly on your way like an eagle Fly as high as the sun On your wings like an eagle Fly and touch the sun
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
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.
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
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
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
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).
|
| Sign In·View Thread·PermaLink | 4.00/5 (1 vote) |
|
|
|
 |
|
|
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
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
 |
|
|
 |
|
|
Did you ever find out why some icons (MSN Messenger for example) do not draw properly?
I am having the same problem in a similar project and i dont know why some icons dont return a valid hIcon.
|
| Sign In·View Thread·PermaLink | 5.00/5 (2 votes) |
|
|
|
 |
|
|
I am attempting to write a function to determine if my tray icon is still present. It irritates me greatly when explorer crashes and drops my icon and I feel it is a poor solution to simply delete and re-add the icon periodically as has been suggested to me. I am implementing this in a non-MFC application, so I am afraid I cannot use the CProcessData class you have used in your example. I was wondering if you could explain how to obtain the specific TRAYDATA object for each icon without using the CProcessData class. Here is my code:
bool IsTrayIconPresent(UINT uid) { TBBUTTON tb = {0}; TRAYDATA td = {0}; HWND hWnd = ::FindWindow("Shell_TrayWnd", NULL); if(hWnd) { hWnd = ::FindWindowEx(hWnd, NULL, "TrayNotifyWnd", NULL); if(hWnd) { hWnd = ::FindWindowEx(hWnd, NULL, "SysPager", NULL); if(hWnd) { hWnd = ::FindWindowEx(hWnd, NULL, "ToolbarWindow32", NULL); } } }
if(!hWnd) //error could not find Toolbar return false;
int count = (int)::SendMessage(hWnd, TB_BUTTONCOUNT, 0, 0); for(int i=0;i ::SendMessage(hWnd, TB_GETBUTTON, i, (LPARAM)(LPTBBUTTON)&tb); memcpy(&td, (LPCVOID)tb.dwData, sizeof(td)); //does not work
if(td.uID == uid) return true; }
return false; }
This code of course needs to be optimized, but this is just the prototype. Any help would be kindly appreciated.
Thanks, Nate
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
spamna wrote: I am implementing this in a non-MFC application, so I am afraid I cannot use the CProcessData class you have used in your example.
CProcessData is not MFC dependent. You can use it for non-MFC projects too.
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
Thank you for your quick reply. I saw your class name was prefixed with "C" and assumed it was MFC. Here is the resulting code in case anyone else wanted a simple and quick snippet to determine if their tray icon was still in the tray:
//determines if the icon with given uid is present in the tray bool IsTrayIconPresent(UINT uid) { register int i; int count; TBBUTTON tb = {0}; TRAYDATA td = {0}; DWORD dwTrayPid = 0; HANDLE hTrayProc = 0; LPVOID lpData = 0;
HWND hWnd = ::FindWindow("Shell_TrayWnd", NULL); if(hWnd) { hWnd = ::FindWindowEx(hWnd, NULL, "TrayNotifyWnd", NULL); if(hWnd) { hWnd = ::FindWindowEx(hWnd, NULL, "SysPager", NULL); if(hWnd) { hWnd = ::FindWindowEx(hWnd, NULL, "ToolbarWindow32", NULL); } } }
if(!hWnd) //error could not find Toolbar return false;
//get the pid if(GetWindowThreadProcessId(hWnd, &dwTrayPid) == NULL) return false;
//open the corrent process if((hTrayProc = OpenProcess(PROCESS_ALL_ACCESS, FALSE, dwTrayPid)) == NULL) return false;
//virtualalloc if((lpData = VirtualAllocEx(hTrayProc, NULL, sizeof(TBBUTTON), MEM_COMMIT, PAGE_READWRITE)) == NULL) goto cleanup;
count = (int)::SendMessage(hWnd, TB_BUTTONCOUNT, 0, 0); for(i=0;i ::SendMessage(hWnd, TB_GETBUTTON, i, (LPARAM)lpData); //read the tb data ReadProcessMemory(hTrayProc, lpData, (LPVOID)&tb, sizeof(TBBUTTON), NULL); //read the td data ReadProcessMemory(hTrayProc, (LPCVOID)tb.dwData, (LPVOID)&td, sizeof(TRAYDATA), NULL);
if((UINT)td.uID == uid) return true; }
VirtualFreeEx(hTrayProc, lpData, NULL, MEM_RELEASE);
cleanup: CloseHandle(hTrayProc);
return false; }
Thanks for this excellent article. Now I have to figure out how to manipulate buttons on the taskbar = P
Regards, Nate
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
Hello. You do not need to periodically query the Tray area to know if your icon is still present. There's a better way.
Every time explorer.exe crashes, it destroys and recreates the Taskbar. As soon as the Taskbar window has been fully rebuilt, the System broadcasts a special message to all the applications that have requested to receive the notification.
The message is: "TaskbarCreated" (<-- case sensitive)
and you request the notification simply by calling: UINT iMyMsg = RegisterWindowMessage ("TaskbarCreated");
where iMyMsg is your own variable that receives the Sistem-wide value uniquely identifiyng the "TaskbarCreated" message. Then, inside your WndProc() you handle this message like any other. For example:
LRESULT CALLBACK WndProc (HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam) { if (Msg == iMyMsg) { // The Windows Taskbar has just been rebuilt.
// Do your stuff here (like adding a new tray icon). . . . }
// Other WM_*** messages, as usual. switch (Msg) { case ... }
return DefWindowProc (hWnd, Msg, wParam, lParam); }
Being the Taskbar brand new, there's no old tray icon to remove. Simply add a new one.
A few notes: You need to request the "TaskbarCreated" message notification only once. Requesting it multiple times is useless, and always returns the same identifier. If different apps request the notification, they all get returned the same identifier. There's no way (and no mean) to unregister the message once your app terminates. So don't worry about it.
Hope it helps.
Regards, Ciao ciao 
[edit:] I forgot to mention that the "TaskbarCreated" message hasn't changed since Win98. I don't know about Win95.
Ciao ciao
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
Nishant Sivakumar wrote: CProcessData is not MFC dependent. You can use it for non-MFC projects too.
How so?
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
Hi,
GetIconInfo creates bitmaps for the hbmMask and hbmColor members of ICONINFO. The calling application must manage these bitmaps and delete them when they are no longer necessary.
//---- old code --- // int iconindex = 0; ICONINFO iinfo; if(GetIconInfo(tray.hIcon,&iinfo) != 0) { iconindex = m_Image16List.Add(tray.hIcon); }
//--- new code ---- // int iconindex = 0; ICONINFO iinfo; if(GetIconInfo(tray.hIcon,&iinfo) != 0) { iconindex = m_Image16List.Add(tray.hIcon);
if (iinfo.hbmMask != NULL) DeleteObject(iinfo.hbmMask); if (iinfo.hbmColor != NULL) DeleteObject(iinfo.hbmColor); }
Cheers, michi
|
| Sign In·View Thread·PermaLink | 5.00/5 (2 votes) |
|
|
|
 |
|
|
after starting the binary direct this error message will shown:
GetProcessImageFileNameA not in PSAPI.DLL
I use WinXP (without any servicepack) german edition
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
Country Man wrote: GetProcessImageFileNameA not in PSAPI.DLL
I use WinXP (without any servicepack) german edition
You'd need the Unicode build.
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
Hi,
First, excuse my bad english, it's not my native language (I'm French). Then, is it possible, starting with your code, to create a function which could hide/unhide any process' icon in the system tray too (not only in the taskbar)??? How must I do?
Regards.
|
| Sign In·View Thread·PermaLink | 2.00/5 (1 vote) |
|
|
|
 |
|
|
NOTIFYICONDATA tbc; tbc.cbSize=sizeof(NOTIFYICONDATA); tbc.hWnd=m_TifoVec[index].hwnd; tbc.uID=m_TifoVec[index].uID; tbc.uCallbackMessage=m_TifoVec[index].uCallbackMessage; tbc.uFlags=NIF_STATE; tbc.dwState=NIS_HIDDEN; tbc.dwStateMask=NIS_HIDDEN ; Shell_NotifyIcon(NIM_DELETE ,&tbc);
this is one code another is ::SendMessage(hwnd, TB_DELETEBUTTON, index, 0);
try let me know.
|
| Sign In·View Thread·PermaLink | 5.00/5 (1 vote) |
|
|
|
 |
|
|
This is excellent - with this and my Taskbar Sorter[^], I can spend hours rearranging everything to be just the way I like it!
"The way of a fool seems right to him, but a wise man listens to advice" - Proverbs 12:15 (NIV)
|
| Sign In·View Thread·PermaLink | 1.00/5 (1 vote) |
|
|
|
 |
|
|
General News Question Answer Joke Rant Admin
|