X-Git-Url: http://review.tizen.org/git/?a=blobdiff_plain;f=gio%2Fgicon.c;h=b5080dae28e195c0daa29310e5feff40e155033f;hb=e608ec7b2e47d29fa189fca6e97f484f41c115a4;hp=c78acd36215b6688730ec744c6ddcdbe7db6e354;hpb=d8ca6404229e5b64d2bf2e1a3660ad9fe7feefdd;p=platform%2Fupstream%2Fglib.git diff --git a/gio/gicon.c b/gio/gicon.c index c78acd3..b5080da 100644 --- a/gio/gicon.c +++ b/gio/gicon.c @@ -13,9 +13,7 @@ * 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, write to the - * Free Software Foundation, Inc., 59 Temple Place, Suite 330, - * Boston, MA 02111-1307, USA. + * Public License along with this library; if not, see . * * Author: Alexander Larsson */ @@ -28,8 +26,11 @@ #include "gthemedicon.h" #include "gfileicon.h" #include "gemblemedicon.h" +#include "gbytesicon.h" #include "gfile.h" #include "gioerror.h" +#include "gioenumtypes.h" +#include "gvfs.h" #include "glibintl.h" @@ -49,17 +50,27 @@ * #GIcon does not provide the actual pixmap for the icon as this is out * of GIO's scope, however implementations of #GIcon may contain the name * of an icon (see #GThemedIcon), or the path to an icon (see #GLoadableIcon). - * + * * To obtain a hash of a #GIcon, see g_icon_hash(). - * + * * To check if two #GIcons are equal, see g_icon_equal(). * - * For serializing a #GIcon, use g_icon_to_string() and - * g_icon_new_for_string(). + * For serializing a #GIcon, use g_icon_serialize() and + * g_icon_deserialize(). + * + * If you want to consume #GIcon (for example, in a toolkit) you must + * be prepared to handle at least the three following cases: + * #GLoadableIcon, #GThemedIcon and #GEmblemedIcon. It may also make + * sense to have fast-paths for other cases (like handling #GdkPixbuf + * directly, for example) but all compliant #GIcon implementations + * outside of GIO must implement #GLoadableIcon. * * If your application or library provides one or more #GIcon - * implementations you need to ensure that each #GType is registered - * with the type system prior to calling g_icon_new_for_string(). + * implementations you need to ensure that your new implementation also + * implements #GLoadableIcon. Additionally, you must provide an + * implementation of g_icon_serialize() that gives a result that is + * understood by g_icon_deserialize(), yielding one of the built-in icon + * types. **/ typedef GIconIface GIconInterface; @@ -94,8 +105,8 @@ g_icon_hash (gconstpointer icon) /** * g_icon_equal: - * @icon1: pointer to the first #GIcon. - * @icon2: pointer to the second #GIcon. + * @icon1: (allow-none): pointer to the first #GIcon. + * @icon2: (allow-none): pointer to the second #GIcon. * * Checks if two icons are equal. * @@ -124,7 +135,6 @@ g_icon_equal (GIcon *icon1, static gboolean g_icon_to_string_tokenized (GIcon *icon, GString *s) { - char *ret; GPtrArray *tokens; gint version; GIconIface *icon_iface; @@ -133,8 +143,6 @@ g_icon_to_string_tokenized (GIcon *icon, GString *s) g_return_val_if_fail (icon != NULL, FALSE); g_return_val_if_fail (G_IS_ICON (icon), FALSE); - ret = NULL; - icon_iface = G_ICON_GET_IFACE (icon); if (icon_iface->to_tokens == NULL) return FALSE; @@ -185,23 +193,18 @@ g_icon_to_string_tokenized (GIcon *icon, GString *s) * The encoding of the returned string is proprietary to #GIcon except * in the following two cases * - * - * - * If @icon is a #GFileIcon, the returned string is a native path - * (such as /path/to/my icon.png) without escaping - * if the #GFile for @icon is a native file. If the file is not - * native, the returned string is the result of g_file_get_uri() - * (such as sftp://path/to/my%%20icon.png). - * - * - * If @icon is a #GThemedIcon with exactly one name, the encoding is - * simply the name (such as network-server). - * - * + * - If @icon is a #GFileIcon, the returned string is a native path + * (such as `/path/to/my icon.png`) without escaping + * if the #GFile for @icon is a native file. If the file is not + * native, the returned string is the result of g_file_get_uri() + * (such as `sftp://path/to/my%20icon.png`). + * + * - If @icon is a #GThemedIcon with exactly one name, the encoding is + * simply the name (such as `network-server`). * * Virtual: to_tokens - * Returns: An allocated NUL-terminated UTF8 string or %NULL if @icon can't - * be serialized. Use g_free() to free. + * Returns: (nullable): An allocated NUL-terminated UTF8 string or + * %NULL if @icon can't be serialized. Use g_free() to free. * * Since: 2.20 */ @@ -378,11 +381,37 @@ g_icon_new_from_tokens (char **tokens, static void ensure_builtin_icon_types (void) { - static volatile GType t; - t = g_themed_icon_get_type (); - t = g_file_icon_get_type (); - t = g_emblemed_icon_get_type (); - t = g_emblem_get_type (); + g_type_ensure (G_TYPE_THEMED_ICON); + g_type_ensure (G_TYPE_FILE_ICON); + g_type_ensure (G_TYPE_EMBLEMED_ICON); + g_type_ensure (G_TYPE_EMBLEM); +} + +/* handles the 'simple' cases: GFileIcon and GThemedIcon */ +static GIcon * +g_icon_new_for_string_simple (const gchar *str) +{ + gchar *scheme; + GIcon *icon; + + if (str[0] == '.') + return NULL; + + /* handle special GFileIcon and GThemedIcon cases */ + scheme = g_uri_parse_scheme (str); + if (scheme != NULL || str[0] == '/' || str[0] == G_DIR_SEPARATOR) + { + GFile *location; + location = g_file_new_for_commandline_arg (str); + icon = g_file_icon_new (location); + g_object_unref (location); + } + else + icon = g_themed_icon_new (str); + + g_free (scheme); + + return icon; } /** @@ -406,48 +435,248 @@ GIcon * g_icon_new_for_string (const gchar *str, GError **error) { - GIcon *icon; + GIcon *icon = NULL; g_return_val_if_fail (str != NULL, NULL); - ensure_builtin_icon_types (); + icon = g_icon_new_for_string_simple (str); + if (icon) + return icon; - icon = NULL; + ensure_builtin_icon_types (); - if (*str == '.') + if (g_str_has_prefix (str, G_ICON_SERIALIZATION_MAGIC0)) { - if (g_str_has_prefix (str, G_ICON_SERIALIZATION_MAGIC0)) - { - gchar **tokens; - - /* handle tokenized encoding */ - tokens = g_strsplit (str + sizeof (G_ICON_SERIALIZATION_MAGIC0) - 1, " ", 0); - icon = g_icon_new_from_tokens (tokens, error); - g_strfreev (tokens); - } - else - g_set_error_literal (error, - G_IO_ERROR, - G_IO_ERROR_INVALID_ARGUMENT, - _("Can't handle the supplied version the icon encoding")); + gchar **tokens; + + /* handle tokenized encoding */ + tokens = g_strsplit (str + sizeof (G_ICON_SERIALIZATION_MAGIC0) - 1, " ", 0); + icon = g_icon_new_from_tokens (tokens, error); + g_strfreev (tokens); } else + g_set_error_literal (error, + G_IO_ERROR, + G_IO_ERROR_INVALID_ARGUMENT, + _("Can't handle the supplied version of the icon encoding")); + + return icon; +} + +static GEmblem * +g_icon_deserialize_emblem (GVariant *value) +{ + GVariant *emblem_metadata; + GVariant *emblem_data; + const gchar *origin_nick; + GIcon *emblem_icon; + GEmblem *emblem; + + g_variant_get (value, "(v@a{sv})", &emblem_data, &emblem_metadata); + + emblem = NULL; + + emblem_icon = g_icon_deserialize (emblem_data); + if (emblem_icon != NULL) + { + /* Check if we should create it with an origin. */ + if (g_variant_lookup (emblem_metadata, "origin", "&s", &origin_nick)) + { + GEnumClass *origin_class; + GEnumValue *origin_value; + + origin_class = g_type_class_ref (G_TYPE_EMBLEM_ORIGIN); + origin_value = g_enum_get_value_by_nick (origin_class, origin_nick); + if (origin_value) + emblem = g_emblem_new_with_origin (emblem_icon, origin_value->value); + g_type_class_unref (origin_class); + } + + /* We didn't create it with an origin, so do it without. */ + if (emblem == NULL) + emblem = g_emblem_new (emblem_icon); + + g_object_unref (emblem_icon); + } + + g_variant_unref (emblem_metadata); + g_variant_unref (emblem_data); + + return emblem; +} + +static GIcon * +g_icon_deserialize_emblemed (GVariant *value) +{ + GVariantIter *emblems; + GVariant *icon_data; + GIcon *main_icon; + GIcon *icon; + + g_variant_get (value, "(va(va{sv}))", &icon_data, &emblems); + main_icon = g_icon_deserialize (icon_data); + + if (main_icon) { - gchar *scheme; + GVariant *emblem_data; + + icon = g_emblemed_icon_new (main_icon, NULL); - /* handle special GFileIcon and GThemedIcon cases */ - scheme = g_uri_parse_scheme (str); - if (scheme != NULL || str[0] == '/') + while ((emblem_data = g_variant_iter_next_value (emblems))) { - GFile *location; - location = g_file_new_for_commandline_arg (str); - icon = g_file_icon_new (location); - g_object_unref (location); + GEmblem *emblem; + + emblem = g_icon_deserialize_emblem (emblem_data); + + if (emblem) + { + g_emblemed_icon_add_emblem (G_EMBLEMED_ICON (icon), emblem); + g_object_unref (emblem); + } + + g_variant_unref (emblem_data); } - else - icon = g_themed_icon_new (str); - g_free (scheme); + + g_object_unref (main_icon); + } + else + icon = NULL; + + g_variant_iter_free (emblems); + g_variant_unref (icon_data); + + return icon; +} + +/** + * g_icon_deserialize: + * @value: a #GVariant created with g_icon_serialize() + * + * Deserializes a #GIcon previously serialized using g_icon_serialize(). + * + * Returns: (transfer full): a #GIcon, or %NULL when deserialization fails. + * + * Since: 2.38 + */ +GIcon * +g_icon_deserialize (GVariant *value) +{ + const gchar *tag; + GVariant *val; + GIcon *icon; + + g_return_val_if_fail (value != NULL, NULL); + g_return_val_if_fail (g_variant_is_of_type (value, G_VARIANT_TYPE_STRING) || + g_variant_is_of_type (value, G_VARIANT_TYPE ("(sv)")), NULL); + + /* Handle some special cases directly so that people can hard-code + * stuff into GMenuModel xml files without resorting to using GVariant + * text format to describe one of the explicitly-tagged possibilities + * below. + */ + if (g_variant_is_of_type (value, G_VARIANT_TYPE_STRING)) + return g_icon_new_for_string_simple (g_variant_get_string (value, NULL)); + + /* Otherwise, use the tagged union format */ + g_variant_get (value, "(&sv)", &tag, &val); + + icon = NULL; + + if (g_str_equal (tag, "file") && g_variant_is_of_type (val, G_VARIANT_TYPE_STRING)) + { + GFile *file; + + file = g_file_new_for_commandline_arg (g_variant_get_string (val, NULL)); + icon = g_file_icon_new (file); + g_object_unref (file); + } + else if (g_str_equal (tag, "themed") && g_variant_is_of_type (val, G_VARIANT_TYPE_STRING_ARRAY)) + { + const gchar **names; + gsize size; + + names = g_variant_get_strv (val, &size); + icon = g_themed_icon_new_from_names ((gchar **) names, size); + g_free (names); + } + else if (g_str_equal (tag, "bytes") && g_variant_is_of_type (val, G_VARIANT_TYPE_BYTESTRING)) + { + GBytes *bytes; + + bytes = g_variant_get_data_as_bytes (val); + icon = g_bytes_icon_new (bytes); + g_bytes_unref (bytes); + } + else if (g_str_equal (tag, "emblem") && g_variant_is_of_type (val, G_VARIANT_TYPE ("(va{sv})"))) + { + GEmblem *emblem; + + emblem = g_icon_deserialize_emblem (val); + if (emblem) + icon = G_ICON (emblem); + } + else if (g_str_equal (tag, "emblemed") && g_variant_is_of_type (val, G_VARIANT_TYPE ("(va(va{sv}))"))) + { + icon = g_icon_deserialize_emblemed (val); + } + else if (g_str_equal (tag, "gvfs")) + { + GVfsClass *class; + GVfs *vfs; + + vfs = g_vfs_get_default (); + class = G_VFS_GET_CLASS (vfs); + if (class->deserialize_icon) + icon = (* class->deserialize_icon) (vfs, val); } + g_variant_unref (val); + return icon; } + +/** + * g_icon_serialize: + * @icon: a #GIcon + * + * Serializes a #GIcon into a #GVariant. An equivalent #GIcon can be retrieved + * back by calling g_icon_deserialize() on the returned value. + * As serialization will avoid using raw icon data when possible, it only + * makes sense to transfer the #GVariant between processes on the same machine, + * (as opposed to over the network), and within the same file system namespace. + * + * Returns: (transfer full): a #GVariant, or %NULL when serialization fails. + * + * Since: 2.38 + */ +GVariant * +g_icon_serialize (GIcon *icon) +{ + GIconInterface *iface; + GVariant *result; + + iface = G_ICON_GET_IFACE (icon); + + if (!iface->serialize) + { + g_critical ("g_icon_serialize() on icon type '%s' is not implemented", G_OBJECT_TYPE_NAME (icon)); + return NULL; + } + + result = (* iface->serialize) (icon); + + if (result) + { + g_variant_take_ref (result); + + if (!g_variant_is_of_type (result, G_VARIANT_TYPE ("(sv)"))) + { + g_critical ("g_icon_serialize() on icon type '%s' returned GVariant of type '%s' but it must return " + "one with type '(sv)'", G_OBJECT_TYPE_NAME (icon), g_variant_get_type_string (result)); + g_variant_unref (result); + result = NULL; + } + } + + return result; +}