summaryrefslogtreecommitdiff
path: root/libcaja-private/caja-desktop-directory-file.c
diff options
context:
space:
mode:
Diffstat (limited to 'libcaja-private/caja-desktop-directory-file.c')
-rw-r--r--libcaja-private/caja-desktop-directory-file.c731
1 files changed, 731 insertions, 0 deletions
diff --git a/libcaja-private/caja-desktop-directory-file.c b/libcaja-private/caja-desktop-directory-file.c
new file mode 100644
index 00000000..ebfa47ce
--- /dev/null
+++ b/libcaja-private/caja-desktop-directory-file.c
@@ -0,0 +1,731 @@
+/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*-
+
+ caja-desktop-directory-file.c: Subclass of CajaFile to help implement the
+ virtual desktop.
+
+ Copyright (C) 2003 Red Hat, Inc.
+
+ 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 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, write to the
+ Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ Boston, MA 02111-1307, USA.
+
+ Author: Alexander Larsson <[email protected]>
+*/
+
+#include <config.h>
+#include "caja-desktop-directory-file.h"
+
+#include "caja-directory-notify.h"
+#include "caja-directory-private.h"
+#include "caja-file-attributes.h"
+#include "caja-file-private.h"
+#include "caja-file-utilities.h"
+#include <eel/eel-glib-extensions.h>
+#include <eel/eel-gtk-macros.h>
+#include <mateconf/mateconf-client.h>
+#include <mateconf/mateconf-value.h>
+#include "caja-desktop-directory.h"
+#include "caja-metadata.h"
+#include <gtk/gtk.h>
+#include <glib/gi18n.h>
+#include <string.h>
+
+struct CajaDesktopDirectoryFileDetails
+{
+ CajaDesktopDirectory *desktop_directory;
+
+ CajaFile *real_dir_file;
+
+ GHashTable *callbacks;
+ GHashTable *monitors;
+};
+
+typedef struct
+{
+ CajaDesktopDirectoryFile *desktop_file;
+ CajaFileCallback callback;
+ gpointer callback_data;
+
+ CajaFileAttributes delegated_attributes;
+ CajaFileAttributes non_delegated_attributes;
+
+ GList *non_ready_files;
+
+ gboolean initializing;
+} DesktopCallback;
+
+typedef struct
+{
+ CajaDesktopDirectoryFile *desktop_file;
+
+ CajaFileAttributes delegated_attributes;
+ CajaFileAttributes non_delegated_attributes;
+} DesktopMonitor;
+
+
+static void caja_desktop_directory_file_init (gpointer object,
+ gpointer klass);
+static void caja_desktop_directory_file_class_init (gpointer klass);
+
+EEL_CLASS_BOILERPLATE (CajaDesktopDirectoryFile,
+ caja_desktop_directory_file,
+ CAJA_TYPE_FILE)
+
+static guint
+desktop_callback_hash (gconstpointer desktop_callback_as_pointer)
+{
+ const DesktopCallback *desktop_callback;
+
+ desktop_callback = desktop_callback_as_pointer;
+ return GPOINTER_TO_UINT (desktop_callback->callback)
+ ^ GPOINTER_TO_UINT (desktop_callback->callback_data);
+}
+
+static gboolean
+desktop_callback_equal (gconstpointer desktop_callback_as_pointer,
+ gconstpointer desktop_callback_as_pointer_2)
+{
+ const DesktopCallback *desktop_callback, *desktop_callback_2;
+
+ desktop_callback = desktop_callback_as_pointer;
+ desktop_callback_2 = desktop_callback_as_pointer_2;
+
+ return desktop_callback->callback == desktop_callback_2->callback
+ && desktop_callback->callback_data == desktop_callback_2->callback_data;
+}
+
+
+static void
+real_file_changed_callback (CajaFile *real_file,
+ gpointer callback_data)
+{
+ CajaDesktopDirectoryFile *desktop_file;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (callback_data);
+ caja_file_changed (CAJA_FILE (desktop_file));
+}
+
+static CajaFileAttributes
+get_delegated_attributes_mask (void)
+{
+ return CAJA_FILE_ATTRIBUTE_DEEP_COUNTS |
+ CAJA_FILE_ATTRIBUTE_DIRECTORY_ITEM_COUNT |
+ CAJA_FILE_ATTRIBUTE_DIRECTORY_ITEM_MIME_TYPES;
+}
+
+static void
+partition_attributes (CajaFileAttributes attributes,
+ CajaFileAttributes *delegated_attributes,
+ CajaFileAttributes *non_delegated_attributes)
+{
+ CajaFileAttributes mask;
+
+ mask = get_delegated_attributes_mask ();
+
+ *delegated_attributes = attributes & mask;
+ *non_delegated_attributes = attributes & ~mask;
+}
+
+static void
+desktop_directory_file_monitor_add (CajaFile *file,
+ gconstpointer client,
+ CajaFileAttributes attributes)
+{
+ CajaDesktopDirectoryFile *desktop_file;
+ DesktopMonitor *monitor;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (file);
+
+ /* Map the client to a unique value so this doesn't interfere
+ * with direct monitoring of the file by the same client.
+ */
+ monitor = g_hash_table_lookup (desktop_file->details->monitors, client);
+ if (monitor != NULL)
+ {
+ g_assert (monitor->desktop_file == desktop_file);
+ }
+ else
+ {
+ monitor = g_new0 (DesktopMonitor, 1);
+ monitor->desktop_file = desktop_file;
+ g_hash_table_insert (desktop_file->details->monitors,
+ (gpointer) client, monitor);
+ }
+
+ partition_attributes (attributes,
+ &monitor->delegated_attributes,
+ &monitor->non_delegated_attributes);
+
+ /* Pawn off partioned attributes to real dir file */
+ caja_file_monitor_add (desktop_file->details->real_dir_file,
+ monitor, monitor->delegated_attributes);
+
+ /* Do the rest ourself */
+ caja_directory_monitor_add_internal
+ (file->details->directory, file,
+ client, TRUE, TRUE,
+ monitor->non_delegated_attributes,
+ NULL, NULL);
+}
+
+static void
+desktop_directory_file_monitor_remove (CajaFile *file,
+ gconstpointer client)
+{
+ CajaDesktopDirectoryFile *desktop_file;
+ DesktopMonitor *monitor;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (file);
+
+ /* Map the client to the value used by the earlier add call. */
+ monitor = g_hash_table_lookup (desktop_file->details->monitors, client);
+ if (monitor == NULL)
+ {
+ return;
+ }
+
+ /* Call through to the real file remove calls. */
+ g_hash_table_remove (desktop_file->details->monitors, client);
+
+ /* Remove the locally handled parts */
+ caja_directory_monitor_remove_internal
+ (file->details->directory, file, client);
+}
+
+static void
+desktop_callback_destroy (DesktopCallback *desktop_callback)
+{
+ g_assert (desktop_callback != NULL);
+ g_assert (CAJA_IS_DESKTOP_DIRECTORY_FILE (desktop_callback->desktop_file));
+
+ caja_file_unref (CAJA_FILE (desktop_callback->desktop_file));
+ g_list_free (desktop_callback->non_ready_files);
+ g_free (desktop_callback);
+}
+
+static void
+desktop_callback_check_done (DesktopCallback *desktop_callback)
+{
+ /* Check if we are ready. */
+ if (desktop_callback->initializing ||
+ desktop_callback->non_ready_files != NULL)
+ {
+ return;
+ }
+
+ /* Remove from the hash table before sending it. */
+ g_hash_table_remove (desktop_callback->desktop_file->details->callbacks,
+ desktop_callback);
+
+ /* We are ready, so do the real callback. */
+ (* desktop_callback->callback) (CAJA_FILE (desktop_callback->desktop_file),
+ desktop_callback->callback_data);
+
+ /* And we are done. */
+ desktop_callback_destroy (desktop_callback);
+}
+
+static void
+desktop_callback_remove_file (DesktopCallback *desktop_callback,
+ CajaFile *file)
+{
+ desktop_callback->non_ready_files = g_list_remove
+ (desktop_callback->non_ready_files, file);
+ desktop_callback_check_done (desktop_callback);
+}
+
+static void
+ready_callback (CajaFile *file,
+ gpointer callback_data)
+{
+ DesktopCallback *desktop_callback;
+
+ g_assert (CAJA_IS_FILE (file));
+ g_assert (callback_data != NULL);
+
+ desktop_callback = callback_data;
+ g_assert (g_list_find (desktop_callback->non_ready_files, file) != NULL);
+
+ desktop_callback_remove_file (desktop_callback, file);
+}
+
+static void
+desktop_directory_file_call_when_ready (CajaFile *file,
+ CajaFileAttributes attributes,
+ CajaFileCallback callback,
+ gpointer callback_data)
+
+{
+ CajaDesktopDirectoryFile *desktop_file;
+ DesktopCallback search_key, *desktop_callback;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (file);
+
+ /* Check to be sure we aren't overwriting. */
+ search_key.callback = callback;
+ search_key.callback_data = callback_data;
+ if (g_hash_table_lookup (desktop_file->details->callbacks, &search_key) != NULL)
+ {
+ g_warning ("tried to add a new callback while an old one was pending");
+ return;
+ }
+
+ /* Create a desktop_callback record. */
+ desktop_callback = g_new0 (DesktopCallback, 1);
+ caja_file_ref (file);
+ desktop_callback->desktop_file = desktop_file;
+ desktop_callback->callback = callback;
+ desktop_callback->callback_data = callback_data;
+ desktop_callback->initializing = TRUE;
+
+ partition_attributes (attributes,
+ &desktop_callback->delegated_attributes,
+ &desktop_callback->non_delegated_attributes);
+
+ desktop_callback->non_ready_files = g_list_prepend
+ (desktop_callback->non_ready_files, file);
+ desktop_callback->non_ready_files = g_list_prepend
+ (desktop_callback->non_ready_files, desktop_file->details->real_dir_file);
+
+ /* Put it in the hash table. */
+ g_hash_table_insert (desktop_file->details->callbacks,
+ desktop_callback, desktop_callback);
+
+ /* Now connect to each file's call_when_ready. */
+ caja_directory_call_when_ready_internal
+ (file->details->directory, file,
+ desktop_callback->non_delegated_attributes,
+ FALSE, NULL, ready_callback, desktop_callback);
+ caja_file_call_when_ready
+ (desktop_file->details->real_dir_file,
+ desktop_callback->delegated_attributes,
+ ready_callback, desktop_callback);
+
+ desktop_callback->initializing = FALSE;
+
+ /* Check if any files became read while we were connecting up
+ * the call_when_ready callbacks (also handles the pathological
+ * case where there are no files at all).
+ */
+ desktop_callback_check_done (desktop_callback);
+
+}
+
+static void
+desktop_directory_file_cancel_call_when_ready (CajaFile *file,
+ CajaFileCallback callback,
+ gpointer callback_data)
+{
+ CajaDesktopDirectoryFile *desktop_file;
+ DesktopCallback search_key, *desktop_callback;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (file);
+
+ /* Find the entry in the table. */
+ search_key.callback = callback;
+ search_key.callback_data = callback_data;
+ desktop_callback = g_hash_table_lookup (desktop_file->details->callbacks, &search_key);
+ if (desktop_callback == NULL)
+ {
+ return;
+ }
+
+ /* Remove from the hash table before working with it. */
+ g_hash_table_remove (desktop_callback->desktop_file->details->callbacks, desktop_callback);
+
+ /* Tell the real directory to cancel the call. */
+ caja_directory_cancel_callback_internal
+ (file->details->directory, file,
+ NULL, ready_callback, desktop_callback);
+
+ caja_file_cancel_call_when_ready
+ (desktop_file->details->real_dir_file,
+ ready_callback, desktop_callback);
+
+ desktop_callback_destroy (desktop_callback);
+}
+
+static gboolean
+real_check_if_ready (CajaFile *file,
+ CajaFileAttributes attributes)
+{
+ return caja_directory_check_if_ready_internal
+ (file->details->directory, file,
+ attributes);
+}
+
+static gboolean
+desktop_directory_file_check_if_ready (CajaFile *file,
+ CajaFileAttributes attributes)
+{
+ CajaFileAttributes delegated_attributes, non_delegated_attributes;
+ CajaDesktopDirectoryFile *desktop_file;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (file);
+
+ partition_attributes (attributes,
+ &delegated_attributes,
+ &non_delegated_attributes);
+
+ return real_check_if_ready (file, non_delegated_attributes) &&
+ caja_file_check_if_ready (desktop_file->details->real_dir_file,
+ delegated_attributes);
+}
+
+static gboolean
+desktop_directory_file_get_item_count (CajaFile *file,
+ guint *count,
+ gboolean *count_unreadable)
+{
+ CajaDesktopDirectoryFile *desktop_file;
+ gboolean got_count;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (file);
+
+ got_count = caja_file_get_directory_item_count (desktop_file->details->real_dir_file,
+ count,
+ count_unreadable);
+
+ if (count)
+ {
+ *count += g_list_length (file->details->directory->details->file_list);
+ }
+
+ return got_count;
+}
+
+static CajaRequestStatus
+desktop_directory_file_get_deep_counts (CajaFile *file,
+ guint *directory_count,
+ guint *file_count,
+ guint *unreadable_directory_count,
+ goffset *total_size)
+{
+ CajaDesktopDirectoryFile *desktop_file;
+ CajaRequestStatus status;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (file);
+
+ status = caja_file_get_deep_counts (desktop_file->details->real_dir_file,
+ directory_count,
+ file_count,
+ unreadable_directory_count,
+ total_size,
+ TRUE);
+
+ if (file_count)
+ {
+ *file_count += g_list_length (file->details->directory->details->file_list);
+ }
+
+ return status;
+}
+
+static gboolean
+desktop_directory_file_get_date (CajaFile *file,
+ CajaDateType date_type,
+ time_t *date)
+{
+ CajaDesktopDirectoryFile *desktop_file;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (file);
+
+ return caja_file_get_date (desktop_file->details->real_dir_file,
+ date_type,
+ date);
+}
+
+static char *
+desktop_directory_file_get_where_string (CajaFile *file)
+{
+ return g_strdup (_("on the desktop"));
+}
+
+
+static void
+monitor_destroy (gpointer data)
+{
+ DesktopMonitor *monitor = data;
+
+ caja_file_monitor_remove
+ (CAJA_FILE (monitor->desktop_file->details->real_dir_file), monitor);
+ g_free (monitor);
+}
+
+static char *
+get_metadata_mateconf_path (const char *name,
+ const char *key)
+{
+ char *res, *escaped_name;
+
+ escaped_name = mateconf_escape_key (name, -1);
+ res = g_build_filename (CAJA_DESKTOP_METADATA_MATECONF_PATH, escaped_name, key, NULL);
+ g_free (escaped_name);
+
+ return res;
+}
+
+void
+caja_desktop_set_metadata_string (CajaFile *file,
+ const char *name,
+ const char *key,
+ const char *string)
+{
+ MateConfClient *client;
+ char *mateconf_key;
+
+ client = mateconf_client_get_default ();
+ mateconf_key = get_metadata_mateconf_path (name, key);
+
+ if (string)
+ {
+ mateconf_client_set_string (client, mateconf_key, string, NULL);
+ }
+ else
+ {
+ mateconf_client_unset (client, mateconf_key, NULL);
+ }
+
+ g_free (mateconf_key);
+ g_object_unref (client);
+
+ if (caja_desktop_update_metadata_from_mateconf (file, name))
+ {
+ caja_file_changed (file);
+ }
+}
+
+void
+caja_desktop_set_metadata_stringv (CajaFile *file,
+ const char *name,
+ const char *key,
+ char **stringv)
+{
+ MateConfClient *client;
+ char *mateconf_key;
+ GSList *list;
+ int i;
+
+ client = mateconf_client_get_default ();
+ mateconf_key = get_metadata_mateconf_path (name, key);
+
+ list = NULL;
+ for (i = 0; stringv[i] != NULL; i++)
+ {
+ list = g_slist_prepend (list, stringv[i]);
+ }
+ list = g_slist_reverse (list);
+
+ mateconf_client_set_list (client, mateconf_key,
+ MATECONF_VALUE_STRING,
+ list, NULL);
+
+ g_slist_free (list);
+ g_free (mateconf_key);
+ g_object_unref (client);
+
+ if (caja_desktop_update_metadata_from_mateconf (file, name))
+ {
+ caja_file_changed (file);
+ }
+}
+
+gboolean
+caja_desktop_update_metadata_from_mateconf (CajaFile *file,
+ const char *name)
+{
+ MateConfClient *client;
+ GSList *entries, *l;
+ char *dir;
+ const char *key;
+ MateConfEntry *entry;
+ MateConfValue *value;
+ GFileInfo *info;
+ gboolean changed;
+ char *gio_key;
+ GSList *value_list;
+ char **strv;
+ int i;
+
+ client = mateconf_client_get_default ();
+
+ dir = get_metadata_mateconf_path (name, NULL);
+ entries = mateconf_client_all_entries (client, dir, NULL);
+ g_free (dir);
+
+ info = g_file_info_new ();
+
+ for (l = entries; l != NULL; l = l->next)
+ {
+ entry = l->data;
+
+ key = mateconf_entry_get_key (entry);
+ value = mateconf_entry_get_value (entry);
+
+ if (value == NULL)
+ {
+ continue;
+ }
+ key = strrchr (key, '/') + 1;
+
+ gio_key = g_strconcat ("metadata::", key, NULL);
+ if (value->type == MATECONF_VALUE_STRING)
+ {
+ g_file_info_set_attribute_string (info, gio_key,
+ mateconf_value_get_string (value));
+ }
+ else if (value->type == MATECONF_VALUE_LIST &&
+ mateconf_value_get_list_type (value) == MATECONF_VALUE_STRING)
+ {
+ value_list = mateconf_value_get_list (value);
+ strv = g_new (char *, g_slist_length (value_list) + 1);
+ for (i = 0; value_list != NULL; i++, value_list = value_list->next)
+ {
+ strv[i] = l->data;
+ }
+ strv[i] = NULL;
+ g_file_info_set_attribute_stringv (info, gio_key, strv);
+ g_free (strv);
+ }
+
+ g_free (gio_key);
+
+ mateconf_entry_unref (entry);
+ }
+ g_slist_free (entries);
+
+ changed = caja_file_update_metadata_from_info (file, info);
+
+ g_object_unref (info);
+ g_object_unref (client);
+
+ return changed;
+}
+
+static void
+caja_desktop_directory_file_set_metadata (CajaFile *file,
+ const char *key,
+ const char *value)
+{
+ caja_desktop_set_metadata_string (file, "directory", key, value);
+}
+
+static void
+caja_desktop_directory_file_set_metadata_as_list (CajaFile *file,
+ const char *key,
+ char **value)
+{
+ caja_desktop_set_metadata_stringv (file, "directory", key, value);
+}
+
+static void
+caja_desktop_directory_file_init (gpointer object, gpointer klass)
+{
+ CajaDesktopDirectoryFile *desktop_file;
+ CajaDesktopDirectory *desktop_directory;
+ CajaDirectory *real_dir;
+ CajaFile *real_dir_file;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (object);
+
+ desktop_directory = CAJA_DESKTOP_DIRECTORY (caja_directory_get_by_uri (EEL_DESKTOP_URI));
+
+ desktop_file->details = g_new0 (CajaDesktopDirectoryFileDetails, 1);
+ desktop_file->details->desktop_directory = desktop_directory;
+
+ desktop_file->details->callbacks = g_hash_table_new
+ (desktop_callback_hash, desktop_callback_equal);
+ desktop_file->details->monitors = g_hash_table_new_full (NULL, NULL,
+ NULL, monitor_destroy);
+
+ real_dir = caja_desktop_directory_get_real_directory (desktop_directory);
+ real_dir_file = caja_directory_get_corresponding_file (real_dir);
+ caja_directory_unref (real_dir);
+
+ desktop_file->details->real_dir_file = real_dir_file;
+
+ caja_desktop_update_metadata_from_mateconf (CAJA_FILE (desktop_file), "directory");
+
+ g_signal_connect_object (real_dir_file, "changed",
+ G_CALLBACK (real_file_changed_callback), desktop_file, 0);
+}
+
+
+static void
+desktop_callback_remove_file_cover (gpointer key,
+ gpointer value,
+ gpointer callback_data)
+{
+ desktop_callback_remove_file
+ (value, CAJA_FILE (callback_data));
+}
+
+
+static void
+desktop_finalize (GObject *object)
+{
+ CajaDesktopDirectoryFile *desktop_file;
+ CajaDesktopDirectory *desktop_directory;
+
+ desktop_file = CAJA_DESKTOP_DIRECTORY_FILE (object);
+ desktop_directory = desktop_file->details->desktop_directory;
+
+ /* Todo: ghash now safe? */
+ eel_g_hash_table_safe_for_each
+ (desktop_file->details->callbacks,
+ desktop_callback_remove_file_cover,
+ desktop_file->details->real_dir_file);
+
+ if (g_hash_table_size (desktop_file->details->callbacks) != 0)
+ {
+ g_warning ("call_when_ready still pending when desktop virtual file is destroyed");
+ }
+
+ g_hash_table_destroy (desktop_file->details->callbacks);
+ g_hash_table_destroy (desktop_file->details->monitors);
+
+ caja_file_unref (desktop_file->details->real_dir_file);
+
+ g_free (desktop_file->details);
+
+ caja_directory_unref (CAJA_DIRECTORY (desktop_directory));
+
+ EEL_CALL_PARENT (G_OBJECT_CLASS, finalize, (object));
+}
+
+static void
+caja_desktop_directory_file_class_init (gpointer klass)
+{
+ GObjectClass *object_class;
+ CajaFileClass *file_class;
+
+ object_class = G_OBJECT_CLASS (klass);
+ file_class = CAJA_FILE_CLASS (klass);
+
+ object_class->finalize = desktop_finalize;
+
+ file_class->default_file_type = G_FILE_TYPE_DIRECTORY;
+
+ file_class->monitor_add = desktop_directory_file_monitor_add;
+ file_class->monitor_remove = desktop_directory_file_monitor_remove;
+ file_class->call_when_ready = desktop_directory_file_call_when_ready;
+ file_class->cancel_call_when_ready = desktop_directory_file_cancel_call_when_ready;
+ file_class->check_if_ready = desktop_directory_file_check_if_ready;
+ file_class->get_item_count = desktop_directory_file_get_item_count;
+ file_class->get_deep_counts = desktop_directory_file_get_deep_counts;
+ file_class->get_date = desktop_directory_file_get_date;
+ file_class->get_where_string = desktop_directory_file_get_where_string;
+ file_class->set_metadata = caja_desktop_directory_file_set_metadata;
+ file_class->set_metadata_as_list = caja_desktop_directory_file_set_metadata_as_list;
+}