diff options
| author | Ted Gould <ted@gould.cx> | 2010-02-05 11:00:07 -0800 | 
|---|---|---|
| committer | Ted Gould <ted@gould.cx> | 2010-02-05 11:00:07 -0800 | 
| commit | c77ed1ccc280c56949b0a5a3f2d8d08bae498d29 (patch) | |
| tree | 4b8cc5313c3c6828565df8efad57ab85d53f4d3a /libdbusmenu-glib | |
| parent | a61a7ba8cc024d3d75f73e336ba83128631d3019 (diff) | |
| parent | f3a13c8b0be55927b8b91ff37ba922ba323b44b7 (diff) | |
| download | libdbusmenu-c77ed1ccc280c56949b0a5a3f2d8d08bae498d29.tar.gz libdbusmenu-c77ed1ccc280c56949b0a5a3f2d8d08bae498d29.tar.bz2 libdbusmenu-c77ed1ccc280c56949b0a5a3f2d8d08bae498d29.zip | |
* Upstrem Merge
  * Fixing handing of root IDs being zero
  * Handling syncing of properties when reusing items
  * Shipping new JSON files and READMEs for testing.
Diffstat (limited to 'libdbusmenu-glib')
| -rw-r--r-- | libdbusmenu-glib/client.c | 38 | ||||
| -rw-r--r-- | libdbusmenu-glib/dbus-menu.xml | 17 | ||||
| -rw-r--r-- | libdbusmenu-glib/menuitem.c | 69 | ||||
| -rw-r--r-- | libdbusmenu-glib/menuitem.h | 3 | ||||
| -rw-r--r-- | libdbusmenu-glib/server.c | 4 | 
5 files changed, 109 insertions, 22 deletions
| diff --git a/libdbusmenu-glib/client.c b/libdbusmenu-glib/client.c index 0f7fd65..2302110 100644 --- a/libdbusmenu-glib/client.c +++ b/libdbusmenu-glib/client.c @@ -339,6 +339,7 @@ id_prop_update (DBusGProxy * proxy, gint id, gchar * property, GValue * value, D  	g_return_if_fail(menuitem != NULL);  	dbusmenu_menuitem_property_set_value(menuitem, property, value); +  	return;  } @@ -569,6 +570,22 @@ menuitem_get_properties_cb (DBusGProxy * proxy, GHashTable * properties, GError  	return;  } +static void +menuitem_get_properties_replace_cb (DBusGProxy * proxy, GHashTable * properties, GError * error, gpointer data) +{ +	GList * current_props = NULL; + +	for (current_props = dbusmenu_menuitem_properties_list(DBUSMENU_MENUITEM(data)); +			current_props != NULL ; current_props = g_list_next(current_props)) { +		if (g_hash_table_lookup(properties, current_props->data) == NULL) { +			dbusmenu_menuitem_property_remove(DBUSMENU_MENUITEM(data), (const gchar *)current_props->data); +		} +	} + +	menuitem_get_properties_cb(proxy, properties, error, data); +	return; +} +  /* This is a different get properites call back that also sends     new signals.  It basically is a small wrapper around the original. */  static void @@ -647,6 +664,7 @@ parse_layout_xml(DbusmenuClient * client, xmlNodePtr node, DbusmenuMenuitem * it  	#ifdef MASSIVEDEBUGGING  	g_debug("Client looking at node with id: %d", id);  	#endif +	/* If we don't have any item, or the IDs don't match */  	if (item == NULL || dbusmenu_menuitem_get_id(item) != id) {  		if (item != NULL) {  			if (parent != NULL) { @@ -675,7 +693,11 @@ parse_layout_xml(DbusmenuClient * client, xmlNodePtr node, DbusmenuMenuitem * it  		} else {  			g_warning("Unable to allocate memory to get properties for menuitem.  This menuitem will never be realized.");  		} -	}  +	} else { +		/* Refresh the properties */ +		gchar * properties[1] = {NULL}; /* This gets them all */ +		org_ayatana_dbusmenu_get_properties_async(proxy, id, (const gchar **)properties, menuitem_get_properties_replace_cb, item); +	}  	xmlNodePtr children;  	guint position; @@ -736,6 +758,10 @@ parse_layout (DbusmenuClient * client, const gchar * layout)  	xmlNodePtr root = xmlDocGetRootElement(xmldoc);  	DbusmenuMenuitem * oldroot = priv->root; +	if (oldroot != NULL) { +		g_object_ref(oldroot); +	} +  	priv->root = parse_layout_xml(client, root, priv->root, NULL, priv->menuproxy);  	xmlFreeDoc(xmldoc); @@ -747,6 +773,16 @@ parse_layout (DbusmenuClient * client, const gchar * layout)  		#ifdef MASSIVEDEBUGGING  		g_debug("Client signaling root changed.");  		#endif  + +		/* Switch the root around */ +		g_object_ref(priv->root); +		dbusmenu_menuitem_set_root(priv->root, TRUE); + +		if (oldroot != NULL) { +			dbusmenu_menuitem_set_root(oldroot, FALSE); +			g_object_unref(oldroot); +		} +  		g_signal_emit(G_OBJECT(client), signals[ROOT_CHANGED], 0, priv->root, TRUE);  	} diff --git a/libdbusmenu-glib/dbus-menu.xml b/libdbusmenu-glib/dbus-menu.xml index 121725e..9af78fa 100644 --- a/libdbusmenu-glib/dbus-menu.xml +++ b/libdbusmenu-glib/dbus-menu.xml @@ -1,6 +1,6 @@  <?xml version="1.0" encoding="UTF-8"?>  <!-- -A library to allow applictions to provide simple indications of +A library to allow applications to provide simple indications of  information to be displayed to users of the application through the  interface shell. @@ -8,7 +8,7 @@ Copyright 2009 Canonical Ltd.  Authors:      Ted Gould <ted@canonical.com> -    Aurelien Gateau <ted@canonical.com> +    Aurélien Gâteau <aurelien.gateau@canonical.com>  This program is free software: you can redistribute it and/or modify it   under the terms of either or both of the following licenses: @@ -37,6 +37,9 @@ License version 3 and version 2.1 along with this program.  If not, see  		Items are represented with a unique numeric id and a dictionary of  		properties. +		To reduce the amount of DBus traffic, a property should only be returned +		if its value is not the default value. +  		Available properties are:  		<table> @@ -151,13 +154,13 @@ License version 3 and version 2.1 along with this program.  If not, see  			XML syntax:  			@verbatim -<menu id="1" revision="2"> # Root container -	<menu id="2" revision="2"> # First level menu, for example "File" -		<menu id="3" revision="2"/> ~ Second level menu, for example "Open" -		<menu id="4" revision="3"/> +<menu id="0"> # Root container +	<menu id="1"> # First level menu, for example "File" +		<menu id="2"/> ~ Second level menu, for example "Open" +		<menu id="3"/>  		...  	</menu> -	<menu id="5" revision="2"> # Another first level menu, say "Edit" +	<menu id="4"> # Another first level menu, say "Edit"  		...  	</menu>  	... diff --git a/libdbusmenu-glib/menuitem.c b/libdbusmenu-glib/menuitem.c index 14bc2db..a2d2682 100644 --- a/libdbusmenu-glib/menuitem.c +++ b/libdbusmenu-glib/menuitem.c @@ -80,6 +80,8 @@ enum {  	PROP_ID,  }; +#define PROP_ID_S  "id" +  #define DBUSMENU_MENUITEM_GET_PRIVATE(o) \  (G_TYPE_INSTANCE_GET_PRIVATE ((o), DBUSMENU_TYPE_MENUITEM, DbusmenuMenuitemPrivate)) @@ -210,9 +212,9 @@ dbusmenu_menuitem_class_init (DbusmenuMenuitemClass *klass)  	                                           G_TYPE_NONE, 0, G_TYPE_NONE);  	g_object_class_install_property (object_class, PROP_ID, -	                                 g_param_spec_int("id", "ID for the menu item", +	                                 g_param_spec_int(PROP_ID_S, "ID for the menu item",  	                                              "This is a unique indentifier for the menu item.", -												  0, 30000, 0, +												  -1, 30000, -1,  	                                              G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));  	/* Check transfer functions for GValue */ @@ -249,7 +251,7 @@ g_value_transform_STRING_INT (const GValue * in, GValue * out)  	return;  } -static gint menuitem_next_id = 0; +static gint menuitem_next_id = 1;  /* A small little function to both clear the insides of a      value as well as the memory it itself uses. */ @@ -270,7 +272,7 @@ dbusmenu_menuitem_init (DbusmenuMenuitem *self)  {  	DbusmenuMenuitemPrivate * priv = DBUSMENU_MENUITEM_GET_PRIVATE(self); -	priv->id = 0;  +	priv->id = -1;   	priv->children = NULL;  	priv->properties = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, _g_value_free); @@ -323,6 +325,9 @@ set_property (GObject * obj, guint id, const GValue * value, GParamSpec * pspec)  			menuitem_next_id = priv->id + 1;  		}  		break; +	default: +		G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, id, pspec); +		break;  	}  	return; @@ -335,7 +340,17 @@ get_property (GObject * obj, guint id, GValue * value, GParamSpec * pspec)  	switch (id) {  	case PROP_ID: -		g_value_set_int(value, priv->id); +		if (priv->id == -1) { +			priv->id = menuitem_next_id++; +		} +		if (dbusmenu_menuitem_get_root(DBUSMENU_MENUITEM(obj))) { +			g_value_set_int(value, 0); +		} else { +			g_value_set_int(value, priv->id); +		} +		break; +	default: +		G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, id, pspec);  		break;  	} @@ -365,7 +380,7 @@ handle_event (DbusmenuMenuitem * mi, const gchar * name, const GValue * value, g  DbusmenuMenuitem *  dbusmenu_menuitem_new (void)  { -	return g_object_new(DBUSMENU_TYPE_MENUITEM, "id", menuitem_next_id++, NULL); +	return g_object_new(DBUSMENU_TYPE_MENUITEM, NULL);  }  /** @@ -379,7 +394,7 @@ dbusmenu_menuitem_new (void)  DbusmenuMenuitem *  dbusmenu_menuitem_new_with_id (gint id)  { -	DbusmenuMenuitem * mi = g_object_new(DBUSMENU_TYPE_MENUITEM, "id", id, NULL); +	DbusmenuMenuitem * mi = g_object_new(DBUSMENU_TYPE_MENUITEM, PROP_ID_S, id, NULL);  	/* g_debug("New Menuitem id %d goal id %d", dbusmenu_menuitem_get_id(mi), id); */  	return mi;  } @@ -392,15 +407,19 @@ dbusmenu_menuitem_new_with_id (gint id)  	Return value: The ID of the @mi.  */ -guint +gint  dbusmenu_menuitem_get_id (DbusmenuMenuitem * mi)  { -	g_return_val_if_fail(DBUSMENU_IS_MENUITEM(mi), 0); +	g_return_val_if_fail(DBUSMENU_IS_MENUITEM(mi), -1);  	GValue retval = {0}; -	g_value_init(&retval, G_TYPE_UINT); -	g_object_get_property(G_OBJECT(mi), "id", &retval); -	return g_value_get_uint(&retval); +	g_value_init(&retval, G_TYPE_INT); +	g_object_get_property(G_OBJECT(mi), PROP_ID_S, &retval); +	gint ret = g_value_get_int(&retval); +	#ifdef MASSIVEDEBUGGING +	g_debug("Getting menuitem ID: %d", ret); +	#endif +	return ret;  }  /** @@ -710,6 +729,12 @@ DbusmenuMenuitem *  dbusmenu_menuitem_find_id (DbusmenuMenuitem * mi, gint id)  {  	g_return_val_if_fail(DBUSMENU_IS_MENUITEM(mi), NULL); +	if (id == 0) { +		if (dbusmenu_menuitem_get_root(mi) == FALSE) { +			g_warning("Getting a menuitem with id zero, but it's not set as root."); +		} +		return mi; +	}  	find_id_t find_id = {mi: NULL, id: id};  	find_id_helper(mi, &find_id);  	return find_id.mi; @@ -963,6 +988,26 @@ dbusmenu_menuitem_property_exist (DbusmenuMenuitem * mi, const gchar * property)  }  /** +	dbusmenu_menuitem_property_remove: +	@mi: The #DbusmenuMenuitem to remove the property on. +	@property: The property to look for. + +	Removes a property from the menuitem. +*/ +void +dbusmenu_menuitem_property_remove (DbusmenuMenuitem * mi, const gchar * property) +{ +	g_return_if_fail(DBUSMENU_IS_MENUITEM(mi)); +	g_return_if_fail(property != NULL); + +	DbusmenuMenuitemPrivate * priv = DBUSMENU_MENUITEM_GET_PRIVATE(mi); + +	g_hash_table_remove(priv->properties, property); + +	return; +} + +/**  	dbusmenu_menuitem_properties_list:  	@mi: #DbusmenuMenuitem to list the properties on diff --git a/libdbusmenu-glib/menuitem.h b/libdbusmenu-glib/menuitem.h index 4fc144e..1382335 100644 --- a/libdbusmenu-glib/menuitem.h +++ b/libdbusmenu-glib/menuitem.h @@ -127,7 +127,7 @@ GType dbusmenu_menuitem_get_type (void);  DbusmenuMenuitem * dbusmenu_menuitem_new (void) G_GNUC_WARN_UNUSED_RESULT;  DbusmenuMenuitem * dbusmenu_menuitem_new_with_id (gint id) G_GNUC_WARN_UNUSED_RESULT; -guint dbusmenu_menuitem_get_id (DbusmenuMenuitem * mi); +gint dbusmenu_menuitem_get_id (DbusmenuMenuitem * mi);  GList * dbusmenu_menuitem_get_children (DbusmenuMenuitem * mi);  GList * dbusmenu_menuitem_take_children (DbusmenuMenuitem * mi) G_GNUC_WARN_UNUSED_RESULT; @@ -152,6 +152,7 @@ gint dbusmenu_menuitem_property_get_int (DbusmenuMenuitem * mi, const gchar * pr  gboolean dbusmenu_menuitem_property_exist (DbusmenuMenuitem * mi, const gchar * property);  GList * dbusmenu_menuitem_properties_list (DbusmenuMenuitem * mi) G_GNUC_WARN_UNUSED_RESULT;  GHashTable * dbusmenu_menuitem_properties_copy (DbusmenuMenuitem * mi); +void dbusmenu_menuitem_property_remove (DbusmenuMenuitem * mi, const gchar * property);  void dbusmenu_menuitem_set_root (DbusmenuMenuitem * mi, gboolean root);  gboolean dbusmenu_menuitem_get_root (DbusmenuMenuitem * mi); diff --git a/libdbusmenu-glib/server.c b/libdbusmenu-glib/server.c index 7d8e5c3..f8267c2 100644 --- a/libdbusmenu-glib/server.c +++ b/libdbusmenu-glib/server.c @@ -237,12 +237,14 @@ set_property (GObject * obj, guint id, const GValue * value, GParamSpec * pspec)  	case PROP_ROOT_NODE:  		if (priv->root != NULL) {  			dbusmenu_menuitem_foreach(priv->root, menuitem_signals_remove, obj); +			dbusmenu_menuitem_set_root(priv->root, FALSE);  			g_object_unref(G_OBJECT(priv->root));  			priv->root = NULL;  		}  		priv->root = DBUSMENU_MENUITEM(g_value_get_object(value));  		if (priv->root != NULL) {  			g_object_ref(G_OBJECT(priv->root)); +			dbusmenu_menuitem_set_root(priv->root, TRUE);  			dbusmenu_menuitem_foreach(priv->root, menuitem_signals_create, obj);  		} else {  			g_debug("Setting root node to NULL"); @@ -516,7 +518,7 @@ static gboolean  _dbusmenu_server_get_children (DbusmenuServer * server, gint id, GPtrArray * properties, GPtrArray ** output, GError ** error)  {  	DbusmenuServerPrivate * priv = DBUSMENU_SERVER_GET_PRIVATE(server); -	DbusmenuMenuitem * mi = id == 0 ? priv->root : dbusmenu_menuitem_find_id(priv->root, id); +	DbusmenuMenuitem * mi = dbusmenu_menuitem_find_id(priv->root, id);  	if (mi == NULL) {  		if (error != NULL) { | 
