1 /* GIO - GLib Input, Output and Streaming Library
3 * Copyright (C) 2006-2007 Red Hat, Inc.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General
16 * Public License along with this library; if not, write to the
17 * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
18 * Boston, MA 02111-1307, USA.
20 * Author: Alexander Larsson <alexl@redhat.com>
28 #include "gthemedicon.h"
29 #include "gfileicon.h"
30 #include "gemblemedicon.h"
38 /* There versioning of this is implicit, version 1 would be ".1 " */
39 #define G_ICON_SERIALIZATION_MAGIC0 ". "
43 * @short_description: Interface for icons
46 * #GIcon is a very minimal interface for icons. It provides functions
47 * for checking the equality of two icons, hashing of icons and
48 * serializing an icon to and from strings.
50 * #GIcon does not provide the actual pixmap for the icon as this is out
51 * of GIO's scope, however implementations of #GIcon may contain the name
52 * of an icon (see #GThemedIcon), or the path to an icon (see #GLoadableIcon).
54 * To obtain a hash of a #GIcon, see g_icon_hash().
56 * To check if two #GIcons are equal, see g_icon_equal().
58 * For serializing a #GIcon, use g_icon_to_string() and
59 * g_icon_new_for_string().
61 * If your application or library provides one or more #GIcon
62 * implementations you need to ensure that each #GType is registered
63 * with the type system prior to calling g_icon_new_for_string().
66 typedef GIconIface GIconInterface;
67 G_DEFINE_INTERFACE(GIcon, g_icon, G_TYPE_OBJECT)
70 g_icon_default_init (GIconInterface *iface)
76 * @icon: #gconstpointer to an icon object.
78 * Gets a hash for an icon.
80 * Returns: a #guint containing a hash for the @icon, suitable for
81 * use in a #GHashTable or similar data structure.
84 g_icon_hash (gconstpointer icon)
88 g_return_val_if_fail (G_IS_ICON (icon), 0);
90 iface = G_ICON_GET_IFACE (icon);
92 return (* iface->hash) ((GIcon *)icon);
97 * @icon1: pointer to the first #GIcon.
98 * @icon2: pointer to the second #GIcon.
100 * Checks if two icons are equal.
102 * Returns: %TRUE if @icon1 is equal to @icon2. %FALSE otherwise.
105 g_icon_equal (GIcon *icon1,
110 if (icon1 == NULL && icon2 == NULL)
113 if (icon1 == NULL || icon2 == NULL)
116 if (G_TYPE_FROM_INSTANCE (icon1) != G_TYPE_FROM_INSTANCE (icon2))
119 iface = G_ICON_GET_IFACE (icon1);
121 return (* iface->equal) (icon1, icon2);
125 g_icon_to_string_tokenized (GIcon *icon, GString *s)
130 GIconIface *icon_iface;
133 g_return_val_if_fail (icon != NULL, FALSE);
134 g_return_val_if_fail (G_IS_ICON (icon), FALSE);
138 icon_iface = G_ICON_GET_IFACE (icon);
139 if (icon_iface->to_tokens == NULL)
142 tokens = g_ptr_array_new ();
143 if (!icon_iface->to_tokens (icon, tokens, &version))
145 g_ptr_array_free (tokens, TRUE);
149 /* format: TypeName[.Version] <token_0> .. <token_N-1>
150 version 0 is implicit and can be omitted
151 all the tokens are url escaped to ensure they have no spaces in them */
153 g_string_append (s, g_type_name_from_instance ((GTypeInstance *)icon));
155 g_string_append_printf (s, ".%d", version);
157 for (i = 0; i < tokens->len; i++)
161 token = g_ptr_array_index (tokens, i);
163 g_string_append_c (s, ' ');
164 /* We really only need to escape spaces here, so allow lots of otherwise reserved chars */
165 g_string_append_uri_escaped (s, token,
166 G_URI_RESERVED_CHARS_ALLOWED_IN_PATH, TRUE);
171 g_ptr_array_free (tokens, TRUE);
180 * Generates a textual representation of @icon that can be used for
181 * serialization such as when passing @icon to a different process or
182 * saving it to persistent storage. Use g_icon_new_for_string() to
183 * get @icon back from the returned string.
185 * The encoding of the returned string is proprietary to #GIcon except
186 * in the following two cases
190 * If @icon is a #GFileIcon, the returned string is a native path
191 * (such as <literal>/path/to/my icon.png</literal>) without escaping
192 * if the #GFile for @icon is a native file. If the file is not
193 * native, the returned string is the result of g_file_get_uri()
194 * (such as <literal>sftp://path/to/my%%20icon.png</literal>).
197 * If @icon is a #GThemedIcon with exactly one name, the encoding is
198 * simply the name (such as <literal>network-server</literal>).
202 * Returns: An allocated NUL-terminated UTF8 string or %NULL if @icon can't
203 * be serialized. Use g_free() to free.
208 g_icon_to_string (GIcon *icon)
212 g_return_val_if_fail (icon != NULL, NULL);
213 g_return_val_if_fail (G_IS_ICON (icon), NULL);
217 if (G_IS_FILE_ICON (icon))
221 file = g_file_icon_get_file (G_FILE_ICON (icon));
222 if (g_file_is_native (file))
224 ret = g_file_get_path (file);
225 if (!g_utf8_validate (ret, -1, NULL))
232 ret = g_file_get_uri (file);
234 else if (G_IS_THEMED_ICON (icon))
236 const char * const *names;
238 names = g_themed_icon_get_names (G_THEMED_ICON (icon));
241 names[0][0] != '.' && /* Allowing icons starting with dot would break G_ICON_SERIALIZATION_MAGIC0 */
242 g_utf8_validate (names[0], -1, NULL) && /* Only return utf8 strings */
244 ret = g_strdup (names[0]);
251 s = g_string_new (G_ICON_SERIALIZATION_MAGIC0);
253 if (g_icon_to_string_tokenized (icon, s))
254 ret = g_string_free (s, FALSE);
256 g_string_free (s, TRUE);
263 g_icon_new_from_tokens (char **tokens,
267 char *typename, *version_str;
270 GIconIface *icon_iface;
279 num_tokens = g_strv_length (tokens);
285 G_IO_ERROR_INVALID_ARGUMENT,
286 _("Wrong number of tokens (%d)"),
291 typename = tokens[0];
292 version_str = strchr (typename, '.');
300 type = g_type_from_name (tokens[0]);
305 G_IO_ERROR_INVALID_ARGUMENT,
306 _("No type for class name %s"),
311 if (!g_type_is_a (type, G_TYPE_ICON))
315 G_IO_ERROR_INVALID_ARGUMENT,
316 _("Type %s does not implement the GIcon interface"),
321 klass = g_type_class_ref (type);
326 G_IO_ERROR_INVALID_ARGUMENT,
327 _("Type %s is not classed"),
335 version = strtol (version_str, &endp, 10);
336 if (endp == NULL || *endp != '\0')
340 G_IO_ERROR_INVALID_ARGUMENT,
341 _("Malformed version number: %s"),
347 icon_iface = g_type_interface_peek (klass, G_TYPE_ICON);
348 g_assert (icon_iface != NULL);
350 if (icon_iface->from_tokens == NULL)
354 G_IO_ERROR_INVALID_ARGUMENT,
355 _("Type %s does not implement from_tokens() on the GIcon interface"),
360 for (i = 1; i < num_tokens; i++)
365 tokens[i] = g_uri_unescape_string (escaped, NULL);
369 icon = icon_iface->from_tokens (tokens + 1, num_tokens - 1, version, error);
373 g_type_class_unref (klass);
378 ensure_builtin_icon_types (void)
380 static volatile GType t;
381 t = g_themed_icon_get_type ();
382 t = g_file_icon_get_type ();
383 t = g_emblemed_icon_get_type ();
384 t = g_emblem_get_type ();
388 * g_icon_new_for_string:
389 * @str: A string obtained via g_icon_to_string().
390 * @error: Return location for error.
392 * Generate a #GIcon instance from @str. This function can fail if
393 * @str is not valid - see g_icon_to_string() for discussion.
395 * If your application or library provides one or more #GIcon
396 * implementations you need to ensure that each #GType is registered
397 * with the type system prior to calling g_icon_new_for_string().
399 * Returns: An object implementing the #GIcon interface or %NULL if
405 g_icon_new_for_string (const gchar *str,
410 g_return_val_if_fail (str != NULL, NULL);
412 ensure_builtin_icon_types ();
418 if (g_str_has_prefix (str, G_ICON_SERIALIZATION_MAGIC0))
422 /* handle tokenized encoding */
423 tokens = g_strsplit (str + sizeof (G_ICON_SERIALIZATION_MAGIC0) - 1, " ", 0);
424 icon = g_icon_new_from_tokens (tokens, error);
428 g_set_error_literal (error,
430 G_IO_ERROR_INVALID_ARGUMENT,
431 _("Can't handle the supplied version the icon encoding"));
437 /* handle special GFileIcon and GThemedIcon cases */
438 scheme = g_uri_parse_scheme (str);
439 if (scheme != NULL || str[0] == '/')
442 location = g_file_new_for_commandline_arg (str);
443 icon = g_file_icon_new (location);
444 g_object_unref (location);
447 icon = g_themed_icon_new (str);
456 #include "gioaliasdef.c"