mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-10-31 00:12:19 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			309 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			309 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* GIO - GLib Input, Output and Streaming Library
 | |
|  * 
 | |
|  * Copyright (C) 2006-2007 Red Hat, Inc.
 | |
|  *
 | |
|  * SPDX-License-Identifier: LGPL-2.1-or-later
 | |
|  *
 | |
|  * 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.1 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, see <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  * Author: Alexander Larsson <alexl@redhat.com>
 | |
|  */
 | |
| 
 | |
| #include "config.h"
 | |
| #include <string.h>
 | |
| 
 | |
| #include "gfile.h"
 | |
| #include "gfilemonitor.h"
 | |
| #include "gioenumtypes.h"
 | |
| #include "glib.h"
 | |
| #include "glibintl.h"
 | |
| #include "gmarshal-internal.h"
 | |
| #include "gvfs.h"
 | |
| 
 | |
| /**
 | |
|  * GFileMonitor:
 | |
|  *
 | |
|  * Monitors a file or directory for changes.
 | |
|  *
 | |
|  * To obtain a `GFileMonitor` for a file or directory, use
 | |
|  * [method@Gio.File.monitor], [method@Gio.File.monitor_file], or
 | |
|  * [method@Gio.File.monitor_directory].
 | |
|  *
 | |
|  * To get informed about changes to the file or directory you are
 | |
|  * monitoring, connect to the [signal@Gio.FileMonitor::changed] signal. The
 | |
|  * signal will be emitted in the thread-default main context (see
 | |
|  * [method@GLib.MainContext.push_thread_default]) of the thread that the monitor
 | |
|  * was created in (though if the global default main context is blocked, this
 | |
|  * may cause notifications to be blocked even if the thread-default
 | |
|  * context is still running).
 | |
|  **/
 | |
| 
 | |
| #define DEFAULT_RATE_LIMIT_MSECS 800
 | |
| 
 | |
| typedef enum {
 | |
|   CANCEL_STATE_NONE,
 | |
|   CANCEL_STATE_CANCELLING,
 | |
|   CANCEL_STATE_CANCELLED,
 | |
| } GFileMonitorCancelState;
 | |
| 
 | |
| struct _GFileMonitorPrivate
 | |
| {
 | |
|   int cancelled; /* atomic */
 | |
| };
 | |
| 
 | |
| G_DEFINE_ABSTRACT_TYPE_WITH_PRIVATE (GFileMonitor, g_file_monitor, G_TYPE_OBJECT)
 | |
| 
 | |
| typedef enum
 | |
| {
 | |
|   PROP_RATE_LIMIT = 1,
 | |
|   PROP_CANCELLED
 | |
| } GFileMonitorProperty;
 | |
| 
 | |
| static GParamSpec *props[PROP_CANCELLED + 1];
 | |
| static guint g_file_monitor_changed_signal;
 | |
| 
 | |
| static void
 | |
| g_file_monitor_set_property (GObject      *object,
 | |
|                              guint         prop_id,
 | |
|                              const GValue *value,
 | |
|                              GParamSpec   *pspec)
 | |
