summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJasper St. Pierre <jstpierre@mecheye.net>2016-06-10 13:40:12 -0700
committerJasper St. Pierre <jstpierre@mecheye.net>2016-06-20 17:47:18 -0700
commit0343349dfd2ee757ee1e4181482f8c7df39aa454 (patch)
tree621fa3a6dfde3fe1db1cff8975d9725938f8f259
parent12c5a9dafc1604353fbedc7c099dbaa4e3f82f89 (diff)
downloadglib-wip/jstpierre/gvfs.tar.gz
gvfs: additional schemeswip/jstpierre/gvfs
-rw-r--r--gio/gvfs.c122
-rw-r--r--gio/gvfs.h10
2 files changed, 123 insertions, 9 deletions
diff --git a/gio/gvfs.c b/gio/gvfs.c
index 2390b7c1e..ced5e599b 100644
--- a/gio/gvfs.c
+++ b/gio/gvfs.c
@@ -37,16 +37,70 @@
*
*/
-G_DEFINE_TYPE (GVfs, g_vfs, G_TYPE_OBJECT);
+/**
+ * GVfsURILookupFunc:
+ * @vfs: A #GVfs
+ * @uri: A URI to look up
+ * @user_data: User data passed to g_vfs_register_uri_scheme().
+ *
+ * Returns: (transfer full): A #GFile for the passed in @uri.
+ *
+ * Since: 2.50
+ */
+
+typedef struct {
+ GVfsURILookupFunc func;
+ gpointer user_data;
+ GDestroyNotify destroy;
+} GVfsURILookupFuncClosure;
+
+struct _GVfsPrivate {
+ GHashTable *additional_schemes;
+};
+
+G_DEFINE_TYPE_WITH_PRIVATE (GVfs, g_vfs, G_TYPE_OBJECT);
+
+static void
+g_vfs_dispose (GObject *object)
+{
+ GVfs *vfs = G_VFS (object);
+ GVfsPrivate *priv = g_vfs_get_instance_private (vfs);
+
+ g_clear_pointer (&priv->additional_schemes, g_hash_table_destroy);
+
+ G_OBJECT_CLASS (g_vfs_parent_class)->dispose (object);
+}
static void
g_vfs_class_init (GVfsClass *klass)
{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ object_class->dispose = g_vfs_dispose;
+}
+
+static GFile *
+resource_get_file_for_uri (GVfs *vfs, const char *uri, gpointer user_data)
+{
+ return _g_resource_file_new (uri);
+}
+
+static void
+g_vfs_uri_lookup_func_closure_free (GVfsURILookupFuncClosure *closure)
+{
+ if (closure->destroy)
+ closure->destroy (closure->user_data);
+ g_free (closure);
}
static void
g_vfs_init (GVfs *vfs)
{
+ GVfsPrivate *priv = g_vfs_get_instance_private (vfs);
+
+ priv->additional_schemes = g_hash_table_new_full (g_str_hash, g_str_equal,
+ g_free, (GDestroyNotify) g_vfs_uri_lookup_func_closure_free);
+
+ g_vfs_register_uri_scheme (vfs, "resource", resource_get_file_for_uri, NULL, NULL);
}
/**
@@ -70,7 +124,6 @@ g_vfs_is_active (GVfs *vfs)
return (* class->is_active) (vfs);
}
-
/**
* g_vfs_get_file_for_path:
* @vfs: a #GVfs.
@@ -95,6 +148,26 @@ g_vfs_get_file_for_path (GVfs *vfs,
return (* class->get_file_for_path) (vfs, path);
}
+static GFile *
+get_file_for_uri_internal (GVfs *vfs, const char *uri)
+{
+ GVfsPrivate *priv = g_vfs_get_instance_private (vfs);
+ char *scheme;
+ GVfsURILookupFuncClosure *closure;
+
+ scheme = g_uri_parse_scheme (uri);
+ if (scheme == NULL)
+ return NULL;
+
+ closure = g_hash_table_lookup (priv->additional_schemes, scheme);
+ g_free (scheme);
+
+ if (closure)
+ return closure->func (vfs, uri, closure->user_data);
+ else
+ return NULL;
+}
+
/**
* g_vfs_get_file_for_uri:
* @vfs: a#GVfs.
@@ -114,19 +187,16 @@ g_vfs_get_file_for_uri (GVfs *vfs,
const char *uri)
{
GVfsClass *class;
+ GFile *ret;
g_return_val_if_fail (G_IS_VFS (vfs), NULL);
g_return_val_if_fail (uri != NULL, NULL);
class = G_VFS_GET_CLASS (vfs);
- /* This is an unfortunate placement, but we really
- * need to check this before chaining to the vfs,
- * because we want to support resource uris for
- * all vfs:es, even those that predate resources.
- */
- if (g_str_has_prefix (uri, "resource:"))
- return _g_resource_file_new (uri);
+ ret = get_file_for_uri_internal (vfs, uri);
+ if (ret)
+ return ret;
return (* class->get_file_for_uri) (vfs, uri);
}
@@ -216,3 +286,37 @@ g_vfs_get_local (void)
return G_VFS (vfs);
}
+
+/**
+ * g_vfs_register_uri_scheme:
+ * @vfs: A #GVfs
+ * @scheme: The scheme to register a URI handler on.
+ * @func: The lookup function to register.
+ * @user_data: (nullable): User data to call the lookup function with.
+ * @destroy: (nullable): Called when the closure is destroyed.
+ *
+ * Registers the scheme @scheme so that when URIs with this scheme are
+ * looked up, the registered @func is called. There is currently no way
+ * to unregister a scheme. It is undefined if two pieces of code try to
+ * register the same scheme.
+ *
+ * Since: 2.50
+ */
+void
+g_vfs_register_uri_scheme (GVfs *vfs,
+ const char *scheme,
+ GVfsURILookupFunc func,
+ gpointer user_data,
+ GDestroyNotify destroy)
+{
+ GVfsPrivate *priv = g_vfs_get_instance_private (vfs);
+ GVfsURILookupFuncClosure *closure;
+
+ closure = g_new (GVfsURILookupFuncClosure, 1);
+ closure->func = func;
+ closure->user_data = user_data;
+ closure->destroy = destroy;
+
+ g_hash_table_replace (priv->additional_schemes, g_strdup (scheme), closure);
+}
+
diff --git a/gio/gvfs.h b/gio/gvfs.h
index e67e2d9a1..11359ae9e 100644
--- a/gio/gvfs.h
+++ b/gio/gvfs.h
@@ -36,6 +36,8 @@ G_BEGIN_DECLS
#define G_IS_VFS(o) (G_TYPE_CHECK_INSTANCE_TYPE ((o), G_TYPE_VFS))
#define G_IS_VFS_CLASS(k) (G_TYPE_CHECK_CLASS_TYPE ((k), G_TYPE_VFS))
+typedef GFile * (* GVfsURILookupFunc) (GVfs *vfs, const char *uri, gpointer user_data);
+
/**
* G_VFS_EXTENSION_POINT_NAME:
*
@@ -50,6 +52,7 @@ G_BEGIN_DECLS
* Virtual File System object.
**/
typedef struct _GVfsClass GVfsClass;
+typedef struct _GVfsPrivate GVfsPrivate;
struct _GVfs
{
@@ -127,6 +130,13 @@ GVfs * g_vfs_get_default (void);
GLIB_AVAILABLE_IN_ALL
GVfs * g_vfs_get_local (void);
+GLIB_AVAILABLE_IN_2_50
+void g_vfs_register_uri_scheme (GVfs *vfs,
+ const char *scheme,
+ GVfsURILookupFunc func,
+ gpointer user_data,
+ GDestroyNotify destroy);
+
G_END_DECLS
#endif /* __G_VFS_H__ */