| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | /* GIO - GLib Input, Output and Streaming Library
 | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * Copyright (C) 2006-2007 Red Hat, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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 License, 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, write to the | 
					
						
							|  |  |  |  * Free Software Foundation, Inc., 59 Temple Place, Suite 330, | 
					
						
							|  |  |  |  * Boston, MA 02111-1307, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Author: Alexander Larsson <alexl@redhat.com> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-22 15:10:51 +00:00
										 |  |  | #include "config.h"
 | 
					
						
							| 
									
										
										
										
											2008-07-01 06:32:35 +00:00
										 |  |  | #include "gasyncresult.h"
 | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | #include "gsimpleasyncresult.h"
 | 
					
						
							| 
									
										
										
										
											2008-07-01 06:32:35 +00:00
										 |  |  | #include "gicon.h"
 | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | #include "gloadableicon.h"
 | 
					
						
							|  |  |  | #include "glibintl.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-28 12:39:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-27 14:00:13 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * SECTION:gloadableicon | 
					
						
							|  |  |  |  * @short_description: Loadable Icons | 
					
						
							| 
									
										
										
										
											2008-02-21 18:20:17 +00:00
										 |  |  |  * @include: gio/gio.h | 
					
						
							| 
									
										
										
										
											2007-11-27 14:00:13 +00:00
										 |  |  |  * @see_also: #GIcon, #GThemedIcon | 
					
						
							|  |  |  |  *  | 
					
						
							| 
									
										
										
										
											2007-12-18 02:52:11 +00:00
										 |  |  |  * Extends the #GIcon interface and adds the ability to  | 
					
						
							|  |  |  |  * load icons from streams. | 
					
						
							| 
									
										
										
										
											2007-11-27 14:00:13 +00:00
										 |  |  |  **/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | static void          g_loadable_icon_real_load_async  (GLoadableIcon        *icon, | 
					
						
							|  |  |  | 						       int                   size, | 
					
						
							|  |  |  | 						       GCancellable         *cancellable, | 
					
						
							|  |  |  | 						       GAsyncReadyCallback   callback, | 
					
						
							|  |  |  | 						       gpointer              user_data); | 
					
						
							|  |  |  | static GInputStream *g_loadable_icon_real_load_finish (GLoadableIcon        *icon, | 
					
						
							|  |  |  | 						       GAsyncResult         *res, | 
					
						
							|  |  |  | 						       char                **type, | 
					
						
							|  |  |  | 						       GError              **error); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 10:42:58 +01:00
										 |  |  | typedef GLoadableIconIface GLoadableIconInterface; | 
					
						
							|  |  |  | G_DEFINE_INTERFACE(GLoadableIcon, g_loadable_icon, G_TYPE_ICON) | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2009-12-01 10:42:58 +01:00
										 |  |  | g_loadable_icon_default_init (GLoadableIconIface *iface) | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   iface->load_async = g_loadable_icon_real_load_async; | 
					
						
							|  |  |  |   iface->load_finish = g_loadable_icon_real_load_finish; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * g_loadable_icon_load: | 
					
						
							| 
									
										
										
										
											2007-11-27 14:00:13 +00:00
										 |  |  |  * @icon: a #GLoadableIcon. | 
					
						
							|  |  |  |  * @size: an integer. | 
					
						
							| 
									
										
										
										
											2010-12-27 16:08:46 +01:00
										 |  |  |  * @type: (out) (allow-none): a location to store the type of the | 
					
						
							|  |  |  |  *        loaded icon, %NULL to ignore. | 
					
						
							|  |  |  |  * @cancellable: (allow-none): optional #GCancellable object, %NULL to ignore.  | 
					
						
							| 
									
										
										
										
											2011-08-29 14:49:32 -04:00
										 |  |  |  * @error: a #GError location to store the error occurring, or %NULL to  | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  |  * ignore. | 
					
						
							| 
									
										
										
										
											2007-11-27 14:00:13 +00:00
										 |  |  |  *  | 
					
						
							|  |  |  |  * Loads a loadable icon. For the asynchronous version of this function,  | 
					
						
							|  |  |  |  * see g_loadable_icon_load_async(). | 
					
						
							|  |  |  |  *  | 
					
						
							| 
									
										
										
										
											2010-09-24 18:24:41 -03:00
										 |  |  |  * Returns: (transfer full): a #GInputStream to read the icon from. | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  |  **/ | 
					
						
							|  |  |  | GInputStream * | 
					
						
							| 
									
										
										
										
											2007-11-29 07:17:59 +00:00
										 |  |  | g_loadable_icon_load (GLoadableIcon  *icon, | 
					
						
							|  |  |  | 		      int             size, | 
					
						
							|  |  |  | 		      char          **type, | 
					
						
							|  |  |  | 		      GCancellable   *cancellable, | 
					
						
							|  |  |  | 		      GError        **error) | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   GLoadableIconIface *iface; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   g_return_val_if_fail (G_IS_LOADABLE_ICON (icon), NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   iface = G_LOADABLE_ICON_GET_IFACE (icon); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (* iface->load) (icon, size, type, cancellable, error); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * g_loadable_icon_load_async: | 
					
						
							| 
									
										
										
										
											2007-11-27 14:00:13 +00:00
										 |  |  |  * @icon: a #GLoadableIcon. | 
					
						
							|  |  |  |  * @size: an integer. | 
					
						
							| 
									
										
										
										
											2010-12-27 16:08:46 +01:00
										 |  |  |  * @cancellable: (allow-none): optional #GCancellable object, %NULL to ignore.  | 
					
						
							|  |  |  |  * @callback: (scope async): a #GAsyncReadyCallback to call when the | 
					
						
							|  |  |  |  *            request is satisfied | 
					
						
							|  |  |  |  * @user_data: (closure): the data to pass to callback function | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  |  *  | 
					
						
							| 
									
										
										
										
											2007-11-27 14:00:13 +00:00
										 |  |  |  * Loads an icon asynchronously. To finish this function, see  | 
					
						
							|  |  |  |  * g_loadable_icon_load_finish(). For the synchronous, blocking  | 
					
						
							|  |  |  |  * version of this function, see g_loadable_icon_load(). | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  |  **/ | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2007-11-29 07:17:59 +00:00
										 |  |  | g_loadable_icon_load_async (GLoadableIcon       *icon, | 
					
						
							|  |  |  |                             int                  size, | 
					
						
							|  |  |  |                             GCancellable        *cancellable, | 
					
						
							|  |  |  |                             GAsyncReadyCallback  callback, | 
					
						
							|  |  |  |                             gpointer             user_data) | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   GLoadableIconIface *iface; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   g_return_if_fail (G_IS_LOADABLE_ICON (icon)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   iface = G_LOADABLE_ICON_GET_IFACE (icon); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   (* iface->load_async) (icon, size, cancellable, callback, user_data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * g_loadable_icon_load_finish: | 
					
						
							| 
									
										
										
										
											2007-11-27 14:00:13 +00:00
										 |  |  |  * @icon: a #GLoadableIcon. | 
					
						
							|  |  |  |  * @res: a #GAsyncResult. | 
					
						
							| 
									
										
										
										
											2007-12-09 15:51:12 +00:00
										 |  |  |  * @type: a location to store the type of the loaded icon, %NULL to ignore. | 
					
						
							| 
									
										
										
										
											2011-08-29 14:49:32 -04:00
										 |  |  |  * @error: a #GError location to store the error occurring, or %NULL to  | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  |  * ignore. | 
					
						
							| 
									
										
										
										
											2007-11-27 14:00:13 +00:00
										 |  |  |  *  | 
					
						
							|  |  |  |  * Finishes an asynchronous icon load started in g_loadable_icon_load_async(). | 
					
						
							|  |  |  |  *  | 
					
						
							| 
									
										
										
										
											2010-09-24 18:24:41 -03:00
										 |  |  |  * Returns: (transfer full): a #GInputStream to read the icon from. | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  |  **/ | 
					
						
							|  |  |  | GInputStream * | 
					
						
							| 
									
										
										
										
											2007-11-29 07:17:59 +00:00
										 |  |  | g_loadable_icon_load_finish (GLoadableIcon  *icon, | 
					
						
							|  |  |  | 			     GAsyncResult   *res, | 
					
						
							|  |  |  | 			     char          **type, | 
					
						
							|  |  |  | 			     GError        **error) | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   GLoadableIconIface *iface; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   g_return_val_if_fail (G_IS_LOADABLE_ICON (icon), NULL); | 
					
						
							|  |  |  |   g_return_val_if_fail (G_IS_ASYNC_RESULT (res), NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-10 09:00:45 -04:00
										 |  |  |   if (g_async_result_legacy_propagate_error (res, error)) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  |   iface = G_LOADABLE_ICON_GET_IFACE (icon); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (* iface->load_finish) (icon, res, type, error); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /********************************************
 | 
					
						
							|  |  |  |  *   Default implementation of async load   * | 
					
						
							|  |  |  |  ********************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |   int size; | 
					
						
							|  |  |  |   char *type; | 
					
						
							|  |  |  |   GInputStream *stream; | 
					
						
							|  |  |  | } LoadData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | load_data_free (LoadData *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (data->stream) | 
					
						
							|  |  |  |     g_object_unref (data->stream); | 
					
						
							|  |  |  |   g_free (data->type); | 
					
						
							|  |  |  |   g_free (data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | load_async_thread (GSimpleAsyncResult *res, | 
					
						
							| 
									
										
										
										
											2007-11-29 07:17:59 +00:00
										 |  |  | 		   GObject            *object, | 
					
						
							|  |  |  | 		   GCancellable       *cancellable) | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   GLoadableIconIface *iface; | 
					
						
							|  |  |  |   GInputStream *stream; | 
					
						
							|  |  |  |   LoadData *data; | 
					
						
							|  |  |  |   GError *error = NULL; | 
					
						
							|  |  |  |   char *type = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   data = g_simple_async_result_get_op_res_gpointer (res); | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   iface = G_LOADABLE_ICON_GET_IFACE (object); | 
					
						
							|  |  |  |   stream = iface->load (G_LOADABLE_ICON (object), data->size, &type, cancellable, &error); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (stream == NULL) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-09-23 13:02:50 +02:00
										 |  |  |       g_simple_async_result_take_error (res, error); | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       data->stream = stream; | 
					
						
							|  |  |  |       data->type = type; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2007-11-29 07:17:59 +00:00
										 |  |  | g_loadable_icon_real_load_async (GLoadableIcon       *icon, | 
					
						
							|  |  |  | 				 int                  size, | 
					
						
							|  |  |  | 				 GCancellable        *cancellable, | 
					
						
							|  |  |  | 				 GAsyncReadyCallback  callback, | 
					
						
							|  |  |  | 				 gpointer             user_data) | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   GSimpleAsyncResult *res; | 
					
						
							|  |  |  |   LoadData *data; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   res = g_simple_async_result_new (G_OBJECT (icon), callback, user_data, g_loadable_icon_real_load_async); | 
					
						
							|  |  |  |   data = g_new0 (LoadData, 1); | 
					
						
							|  |  |  |   g_simple_async_result_set_op_res_gpointer (res, data, (GDestroyNotify) load_data_free); | 
					
						
							|  |  |  |   g_simple_async_result_run_in_thread (res, load_async_thread, 0, cancellable); | 
					
						
							|  |  |  |   g_object_unref (res); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static GInputStream * | 
					
						
							|  |  |  | g_loadable_icon_real_load_finish (GLoadableIcon        *icon, | 
					
						
							|  |  |  | 				  GAsyncResult         *res, | 
					
						
							|  |  |  | 				  char                **type, | 
					
						
							|  |  |  | 				  GError              **error) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GSimpleAsyncResult *simple = G_SIMPLE_ASYNC_RESULT (res); | 
					
						
							|  |  |  |   LoadData *data; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-10 14:07:42 +00:00
										 |  |  |   g_warn_if_fail (g_simple_async_result_get_source_tag (simple) == g_loadable_icon_real_load_async); | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-11 13:44:19 -04:00
										 |  |  |   if (g_simple_async_result_propagate_error (simple, error)) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-26 16:13:05 +00:00
										 |  |  |   data = g_simple_async_result_get_op_res_gpointer (simple); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       *type = data->type; | 
					
						
							|  |  |  |       data->type = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return g_object_ref (data->stream); | 
					
						
							|  |  |  | } |