| {
 | |
|   switch ((GFileMonitorProperty) prop_id)
 | |
|     {
 | |
|     case PROP_RATE_LIMIT:
 | |
|       /* not supported by default */
 | |
|       break;
 | |
| 
 | |
|     case PROP_CANCELLED:
 | |
|       /* Read only */
 | |
|       g_assert_not_reached ();
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
 | |
|       break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_file_monitor_get_property (GObject    *object,
 | |
|                              guint       prop_id,
 | |
|                              GValue     *value,
 | |
|                              GParamSpec *pspec)
 | |
| {
 | |
|   GFileMonitor *self = G_FILE_MONITOR (object);
 | |
| 
 | |
|   switch ((GFileMonitorProperty) prop_id)
 | |
|     {
 | |
|     case PROP_RATE_LIMIT:
 | |
|       /* we expect this to be overridden... */
 | |
|       g_value_set_int (value, DEFAULT_RATE_LIMIT_MSECS);
 | |
|       break;
 | |
| 
 | |
|     case PROP_CANCELLED:
 | |
|       g_value_set_boolean (value, g_file_monitor_is_cancelled (self));
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
 | |
|       break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_file_monitor_dispose (GObject *object)
 | |
| {
 | |
|   GFileMonitor *monitor = G_FILE_MONITOR (object);
 | |
| 
 | |
|   /* Make sure we cancel on last unref */
 | |
|   g_file_monitor_cancel (monitor);
 | |
| 
 | |
|   G_OBJECT_CLASS (g_file_monitor_parent_class)->dispose (object);
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_file_monitor_init (GFileMonitor *monitor)
 | |
| {
 | |
|   monitor->priv = g_file_monitor_get_instance_private (monitor);
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_file_monitor_class_init (GFileMonitorClass *klass)
 | |
| {
 | |
|   GObjectClass *object_class;
 | |
| 
 | |
|   object_class = G_OBJECT_CLASS (klass);
 | |
|   object_class->dispose = g_file_monitor_dispose;
 | |
|   object_class->get_property = g_file_monitor_get_property;
 | |
|   object_class->set_property = g_file_monitor_set_property;
 | |
| 
 | |
|   /**
 | |
|    * GFileMonitor::changed:
 | |
|    * @monitor: a #GFileMonitor.
 | |
|    * @file: a #GFile.
 | |
|    * @other_file: (nullable): a #GFile or #NULL.
 | |
|    * @event_type: a #GFileMonitorEvent.
 | |
|    *
 | |
|    * Emitted when @file has been changed.
 | |
|    *
 | |
|    * If using %G_FILE_MONITOR_WATCH_MOVES on a directory monitor, and
 | |
|    * the information is available (and if supported by the backend),
 | |
|    * @event_type may be %G_FILE_MONITOR_EVENT_RENAMED,
 | |
|    * %G_FILE_MONITOR_EVENT_MOVED_IN or %G_FILE_MONITOR_EVENT_MOVED_OUT.
 | |
|    *
 | |
|    * In all cases @file will be a child of the monitored directory.  For
 | |
|    * renames, @file will be the old name and @other_file is the new
 | |
|    * name.  For "moved in" events, @file is the name of the file that
 | |
|    * appeared and @other_file is the old name that it was moved from (in
 | |
|    * another directory).  For "moved out" events, @file is the name of
 | |
|    * the file that used to be in this directory and @other_file is the
 | |
|    * name of the file at its new location.
 | |
|    *
 | |
|    * It makes sense to treat %G_FILE_MONITOR_EVENT_MOVED_IN as
 | |
|    * equivalent to %G_FILE_MONITOR_EVENT_CREATED and
 | |
|    * %G_FILE_MONITOR_EVENT_MOVED_OUT as equivalent to
 | |
|    * %G_FILE_MONITOR_EVENT_DELETED, with extra information.
 | |
|    * %G_FILE_MONITOR_EVENT_RENAMED is equivalent to a delete/create
 | |
|    * pair.  This is exactly how the events will be reported in the case
 | |
|    * that the %G_FILE_MONITOR_WATCH_MOVES flag is not in use.
 | |
|    *
 | |
|    * If using the deprecated flag %G_FILE_MONITOR_SEND_MOVED flag and @event_type is
 | |
|    * %G_FILE_MONITOR_EVENT_MOVED, @file will be set to a #GFile containing the
 | |
|    * old path, and @other_file will be set to a #GFile containing the new path.
 | |
|    *
 | |
|    * In all the other cases, @other_file will be set to #NULL.
 | |
|    **/
 | |
|   g_file_monitor_changed_signal = g_signal_new (I_("changed"),
 | |
|                                                 G_TYPE_FILE_MONITOR,
 | |
|                                                 G_SIGNAL_RUN_LAST,
 | |
|                                                 G_STRUCT_OFFSET (GFileMonitorClass, changed),
 | |
|                                                 NULL, NULL,
 | |
|                                                 _g_cclosure_marshal_VOID__OBJECT_OBJECT_ENUM,
 | |
|                                                 G_TYPE_NONE, 3,
 | |
|                                                 G_TYPE_FILE, G_TYPE_FILE, G_TYPE_FILE_MONITOR_EVENT);
 | |
|   g_signal_set_va_marshaller (g_file_monitor_changed_signal,
 | |
|                               G_TYPE_FROM_CLASS (klass),
 | |
|                               _g_cclosure_marshal_VOID__OBJECT_OBJECT_ENUMv);
 | |
| 
 | |
|   /**
 | |
|    * GFileMonitor:rate-limit:
 | |
|    *
 | |
|    * The limit of the monitor to watch for changes, in milliseconds.
 | |
|    */
 | |
|   props[PROP_RATE_LIMIT] =
 | |
|       g_param_spec_int ("rate-limit", NULL, NULL,
 | |
|                         0, G_MAXINT, DEFAULT_RATE_LIMIT_MSECS, G_PARAM_READWRITE |
 | |
|                         G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
 | |
| 
 | |
|   /**
 | |
|    * GFileMonitor:cancelled:
 | |
|    *
 | |
|    * Whether the monitor has been cancelled.
 | |
|    */
 | |
|   props[PROP_CANCELLED] =
 | |
|       g_param_spec_boolean ("cancelled", NULL, NULL,
 | |
|                             FALSE, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
 | |
| 
 | |
|   g_object_class_install_properties (object_class, G_N_ELEMENTS (props), props);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * g_file_monitor_is_cancelled:
 | |
|  * @monitor: a #GFileMonitor
 | |
|  * 
 | |
|  * Returns whether the monitor is canceled.
 | |
|  *
 | |
|  * Returns: %TRUE if monitor is canceled. %FALSE otherwise.
 | |
|  **/
 | |
| gboolean
 | |
| g_file_monitor_is_cancelled (GFileMonitor *monitor)
 | |
| {
 | |
|   g_return_val_if_fail (G_IS_FILE_MONITOR (monitor), FALSE);
 | |
| 
 | |
|   return g_atomic_int_get (&monitor->priv->cancelled) == CANCEL_STATE_CANCELLED;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * g_file_monitor_cancel:
 | |
|  * @monitor: a #GFileMonitor.
 | |
|  *
 | |
|  * Cancels a file monitor.
 | |
|  *
 | |
|  * Returns: always %TRUE
 | |
|  **/
 | |
| gboolean
 | |
| g_file_monitor_cancel (GFileMonitor *monitor)
 | |
| {
 | |
|   g_return_val_if_fail (G_IS_FILE_MONITOR (monitor), FALSE);
 | |
| 
 | |
|   if (g_atomic_int_compare_and_exchange (&monitor->priv->cancelled,
 | |
|                                          CANCEL_STATE_NONE,
 | |
|                                          CANCEL_STATE_CANCELLING))
 | |
|     {
 | |
|       G_FILE_MONITOR_GET_CLASS (monitor)->cancel (monitor);
 | |
| 
 | |
|       g_atomic_int_set (&monitor->priv->cancelled, CANCEL_STATE_CANCELLED);
 | |
|       g_object_notify_by_pspec (G_OBJECT (monitor), props[PROP_CANCELLED]);
 | |
|     }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * g_file_monitor_set_rate_limit:
 | |
|  * @monitor: a #GFileMonitor.
 | |
|  * @limit_msecs: a non-negative integer with the limit in milliseconds
 | |
|  *     to poll for changes
 | |
|  *
 | |
|  * Sets the rate limit to which the @monitor will report
 | |
|  * consecutive change events to the same file.
 | |
|  */
 | |
| void
 | |
| g_file_monitor_set_rate_limit (GFileMonitor *monitor,
 | |
|                                gint          limit_msecs)
 | |
| {
 | |
|   g_object_set (monitor, "rate-limit", limit_msecs, NULL);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * g_file_monitor_emit_event:
 | |
|  * @monitor: a #GFileMonitor.
 | |
|  * @child: a #GFile.
 | |
|  * @other_file: (nullable): a #GFile, or %NULL.
 | |
|  * @event_type: a set of #GFileMonitorEvent flags.
 | |
|  *
 | |
|  * Emits the #GFileMonitor::changed signal if a change
 | |
|  * has taken place. Should be called from file monitor
 | |
|  * implementations only.
 | |
|  *
 | |
|  * Implementations are responsible to call this method from the
 | |
|  * thread-default main context (see [method@GLib.MainContext.push_thread_default])
 | |
|  * of the thread that the monitor was created in.
 | |
|  **/
 | |
| void
 | |
| g_file_monitor_emit_event (GFileMonitor      *monitor,
 | |
|                            GFile             *child,
 | |
|                            GFile             *other_file,
 | |
|                            GFileMonitorEvent  event_type)
 | |
| {
 | |
|   g_return_if_fail (G_IS_FILE_MONITOR (monitor));
 | |
|   g_return_if_fail (G_IS_FILE (child));
 | |
|   g_return_if_fail (!other_file || G_IS_FILE (other_file));
 | |
| 
 | |
|   if (g_atomic_int_get (&monitor->priv->cancelled) != CANCEL_STATE_NONE)
 | |
|     return;
 | |
| 
 | |
|   g_signal_emit (monitor, g_file_monitor_changed_signal, 0, child, other_file, event_type);
 | |
| }
 |