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

Finding the position and dimensions of the Windows system tray

By , 20 Sep 2000
 

Introduction

In my Tray Calendar application I wanted to make the "About" box and Options dialog appear to expand and contract into and out of the system tray. The animation was taken care of by using the DrawAnimatedRects function (see the article "Using the DrawAnimatedRects() function"), but I need to know where the system tray was.

Finding the System Tray

My first thought was to simply use FindWindow using the window name "TrayNotifyWnd", but unfortunately this did not work as planned. Instead I was able to get a handle to "Shell_TrayWnd", and from there work my way down using EnumChildWindows to get a hold of the system tray and the clock window. Subtracting the size of the system clock window from that of the system tray then gave me the working area of the system tray.

The code is shown below:

BOOL CALLBACK FindTrayWnd(HWND hwnd, LPARAM lParam)
{
    // lParam will contain a pointer to a CRect structure that will be used to 
    // store the coordinates of the tray

    TCHAR szClassName[256];
    GetClassName(hwnd, szClassName, 255);

    // Did we find the Main System Tray? If so, then get its size and keep going
    if (_tcscmp(szClassName, _T("TrayNotifyWnd")) == 0)
    {
        CRect *pRect = (CRect*) lParam;
        ::GetWindowRect(hwnd, pRect);
        return TRUE;
    }

    // Did we find the System Clock? If so, then adjust the size of 
    // the rectangle we have and quit (clock will be found after the 
    // system tray)
    if (_tcscmp(szClassName, _T("TrayClockWClass")) == 0)
    {
        CRect *pRect = (CRect*) lParam;
        CRect rectClock;
        ::GetWindowRect(hwnd, rectClock);
        
        // The system clock may be either to the right or above the system 
        // tray area. Adjust accordingly
        if (rectClock.bottom < lpRect->bottom-5)  // 5 = fudge factor
            lpRect->top = rectClock.bottom;
        else
            lpRect->right = rectClock.left;
        return FALSE;
    }
 
    return TRUE;
}
 
void GetTrayWndRect(LPRECT lprect)
{
#define DEFAULT_RECT_WIDTH 150
#define DEFAULT_RECT_HEIGHT 30

    HWND hShellTrayWnd = ::FindWindow(_T("Shell_TrayWnd"), NULL);
    if (hShellTrayWnd)
    {
        ::GetWindowRect(hShellTrayWnd, lprect);
        EnumChildWindows(hShellTrayWnd, FindTrayWnd, (LPARAM)lprect);
        return;
    }
    // OK, we failed to get the rect from the quick hack. Either explorer isn't
    // running or it's a new version of the shell with the window class names
    // changed (how dare Microsoft change these undocumented class names!) So, we
    // try to find out what side of the screen the taskbar is connected to. We
    // know that the system tray is either on the right or the bottom of the
    // taskbar, so we can make a good guess at where to minimize to
    APPBARDATA appBarData;
    appBarData.cbSize=sizeof(appBarData);
    if (SHAppBarMessage(ABM_GETTASKBARPOS,&appBarData))
    {
        // We know the edge the taskbar is connected to, so guess the rect of the
        // system tray. Use various fudge factor to make it look good
        switch(appBarData.uEdge)
        {
        case ABE_LEFT:
        case ABE_RIGHT:
            // We want to minimize to the bottom of the taskbar
            lprect->top    = appBarData.rc.bottom-100;
            lprect->bottom = appBarData.rc.bottom-16;
            lprect->left   = appBarData.rc.left;
            lprect->right  = appBarData.rc.right;
            break;
            
        case ABE_TOP:
        case ABE_BOTTOM:
            // We want to minimize to the right of the taskbar
            lprect->top    = appBarData.rc.top;
            lprect->bottom = appBarData.rc.bottom;
            lprect->left   = appBarData.rc.right-100;
            lprect->right  = appBarData.rc.right-16;
            break;
        }
        return;
    }
    
    // Blimey, we really aren't in luck. It's possible that a third party shell
    // is running instead of explorer. This shell might provide support for the
    // system tray, by providing a Shell_TrayWnd window (which receives the
    // messages for the icons) So, look for a Shell_TrayWnd window and work out
    // the rect from that. Remember that explorer's taskbar is the Shell_TrayWnd,
    // and stretches either the width or the height of the screen. We can't rely
    // on the 3rd party shell's Shell_TrayWnd doing the same, in fact, we can't
    // rely on it being any size. The best we can do is just blindly use the
    // window rect, perhaps limiting the width and height to, say 150 square.
    // Note that if the 3rd party shell supports the same configuraion as
    // explorer (the icons hosted in NotifyTrayWnd, which is a child window of
    // Shell_TrayWnd), we would already have caught it above
    if (hShellTrayWnd)
    {
        ::GetWindowRect(hShellTrayWnd, lprect);
        if (lprect->right - lprect->left > DEFAULT_RECT_WIDTH)
            lprect->left = lprect->right - DEFAULT_RECT_WIDTH;
        if (lprect->bottom - lprect->top > DEFAULT_RECT_HEIGHT)
            lprect->top = lprect->bottom - DEFAULT_RECT_HEIGHT;
        
        return;
    }
    
    // OK. Haven't found a thing. Provide a default rect based on the current work
    // area
    SystemParametersInfo(SPI_GETWORKAREA,0,lprect, 0);
    lprect->left = lprect->right - DEFAULT_RECT_WIDTH;
    lprect->top  = lprect->bottom - DEFAULT_RECT_HEIGHT;
}

