1238 lines
32 KiB
C
1238 lines
32 KiB
C
/*
|
|
* moofolder.c
|
|
*
|
|
* Copyright (C) 2004-2010 by Yevgen Muntyan <emuntyan@users.sourceforge.net>
|
|
*
|
|
* This file is part of medit. medit 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.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#define MOO_FILE_VIEW_COMPILATION
|
|
#include "moofileview/moofilesystem.h"
|
|
#include "moofileview/moofolder-private.h"
|
|
#include "mooutils/mooutils-fs.h"
|
|
#include "mooutils/mooutils-misc.h"
|
|
#include "marshals.h"
|
|
#include <glib/gstdio.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
#include <errno.h>
|
|
#include <time.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#define NORMAL_PRIORITY G_PRIORITY_DEFAULT_IDLE
|
|
#define NORMAL_TIMEOUT 0.04
|
|
#define BACKGROUND_PRIORITY G_PRIORITY_LOW
|
|
#define BACKGROUND_TIMEOUT 0.001
|
|
|
|
#define TIMER_CLEAR(timer) \
|
|
G_STMT_START { \
|
|
g_timer_start (timer); \
|
|
g_timer_stop (timer); \
|
|
} G_STMT_END
|
|
|
|
#if 0
|
|
#define PRINT_TIMES g_print
|
|
#else
|
|
static void G_GNUC_PRINTF(1,2) PRINT_TIMES (G_GNUC_UNUSED const char *format, ...)
|
|
{
|
|
}
|
|
#endif
|
|
|
|
static void moo_folder_dispose (GObject *object);
|
|
|
|
static void moo_folder_deleted (MooFolderImpl *impl);
|
|
static gboolean moo_folder_do_reload (MooFolderImpl *impl);
|
|
|
|
static void stop_populate (MooFolderImpl *impl);
|
|
|
|
static void files_list_free (GSList **list);
|
|
|
|
static gboolean get_icons_a_bit (MooFolderImpl *impl);
|
|
static gboolean get_stat_a_bit (MooFolderImpl *impl);
|
|
static double get_names (MooFolderImpl *impl);
|
|
|
|
#define FILE_PATH(folder,file) g_build_filename (folder->impl->path, file->name, NULL)
|
|
|
|
static void start_monitor (MooFolderImpl *impl);
|
|
static void stop_monitor (MooFolderImpl *impl);
|
|
|
|
static GSList *hash_table_to_file_list (GHashTable *files);
|
|
static void diff_hash_tables (GHashTable *table1,
|
|
GHashTable *table2,
|
|
GSList **only_1,
|
|
GSList **only_2);
|
|
|
|
|
|
/* MOO_TYPE_FOLDER */
|
|
G_DEFINE_TYPE (MooFolder, _moo_folder, G_TYPE_OBJECT)
|
|
|
|
enum {
|
|
DELETED,
|
|
FILES_ADDED,
|
|
FILES_REMOVED,
|
|
FILES_CHANGED,
|
|
NUM_SIGNALS
|
|
};
|
|
|
|
static guint signals[NUM_SIGNALS];
|
|
|
|
static void
|
|
_moo_folder_class_init (MooFolderClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
gobject_class->dispose = moo_folder_dispose;
|
|
|
|
signals[DELETED] =
|
|
g_signal_new ("deleted",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (MooFolderClass, deleted),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__VOID,
|
|
G_TYPE_NONE, 0);
|
|
|
|
signals[FILES_ADDED] =
|
|
g_signal_new ("files-added",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (MooFolderClass, files_added),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__POINTER,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_POINTER);
|
|
|
|
signals[FILES_REMOVED] =
|
|
g_signal_new ("files-removed",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (MooFolderClass, files_removed),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__POINTER,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_POINTER);
|
|
|
|
signals[FILES_CHANGED] =
|
|
g_signal_new ("files-changed",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (MooFolderClass, files_changed),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__POINTER,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_POINTER);
|
|
}
|
|
|
|
|
|
static void
|
|
_moo_folder_init (G_GNUC_UNUSED MooFolder *folder)
|
|
{
|
|
}
|
|
|
|
|
|
static MooFolderImpl *
|
|
moo_folder_impl_new (MooFileSystem *fs,
|
|
const char *path,
|
|
GDir *dir)
|
|
{
|
|
MooFolderImpl *impl;
|
|
|
|
impl = g_new0 (MooFolderImpl, 1);
|
|
impl->deleted = FALSE;
|
|
impl->done = 0;
|
|
impl->wanted = 0;
|
|
impl->fs = fs;
|
|
impl->dir = dir;
|
|
impl->files_copy = NULL;
|
|
impl->path = g_strdup (path);
|
|
impl->populate_func = NULL;
|
|
impl->populate_idle_id = 0;
|
|
impl->populate_timeout = BACKGROUND_TIMEOUT;
|
|
impl->timer = g_timer_new ();
|
|
g_timer_stop (impl->timer);
|
|
|
|
impl->files = g_hash_table_new_full (g_str_hash, g_str_equal, g_free,
|
|
(GDestroyNotify) _moo_file_unref);
|
|
|
|
return impl;
|
|
}
|
|
|
|
|
|
static void
|
|
folder_shutdown (MooFolderImpl *impl)
|
|
{
|
|
stop_populate (impl);
|
|
stop_monitor (impl);
|
|
files_list_free (&impl->files_copy);
|
|
|
|
if (impl->reload_idle)
|
|
g_source_remove (impl->reload_idle);
|
|
impl->reload_idle = 0;
|
|
|
|
if (impl->files)
|
|
g_hash_table_destroy (impl->files);
|
|
impl->files = NULL;
|
|
|
|
if (impl->dir)
|
|
g_dir_close (impl->dir);
|
|
impl->dir = NULL;
|
|
|
|
if (impl->populate_idle_id)
|
|
g_source_remove (impl->populate_idle_id);
|
|
impl->populate_idle_id = 0;
|
|
|
|
if (impl->timer)
|
|
g_timer_destroy (impl->timer);
|
|
impl->timer = NULL;
|
|
}
|
|
|
|
|
|
void
|
|
_moo_folder_impl_free (MooFolderImpl *impl)
|
|
{
|
|
g_return_if_fail (impl != NULL);
|
|
folder_shutdown (impl);
|
|
g_free (impl->path);
|
|
g_free (impl);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_folder_dispose (GObject *object)
|
|
{
|
|
MooFolder *folder = MOO_FOLDER (object);
|
|
|
|
if (folder->impl)
|
|
{
|
|
MooFileSystem *fs = folder->impl->fs;
|
|
folder->impl->proxy = NULL;
|
|
_moo_file_system_folder_finalized (fs, folder);
|
|
folder->impl = NULL;
|
|
g_object_unref (fs);
|
|
}
|
|
|
|
G_OBJECT_CLASS (_moo_folder_parent_class)->finalize (object);
|
|
}
|
|
|
|
|
|
static void
|
|
add_file_size (G_GNUC_UNUSED const char *filename,
|
|
MooFile *file,
|
|
gsize *mem)
|
|
{
|
|
*mem += sizeof *file;
|
|
#define STRING_SIZE(s) ((s) ? (strlen (s) + 1) : 0)
|
|
*mem += STRING_SIZE (file->name);
|
|
*mem += STRING_SIZE (file->link_target);
|
|
*mem += STRING_SIZE (file->display_name);
|
|
*mem += STRING_SIZE (file->case_display_name);
|
|
*mem += _moo_collation_key_size (file->collation_key);
|
|
#undef STRING_SIZE
|
|
}
|
|
|
|
gsize
|
|
_moo_folder_mem_usage (MooFolder *folder)
|
|
{
|
|
gsize mem = 0;
|
|
mem += sizeof (MooFolderImpl);
|
|
g_hash_table_foreach (folder->impl->files, (GHFunc) add_file_size, &mem);
|
|
return mem;
|
|
}
|
|
|
|
|
|
MooFolder *
|
|
_moo_folder_new_with_impl (MooFolderImpl *impl)
|
|
{
|
|
MooFolder *folder = MOO_FOLDER (g_object_new (MOO_TYPE_FOLDER, (const char*) NULL));
|
|
folder->impl = impl;
|
|
impl->proxy = folder;
|
|
g_object_ref (impl->fs);
|
|
return folder;
|
|
}
|
|
|
|
|
|
MooFolder *
|
|
_moo_folder_new (MooFileSystem *fs,
|
|
const char *path,
|
|
MooFileFlags wanted,
|
|
GError **error)
|
|
{
|
|
GDir *dir;
|
|
MooFolderImpl *impl;
|
|
MooFolder *folder;
|
|
GError *file_error = NULL;
|
|
|
|
g_return_val_if_fail (MOO_IS_FILE_SYSTEM (fs), NULL);
|
|
g_return_val_if_fail (path != NULL, NULL);
|
|
|
|
dir = g_dir_open (path, 0, &file_error);
|
|
|
|
if (!dir)
|
|
{
|
|
if (file_error->domain != G_FILE_ERROR)
|
|
{
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
MOO_FILE_ERROR_FAILED,
|
|
"%s", file_error->message);
|
|
}
|
|
else switch (file_error->code)
|
|
{
|
|
case G_FILE_ERROR_NOENT:
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
MOO_FILE_ERROR_NONEXISTENT,
|
|
"%s", file_error->message);
|
|
break;
|
|
case G_FILE_ERROR_NOTDIR:
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
MOO_FILE_ERROR_NOT_FOLDER,
|
|
"%s", file_error->message);
|
|
break;
|
|
case G_FILE_ERROR_NAMETOOLONG:
|
|
case G_FILE_ERROR_LOOP:
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
MOO_FILE_ERROR_BAD_FILENAME,
|
|
"%s", file_error->message);
|
|
break;
|
|
default:
|
|
g_set_error (error, MOO_FILE_ERROR,
|
|
MOO_FILE_ERROR_FAILED,
|
|
"%s", file_error->message);
|
|
break;
|
|
}
|
|
|
|
g_error_free (file_error);
|
|
return NULL;
|
|
}
|
|
|
|
impl = moo_folder_impl_new (fs, path, dir);
|
|
|
|
folder = _moo_folder_new_with_impl (impl);
|
|
get_names (folder->impl);
|
|
_moo_folder_set_wanted (folder, wanted, TRUE);
|
|
|
|
return folder;
|
|
}
|
|
|
|
|
|
static void
|
|
moo_folder_deleted (MooFolderImpl *impl)
|
|
{
|
|
folder_shutdown (impl);
|
|
impl->deleted = TRUE;
|
|
_moo_file_system_folder_deleted (impl->fs, impl);
|
|
}
|
|
|
|
|
|
void
|
|
_moo_folder_set_wanted (MooFolder *folder,
|
|
MooFileFlags wanted,
|
|
gboolean bit_now)
|
|
{
|
|
Stage wanted_stage = STAGE_NAMES;
|
|
|
|
g_return_if_fail (MOO_IS_FOLDER (folder));
|
|
g_return_if_fail (!folder->impl->deleted);
|
|
|
|
if (wanted & MOO_FILE_HAS_ICON)
|
|
wanted_stage = STAGE_MIME_TYPE;
|
|
else if (wanted & MOO_FILE_HAS_MIME_TYPE)
|
|
wanted_stage = STAGE_MIME_TYPE;
|
|
else if (wanted & MOO_FILE_HAS_STAT)
|
|
wanted_stage = STAGE_STAT;
|
|
|
|
if (wanted_stage <= folder->impl->done)
|
|
return;
|
|
|
|
if (folder->impl->wanted > folder->impl->done)
|
|
{
|
|
g_assert (folder->impl->populate_idle_id != 0);
|
|
folder->impl->wanted = MAX (folder->impl->wanted, wanted_stage);
|
|
return;
|
|
}
|
|
|
|
folder->impl->wanted = wanted_stage;
|
|
|
|
if (folder->impl->wanted_bg != 0)
|
|
{
|
|
g_assert (folder->impl->populate_idle_id != 0);
|
|
g_assert (folder->impl->populate_func != NULL);
|
|
g_assert (folder->impl->populate_priority != 0);
|
|
g_source_remove (folder->impl->populate_idle_id);
|
|
}
|
|
else
|
|
{
|
|
g_assert (folder->impl->populate_idle_id == 0);
|
|
|
|
switch (folder->impl->done)
|
|
{
|
|
case STAGE_NAMES:
|
|
g_assert (folder->impl->dir == NULL);
|
|
folder->impl->populate_func = (GSourceFunc) get_stat_a_bit;
|
|
break;
|
|
case STAGE_STAT:
|
|
g_assert (folder->impl->dir == NULL);
|
|
folder->impl->populate_func = (GSourceFunc) get_icons_a_bit;
|
|
break;
|
|
default:
|
|
g_assert_not_reached ();
|
|
}
|
|
}
|
|
|
|
folder->impl->wanted_bg = STAGE_MIME_TYPE;
|
|
folder->impl->populate_timeout = NORMAL_TIMEOUT;
|
|
folder->impl->populate_priority = NORMAL_PRIORITY;
|
|
|
|
TIMER_CLEAR (folder->impl->timer);
|
|
|
|
g_object_ref (folder);
|
|
|
|
if (!bit_now ||
|
|
folder->impl->populate_func (folder->impl))
|
|
{
|
|
folder->impl->populate_idle_id =
|
|
gdk_threads_add_timeout_full (folder->impl->populate_priority,
|
|
folder->impl->populate_timeout,
|
|
folder->impl->populate_func,
|
|
folder->impl, NULL);
|
|
}
|
|
|
|
g_object_unref (folder);
|
|
}
|
|
|
|
|
|
static void
|
|
stop_populate (MooFolderImpl *impl)
|
|
{
|
|
if (impl->populate_idle_id)
|
|
g_source_remove (impl->populate_idle_id);
|
|
impl->populate_idle_id = 0;
|
|
impl->populate_func = NULL;
|
|
if (impl->dir)
|
|
g_dir_close (impl->dir);
|
|
impl->dir = NULL;
|
|
files_list_free (&impl->files_copy);
|
|
}
|
|
|
|
|
|
static void
|
|
folder_emit_deleted (MooFolderImpl *impl)
|
|
{
|
|
MooFolder *folder;
|
|
|
|
/* moo_folder_deleted may call impl_free() */
|
|
folder = impl->proxy;
|
|
moo_folder_deleted (impl);
|
|
|
|
if (folder)
|
|
g_signal_emit (folder, signals[DELETED], 0);
|
|
}
|
|
|
|
|
|
static void
|
|
folder_emit_files (MooFolderImpl *impl,
|
|
guint sig,
|
|
GSList *files)
|
|
{
|
|
if (files && impl->proxy)
|
|
g_signal_emit (impl->proxy, signals[sig], 0, files);
|
|
}
|
|
|
|
|
|
GSList *
|
|
_moo_folder_list_files (MooFolder *folder)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
g_return_val_if_fail (!folder->impl->deleted, NULL);
|
|
return hash_table_to_file_list (folder->impl->files);
|
|
}
|
|
|
|
|
|
static double
|
|
get_names (MooFolderImpl *impl)
|
|
{
|
|
GTimer *timer;
|
|
GSList *added = NULL;
|
|
const char *name;
|
|
MooFile *file;
|
|
double elapsed;
|
|
|
|
g_assert (impl->path != NULL);
|
|
g_assert (impl->dir != NULL);
|
|
g_assert (g_hash_table_size (impl->files) == 0);
|
|
|
|
timer = g_timer_new ();
|
|
|
|
file = _moo_file_new (impl->path, "..");
|
|
file->flags = MOO_FILE_HAS_MIME_TYPE | MOO_FILE_HAS_ICON;
|
|
file->info = MOO_FILE_INFO_EXISTS | MOO_FILE_INFO_IS_DIR;
|
|
file->icon = _moo_file_get_icon_type (file, impl->path);
|
|
|
|
g_hash_table_insert (impl->files, g_strdup (".."), file);
|
|
added = g_slist_prepend (added, file);
|
|
|
|
for (name = g_dir_read_name (impl->dir);
|
|
name != NULL;
|
|
name = g_dir_read_name (impl->dir))
|
|
{
|
|
file = _moo_file_new (impl->path, name);
|
|
|
|
if (file)
|
|
{
|
|
file->icon = _moo_file_icon_blank ();
|
|
g_hash_table_insert (impl->files, g_strdup (name), file);
|
|
added = g_slist_prepend (added, file);
|
|
_moo_file_stat (file, impl->path);
|
|
}
|
|
else
|
|
{
|
|
g_critical ("_moo_file_new() failed for '%s'", name);
|
|
}
|
|
}
|
|
|
|
folder_emit_files (impl, FILES_ADDED, added);
|
|
g_slist_free (added);
|
|
|
|
elapsed = impl->debug.names_timer = g_timer_elapsed (timer, NULL);
|
|
g_timer_destroy (timer);
|
|
|
|
g_dir_close (impl->dir);
|
|
impl->dir = NULL;
|
|
|
|
impl->done = STAGE_NAMES;
|
|
|
|
PRINT_TIMES ("names folder %s: %f sec\n",
|
|
impl->path,
|
|
impl->debug.names_timer);
|
|
|
|
return elapsed;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
get_stat_a_bit (MooFolderImpl *impl)
|
|
{
|
|
gboolean done = FALSE;
|
|
double elapsed;
|
|
|
|
g_assert (impl->dir == NULL);
|
|
g_assert (impl->done == STAGE_NAMES);
|
|
g_assert (impl->path != NULL);
|
|
|
|
elapsed = g_timer_elapsed (impl->timer, NULL);
|
|
g_timer_continue (impl->timer);
|
|
|
|
if (!impl->files_copy)
|
|
impl->files_copy = hash_table_to_file_list (impl->files);
|
|
if (!impl->files_copy)
|
|
done = TRUE;
|
|
|
|
while (!done)
|
|
{
|
|
GSList *changed = impl->files_copy;
|
|
MooFile *file = changed->data;
|
|
impl->files_copy = g_slist_remove_link (impl->files_copy, impl->files_copy);
|
|
|
|
if (!(file->flags & MOO_FILE_HAS_STAT))
|
|
{
|
|
_moo_file_stat (file, impl->path);
|
|
folder_emit_files (impl, FILES_CHANGED, changed);
|
|
}
|
|
else
|
|
{
|
|
_moo_file_unref (file);
|
|
}
|
|
|
|
g_slist_free_1 (changed);
|
|
|
|
if (!impl->files_copy)
|
|
done = TRUE;
|
|
|
|
if (g_timer_elapsed (impl->timer, NULL) > impl->populate_timeout)
|
|
break;
|
|
}
|
|
|
|
elapsed = g_timer_elapsed (impl->timer, NULL) - elapsed;
|
|
impl->debug.stat_timer += elapsed;
|
|
impl->debug.stat_counter += 1;
|
|
g_timer_stop (impl->timer);
|
|
|
|
if (!done)
|
|
{
|
|
TIMER_CLEAR (impl->timer);
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
g_assert (impl->files_copy == NULL);
|
|
impl->populate_idle_id = 0;
|
|
|
|
PRINT_TIMES ("stat folder %s: %d iterations, %f sec\n",
|
|
impl->path,
|
|
impl->debug.stat_counter,
|
|
impl->debug.stat_timer);
|
|
|
|
impl->done = STAGE_STAT;
|
|
|
|
if (impl->wanted >= STAGE_MIME_TYPE || impl->wanted_bg >= STAGE_MIME_TYPE)
|
|
{
|
|
if (impl->wanted >= STAGE_MIME_TYPE)
|
|
{
|
|
impl->populate_priority = NORMAL_PRIORITY;
|
|
impl->populate_timeout = NORMAL_TIMEOUT;
|
|
}
|
|
else if (impl->wanted_bg >= STAGE_MIME_TYPE)
|
|
{
|
|
impl->populate_priority = BACKGROUND_PRIORITY;
|
|
impl->populate_timeout = BACKGROUND_TIMEOUT;
|
|
}
|
|
|
|
if (impl->populate_idle_id)
|
|
g_source_remove (impl->populate_idle_id);
|
|
impl->populate_idle_id = 0;
|
|
impl->populate_func = (GSourceFunc) get_icons_a_bit;
|
|
|
|
if (g_timer_elapsed (impl->timer, NULL) < impl->populate_timeout)
|
|
{
|
|
/* in this case we may block for as much as twice TIMEOUT, but usually
|
|
it allows stat and loading icons in one iteration */
|
|
TIMER_CLEAR (impl->timer);
|
|
if (impl->populate_func (impl))
|
|
impl->populate_idle_id =
|
|
gdk_threads_add_timeout_full (impl->populate_priority,
|
|
impl->populate_timeout,
|
|
impl->populate_func,
|
|
impl, NULL);
|
|
}
|
|
else
|
|
{
|
|
TIMER_CLEAR (impl->timer);
|
|
impl->populate_idle_id =
|
|
gdk_threads_add_timeout_full (impl->populate_priority,
|
|
impl->populate_timeout,
|
|
impl->populate_func,
|
|
impl, NULL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
impl->populate_func = NULL;
|
|
impl->populate_priority = 0;
|
|
impl->populate_timeout = 0;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
static gboolean
|
|
get_icons_a_bit (MooFolderImpl *impl)
|
|
{
|
|
gboolean done = FALSE;
|
|
double elapsed;
|
|
|
|
g_assert (impl->dir == NULL);
|
|
g_assert (impl->done == STAGE_STAT);
|
|
g_assert (impl->path != NULL);
|
|
|
|
elapsed = g_timer_elapsed (impl->timer, NULL);
|
|
g_timer_continue (impl->timer);
|
|
|
|
if (!impl->files_copy)
|
|
impl->files_copy = hash_table_to_file_list (impl->files);
|
|
if (!impl->files_copy)
|
|
done = TRUE;
|
|
|
|
while (!done)
|
|
{
|
|
GSList *changed = impl->files_copy;
|
|
MooFile *file = changed->data;
|
|
|
|
impl->files_copy = g_slist_remove_link (impl->files_copy, changed);
|
|
|
|
#ifndef __WIN32__
|
|
if (file->info & MOO_FILE_INFO_EXISTS &&
|
|
!(file->flags & MOO_FILE_HAS_MIME_TYPE))
|
|
{
|
|
char *path = g_build_filename (impl->path, file->name, NULL);
|
|
|
|
_moo_file_find_mime_type (file, path);
|
|
|
|
file->flags |= MOO_FILE_HAS_ICON;
|
|
file->icon = _moo_file_get_icon_type (file, impl->path);
|
|
folder_emit_files (impl, FILES_CHANGED, changed);
|
|
g_free (path);
|
|
}
|
|
#endif
|
|
|
|
_moo_file_free_statbuf (file);
|
|
_moo_file_unref (file);
|
|
g_slist_free (changed);
|
|
|
|
if (!impl->files_copy)
|
|
done = TRUE;
|
|
|
|
if (g_timer_elapsed (impl->timer, NULL) > impl->populate_timeout)
|
|
break;
|
|
}
|
|
|
|
elapsed = g_timer_elapsed (impl->timer, NULL) - elapsed;
|
|
impl->debug.icons_timer += elapsed;
|
|
impl->debug.icons_counter += 1;
|
|
TIMER_CLEAR (impl->timer);
|
|
|
|
if (done)
|
|
{
|
|
PRINT_TIMES ("icons folder %s: %d iterations, %f sec\n",
|
|
impl->path,
|
|
impl->debug.icons_counter,
|
|
impl->debug.icons_timer);
|
|
|
|
g_assert (impl->files_copy == NULL);
|
|
impl->populate_idle_id = 0;
|
|
impl->done = STAGE_MIME_TYPE;
|
|
impl->populate_func = NULL;
|
|
impl->populate_priority = 0;
|
|
impl->populate_timeout = 0;
|
|
|
|
start_monitor (impl);
|
|
}
|
|
|
|
return !done;
|
|
}
|
|
|
|
|
|
#if 0
|
|
MooFile *
|
|
_moo_folder_get_file (MooFolder *folder,
|
|
const char *basename)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
g_return_val_if_fail (!folder->impl->deleted, NULL);
|
|
g_return_val_if_fail (basename != NULL, NULL);
|
|
return g_hash_table_lookup (folder->impl->files, basename);
|
|
}
|
|
#endif
|
|
|
|
|
|
char *
|
|
_moo_folder_get_file_path (MooFolder *folder,
|
|
MooFile *file)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
return FILE_PATH (folder, file);
|
|
}
|
|
|
|
|
|
char *
|
|
_moo_folder_get_file_uri (MooFolder *folder,
|
|
MooFile *file)
|
|
{
|
|
char *path, *uri;
|
|
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
|
|
path = FILE_PATH (folder, file);
|
|
g_return_val_if_fail (path != NULL, NULL);
|
|
|
|
uri = g_filename_to_uri (path, NULL, NULL);
|
|
|
|
g_free (path);
|
|
return uri;
|
|
}
|
|
|
|
|
|
MooFolder *
|
|
_moo_folder_get_parent (MooFolder *folder,
|
|
MooFileFlags wanted)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
g_return_val_if_fail (!folder->impl->deleted, NULL);
|
|
return _moo_file_system_get_parent_folder (folder->impl->fs,
|
|
folder, wanted);
|
|
}
|
|
|
|
|
|
MooFileSystem *
|
|
_moo_folder_get_file_system (MooFolder *folder)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
return folder->impl->fs;
|
|
}
|
|
|
|
|
|
/*****************************************************************************/
|
|
/* Monitoring
|
|
*/
|
|
|
|
static void file_deleted (MooFolderImpl *folder,
|
|
const char *name);
|
|
static void file_changed (MooFolderImpl *folder,
|
|
const char *name);
|
|
static void file_created (MooFolderImpl *folder,
|
|
const char *name);
|
|
|
|
static void
|
|
fam_callback (MooFileWatch *watch,
|
|
MooFileEvent *event,
|
|
gpointer data)
|
|
{
|
|
MooFolderImpl *impl = data;
|
|
|
|
g_return_if_fail (watch == impl->fam);
|
|
g_return_if_fail (event->monitor_id == impl->fam_request);
|
|
|
|
switch (event->code)
|
|
{
|
|
case MOO_FILE_EVENT_CHANGED:
|
|
file_changed (impl, event->filename);
|
|
break;
|
|
case MOO_FILE_EVENT_CREATED:
|
|
file_created (impl, event->filename);
|
|
break;
|
|
case MOO_FILE_EVENT_DELETED:
|
|
file_deleted (impl, event->filename);
|
|
break;
|
|
case MOO_FILE_EVENT_ERROR:
|
|
stop_monitor (impl);
|
|
file_changed (impl, impl->path);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
start_monitor (MooFolderImpl *impl)
|
|
{
|
|
GError *error = NULL;
|
|
|
|
g_return_if_fail (!impl->deleted);
|
|
g_return_if_fail (impl->fam_request == 0);
|
|
impl->fam = _moo_file_system_get_file_watch (impl->fs);
|
|
g_return_if_fail (impl->fam != NULL);
|
|
|
|
impl->fam_request =
|
|
moo_file_watch_create_monitor (impl->fam, impl->path,
|
|
fam_callback, impl,
|
|
NULL, &error);
|
|
|
|
if (!impl->fam_request)
|
|
{
|
|
g_warning ("moo_file_watch_create_monitor failed for path '%s': %s",
|
|
impl->path, moo_error_message (error));
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
stop_monitor (MooFolderImpl *impl)
|
|
{
|
|
if (impl->fam_request)
|
|
{
|
|
moo_file_watch_cancel_monitor (impl->fam, impl->fam_request);
|
|
impl->fam = NULL;
|
|
impl->fam_request = 0;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
file_deleted (MooFolderImpl *impl,
|
|
const char *name)
|
|
{
|
|
MooFile *file;
|
|
GSList *list;
|
|
|
|
g_return_if_fail (!impl->deleted);
|
|
|
|
if (!strcmp (name, impl->path))
|
|
{
|
|
folder_emit_deleted (impl);
|
|
return;
|
|
}
|
|
|
|
file = g_hash_table_lookup (impl->files, name);
|
|
if (!file) return;
|
|
|
|
_moo_file_ref (file);
|
|
g_hash_table_remove (impl->files, name);
|
|
|
|
list = g_slist_append (NULL, file);
|
|
folder_emit_files (impl, FILES_REMOVED, list);
|
|
|
|
g_slist_free (list);
|
|
_moo_file_unref (file);
|
|
}
|
|
|
|
|
|
void
|
|
_moo_folder_reload (MooFolder *folder)
|
|
{
|
|
g_return_if_fail (MOO_IS_FOLDER (folder));
|
|
|
|
if (folder->impl->reload_idle)
|
|
g_source_remove (folder->impl->reload_idle);
|
|
folder->impl->reload_idle = 0;
|
|
|
|
moo_folder_do_reload (folder->impl);
|
|
}
|
|
|
|
|
|
static void
|
|
file_changed (MooFolderImpl *impl,
|
|
const char *name)
|
|
{
|
|
g_return_if_fail (!impl->deleted);
|
|
|
|
if (!strcmp (name, impl->path))
|
|
{
|
|
if (!impl->reload_idle)
|
|
impl->reload_idle = gdk_threads_add_idle ((GSourceFunc) moo_folder_do_reload, impl);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
file_created (MooFolderImpl *impl,
|
|
const char *name)
|
|
{
|
|
MooFile *file;
|
|
GSList *list;
|
|
|
|
g_return_if_fail (!impl->deleted);
|
|
|
|
file = _moo_file_new (impl->path, name);
|
|
g_return_if_fail (file != NULL);
|
|
|
|
file->icon = _moo_file_get_icon_type (file, impl->path);
|
|
_moo_file_stat (file, impl->path);
|
|
|
|
#ifndef __WIN32__
|
|
if (file->info & MOO_FILE_INFO_EXISTS &&
|
|
!(file->flags & MOO_FILE_HAS_MIME_TYPE))
|
|
{
|
|
char *path = g_build_filename (impl->path, file->name, NULL);
|
|
|
|
_moo_file_find_mime_type (file, path);
|
|
|
|
file->flags |= MOO_FILE_HAS_ICON;
|
|
file->icon = _moo_file_get_icon_type (file, impl->path);
|
|
g_free (path);
|
|
}
|
|
#endif
|
|
|
|
_moo_file_free_statbuf (file);
|
|
|
|
g_hash_table_insert (impl->files, g_strdup (name), file);
|
|
list = g_slist_append (NULL, file);
|
|
folder_emit_files (impl, FILES_ADDED, list);
|
|
|
|
g_slist_free (list);
|
|
}
|
|
|
|
|
|
void
|
|
_moo_folder_check_exists (MooFolder *folder,
|
|
const char *name)
|
|
{
|
|
MooFile *file;
|
|
char *path;
|
|
gboolean exists;
|
|
|
|
g_return_if_fail (MOO_IS_FOLDER (folder));
|
|
g_return_if_fail (name != NULL);
|
|
|
|
file = g_hash_table_lookup (folder->impl->files, name);
|
|
path = g_build_filename (folder->impl->path, name, NULL);
|
|
exists = g_file_test (path, G_FILE_TEST_EXISTS);
|
|
|
|
if (exists && !file)
|
|
file_created (folder->impl, name);
|
|
else if (!exists && file)
|
|
file_deleted (folder->impl, name);
|
|
|
|
g_free (path);
|
|
}
|
|
|
|
|
|
/* TODO */
|
|
static gboolean
|
|
moo_folder_do_reload (MooFolderImpl *impl)
|
|
{
|
|
GHashTable *files;
|
|
GDir *dir;
|
|
GError *error = NULL;
|
|
const char *name;
|
|
GSList *new = NULL, *deleted = NULL, *l;
|
|
|
|
g_return_val_if_fail (!impl->deleted, FALSE);
|
|
impl->reload_idle = 0;
|
|
|
|
dir = g_dir_open (impl->path, 0, &error);
|
|
|
|
if (!dir)
|
|
{
|
|
g_warning ("could not open directory %s: %s", impl->path, moo_error_message (error));
|
|
g_error_free (error);
|
|
folder_emit_deleted (impl);
|
|
return FALSE;
|
|
}
|
|
|
|
files = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
|
|
|
|
while ((name = g_dir_read_name (dir)))
|
|
g_hash_table_insert (files, g_strdup (name), NULL);
|
|
|
|
diff_hash_tables (files, impl->files, &new, &deleted);
|
|
|
|
for (l = new; l != NULL; l = l->next)
|
|
file_created (impl, l->data);
|
|
|
|
for (l = deleted; l != NULL; l = l->next)
|
|
file_deleted (impl, l->data);
|
|
|
|
g_slist_foreach (new, (GFunc) g_free, NULL);
|
|
g_slist_foreach (deleted, (GFunc) g_free, NULL);
|
|
g_slist_free (new);
|
|
g_slist_free (deleted);
|
|
g_hash_table_destroy (files);
|
|
g_dir_close (dir);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/* XXX */
|
|
static char *
|
|
moo_file_get_type_string (MooFile *file)
|
|
{
|
|
g_return_val_if_fail (MOO_FILE_EXISTS (file), NULL);
|
|
|
|
if (MOO_FILE_IS_DIR (file))
|
|
return g_strdup ("folder");
|
|
else if (file->mime_type)
|
|
return g_strdup (file->mime_type);
|
|
else
|
|
return g_strdup ("file");
|
|
}
|
|
|
|
|
|
/* XXX */
|
|
static char *
|
|
get_size_string (struct stat *statbuf)
|
|
{
|
|
g_return_val_if_fail (statbuf != NULL, NULL);
|
|
return g_strdup_printf ("%" G_GINT64_FORMAT, (gint64) statbuf->st_size);
|
|
}
|
|
|
|
|
|
/* XXX */
|
|
static char *
|
|
moo_file_get_mtime_string (MooFile *file)
|
|
{
|
|
static char buf[1024];
|
|
|
|
if (!MOO_FILE_EXISTS (file))
|
|
return NULL;
|
|
|
|
#ifdef __WIN32__
|
|
if (MOO_FILE_IS_DIR (file))
|
|
return NULL;
|
|
#endif
|
|
|
|
g_return_val_if_fail (file->statbuf != NULL, NULL);
|
|
|
|
if (strftime (buf, 1024, "%x %X", localtime (&file->statbuf->st_mtime)))
|
|
return g_strdup (buf);
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
|
|
char **
|
|
_moo_folder_get_file_info (MooFolder *folder,
|
|
MooFile *file)
|
|
{
|
|
GPtrArray *array;
|
|
GSList *list;
|
|
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
g_return_val_if_fail (!folder->impl->deleted, NULL);
|
|
g_return_val_if_fail (folder->impl->files != NULL, NULL);
|
|
g_return_val_if_fail (g_hash_table_lookup (folder->impl->files,
|
|
_moo_file_name (file)) == file, NULL);
|
|
|
|
_moo_file_stat (file, folder->impl->path);
|
|
|
|
if (file->info & MOO_FILE_INFO_EXISTS &&
|
|
!(file->flags & MOO_FILE_HAS_MIME_TYPE))
|
|
{
|
|
char *path = FILE_PATH (folder, file);
|
|
|
|
_moo_file_find_mime_type (file, path);
|
|
|
|
file->flags |= MOO_FILE_HAS_ICON;
|
|
file->icon = _moo_file_get_icon_type (file, folder->impl->path);
|
|
g_free (path);
|
|
}
|
|
|
|
array = g_ptr_array_new ();
|
|
|
|
if (file->info & MOO_FILE_INFO_EXISTS)
|
|
{
|
|
char *type, *mtime, *location;
|
|
|
|
g_ptr_array_add (array, g_strdup ("Type:"));
|
|
type = moo_file_get_type_string (file);
|
|
|
|
if (file->info & MOO_FILE_INFO_IS_LINK)
|
|
{
|
|
g_ptr_array_add (array, g_strdup_printf ("link to %s", type));
|
|
g_free (type);
|
|
}
|
|
else
|
|
{
|
|
g_ptr_array_add (array, type);
|
|
}
|
|
|
|
location = g_filename_display_name (_moo_folder_get_path (folder));
|
|
g_ptr_array_add (array, g_strdup ("Location:"));
|
|
g_ptr_array_add (array, location);
|
|
|
|
if (!(file->info & MOO_FILE_INFO_IS_DIR))
|
|
{
|
|
g_ptr_array_add (array, g_strdup ("Size:"));
|
|
g_ptr_array_add (array, get_size_string (file->statbuf));
|
|
}
|
|
|
|
mtime = moo_file_get_mtime_string (file);
|
|
|
|
if (mtime)
|
|
{
|
|
g_ptr_array_add (array, g_strdup ("Modified:"));
|
|
g_ptr_array_add (array, mtime);
|
|
}
|
|
}
|
|
else if (file->info & MOO_FILE_INFO_IS_LINK)
|
|
{
|
|
g_ptr_array_add (array, g_strdup ("Type:"));
|
|
g_ptr_array_add (array, g_strdup ("broken symbolic link"));
|
|
}
|
|
|
|
#ifndef __WIN32__
|
|
if ((file->info & MOO_FILE_INFO_IS_LINK) &&
|
|
_moo_file_link_get_target (file))
|
|
{
|
|
g_ptr_array_add (array, g_strdup ("Points to:"));
|
|
g_ptr_array_add (array, g_strdup (_moo_file_link_get_target (file)));
|
|
}
|
|
#endif
|
|
|
|
list = g_slist_append (NULL, _moo_file_ref (file));
|
|
g_object_ref (folder);
|
|
folder_emit_files (folder->impl, FILES_CHANGED, list);
|
|
g_object_unref (folder);
|
|
_moo_file_unref (file);
|
|
g_slist_free (list);
|
|
|
|
g_ptr_array_add (array, NULL);
|
|
return (char**) g_ptr_array_free (array, FALSE);
|
|
}
|
|
|
|
|
|
const char *
|
|
_moo_folder_get_path (MooFolder *folder)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
return folder->impl->path;
|
|
}
|
|
|
|
|
|
static void
|
|
files_list_free (GSList **list)
|
|
{
|
|
g_slist_foreach (*list, (GFunc) _moo_file_unref, NULL);
|
|
g_slist_free (*list);
|
|
*list = NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
prepend_file (G_GNUC_UNUSED gpointer key,
|
|
MooFile *file,
|
|
GSList **list)
|
|
{
|
|
*list = g_slist_prepend (*list, _moo_file_ref (file));
|
|
}
|
|
|
|
static GSList *
|
|
hash_table_to_file_list (GHashTable *files)
|
|
{
|
|
GSList *list = NULL;
|
|
g_return_val_if_fail (files != NULL, NULL);
|
|
g_hash_table_foreach (files, (GHFunc) prepend_file, &list);
|
|
return list;
|
|
}
|
|
|
|
|
|
static void
|
|
check_unique (const char *key,
|
|
G_GNUC_UNUSED gpointer whatever,
|
|
gpointer user_data)
|
|
{
|
|
struct {
|
|
GSList *list;
|
|
GHashTable *table2;
|
|
} *data = user_data;
|
|
gpointer orig_key, value;
|
|
|
|
if (!g_hash_table_lookup_extended (data->table2, key, &orig_key, &value))
|
|
data->list = g_slist_prepend (data->list, g_strdup (key));
|
|
}
|
|
|
|
static void
|
|
get_unique (GHashTable *table1,
|
|
GHashTable *table2,
|
|
GSList **only_1)
|
|
{
|
|
struct {
|
|
GSList *list;
|
|
GHashTable *table2;
|
|
} data;
|
|
|
|
data.list = NULL;
|
|
data.table2 = table2;
|
|
g_hash_table_foreach (table1, (GHFunc) check_unique, &data);
|
|
|
|
*only_1 = data.list;
|
|
}
|
|
|
|
static void
|
|
diff_hash_tables (GHashTable *table1,
|
|
GHashTable *table2,
|
|
GSList **only_1,
|
|
GSList **only_2)
|
|
{
|
|
get_unique (table1, table2, only_1);
|
|
get_unique (table2, table1, only_2);
|
|
}
|