From 70a3434b5af5d66260b03337a17f35377302da11 Mon Sep 17 00:00:00 2001 From: Colin Walters <walters@src.gnome.org> Date: Mon, 19 Jan 2009 23:21:57 +0000 Subject: [PATCH] Add a panel containing tasklist to bottom screen This is implemented as a separate process, since creating and running toplevel windows from inside Metacity has issues. We now grab a DBus name, and exec the child process. The child monitors our name to know when to exit. svn path=/trunk/; revision=153 --- js/ui/main.js | 4 +- scripts/launcher.py | 1 + src/gnomeshell-taskpanel.c | 24 +++++- src/shell-global.c | 50 +++++------ src/shell-global.h | 4 +- src/shell-panel-window.c | 168 ++++++++++++++++--------------------- 6 files changed, 128 insertions(+), 123 deletions(-) diff --git a/js/ui/main.js b/js/ui/main.js index 9f132c645..fef838aee 100644 --- a/js/ui/main.js +++ b/js/ui/main.js @@ -98,8 +98,8 @@ Signals.addSignalMethods(ClutterFrameTicker.prototype); function start() { let global = Shell.Global.get(); - // Here we grab our DBus name, etc. - global.late_init(); + global.grab_dbus_service(); + global.start_task_panel(); Tweener.setFrameTicker(new ClutterFrameTicker()); diff --git a/scripts/launcher.py b/scripts/launcher.py index 28c5d91c1..0efd6cb3f 100644 --- a/scripts/launcher.py +++ b/scripts/launcher.py @@ -93,6 +93,7 @@ class Launcher: env.update({'GNOME_SHELL_JS' : self.js_dir, 'GNOME_SHELL_DATADIR' : self.data_dir, 'GI_TYPELIB_PATH' : self.plugin_dir, + 'PATH' : os.environ.get('PATH', '') + ':' + self.plugin_dir, 'LD_LIBRARY_PATH' : os.environ.get('LD_LIBRARY_PATH', '') + ':' + self.plugin_dir, 'GNOME_DISABLE_CRASH_DIALOG' : '1'}) diff --git a/src/gnomeshell-taskpanel.c b/src/gnomeshell-taskpanel.c index 1dca7808d..5e9d2177c 100644 --- a/src/gnomeshell-taskpanel.c +++ b/src/gnomeshell-taskpanel.c @@ -18,10 +18,12 @@ on_name_owner_changed (DBusGProxy *proxy, } static void -monitor_main_shell () +monitor_main_shell (const char *shell_name) { DBusGConnection *session; DBusGProxy *driver; + GError *error = NULL; + gboolean have_shell; session = dbus_g_bus_get (DBUS_BUS_SESSION, NULL); @@ -29,6 +31,19 @@ monitor_main_shell () DBUS_SERVICE_DBUS, DBUS_PATH_DBUS, DBUS_INTERFACE_DBUS); + + if (!dbus_g_proxy_call (driver, "NameHasOwner", &error, G_TYPE_STRING, + shell_name, G_TYPE_INVALID, G_TYPE_BOOLEAN, + &have_shell, G_TYPE_INVALID)) + { + /* Shouldn't happen */ + exit (1); + } + if (!have_shell) + { + /* Shell doesn't exist; either crashed or was restarted. Just abort. */ + exit (0); + } dbus_g_proxy_add_signal (driver, "NameOwnerChanged", @@ -53,7 +68,12 @@ main (int argc, char **argv) gtk_init (&argc, &argv); - monitor_main_shell (); + if (argc != 2) { + g_printerr ("Usage: gnomeshell-taskpanel [PARENT_DBUS_SERVICE]\n"); + exit (1); + } + + monitor_main_shell (argv[1]); panel = shell_panel_window_new (); diff --git a/src/shell-global.c b/src/shell-global.c index 22ccd3a43..3aef2aea5 100644 --- a/src/shell-global.c +++ b/src/shell-global.c @@ -14,6 +14,8 @@ #include <dbus/dbus-glib.h> #include <libgnomeui/gnome-thumbnail.h> +#define SHELL_DBUS_SERVICE "org.gnome.Shell" + struct _ShellGlobal { GObject parent; @@ -460,25 +462,13 @@ shell_global_reexec_self (ShellGlobal *global) g_ptr_array_free (arr, TRUE); } -/** - * shell_global_late_init - * - * Perform once-only global initialization; currently this executes - * the external tasklist process and grabs the org.gnome.Shell DBus name. - */ void -shell_global_late_init (ShellGlobal *global) +shell_global_grab_dbus_service (ShellGlobal *global) { - static gboolean initialized = FALSE; GError *error = NULL; DBusGConnection *session; DBusGProxy *bus; guint32 request_name_result; - const char* panel_args[] = {"gnomeshell-taskpanel", NULL}; - - if (initialized) - return; - initialized = TRUE; session = dbus_g_bus_get (DBUS_BUS_SESSION, NULL); @@ -488,20 +478,32 @@ shell_global_late_init (ShellGlobal *global) DBUS_INTERFACE_DBUS); if (!dbus_g_proxy_call (bus, "RequestName", &error, - G_TYPE_STRING, "org.gnome.Shell", + G_TYPE_STRING, SHELL_DBUS_SERVICE, G_TYPE_UINT, 0, G_TYPE_INVALID, G_TYPE_UINT, &request_name_result, - G_TYPE_INVALID)) { - g_print ("failed to acquire org.gnome.Shell: %s\n", error->message); - exit (0); - } - - if (!g_spawn_async (NULL, &panel_args, NULL, G_SPAWN_SEARCH_PATH, NULL, - NULL, NULL, &error)) { - g_critical ("failed to execute %s: %s", panel_args[0], error->message); - g_clear_error (&error); - } + G_TYPE_INVALID)) + { + g_print ("failed to acquire org.gnome.Shell: %s\n", error->message); + /* If we somehow got started again, it's not an error to be running + * already. So just exit 0. + */ + exit (0); + } g_object_unref (bus); +} + +void +shell_global_start_task_panel (ShellGlobal *global) +{ + const char* panel_args[] = {"gnomeshell-taskpanel", SHELL_DBUS_SERVICE, NULL}; + GError *error = NULL; + + if (!g_spawn_async (NULL, (char**) panel_args, NULL, G_SPAWN_SEARCH_PATH, NULL, + NULL, NULL, &error)) + { + g_critical ("failed to execute %s: %s", panel_args[0], error->message); + g_clear_error (&error); + } } \ No newline at end of file diff --git a/src/shell-global.h b/src/shell-global.h index a9119c13b..a6e268a80 100644 --- a/src/shell-global.h +++ b/src/shell-global.h @@ -38,7 +38,9 @@ GdkPixbuf *shell_get_thumbnail_for_recent_info(GtkRecentInfo *recent_info); ShellGlobal *shell_global_get (void); -void shell_global_late_init (ShellGlobal *global); +void shell_global_grab_dbus_service (ShellGlobal *global); + +void shell_global_start_task_panel (ShellGlobal *global); void shell_global_set_stage_input_area (ShellGlobal *global, int x, diff --git a/src/shell-panel-window.c b/src/shell-panel-window.c index 04e9fbd91..a0fd2ac00 100644 --- a/src/shell-panel-window.c +++ b/src/shell-panel-window.c @@ -10,16 +10,7 @@ enum { }; -static void shell_panel_window_dispose (GObject *object); static void shell_panel_window_finalize (GObject *object); -static void shell_panel_window_set_property ( GObject *object, - guint property_id, - const GValue *value, - GParamSpec *pspec ); -static void shell_panel_window_get_property( GObject *object, - guint property_id, - GValue *value, - GParamSpec *pspec ); static void shell_panel_window_size_request (GtkWidget *self, GtkRequisition *req); static void shell_panel_window_size_allocate (GtkWidget *self, GtkAllocation *allocation); static void shell_panel_window_realize (GtkWidget *self); @@ -36,6 +27,8 @@ G_DEFINE_TYPE(ShellPanelWindow, shell_panel_window, GTK_TYPE_WINDOW); struct ShellPanelWindowPrivate { GtkAllocation workarea; guint width; + guint height; + Atom workarea_atom; }; static void @@ -44,11 +37,8 @@ shell_panel_window_class_init(ShellPanelWindowClass *klass) GObjectClass *gobject_class = (GObjectClass *)klass; GtkWidgetClass *widget_class = (GtkWidgetClass *)klass; - gobject_class->dispose = shell_panel_window_dispose; gobject_class->finalize = shell_panel_window_finalize; - gobject_class->set_property = shell_panel_window_set_property; - gobject_class->get_property = shell_panel_window_get_property; - + widget_class->realize = shell_panel_window_realize; widget_class->size_request = shell_panel_window_size_request; widget_class->size_allocate = shell_panel_window_size_allocate; @@ -59,18 +49,13 @@ static void shell_panel_window_init (ShellPanelWindow *self) { self->priv = g_new0 (ShellPanelWindowPrivate, 1); + self->priv->workarea_atom = gdk_x11_get_xatom_by_name_for_display (gdk_display_get_default (), "_NET_WORKAREA"); + gtk_window_set_type_hint (GTK_WINDOW (self), GDK_WINDOW_TYPE_HINT_DOCK); gtk_window_set_focus_on_map (GTK_WINDOW (self), FALSE); gdk_window_add_filter (NULL, filter_func, self); } -static void shell_panel_window_dispose (GObject *object) -{ - ShellPanelWindow *self = (ShellPanelWindow*)object; - - G_OBJECT_CLASS (shell_panel_window_parent_class)->dispose(object); -} - static void shell_panel_window_finalize (GObject *object) { ShellPanelWindow *self = (ShellPanelWindow*)object; @@ -80,53 +65,24 @@ static void shell_panel_window_finalize (GObject *object) G_OBJECT_CLASS (shell_panel_window_parent_class)->finalize(object); } -static void shell_panel_window_set_property ( GObject *object, - guint property_id, - const GValue *value, - GParamSpec *pspec ) -{ - ShellPanelWindow* self = SHELL_PANEL_WINDOW(object); - switch (property_id) { - default: - G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); - break; - } -} - - -static void shell_panel_window_get_property ( GObject *object, - guint property_id, - GValue *value, - GParamSpec *pspec ) -{ - ShellPanelWindow* self = SHELL_PANEL_WINDOW(object); - switch (property_id) { - default: - G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); - break; - } -} - -ShellPanelWindow* shell_panel_window_new(void) -{ +ShellPanelWindow* shell_panel_window_new(void) { return (ShellPanelWindow*) g_object_new(SHELL_TYPE_PANEL_WINDOW, "type", GTK_WINDOW_TOPLEVEL, NULL); } - static void set_strut (ShellPanelWindow *self) { long *buf; - int x, y; int strut_size; strut_size = GTK_WIDGET (self)->allocation.height; - gtk_window_get_position (GTK_WINDOW (self), &x, &y); - - buf = g_new0(long, 4); - buf[3] = strut_size; + buf = g_new0 (long, 4); + buf[0] = 0; /* left */ + buf[1] = 0; /* right */ + buf[2] = 0; /* top */ + buf[3] = strut_size; /* bottom */ gdk_property_change (GTK_WIDGET (self)->window, gdk_atom_intern_static_string ("_NET_WM_STRUT"), gdk_atom_intern_static_string ("CARDINAL"), 32, GDK_PROP_MODE_REPLACE, @@ -171,26 +127,44 @@ shell_panel_window_show (GtkWidget *widget) static void handle_new_workarea (ShellPanelWindow *self) { - int monitor; GtkRequisition requisition; - GdkRectangle monitor_geometry; int x, y; int width; + int height; + int x_target, y_target; - monitor = gdk_screen_get_monitor_at_point (gdk_screen_get_default (), - self->priv->workarea.x, - self->priv->workarea.y); - gdk_screen_get_monitor_geometry (gdk_screen_get_default (), - monitor, &monitor_geometry); gtk_widget_size_request (GTK_WIDGET (self), &requisition); - x = self->priv->workarea.x; - y = monitor_geometry.y + monitor_geometry.height - requisition.height; - width = monitor_geometry.width - x; + /* If we don't have a workarea, just use monitor */ + if (self->priv->workarea.width == 0) + { + int monitor; + GdkRectangle monitor_geometry; + + monitor = gdk_screen_get_monitor_at_point (gdk_screen_get_default (), + 0, 0); + gdk_screen_get_monitor_geometry (gdk_screen_get_default (), + monitor, &monitor_geometry); + x = monitor_geometry.x; + y = monitor_geometry.y; + width = monitor_geometry.width; + height = monitor_geometry.height; + } + else + { + x = self->priv->workarea.x; + y = self->priv->workarea.y; + width = self->priv->workarea.width; + height = self->priv->workarea.height; + } + + x_target = x; + y_target = y + height - requisition.height; self->priv->width = width; - gtk_widget_set_size_request (GTK_WIDGET (self), width, PANEL_HEIGHT); - gtk_window_move (GTK_WINDOW (self), x, y); + self->priv->height = height; + gtk_widget_set_size_request (GTK_WIDGET (self), width - x_target, PANEL_HEIGHT); + gtk_window_move (GTK_WINDOW (self), x_target, y_target); } static void @@ -207,31 +181,38 @@ on_workarea_changed (ShellPanelWindow *self) workarea, 0, 4, FALSE, workarea, &type, &format, &nitems, &bytes_after, &data); - if ((format == 32) && (nitems == 4) && (bytes_after == 0)) { - int x, y, width, height; - data32 = (long*)data; - x = data32[0]; y = data32[1]; - width = data32[2]; height = data32[3]; - if (x == self->priv->workarea.x && - y == self->priv->workarea.y && - width == self->priv->workarea.width && - height == self->priv->workarea.height) - return; + if ((format == 32) && (nitems == 4) && (bytes_after == 0)) + { + int x, y, width, height; + data32 = (long*) data; + x = data32[0]; + y = data32[1]; + width = data32[2]; + height = data32[3]; + if (x == self->priv->workarea.x && y == self->priv->workarea.y + && width == self->priv->workarea.width + && height == self->priv->workarea.height) + return; - self->priv->workarea.x = x; - self->priv->workarea.y = y; - self->priv->workarea.width = width; - self->priv->workarea.height = height; + self->priv->workarea.x = x; + self->priv->workarea.y = y; + self->priv->workarea.width = width; + self->priv->workarea.height = height; - handle_new_workarea (self); - } else if (nitems == 0) { - self->priv->workarea.x = self->priv->workarea.y = 0; - self->priv->workarea.width = self->priv->workarea.height = -1; - handle_new_workarea (self); - } else { - g_printerr ("unexpected return from XGetWindowProperty: %d %ld %ld\n", - format, nitems, bytes_after); - } + handle_new_workarea (self); + } + else if (nitems == 0) + { + /* We have no workarea set; assume there are no other panels at this time */ + self->priv->workarea.x = self->priv->workarea.y = 0; + self->priv->workarea.width = self->priv->workarea.height = 0; + handle_new_workarea (self); + } + else + { + g_printerr ("unexpected return from XGetWindowProperty: %d %ld %ld\n", + format, nitems, bytes_after); + } } static GdkFilterReturn @@ -242,14 +223,13 @@ filter_func (GdkXEvent *gdk_xevent, ShellPanelWindow *self = SHELL_PANEL_WINDOW (data); GdkFilterReturn ret = GDK_FILTER_CONTINUE; XEvent *xevent = (XEvent *) event; - Atom workarea = gdk_x11_get_xatom_by_name_for_display (gdk_display_get_default (), "_NET_WORKAREA"); - switch (xevent->type) { + switch (xevent->type) + { case PropertyNotify: { - if (xevent->xproperty.atom != workarea) + if (xevent->xproperty.atom != self->priv->workarea_atom) break; - on_workarea_changed (self); } break;