 d3b34d3cf2
			
		
	
	d3b34d3cf2
	
	
	
		
			
			Applications can call this to determine if they should an app menu. This will be %FALSE on desktop environments that do not have an application menu like the one in gnome-shell. It is %FALSE on Windows and Mac OS. Applications are completely free to totally ignore this API -- it is only provided as a hint to help applications that may be interested in supporting non-GNOME platforms with a more native 'look and feel'. https://bugzilla.gnome.org/show_bug.cgi?id=722092
		
			
				
	
	
		
			498 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			498 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright © 2010 Codethink Limited
 | |
|  * Copyright © 2012 Red Hat, Inc.
 | |
|  * Copyright © 2013 Canonical Limited
 | |
|  *
 | |
|  * 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 licence, 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 <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  * Author: Ryan Lortie <desrt@desrt.ca>
 | |
|  *         Matthias Clasen <mclasen@redhat.com>
 | |
|  */
 | |
| 
 | |
| #include "config.h"
 | |
| 
 | |
| #include "gtkapplicationprivate.h"
 | |
| #include "gtksettings.h"
 | |
| 
 | |
| G_DEFINE_TYPE (GtkApplicationImplDBus, gtk_application_impl_dbus, GTK_TYPE_APPLICATION_IMPL)
 | |
| 
 | |
| 
 | |
| static void
 | |
| unregister_client (GtkApplicationImplDBus *dbus)
 | |
| {
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   g_debug ("Unregistering client");
 | |
| 
 | |
|   g_dbus_proxy_call_sync (dbus->sm_proxy,
 | |
|                           "UnregisterClient",
 | |
|                           g_variant_new ("(o)", dbus->client_path),
 | |
|                           G_DBUS_CALL_FLAGS_NONE,
 | |
|                           G_MAXINT,
 | |
|                           NULL,
 | |
|                           &error);
 | |
| 
 | |
|   if (error)
 | |
|     {
 | |
|       g_warning ("Failed to unregister client: %s", error->message);
 | |
|       g_error_free (error);
 | |
|     }
 | |
| 
 | |
|   g_clear_object (&dbus->client_proxy);
 | |
| 
 | |
|   g_free (dbus->client_path);
 | |
|   dbus->client_path = NULL;
 | |
| }
 | |
| 
 | |
| static void
 | |
| send_quit_response (GtkApplicationImplDBus *dbus,
 | |
|                     gboolean                will_quit,
 | |
|                     const gchar            *reason)
 | |
| {
 | |
|   g_debug ("Calling EndSessionResponse %d '%s'", will_quit, reason);
 | |
| 
 | |
|   g_dbus_proxy_call (dbus->client_proxy,
 | |
|                      "EndSessionResponse",
 | |
|                      g_variant_new ("(bs)", will_quit, reason ? reason : ""),
 | |
|                      G_DBUS_CALL_FLAGS_NONE,
 | |
|                      G_MAXINT,
 | |
|                      NULL, NULL, NULL);
 | |
| }
 | |
| 
 | |
| static void
 | |
| client_proxy_signal (GDBusProxy  *proxy,
 | |
|                      const gchar *sender_name,
 | |
|                      const gchar *signal_name,
 | |
|                      GVariant    *parameters,
 | |
|                      gpointer     user_data)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = user_data;
 | |
| 
 | |
|   if (g_str_equal (signal_name, "QueryEndSession"))
 | |
|     {
 | |
|       g_debug ("Received QueryEndSession");
 | |
|       send_quit_response (dbus, TRUE, NULL);
 | |
|     }
 | |
|   else if (g_str_equal (signal_name, "CancelEndSession"))
 | |
|     {
 | |
|       g_debug ("Received CancelEndSession");
 | |
|     }
 | |
|   else if (g_str_equal (signal_name, "EndSession"))
 | |
|     {
 | |
|       g_debug ("Received EndSession");
 | |
|       send_quit_response (dbus, TRUE, NULL);
 | |
|       unregister_client (dbus);
 | |
|       g_application_quit (G_APPLICATION (dbus->impl.application));
 | |
|     }
 | |
|   else if (g_str_equal (signal_name, "Stop"))
 | |