Updates

Compensation was made for the case where the system clock is above the system tray, and Matthew Ellis improved the GetTrayWndRect to provide more fallbacks.

License

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

About the Author

Chris Maunder
Founder CodeProject
Canada Canada
Member
Chris is the Co-founder, Administrator, Architect, Chief Editor and Shameless Hack who wrote and runs The Code Project. He's been programming since 1988 while pretending to be, in various guises, an astrophysicist, mathematician, physicist, hydrologist, geomorphologist, defence intelligence researcher and then, when all that got a bit rough on the nerves, a web developer. He is a Microsoft Visual C++ MVP both globally and for Canada locally.
 
His programming experience includes C/C++, C#, SQL, MFC, ASP, ASP.NET, and far, far too much FORTRAN. He has worked on PocketPCs, AIX mainframes, Sun workstations, and a CRAY YMP C90 behemoth but finds notebooks take up less desk space.
 
He dodges, he weaves, and he never gets enough sleep. He is kind to small animals.
 
Chris was born and bred in Australia but splits his time between Toronto and Melbourne, depending on the weather. For relaxation he is into road cycling, snowboarding, rock climbing, and storm chasing.

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   
QuestionHow can i Minimize to system tray at startup?memberMattias19 Jun '01 - 3:00 
I want to minimize my program to the tray at startup if the user selects "start minimized" in the shortcut properties. My current approach for minimizing to the tray is calling ShowWindow(SW_HIDE) in OnSize() (CDialog derived class). Once the program is running, this takes care of all minimize-operations (EVEN the "minimize all windows" feature, which does not send any SYS_COMMAND SC_MINIMIZE at all. That's a problem with other approaches, eg, try it on mIRC, it will minimize to the taskbar). It doesn't have the fancy DrawAnimatedRects() feature, but I can
live with that. The problem is with "start minimized".
 
At program startup, OnSize() is indeed called with the MINIMIZE parameter, just as it should be when "start minimized" is activated. But instead of hiding the window, the call to ShowWindow(SW_HIDE) just makes it pop up again, restored to normal size, and visible. A click on the
minimize button makes it go away into the tray, but this is what I'd like to get rid of.
Any suggestions/pointers to info?

AnswerRe: How can i Minimize to system tray at startup?memberPeter Maria Engeli6 Apr '06 - 7:42 
hi
 
it's all about timing in windows programming.
 
we do similar thing in the hideStatus() function. buy you can do also just a minimize.
user defined messages are all the time helpfull to work with the windows message que.
 

#define WM_POST_CREATE (WM_USER+101)
 

BOOL WatchdogClientDlg::OnInitDialog()
{
 
...
// normaly at the end of this function
PostMessage(WM_POST_CREATE);
return (TRUE);
}
 

LRESULT WatchdogClientDlg::DefWindowProc
(
UINT message,
WPARAM wParam,
LPARAM lParam
)
{
if (message == WM_POST_CREATE)
{
hideStatus();
return (TRUE);
}
return CDialog::DefWindowProc(message, wParam, lParam);
}
 

void WatchdogClientDlg::hideStatus()
{
ShowWindow(SW_HIDE);
}
 

 

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

Permalink | Advertise | Privacy | Mobile
Web01 | 2.6.130523.1 | Last Updated 21 Sep 2000
Article Copyright 2000 by Chris Maunder
Everything else Copyright © CodeProject, 1999-2013
Terms of Use
Layout: fixed | fluid