2 * Copyright (C) 2004 Mike Hearn, for CodeWeavers
3 * Copyright (C) 2005 Robert Shearman
4 * Copyright (C) 2008 Alexandre Julliard
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
23 #define NONAMELESSUNION
24 #define _WIN32_IE 0x500
29 #include <wine/debug.h>
30 #include <wine/list.h>
32 #include "explorer_private.h"
35 WINE_DEFAULT_DEBUG_CHANNEL(systray
);
37 struct notify_data
/* platform-independent format for NOTIFYICONDATA */
42 UINT uCallbackMessage
;
51 WCHAR szInfoTitle
[64];
54 /* data for the icon bitmap */
61 static int (CDECL
*wine_notify_icon
)(DWORD
,NOTIFYICONDATAW
*);
63 /* an individual systray icon, unpacked from the NOTIFYICONDATA and always in unicode */
67 HICON image
; /* the image to render */
68 HWND owner
; /* the HWND passed in to the Shell_NotifyIcon call */
69 HWND tooltip
; /* Icon tooltip */
70 UINT state
; /* state flags */
71 UINT id
; /* the unique id given by the app */
72 UINT callback_message
;
73 int display
; /* index in display list, or -1 if hidden */
74 WCHAR tiptext
[128]; /* Tooltip text. If empty => tooltip disabled */
75 WCHAR info_text
[256]; /* info balloon text */
76 WCHAR info_title
[64]; /* info balloon title */
77 UINT info_flags
; /* flags for info balloon */
78 UINT info_timeout
; /* timeout for info balloon */
79 HICON info_icon
; /* info balloon icon */
82 static struct list icon_list
= LIST_INIT( icon_list
);
83 static HWND tray_window
;
85 static unsigned int alloc_displayed
;
86 static unsigned int nb_displayed
;
87 static struct icon
**displayed
; /* array of currently displayed icons */
89 static BOOL hide_systray
, enable_shell
;
90 static int icon_cx
, icon_cy
, tray_width
;
92 static struct icon
*balloon_icon
;
93 static HWND balloon_window
;
95 static HWND start_button
;
97 #define MIN_DISPLAYED 8
100 #define BALLOON_CREATE_TIMER 1
101 #define BALLOON_SHOW_TIMER 2
103 #define BALLOON_CREATE_TIMEOUT 2000
104 #define BALLOON_SHOW_MIN_TIMEOUT 10000
105 #define BALLOON_SHOW_MAX_TIMEOUT 30000
107 /* Retrieves icon record by owner window and ID */
108 static struct icon
*get_icon(HWND owner
, UINT id
)
112 /* search for the icon */
113 LIST_FOR_EACH_ENTRY( this, &icon_list
, struct icon
, entry
)
114 if ((this->id
== id
) && (this->owner
== owner
)) return this;
119 static RECT
get_icon_rect( struct icon
*icon
)
123 rect
.right
= tray_width
- icon_cx
* icon
->display
;
124 rect
.left
= rect
.right
- icon_cx
;
126 rect
.bottom
= icon_cy
;
130 static void init_common_controls(void)
132 static BOOL initialized
= FALSE
;
136 INITCOMMONCONTROLSEX init_tooltip
;
138 init_tooltip
.dwSize
= sizeof(INITCOMMONCONTROLSEX
);
139 init_tooltip
.dwICC
= ICC_TAB_CLASSES
|ICC_STANDARD_CLASSES
;
141 InitCommonControlsEx(&init_tooltip
);
146 /* Creates tooltip window for icon. */
147 static void create_tooltip(struct icon
*icon
)
151 init_common_controls();
152 icon
->tooltip
= CreateWindowExW(WS_EX_TOPMOST
, TOOLTIPS_CLASSW
, NULL
,
153 WS_POPUP
| TTS_ALWAYSTIP
,
154 CW_USEDEFAULT
, CW_USEDEFAULT
,
155 CW_USEDEFAULT
, CW_USEDEFAULT
,
156 tray_window
, NULL
, NULL
, NULL
);
158 ZeroMemory(&ti
, sizeof(ti
));
159 ti
.cbSize
= sizeof(TTTOOLINFOW
);
160 ti
.hwnd
= tray_window
;
161 ti
.lpszText
= icon
->tiptext
;
162 if (icon
->display
!= -1) ti
.rect
= get_icon_rect( icon
);
163 SendMessageW(icon
->tooltip
, TTM_ADDTOOLW
, 0, (LPARAM
)&ti
);
166 static void set_balloon_position( struct icon
*icon
)
168 RECT rect
= get_icon_rect( icon
);
171 MapWindowPoints( tray_window
, 0, (POINT
*)&rect
, 2 );
172 pos
.x
= (rect
.left
+ rect
.right
) / 2;
173 pos
.y
= (rect
.top
+ rect
.bottom
) / 2;
174 SendMessageW( balloon_window
, TTM_TRACKPOSITION
, 0, MAKELONG( pos
.x
, pos
.y
));
177 static void balloon_create_timer(void)
181 init_common_controls();
182 balloon_window
= CreateWindowExW( WS_EX_TOPMOST
, TOOLTIPS_CLASSW
, NULL
,
183 WS_POPUP
| TTS_ALWAYSTIP
| TTS_NOPREFIX
| TTS_BALLOON
| TTS_CLOSE
,
184 CW_USEDEFAULT
, CW_USEDEFAULT
, CW_USEDEFAULT
, CW_USEDEFAULT
,
185 tray_window
, NULL
, NULL
, NULL
);
187 memset( &ti
, 0, sizeof(ti
) );
188 ti
.cbSize
= sizeof(TTTOOLINFOW
);
189 ti
.hwnd
= tray_window
;
190 ti
.uFlags
= TTF_TRACK
;
191 ti
.lpszText
= balloon_icon
->info_text
;
192 SendMessageW( balloon_window
, TTM_ADDTOOLW
, 0, (LPARAM
)&ti
);
193 if ((balloon_icon
->info_flags
& NIIF_ICONMASK
) == NIIF_USER
)
194 SendMessageW( balloon_window
, TTM_SETTITLEW
, (WPARAM
)balloon_icon
->info_icon
,
195 (LPARAM
)balloon_icon
->info_title
);
197 SendMessageW( balloon_window
, TTM_SETTITLEW
, balloon_icon
->info_flags
,
198 (LPARAM
)balloon_icon
->info_title
);
199 set_balloon_position( balloon_icon
);
200 SendMessageW( balloon_window
, TTM_TRACKACTIVATE
, TRUE
, (LPARAM
)&ti
);
201 KillTimer( tray_window
, BALLOON_CREATE_TIMER
);
202 SetTimer( tray_window
, BALLOON_SHOW_TIMER
, balloon_icon
->info_timeout
, NULL
);
205 static BOOL
show_balloon( struct icon
*icon
)
207 if (icon
->display
== -1) return FALSE
; /* not displayed */
208 if (!icon
->info_text
[0]) return FALSE
; /* no balloon */
210 SetTimer( tray_window
, BALLOON_CREATE_TIMER
, BALLOON_CREATE_TIMEOUT
, NULL
);
214 static void hide_balloon(void)
216 if (!balloon_icon
) return;
219 KillTimer( tray_window
, BALLOON_SHOW_TIMER
);
220 DestroyWindow( balloon_window
);
223 else KillTimer( tray_window
, BALLOON_CREATE_TIMER
);
227 static void show_next_balloon(void)
231 LIST_FOR_EACH_ENTRY( icon
, &icon_list
, struct icon
, entry
)
232 if (show_balloon( icon
)) break;
235 static void update_balloon( struct icon
*icon
)
237 if (balloon_icon
== icon
)
240 show_balloon( icon
);
242 else if (!balloon_icon
)
244 if (!show_balloon( icon
)) return;
246 if (!balloon_icon
) show_next_balloon();
249 static void balloon_timer(void)
251 if (balloon_icon
) balloon_icon
->info_text
[0] = 0; /* clear text now that balloon has been shown */
256 /* Synchronize tooltip text with tooltip window */
257 static void update_tooltip_text(struct icon
*icon
)
261 ZeroMemory(&ti
, sizeof(ti
));
262 ti
.cbSize
= sizeof(TTTOOLINFOW
);
263 ti
.hwnd
= tray_window
;
264 ti
.lpszText
= icon
->tiptext
;
266 SendMessageW(icon
->tooltip
, TTM_UPDATETIPTEXTW
, 0, (LPARAM
)&ti
);
269 /* synchronize tooltip position with tooltip window */
270 static void update_tooltip_position( struct icon
*icon
)
274 ZeroMemory(&ti
, sizeof(ti
));
275 ti
.cbSize
= sizeof(TTTOOLINFOW
);
276 ti
.hwnd
= tray_window
;
277 if (icon
->display
!= -1) ti
.rect
= get_icon_rect( icon
);
278 SendMessageW( icon
->tooltip
, TTM_NEWTOOLRECTW
, 0, (LPARAM
)&ti
);
279 if (balloon_icon
== icon
) set_balloon_position( icon
);
282 /* find the icon located at a certain point in the tray window */
283 static struct icon
*icon_from_point( int x
, int y
)
285 if (y
< 0 || y
>= icon_cy
) return NULL
;
287 if (x
< 0 || x
>= icon_cx
* nb_displayed
) return NULL
;
288 return displayed
[x
/ icon_cx
];
291 /* invalidate the portion of the tray window that contains the specified icons */
292 static void invalidate_icons( unsigned int start
, unsigned int end
)
296 rect
.left
= tray_width
- (end
+ 1) * icon_cx
;
298 rect
.right
= tray_width
- start
* icon_cx
;
299 rect
.bottom
= icon_cy
;
300 InvalidateRect( tray_window
, &rect
, TRUE
);
303 /* make an icon visible */
304 static BOOL
show_icon(struct icon
*icon
)
306 WINE_TRACE("id=0x%x, hwnd=%p\n", icon
->id
, icon
->owner
);
308 if (icon
->display
!= -1) return TRUE
; /* already displayed */
310 if (nb_displayed
>= alloc_displayed
)
312 unsigned int new_count
= max( alloc_displayed
* 2, 32 );
314 if (displayed
) ptr
= HeapReAlloc( GetProcessHeap(), 0, displayed
, new_count
* sizeof(*ptr
) );
315 else ptr
= HeapAlloc( GetProcessHeap(), 0, new_count
* sizeof(*ptr
) );
316 if (!ptr
) return FALSE
;
318 alloc_displayed
= new_count
;
321 icon
->display
= nb_displayed
;
322 displayed
[nb_displayed
++] = icon
;
323 update_tooltip_position( icon
);
324 invalidate_icons( nb_displayed
-1, nb_displayed
-1 );
326 if (nb_displayed
== 1 && !hide_systray
) ShowWindow( tray_window
, SW_SHOWNA
);
328 create_tooltip(icon
);
329 update_balloon( icon
);
333 /* make an icon invisible */
334 static BOOL
hide_icon(struct icon
*icon
)
338 WINE_TRACE("id=0x%x, hwnd=%p\n", icon
->id
, icon
->owner
);
340 if (icon
->display
== -1) return TRUE
; /* already hidden */
342 assert( nb_displayed
);
343 for (i
= icon
->display
; i
< nb_displayed
- 1; i
++)
345 displayed
[i
] = displayed
[i
+ 1];
346 displayed
[i
]->display
= i
;
347 update_tooltip_position( displayed
[i
] );
350 invalidate_icons( icon
->display
, nb_displayed
);
353 if (!nb_displayed
&& !enable_shell
) ShowWindow( tray_window
, SW_HIDE
);
355 update_balloon( icon
);
356 update_tooltip_position( icon
);
360 /* Modifies an existing icon record */
361 static BOOL
modify_icon( struct icon
*icon
, NOTIFYICONDATAW
*nid
)
363 WINE_TRACE("id=0x%x, hwnd=%p\n", nid
->uID
, nid
->hWnd
);
365 /* demarshal the request from the NID */
368 WINE_WARN("Invalid icon ID (0x%x) for HWND %p\n", nid
->uID
, nid
->hWnd
);
372 if (nid
->uFlags
& NIF_STATE
)
374 icon
->state
= (icon
->state
& ~nid
->dwStateMask
) | (nid
->dwState
& nid
->dwStateMask
);
377 if (nid
->uFlags
& NIF_ICON
)
379 if (icon
->image
) DestroyIcon(icon
->image
);
380 icon
->image
= CopyIcon(nid
->hIcon
);
381 if (icon
->display
!= -1) invalidate_icons( icon
->display
, icon
->display
);
384 if (nid
->uFlags
& NIF_MESSAGE
)
386 icon
->callback_message
= nid
->uCallbackMessage
;
388 if (nid
->uFlags
& NIF_TIP
)
390 lstrcpynW(icon
->tiptext
, nid
->szTip
, sizeof(icon
->tiptext
)/sizeof(WCHAR
));
391 if (icon
->display
!= -1) update_tooltip_text(icon
);
393 if (nid
->uFlags
& NIF_INFO
&& nid
->cbSize
>= NOTIFYICONDATAA_V2_SIZE
)
395 lstrcpynW( icon
->info_text
, nid
->szInfo
, sizeof(icon
->info_text
)/sizeof(WCHAR
) );
396 lstrcpynW( icon
->info_title
, nid
->szInfoTitle
, sizeof(icon
->info_title
)/sizeof(WCHAR
) );
397 icon
->info_flags
= nid
->dwInfoFlags
;
398 icon
->info_timeout
= max(min(nid
->u
.uTimeout
, BALLOON_SHOW_MAX_TIMEOUT
), BALLOON_SHOW_MIN_TIMEOUT
);
399 icon
->info_icon
= nid
->hBalloonIcon
;
400 update_balloon( icon
);
402 if (icon
->state
& NIS_HIDDEN
) hide_icon( icon
);
403 else show_icon( icon
);
407 /* Adds a new icon record to the list */
408 static BOOL
add_icon(NOTIFYICONDATAW
*nid
)
412 WINE_TRACE("id=0x%x, hwnd=%p\n", nid
->uID
, nid
->hWnd
);
414 if ((icon
= get_icon(nid
->hWnd
, nid
->uID
)))
416 WINE_WARN("duplicate tray icon add, buggy app?\n");
420 if (!(icon
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(*icon
))))
422 WINE_ERR("out of memory\n");
426 ZeroMemory(icon
, sizeof(struct icon
));
428 icon
->owner
= nid
->hWnd
;
431 list_add_tail(&icon_list
, &icon
->entry
);
433 return modify_icon( icon
, nid
);
436 /* Deletes tray icon window and icon record */
437 static BOOL
delete_icon(struct icon
*icon
)
440 list_remove(&icon
->entry
);
441 DestroyIcon(icon
->image
);
442 HeapFree(GetProcessHeap(), 0, icon
);
446 /* cleanup icons belonging to a window that has been destroyed */
447 void cleanup_systray_window( HWND hwnd
)
449 struct icon
*icon
, *next
;
451 LIST_FOR_EACH_ENTRY_SAFE( icon
, next
, &icon_list
, struct icon
, entry
)
452 if (icon
->owner
== hwnd
) delete_icon( icon
);
454 if (wine_notify_icon
)
456 NOTIFYICONDATAW nid
= { sizeof(nid
), hwnd
};
457 wine_notify_icon( 0xdead, &nid
);
461 static BOOL
handle_incoming(HWND hwndSource
, COPYDATASTRUCT
*cds
)
463 struct icon
*icon
= NULL
;
464 const struct notify_data
*data
;
468 if (cds
->cbData
< sizeof(*data
)) return FALSE
;
471 nid
.cbSize
= sizeof(nid
);
472 nid
.hWnd
= LongToHandle( data
->hWnd
);
474 nid
.uFlags
= data
->uFlags
;
475 nid
.uCallbackMessage
= data
->uCallbackMessage
;
477 nid
.dwState
= data
->dwState
;
478 nid
.dwStateMask
= data
->dwStateMask
;
479 nid
.u
.uTimeout
= data
->u
.uTimeout
;
480 nid
.dwInfoFlags
= data
->dwInfoFlags
;
481 nid
.guidItem
= data
->guidItem
;
482 lstrcpyW( nid
.szTip
, data
->szTip
);
483 lstrcpyW( nid
.szInfo
, data
->szInfo
);
484 lstrcpyW( nid
.szInfoTitle
, data
->szInfoTitle
);
485 nid
.hBalloonIcon
= 0;
487 /* FIXME: if statement only needed because we don't support interprocess
489 if ((nid
.uFlags
& NIF_ICON
) && cds
->cbData
> sizeof(*data
))
493 const char *buffer
= (const char *)(data
+ 1);
495 cbMaskBits
= (data
->width
* data
->height
+ 15) / 16 * 2;
496 cbColourBits
= (data
->planes
* data
->width
* data
->height
* data
->bpp
+ 15) / 16 * 2;
498 if (cds
->cbData
< sizeof(*data
) + cbMaskBits
+ cbColourBits
)
500 WINE_ERR("buffer underflow\n");
503 nid
.hIcon
= CreateIcon(NULL
, data
->width
, data
->height
, data
->planes
, data
->bpp
,
504 buffer
, buffer
+ cbMaskBits
);
507 /* try forward to x11drv first */
508 if (cds
->dwData
== NIM_ADD
|| !(icon
= get_icon( nid
.hWnd
, nid
.uID
)))
510 if (wine_notify_icon
&& ((ret
= wine_notify_icon( cds
->dwData
, &nid
)) != -1))
512 if (nid
.hIcon
) DestroyIcon( nid
.hIcon
);
521 ret
= add_icon(&nid
);
524 if (icon
) ret
= delete_icon( icon
);
527 if (icon
) ret
= modify_icon( icon
, &nid
);
530 WINE_FIXME("unhandled tray message: %ld\n", cds
->dwData
);
534 if (nid
.hIcon
) DestroyIcon( nid
.hIcon
);
538 static void do_hide_systray(void)
540 SetWindowPos( tray_window
, 0,
541 GetSystemMetrics(SM_XVIRTUALSCREEN
) + GetSystemMetrics(SM_CXVIRTUALSCREEN
),
542 GetSystemMetrics(SM_YVIRTUALSCREEN
) + GetSystemMetrics(SM_CYVIRTUALSCREEN
),
543 0, 0, SWP_NOSIZE
| SWP_NOZORDER
| SWP_NOACTIVATE
);
546 static LRESULT WINAPI
tray_wndproc( HWND hwnd
, UINT msg
, WPARAM wparam
, LPARAM lparam
)
551 return handle_incoming((HWND
)wparam
, (COPYDATASTRUCT
*)lparam
);
553 case WM_DISPLAYCHANGE
:
554 if (hide_systray
) do_hide_systray();
557 tray_width
= GetSystemMetrics( SM_CXSCREEN
);
558 SetWindowPos( tray_window
, 0, 0, GetSystemMetrics( SM_CYSCREEN
) - icon_cy
,
559 tray_width
, icon_cy
, SWP_NOZORDER
| SWP_NOACTIVATE
);
566 case BALLOON_CREATE_TIMER
: balloon_create_timer(); break;
567 case BALLOON_SHOW_TIMER
: balloon_timer(); break;
577 hdc
= BeginPaint( hwnd
, &ps
);
578 for (i
= 0; i
< nb_displayed
; i
++)
580 RECT dummy
, rect
= get_icon_rect( displayed
[i
] );
581 if (IntersectRect( &dummy
, &rect
, &ps
.rcPaint
))
582 DrawIconEx( hdc
, rect
.left
+ ICON_BORDER
, rect
.top
+ ICON_BORDER
, displayed
[i
]->image
,
583 icon_cx
- 2*ICON_BORDER
, icon_cy
- 2*ICON_BORDER
,
584 0, 0, DI_DEFAULTSIZE
|DI_NORMAL
);
586 EndPaint( hwnd
, &ps
);
597 case WM_LBUTTONDBLCLK
:
598 case WM_RBUTTONDBLCLK
:
599 case WM_MBUTTONDBLCLK
:
602 struct icon
*icon
= icon_from_point( (short)LOWORD(lparam
), (short)HIWORD(lparam
) );
605 /* notify the owner hwnd of the message */
606 WINE_TRACE("relaying 0x%x\n", msg
);
609 message
.message
= msg
;
610 message
.wParam
= wparam
;
611 message
.lParam
= lparam
;
612 SendMessageW( icon
->tooltip
, TTM_RELAYEVENT
, 0, (LPARAM
)&message
);
614 if (!PostMessageW( icon
->owner
, icon
->callback_message
, (WPARAM
) icon
->id
, (LPARAM
) msg
) &&
615 GetLastError() == ERROR_INVALID_WINDOW_HANDLE
)
617 WINE_WARN("application window was destroyed without removing "
618 "notification icon, removing automatically\n");
625 /* don't destroy the tray window, just hide it */
626 ShowWindow( hwnd
, SW_HIDE
);
630 if ((HWND
)lparam
== start_button
&& HIWORD(wparam
) == BN_CLICKED
)
634 case WM_INITMENUPOPUP
:
636 return menu_wndproc(hwnd
, msg
, wparam
, lparam
);
639 return DefWindowProcW( hwnd
, msg
, wparam
, lparam
);
644 static void get_system_text_size( const WCHAR
*text
, SIZE
*size
)
646 /* FIXME: Implement BCM_GETIDEALSIZE and use that instead. */
647 HDC hdc
= GetDC( 0 );
649 GetTextExtentPointW(hdc
, text
, lstrlenW(text
), size
);
654 /* this function creates the listener window */
655 void initialize_systray( HMODULE graphics_driver
, BOOL using_root
, BOOL arg_enable_shell
)
658 static const WCHAR classname
[] = {'S','h','e','l','l','_','T','r','a','y','W','n','d',0};
659 static const WCHAR button_class
[] = {'B','u','t','t','o','n',0};
660 WCHAR start_label
[50];
661 SIZE start_text_size
;
663 wine_notify_icon
= (void *)GetProcAddress( graphics_driver
, "wine_notify_icon" );
665 icon_cx
= GetSystemMetrics( SM_CXSMICON
) + 2*ICON_BORDER
;
666 icon_cy
= GetSystemMetrics( SM_CYSMICON
) + 2*ICON_BORDER
;
667 hide_systray
= using_root
;
668 enable_shell
= arg_enable_shell
;
670 /* register the systray listener window class */
671 ZeroMemory(&class, sizeof(class));
672 class.cbSize
= sizeof(class);
673 class.style
= CS_DBLCLKS
| CS_HREDRAW
;
674 class.lpfnWndProc
= tray_wndproc
;
675 class.hInstance
= NULL
;
676 class.hIcon
= LoadIconW(0, (LPCWSTR
)IDI_WINLOGO
);
677 class.hCursor
= LoadCursorW(0, (LPCWSTR
)IDC_ARROW
);
678 class.hbrBackground
= (HBRUSH
) COLOR_WINDOW
;
679 class.lpszClassName
= classname
;
681 if (!RegisterClassExW(&class))
683 WINE_ERR("Could not register SysTray window class\n");
687 tray_width
= GetSystemMetrics( SM_CXSCREEN
);
688 tray_window
= CreateWindowExW( WS_EX_NOACTIVATE
, classname
, NULL
, WS_POPUP
,
689 0, GetSystemMetrics( SM_CYSCREEN
) - icon_cy
,
690 tray_width
, icon_cy
, 0, 0, 0, 0 );
693 WINE_ERR("Could not create tray window\n");
697 LoadStringW( NULL
, IDS_START_LABEL
, start_label
, sizeof(start_label
)/sizeof(WCHAR
) );
699 get_system_text_size( start_label
, &start_text_size
);
701 start_button
= CreateWindowW( button_class
, start_label
, WS_CHILD
|WS_VISIBLE
|BS_PUSHBUTTON
,
702 0, 0, start_text_size
.cx
+ 8, icon_cy
, tray_window
, 0, 0, 0 );
704 if (enable_shell
&& !hide_systray
) ShowWindow( tray_window
, SW_SHOWNA
);
706 if (hide_systray
) do_hide_systray();