/* vi:set et ai sw=2 sts=2 ts=2: */ /*- * Copyright (c) 2009-2011 Jannis Pohlmann * Copyright (c) 2012 Nick Schermer * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public * License along with this program; if not, write to the Free * Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include /** * WARNING: The source code in this file may do harm to animals. Dead kittens * are to be expected. * * * ThunarThumbnailer is a class for requesting thumbnails from org.xfce.tumbler.* * D-Bus services. This header contains an in-depth description of its design to make * it easier to understand why things are the way they are. * * Please note that all D-Bus calls are performed asynchronously. * * The proxy and the supported file types are intialized asynchronously, but * this is nearly completely transparent to the calling code because all requests * that arrive beforehand will be cached and sent out as soon as the initialization * is completed. * * When a request call is sent out, an internal request ID is created and tied * to the dbus call. While the call is running, an additional reference is held * to the thumbnailer so it cannot be finalized. * * The D-Bus reply handler then checks if there was an delivery error or * not. If the request method was sent successfully, the handle returned by the * D-Bus thumbnailer is associated bidirectionally with the internal request ID via * the request and handle values in the job structure. * * * Ready / Error * ============= * * The Ready and Error signal handlers work exactly like Started except that * the Ready idle function sets the thumb state of the corresponding * ThunarFile objects to _READY and the Error signal sets the state to _NONE. * * * Finished * ======== * * The Finished signal handler looks up the internal request ID based on * the D-Bus thumbnailer handle. It then drops all corresponding information * from handle_request_mapping and request_handle_mapping. */ typedef enum { THUNAR_THUMBNAILER_IDLE_ERROR, THUNAR_THUMBNAILER_IDLE_READY, } ThunarThumbnailerIdleType; typedef enum { THUNAR_THUMBNAILER_PROXY_WAITING = 0, THUNAR_THUMBNAILER_PROXY_AVAILABLE, THUNAR_THUMBNAILER_PROXY_FAILED } ThunarThumbnailerProxyState; typedef struct _ThunarThumbnailerJob ThunarThumbnailerJob; typedef struct _ThunarThumbnailerIdle ThunarThumbnailerIdle; /* Signal identifiers */ enum { REQUEST_FINISHED, LAST_SIGNAL, }; /* Property identifiers */ enum { PROP_0, PROP_THUMBNAIL_SIZE, }; static void thunar_thumbnailer_finalize (GObject *object); static void thunar_thumbnailer_init_thumbnailer_proxy (ThunarThumbnailer *thumbnailer); static gboolean thunar_thumbnailer_file_is_supported (ThunarThumbnailer *thumbnailer, ThunarFile *file); static void thunar_thumbnailer_thumbnailer_finished (GDBusProxy *proxy, guint handle, ThunarThumbnailer *thumbnailer); static void thunar_thumbnailer_thumbnailer_error (GDBusProxy *proxy, guint handle, const gchar **uris, gint code, const gchar *message, ThunarThumbnailer *thumbnailer); static void thunar_thumbnailer_thumbnailer_ready (GDBusProxy *proxy, guint32 handle, const gchar **uris, ThunarThumbnailer *thumbnailer); static void thunar_thumbnailer_idle (ThunarThumbnailer *thumbnailer, guint handle, ThunarThumbnailerIdleType type, const gchar **uris); static gboolean thunar_thumbnailer_idle_func (gpointer user_data); static void thunar_thumbnailer_idle_free (gpointer data); static void thunar_thumbnailer_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static void thunar_thumbnailer_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); #define _thumbnailer_lock(thumbnailer) g_mutex_lock (&((thumbnailer)->lock)) #define _thumbnailer_unlock(thumbnailer) g_mutex_unlock (&((thumbnailer)->lock)) #define _thumbnailer_trylock(thumbnailer) g_mutex_trylock (&((thumbnailer)->lock)) struct _ThunarThumbnailerClass { GObjectClass __parent__; }; struct _ThunarThumbnailer { GObject __parent__; /* proxies to communicate with D-Bus services */ ThunarThumbnailerDBus *thumbnailer_proxy; ThunarThumbnailerProxyState proxy_state; /* running jobs */ GSList *jobs; GMutex lock; /* cached MIME types -> URI schemes for which thumbs can be generated */ GHashTable *supported; /* last ThunarThumbnailer request ID */ guint last_request; /* size to use to store thumbnails */ ThunarThumbnailSize thumbnail_size; /* IDs of idle functions */ GSList *idles; }; struct _ThunarThumbnailerJob { ThunarThumbnailer *thumbnailer; /* if this job is cancelled */ guint cancelled : 1; guint lazy_checks : 1; /* data is saved here in case the queueing is delayed */ /* If this is NULL, the request has been sent off. */ GList *files; /* element type: ThunarFile */ /* request number returned by ThunarThumbnailer */ guint request; /* handle returned by the tumbler dbus service */ guint handle; }; struct _ThunarThumbnailerIdle { ThunarThumbnailerIdleType type; ThunarThumbnailer *thumbnailer; guint id; gchar **uris; }; static guint thumbnailer_signals[LAST_SIGNAL]; G_DEFINE_TYPE (ThunarThumbnailer, thunar_thumbnailer, G_TYPE_OBJECT); static void thunar_thumbnailer_class_init (ThunarThumbnailerClass *klass) { GObjectClass *gobject_class; gobject_class = G_OBJECT_CLASS (klass); gobject_class->finalize = thunar_thumbnailer_finalize; gobject_class->get_property = thunar_thumbnailer_get_property; gobject_class->set_property = thunar_thumbnailer_set_property; /** * ThunarThumbnailer:request-finished: * @thumbnailer : a #ThunarThumbnailer. * @request : id of the request that is finished. * * Emitted by @thumbnailer, when a request is finished * by the thumbnail generator **/ thumbnailer_signals[REQUEST_FINISHED] = g_signal_new (I_("request-finished"), G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, 0, NULL, NULL, g_cclosure_marshal_VOID__UINT, G_TYPE_NONE, 1, G_TYPE_UINT); /** * ThunarIconFactory:thumbnail-size: * * Size of the thumbnails to load **/ g_object_class_install_property (gobject_class, PROP_THUMBNAIL_SIZE, g_param_spec_enum ("thumbnail-size", "thumbnail-size", "thumbnail-size", THUNAR_TYPE_THUMBNAIL_SIZE, THUNAR_THUMBNAIL_SIZE_NORMAL, EXO_PARAM_READWRITE)); } static void thunar_thumbnailer_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { ThunarThumbnailer *thumbnailer = THUNAR_THUMBNAILER (object); switch (prop_id) { case PROP_THUMBNAIL_SIZE: g_value_set_enum (value, thumbnailer->thumbnail_size); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void thunar_thumbnailer_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { ThunarThumbnailer *thumbnailer = THUNAR_THUMBNAILER (object); switch (prop_id) { case PROP_THUMBNAIL_SIZE: thumbnailer->thumbnail_size = g_value_get_enum (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void thunar_thumbnailer_free_job (ThunarThumbnailerJob *job) { if (job->files) g_list_free_full (job->files, g_object_unref); if (job->thumbnailer && job->thumbnailer->thumbnailer_proxy && job->handle) thunar_thumbnailer_dbus_call_dequeue (job->thumbnailer->thumbnailer_proxy, job->handle, NULL, NULL, NULL); g_slice_free (ThunarThumbnailerJob, job); } static void thunar_thumbnailer_queue_async_reply (GObject *proxy, GAsyncResult *res, gpointer user_data) { ThunarThumbnailerJob *job = user_data; ThunarThumbnailer *thumbnailer; GError *error = NULL; guint handle; _thunar_return_if_fail (THUNAR_IS_THUMBNAILER_DBUS (proxy)); _thunar_return_if_fail (job != NULL); thumbnailer = THUNAR_THUMBNAILER (job->thumbnailer); _thunar_return_if_fail (THUNAR_IS_THUMBNAILER (thumbnailer)); _thumbnailer_lock (thumbnailer); thunar_thumbnailer_dbus_call_queue_finish (THUNAR_THUMBNAILER_DBUS (proxy), &handle, res, &error); if (job->cancelled) { /* job is cancelled while there was no handle jet, so dequeue it now */ thunar_thumbnailer_dbus_call_dequeue (THUNAR_THUMBNAILER_DBUS (proxy), handle, NULL, NULL, NULL); /* cleanup */ thumbnailer->jobs = g_slist_remove (thumbnailer->jobs, job); thunar_thumbnailer_free_job (job); } else if (error == NULL) { if (handle == 0) { g_printerr ("ThunarThumbnailer: got 0 handle (Queue)\n"); } else { /* store the handle returned by tumbler */ job->handle = handle; } } else { g_printerr ("ThunarThumbnailer: Queue failed: %s\n", error->message); } _thumbnailer_unlock (thumbnailer); g_clear_error (&error); /* remove the additional reference we held during the call */ g_object_unref (thumbnailer); } /* NOTE: assumes that the lock is held by the caller */ static gboolean thunar_thumbnailer_begin_job (ThunarThumbnailer *thumbnailer, ThunarThumbnailerJob *job) { gboolean success = FALSE; const gchar **mime_hints; gchar **uris; GList *lp; GList *supported_files = NULL; guint n; guint n_items = 0; ThunarFileThumbState thumb_state; const gchar *thumbnail_path; gint request_no; if (thumbnailer->proxy_state == THUNAR_THUMBNAILER_PROXY_WAITING) { /* all pending jobs will be queued automatically once the proxy is available */ return TRUE; } else if (thumbnailer->proxy_state == THUNAR_THUMBNAILER_PROXY_FAILED) { /* the job has no chance to be completed - ever */ return FALSE; } /* collect all supported files from the list that are neither in the * about to be queued (wait queue), nor already queued, nor already * processed (and awaiting to be refreshed) */ for (lp = job->files; lp != NULL; lp = lp->next) { /* the icon factory only loads icons for regular files and folders */ if (!thunar_file_is_regular (lp->data) && !thunar_file_is_directory (lp->data)) { thunar_file_set_thumb_state (lp->data, THUNAR_FILE_THUMB_STATE_NONE); continue; } /* get the current thumb state */ thumb_state = thunar_file_get_thumb_state (lp->data); if (job->lazy_checks) { /* in lazy mode, don't both for files that have already * been loaded or are not supported */ if (thumb_state == THUNAR_FILE_THUMB_STATE_NONE || thumb_state == THUNAR_FILE_THUMB_STATE_READY) continue; } /* check if the file is supported, assume it is when the state was ready previously */ if (thumb_state == THUNAR_FILE_THUMB_STATE_READY || thunar_thumbnailer_file_is_supported (thumbnailer, lp->data)) { supported_files = g_list_prepend (supported_files, lp->data); n_items++; } else { /* still a regular file, but the type is now known to tumbler but * maybe the application created a thumbnail */ thumbnail_path = thunar_file_get_thumbnail_path (lp->data, thumbnailer->thumbnail_size); /* test if a thumbnail can be found */ if (thumbnail_path != NULL && g_file_test (thumbnail_path, G_FILE_TEST_EXISTS)) thunar_file_set_thumb_state (lp->data, THUNAR_FILE_THUMB_STATE_READY); else thunar_file_set_thumb_state (lp->data, THUNAR_FILE_THUMB_STATE_NONE); } } /* check if we have any supported files */ if (n_items > 0) { /* allocate arrays for URIs and mime hints */ uris = g_new0 (gchar *, n_items + 1); mime_hints = g_new0 (const gchar *, n_items + 1); /* fill URI and MIME hint arrays with items from the wait queue */ for (lp = supported_files, n = 0; lp != NULL; lp = lp->next, ++n) { /* set the thumbnail state to loading */ thunar_file_set_thumb_state (lp->data, THUNAR_FILE_THUMB_STATE_LOADING); /* save URI and MIME hint in the arrays */ uris[n] = thunar_file_dup_uri (lp->data); mime_hints[n] = thunar_file_get_content_type (lp->data); } /* NULL-terminate both arrays */ uris[n] = NULL; mime_hints[n] = NULL; /* queue a thumbnail request for the URIs from the wait queue */ /* compute the next request ID, making sure it's never 0 */ request_no = thumbnailer->last_request + 1; request_no = MAX (request_no, 1); /* remember the ID for the next request */ thumbnailer->last_request = request_no; /* save the request number */ job->request = request_no; /* increase the reference count while the dbus call is running */ g_object_ref (thumbnailer); /* queue the request - asynchronously, of course */ thunar_thumbnailer_dbus_call_queue (thumbnailer->thumbnailer_proxy, (const gchar *const *)uris, (const gchar *const *)mime_hints, thunar_thumbnail_size_get_nick (thumbnailer->thumbnail_size), "foreground", 0, NULL, thunar_thumbnailer_queue_async_reply, job); /* free mime hints array */ g_free (mime_hints); g_strfreev (uris); /* free the list of supported files */ g_list_free (supported_files); /* free the list of files passed in */ g_list_free_full (job->files, g_object_unref); job->files = NULL; /* we assume success if we've come so far */ success = TRUE; } return success; } static void thunar_thumbnailer_init (ThunarThumbnailer *thumbnailer) { g_mutex_init (&thumbnailer->lock); /* initialize the proxies */ thunar_thumbnailer_init_thumbnailer_proxy (thumbnailer); } static void thunar_thumbnailer_finalize (GObject *object) { ThunarThumbnailer *thumbnailer = THUNAR_THUMBNAILER (object); ThunarThumbnailerIdle *idle; GSList *lp; /* acquire the thumbnailer lock */ _thumbnailer_lock (thumbnailer); if (thumbnailer->thumbnailer_proxy != NULL) { /* disconnect from the thumbnailer proxy */ g_signal_handlers_disconnect_matched (thumbnailer->thumbnailer_proxy, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, thumbnailer); } /* abort all pending idle functions */ for (lp = thumbnailer->idles; lp != NULL; lp = lp->next) { idle = lp->data; g_source_remove (idle->id); } g_slist_free (thumbnailer->idles); /* remove all jobs */ g_slist_free_full (thumbnailer->jobs, (GDestroyNotify)thunar_thumbnailer_free_job); /* release the thumbnailer proxy */ if (thumbnailer->thumbnailer_proxy != NULL) g_object_unref (thumbnailer->thumbnailer_proxy); /* free the cached URI schemes and MIME types table */ if (thumbnailer->supported != NULL) g_hash_table_unref (thumbnailer->supported); /* release the thumbnailer lock */ _thumbnailer_unlock (thumbnailer); /* release the mutex */ g_mutex_clear (&thumbnailer->lock); (*G_OBJECT_CLASS (thunar_thumbnailer_parent_class)->finalize) (object); } static gint thunar_thumbnailer_file_schemes_compare (gconstpointer a, gconstpointer b) { const gchar *scheme_a = *(gconstpointer *) a; const gchar *scheme_b = *(gconstpointer *) b; /* sort file before other schemes */ if (strcmp (scheme_a, "file") == 0) return -1; if (strcmp (scheme_b, "file") == 0) return 1; /* sort trash before other schemes */ if (strcmp (scheme_a, "trash") == 0) return -1; if (strcmp (scheme_b, "trash") == 0) return 1; /* other order is just fine */ return 0; } static void thunar_thumbnailer_file_sort_schemes (gpointer mime_type, gpointer schemes, gpointer user_data) { g_ptr_array_sort (schemes, thunar_thumbnailer_file_schemes_compare); } static void thunar_thumbnailer_received_supported_types (ThunarThumbnailerDBus *proxy, GAsyncResult *result, ThunarThumbnailer *thumbnailer) { guint n; gchar **schemes = NULL; gchar **types = NULL; GPtrArray *schemes_array; GSList *lp = NULL; GError *error = NULL; _thunar_return_if_fail (THUNAR_IS_THUMBNAILER (thumbnailer)); _thunar_return_if_fail (THUNAR_IS_THUMBNAILER_DBUS (proxy)); _thumbnailer_lock (thumbnailer); if (!thunar_thumbnailer_dbus_call_get_supported_finish (proxy, &schemes, &types, result, &error)) { g_slist_free_full (thumbnailer->jobs, (GDestroyNotify)thunar_thumbnailer_free_job); thumbnailer->jobs = NULL; g_printerr ("ThunarThumbnailer: Failed to retrieve supported types: %s\n", error->message); g_clear_error (&error); thumbnailer->proxy_state = THUNAR_THUMBNAILER_PROXY_FAILED; g_object_unref (proxy); _thumbnailer_unlock (thumbnailer); g_object_unref (thumbnailer); return; } if (G_LIKELY (schemes != NULL && types != NULL)) { /* combine content types and uri schemes */ for (n = 0; types[n] != NULL; ++n) { schemes_array = g_hash_table_lookup (thumbnailer->supported, types[n]); if (G_UNLIKELY (schemes_array == NULL)) { /* create an array for the uri schemes this content type supports */ schemes_array = g_ptr_array_new_with_free_func (g_free); g_ptr_array_add (schemes_array, schemes[n]); g_hash_table_insert (thumbnailer->supported, types[n], schemes_array); } else { /* add the uri scheme to the array of the content type */ g_ptr_array_add (schemes_array, schemes[n]); /* cleanup */ g_free (types[n]); } } /* remove arrays, we stole the values */ g_free (types); g_free (schemes); /* sort array to optimize for local files */ g_hash_table_foreach (thumbnailer->supported, thunar_thumbnailer_file_sort_schemes, NULL); } thumbnailer->proxy_state = THUNAR_THUMBNAILER_PROXY_AVAILABLE; thumbnailer->thumbnailer_proxy = proxy; /* now start delayed jobs */ for (lp = thumbnailer->jobs; lp; lp = lp->next) { if (!thunar_thumbnailer_begin_job (thumbnailer, lp->data)) { thunar_thumbnailer_free_job (lp->data); lp->data = NULL; } } thumbnailer->jobs = g_slist_remove_all (thumbnailer->jobs, NULL); g_clear_error (&error); _thumbnailer_unlock (thumbnailer); g_object_unref (thumbnailer); } static void thunar_thumbnailer_proxy_created (GObject *object, GAsyncResult *result, gpointer userdata) { ThunarThumbnailer *thumbnailer = THUNAR_THUMBNAILER (userdata); GError *error = NULL; ThunarThumbnailerDBus *proxy; proxy = thunar_thumbnailer_dbus_proxy_new_finish (result, &error); if (!proxy) { _thumbnailer_lock (thumbnailer); thumbnailer->proxy_state = THUNAR_THUMBNAILER_PROXY_FAILED; g_printerr ("ThunarThumbnailer: failed to create proxy: %s", error->message); g_clear_error (&error); g_slist_free_full (thumbnailer->jobs, (GDestroyNotify)thunar_thumbnailer_free_job); thumbnailer->jobs = NULL; _thumbnailer_unlock (thumbnailer); g_object_unref (thumbnailer); return; } /* setup signals */ g_signal_connect (proxy, "error", G_CALLBACK (thunar_thumbnailer_thumbnailer_error), thumbnailer); g_signal_connect (proxy, "finished", G_CALLBACK (thunar_thumbnailer_thumbnailer_finished), thumbnailer); g_signal_connect (proxy, "ready", G_CALLBACK (thunar_thumbnailer_thumbnailer_ready), thumbnailer); /* begin retrieving supported file types */ _thumbnailer_lock (thumbnailer); /* prepare table */ g_clear_pointer (&thumbnailer->supported, g_hash_table_unref); thumbnailer->supported = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, (GDestroyNotify) g_ptr_array_unref); /* request the supported types from the thumbnailer D-Bus service. */ thunar_thumbnailer_dbus_call_get_supported (proxy, NULL, (GAsyncReadyCallback)thunar_thumbnailer_received_supported_types, thumbnailer); _thumbnailer_unlock (thumbnailer); } static void thunar_thumbnailer_init_thumbnailer_proxy (ThunarThumbnailer *thumbnailer) { _thumbnailer_lock (thumbnailer); thumbnailer->thumbnailer_proxy = NULL; thumbnailer->proxy_state = THUNAR_THUMBNAILER_PROXY_WAITING; /* create the thumbnailer proxy */ thunar_thumbnailer_dbus_proxy_new_for_bus (G_BUS_TYPE_SESSION, G_DBUS_PROXY_FLAGS_GET_INVALIDATED_PROPERTIES, "org.freedesktop.thumbnails.Thumbnailer1", "/org/freedesktop/thumbnails/Thumbnailer1", NULL, (GAsyncReadyCallback)thunar_thumbnailer_proxy_created, thumbnailer); _thumbnailer_unlock (thumbnailer); } /* NOTE: assumes the lock is being held by the caller*/ static gboolean thunar_thumbnailer_file_is_supported (ThunarThumbnailer *thumbnailer, ThunarFile *file) { const gchar *content_type; gboolean supported = FALSE; guint n; GPtrArray *schemes_array; const gchar *scheme; _thunar_return_val_if_fail (THUNAR_IS_THUMBNAILER (thumbnailer), FALSE); _thunar_return_val_if_fail (THUNAR_IS_FILE (file), FALSE); _thunar_return_val_if_fail (THUNAR_IS_THUMBNAILER_DBUS (thumbnailer->thumbnailer_proxy), FALSE); _thunar_return_val_if_fail (thumbnailer->supported != NULL, FALSE); /* determine the content type of the passed file */ content_type = thunar_file_get_content_type (file); /* abort if the content type is unknown */ if (content_type == NULL) return FALSE; /* lazy lookup the content type, no difficult parent type matching here */ schemes_array = g_hash_table_lookup (thumbnailer->supported, content_type); if (schemes_array != NULL) { /* go through all the URI schemes this type supports */ for (n = 0; !supported && n < schemes_array->len; ++n) { /* check if the file has the current URI scheme */ scheme = g_ptr_array_index (schemes_array, n); if (thunar_file_has_uri_scheme (file, scheme)) supported = TRUE; } } return supported; } static void thunar_thumbnailer_thumbnailer_error (GDBusProxy *proxy, guint handle, const gchar **uris, gint code, const gchar *message, ThunarThumbnailer *thumbnailer) { _thunar_return_if_fail (G_IS_DBUS_PROXY (proxy)); _thunar_return_if_fail (THUNAR_IS_THUMBNAILER (thumbnailer)); /* check if we have any ready URIs */ thunar_thumbnailer_idle (thumbnailer, handle, THUNAR_THUMBNAILER_IDLE_ERROR, uris); } static void thunar_thumbnailer_thumbnailer_ready (GDBusProxy *proxy, guint32 handle, const gchar **uris, ThunarThumbnailer *thumbnailer) { _thunar_return_if_fail (G_IS_DBUS_PROXY (proxy)); _thunar_return_if_fail (THUNAR_IS_THUMBNAILER (thumbnailer)); thunar_thumbnailer_idle (thumbnailer, handle, THUNAR_THUMBNAILER_IDLE_READY, uris); } static void thunar_thumbnailer_thumbnailer_finished (GDBusProxy *proxy, guint handle, ThunarThumbnailer *thumbnailer) { ThunarThumbnailerJob *job; GSList *lp; _thunar_return_if_fail (G_IS_DBUS_PROXY (proxy)); _thunar_return_if_fail (THUNAR_IS_THUMBNAILER (thumbnailer)); if (handle == 0) { g_printerr ("ThunarThumbnailer: got 0 handle (Finished)\n"); return; } _thumbnailer_lock (thumbnailer); for (lp = thumbnailer->jobs; lp != NULL; lp = lp->next) { job = lp->data; if (job->handle == handle) { /* this job is finished, forget about the handle */ job->handle = 0; /* tell everybody we're done here */ g_signal_emit (G_OBJECT (thumbnailer), thumbnailer_signals[REQUEST_FINISHED], 0, job->request); /* remove job from the list */ thumbnailer->jobs = g_slist_delete_link (thumbnailer->jobs, lp); thunar_thumbnailer_free_job (job); break; } } _thumbnailer_unlock (thumbnailer); } static void thunar_thumbnailer_idle (ThunarThumbnailer *thumbnailer, guint handle, ThunarThumbnailerIdleType type, const gchar **uris) { GSList *lp; ThunarThumbnailerIdle *idle; ThunarThumbnailerJob *job; /* leave if there are no uris */ if (G_UNLIKELY (uris == NULL)) return; if (handle == 0) { g_printerr ("ThunarThumbnailer: got 0 handle (Error or Ready)\n"); return; } _thumbnailer_lock (thumbnailer); /* look for the job so we don't emit unknown handles, the reason * we do this is when you have multiple windows opened, you don't * want each window (because they all have a connection to the * same proxy) emit the file change, only the window that requested * the data */ for (lp = thumbnailer->jobs; lp != NULL; lp = lp->next) { job = lp->data; if (job->handle == handle) { /* allocate a new idle struct */ idle = g_slice_new0 (ThunarThumbnailerIdle); idle->type = type; idle->thumbnailer = thumbnailer; /* copy the URI array because we need it in the idle function */ idle->uris = g_strdupv ((gchar **)uris); /* remember the idle struct because we might have to remove it in finalize() */ thumbnailer->idles = g_slist_prepend (thumbnailer->idles, idle); /* call the idle function when we have the time */ idle->id = g_idle_add_full (G_PRIORITY_LOW, thunar_thumbnailer_idle_func, idle, thunar_thumbnailer_idle_free); break; } } _thumbnailer_unlock (thumbnailer); } static gboolean thunar_thumbnailer_idle_func (gpointer user_data) { ThunarThumbnailerIdle *idle = user_data; ThunarFile *file; GFile *gfile; guint n; _thunar_return_val_if_fail (idle != NULL, FALSE); _thunar_return_val_if_fail (THUNAR_IS_THUMBNAILER (idle->thumbnailer), FALSE); /* iterate over all failed URIs */ for (n = 0; idle->uris != NULL && idle->uris[n] != NULL; ++n) { /* look up the corresponding ThunarFile from the cache */ gfile = g_file_new_for_uri (idle->uris[n]); file = thunar_file_cache_lookup (gfile); g_object_unref (gfile); /* check if we have a file for this URI in the cache */ if (file != NULL) { if (idle->type == THUNAR_THUMBNAILER_IDLE_ERROR) { /* set thumbnail state to none unless the thumbnail has already been created. * This is to prevent race conditions with the other idle functions */ if (thunar_file_get_thumb_state (file) != THUNAR_FILE_THUMB_STATE_READY) thunar_file_set_thumb_state (file, THUNAR_FILE_THUMB_STATE_NONE); } else if (idle->type == THUNAR_THUMBNAILER_IDLE_READY) { /* set thumbnail state to ready - we now have a thumbnail */ thunar_file_set_thumb_state (file, THUNAR_FILE_THUMB_STATE_READY); } else { _thunar_assert_not_reached (); } g_object_unref (file); } } /* remove the idle struct */ _thumbnailer_lock (idle->thumbnailer); idle->thumbnailer->idles = g_slist_remove (idle->thumbnailer->idles, idle); _thumbnailer_unlock (idle->thumbnailer); /* remove the idle source, which also destroys the idle struct */ return FALSE; } static void thunar_thumbnailer_idle_free (gpointer data) { ThunarThumbnailerIdle *idle = data; _thunar_return_if_fail (idle != NULL); /* free the URI array if necessary */ if (idle->type == THUNAR_THUMBNAILER_IDLE_READY || idle->type == THUNAR_THUMBNAILER_IDLE_ERROR) { g_strfreev (idle->uris); } /* free the struct */ g_slice_free (ThunarThumbnailerIdle, idle); } /** * thunar_thumbnailer_get: * * Return a shared new #ThunarThumbnailer object, which can be used to * generate and store thumbnails for files. * * The caller is responsible to free the returned * object using g_object_unref() when no longer needed. * * Return value: a #ThunarThumbnailer. **/ ThunarThumbnailer* thunar_thumbnailer_get (void) { static ThunarThumbnailer *thumbnailer = NULL; if (G_UNLIKELY (thumbnailer == NULL)) { thumbnailer = g_object_new (THUNAR_TYPE_THUMBNAILER, NULL); g_object_add_weak_pointer (G_OBJECT (thumbnailer), (gpointer) &thumbnailer); } return g_object_ref_sink (thumbnailer); } gboolean thunar_thumbnailer_queue_file (ThunarThumbnailer *thumbnailer, ThunarFile *file, guint *request) { GList files; _thunar_return_val_if_fail (THUNAR_IS_THUMBNAILER (thumbnailer), FALSE); _thunar_return_val_if_fail (THUNAR_IS_FILE (file), FALSE); /* fake a file list */ files.data = file; files.next = NULL; files.prev = NULL; /* queue a thumbnail request for the file */ return thunar_thumbnailer_queue_files (thumbnailer, FALSE, &files, request); } gboolean thunar_thumbnailer_queue_files (ThunarThumbnailer *thumbnailer, gboolean lazy_checks, GList *files, guint *request) { gboolean success = FALSE; ThunarThumbnailerJob *job = NULL; _thunar_return_val_if_fail (THUNAR_IS_THUMBNAILER (thumbnailer), FALSE); _thunar_return_val_if_fail (files != NULL, FALSE); /* acquire the thumbnailer lock */ _thumbnailer_lock (thumbnailer); /* allocate a job */ job = g_slice_new0 (ThunarThumbnailerJob); job->thumbnailer = thumbnailer; job->files = g_list_copy_deep (files, (GCopyFunc) (void (*)(void)) g_object_ref, NULL); job->lazy_checks = lazy_checks ? 1 : 0; success = thunar_thumbnailer_begin_job (thumbnailer, job); if (success) { thumbnailer->jobs = g_slist_prepend (thumbnailer->jobs, job); if (*request) *request = job->request; } else { thunar_thumbnailer_free_job (job); } /* release the lock */ _thumbnailer_unlock (thumbnailer); return success; } void thunar_thumbnailer_dequeue (ThunarThumbnailer *thumbnailer, guint request) { ThunarThumbnailerJob *job; GSList *lp; _thunar_return_if_fail (THUNAR_IS_THUMBNAILER (thumbnailer)); /* acquire the thumbnailer lock */ _thumbnailer_lock (thumbnailer); for (lp = thumbnailer->jobs; lp != NULL; lp = lp->next) { job = lp->data; /* find the request in the list */ if (job->request == request) { /* this job is cancelled */ job->cancelled = TRUE; if (job->handle != 0) { /* remove job */ thumbnailer->jobs = g_slist_delete_link (thumbnailer->jobs, lp); thunar_thumbnailer_free_job (job); } break; } } /* release the thumbnailer lock */ _thumbnailer_unlock (thumbnailer); }