2 * Copyright (C) 2002 Anders Carlsson <andersca@gnu.org>
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
21 * This is an implementation of the freedesktop.org "system tray" spec,
22 * http://www.freedesktop.org/wiki/Standards/systemtray-spec
29 #include "gtkprivate.h"
30 #include "gtktrayicon.h"
35 #include <X11/Xatom.h>
37 #define SYSTEM_TRAY_REQUEST_DOCK 0
38 #define SYSTEM_TRAY_BEGIN_MESSAGE 1
39 #define SYSTEM_TRAY_CANCEL_MESSAGE 2
41 #define SYSTEM_TRAY_ORIENTATION_HORZ 0
42 #define SYSTEM_TRAY_ORIENTATION_VERT 1
53 struct _GtkTrayIconPrivate
59 Atom system_tray_opcode_atom;
60 Atom orientation_atom;
63 Window manager_window;
64 GdkVisual *manager_visual;
65 gboolean manager_visual_rgba;
67 GtkOrientation orientation;
70 GdkColor warning_color;
71 GdkColor success_color;
74 static void gtk_tray_icon_constructed (GObject *object);
75 static void gtk_tray_icon_dispose (GObject *object);
77 static void gtk_tray_icon_get_property (GObject *object,
82 static void gtk_tray_icon_realize (GtkWidget *widget);
83 static void gtk_tray_icon_style_set (GtkWidget *widget,
84 GtkStyle *previous_style);
85 static gboolean gtk_tray_icon_delete (GtkWidget *widget,
87 static gboolean gtk_tray_icon_expose (GtkWidget *widget,
88 GdkEventExpose *event);
90 static void gtk_tray_icon_clear_manager_window (GtkTrayIcon *icon);
91 static void gtk_tray_icon_update_manager_window (GtkTrayIcon *icon);
92 static void gtk_tray_icon_manager_window_destroyed (GtkTrayIcon *icon);
94 static GdkFilterReturn gtk_tray_icon_manager_filter (GdkXEvent *xevent,
99 G_DEFINE_TYPE (GtkTrayIcon, gtk_tray_icon, GTK_TYPE_PLUG)
102 gtk_tray_icon_class_init (GtkTrayIconClass *class)
104 GObjectClass *gobject_class = (GObjectClass *)class;
105 GtkWidgetClass *widget_class = (GtkWidgetClass *)class;
107 gobject_class->get_property = gtk_tray_icon_get_property;
108 gobject_class->constructed = gtk_tray_icon_constructed;
109 gobject_class->dispose = gtk_tray_icon_dispose;
111 widget_class->realize = gtk_tray_icon_realize;
112 widget_class->style_set = gtk_tray_icon_style_set;
113 widget_class->delete_event = gtk_tray_icon_delete;
114 widget_class->expose_event = gtk_tray_icon_expose;
116 g_object_class_install_property (gobject_class,
118 g_param_spec_enum ("orientation",
120 P_("The orientation of the tray"),
121 GTK_TYPE_ORIENTATION,
122 GTK_ORIENTATION_HORIZONTAL,
123 GTK_PARAM_READABLE));
125 g_object_class_install_property (gobject_class,
127 g_param_spec_boxed ("fg-color",
128 P_("Foreground color"),
129 P_("Foreground color for symbolic icons"),
131 GTK_PARAM_READABLE));
133 g_object_class_install_property (gobject_class,
135 g_param_spec_boxed ("error-color",
137 P_("Error color for symbolic icons"),
139 GTK_PARAM_READABLE));
141 g_object_class_install_property (gobject_class,
143 g_param_spec_boxed ("warning-color",
145 P_("Warning color for symbolic icons"),
147 GTK_PARAM_READABLE));
149 g_object_class_install_property (gobject_class,
151 g_param_spec_boxed ("success-color",
153 P_("Success color for symbolic icons"),
155 GTK_PARAM_READABLE));
157 g_type_class_add_private (class, sizeof (GtkTrayIconPrivate));
161 gtk_tray_icon_init (GtkTrayIcon *icon)
163 icon->priv = G_TYPE_INSTANCE_GET_PRIVATE (icon, GTK_TYPE_TRAY_ICON,
166 icon->priv->stamp = 1;
167 icon->priv->orientation = GTK_ORIENTATION_HORIZONTAL;
168 icon->priv->fg_color.red = 0x0000;
169 icon->priv->fg_color.green = 0x0000;
170 icon->priv->fg_color.blue = 0x0000;
171 icon->priv->error_color.red = 0xcc00;
172 icon->priv->error_color.green = 0x0000;
173 icon->priv->error_color.blue = 0x0000;
174 icon->priv->warning_color.red = 0xf500;
175 icon->priv->warning_color.green = 0x7900;
176 icon->priv->warning_color.blue = 0x3e00;
177 icon->priv->success_color.red = 0x4e00;
178 icon->priv->success_color.green = 0x9a00;
179 icon->priv->success_color.blue = 0x0600;
181 gtk_widget_set_app_paintable (GTK_WIDGET (icon), TRUE);
182 gtk_widget_add_events (GTK_WIDGET (icon), GDK_PROPERTY_CHANGE_MASK);
186 gtk_tray_icon_constructed (GObject *object)
188 /* Do setup that depends on the screen; screen has been set at this point */
190 GtkTrayIcon *icon = GTK_TRAY_ICON (object);
191 GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (object));
192 GdkWindow *root_window = gdk_screen_get_root_window (screen);
193 GdkDisplay *display = gtk_widget_get_display (GTK_WIDGET (object));
194 Display *xdisplay = gdk_x11_display_get_xdisplay (display);
197 g_snprintf (buffer, sizeof (buffer),
198 "_NET_SYSTEM_TRAY_S%d",
199 gdk_screen_get_number (screen));
201 icon->priv->selection_atom = XInternAtom (xdisplay, buffer, False);
203 icon->priv->manager_atom = XInternAtom (xdisplay, "MANAGER", False);
205 icon->priv->system_tray_opcode_atom = XInternAtom (xdisplay,
206 "_NET_SYSTEM_TRAY_OPCODE",
209 icon->priv->orientation_atom = XInternAtom (xdisplay,
210 "_NET_SYSTEM_TRAY_ORIENTATION",
213 icon->priv->visual_atom = XInternAtom (xdisplay,
214 "_NET_SYSTEM_TRAY_VISUAL",
217 icon->priv->colors_atom = XInternAtom (xdisplay,
218 "_NET_SYSTEM_TRAY_COLORS",
221 /* Add a root window filter so that we get changes on MANAGER */
222 gdk_window_add_filter (root_window,
223 gtk_tray_icon_manager_filter, icon);
225 gtk_tray_icon_update_manager_window (icon);
229 gtk_tray_icon_clear_manager_window (GtkTrayIcon *icon)
231 GdkDisplay *display = gtk_widget_get_display (GTK_WIDGET (icon));
233 if (icon->priv->manager_window != None)
237 gdkwin = gdk_window_lookup_for_display (display,
238 icon->priv->manager_window);
240 gdk_window_remove_filter (gdkwin, gtk_tray_icon_manager_filter, icon);
242 icon->priv->manager_window = None;
243 icon->priv->manager_visual = NULL;
248 gtk_tray_icon_dispose (GObject *object)
250 GtkTrayIcon *icon = GTK_TRAY_ICON (object);
251 GtkWidget *widget = GTK_WIDGET (object);
252 GdkWindow *root_window = gdk_screen_get_root_window (gtk_widget_get_screen (widget));
254 gtk_tray_icon_clear_manager_window (icon);
256 gdk_window_remove_filter (root_window, gtk_tray_icon_manager_filter, icon);
260 gtk_tray_icon_get_property (GObject *object,
265 GtkTrayIcon *icon = GTK_TRAY_ICON (object);
269 case PROP_ORIENTATION:
270 g_value_set_enum (value, icon->priv->orientation);
273 g_value_set_boxed (value, &icon->priv->fg_color);
275 case PROP_ERROR_COLOR:
276 g_value_set_boxed (value, &icon->priv->error_color);
278 case PROP_WARNING_COLOR:
279 g_value_set_boxed (value, &icon->priv->warning_color);
281 case PROP_SUCCESS_COLOR:
282 g_value_set_boxed (value, &icon->priv->success_color);
285 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
291 gtk_tray_icon_expose (GtkWidget *widget,
292 GdkEventExpose *event)
294 GtkTrayIcon *icon = GTK_TRAY_ICON (widget);
295 GtkWidget *focus_child;
296 gint border_width, x, y, width, height;
297 gboolean retval = FALSE;
299 if (icon->priv->manager_visual_rgba)
301 /* Clear to transparent */
302 cairo_t *cr = gdk_cairo_create (widget->window);
303 cairo_set_source_rgba (cr, 0, 0, 0, 0);
304 cairo_set_operator (cr, CAIRO_OPERATOR_SOURCE);
305 gdk_cairo_region (cr, event->region);
311 /* Clear to parent-relative pixmap */
312 gdk_window_clear_area (widget->window, event->area.x, event->area.y,
313 event->area.width, event->area.height);
316 if (GTK_WIDGET_CLASS (gtk_tray_icon_parent_class)->expose_event)
317 retval = GTK_WIDGET_CLASS (gtk_tray_icon_parent_class)->expose_event (widget, event);
319 focus_child = GTK_CONTAINER (widget)->focus_child;
320 if (focus_child && gtk_widget_has_focus (focus_child))
322 border_width = GTK_CONTAINER (widget)->border_width;
324 x = widget->allocation.x + border_width;
325 y = widget->allocation.y + border_width;
327 width = widget->allocation.width - 2 * border_width;
328 height = widget->allocation.height - 2 * border_width;
330 gtk_paint_focus (widget->style, widget->window,
331 gtk_widget_get_state (widget),
332 &event->area, widget, "tray_icon",
333 x, y, width, height);
340 gtk_tray_icon_get_orientation_property (GtkTrayIcon *icon)
342 GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon));
343 GdkDisplay *display = gdk_screen_get_display (screen);
344 Display *xdisplay = GDK_DISPLAY_XDISPLAY (display);
356 g_assert (icon->priv->manager_window != None);
358 gdk_error_trap_push ();
360 result = XGetWindowProperty (xdisplay,
361 icon->priv->manager_window,
362 icon->priv->orientation_atom,
365 &type, &format, &nitems,
366 &bytes_after, &(prop.prop_ch));
367 error = gdk_error_trap_pop ();
369 if (error || result != Success)
372 if (type == XA_CARDINAL && nitems == 1 && format == 32)
374 GtkOrientation orientation;
376 orientation = (prop.prop [0] == SYSTEM_TRAY_ORIENTATION_HORZ) ?
377 GTK_ORIENTATION_HORIZONTAL :
378 GTK_ORIENTATION_VERTICAL;
380 if (icon->priv->orientation != orientation)
382 icon->priv->orientation = orientation;
384 g_object_notify (G_OBJECT (icon), "orientation");
393 gtk_tray_icon_get_visual_property (GtkTrayIcon *icon)
395 GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon));
396 GdkDisplay *display = gdk_screen_get_display (screen);
397 Display *xdisplay = GDK_DISPLAY_XDISPLAY (display);
410 g_assert (icon->priv->manager_window != None);
412 gdk_error_trap_push ();
414 result = XGetWindowProperty (xdisplay,
415 icon->priv->manager_window,
416 icon->priv->visual_atom,
419 &type, &format, &nitems,
420 &bytes_after, &(prop.prop_ch));
421 error = gdk_error_trap_pop ();
425 if (!error && result == Success &&
426 type == XA_VISUALID && nitems == 1 && format == 32)
428 VisualID visual_id = prop.prop[0];
429 visual = gdk_x11_screen_lookup_visual (screen, visual_id);
432 icon->priv->manager_visual = visual;
433 icon->priv->manager_visual_rgba = visual != NULL &&
434 (visual->red_prec + visual->blue_prec + visual->green_prec < visual->depth);
436 /* For the background-relative hack we use when we aren't using a real RGBA
437 * visual, we can't be double-buffered */
438 gtk_widget_set_double_buffered (GTK_WIDGET (icon), icon->priv->manager_visual_rgba);
445 gtk_tray_icon_get_colors_property (GtkTrayIcon *icon)
447 GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon));
448 GdkDisplay *display = gdk_screen_get_display (screen);
449 Display *xdisplay = GDK_DISPLAY_XDISPLAY (display);
461 g_assert (icon->priv->manager_window != None);
463 gdk_error_trap_push ();
465 result = XGetWindowProperty (xdisplay,
466 icon->priv->manager_window,
467 icon->priv->colors_atom,
470 &type, &format, &nitems,
471 &bytes_after, &(prop.prop_ch));
472 error = gdk_error_trap_pop ();
474 if (error || result != Success)
477 if (type == XA_CARDINAL && nitems == 12 && format == 32)
481 g_object_freeze_notify (G_OBJECT (icon));
483 color.red = prop.prop[0];
484 color.green = prop.prop[1];
485 color.blue = prop.prop[2];
487 if (!gdk_color_equal (&icon->priv->fg_color, &color))
489 icon->priv->fg_color = color;
491 g_object_notify (G_OBJECT (icon), "fg-color");
494 color.red = prop.prop[3];
495 color.green = prop.prop[4];
496 color.blue = prop.prop[5];
498 if (!gdk_color_equal (&icon->priv->error_color, &color))
500 icon->priv->error_color = color;
502 g_object_notify (G_OBJECT (icon), "error-color");
505 g_object_thaw_notify (G_OBJECT (icon));
507 color.red = prop.prop[6];
508 color.green = prop.prop[7];
509 color.blue = prop.prop[8];
511 if (!gdk_color_equal (&icon->priv->warning_color, &color))
513 icon->priv->warning_color = color;
515 g_object_notify (G_OBJECT (icon), "warning-color");
518 g_object_thaw_notify (G_OBJECT (icon));
520 color.red = prop.prop[9];
521 color.green = prop.prop[10];
522 color.blue = prop.prop[11];
524 if (!gdk_color_equal (&icon->priv->success_color, &color))
526 icon->priv->success_color = color;
528 g_object_notify (G_OBJECT (icon), "success-color");
531 g_object_thaw_notify (G_OBJECT (icon));
538 static GdkFilterReturn
539 gtk_tray_icon_manager_filter (GdkXEvent *xevent,
543 GtkTrayIcon *icon = user_data;
544 XEvent *xev = (XEvent *)xevent;
546 if (xev->xany.type == ClientMessage &&
547 xev->xclient.message_type == icon->priv->manager_atom &&
548 xev->xclient.data.l[1] == icon->priv->selection_atom)
550 GTK_NOTE (PLUGSOCKET,
551 g_print ("GtkStatusIcon %p: tray manager appeared\n", icon));
553 gtk_tray_icon_update_manager_window (icon);
555 else if (xev->xany.window == icon->priv->manager_window)
557 if (xev->xany.type == PropertyNotify &&
558 xev->xproperty.atom == icon->priv->orientation_atom)
560 GTK_NOTE (PLUGSOCKET,
561 g_print ("GtkStatusIcon %p: got PropertyNotify on manager window for orientation atom\n", icon));
563 gtk_tray_icon_get_orientation_property (icon);
565 else if (xev->xany.type == PropertyNotify &&
566 xev->xproperty.atom == icon->priv->colors_atom)
568 GTK_NOTE (PLUGSOCKET,
569 g_print ("GtkStatusIcon %p: got PropertyNotify on manager window for colors atom\n", icon));
571 gtk_tray_icon_get_colors_property (icon);
573 else if (xev->xany.type == DestroyNotify)
575 GTK_NOTE (PLUGSOCKET,
576 g_print ("GtkStatusIcon %p: got DestroyNotify for manager window\n", icon));
578 gtk_tray_icon_manager_window_destroyed (icon);
581 GTK_NOTE (PLUGSOCKET,
582 g_print ("GtkStatusIcon %p: got other message on manager window\n", icon));
585 return GDK_FILTER_CONTINUE;
589 gtk_tray_icon_send_manager_message (GtkTrayIcon *icon,
596 XClientMessageEvent ev;
599 memset (&ev, 0, sizeof (ev));
600 ev.type = ClientMessage;
602 ev.message_type = icon->priv->system_tray_opcode_atom;
604 ev.data.l[0] = gdk_x11_get_server_time (GTK_WIDGET (icon)->window);
605 ev.data.l[1] = message;
606 ev.data.l[2] = data1;
607 ev.data.l[3] = data2;
608 ev.data.l[4] = data3;
610 display = GDK_DISPLAY_XDISPLAY (gtk_widget_get_display (GTK_WIDGET (icon)));
612 gdk_error_trap_push ();
614 icon->priv->manager_window, False, NoEventMask, (XEvent *)&ev);
615 gdk_display_sync (gtk_widget_get_display (GTK_WIDGET (icon)));
616 gdk_error_trap_pop ();
620 gtk_tray_icon_send_dock_request (GtkTrayIcon *icon)
622 GTK_NOTE (PLUGSOCKET,
623 g_print ("GtkStatusIcon %p: sending dock request to manager window %lx\n",
624 icon, (gulong) icon->priv->manager_window));
626 gtk_tray_icon_send_manager_message (icon,
627 SYSTEM_TRAY_REQUEST_DOCK,
628 icon->priv->manager_window,
629 gtk_plug_get_id (GTK_PLUG (icon)),
634 gtk_tray_icon_update_manager_window (GtkTrayIcon *icon)
636 GtkWidget *widget = GTK_WIDGET (icon);
637 GdkScreen *screen = gtk_widget_get_screen (widget);
638 GdkDisplay *display = gdk_screen_get_display (screen);
639 Display *xdisplay = GDK_DISPLAY_XDISPLAY (display);
641 GTK_NOTE (PLUGSOCKET,
642 g_print ("GtkStatusIcon %p: updating tray icon manager window, current manager window: %lx\n",
643 icon, (gulong) icon->priv->manager_window));
645 if (icon->priv->manager_window != None)
648 GTK_NOTE (PLUGSOCKET,
649 g_print ("GtkStatusIcon %p: trying to find manager window\n", icon));
651 XGrabServer (xdisplay);
653 icon->priv->manager_window = XGetSelectionOwner (xdisplay,
654 icon->priv->selection_atom);
656 if (icon->priv->manager_window != None)
657 XSelectInput (xdisplay,
658 icon->priv->manager_window, StructureNotifyMask|PropertyChangeMask);
660 XUngrabServer (xdisplay);
663 if (icon->priv->manager_window != None)
667 GTK_NOTE (PLUGSOCKET,
668 g_print ("GtkStatusIcon %p: is being managed by window %lx\n",
669 icon, (gulong) icon->priv->manager_window));
671 gdkwin = gdk_window_lookup_for_display (display,
672 icon->priv->manager_window);
674 gdk_window_add_filter (gdkwin, gtk_tray_icon_manager_filter, icon);
676 gtk_tray_icon_get_orientation_property (icon);
677 gtk_tray_icon_get_visual_property (icon);
678 gtk_tray_icon_get_colors_property (icon);
680 if (gtk_widget_get_realized (GTK_WIDGET (icon)))
682 if ((icon->priv->manager_visual == NULL &&
683 gtk_widget_get_visual (widget) == gdk_screen_get_system_visual (screen)) ||
684 (icon->priv->manager_visual == gtk_widget_get_visual (widget)))
686 /* Already have the right visual, can just dock
688 gtk_tray_icon_send_dock_request (icon);
692 /* Need to re-realize the widget to get the right visual
694 gtk_widget_hide (widget);
695 gtk_widget_unrealize (widget);
696 gtk_widget_show (widget);
701 GTK_NOTE (PLUGSOCKET,
702 g_print ("GtkStatusIcon %p: no tray manager found\n", icon));
706 gtk_tray_icon_manager_window_destroyed (GtkTrayIcon *icon)
708 g_return_if_fail (icon->priv->manager_window != None);
710 GTK_NOTE (PLUGSOCKET,
711 g_print ("GtkStatusIcon %p: tray manager window destroyed\n", icon));
713 gtk_tray_icon_clear_manager_window (icon);
717 gtk_tray_icon_delete (GtkWidget *widget,
720 #ifdef G_ENABLE_DEBUG
721 GtkTrayIcon *icon = GTK_TRAY_ICON (widget);
724 GTK_NOTE (PLUGSOCKET,
725 g_print ("GtkStatusIcon %p: delete notify, tray manager window %lx\n",
726 icon, (gulong) icon->priv->manager_window));
728 /* A bug in X server versions up to x.org 1.5.0 means that:
729 * XFixesChangeSaveSet(...., SaveSetRoot, SaveSetUnmap) doesn't work properly
730 * and we'll left mapped in a separate toplevel window if the tray is destroyed.
731 * For simplicity just get rid of our X window and start over.
733 gtk_widget_hide (widget);
734 gtk_widget_unrealize (widget);
735 gtk_widget_show (widget);
737 /* Handled it, don't destroy the tray icon */
742 gtk_tray_icon_set_colormap (GtkTrayIcon *icon)
744 GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon));
745 GdkColormap *colormap;
746 GdkVisual *visual = icon->priv->manager_visual;
747 gboolean new_colormap = FALSE;
749 /* To avoid uncertainty about colormaps, _NET_SYSTEM_TRAY_VISUAL is supposed
750 * to be either the screen default visual or a TrueColor visual; ignore it
751 * if it is something else
753 if (visual && visual->type != GDK_VISUAL_TRUE_COLOR)
756 if (visual == NULL || visual == gdk_screen_get_system_visual (screen))
757 colormap = gdk_screen_get_system_colormap (screen);
758 else if (visual == gdk_screen_get_rgb_visual (screen))
759 colormap = gdk_screen_get_rgb_colormap (screen);
760 else if (visual == gdk_screen_get_rgba_visual (screen))
761 colormap = gdk_screen_get_rgba_colormap (screen);
764 colormap = gdk_colormap_new (visual, FALSE);
768 gtk_widget_set_colormap (GTK_WIDGET (icon), colormap);
771 g_object_unref (colormap);
775 gtk_tray_icon_realize (GtkWidget *widget)
777 GtkTrayIcon *icon = GTK_TRAY_ICON (widget);
779 /* Set our colormap before realizing */
780 gtk_tray_icon_set_colormap (icon);
782 GTK_WIDGET_CLASS (gtk_tray_icon_parent_class)->realize (widget);
783 if (icon->priv->manager_visual_rgba)
785 /* Set a transparent background */
786 GdkColor transparent = { 0, 0, 0, 0 }; /* Only pixel=0 matters */
787 gdk_window_set_background (widget->window, &transparent);
791 /* Set a parent-relative background pixmap */
792 gdk_window_set_back_pixmap (widget->window, NULL, TRUE);
795 GTK_NOTE (PLUGSOCKET,
796 g_print ("GtkStatusIcon %p: realized, window: %lx, socket window: %lx\n",
798 (gulong) GDK_WINDOW_XWINDOW (widget->window),
799 GTK_PLUG (icon)->socket_window ?
800 (gulong) GDK_WINDOW_XWINDOW (GTK_PLUG (icon)->socket_window) : 0UL));
802 if (icon->priv->manager_window != None)
803 gtk_tray_icon_send_dock_request (icon);
807 gtk_tray_icon_style_set (GtkWidget *widget,
808 GtkStyle *previous_style)
810 /* The default handler resets the background according to the style. We either
811 * use a transparent background or a parent-relative background and ignore the
812 * style background. So, just don't chain up.
817 _gtk_tray_icon_send_message (GtkTrayIcon *icon,
819 const gchar *message,
825 g_return_val_if_fail (GTK_IS_TRAY_ICON (icon), 0);
826 g_return_val_if_fail (timeout >= 0, 0);
827 g_return_val_if_fail (message != NULL, 0);
829 if (icon->priv->manager_window == None)
833 len = strlen (message);
835 stamp = icon->priv->stamp++;
837 /* Get ready to send the message */
838 gtk_tray_icon_send_manager_message (icon, SYSTEM_TRAY_BEGIN_MESSAGE,
839 (Window)gtk_plug_get_id (GTK_PLUG (icon)),
840 timeout, len, stamp);
842 /* Now to send the actual message */
843 xdisplay = GDK_DISPLAY_XDISPLAY (gtk_widget_get_display (GTK_WIDGET (icon)));
844 gdk_error_trap_push ();
847 XClientMessageEvent ev;
849 memset (&ev, 0, sizeof (ev));
850 ev.type = ClientMessage;
851 ev.window = (Window)gtk_plug_get_id (GTK_PLUG (icon));
853 ev.message_type = XInternAtom (xdisplay,
854 "_NET_SYSTEM_TRAY_MESSAGE_DATA", False);
857 memcpy (&ev.data, message, 20);
863 memcpy (&ev.data, message, len);
867 XSendEvent (xdisplay,
868 icon->priv->manager_window, False,
869 StructureNotifyMask, (XEvent *)&ev);
871 gdk_display_sync (gtk_widget_get_display (GTK_WIDGET (icon)));
872 gdk_error_trap_pop ();
878 _gtk_tray_icon_cancel_message (GtkTrayIcon *icon,
881 g_return_if_fail (GTK_IS_TRAY_ICON (icon));
882 g_return_if_fail (id > 0);
884 gtk_tray_icon_send_manager_message (icon, SYSTEM_TRAY_CANCEL_MESSAGE,
885 (Window)gtk_plug_get_id (GTK_PLUG (icon)),
890 _gtk_tray_icon_new_for_screen (GdkScreen *screen,
893 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
895 return g_object_new (GTK_TYPE_TRAY_ICON,
902 _gtk_tray_icon_new (const gchar *name)
904 return g_object_new (GTK_TYPE_TRAY_ICON,
910 _gtk_tray_icon_get_orientation (GtkTrayIcon *icon)
912 g_return_val_if_fail (GTK_IS_TRAY_ICON (icon), GTK_ORIENTATION_HORIZONTAL);
914 return icon->priv->orientation;
918 #define __GTK_TRAY_ICON_X11_C__
919 #include "gtkaliasdef.c"