267 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			267 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* GTK - The GIMP Toolkit
 | 
						|
 * gtktrashmonitor.h: Monitor the trash:/// folder to see if there is trash or not
 | 
						|
 * Copyright (C) 2011 Suse
 | 
						|
 *
 | 
						|
 * This program 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 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, see <http://www.gnu.org/licenses/>.
 | 
						|
 *
 | 
						|
 * Authors: Federico Mena Quintero <federico@gnome.org>
 | 
						|
 */
 | 
						|
 | 
						|
#include "config.h"
 | 
						|
 | 
						|
#include "gtkintl.h"
 | 
						|
#include "gtkmarshalers.h"
 | 
						|
#include "gtktrashmonitor.h"
 | 
						|
 | 
						|
struct _GtkTrashMonitor
 | 
						|
{
 | 
						|
  GObject parent;
 | 
						|
 | 
						|
  GFileMonitor *file_monitor;
 | 
						|
  gulong file_monitor_changed_id;
 | 
						|
  
 | 
						|
  guint has_trash : 1;
 | 
						|
};
 | 
						|
 | 
						|
struct _GtkTrashMonitorClass
 | 
						|
{
 | 
						|
  GObjectClass parent_class;
 | 
						|
 | 
						|
  void (* trash_state_changed) (GtkTrashMonitor *monitor);
 | 
						|
};
 | 
						|
 | 
						|
enum {
 | 
						|
  TRASH_STATE_CHANGED,
 | 
						|
  LAST_SIGNAL
 | 
						|
};
 | 
						|
 | 
						|
static guint signals[LAST_SIGNAL];
 | 
						|
 | 
						|
G_DEFINE_TYPE (GtkTrashMonitor, _gtk_trash_monitor, G_TYPE_OBJECT)
 | 
						|
 | 
						|
static GtkTrashMonitor *the_trash_monitor;
 | 
						|
 | 
						|
#define ICON_NAME_TRASH_EMPTY "user-trash-symbolic"
 | 
						|
#define ICON_NAME_TRASH_FULL  "user-trash-full-symbolic"
 | 
						|
 | 
						|
static void
 | 
						|
