diff options
| author | Ted Gould <ted@gould.cx> | 2009-12-02 12:02:35 -0600 | 
|---|---|---|
| committer | Ted Gould <ted@gould.cx> | 2009-12-02 12:02:35 -0600 | 
| commit | 984759eaf0354ee99a177b185fed21ba0e5229ad (patch) | |
| tree | b246e31d89f79defb4d3f2e8ba26a7c716e6f816 /src/indicator-application.c | |
| parent | 47741f6a4e84f187858caa54dc625a9d3d5e2b07 (diff) | |
| parent | f0881be3b8f779e0411958e6637f8f29e5697cbb (diff) | |
| download | ayatana-indicator-application-984759eaf0354ee99a177b185fed21ba0e5229ad.tar.gz ayatana-indicator-application-984759eaf0354ee99a177b185fed21ba0e5229ad.tar.bz2 ayatana-indicator-application-984759eaf0354ee99a177b185fed21ba0e5229ad.zip | |
Changing the name from indicator-custom to indicator-application
Diffstat (limited to 'src/indicator-application.c')
| -rw-r--r-- | src/indicator-application.c | 304 | 
1 files changed, 304 insertions, 0 deletions
| diff --git a/src/indicator-application.c b/src/indicator-application.c new file mode 100644 index 0000000..df45d13 --- /dev/null +++ b/src/indicator-application.c @@ -0,0 +1,304 @@ + +/* G Stuff */ +#include <glib.h> +#include <glib-object.h> +#include <gtk/gtk.h> + +/* DBus Stuff */ +#include <dbus/dbus-glib.h> +#include <libdbusmenu-gtk/menu.h> + +/* Indicator Stuff */ +#include <libindicator/indicator.h> +#include <libindicator/indicator-object.h> +#include <libindicator/indicator-service-manager.h> + +/* Local Stuff */ +#include "dbus-shared.h" +#include "application-service-client.h" +#include "application-service-marshal.h" + +#define INDICATOR_APPLICATION_TYPE            (indicator_application_get_type ()) +#define INDICATOR_APPLICATION(obj)            (G_TYPE_CHECK_INSTANCE_CAST ((obj), INDICATOR_APPLICATION_TYPE, IndicatorApplication)) +#define INDICATOR_APPLICATION_CLASS(klass)    (G_TYPE_CHECK_CLASS_CAST ((klass), INDICATOR_APPLICATION_TYPE, IndicatorApplicationClass)) +#define IS_INDICATOR_APPLICATION(obj)         (G_TYPE_CHECK_INSTANCE_TYPE ((obj), INDICATOR_APPLICATION_TYPE)) +#define IS_INDICATOR_APPLICATION_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), INDICATOR_APPLICATION_TYPE)) +#define INDICATOR_APPLICATION_GET_CLASS(obj)  (G_TYPE_INSTANCE_GET_CLASS ((obj), INDICATOR_APPLICATION_TYPE, IndicatorApplicationClass)) + +typedef struct _IndicatorApplication      IndicatorApplication; +typedef struct _IndicatorApplicationClass IndicatorApplicationClass; + +struct _IndicatorApplicationClass { +	IndicatorObjectClass parent_class; +}; + +struct _IndicatorApplication { +	IndicatorObject parent; +}; + +GType indicator_application_get_type (void); + +INDICATOR_SET_VERSION +INDICATOR_SET_TYPE(INDICATOR_APPLICATION_TYPE) + + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +typedef struct _IndicatorApplicationPrivate IndicatorApplicationPrivate; +struct _IndicatorApplicationPrivate { +	IndicatorServiceManager * sm; +	DBusGConnection * bus; +	DBusGProxy * service_proxy; +	GList * applications; +}; + +typedef struct _ApplicationEntry ApplicationEntry; +struct _ApplicationEntry { +	IndicatorObjectEntry entry; +}; + +#define INDICATOR_APPLICATION_GET_PRIVATE(o) \ +(G_TYPE_INSTANCE_GET_PRIVATE ((o), INDICATOR_APPLICATION_TYPE, IndicatorApplicationPrivate)) + +static void indicator_application_class_init (IndicatorApplicationClass *klass); +static void indicator_application_init       (IndicatorApplication *self); +static void indicator_application_dispose    (GObject *object); +static void indicator_application_finalize   (GObject *object); +static GList * get_entries (IndicatorObject * io); +static void connected (IndicatorServiceManager * sm, gboolean connected, IndicatorApplication * application); +static void application_added (DBusGProxy * proxy, const gchar * iconname, gint position, const gchar * dbusaddress, const gchar * dbusobject, IndicatorApplication * application); +static void application_removed (DBusGProxy * proxy, gint position , IndicatorApplication * application); +static void get_applications (DBusGProxy *proxy, GPtrArray *OUT_applications, GError *error, gpointer userdata); + +G_DEFINE_TYPE (IndicatorApplication, indicator_application, INDICATOR_OBJECT_TYPE); + +static void +indicator_application_class_init (IndicatorApplicationClass *klass) +{ +	GObjectClass *object_class = G_OBJECT_CLASS (klass); + +	g_type_class_add_private (klass, sizeof (IndicatorApplicationPrivate)); + +	object_class->dispose = indicator_application_dispose; +	object_class->finalize = indicator_application_finalize; + +	IndicatorObjectClass * io_class = INDICATOR_OBJECT_CLASS(klass); + +	io_class->get_entries = get_entries; + +	dbus_g_object_register_marshaller(_application_service_marshal_VOID__STRING_INT_STRING_STRING, +	                                  G_TYPE_NONE, +	                                  G_TYPE_STRING, +	                                  G_TYPE_INT, +	                                  G_TYPE_STRING, +	                                  G_TYPE_STRING, +	                                  G_TYPE_INVALID); + +	return; +} + +static void +indicator_application_init (IndicatorApplication *self) +{ +	IndicatorApplicationPrivate * priv = INDICATOR_APPLICATION_GET_PRIVATE(self); + +	/* These are built in the connection phase */ +	priv->bus = NULL; +	priv->service_proxy = NULL; + +	priv->sm = indicator_service_manager_new(INDICATOR_APPLICATION_DBUS_ADDR);	 +	g_signal_connect(G_OBJECT(priv->sm), INDICATOR_SERVICE_MANAGER_SIGNAL_CONNECTION_CHANGE, G_CALLBACK(connected), self); + +	priv->applications = NULL; + +	return; +} + +static void +indicator_application_dispose (GObject *object) +{ +	IndicatorApplicationPrivate * priv = INDICATOR_APPLICATION_GET_PRIVATE(object); + +	while (priv->applications != NULL) { +		application_removed(priv->service_proxy, +		                    0, +		                    INDICATOR_APPLICATION(object)); +	} + +	if (priv->sm != NULL) { +		g_object_unref(priv->sm); +		priv->sm = NULL; +	} + +	if (priv->bus != NULL) { +		/* We're not incrementing the ref count on this one. */ +		priv->bus = NULL; +	} + +	if (priv->service_proxy != NULL) { +		g_object_unref(G_OBJECT(priv->service_proxy)); +		priv->service_proxy = NULL; +	} + +	G_OBJECT_CLASS (indicator_application_parent_class)->dispose (object); +	return; +} + +static void +indicator_application_finalize (GObject *object) +{ + +	G_OBJECT_CLASS (indicator_application_parent_class)->finalize (object); +	return; +} + +void +connected (IndicatorServiceManager * sm, gboolean connected, IndicatorApplication * application) +{ +	IndicatorApplicationPrivate * priv = INDICATOR_APPLICATION_GET_PRIVATE(application); +	g_debug("Connected to Application Indicator Service."); + +	GError * error = NULL; + +	/* Grab the session bus */ +	if (priv->bus == NULL) { +		priv->bus = dbus_g_bus_get(DBUS_BUS_SESSION, &error); + +		if (error != NULL) { +			g_error("Unable to get session bus: %s", error->message); +			g_error_free(error); +			return; +		} +	} + +	/* Build the service proxy */ +	priv->service_proxy = dbus_g_proxy_new_for_name_owner(priv->bus, +	                                                      INDICATOR_APPLICATION_DBUS_ADDR, +	                                                      INDICATOR_APPLICATION_DBUS_OBJ, +	                                                      INDICATOR_APPLICATION_DBUS_IFACE, +	                                                      &error); + +	/* Set up proxy signals */ +	g_debug("Setup proxy signals"); +	dbus_g_proxy_add_signal(priv->service_proxy, +	                        "ApplicationAdded", +	                        G_TYPE_STRING, +	                        G_TYPE_INT, +	                        G_TYPE_STRING, +	                        G_TYPE_STRING, +	                        G_TYPE_INVALID); +	dbus_g_proxy_add_signal(priv->service_proxy, +	                        "ApplicationRemoved", +	                        G_TYPE_INT, +	                        G_TYPE_INVALID); + +	/* Connect to them */ +	g_debug("Connect to them."); +	dbus_g_proxy_connect_signal(priv->service_proxy, +	                            "ApplicationAdded", +	                            G_CALLBACK(application_added), +	                            application, +	                            NULL /* Disconnection Signal */); +	dbus_g_proxy_connect_signal(priv->service_proxy, +	                            "ApplicationRemoved", +	                            G_CALLBACK(application_removed), +	                            application, +	                            NULL /* Disconnection Signal */); + +	/* Query it for existing applications */ +	g_debug("Request current apps"); +	org_ayatana_indicator_application_service_get_applications_async(priv->service_proxy, +	                                                                 get_applications, +	                                                                 application); + +	return; +} + +/* Goes through the list of applications that we're maintaining and +   pulls out the IndicatorObjectEntry and returns that in a list +   for the caller. */ +static GList * +get_entries (IndicatorObject * io) +{ +	g_return_val_if_fail(IS_INDICATOR_APPLICATION(io), NULL); + +	IndicatorApplicationPrivate * priv = INDICATOR_APPLICATION_GET_PRIVATE(io); +	GList * retval = NULL; +	GList * apppointer = NULL; + +	for (apppointer = priv->applications; apppointer != NULL; apppointer = g_list_next(apppointer)) { +		IndicatorObjectEntry * entry = &(((ApplicationEntry *)apppointer->data)->entry); +		retval = g_list_prepend(retval, entry); +	} + +	if (retval != NULL) { +		retval = g_list_reverse(retval); +	} + +	return retval; +} + +/* Here we respond to new applications by building up the +   ApplicationEntry and signaling the indicator host that +   we've got a new indicator. */ +static void +application_added (DBusGProxy * proxy, const gchar * iconname, gint position, const gchar * dbusaddress, const gchar * dbusobject, IndicatorApplication * application) +{ +	g_debug("Building new application entry: %s  with icon: %s", dbusaddress, iconname); +	IndicatorApplicationPrivate * priv = INDICATOR_APPLICATION_GET_PRIVATE(application); +	ApplicationEntry * app = g_new(ApplicationEntry, 1); + +	app->entry.image = GTK_IMAGE(gtk_image_new_from_icon_name(iconname, GTK_ICON_SIZE_MENU)); +	app->entry.label = NULL; +	app->entry.menu = GTK_MENU(dbusmenu_gtkmenu_new((gchar *)dbusaddress, (gchar *)dbusobject)); + +	gtk_widget_show(GTK_WIDGET(app->entry.image)); + +	priv->applications = g_list_insert(priv->applications, app, position); + +	/* TODO: Need to deal with position here somehow */ +	g_signal_emit(G_OBJECT(application), INDICATOR_OBJECT_SIGNAL_ENTRY_ADDED_ID, 0, &(app->entry), TRUE); +	return; +} + +/* This removes the application from the list and free's all +   of the memory associated with it. */ +static void +application_removed (DBusGProxy * proxy, gint position, IndicatorApplication * application) +{ +	IndicatorApplicationPrivate * priv = INDICATOR_APPLICATION_GET_PRIVATE(application); +	ApplicationEntry * app = (ApplicationEntry *)g_list_nth_data(priv->applications, position); + +	if (app == NULL) { +		g_warning("Unable to find application at position: %d", position); +		return; +	} + +	priv->applications = g_list_remove(priv->applications, app); +	g_signal_emit(G_OBJECT(application), INDICATOR_OBJECT_SIGNAL_ENTRY_REMOVED_ID, 0, &(app->entry), TRUE); + +	if (app->entry.image != NULL) { +		g_object_unref(G_OBJECT(app->entry.image)); +	} +	if (app->entry.label != NULL) { +		g_warning("Odd, an application indicator with a label?"); +		g_object_unref(G_OBJECT(app->entry.label)); +	} +	if (app->entry.menu != NULL) { +		g_object_unref(G_OBJECT(app->entry.menu)); +	} +	g_free(app); + +	return; +} + +/* This repsonds to the list of applications that the service +   has and calls application_added on each one of them. */ +static void +get_applications (DBusGProxy *proxy, GPtrArray *OUT_applications, GError *error, gpointer userdata) +{ + +	return; +} | 
