aboutsummaryrefslogtreecommitdiffstats
path: root/embed/ephy-file-monitor.c
diff options
context:
space:
mode:
authorXan Lopez <xan@igalia.com>2012-06-27 18:47:18 +0800
committerXan Lopez <xan@igalia.com>2012-06-27 18:47:18 +0800
commit8118e5fb00fc3514940ce629ca6e1a7aeb5a3b1d (patch)
tree86d05c3baba077d4106125ccb13f63610f0722fb /embed/ephy-file-monitor.c
parent46f5f85b4a6bccfd8260935142660b4b324e94c7 (diff)
downloadgsoc2013-epiphany-8118e5fb00fc3514940ce629ca6e1a7aeb5a3b1d.tar
gsoc2013-epiphany-8118e5fb00fc3514940ce629ca6e1a7aeb5a3b1d.tar.gz
gsoc2013-epiphany-8118e5fb00fc3514940ce629ca6e1a7aeb5a3b1d.tar.bz2
gsoc2013-epiphany-8118e5fb00fc3514940ce629ca6e1a7aeb5a3b1d.tar.lz
gsoc2013-epiphany-8118e5fb00fc3514940ce629ca6e1a7aeb5a3b1d.tar.xz
gsoc2013-epiphany-8118e5fb00fc3514940ce629ca6e1a7aeb5a3b1d.tar.zst
gsoc2013-epiphany-8118e5fb00fc3514940ce629ca6e1a7aeb5a3b1d.zip
ephy-web-view: move the file monitoring code to its own class, EphyFileMonitor.
Since the vast majority of the code was really independent from EphyWebView.
Diffstat (limited to 'embed/ephy-file-monitor.c')
-rw-r--r--embed/ephy-file-monitor.c301
1 files changed, 301 insertions, 0 deletions
diff --git a/embed/ephy-file-monitor.c b/embed/ephy-file-monitor.c
new file mode 100644
index 000000000..3580e3d38
--- /dev/null
+++ b/embed/ephy-file-monitor.c
@@ -0,0 +1,301 @@
+/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set sw=2 ts=2 sts=2 et: */
+/*
+ * Copyright © 2012 Igalia S.L.
+ *
+ * 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, 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.
+ *
+ */
+
+#include "config.h"
+#include "ephy-file-monitor.h"
+
+#include "ephy-debug.h"
+
+#include <string.h>
+
+#define RELOAD_DELAY 250 /* ms */
+#define RELOAD_DELAY_MAX_TICKS 40 /* RELOAD_DELAY * RELOAD_DELAY_MAX_TICKS = 10 s */
+
+struct _EphyFileMonitorPrivate {
+ GFileMonitor *monitor;
+ gboolean monitor_directory;
+ guint reload_scheduled_id;
+ guint reload_delay_ticks;
+
+ EphyWebView *view;
+};
+
+enum {
+ PROP_0,
+
+ PROP_VIEW
+};
+
+#define EPHY_FILE_MONITOR_GET_PRIVATE(object) (G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_FILE_MONITOR, EphyFileMonitorPrivate))
+
+G_DEFINE_TYPE (EphyFileMonitor, ephy_file_monitor, G_TYPE_OBJECT)
+
+static void
+ephy_file_monitor_cancel (EphyFileMonitor *monitor)
+{
+ EphyFileMonitorPrivate *priv;
+
+ g_return_if_fail (EPHY_IS_FILE_MONITOR (monitor));
+
+ priv = monitor->priv;
+
+ if (priv->monitor != NULL) {
+ LOG ("Cancelling file monitor");
+
+ g_file_monitor_cancel (G_FILE_MONITOR (priv->monitor));
+ g_object_unref (priv->monitor);
+ priv->monitor = NULL;
+ }
+
+ if (priv->reload_scheduled_id != 0) {
+ LOG ("Cancelling scheduled reload");
+
+ g_source_remove (priv->reload_scheduled_id);
+ priv->reload_scheduled_id = 0;
+ }
+
+ priv->reload_delay_ticks = 0;
+}
+
+static gboolean
+ephy_file_monitor_reload_cb (EphyFileMonitor *monitor)
+{
+ EphyFileMonitorPrivate *priv = monitor->priv;
+
+ if (priv->reload_delay_ticks > 0) {
+ priv->reload_delay_ticks--;
+
+ /* Run again. */
+ return TRUE;
+ }
+
+ if (ephy_web_view_is_loading (priv->view)) {
+ /* Wait a bit to reload if we're still loading! */
+ priv->reload_delay_ticks = RELOAD_DELAY_MAX_TICKS / 2;
+
+ /* Run again. */
+ return TRUE;
+ }
+
+ priv->reload_scheduled_id = 0;
+
+ LOG ("Reloading file '%s'", ephy_web_view_get_address (priv->view));
+ webkit_web_view_reload (WEBKIT_WEB_VIEW (priv->view));
+
+ /* Don't run again. */
+ return FALSE;
+}
+
+static void
+ephy_file_monitor_changed_cb (GFileMonitor *monitor,
+ GFile *file,
+ GFile *other_file,
+ GFileMonitorEvent event_type,
+ EphyFileMonitor *file_monitor)
+{
+ gboolean should_reload;
+ EphyFileMonitorPrivate *priv = file_monitor->priv;
+
+ switch (event_type) {
+ /* These events will always trigger a reload: */
+ case G_FILE_MONITOR_EVENT_CHANGED:
+ case G_FILE_MONITOR_EVENT_CREATED:
+ should_reload = TRUE;
+ break;
+
+ /* These events will only trigger a reload for directories: */
+ case G_FILE_MONITOR_EVENT_DELETED:
+ case G_FILE_MONITOR_EVENT_ATTRIBUTE_CHANGED:
+ should_reload = priv->monitor_directory;
+ break;
+
+ /* These events don't trigger a reload: */
+ case G_FILE_MONITOR_EVENT_PRE_UNMOUNT:
+ case G_FILE_MONITOR_EVENT_UNMOUNTED:
+ case G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT:
+ default:
+ should_reload = FALSE;
+ break;
+ }
+
+ if (should_reload) {
+ /* We make a lot of assumptions here, but basically we know
+ * that we just have to reload, by construction.
+ * Delay the reload a little bit so we don't endlessly
+ * reload while a file is written.
+ */
+ if (priv->reload_delay_ticks == 0)
+ priv->reload_delay_ticks = 1;
+ else {
+ /* Exponential backoff. */
+ priv->reload_delay_ticks = MIN (priv->reload_delay_ticks * 2,
+ RELOAD_DELAY_MAX_TICKS);
+ }
+
+ if (priv->reload_scheduled_id == 0) {
+ priv->reload_scheduled_id =
+ g_timeout_add (RELOAD_DELAY,
+ (GSourceFunc)ephy_file_monitor_reload_cb, file_monitor);
+ }
+ }
+}
+
+void
+ephy_file_monitor_update_location (EphyFileMonitor *file_monitor,
+ const char *address)
+{
+ EphyFileMonitorPrivate *priv;
+ gboolean local;
+ char *anchor;
+ char *url;
+ GFile *file;
+ GFileType file_type;
+ GFileInfo *file_info;
+
+ g_return_if_fail (EPHY_IS_FILE_MONITOR (file_monitor));
+ g_return_if_fail (address != NULL);
+
+ priv = file_monitor->priv;
+
+ ephy_file_monitor_cancel (file_monitor);
+
+ local = g_str_has_prefix (address, "file://");
+ if (local == FALSE)
+ return;
+
+ /* strip off anchors */
+ anchor = strchr (address, '#');
+ if (anchor != NULL)
+ url = g_strndup (address, anchor - address);
+ else
+ url = g_strdup (address);
+
+ file = g_file_new_for_uri (url);
+ file_info = g_file_query_info (file,
+ G_FILE_ATTRIBUTE_STANDARD_TYPE,
+ 0, NULL, NULL);
+ if (file_info == NULL) {
+ g_object_unref (file);
+ g_free (url);
+ return;
+ }
+
+ file_type = g_file_info_get_file_type (file_info);
+ g_object_unref (file_info);
+
+ if (file_type == G_FILE_TYPE_DIRECTORY) {
+ priv->monitor = g_file_monitor_directory (file, 0, NULL, NULL);
+ g_signal_connect (priv->monitor, "changed",
+ G_CALLBACK (ephy_file_monitor_changed_cb),
+ file_monitor);
+ priv->monitor_directory = TRUE;
+ LOG ("Installed monitor for directory '%s'", url);
+ }
+ else if (file_type == G_FILE_TYPE_REGULAR) {
+ priv->monitor = g_file_monitor_file (file, 0, NULL, NULL);
+ g_signal_connect (priv->monitor, "changed",
+ G_CALLBACK (ephy_file_monitor_changed_cb),
+ file_monitor);
+ priv->monitor_directory = FALSE;
+ LOG ("Installed monitor for file '%s'", url);
+ }
+
+ g_object_unref (file);
+ g_free (url);
+}
+
+static void
+ephy_file_monitor_dispose (GObject *object)
+{
+ ephy_file_monitor_cancel (EPHY_FILE_MONITOR (object));
+
+ G_OBJECT_CLASS (ephy_file_monitor_parent_class)->dispose (object);
+}
+
+static void
+ephy_file_monitor_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ EphyFileMonitorPrivate *priv = EPHY_FILE_MONITOR (object)->priv;
+
+ switch (prop_id) {
+ case PROP_VIEW:
+ g_value_set_object (value, priv->view);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+ephy_file_monitor_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ EphyFileMonitorPrivate *priv = EPHY_FILE_MONITOR (object)->priv;
+
+ switch (prop_id) {
+ case PROP_VIEW:
+ priv->view = g_value_get_object (value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+ephy_file_monitor_class_init (EphyFileMonitorClass *klass)
+{
+ GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
+
+ gobject_class->dispose = ephy_file_monitor_dispose;
+ gobject_class->get_property = ephy_file_monitor_get_property;
+ gobject_class->set_property = ephy_file_monitor_set_property;
+
+ g_object_class_install_property (gobject_class,
+ PROP_VIEW,
+ g_param_spec_object ("view",
+ "View",
+ "The file monitor's associated view",
+ EPHY_TYPE_WEB_VIEW,
+ G_PARAM_READWRITE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB | G_PARAM_CONSTRUCT_ONLY));
+
+ g_type_class_add_private (gobject_class, sizeof (EphyFileMonitorPrivate));
+}
+
+static void
+ephy_file_monitor_init (EphyFileMonitor *monitor)
+{
+ monitor->priv = EPHY_FILE_MONITOR_GET_PRIVATE (monitor);
+}
+
+EphyFileMonitor *
+ephy_file_monitor_new (EphyWebView *view)
+{
+ return g_object_new (EPHY_TYPE_FILE_MONITOR,
+ "view", view,
+ NULL);
+}