gtk_trash_monitor_dispose (GObject *object)
 | 
						|
{
 | 
						|
  GtkTrashMonitor *monitor;
 | 
						|
 | 
						|
  monitor = GTK_TRASH_MONITOR (object);
 | 
						|
 | 
						|
  if (monitor->file_monitor)
 | 
						|
    {
 | 
						|
      g_signal_handler_disconnect (monitor->file_monitor, monitor->file_monitor_changed_id);
 | 
						|
      monitor->file_monitor_changed_id = 0;
 | 
						|
 | 
						|
      g_clear_object (&monitor->file_monitor);
 | 
						|
    }
 | 
						|
 | 
						|
  G_OBJECT_CLASS (_gtk_trash_monitor_parent_class)->dispose (object);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
_gtk_trash_monitor_class_init (GtkTrashMonitorClass *class)
 | 
						|
{
 | 
						|
  GObjectClass *gobject_class;
 | 
						|
 | 
						|
  gobject_class = (GObjectClass *) class;
 | 
						|
 | 
						|
  gobject_class->dispose = gtk_trash_monitor_dispose;
 | 
						|
 | 
						|
  signals[TRASH_STATE_CHANGED] =
 | 
						|
    g_signal_new (I_("trash-state-changed"),
 | 
						|
		  G_OBJECT_CLASS_TYPE (gobject_class),
 | 
						|
		  G_SIGNAL_RUN_FIRST,
 | 
						|
		  G_STRUCT_OFFSET (GtkTrashMonitorClass, trash_state_changed),
 | 
						|
		  NULL, NULL,
 | 
						|
		  _gtk_marshal_VOID__VOID,
 | 
						|
		  G_TYPE_NONE, 0);
 | 
						|
}
 | 
						|
 | 
						|
/* Updates the internal has_trash flag and emits the "trash-state-changed" signal */
 | 
						|
static void
 | 
						|
update_has_trash_and_notify (GtkTrashMonitor *monitor,
 | 
						|
			     gboolean has_trash)
 | 
						|
{
 | 
						|
  monitor->has_trash = !!has_trash;
 | 
						|
 | 
						|
  g_signal_emit (monitor, signals[TRASH_STATE_CHANGED], 0); 
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
trash_enumerate_next_files_cb (GObject *source,
 | 
						|
			       GAsyncResult *result,
 | 
						|
			       gpointer user_data)
 | 
						|
{
 | 
						|
  GtkTrashMonitor *monitor = GTK_TRASH_MONITOR (user_data);
 | 
						|
  GFileEnumerator *enumerator;
 | 
						|
  GList *infos;
 | 
						|
 | 
						|
  enumerator = G_FILE_ENUMERATOR (source);
 | 
						|
 | 
						|
  infos = g_file_enumerator_next_files_finish (enumerator, result, NULL);
 | 
						|
  if (infos)
 | 
						|
    {
 | 
						|
      update_has_trash_and_notify (monitor, TRUE);
 | 
						|
      g_list_free_full (infos, g_object_unref);
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      update_has_trash_and_notify (monitor, FALSE);
 | 
						|
    }
 | 
						|
 | 
						|
  g_object_unref (monitor); /* was reffed in recompute_trash_state() */
 | 
						|
}
 | 
						|
 | 
						|
/* Callback used from g_file_enumerate_children_async() - this is what enumerates "trash:///" */
 | 
						|
static void
 | 
						|
trash_enumerate_children_cb (GObject *source,
 | 
						|
			     GAsyncResult *result,
 | 
						|
			     gpointer user_data)
 | 
						|
{
 | 
						|
  GtkTrashMonitor *monitor = GTK_TRASH_MONITOR (user_data);
 | 
						|
  GFileEnumerator *enumerator;
 | 
						|
 | 
						|
  enumerator = g_file_enumerate_children_finish (G_FILE (source), result, NULL);
 | 
						|
  if (enumerator)
 | 
						|
    {
 | 
						|
      g_file_enumerator_next_files_async (enumerator,
 | 
						|
					  1,
 | 
						|
					  G_PRIORITY_DEFAULT,
 | 
						|
					  NULL,
 | 
						|
					  trash_enumerate_next_files_cb,
 | 
						|
					  monitor);
 | 
						|
      g_object_unref (enumerator);
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      update_has_trash_and_notify (monitor, FALSE);
 | 
						|
      g_object_unref (monitor); /* was reffed in recompute_trash_state() */
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/* Asynchronously recomputes whether there is trash or not */
 | 
						|
static void
 | 
						|
recompute_trash_state (GtkTrashMonitor *monitor)
 | 
						|
{
 | 
						|
  GFile *file;
 | 
						|
 | 
						|
  g_object_ref (monitor);
 | 
						|
 | 
						|
  file = g_file_new_for_uri ("trash:///");
 | 
						|
  g_file_enumerate_children_async (file,
 | 
						|
				   G_FILE_ATTRIBUTE_STANDARD_TYPE,
 | 
						|
				   G_FILE_QUERY_INFO_NONE,
 | 
						|
				   G_PRIORITY_DEFAULT,
 | 
						|
				   NULL,
 | 
						|
				   trash_enumerate_children_cb,
 | 
						|
				   monitor);
 | 
						|
 | 
						|
  g_object_unref (file);
 | 
						|
}
 | 
						|
 | 
						|
/* Callback used when the "trash:///" file monitor changes; we just recompute the trash state
 | 
						|
 * whenever something happens.
 | 
						|
 */
 | 
						|
static void
 | 
						|
file_monitor_changed_cb (GFileMonitor	   *file_monitor,
 | 
						|
			 GFile		   *child,
 | 
						|
			 GFile		   *other_file,
 | 
						|
			 GFileMonitorEvent  event_type,
 | 
						|
			 GtkTrashMonitor   *monitor)
 | 
						|
{
 | 
						|
  recompute_trash_state (monitor);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
_gtk_trash_monitor_init (GtkTrashMonitor *monitor)
 | 
						|
{
 | 
						|
  GFile *file;
 | 
						|
 | 
						|
  file = g_file_new_for_uri ("trash:///");
 | 
						|
 | 
						|
  monitor->file_monitor = g_file_monitor_file (file, G_FILE_MONITOR_NONE, NULL, NULL);
 | 
						|
 | 
						|
  g_object_unref (file);
 | 
						|
 | 
						|
  if (monitor->file_monitor)
 | 
						|
    monitor->file_monitor_changed_id = g_signal_connect (monitor->file_monitor, "changed",
 | 
						|
							 G_CALLBACK (file_monitor_changed_cb), monitor);
 | 
						|
 | 
						|
  recompute_trash_state (monitor);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * _gtk_trash_monitor_get:
 | 
						|
 *
 | 
						|
 * Return value: (transfer full): a new reference to the singleton
 | 
						|
 * #GtkTrashMonitor object.  Be sure to call g_object_unref() on it when you are
 | 
						|
 * done with the trash monitor.
 | 
						|
 */
 | 
						|
GtkTrashMonitor *
 | 
						|
_gtk_trash_monitor_get (void)
 | 
						|
{
 | 
						|
  if (the_trash_monitor != NULL)
 | 
						|
    {
 | 
						|
      g_object_ref (the_trash_monitor);
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      the_trash_monitor = g_object_new (GTK_TYPE_TRASH_MONITOR, NULL);
 | 
						|
      g_object_add_weak_pointer (G_OBJECT (the_trash_monitor), (gpointer *) &the_trash_monitor);
 | 
						|
    }
 | 
						|
 | 
						|
  return the_trash_monitor;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * _gtk_trash_monitor_get_icon:
 | 
						|
 * @monitor: a #GtkTrashMonitor
 | 
						|
 *
 | 
						|
 * Return value: (transfer full): the #GIcon that should be used to represent
 | 
						|
 * the state of the trash folder on screen, based on whether there is trash or
 | 
						|
 * not.
 | 
						|
 */
 | 
						|
GIcon *
 | 
						|
_gtk_trash_monitor_get_icon (GtkTrashMonitor *monitor)
 | 
						|
{
 | 
						|
  const char *icon_name;
 | 
						|
 | 
						|
  g_return_val_if_fail (GTK_IS_TRASH_MONITOR (monitor), NULL);
 | 
						|
 | 
						|
  if (monitor->has_trash)
 | 
						|
    icon_name = ICON_NAME_TRASH_FULL;
 | 
						|
  else
 | 
						|
    icon_name = ICON_NAME_TRASH_EMPTY;
 | 
						|
 | 
						|
  return g_themed_icon_new (icon_name);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * _gtk_trash_monitor_get_has_trash:
 | 
						|
 * @monitor: a #GtkTrashMonitor
 | 
						|
 *
 | 
						|
 * Return value: #TRUE if there is trash in the trash:/// folder, or #FALSE otherwise.
 | 
						|
 */
 | 
						|
gboolean
 | 
						|
_gtk_trash_monitor_get_has_trash (GtkTrashMonitor *monitor)
 | 
						|
{
 | 
						|
  g_return_val_if_fail (GTK_IS_TRASH_MONITOR (monitor), FALSE);
 | 
						|
 | 
						|
  return monitor->has_trash;
 | 
						|
}
 |