|     {
 | |
|       g_debug ("Received Stop");
 | |
|       unregister_client (dbus);
 | |
|       g_application_quit (G_APPLICATION (dbus->impl.application));
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_startup (GtkApplicationImpl *impl,
 | |
|                                    gboolean            register_session)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = (GtkApplicationImplDBus *) impl;
 | |
|   static gchar *client_id;
 | |
|   GError *error = NULL;
 | |
|   GVariant *res;
 | |
| 
 | |
|   dbus->session = g_application_get_dbus_connection (G_APPLICATION (impl->application));
 | |
| 
 | |
|   if (!dbus->session)
 | |
|     return;
 | |
| 
 | |
|   dbus->application_id = g_application_get_application_id (G_APPLICATION (impl->application));
 | |
|   dbus->object_path = g_application_get_dbus_object_path (G_APPLICATION (impl->application));
 | |
|   dbus->unique_name = g_dbus_connection_get_unique_name (dbus->session);
 | |
| 
 | |
|   if (client_id == NULL)
 | |
|     {
 | |
|       const gchar *desktop_autostart_id;
 | |
| 
 | |
|       desktop_autostart_id = g_getenv ("DESKTOP_AUTOSTART_ID");
 | |
|       /* Unset DESKTOP_AUTOSTART_ID in order to avoid child processes to
 | |
|        * use the same client id.
 | |
|        */
 | |
|       g_unsetenv ("DESKTOP_AUTOSTART_ID");
 | |
|       client_id = g_strdup (desktop_autostart_id ? desktop_autostart_id : "");
 | |
|     }
 | |
| 
 | |
|   g_debug ("Connecting to session manager");
 | |
| 
 | |
|   dbus->sm_proxy = g_dbus_proxy_new_sync (dbus->session,
 | |
|                                           G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES |
 | |
|                                           G_DBUS_PROXY_FLAGS_DO_NOT_CONNECT_SIGNALS,
 | |
|                                           NULL,
 | |
|                                           "org.gnome.SessionManager",
 | |
|                                           "/org/gnome/SessionManager",
 | |
|                                           "org.gnome.SessionManager",
 | |
|                                           NULL,
 | |
|                                           &error);
 | |
| 
 | |
|   if (error)
 | |
|     {
 | |
|       g_warning ("Failed to get a session proxy: %s", error->message);
 | |
|       g_error_free (error);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|   /* FIXME: should we reuse the D-Bus application id here ? */
 | |
|   dbus->app_id = g_strdup (g_get_prgname ());
 | |
| 
 | |
|   if (!register_session)
 | |
|     return;
 | |
| 
 | |
|   g_debug ("Registering client '%s' '%s'", dbus->app_id, client_id);
 | |
| 
 | |
|   res = g_dbus_proxy_call_sync (dbus->sm_proxy,
 | |
|                                 "RegisterClient",
 | |
|                                 g_variant_new ("(ss)", dbus->app_id, client_id),
 | |
|                                 G_DBUS_CALL_FLAGS_NONE,
 | |
|                                 G_MAXINT,
 | |
|                                 NULL,
 | |
|                                 &error);
 | |
| 
 | |
|   if (error)
 | |
|     {
 | |
|       g_warning ("Failed to register client: %s", error->message);
 | |
|       g_error_free (error);
 | |
|       g_clear_object (&dbus->sm_proxy);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|   g_variant_get (res, "(o)", &dbus->client_path);
 | |
|   g_variant_unref (res);
 | |
| 
 | |
|   g_debug ("Registered client at '%s'", dbus->client_path);
 | |
| 
 | |
|   dbus->client_proxy = g_dbus_proxy_new_sync (dbus->session, 0,
 | |
|                                               NULL,
 | |
|                                               "org.gnome.SessionManager",
 | |
|                                               dbus->client_path,
 | |
|                                               "org.gnome.SessionManager.ClientPrivate",
 | |
|                                               NULL,
 | |
|                                               &error);
 | |
|   if (error)
 | |
|     {
 | |
|       g_warning ("Failed to get client proxy: %s", error->message);
 | |
|       g_error_free (error);
 | |
|       g_clear_object (&dbus->sm_proxy);
 | |
|       g_free (dbus->client_path);
 | |
|       dbus->client_path = NULL;
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|   g_signal_connect (dbus->client_proxy, "g-signal", G_CALLBACK (client_proxy_signal), dbus);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_shutdown (GtkApplicationImpl *impl)
 | |
| {
 | |
| }
 | |
| 
 | |
| G_DEFINE_QUARK (GtkApplicationImplDBus export id, gtk_application_impl_dbus_export_id)
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_window_added (GtkApplicationImpl *impl,
 | |
|                                         GtkWindow          *window)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = (GtkApplicationImplDBus *) impl;
 | |
|   GActionGroup *actions;
 | |
|   gchar *path;
 | |
|   guint id;
 | |
| 
 | |
|   if (!dbus->session || !GTK_IS_APPLICATION_WINDOW (window))
 | |
|     return;
 | |
| 
 | |
|   /* Export the action group of this window, based on its id */
 | |
|   actions = gtk_application_window_get_action_group (GTK_APPLICATION_WINDOW (window));
 | |
| 
 | |
|   path = gtk_application_impl_dbus_get_window_path (dbus, window);
 | |
|   id = g_dbus_connection_export_action_group (dbus->session, path, actions, NULL);
 | |
|   g_free (path);
 | |
| 
 | |
|   g_object_set_qdata (G_OBJECT (window), gtk_application_impl_dbus_export_id_quark (), GUINT_TO_POINTER (id));
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_window_removed (GtkApplicationImpl *impl,
 | |
|                                           GtkWindow          *window)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = (GtkApplicationImplDBus *) impl;
 | |
|   guint id;
 | |
| 
 | |
|   id = GPOINTER_TO_UINT (g_object_get_qdata (G_OBJECT (window), gtk_application_impl_dbus_export_id_quark ()));
 | |
|   if (id)
 | |
|     {
 | |
|       g_dbus_connection_unexport_action_group (dbus->session, id);
 | |
|       g_object_set_qdata (G_OBJECT (window), gtk_application_impl_dbus_export_id_quark (), NULL);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_active_window_changed (GtkApplicationImpl *impl,
 | |
|                                                  GtkWindow          *window)
 | |
| {
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_publish_menu (GtkApplicationImplDBus  *dbus,
 | |
|                                         const gchar             *type,
 | |
|                                         GMenuModel              *model,
 | |
|                                         guint                   *id,
 | |
|                                         gchar                  **path)
 | |
| {
 | |
|   gint i;
 | |
| 
 | |
|   if (dbus->session == NULL)
 | |
|     return;
 | |
| 
 | |
|   /* unexport any existing menu */
 | |
|   if (*id)
 | |
|     {
 | |
|       g_dbus_connection_unexport_menu_model (dbus->session, *id);
 | |
|       g_free (*path);
 | |
|       *path = NULL;
 | |
|       *id = 0;
 | |
|     }
 | |
| 
 | |
|   /* export the new menu, if there is one */
 | |
|   if (model != NULL)
 | |
|     {
 | |
|       /* try getting the preferred name */
 | |
|       *path = g_strconcat (dbus->object_path, "/menus/", type, NULL);
 | |
|       *id = g_dbus_connection_export_menu_model (dbus->session, *path, model, NULL);
 | |
| 
 | |
|       /* keep trying until we get a working name... */
 | |
|       for (i = 0; *id == 0; i++)
 | |
|         {
 | |
|           g_free (*path);
 | |
|           *path = g_strdup_printf ("%s/menus/%s%d", dbus->object_path, type, i);
 | |
|           *id = g_dbus_connection_export_menu_model (dbus->session, *path, model, NULL);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_set_app_menu (GtkApplicationImpl *impl,
 | |
|                                         GMenuModel         *app_menu)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = (GtkApplicationImplDBus *) impl;
 | |
| 
 | |
|   gtk_application_impl_dbus_publish_menu (dbus, "appmenu", app_menu, &dbus->app_menu_id, &dbus->app_menu_path);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_set_menubar (GtkApplicationImpl *impl,
 | |
|                                        GMenuModel         *menubar)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = (GtkApplicationImplDBus *) impl;
 | |
| 
 | |
|   gtk_application_impl_dbus_publish_menu (dbus, "menubar", menubar, &dbus->menubar_id, &dbus->menubar_path);
 | |
| }
 | |
| 
 | |
| static GVariant *
 | |
| gtk_application_impl_dbus_real_get_window_system_id (GtkApplicationImplDBus *dbus,
 | |
|                                                      GtkWindow              *window)
 | |
| {
 | |
|   return g_variant_new_uint32 (0);
 | |
| }
 | |
| 
 | |
| /* returns floating */
 | |
| static GVariant *
 | |
| gtk_application_impl_dbus_get_window_system_id (GtkApplicationImplDBus *dbus,
 | |
|                                                 GtkWindow              *window)
 | |
| {
 | |
|   return GTK_APPLICATION_IMPL_DBUS_GET_CLASS (dbus)->get_window_system_id (dbus, window);
 | |
| }
 | |
| 
 | |
| static guint
 | |
| gtk_application_impl_dbus_inhibit (GtkApplicationImpl         *impl,
 | |
|                                    GtkWindow                  *window,
 | |
|                                    GtkApplicationInhibitFlags  flags,
 | |
|                                    const gchar                *reason)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = (GtkApplicationImplDBus *) impl;
 | |
|   GVariant *res;
 | |
|   GError *error = NULL;
 | |
|   guint cookie;
 | |
| 
 | |
|   if (dbus->sm_proxy == NULL)
 | |
|     return 0;
 | |
| 
 | |
|   res = g_dbus_proxy_call_sync (dbus->sm_proxy,
 | |
|                                 "Inhibit",
 | |
|                                 g_variant_new ("(s@usu)",
 | |
|                                                dbus->app_id,
 | |
|                                                window ? gtk_application_impl_dbus_get_window_system_id (dbus, window) : g_variant_new_uint32 (0),
 | |
|                                                reason,
 | |
|                                                flags),
 | |
|                                 G_DBUS_CALL_FLAGS_NONE,
 | |
|                                 G_MAXINT,
 | |
|                                 NULL,
 | |
|                                 &error);
 | |
| 
 | |
|  if (error)
 | |
|     {
 | |
|       g_warning ("Calling Inhibit failed: %s", error->message);
 | |
|       g_error_free (error);
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|   g_variant_get (res, "(u)", &cookie);
 | |
|   g_variant_unref (res);
 | |
| 
 | |
|   return cookie;
 | |
| 
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_uninhibit (GtkApplicationImpl *impl,
 | |
|                                      guint               cookie)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = (GtkApplicationImplDBus *) impl;
 | |
| 
 | |
|   /* Application could only obtain a cookie through a session
 | |
|    * manager proxy, so it's valid to assert its presence here. */
 | |
|   g_return_if_fail (dbus->sm_proxy != NULL);
 | |
| 
 | |
|   g_dbus_proxy_call (dbus->sm_proxy,
 | |
|                      "Uninhibit",
 | |
|                      g_variant_new ("(u)", cookie),
 | |
|                      G_DBUS_CALL_FLAGS_NONE,
 | |
|                      G_MAXINT,
 | |
|                      NULL, NULL, NULL);
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| gtk_application_impl_dbus_is_inhibited (GtkApplicationImpl         *impl,
 | |
|                                         GtkApplicationInhibitFlags  flags)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = (GtkApplicationImplDBus *) impl;
 | |
|   GVariant *res;
 | |
|   GError *error = NULL;
 | |
|   gboolean inhibited;
 | |
| 
 | |
|   if (dbus->sm_proxy == NULL)
 | |
|     return FALSE;
 | |
| 
 | |
|   res = g_dbus_proxy_call_sync (dbus->sm_proxy,
 | |
|                                 "IsInhibited",
 | |
|                                 g_variant_new ("(u)", flags),
 | |
|                                 G_DBUS_CALL_FLAGS_NONE,
 | |
|                                 G_MAXINT,
 | |
|                                 NULL,
 | |
|                                 &error);
 | |
|   if (error)
 | |
|     {
 | |
|       g_warning ("Calling IsInhibited failed: %s", error->message);
 | |
|       g_error_free (error);
 | |
|       return FALSE;
 | |
|     }
 | |
| 
 | |
|   g_variant_get (res, "(b)", &inhibited);
 | |
|   g_variant_unref (res);
 | |
| 
 | |
|   return inhibited;
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| gtk_application_impl_dbus_prefers_app_menu (GtkApplicationImpl *impl)
 | |
| {
 | |
|   static gboolean decided;
 | |
|   static gboolean result;
 | |
| 
 | |
|   /* We do not support notifying if/when the result changes, so make
 | |
|    * sure that once we give an answer, we will always give the same one.
 | |
|    */
 | |
|   if (!decided)
 | |
|     {
 | |
|       GtkSettings *gtk_settings;
 | |
|       gboolean show_app_menu;
 | |
|       gboolean show_menubar;
 | |
| 
 | |
|       gtk_settings = gtk_settings_get_default ();
 | |
|       g_object_get (G_OBJECT (gtk_settings),
 | |
|                     "gtk-shell-shows-app-menu", &show_app_menu,
 | |
|                     "gtk-shell-shows-menubar", &show_menubar,
 | |
|                     NULL);
 | |
| 
 | |
|       /* We prefer traditional menus when we have a shell that doesn't
 | |
|        * show the appmenu or we have a shell that shows menubars
 | |
|        * (ie: Unity)
 | |
|        */
 | |
|       result = show_app_menu && !show_menubar;
 | |
|       decided = TRUE;
 | |
|     }
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_init (GtkApplicationImplDBus *dbus)
 | |
| {
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_finalize (GObject *object)
 | |
| {
 | |
|   GtkApplicationImplDBus *dbus = (GtkApplicationImplDBus *) object;
 | |
| 
 | |
|   g_free (dbus->app_menu_path);
 | |
|   g_free (dbus->menubar_path);
 | |
|   g_free (dbus->app_id);
 | |
| 
 | |
|   G_OBJECT_CLASS (gtk_application_impl_dbus_parent_class)->finalize (object);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_application_impl_dbus_class_init (GtkApplicationImplDBusClass *class)
 | |
| {
 | |
|   GtkApplicationImplClass *impl_class = GTK_APPLICATION_IMPL_CLASS (class);
 | |
|   GObjectClass *gobject_class = G_OBJECT_CLASS (class);
 | |
| 
 | |
|   class->get_window_system_id = gtk_application_impl_dbus_real_get_window_system_id;
 | |
| 
 | |
|   impl_class->startup = gtk_application_impl_dbus_startup;
 | |
|   impl_class->shutdown = gtk_application_impl_dbus_shutdown;
 | |
|   impl_class->window_added = gtk_application_impl_dbus_window_added;
 | |
|   impl_class->window_removed = gtk_application_impl_dbus_window_removed;
 | |
|   impl_class->active_window_changed = gtk_application_impl_dbus_active_window_changed;
 | |
|   impl_class->set_app_menu = gtk_application_impl_dbus_set_app_menu;
 | |
|   impl_class->set_menubar = gtk_application_impl_dbus_set_menubar;
 | |
|   impl_class->inhibit = gtk_application_impl_dbus_inhibit;
 | |
|   impl_class->uninhibit = gtk_application_impl_dbus_uninhibit;
 | |
|   impl_class->is_inhibited = gtk_application_impl_dbus_is_inhibited;
 | |
|   impl_class->prefers_app_menu = gtk_application_impl_dbus_prefers_app_menu;
 | |
| 
 | |
|   gobject_class->finalize = gtk_application_impl_dbus_finalize;
 | |
| }
 | |
| 
 | |
| gchar *
 | |
| gtk_application_impl_dbus_get_window_path (GtkApplicationImplDBus *dbus,
 | |
|                                            GtkWindow *window)
 | |
| {
 | |
|   if (dbus->session && GTK_IS_APPLICATION_WINDOW (window))
 | |
|     return g_strdup_printf ("%s/window/%d",
 | |
|                             dbus->object_path,
 | |
|                             gtk_application_window_get_id (GTK_APPLICATION_WINDOW (window)));
 | |
|   else
 | |
|     return NULL;
 | |
| }
 |