534 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			534 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* gtkrecentchooserutils.h - Private utility functions for implementing a
 | |
|  *                           GtkRecentChooser interface
 | |
|  *
 | |
|  * Copyright (C) 2006 Emmanuele Bassi
 | |
|  *
 | |
|  * All rights reserved
 | |
|  *
 | |
|  * This library is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU Library 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
 | |
|  * Library General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Library 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.
 | |
|  *
 | |
|  * Based on gtkfilechooserutils.c:
 | |
|  *	Copyright (C) 2003 Red Hat, Inc.
 | |
|  */
 | |
| 
 | |
| #include "config.h"
 | |
| 
 | |
| #include "gtkrecentchooserutils.h"
 | |
| 
 | |
| /* Methods */
 | |
| static void      delegate_set_sort_func              (GtkRecentChooser  *chooser,
 | |
| 						      GtkRecentSortFunc  sort_func,
 | |
| 						      gpointer           sort_data,
 | |
| 						      GDestroyNotify     data_destroy);
 | |
| static void      delegate_add_filter                 (GtkRecentChooser  *chooser,
 | |
| 						      GtkRecentFilter   *filter);
 | |
| static void      delegate_remove_filter              (GtkRecentChooser  *chooser,
 | |
| 						      GtkRecentFilter   *filter);
 | |
| static GSList   *delegate_list_filters               (GtkRecentChooser  *chooser);
 | |
| static gboolean  delegate_select_uri                 (GtkRecentChooser  *chooser,
 | |
| 						      const gchar       *uri,
 | |
| 						      GError           **error);
 | |
| static void      delegate_unselect_uri               (GtkRecentChooser  *chooser,
 | |
| 						      const gchar       *uri);
 | |
| static GList    *delegate_get_items                  (GtkRecentChooser  *chooser);
 | |
| static GtkRecentManager *delegate_get_recent_manager (GtkRecentChooser  *chooser);
 | |
| static void      delegate_select_all                 (GtkRecentChooser  *chooser);
 | |
| static void      delegate_unselect_all               (GtkRecentChooser  *chooser);
 | |
| static gboolean  delegate_set_current_uri            (GtkRecentChooser  *chooser,
 | |
| 						      const gchar       *uri,
 | |
| 						      GError           **error);
 | |
| static gchar *   delegate_get_current_uri            (GtkRecentChooser  *chooser);
 | |
| 
 | |
| /* Signals */
 | |
| static void      delegate_notify            (GObject          *object,
 | |
| 					     GParamSpec       *pspec,
 | |
| 					     gpointer          user_data);
 | |
| static void      delegate_selection_changed (GtkRecentChooser *receiver,
 | |
| 					     gpointer          user_data);
 | |
| static void      delegate_item_activated    (GtkRecentChooser *receiver,
 | |
| 					     gpointer          user_data);
 | |
| 
 | |
| /**
 | |
|  * _gtk_recent_chooser_install_properties:
 | |
|  * @klass: the class structure for a type deriving from #GObject
 | |
|  *
 | |
|  * Installs the necessary properties for a class implementing
 | |
|  * #GtkRecentChooser. A #GtkParamSpecOverride property is installed
 | |
|  * for each property, using the values from the #GtkRecentChooserProp
 | |
|  * enumeration. The caller must make sure itself that the enumeration
 | |
|  * values don't collide with some other property values they
 | |
|  * are using.
 | |
|  */
 | |
| void
 | |
| _gtk_recent_chooser_install_properties (GObjectClass *klass)
 | |
