1238 lines
36 KiB
C
1238 lines
36 KiB
C
/*
|
|
* mooedit/moofile.c
|
|
*
|
|
* Copyright (C) 2004-2005 by Yevgen Muntyan <muntyan@math.tamu.edu>
|
|
*
|
|
* 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.
|
|
*
|
|
* See COPYING file that comes with this distribution.
|
|
*/
|
|
|
|
/*
|
|
* Icon handling code was copied (and modified) from gtk/gtkfilesystemunix.c,
|
|
* Copyright (C) 2003, Red Hat, Inc.
|
|
*
|
|
* Don't ask me why and how rest of code works/done this way, it's all
|
|
* experimental.
|
|
*/
|
|
|
|
#define MOO_FILE_SYSTEM_COMPILATION
|
|
#include "mooedit/moofilesystem.h"
|
|
#include "mooedit/xdgmime/xdgmime.h"
|
|
#include "mooutils/moomarshals.h"
|
|
#include <string.h>
|
|
#include <sys/stat.h>
|
|
#include <errno.h>
|
|
|
|
#define NORMAL_PRIORITY G_PRIORITY_DEFAULT_IDLE
|
|
#define NORMAL_TIMEOUT 0.1
|
|
#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 PRINT_TIMES (G_GNUC_UNUSED const char *format, ...)
|
|
{
|
|
}
|
|
#endif
|
|
|
|
typedef enum {
|
|
STAGE_NAMES = 1,
|
|
STAGE_STAT = 2,
|
|
STAGE_MIME_TYPE = 3
|
|
} Stage;
|
|
|
|
typedef struct {
|
|
double names_timer;
|
|
double stat_timer;
|
|
guint stat_counter;
|
|
double icons_timer;
|
|
guint icons_counter;
|
|
} Debug;
|
|
|
|
struct _MooFolderPrivate {
|
|
guint deleted : 1;
|
|
Stage done;
|
|
Stage wanted;
|
|
Stage wanted_bg;
|
|
MooFileSystem *fs;
|
|
GDir *dir;
|
|
GSList *files;
|
|
GSList *files_copy;
|
|
char *path;
|
|
GSourceFunc populate_func;
|
|
int populate_priority;
|
|
guint populate_idle_id;
|
|
double populate_timeout;
|
|
Debug debug;
|
|
GTimer *timer;
|
|
};
|
|
|
|
|
|
static void moo_folder_finalize (GObject *object);
|
|
|
|
static void moo_folder_deleted (MooFolder *folder);
|
|
|
|
static void folder_emit_deleted (MooFolder *folder);
|
|
static void folder_emit_files (MooFolder *folder,
|
|
guint signal,
|
|
GSList *files);
|
|
|
|
static void stop_populate (MooFolder *folder);
|
|
|
|
static GSList *files_list_copy (GSList *list);
|
|
static void files_list_free (GSList **list);
|
|
|
|
static gboolean get_icons_a_bit (MooFolder *folder);
|
|
static gboolean get_stat_a_bit (MooFolder *folder);
|
|
static double get_names (MooFolder *folder);
|
|
|
|
static const char *get_default_file_icon (void);
|
|
static const char *get_nonexistent_icon (void);
|
|
static const char *get_blank_icon (void);
|
|
static const char *get_icon (MooFolder *folder,
|
|
MooFile *file);
|
|
static const char *get_folder_icon (const char *path);
|
|
static GdkPixbuf *get_named_icon (const char *name,
|
|
GtkWidget *widget,
|
|
GtkIconSize size);
|
|
static GdkPixbuf *create_named_icon (GtkIconTheme *icon_theme,
|
|
const char *name,
|
|
GtkWidget *widget,
|
|
GtkIconSize size);
|
|
static GdkPixbuf *create_fallback_icon (GtkIconTheme *icon_theme,
|
|
const char *name,
|
|
GtkWidget *widget,
|
|
GtkIconSize size);
|
|
static GdkPixbuf *create_icon_for_mime_type (GtkIconTheme *icon_theme,
|
|
const char *mime_type,
|
|
int pixel_size);
|
|
|
|
#define FILE_PATH(folder,file) g_build_filename (folder->priv->path, file->basename, NULL)
|
|
|
|
|
|
/* 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->finalize = moo_folder_finalize;
|
|
|
|
klass->deleted = moo_folder_deleted;
|
|
|
|
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 (MooFolder *folder)
|
|
{
|
|
folder->priv = g_new0 (MooFolderPrivate, 1);
|
|
folder->priv->deleted = FALSE;
|
|
folder->priv->done = 0;
|
|
folder->priv->wanted = 0;
|
|
folder->priv->fs = NULL;
|
|
folder->priv->dir = NULL;
|
|
folder->priv->files = NULL;
|
|
folder->priv->files_copy = NULL;
|
|
folder->priv->path = NULL;
|
|
folder->priv->populate_func = NULL;
|
|
folder->priv->populate_idle_id = 0;
|
|
folder->priv->populate_timeout = BACKGROUND_TIMEOUT;
|
|
folder->priv->timer = g_timer_new ();
|
|
g_timer_stop (folder->priv->timer);
|
|
}
|
|
|
|
|
|
static void moo_folder_finalize (GObject *object)
|
|
{
|
|
MooFolder *folder = MOO_FOLDER (object);
|
|
|
|
files_list_free (&folder->priv->files);
|
|
files_list_free (&folder->priv->files_copy);
|
|
if (folder->priv->dir)
|
|
g_dir_close (folder->priv->dir);
|
|
g_free (folder->priv->path);
|
|
if (folder->priv->populate_idle_id)
|
|
g_source_remove (folder->priv->populate_idle_id);
|
|
g_timer_destroy (folder->priv->timer);
|
|
|
|
g_free (folder->priv);
|
|
folder->priv = NULL;
|
|
|
|
G_OBJECT_CLASS (moo_folder_parent_class)->finalize (object);
|
|
}
|
|
|
|
|
|
MooFolder *moo_folder_new (MooFileSystem *fs,
|
|
const char *path,
|
|
MooFileFlags wanted,
|
|
GError **error)
|
|
{
|
|
GDir *dir;
|
|
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;
|
|
}
|
|
|
|
folder = g_object_new (MOO_TYPE_FOLDER, NULL);
|
|
folder->priv->fs = fs;
|
|
folder->priv->path = g_strdup (path);
|
|
|
|
folder->priv->dir = dir;
|
|
|
|
moo_folder_set_wanted (folder, wanted, TRUE);
|
|
|
|
return folder;
|
|
}
|
|
|
|
|
|
static void moo_folder_deleted (MooFolder *folder)
|
|
{
|
|
stop_populate (folder);
|
|
folder->priv->deleted = TRUE;
|
|
files_list_free (&folder->priv->files);
|
|
}
|
|
|
|
|
|
void moo_folder_set_wanted (MooFolder *folder,
|
|
MooFileFlags wanted,
|
|
gboolean bit_now)
|
|
{
|
|
Stage wanted_stage = STAGE_NAMES;
|
|
double elapsed = 0.0;
|
|
|
|
g_return_if_fail (MOO_IS_FOLDER (folder));
|
|
g_return_if_fail (!folder->priv->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->priv->done)
|
|
return;
|
|
|
|
if (folder->priv->wanted > folder->priv->done)
|
|
{
|
|
g_assert (folder->priv->populate_idle_id != 0);
|
|
folder->priv->wanted = MAX (folder->priv->wanted, wanted_stage);
|
|
return;
|
|
}
|
|
|
|
folder->priv->wanted = wanted_stage;
|
|
|
|
if (!folder->priv->done)
|
|
{
|
|
g_assert (folder->priv->dir != NULL);
|
|
elapsed = get_names (folder);
|
|
}
|
|
|
|
if (folder->priv->wanted_bg != 0)
|
|
{
|
|
g_assert (folder->priv->populate_idle_id != 0);
|
|
g_assert (folder->priv->populate_func != NULL);
|
|
g_assert (folder->priv->populate_priority != 0);
|
|
g_source_remove (folder->priv->populate_idle_id);
|
|
}
|
|
else
|
|
{
|
|
g_assert (folder->priv->populate_idle_id == 0);
|
|
|
|
switch (folder->priv->done)
|
|
{
|
|
case STAGE_NAMES:
|
|
g_assert (folder->priv->dir == NULL);
|
|
folder->priv->populate_func = (GSourceFunc) get_stat_a_bit;
|
|
break;
|
|
case STAGE_STAT:
|
|
g_assert (folder->priv->dir == NULL);
|
|
folder->priv->populate_func = (GSourceFunc) get_icons_a_bit;
|
|
break;
|
|
default:
|
|
g_assert_not_reached ();
|
|
}
|
|
}
|
|
|
|
folder->priv->populate_timeout = NORMAL_TIMEOUT;
|
|
folder->priv->populate_priority = NORMAL_PRIORITY;
|
|
|
|
TIMER_CLEAR (folder->priv->timer);
|
|
|
|
if (!bit_now ||
|
|
elapsed > folder->priv->populate_timeout ||
|
|
folder->priv->populate_func (folder))
|
|
{
|
|
folder->priv->populate_idle_id =
|
|
g_idle_add_full (folder->priv->populate_priority,
|
|
folder->priv->populate_func,
|
|
folder, NULL);
|
|
}
|
|
}
|
|
|
|
|
|
static void stop_populate (MooFolder *folder)
|
|
{
|
|
if (folder->priv->populate_idle_id)
|
|
g_source_remove (folder->priv->populate_idle_id);
|
|
folder->priv->populate_idle_id = 0;
|
|
folder->priv->populate_func = 0;
|
|
if (folder->priv->dir)
|
|
g_dir_close (folder->priv->dir);
|
|
folder->priv->dir = NULL;
|
|
files_list_free (&folder->priv->files_copy);
|
|
}
|
|
|
|
|
|
static void folder_emit_deleted (MooFolder *folder)
|
|
{
|
|
g_signal_emit (folder, signals[DELETED], 0);
|
|
}
|
|
|
|
|
|
static void folder_emit_files (MooFolder *folder,
|
|
guint sig,
|
|
GSList *files)
|
|
{
|
|
if (files)
|
|
g_signal_emit (folder, signals[sig], 0, files);
|
|
}
|
|
|
|
|
|
GSList *moo_folder_list_files (MooFolder *folder)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
return files_list_copy (folder->priv->files);
|
|
}
|
|
|
|
|
|
static double get_names (MooFolder *folder)
|
|
{
|
|
GTimer *timer;
|
|
GSList *added = NULL;
|
|
const char *name;
|
|
MooFile *file;
|
|
double elapsed;
|
|
|
|
g_assert (folder->priv->path != NULL);
|
|
g_assert (folder->priv->dir != NULL);
|
|
|
|
timer = g_timer_new ();
|
|
|
|
file = moo_file_new (folder->priv->path, "..");
|
|
file->icon = get_folder_icon (NULL);
|
|
file->flags = MOO_FILE_HAS_MIME_TYPE | MOO_FILE_HAS_ICON;
|
|
file->info = MOO_FILE_EXISTS | MOO_FILE_IS_FOLDER;
|
|
folder->priv->files = g_slist_prepend (folder->priv->files, file);
|
|
added = g_slist_prepend (added, file);
|
|
|
|
for (name = g_dir_read_name (folder->priv->dir);
|
|
name != NULL;
|
|
name = g_dir_read_name (folder->priv->dir))
|
|
{
|
|
file = moo_file_new (folder->priv->path, name);
|
|
file->icon = get_blank_icon ();
|
|
folder->priv->files = g_slist_prepend (folder->priv->files, file);
|
|
added = g_slist_prepend (added, file);
|
|
}
|
|
|
|
elapsed = folder->priv->debug.names_timer =
|
|
g_timer_elapsed (timer, NULL);
|
|
g_timer_destroy (timer);
|
|
|
|
folder_emit_files (folder, FILES_ADDED, added);
|
|
g_slist_free (added);
|
|
|
|
g_dir_close (folder->priv->dir);
|
|
folder->priv->dir = NULL;
|
|
|
|
folder->priv->done = STAGE_NAMES;
|
|
|
|
PRINT_TIMES ("names folder %s: %f sec\n",
|
|
folder->priv->path,
|
|
folder->priv->debug.names_timer);
|
|
|
|
return elapsed;
|
|
}
|
|
|
|
|
|
static gboolean get_stat_a_bit (MooFolder *folder)
|
|
{
|
|
gboolean done = FALSE;
|
|
double elapsed;
|
|
|
|
g_assert (folder->priv->dir == NULL);
|
|
g_assert (folder->priv->done == STAGE_NAMES);
|
|
g_assert (folder->priv->path != NULL);
|
|
|
|
elapsed = g_timer_elapsed (folder->priv->timer, NULL);
|
|
g_timer_continue (folder->priv->timer);
|
|
|
|
if (!folder->priv->files_copy)
|
|
folder->priv->files_copy =
|
|
files_list_copy (folder->priv->files);
|
|
if (!folder->priv->files_copy)
|
|
done = TRUE;
|
|
|
|
while (!done)
|
|
{
|
|
GSList *changed = folder->priv->files_copy;
|
|
MooFile *file = changed->data;
|
|
folder->priv->files_copy =
|
|
g_slist_remove_link (folder->priv->files_copy,
|
|
folder->priv->files_copy);
|
|
|
|
if (!(file->flags & MOO_FILE_HAS_STAT))
|
|
{
|
|
moo_file_stat (file, folder->priv->path);
|
|
folder_emit_files (folder, FILES_CHANGED, changed);
|
|
}
|
|
else
|
|
{
|
|
moo_file_unref (file);
|
|
}
|
|
|
|
g_slist_free_1 (changed);
|
|
|
|
if (!folder->priv->files_copy)
|
|
done = TRUE;
|
|
|
|
if (g_timer_elapsed (folder->priv->timer, NULL) > folder->priv->populate_timeout)
|
|
break;
|
|
}
|
|
|
|
elapsed = g_timer_elapsed (folder->priv->timer, NULL) - elapsed;
|
|
folder->priv->debug.stat_timer += elapsed;
|
|
folder->priv->debug.stat_counter += 1;
|
|
g_timer_stop (folder->priv->timer);
|
|
|
|
if (!done)
|
|
{
|
|
TIMER_CLEAR (folder->priv->timer);
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
g_assert (folder->priv->files_copy == NULL);
|
|
folder->priv->populate_idle_id = 0;
|
|
|
|
PRINT_TIMES ("stat folder %s: %d iterations, %f sec\n",
|
|
folder->priv->path,
|
|
folder->priv->debug.stat_counter,
|
|
folder->priv->debug.stat_timer);
|
|
|
|
folder->priv->done = STAGE_STAT;
|
|
|
|
if (folder->priv->wanted >= STAGE_MIME_TYPE || folder->priv->wanted_bg >= STAGE_MIME_TYPE)
|
|
{
|
|
if (folder->priv->wanted >= STAGE_MIME_TYPE)
|
|
{
|
|
folder->priv->populate_priority = NORMAL_PRIORITY;
|
|
folder->priv->populate_timeout = NORMAL_TIMEOUT;
|
|
}
|
|
else if (folder->priv->wanted_bg >= STAGE_MIME_TYPE)
|
|
{
|
|
folder->priv->populate_priority = BACKGROUND_PRIORITY;
|
|
folder->priv->populate_timeout = BACKGROUND_TIMEOUT;
|
|
}
|
|
|
|
if (folder->priv->populate_idle_id)
|
|
g_source_remove (folder->priv->populate_idle_id);
|
|
folder->priv->populate_idle_id = 0;
|
|
folder->priv->populate_func = (GSourceFunc) get_icons_a_bit;
|
|
|
|
if (g_timer_elapsed (folder->priv->timer, NULL) < folder->priv->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 (folder->priv->timer);
|
|
if (folder->priv->populate_func (folder))
|
|
folder->priv->populate_idle_id =
|
|
g_idle_add_full (folder->priv->populate_priority,
|
|
folder->priv->populate_func,
|
|
folder, NULL);
|
|
}
|
|
else
|
|
{
|
|
TIMER_CLEAR (folder->priv->timer);
|
|
folder->priv->populate_idle_id =
|
|
g_idle_add_full (folder->priv->populate_priority,
|
|
folder->priv->populate_func,
|
|
folder, NULL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
folder->priv->populate_func = NULL;
|
|
folder->priv->populate_priority = 0;
|
|
folder->priv->populate_timeout = 0;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
static gboolean get_icons_a_bit (MooFolder *folder)
|
|
{
|
|
gboolean done = FALSE;
|
|
double elapsed;
|
|
|
|
g_assert (folder->priv->dir == NULL);
|
|
g_assert (folder->priv->done == STAGE_STAT);
|
|
g_assert (folder->priv->path != NULL);
|
|
|
|
elapsed = g_timer_elapsed (folder->priv->timer, NULL);
|
|
g_timer_continue (folder->priv->timer);
|
|
|
|
if (!folder->priv->files_copy)
|
|
folder->priv->files_copy =
|
|
files_list_copy (folder->priv->files);
|
|
if (!folder->priv->files_copy)
|
|
done = TRUE;
|
|
|
|
while (!done)
|
|
{
|
|
GSList *changed = folder->priv->files_copy;
|
|
MooFile *file = changed->data;
|
|
char *path;
|
|
|
|
folder->priv->files_copy =
|
|
g_slist_remove_link (folder->priv->files_copy,
|
|
changed);
|
|
|
|
if (file->info & MOO_FILE_EXISTS &&
|
|
!(file->flags & MOO_FILE_HAS_MIME_TYPE))
|
|
{
|
|
path = FILE_PATH (folder, file);
|
|
file->mime_type = xdg_mime_get_mime_type_for_file (path);
|
|
file->flags |= MOO_FILE_HAS_MIME_TYPE;
|
|
file->flags |= MOO_FILE_HAS_ICON;
|
|
file->icon = get_icon (folder, file);
|
|
folder_emit_files (folder, FILES_CHANGED, changed);
|
|
g_free (path);
|
|
}
|
|
else
|
|
{
|
|
moo_file_unref (file);
|
|
}
|
|
|
|
g_slist_free_1 (changed);
|
|
|
|
if (!folder->priv->files_copy)
|
|
done = TRUE;
|
|
|
|
if (g_timer_elapsed (folder->priv->timer, NULL) > folder->priv->populate_timeout)
|
|
break;
|
|
}
|
|
|
|
elapsed = g_timer_elapsed (folder->priv->timer, NULL) - elapsed;
|
|
folder->priv->debug.icons_timer += elapsed;
|
|
folder->priv->debug.icons_counter += 1;
|
|
TIMER_CLEAR (folder->priv->timer);
|
|
|
|
if (done)
|
|
{
|
|
PRINT_TIMES ("icons folder %s: %d iterations, %f sec\n",
|
|
folder->priv->path,
|
|
folder->priv->debug.icons_counter,
|
|
folder->priv->debug.icons_timer);
|
|
|
|
g_assert (folder->priv->files_copy == NULL);
|
|
folder->priv->populate_idle_id = 0;
|
|
folder->priv->done = STAGE_MIME_TYPE;
|
|
folder->priv->populate_func = NULL;
|
|
folder->priv->populate_priority = 0;
|
|
folder->priv->populate_timeout = 0;
|
|
}
|
|
|
|
return !done;
|
|
}
|
|
|
|
|
|
/* TODO TODO */
|
|
MooFile *moo_folder_get_file (MooFolder *folder,
|
|
const char *basename)
|
|
{
|
|
GSList *l;
|
|
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
g_return_val_if_fail (basename != NULL, NULL);
|
|
|
|
for (l = folder->priv->files; l != NULL; l = l->next)
|
|
{
|
|
MooFile *file = l->data;
|
|
g_assert (file != NULL && file->basename != NULL);
|
|
if (!strcmp (basename, file->basename))
|
|
return file;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
MooFolder *moo_folder_get_parent (MooFolder *folder,
|
|
MooFileFlags wanted)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
return moo_file_system_get_parent_folder (folder->priv->fs,
|
|
folder, wanted);
|
|
}
|
|
|
|
|
|
MooFileSystem *moo_folder_get_file_system (MooFolder *folder)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
return folder->priv->fs;
|
|
}
|
|
|
|
|
|
/********************************************************************/
|
|
/* MooFile
|
|
*/
|
|
|
|
MooFile *moo_file_new (const char *dirname,
|
|
const char *basename)
|
|
{
|
|
MooFile *file;
|
|
char *path;
|
|
|
|
g_return_val_if_fail (dirname != NULL && basename != NULL, NULL);
|
|
|
|
file = g_new0 (MooFile, 1);
|
|
|
|
file->basename = g_strdup (basename);
|
|
path = g_build_filename (dirname, basename, NULL);
|
|
file->display_basename = g_filename_display_basename (path);
|
|
g_free (path);
|
|
file->ref_count = 1;
|
|
|
|
if (basename[0] == '.')
|
|
file->info = MOO_FILE_IS_HIDDEN;
|
|
|
|
return file;
|
|
}
|
|
|
|
|
|
MooFile *moo_file_ref (MooFile *file)
|
|
{
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
file->ref_count++;
|
|
return file;
|
|
}
|
|
|
|
|
|
void moo_file_unref (MooFile *file)
|
|
{
|
|
if (file && !--file->ref_count)
|
|
{
|
|
g_free (file->basename);
|
|
g_free (file->display_basename);
|
|
g_free (file);
|
|
}
|
|
}
|
|
|
|
|
|
void moo_file_stat (MooFile *file,
|
|
const char *dirname)
|
|
{
|
|
char *fullname;
|
|
|
|
g_return_if_fail (file != NULL);
|
|
|
|
fullname = g_build_filename (dirname, file->basename, NULL);
|
|
|
|
file->info = MOO_FILE_EXISTS;
|
|
file->flags = MOO_FILE_HAS_STAT;
|
|
|
|
if (stat (fullname, &file->statbuf) != 0)
|
|
{
|
|
if (errno == ENOENT && !lstat (fullname, &file->statbuf))
|
|
{
|
|
gchar *display_name = g_filename_display_name (fullname);
|
|
g_message ("%s: file '%s' is a broken link",
|
|
G_STRLOC, display_name);
|
|
g_free (display_name);
|
|
file->info = MOO_FILE_IS_LINK;
|
|
}
|
|
else if (errno == EACCES)
|
|
{
|
|
gchar *display_name = g_filename_display_name (fullname);
|
|
g_message ("%s: error getting information for '%s': %s",
|
|
G_STRLOC, display_name,
|
|
g_strerror (EACCES));
|
|
g_free (display_name);
|
|
file->info = 0;
|
|
file->flags = 0;
|
|
}
|
|
else
|
|
{
|
|
int save_errno = errno;
|
|
gchar *display_name = g_filename_display_name (fullname);
|
|
g_message ("%s: error getting information for '%s': %s",
|
|
G_STRLOC, display_name,
|
|
g_strerror (save_errno));
|
|
g_free (display_name);
|
|
file->info = 0;
|
|
file->flags = 0;
|
|
}
|
|
}
|
|
|
|
if (file->info & MOO_FILE_EXISTS)
|
|
{
|
|
if (S_ISDIR (file->statbuf.st_mode))
|
|
file->info |= MOO_FILE_IS_FOLDER;
|
|
if (S_ISLNK (file->statbuf.st_mode))
|
|
file->info |= MOO_FILE_IS_LINK;
|
|
}
|
|
|
|
if (file->info & MOO_FILE_EXISTS)
|
|
{
|
|
if (file->info & MOO_FILE_IS_FOLDER)
|
|
{
|
|
file->flags |= MOO_FILE_HAS_MIME_TYPE;
|
|
file->flags |= MOO_FILE_HAS_ICON;
|
|
file->icon = get_folder_icon (fullname);
|
|
}
|
|
else
|
|
{
|
|
file->icon = get_default_file_icon ();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
file->icon = get_nonexistent_icon ();
|
|
}
|
|
|
|
if (file->basename[0] == '.')
|
|
file->info |= MOO_FILE_IS_HIDDEN;
|
|
|
|
g_free (fullname);
|
|
}
|
|
|
|
|
|
gboolean moo_file_test (const MooFile *file,
|
|
MooFileInfo test)
|
|
{
|
|
g_return_val_if_fail (file != NULL, FALSE);
|
|
return file->info & test;
|
|
}
|
|
|
|
|
|
const char *moo_file_get_display_basename (const MooFile *file)
|
|
{
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
return file->display_basename;
|
|
}
|
|
|
|
|
|
const char *moo_file_get_basename (const MooFile *file)
|
|
{
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
return file->basename;
|
|
}
|
|
|
|
|
|
const char *moo_file_get_mime_type (const MooFile *file)
|
|
{
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
return file->mime_type;
|
|
}
|
|
|
|
|
|
gconstpointer moo_file_get_stat (const MooFile *file)
|
|
{
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
if (file->flags & MOO_FILE_HAS_STAT && file->info & MOO_FILE_EXISTS)
|
|
return &file->statbuf;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
|
|
GdkPixbuf *moo_file_get_icon (const MooFile *file,
|
|
GtkWidget *widget,
|
|
GtkIconSize size)
|
|
{
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
return get_named_icon (file->icon, widget, size);
|
|
}
|
|
|
|
|
|
GType moo_file_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
|
|
if (!type)
|
|
type = g_boxed_type_register_static ("MooFile",
|
|
(GBoxedCopyFunc) moo_file_ref,
|
|
(GBoxedFreeFunc) moo_file_unref);
|
|
|
|
return type;
|
|
}
|
|
|
|
|
|
GType moo_file_flags_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
|
|
static const GFlagsValue values[] = {
|
|
{ MOO_FILE_HAS_MIME_TYPE, (char*)"MOO_FILE_HAS_MIME_TYPE", (char*)"has-mime-type" },
|
|
{ MOO_FILE_HAS_ICON, (char*)"MOO_FILE_HAS_ICON", (char*)"has-icon" },
|
|
{ MOO_FILE_HAS_STAT, (char*)"MOO_FILE_HAS_STAT", (char*)"has-stat" },
|
|
{ MOO_FILE_ALL_FLAGS, (char*)"MOO_FILE_ALL_FLAGS", (char*)"all-flags" },
|
|
{ 0, NULL, NULL }
|
|
};
|
|
|
|
if (!type)
|
|
type = g_flags_register_static ("MooFileFlags", values);
|
|
|
|
return type;
|
|
}
|
|
|
|
|
|
GType moo_file_info_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
|
|
static const GFlagsValue values[] = {
|
|
{ MOO_FILE_EXISTS, (char*)"MOO_FILE_EXISTS", (char*)"exists" },
|
|
{ MOO_FILE_IS_FOLDER, (char*)"MOO_FILE_IS_FOLDER", (char*)"is-folder" },
|
|
{ MOO_FILE_IS_HIDDEN, (char*)"MOO_FILE_IS_HIDDEN", (char*)"is-hidden" },
|
|
{ MOO_FILE_IS_LINK, (char*)"MOO_FILE_IS_LINK", (char*)"is-link" },
|
|
{ 0, NULL, NULL }
|
|
};
|
|
|
|
if (!type)
|
|
type = g_flags_register_static ("MooFileInfo", values);
|
|
|
|
return type;
|
|
}
|
|
|
|
|
|
const char *moo_folder_get_path (MooFolder *folder)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLDER (folder), NULL);
|
|
return folder->priv->path;
|
|
}
|
|
|
|
|
|
static GSList *files_list_copy (GSList *list)
|
|
{
|
|
GSList *copy, *l;
|
|
for (copy = NULL, l = list; l != NULL; l = l->next)
|
|
copy = g_slist_prepend (copy, moo_file_ref (l->data));
|
|
return g_slist_reverse (copy);
|
|
}
|
|
|
|
|
|
static void files_list_free (GSList **list)
|
|
{
|
|
g_slist_foreach (*list, (GFunc) moo_file_unref, NULL);
|
|
g_slist_free (*list);
|
|
*list = NULL;
|
|
}
|
|
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
#define GNOME_HOME_ICON_NAME "gnome-fs-home"
|
|
#define GNOME_DESKTOP_ICON_NAME "gnome-fs-desktop"
|
|
#define GNOME_TRASH_ICON_NAME "gnome-fs-trash-full"
|
|
#define GNOME_DIRECTORY_ICON_NAME "gnome-fs-directory"
|
|
#define FILE_ICON_NAME "gnome-fs-regular"
|
|
#define BLOCK_DEVICE_ICON_NAME "gnome-fs-blockdev"
|
|
#define EXECUTABLE_ICON_NAME "gnome-fs-executable"
|
|
#define SYMBOLIC_LINK_ICON_NAME "gnome-fs-symlink"
|
|
#define CHARACTER_DEVICE_ICON_NAME "gnome-fs-chardev"
|
|
#define FIFO_ICON_NAME "gnome-fs-fifo"
|
|
#define SOCKET_ICON_NAME "gnome-fs-socket"
|
|
#define NONEXISTENT_ICON_NAME "BROKEN"
|
|
#define BROKEN_LINK_ICON_NAME "BROKEN_LINK"
|
|
#define BLANK_ICON_NAME ""
|
|
|
|
|
|
static const char *get_default_file_icon (void)
|
|
{
|
|
return FILE_ICON_NAME;
|
|
}
|
|
|
|
static const char *get_blank_icon (void)
|
|
{
|
|
return BLANK_ICON_NAME;
|
|
}
|
|
|
|
static const char *get_nonexistent_icon (void)
|
|
{
|
|
return NONEXISTENT_ICON_NAME;
|
|
}
|
|
|
|
|
|
static GdkPixbuf *create_icon_for_mime_type (GtkIconTheme *icon_theme,
|
|
const char *mime_type,
|
|
int pixel_size)
|
|
{
|
|
const char *separator;
|
|
GString *icon_name;
|
|
GdkPixbuf *pixbuf;
|
|
|
|
separator = strchr (mime_type, '/');
|
|
if (!separator)
|
|
return NULL;
|
|
|
|
icon_name = g_string_new ("gnome-mime-");
|
|
g_string_append_len (icon_name, mime_type, separator - mime_type);
|
|
g_string_append_c (icon_name, '-');
|
|
g_string_append (icon_name, separator + 1);
|
|
pixbuf = gtk_icon_theme_load_icon (icon_theme, icon_name->str,
|
|
pixel_size, 0, NULL);
|
|
g_string_free (icon_name, TRUE);
|
|
|
|
if (pixbuf)
|
|
return pixbuf;
|
|
|
|
icon_name = g_string_new ("gnome-mime-");
|
|
g_string_append_len (icon_name, mime_type, separator - mime_type);
|
|
pixbuf = gtk_icon_theme_load_icon (icon_theme, icon_name->str,
|
|
pixel_size, 0, NULL);
|
|
g_string_free (icon_name, TRUE);
|
|
|
|
return pixbuf;
|
|
}
|
|
|
|
|
|
static GdkPixbuf *get_named_icon (const char *name,
|
|
GtkWidget *widget,
|
|
GtkIconSize size)
|
|
{
|
|
GtkIconTheme *icon_theme;
|
|
GHashTable *all_sizes_cache, *cache;
|
|
GdkPixbuf *pixbuf;
|
|
|
|
if (!name)
|
|
return NULL;
|
|
|
|
icon_theme = gtk_icon_theme_get_for_screen (gtk_widget_get_screen (widget));
|
|
all_sizes_cache = g_object_get_data (G_OBJECT (icon_theme), "moo-file-icon-cache");
|
|
|
|
if (!all_sizes_cache)
|
|
{
|
|
all_sizes_cache =
|
|
g_hash_table_new_full (g_direct_hash, g_direct_equal,
|
|
NULL,
|
|
(GDestroyNotify) g_hash_table_destroy);
|
|
g_object_set_data_full (G_OBJECT (icon_theme),
|
|
"moo-file-icon-cache",
|
|
all_sizes_cache,
|
|
(GDestroyNotify) g_hash_table_destroy);
|
|
}
|
|
|
|
cache = g_hash_table_lookup (all_sizes_cache, GINT_TO_POINTER (size));
|
|
|
|
if (!cache)
|
|
{
|
|
cache = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
g_free, g_object_unref);
|
|
g_hash_table_insert (all_sizes_cache, GINT_TO_POINTER (size), cache);
|
|
}
|
|
|
|
pixbuf = g_hash_table_lookup (cache, name);
|
|
|
|
if (pixbuf)
|
|
return pixbuf;
|
|
|
|
pixbuf = create_named_icon (icon_theme, name, widget, size);
|
|
g_assert (pixbuf != NULL);
|
|
g_hash_table_insert (cache, g_strdup (name), pixbuf);
|
|
|
|
return pixbuf;
|
|
}
|
|
|
|
|
|
static GdkPixbuf *create_named_icon (GtkIconTheme *icon_theme,
|
|
const char *name,
|
|
GtkWidget *widget,
|
|
GtkIconSize size)
|
|
{
|
|
int width, height;
|
|
GdkPixbuf *pixbuf;
|
|
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
|
|
if (!strcmp (name, BLANK_ICON_NAME))
|
|
{
|
|
pixbuf = gtk_widget_render_icon (widget, GTK_STOCK_FILE, size, "");
|
|
if (!pixbuf)
|
|
g_warning ("%s: could not create a stock icon for %s",
|
|
G_STRLOC, GTK_STOCK_FILE);
|
|
return pixbuf;
|
|
}
|
|
|
|
if (!strcmp (name, NONEXISTENT_ICON_NAME) ||
|
|
!strcmp (name, BROKEN_LINK_ICON_NAME))
|
|
{
|
|
pixbuf = gtk_widget_render_icon (widget, GTK_STOCK_MISSING_IMAGE, size, "");
|
|
if (!pixbuf)
|
|
g_warning ("%s: could not create a stock icon for %s",
|
|
G_STRLOC, GTK_STOCK_MISSING_IMAGE);
|
|
return pixbuf;
|
|
}
|
|
|
|
if (!gtk_icon_size_lookup (size, &width, &height))
|
|
if (!gtk_icon_size_lookup (GTK_ICON_SIZE_MENU, &width, &height))
|
|
width = height = 16;
|
|
|
|
if (!strncmp (name, "MIME-", 5))
|
|
{
|
|
const char *mime_type = name + 5;
|
|
pixbuf = create_icon_for_mime_type (icon_theme, mime_type, width);
|
|
if (pixbuf)
|
|
return pixbuf;
|
|
else
|
|
return get_named_icon (FILE_ICON_NAME, widget, size);
|
|
}
|
|
|
|
pixbuf = gtk_icon_theme_load_icon (icon_theme, name, width, 0, NULL);
|
|
|
|
if (!pixbuf)
|
|
pixbuf = create_fallback_icon (icon_theme, name, widget, size);
|
|
|
|
return pixbuf;
|
|
}
|
|
|
|
|
|
static const char *get_folder_icon (const char *path)
|
|
{
|
|
static const char *home_path = NULL;
|
|
static char *desktop_path = NULL;
|
|
static char *trash_path = NULL;
|
|
|
|
if (!path)
|
|
return GNOME_DIRECTORY_ICON_NAME;
|
|
|
|
if (!home_path)
|
|
home_path = g_get_home_dir ();
|
|
|
|
if (!home_path)
|
|
return GNOME_DIRECTORY_ICON_NAME;
|
|
|
|
if (!desktop_path)
|
|
desktop_path = g_build_filename (home_path, "Desktop", NULL);
|
|
|
|
if (!trash_path)
|
|
trash_path = g_build_filename (desktop_path, "Trash", NULL);
|
|
|
|
if (strcmp (home_path, path) == 0)
|
|
return GNOME_HOME_ICON_NAME;
|
|
else if (strcmp (desktop_path, path) == 0)
|
|
return GNOME_DESKTOP_ICON_NAME;
|
|
else if (strcmp (trash_path, path) == 0)
|
|
return GNOME_TRASH_ICON_NAME;
|
|
else
|
|
return GNOME_DIRECTORY_ICON_NAME;
|
|
}
|
|
|
|
|
|
static GdkPixbuf *create_fallback_icon (GtkIconTheme *icon_theme,
|
|
const char *name,
|
|
GtkWidget *widget,
|
|
GtkIconSize size)
|
|
{
|
|
const char *stock_name;
|
|
GdkPixbuf *pixbuf;
|
|
|
|
if (!strcmp (name, BLOCK_DEVICE_ICON_NAME))
|
|
stock_name = GTK_STOCK_HARDDISK;
|
|
else if (!strcmp (name, EXECUTABLE_ICON_NAME))
|
|
stock_name = GTK_STOCK_EXECUTE;
|
|
else
|
|
{
|
|
int width, height;
|
|
|
|
if (!gtk_icon_size_lookup (size, &width, &height))
|
|
if (!gtk_icon_size_lookup (GTK_ICON_SIZE_MENU, &width, &height))
|
|
width = height = 16;
|
|
|
|
pixbuf = gtk_icon_theme_load_icon (icon_theme, FILE_ICON_NAME,
|
|
width, 0, NULL);
|
|
|
|
if (pixbuf)
|
|
return pixbuf;
|
|
else
|
|
stock_name = GTK_STOCK_FILE;
|
|
}
|
|
|
|
pixbuf = gtk_widget_render_icon (widget, stock_name, size, NULL);
|
|
|
|
if (!pixbuf)
|
|
g_warning ("%s: could not create a stock icon for %s",
|
|
G_STRLOC, stock_name);
|
|
|
|
return pixbuf;
|
|
}
|
|
|
|
|
|
static const char *get_icon (MooFolder *folder,
|
|
MooFile *file)
|
|
{
|
|
static GHashTable *mime_type_names = NULL;
|
|
|
|
if (!(file->info & MOO_FILE_EXISTS))
|
|
return NONEXISTENT_ICON_NAME;
|
|
|
|
if (file->info & MOO_FILE_IS_FOLDER)
|
|
{
|
|
char *path = FILE_PATH (folder, file);
|
|
const char *name = get_folder_icon (path);
|
|
g_free (path);
|
|
return name;
|
|
}
|
|
|
|
if (file->flags & MOO_FILE_HAS_STAT)
|
|
{
|
|
struct stat *statp = &file->statbuf;
|
|
|
|
if (S_ISBLK (statp->st_mode))
|
|
return BLOCK_DEVICE_ICON_NAME;
|
|
else if (S_ISLNK (statp->st_mode))
|
|
return BROKEN_LINK_ICON_NAME;
|
|
else if (S_ISCHR (statp->st_mode))
|
|
return CHARACTER_DEVICE_ICON_NAME;
|
|
#ifdef S_ISFIFO
|
|
else if (S_ISFIFO (statp->st_mode))
|
|
return FIFO_ICON_NAME;
|
|
#endif
|
|
#ifdef S_ISSOCK
|
|
else if (S_ISSOCK (statp->st_mode))
|
|
return SOCKET_ICON_NAME;
|
|
#endif
|
|
}
|
|
|
|
if (file->flags & MOO_FILE_HAS_MIME_TYPE && file->mime_type)
|
|
{
|
|
char *name;
|
|
|
|
if (!mime_type_names)
|
|
mime_type_names =
|
|
g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
g_free, g_free);
|
|
|
|
name = g_hash_table_lookup (mime_type_names, file->mime_type);
|
|
|
|
if (!name)
|
|
{
|
|
name = g_strdup_printf ("MIME-%s", file->mime_type);
|
|
g_hash_table_insert (mime_type_names,
|
|
g_strdup (file->mime_type),
|
|
name);
|
|
}
|
|
|
|
return name;
|
|
}
|
|
|
|
return FILE_ICON_NAME;
|
|
}
|