/* gtktrayicon.c * Copyright (C) 2002 Anders Carlsson * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . */ /* * This is an implementation of the freedesktop.org “system tray” spec, * http://www.freedesktop.org/wiki/Standards/systemtray-spec */ #include "config.h" #include #include #include "x11/gdkx.h" #include #include #include "gtkintl.h" #include "gtkprivate.h" #include "gtktrayicon.h" #include "gtktestutils.h" #include "gtkdebug.h" #include "gtktypebuiltins.h" #include "gtkrender.h" #define SYSTEM_TRAY_REQUEST_DOCK 0 #define SYSTEM_TRAY_BEGIN_MESSAGE 1 #define SYSTEM_TRAY_CANCEL_MESSAGE 2 #define SYSTEM_TRAY_ORIENTATION_HORZ 0 #define SYSTEM_TRAY_ORIENTATION_VERT 1 enum { PROP_0, PROP_ORIENTATION, PROP_FG_COLOR, PROP_ERROR_COLOR, PROP_WARNING_COLOR, PROP_SUCCESS_COLOR, PROP_PADDING, PROP_ICON_SIZE }; struct _GtkTrayIconPrivate { guint stamp; Atom selection_atom; Atom manager_atom; Atom system_tray_opcode_atom; Atom orientation_atom; Atom visual_atom; Atom colors_atom; Atom padding_atom; Atom icon_size_atom; Window manager_window; GdkVisual *manager_visual; gboolean manager_visual_rgba; GtkOrientation orientation; GdkRGBA fg_color; GdkRGBA error_color; GdkRGBA warning_color; GdkRGBA success_color; gint padding; gint icon_size; }; static void gtk_tray_icon_constructed (GObject *object); static void gtk_tray_icon_dispose (GObject *object); static void gtk_tray_icon_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static void gtk_tray_icon_realize (GtkWidget *widget); static void gtk_tray_icon_style_updated (GtkWidget *widget); static gboolean gtk_tray_icon_delete (GtkWidget *widget, GdkEventAny *event); static gboolean gtk_tray_icon_draw (GtkWidget *widget, cairo_t *cr); static void gtk_tray_icon_clear_manager_window (GtkTrayIcon *icon); static void gtk_tray_icon_update_manager_window (GtkTrayIcon *icon); static void gtk_tray_icon_manager_window_destroyed (GtkTrayIcon *icon); static GdkFilterReturn gtk_tray_icon_manager_filter (GdkXEvent *xevent, GdkEvent *event, gpointer user_data); G_DEFINE_TYPE_WITH_PRIVATE (GtkTrayIcon, gtk_tray_icon, GTK_TYPE_PLUG) static void gtk_tray_icon_class_init (GtkTrayIconClass *class) { GObjectClass *gobject_class = (GObjectClass *)class; GtkWidgetClass *widget_class = (GtkWidgetClass *)class; gobject_class->get_property = gtk_tray_icon_get_property; gobject_class->constructed = gtk_tray_icon_constructed; gobject_class->dispose = gtk_tray_icon_dispose; widget_class->realize = gtk_tray_icon_realize; widget_class->style_updated = gtk_tray_icon_style_updated; widget_class->delete_event = gtk_tray_icon_delete; widget_class->draw = gtk_tray_icon_draw; g_object_class_install_property (gobject_class, PROP_ORIENTATION, g_param_spec_enum ("orientation", P_("Orientation"), P_("The orientation of the tray"), GTK_TYPE_ORIENTATION, GTK_ORIENTATION_HORIZONTAL, GTK_PARAM_READABLE)); g_object_class_install_property (gobject_class, PROP_FG_COLOR, g_param_spec_boxed ("fg-color", P_("Foreground color"), P_("Foreground color for symbolic icons"), GDK_TYPE_RGBA, GTK_PARAM_READABLE)); g_object_class_install_property (gobject_class, PROP_ERROR_COLOR, g_param_spec_boxed ("error-color", P_("Error color"), P_("Error color for symbolic icons"), GDK_TYPE_RGBA, GTK_PARAM_READABLE)); g_object_class_install_property (gobject_class, PROP_WARNING_COLOR, g_param_spec_boxed ("warning-color", P_("Warning color"), P_("Warning color for symbolic icons"), GDK_TYPE_RGBA, GTK_PARAM_READABLE)); g_object_class_install_property (gobject_class, PROP_SUCCESS_COLOR, g_param_spec_boxed ("success-color", P_("Success color"), P_("Success color for symbolic icons"), GDK_TYPE_RGBA, GTK_PARAM_READABLE)); g_object_class_install_property (gobject_class, PROP_PADDING, g_param_spec_int ("padding", P_("Padding"), P_("Padding that should be put around icons in the tray"), 0, G_MAXINT, 0, GTK_PARAM_READABLE)); g_object_class_install_property (gobject_class, PROP_ICON_SIZE, g_param_spec_int ("icon-size", P_("Icon Size"), P_("The pixel size that icons should be forced to, or zero"), 0, G_MAXINT, 0, GTK_PARAM_READABLE)); } static void gtk_tray_icon_init (GtkTrayIcon *icon) { icon->priv = gtk_tray_icon_get_instance_private (icon); icon->priv->stamp = 1; icon->priv->orientation = GTK_ORIENTATION_HORIZONTAL; icon->priv->fg_color.red = 0.0; icon->priv->fg_color.green = 0.0; icon->priv->fg_color.blue = 0.0; icon->priv->fg_color.alpha = 1.0; icon->priv->error_color.red = 0.7968; icon->priv->error_color.green = 0.0; icon->priv->error_color.blue = 0.0; icon->priv->error_color.alpha = 1.0; icon->priv->warning_color.red = 0.9570; icon->priv->warning_color.green = 0.4726; icon->priv->warning_color.blue = 0.2421; icon->priv->warning_color.alpha = 1.0; icon->priv->success_color.red = 0.3047; icon->priv->success_color.green = 0.6016; icon->priv->success_color.blue = 0.0234; icon->priv->success_color.alpha = 1.0; icon->priv->padding = 0; icon->priv->icon_size = 0; gtk_widget_set_app_paintable (GTK_WIDGET (icon), TRUE); gtk_widget_add_events (GTK_WIDGET (icon), GDK_PROPERTY_CHANGE_MASK); } static void gtk_tray_icon_constructed (GObject *object) { /* Do setup that depends on the screen; screen has been set at this point */ GtkTrayIcon *icon = GTK_TRAY_ICON (object); GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (object)); GdkWindow *root_window = gdk_screen_get_root_window (screen); GdkDisplay *display = gtk_widget_get_display (GTK_WIDGET (object)); Display *xdisplay = gdk_x11_display_get_xdisplay (display); char buffer[256]; G_GNUC_BEGIN_IGNORE_DEPRECATIONS g_snprintf (buffer, sizeof (buffer), "_NET_SYSTEM_TRAY_S%d", gdk_screen_get_number (screen)); G_GNUC_END_IGNORE_DEPRECATIONS icon->priv->selection_atom = XInternAtom (xdisplay, buffer, False); icon->priv->manager_atom = XInternAtom (xdisplay, "MANAGER", False); icon->priv->system_tray_opcode_atom = XInternAtom (xdisplay, "_NET_SYSTEM_TRAY_OPCODE", False); icon->priv->orientation_atom = XInternAtom (xdisplay, "_NET_SYSTEM_TRAY_ORIENTATION", False); icon->priv->visual_atom = XInternAtom (xdisplay, "_NET_SYSTEM_TRAY_VISUAL", False); icon->priv->colors_atom = XInternAtom (xdisplay, "_NET_SYSTEM_TRAY_COLORS", False); icon->priv->padding_atom = XInternAtom (xdisplay, "_NET_SYSTEM_TRAY_PADDING", False); icon->priv->icon_size_atom = XInternAtom (xdisplay, "_NET_SYSTEM_TRAY_ICON_SIZE", False); /* Add a root window filter so that we get changes on MANAGER */ gdk_window_add_filter (root_window, gtk_tray_icon_manager_filter, icon); gtk_tray_icon_update_manager_window (icon); } static void gtk_tray_icon_clear_manager_window (GtkTrayIcon *icon) { GdkDisplay *display = gtk_widget_get_display (GTK_WIDGET (icon)); if (icon->priv->manager_window != None) { GdkWindow *gdkwin; gdkwin = gdk_x11_window_lookup_for_display (display, icon->priv->manager_window); gdk_window_remove_filter (gdkwin, gtk_tray_icon_manager_filter, icon); icon->priv->manager_window = None; icon->priv->manager_visual = NULL; } } static void gtk_tray_icon_dispose (GObject *object) { GtkTrayIcon *icon = GTK_TRAY_ICON (object); GtkWidget *widget = GTK_WIDGET (object); GdkWindow *root_window = gdk_screen_get_root_window (gtk_widget_get_screen (widget)); gtk_tray_icon_clear_manager_window (icon); gdk_window_remove_filter (root_window, gtk_tray_icon_manager_filter, icon); G_OBJECT_CLASS (gtk_tray_icon_parent_class)->dispose (object); } static void gtk_tray_icon_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GtkTrayIcon *icon = GTK_TRAY_ICON (object); switch (prop_id) { case PROP_ORIENTATION: g_value_set_enum (value, icon->priv->orientation); break; case PROP_FG_COLOR: g_value_set_boxed (value, &icon->priv->fg_color); break; case PROP_ERROR_COLOR: g_value_set_boxed (value, &icon->priv->error_color); break; case PROP_WARNING_COLOR: g_value_set_boxed (value, &icon->priv->warning_color); break; case PROP_SUCCESS_COLOR: g_value_set_boxed (value, &icon->priv->success_color); break; case PROP_PADDING: g_value_set_int (value, icon->priv->padding); break; case PROP_ICON_SIZE: g_value_set_int (value, icon->priv->icon_size); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static gboolean gtk_tray_icon_draw (GtkWidget *widget, cairo_t *cr) { GtkTrayIcon *icon = GTK_TRAY_ICON (widget); GtkWidget *focus_child; GdkWindow *window; gint border_width; gboolean retval = FALSE; cairo_surface_t *target; window = gtk_widget_get_window (widget); target = cairo_get_group_target (cr); if (icon->priv->manager_visual_rgba || cairo_surface_get_type (target) != CAIRO_SURFACE_TYPE_XLIB || cairo_xlib_surface_get_drawable (target) != GDK_WINDOW_XID (window)) { /* Clear to transparent */ cairo_set_source_rgba (cr, 0, 0, 0, 0); cairo_set_operator (cr, CAIRO_OPERATOR_SOURCE); cairo_paint (cr); } else { GdkRectangle clip; if (gdk_cairo_get_clip_rectangle (cr, &clip)) { /* Clear to parent-relative pixmap * We need to use direct X access here because GDK doesn't know about * the parent realtive pixmap. */ cairo_surface_flush (target); XClearArea (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window), clip.x, clip.y, clip.width, clip.height, False); cairo_surface_mark_dirty_rectangle (target, clip.x, clip.y, clip.width, clip.height); } } if (GTK_WIDGET_CLASS (gtk_tray_icon_parent_class)->draw) retval = GTK_WIDGET_CLASS (gtk_tray_icon_parent_class)->draw (widget, cr); focus_child = gtk_container_get_focus_child (GTK_CONTAINER (widget)); if (focus_child && gtk_widget_has_visible_focus (focus_child)) { GtkStyleContext *context; border_width = gtk_container_get_border_width (GTK_CONTAINER (widget)); context = gtk_widget_get_style_context (widget); gtk_render_focus (context, cr, border_width, border_width, gtk_widget_get_allocated_width (widget) - 2 * border_width, gtk_widget_get_allocated_height (widget) - 2 * border_width); } return retval; } static void gtk_tray_icon_get_orientation_property (GtkTrayIcon *icon) { GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon)); GdkDisplay *display = gdk_screen_get_display (screen); Display *xdisplay = GDK_DISPLAY_XDISPLAY (display); Atom type; int format; union { gulong *prop; guchar *prop_ch; } prop = { NULL }; gulong nitems; gulong bytes_after; int error, result; g_assert (icon->priv->manager_window != None); gdk_error_trap_push (); type = None; result = XGetWindowProperty (xdisplay, icon->priv->manager_window, icon->priv->orientation_atom, 0, G_MAXLONG, FALSE, XA_CARDINAL, &type, &format, &nitems, &bytes_after, &(prop.prop_ch)); error = gdk_error_trap_pop (); if (error || result != Success) return; if (type == XA_CARDINAL && nitems == 1 && format == 32) { GtkOrientation orientation; orientation = (prop.prop [0] == SYSTEM_TRAY_ORIENTATION_HORZ) ? GTK_ORIENTATION_HORIZONTAL : GTK_ORIENTATION_VERTICAL; if (icon->priv->orientation != orientation) { icon->priv->orientation = orientation; g_object_notify (G_OBJECT (icon), "orientation"); } } if (type != None) XFree (prop.prop); } static void gtk_tray_icon_get_visual_property (GtkTrayIcon *icon) { GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon)); GdkDisplay *display = gdk_screen_get_display (screen); Display *xdisplay = GDK_DISPLAY_XDISPLAY (display); Atom type; int format; union { gulong *prop; guchar *prop_ch; } prop = { NULL }; gulong nitems; gulong bytes_after; int error, result; g_assert (icon->priv->manager_window != None); gdk_error_trap_push (); type = None; result = XGetWindowProperty (xdisplay, icon->priv->manager_window, icon->priv->visual_atom, 0, G_MAXLONG, FALSE, XA_VISUALID, &type, &format, &nitems, &bytes_after, &(prop.prop_ch)); error = gdk_error_trap_pop (); if (!error && result == Success && type == XA_VISUALID && nitems == 1 && format == 32) { VisualID visual_id; GdkVisual *visual; gint red_prec, green_prec, blue_prec; visual_id = prop.prop[0]; visual = gdk_x11_screen_lookup_visual (screen, visual_id); gdk_visual_get_red_pixel_details (visual, NULL, NULL, &red_prec); gdk_visual_get_green_pixel_details (visual, NULL, NULL, &green_prec); gdk_visual_get_blue_pixel_details (visual, NULL, NULL, &blue_prec); icon->priv->manager_visual = visual; icon->priv->manager_visual_rgba = (red_prec + blue_prec + green_prec < gdk_visual_get_depth (visual)); } else { icon->priv->manager_visual = NULL; icon->priv->manager_visual_rgba = FALSE; } G_GNUC_BEGIN_IGNORE_DEPRECATIONS /* For the background-relative hack we use when we aren't * using a real RGBA visual, we can't be double-buffered */ gtk_widget_set_double_buffered (GTK_WIDGET (icon), icon->priv->manager_visual_rgba); G_GNUC_END_IGNORE_DEPRECATIONS if (type != None) XFree (prop.prop); } static void gtk_tray_icon_get_colors_property (GtkTrayIcon *icon) { GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon)); GdkDisplay *display = gdk_screen_get_display (screen); Display *xdisplay = GDK_DISPLAY_XDISPLAY (display); Atom type; int format; union { gulong *prop; guchar *prop_ch; } prop = { NULL }; gulong nitems; gulong bytes_after; int error, result; g_assert (icon->priv->manager_window != None); gdk_error_trap_push (); type = None; result = XGetWindowProperty (xdisplay, icon->priv->manager_window, icon->priv->colors_atom, 0, G_MAXLONG, FALSE, XA_CARDINAL, &type, &format, &nitems, &bytes_after, &(prop.prop_ch)); error = gdk_error_trap_pop (); if (error || result != Success) return; if (type == XA_CARDINAL && nitems == 12 && format == 32) { GdkRGBA color; color.alpha = 1.0; g_object_freeze_notify (G_OBJECT (icon)); color.red = prop.prop[0] / 65535.0; color.green = prop.prop[1] / 65535.0; color.blue = prop.prop[2] / 65535.0; if (!gdk_rgba_equal (&icon->priv->fg_color, &color)) { icon->priv->fg_color = color; g_object_notify (G_OBJECT (icon), "fg-color"); } color.red = prop.prop[3] / 65535.0; color.green = prop.prop[4] / 65535.0; color.blue = prop.prop[5] / 65535.0; if (!gdk_rgba_equal (&icon->priv->error_color, &color)) { icon->priv->error_color = color; g_object_notify (G_OBJECT (icon), "error-color"); } color.red = prop.prop[6] / 65535.0; color.green = prop.prop[7] / 65535.0; color.blue = prop.prop[8] / 65535.0; if (!gdk_rgba_equal (&icon->priv->warning_color, &color)) { icon->priv->warning_color = color; g_object_notify (G_OBJECT (icon), "warning-color"); } color.red = prop.prop[9] / 65535.0; color.green = prop.prop[10] / 65535.0; color.blue = prop.prop[11] / 65535.0; if (!gdk_rgba_equal (&icon->priv->success_color, &color)) { icon->priv->success_color = color; g_object_notify (G_OBJECT (icon), "success-color"); } g_object_thaw_notify (G_OBJECT (icon)); } if (type != None) XFree (prop.prop); } static void gtk_tray_icon_get_padding_property (GtkTrayIcon *icon) { GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon)); GdkDisplay *display = gdk_screen_get_display (screen); Display *xdisplay = GDK_DISPLAY_XDISPLAY (display); Atom type; int format; union { gulong *prop; guchar *prop_ch; } prop = { NULL }; gulong nitems; gulong bytes_after; int error, result; g_assert (icon->priv->manager_window != None); gdk_error_trap_push (); type = None; result = XGetWindowProperty (xdisplay, icon->priv->manager_window, icon->priv->padding_atom, 0, G_MAXLONG, FALSE, XA_CARDINAL, &type, &format, &nitems, &bytes_after, &(prop.prop_ch)); error = gdk_error_trap_pop (); if (!error && result == Success && type == XA_CARDINAL && nitems == 1 && format == 32) { gint padding; padding = prop.prop[0]; if (icon->priv->padding != padding) { icon->priv->padding = padding; g_object_notify (G_OBJECT (icon), "padding"); } } if (type != None) XFree (prop.prop); } static void gtk_tray_icon_get_icon_size_property (GtkTrayIcon *icon) { GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon)); GdkDisplay *display = gdk_screen_get_display (screen); Display *xdisplay = GDK_DISPLAY_XDISPLAY (display); Atom type; int format; union { gulong *prop; guchar *prop_ch; } prop = { NULL }; gulong nitems; gulong bytes_after; int error, result; g_assert (icon->priv->manager_window != None); gdk_error_trap_push (); type = None; result = XGetWindowProperty (xdisplay, icon->priv->manager_window, icon->priv->icon_size_atom, 0, G_MAXLONG, FALSE, XA_CARDINAL, &type, &format, &nitems, &bytes_after, &(prop.prop_ch)); error = gdk_error_trap_pop (); if (!error && result == Success && type == XA_CARDINAL && nitems == 1 && format == 32) { gint icon_size; icon_size = prop.prop[0]; if (icon->priv->icon_size != icon_size) { icon->priv->icon_size = icon_size; g_object_notify (G_OBJECT (icon), "icon-size"); } } if (type != None) XFree (prop.prop); } static GdkFilterReturn gtk_tray_icon_manager_filter (GdkXEvent *xevent, GdkEvent *event, gpointer user_data) { GtkTrayIcon *icon = user_data; XEvent *xev = (XEvent *)xevent; if (xev->xany.type == ClientMessage && xev->xclient.message_type == icon->priv->manager_atom && xev->xclient.data.l[1] == icon->priv->selection_atom) { GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: tray manager appeared", icon)); gtk_tray_icon_update_manager_window (icon); } else if (xev->xany.window == icon->priv->manager_window) { if (xev->xany.type == PropertyNotify && xev->xproperty.atom == icon->priv->orientation_atom) { GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: got PropertyNotify on manager window for orientation atom", icon)); gtk_tray_icon_get_orientation_property (icon); } else if (xev->xany.type == PropertyNotify && xev->xproperty.atom == icon->priv->colors_atom) { GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: got PropertyNotify on manager window for colors atom", icon)); gtk_tray_icon_get_colors_property (icon); } else if (xev->xany.type == PropertyNotify && xev->xproperty.atom == icon->priv->padding_atom) { gtk_tray_icon_get_padding_property (icon); } else if (xev->xany.type == PropertyNotify && xev->xproperty.atom == icon->priv->icon_size_atom) { gtk_tray_icon_get_icon_size_property (icon); } else if (xev->xany.type == DestroyNotify) { GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: got DestroyNotify for manager window", icon)); gtk_tray_icon_manager_window_destroyed (icon); } else { GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: got other message on manager window", icon)); } } return GDK_FILTER_CONTINUE; } static void gtk_tray_icon_send_manager_message (GtkTrayIcon *icon, long message, Window window, long data1, long data2, long data3) { GtkWidget *widget; XClientMessageEvent ev; Display *display; widget = GTK_WIDGET (icon); memset (&ev, 0, sizeof (ev)); ev.type = ClientMessage; ev.window = window; ev.message_type = icon->priv->system_tray_opcode_atom; ev.format = 32; ev.data.l[0] = gdk_x11_get_server_time (gtk_widget_get_window (widget)); ev.data.l[1] = message; ev.data.l[2] = data1; ev.data.l[3] = data2; ev.data.l[4] = data3; display = GDK_DISPLAY_XDISPLAY (gtk_widget_get_display (widget)); gdk_error_trap_push (); XSendEvent (display, icon->priv->manager_window, False, NoEventMask, (XEvent *)&ev); gdk_error_trap_pop_ignored (); } static void gtk_tray_icon_send_dock_request (GtkTrayIcon *icon) { GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: sending dock request to manager window %lx", icon, (gulong) icon->priv->manager_window)); gtk_tray_icon_send_manager_message (icon, SYSTEM_TRAY_REQUEST_DOCK, icon->priv->manager_window, gtk_plug_get_id (GTK_PLUG (icon)), 0, 0); } static void gtk_tray_icon_update_manager_window (GtkTrayIcon *icon) { GtkWidget *widget = GTK_WIDGET (icon); GdkScreen *screen = gtk_widget_get_screen (widget); GdkDisplay *display = gdk_screen_get_display (screen); Display *xdisplay = GDK_DISPLAY_XDISPLAY (display); GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: updating tray icon manager window, current manager window: %lx", icon, (gulong) icon->priv->manager_window)); if (icon->priv->manager_window != None) return; GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: trying to find manager window", icon)); XGrabServer (xdisplay); icon->priv->manager_window = XGetSelectionOwner (xdisplay, icon->priv->selection_atom); if (icon->priv->manager_window != None) XSelectInput (xdisplay, icon->priv->manager_window, StructureNotifyMask|PropertyChangeMask); XUngrabServer (xdisplay); XFlush (xdisplay); if (icon->priv->manager_window != None) { GdkWindow *gdkwin; GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: is being managed by window %lx", icon, (gulong) icon->priv->manager_window)); gdkwin = gdk_x11_window_lookup_for_display (display, icon->priv->manager_window); gdk_window_add_filter (gdkwin, gtk_tray_icon_manager_filter, icon); gtk_tray_icon_get_orientation_property (icon); gtk_tray_icon_get_visual_property (icon); gtk_tray_icon_get_colors_property (icon); gtk_tray_icon_get_padding_property (icon); gtk_tray_icon_get_icon_size_property (icon); if (gtk_widget_get_realized (GTK_WIDGET (icon))) { if ((icon->priv->manager_visual == NULL && gtk_widget_get_visual (widget) == gdk_screen_get_system_visual (screen)) || (icon->priv->manager_visual == gtk_widget_get_visual (widget))) { /* Already have the right visual, can just dock */ gtk_tray_icon_send_dock_request (icon); } else { /* Need to re-realize the widget to get the right visual */ gtk_widget_hide (widget); gtk_widget_unrealize (widget); gtk_widget_show (widget); } } } else { GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: no tray manager found", icon)); } } static void gtk_tray_icon_manager_window_destroyed (GtkTrayIcon *icon) { g_return_if_fail (icon->priv->manager_window != None); GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: tray manager window destroyed", icon)); gtk_tray_icon_clear_manager_window (icon); } static gboolean gtk_tray_icon_delete (GtkWidget *widget, GdkEventAny *event) { GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: delete notify, tray manager window %lx", widget, (gulong) GTK_TRAY_ICON (widget)->priv->manager_window)); /* A bug in X server versions up to x.org 1.5.0 means that: * XFixesChangeSaveSet(...., SaveSetRoot, SaveSetUnmap) doesn't work properly * and we'll left mapped in a separate toplevel window if the tray is destroyed. * For simplicity just get rid of our X window and start over. */ gtk_widget_hide (widget); gtk_widget_unrealize (widget); gtk_widget_show (widget); /* Handled it, don't destroy the tray icon */ return TRUE; } static void gtk_tray_icon_set_visual (GtkTrayIcon *icon) { GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET (icon)); GdkVisual *visual = icon->priv->manager_visual; /* To avoid uncertainty about colormaps, _NET_SYSTEM_TRAY_VISUAL is supposed * to be either the screen default visual or a TrueColor visual; ignore it * if it is something else */ if (visual && gdk_visual_get_visual_type (visual) != GDK_VISUAL_TRUE_COLOR) visual = NULL; if (visual == NULL) visual = gdk_screen_get_system_visual (screen); gtk_widget_set_visual (GTK_WIDGET (icon), visual); } static void gtk_tray_icon_realize (GtkWidget *widget) { GtkTrayIcon *icon = GTK_TRAY_ICON (widget); GdkWindow *window; /* Set our visual before realizing */ gtk_tray_icon_set_visual (icon); GTK_WIDGET_CLASS (gtk_tray_icon_parent_class)->realize (widget); window = gtk_widget_get_window (widget); G_GNUC_BEGIN_IGNORE_DEPRECATIONS if (icon->priv->manager_visual_rgba) { /* Set a transparent background */ GdkRGBA transparent = { 0.0, 0.0, 0.0, 0.0 }; gdk_window_set_background_rgba (window, &transparent); } else { /* Set a parent-relative background pixmap */ gdk_window_set_background_pattern (window, NULL); } G_GNUC_END_IGNORE_DEPRECATIONS GTK_NOTE (PLUGSOCKET, g_message ("GtkStatusIcon %p: realized, window: %lx, socket window: %lx", widget, (gulong) GDK_WINDOW_XID (window), gtk_plug_get_socket_window (GTK_PLUG (icon)) ? (gulong) GDK_WINDOW_XID (gtk_plug_get_socket_window (GTK_PLUG (icon))) : 0UL)); if (icon->priv->manager_window != None) gtk_tray_icon_send_dock_request (icon); } static void gtk_tray_icon_style_updated (GtkWidget *widget) { /* The default handler resets the background according to the style. We either * use a transparent background or a parent-relative background and ignore the * style background. So, just don't chain up. */ } guint _gtk_tray_icon_send_message (GtkTrayIcon *icon, gint timeout, const gchar *message, gint len) { guint stamp; Display *xdisplay; g_return_val_if_fail (GTK_IS_TRAY_ICON (icon), 0); g_return_val_if_fail (timeout >= 0, 0); g_return_val_if_fail (message != NULL, 0); if (icon->priv->manager_window == None) return 0; if (len < 0) len = strlen (message); stamp = icon->priv->stamp++; /* Get ready to send the message */ gtk_tray_icon_send_manager_message (icon, SYSTEM_TRAY_BEGIN_MESSAGE, (Window)gtk_plug_get_id (GTK_PLUG (icon)), timeout, len, stamp); /* Now to send the actual message */ xdisplay = GDK_DISPLAY_XDISPLAY (gtk_widget_get_display (GTK_WIDGET (icon))); gdk_error_trap_push (); while (len > 0) { XClientMessageEvent ev; memset (&ev, 0, sizeof (ev)); ev.type = ClientMessage; ev.window = (Window)gtk_plug_get_id (GTK_PLUG (icon)); ev.format = 8; ev.message_type = XInternAtom (xdisplay, "_NET_SYSTEM_TRAY_MESSAGE_DATA", False); if (len > 20) { memcpy (&ev.data, message, 20); len -= 20; message += 20; } else { memcpy (&ev.data, message, len); len = 0; } XSendEvent (xdisplay, icon->priv->manager_window, False, StructureNotifyMask, (XEvent *)&ev); } gdk_error_trap_pop_ignored (); return stamp; } void _gtk_tray_icon_cancel_message (GtkTrayIcon *icon, guint id) { g_return_if_fail (GTK_IS_TRAY_ICON (icon)); g_return_if_fail (id > 0); gtk_tray_icon_send_manager_message (icon, SYSTEM_TRAY_CANCEL_MESSAGE, (Window)gtk_plug_get_id (GTK_PLUG (icon)), id, 0, 0); } GtkTrayIcon * _gtk_tray_icon_new_for_screen (GdkScreen *screen, const gchar *name) { g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL); return g_object_new (GTK_TYPE_TRAY_ICON, "screen", screen, "title", name, NULL); } GtkTrayIcon* _gtk_tray_icon_new (const gchar *name) { return g_object_new (GTK_TYPE_TRAY_ICON, "title", name, NULL); } GtkOrientation _gtk_tray_icon_get_orientation (GtkTrayIcon *icon) { g_return_val_if_fail (GTK_IS_TRAY_ICON (icon), GTK_ORIENTATION_HORIZONTAL); return icon->priv->orientation; } gint _gtk_tray_icon_get_padding (GtkTrayIcon *icon) { g_return_val_if_fail (GTK_IS_TRAY_ICON (icon), 0); return icon->priv->padding; } gint _gtk_tray_icon_get_icon_size (GtkTrayIcon *icon) { g_return_val_if_fail (GTK_IS_TRAY_ICON (icon), 0); return icon->priv->icon_size; }