| {
 | |
|   g_object_class_override_property (klass,
 | |
|   				    GTK_RECENT_CHOOSER_PROP_RECENT_MANAGER,
 | |
|   				    "recent-manager");  				    
 | |
|   g_object_class_override_property (klass,
 | |
|   				    GTK_RECENT_CHOOSER_PROP_SHOW_PRIVATE,
 | |
|   				    "show-private");
 | |
|   g_object_class_override_property (klass,
 | |
|   				    GTK_RECENT_CHOOSER_PROP_SHOW_TIPS,
 | |
|   				    "show-tips");
 | |
|   g_object_class_override_property (klass,
 | |
|   				    GTK_RECENT_CHOOSER_PROP_SHOW_ICONS,
 | |
|   				    "show-icons");
 | |
|   g_object_class_override_property (klass,
 | |
|   				    GTK_RECENT_CHOOSER_PROP_SHOW_NOT_FOUND,
 | |
|   				    "show-not-found");
 | |
|   g_object_class_override_property (klass,
 | |
|   				    GTK_RECENT_CHOOSER_PROP_SELECT_MULTIPLE,
 | |
|   				    "select-multiple");
 | |
|   g_object_class_override_property (klass,
 | |
|   				    GTK_RECENT_CHOOSER_PROP_LIMIT,
 | |
|   				    "limit");
 | |
|   g_object_class_override_property (klass,
 | |
| 		  		    GTK_RECENT_CHOOSER_PROP_LOCAL_ONLY,
 | |
| 				    "local-only");
 | |
|   g_object_class_override_property (klass,
 | |
| 		  		    GTK_RECENT_CHOOSER_PROP_SORT_TYPE,
 | |
| 				    "sort-type");
 | |
|   g_object_class_override_property (klass,
 | |
|   				    GTK_RECENT_CHOOSER_PROP_FILTER,
 | |
|   				    "filter");
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * _gtk_recent_chooser_delegate_iface_init:
 | |
|  * @iface: a #GtkRecentChooserIface
 | |
|  *
 | |
|  * An interface-initialization function for use in cases where
 | |
|  * an object is simply delegating the methods, signals of
 | |
|  * the #GtkRecentChooser interface to another object.
 | |
|  * _gtk_recent_chooser_set_delegate() must be called on each
 | |
|  * instance of the object so that the delegate object can
 | |
|  * be found.
 | |
|  */
 | |
| void
 | |
| _gtk_recent_chooser_delegate_iface_init (GtkRecentChooserIface *iface)
 | |
| {
 | |
|   iface->set_current_uri = delegate_set_current_uri;
 | |
|   iface->get_current_uri = delegate_get_current_uri;
 | |
|   iface->select_uri = delegate_select_uri;
 | |
|   iface->unselect_uri = delegate_unselect_uri;
 | |
|   iface->select_all = delegate_select_all;
 | |
|   iface->unselect_all = delegate_unselect_all;
 | |
|   iface->get_items = delegate_get_items;
 | |
|   iface->get_recent_manager = delegate_get_recent_manager;
 | |
|   iface->set_sort_func = delegate_set_sort_func;
 | |
|   iface->add_filter = delegate_add_filter;
 | |
|   iface->remove_filter = delegate_remove_filter;
 | |
|   iface->list_filters = delegate_list_filters;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * _gtk_recent_chooser_set_delegate:
 | |
|  * @receiver: a #GObject implementing #GtkRecentChooser
 | |
|  * @delegate: another #GObject implementing #GtkRecentChooser
 | |
|  *
 | |
|  * Establishes that calls on @receiver for #GtkRecentChooser
 | |
|  * methods should be delegated to @delegate, and that
 | |
|  * #GtkRecentChooser signals emitted on @delegate should be
 | |
|  * forwarded to @receiver. Must be used in conjunction with
 | |
|  * _gtk_recent_chooser_delegate_iface_init().
 | |
|  */
 | |
| void
 | |
| _gtk_recent_chooser_set_delegate (GtkRecentChooser *receiver,
 | |
| 				  GtkRecentChooser *delegate)
 | |
| {
 | |
|   g_return_if_fail (GTK_IS_RECENT_CHOOSER (receiver));
 | |
|   g_return_if_fail (GTK_IS_RECENT_CHOOSER (delegate));
 | |
|   
 | |
|   g_object_set_data (G_OBJECT (receiver),
 | |
|   		    "gtk-recent-chooser-delegate", delegate);
 | |
|   
 | |
|   g_signal_connect (delegate, "notify",
 | |
|   		    G_CALLBACK (delegate_notify), receiver);
 | |
|   g_signal_connect (delegate, "selection-changed",
 | |
|   		    G_CALLBACK (delegate_selection_changed), receiver);
 | |
|   g_signal_connect (delegate, "item-activated",
 | |
|   		    G_CALLBACK (delegate_item_activated), receiver);
 | |
| }
 | |
| 
 | |
| GQuark
 | |
| _gtk_recent_chooser_delegate_get_quark (void)
 | |
| {
 | |
|   static GQuark quark = 0;
 | |
|   
 | |
|   if (G_UNLIKELY (quark == 0))
 | |
|     quark = g_quark_from_static_string ("gtk-recent-chooser-delegate");
 | |
|   
 | |
|   return quark;
 | |
| }
 | |
| 
 | |
| static GtkRecentChooser *
 | |
| get_delegate (GtkRecentChooser *receiver)
 | |
| {
 | |
|   return g_object_get_qdata (G_OBJECT (receiver),
 | |
|   			     GTK_RECENT_CHOOSER_DELEGATE_QUARK);
 | |
| }
 | |
| 
 | |
| static void
 | |
| delegate_set_sort_func (GtkRecentChooser  *chooser,
 | |
| 			GtkRecentSortFunc  sort_func,
 | |
| 			gpointer           sort_data,
 | |
| 			GDestroyNotify     data_destroy)
 | |
| {
 | |
|   gtk_recent_chooser_set_sort_func (get_delegate (chooser),
 | |
| 		  		    sort_func,
 | |
| 				    sort_data,
 | |
| 				    data_destroy);
 | |
| }
 | |
| 
 | |
| static void
 | |
| delegate_add_filter (GtkRecentChooser *chooser,
 | |
| 		     GtkRecentFilter  *filter)
 | |
| {
 | |
|   gtk_recent_chooser_add_filter (get_delegate (chooser), filter);
 | |
| }
 | |
| 
 | |
| static void
 | |
| delegate_remove_filter (GtkRecentChooser *chooser,
 | |
| 			GtkRecentFilter  *filter)
 | |
| {
 | |
|   gtk_recent_chooser_remove_filter (get_delegate (chooser), filter);
 | |
| }
 | |
| 
 | |
| static GSList *
 | |
| delegate_list_filters (GtkRecentChooser *chooser)
 | |
| {
 | |
|   return gtk_recent_chooser_list_filters (get_delegate (chooser));
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| delegate_select_uri (GtkRecentChooser  *chooser,
 | |
| 		     const gchar       *uri,
 | |
| 		     GError           **error)
 | |
| {
 | |
|   return gtk_recent_chooser_select_uri (get_delegate (chooser), uri, error);
 | |
| }
 | |
| 
 | |
| static void
 | |
| delegate_unselect_uri (GtkRecentChooser *chooser,
 | |
| 		       const gchar      *uri)
 | |
| {
 | |
|  gtk_recent_chooser_unselect_uri (get_delegate (chooser), uri);
 | |
| }
 | |
| 
 | |
| static GList *
 | |
| delegate_get_items (GtkRecentChooser *chooser)
 | |
| {
 | |
|   return gtk_recent_chooser_get_items (get_delegate (chooser));
 | |
| }
 | |
| 
 | |
| static GtkRecentManager *
 | |
| delegate_get_recent_manager (GtkRecentChooser *chooser)
 | |
| {
 | |
|   return _gtk_recent_chooser_get_recent_manager (get_delegate (chooser));
 | |
| }
 | |
| 
 | |
| static void
 | |
| delegate_select_all (GtkRecentChooser *chooser)
 | |
| {
 | |
|   gtk_recent_chooser_select_all (get_delegate (chooser));
 | |
| }
 | |
| 
 | |
| static void
 | |
| delegate_unselect_all (GtkRecentChooser *chooser)
 | |
| {
 | |
|   gtk_recent_chooser_unselect_all (get_delegate (chooser));
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| delegate_set_current_uri (GtkRecentChooser  *chooser,
 | |
| 			  const gchar       *uri,
 | |
| 			  GError           **error)
 | |
| {
 | |
|   return gtk_recent_chooser_set_current_uri (get_delegate (chooser), uri, error);
 | |
| }
 | |
| 
 | |
| static gchar *
 | |
| delegate_get_current_uri (GtkRecentChooser *chooser)
 | |
| {
 | |
|   return gtk_recent_chooser_get_current_uri (get_delegate (chooser));
 | |
| }
 | |
| 
 | |
| static void
 | |
| delegate_notify (GObject    *object,
 | |
| 		 GParamSpec *pspec,
 | |
| 		 gpointer    user_data)
 | |
| {
 | |
|   gpointer iface;
 | |
| 
 | |
|   iface = g_type_interface_peek (g_type_class_peek (G_OBJECT_TYPE (object)),
 | |
| 				 gtk_recent_chooser_get_type ());
 | |
|   if (g_object_interface_find_property (iface, pspec->name))
 | |
|     g_object_notify (user_data, pspec->name);
 | |
| }
 | |
| 
 | |
| static void
 | |
| delegate_selection_changed (GtkRecentChooser *receiver,
 | |
| 			    gpointer          user_data)
 | |
| {
 | |
|   _gtk_recent_chooser_selection_changed (GTK_RECENT_CHOOSER (user_data));
 | |
| }
 | |
| 
 | |
| static void
 | |
| delegate_item_activated (GtkRecentChooser *receiver,
 | |
| 			 gpointer          user_data)
 | |
| {
 | |
|   _gtk_recent_chooser_item_activated (GTK_RECENT_CHOOSER (user_data));
 | |
| }
 | |
| 
 | |
| static gint
 | |
| sort_recent_items_mru (GtkRecentInfo *a,
 | |
| 		       GtkRecentInfo *b,
 | |
| 		       gpointer       unused)
 | |
| {
 | |
|   g_assert (a != NULL && b != NULL);
 | |
|   
 | |
|   return gtk_recent_info_get_modified (b) - gtk_recent_info_get_modified (a);
 | |
| }
 | |
| 
 | |
| static gint
 | |
| sort_recent_items_lru (GtkRecentInfo *a,
 | |
| 		       GtkRecentInfo *b,
 | |
| 		       gpointer       unused)
 | |
| {
 | |
|   g_assert (a != NULL && b != NULL);
 | |
|   
 | |
|   return -1 * (gtk_recent_info_get_modified (b) - gtk_recent_info_get_modified (a));
 | |
| }
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
|   GtkRecentSortFunc func;
 | |
|   gpointer data;
 | |
| } SortRecentData;
 | |
| 
 | |
| /* our proxy sorting function */
 | |
| static gint
 | |
| sort_recent_items_proxy (gpointer *a,
 | |
|                          gpointer *b,
 | |
|                          gpointer  user_data)
 | |
| {
 | |
|   GtkRecentInfo *info_a = (GtkRecentInfo *) a;
 | |
|   GtkRecentInfo *info_b = (GtkRecentInfo *) b;
 | |
|   SortRecentData *sort_recent = user_data;
 | |
| 
 | |
|   if (sort_recent->func)
 | |
|     return (* sort_recent->func) (info_a, info_b, sort_recent->data);
 | |
|   
 | |
|   /* fallback */
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| get_is_recent_filtered (GtkRecentFilter *filter,
 | |
| 			GtkRecentInfo   *info)
 | |
| {
 | |
|   GtkRecentFilterInfo filter_info;
 | |
|   GtkRecentFilterFlags needed;
 | |
|   gboolean retval;
 | |
| 
 | |
|   g_assert (info != NULL);
 | |
|   
 | |
|   needed = gtk_recent_filter_get_needed (filter);
 | |
|   
 | |
|   filter_info.contains = GTK_RECENT_FILTER_URI | GTK_RECENT_FILTER_MIME_TYPE;
 | |
|   
 | |
|   filter_info.uri = gtk_recent_info_get_uri (info);
 | |
|   filter_info.mime_type = gtk_recent_info_get_mime_type (info);
 | |
|   
 | |
|   if (needed & GTK_RECENT_FILTER_DISPLAY_NAME)
 | |
|     {
 | |
|       filter_info.display_name = gtk_recent_info_get_display_name (info);
 | |
|       filter_info.contains |= GTK_RECENT_FILTER_DISPLAY_NAME;
 | |
|     }
 | |
|   else
 | |
|     filter_info.uri = NULL;
 | |
|   
 | |
|   if (needed & GTK_RECENT_FILTER_APPLICATION)
 | |
|     {
 | |
|       filter_info.applications = (const gchar **) gtk_recent_info_get_applications (info, NULL);
 | |
|       filter_info.contains |= GTK_RECENT_FILTER_APPLICATION;
 | |
|     }
 | |
|   else
 | |
|     filter_info.applications = NULL;
 | |
| 
 | |
|   if (needed & GTK_RECENT_FILTER_GROUP)
 | |
|     {
 | |
|       filter_info.groups = (const gchar **) gtk_recent_info_get_groups (info, NULL);
 | |
|       filter_info.contains |= GTK_RECENT_FILTER_GROUP;
 | |
|     }
 | |
|   else
 | |
|     filter_info.groups = NULL;
 | |
| 
 | |
|   if (needed & GTK_RECENT_FILTER_AGE)
 | |
|     {
 | |
|       filter_info.age = gtk_recent_info_get_age (info);
 | |
|       filter_info.contains |= GTK_RECENT_FILTER_AGE;
 | |
|     }
 | |
|   else
 | |
|     filter_info.age = -1;
 | |
|   
 | |
|   retval = gtk_recent_filter_filter (filter, &filter_info);
 | |
|   
 | |
|   /* these we own */
 | |
|   if (filter_info.applications)
 | |
|     g_strfreev ((gchar **) filter_info.applications);
 | |
|   if (filter_info.groups)
 | |
|     g_strfreev ((gchar **) filter_info.groups);
 | |
|   
 | |
|   return !retval;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * _gtk_recent_chooser_get_items:
 | |
|  * @chooser: a #GtkRecentChooser
 | |
|  * @filter: a #GtkRecentFilter
 | |
|  * @sort_func: (allow-none): sorting function, or %NULL
 | |
|  * @sort_data: (allow-none): sorting function data, or %NULL
 | |
|  *
 | |
|  * Default implementation for getting the filtered, sorted and
 | |
|  * clamped list of recently used resources from a #GtkRecentChooser.
 | |
|  * This function should be used by implementations of the
 | |
|  * #GtkRecentChooser interface inside the GtkRecentChooser::get_items
 | |
|  * vfunc.
 | |
|  *
 | |
|  * Return value: a list of #GtkRecentInfo objects
 | |
|  */
 | |
| GList *
 | |
| _gtk_recent_chooser_get_items (GtkRecentChooser  *chooser,
 | |
|                                GtkRecentFilter   *filter,
 | |
|                                GtkRecentSortFunc  sort_func,
 | |
|                                gpointer           sort_data)
 | |
| {
 | |
|   GtkRecentManager *manager;
 | |
|   gint limit;
 | |
|   GtkRecentSortType sort_type;
 | |
|   GList *items;
 | |
|   GCompareDataFunc compare_func;
 | |
|   gint length;
 | |
| 
 | |
|   g_return_val_if_fail (GTK_IS_RECENT_CHOOSER (chooser), NULL);
 | |
| 
 | |
|   manager = _gtk_recent_chooser_get_recent_manager (chooser);
 | |
|   if (!manager)
 | |
|     return NULL;
 | |
| 
 | |
|   items = gtk_recent_manager_get_items (manager);
 | |
|   if (!items)
 | |
|     return NULL;
 | |
| 
 | |
|   limit = gtk_recent_chooser_get_limit (chooser);
 | |
|   if (limit == 0)
 | |
|     return NULL;
 | |
| 
 | |
|   if (filter)
 | |
|     {
 | |
|       GList *filter_items, *l;
 | |
|       gboolean local_only = FALSE;
 | |
|       gboolean show_private = FALSE;
 | |
|       gboolean show_not_found = FALSE;
 | |
| 
 | |
|       g_object_get (G_OBJECT (chooser),
 | |
|                     "local-only", &local_only,
 | |
|                     "show-private", &show_private,
 | |
|                     "show-not-found", &show_not_found,
 | |
|                     NULL);
 | |
| 
 | |
|       filter_items = NULL;
 | |
|       for (l = items; l != NULL; l = l->next)
 | |
|         {
 | |
|           GtkRecentInfo *info = l->data;
 | |
|           gboolean remove_item = FALSE;
 | |
| 
 | |
|           if (get_is_recent_filtered (filter, info))
 | |
|             remove_item = TRUE;
 | |
|           
 | |
|           if (local_only && !gtk_recent_info_is_local (info))
 | |
|             remove_item = TRUE;
 | |
| 
 | |
|           if (!show_private && gtk_recent_info_get_private_hint (info))
 | |
|             remove_item = TRUE;
 | |
| 
 | |
|           if (!show_not_found && !gtk_recent_info_exists (info))
 | |
|             remove_item = TRUE;
 | |
|           
 | |
|           if (!remove_item)
 | |
|             filter_items = g_list_prepend (filter_items, info);
 | |
|           else
 | |
|             gtk_recent_info_unref (info);
 | |
|         }
 | |
|       
 | |
|       g_list_free (items);
 | |
|       items = filter_items;
 | |
|     }
 | |
| 
 | |
|   if (!items)
 | |
|     return NULL;
 | |
| 
 | |
|   sort_type = gtk_recent_chooser_get_sort_type (chooser);
 | |
|   switch (sort_type)
 | |
|     {
 | |
|     case GTK_RECENT_SORT_NONE:
 | |
|       compare_func = NULL;
 | |
|       break;
 | |
|     case GTK_RECENT_SORT_MRU:
 | |
|       compare_func = (GCompareDataFunc) sort_recent_items_mru;
 | |
|       break;
 | |
|     case GTK_RECENT_SORT_LRU:
 | |
|       compare_func = (GCompareDataFunc) sort_recent_items_lru;
 | |
|       break;
 | |
|     case GTK_RECENT_SORT_CUSTOM:
 | |
|       compare_func = (GCompareDataFunc) sort_recent_items_proxy;
 | |
|       break;
 | |
|     default:
 | |
|       g_assert_not_reached ();
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|   if (compare_func)
 | |
|     {
 | |
|       SortRecentData sort_recent;
 | |
| 
 | |
|       sort_recent.func = sort_func;
 | |
|       sort_recent.data = sort_data;
 | |
| 
 | |
|       items = g_list_sort_with_data (items, compare_func, &sort_recent);
 | |
|     }
 | |
|   
 | |
|   length = g_list_length (items);
 | |
|   if ((limit != -1) && (length > limit))
 | |
|     {
 | |
|       GList *clamp, *l;
 | |
|       
 | |
|       clamp = g_list_nth (items, limit - 1);
 | |
|       if (!clamp)
 | |
|         return items;
 | |
|       
 | |
|       l = clamp->next;
 | |
|       clamp->next = NULL;
 | |
|     
 | |
|       g_list_foreach (l, (GFunc) gtk_recent_info_unref, NULL);
 | |
|       g_list_free (l);
 | |
|     }
 | |
| 
 | |
|   return items;
 | |
| }
 | 
