
GLib/GTK+ functions, except for a couple of places were we use GtkCList.] * Makefile.am (INCLUDES): Add `-DG_DISABLE_DEPRECATED -DGTK_DISABLE_DEPRECATED'. * evolution-test-component.c (create_new_folder_selector): No gtk_window_set_policy(). * main.c (quit_box_new): gtk_window_set_resizable() instead of gtk_window_set_policy(). (view_map_callback): g_signal_handlers_disconnect_by_func() instead of gtk_signal_disconnect_by_func(). (main): No need to push the GdkRGB visual/cmap anymore. (new_view_created_callback): Likewise. * e-task-widget.c: Renamed member pixmap of ETaskWidgetPrivate to `image'. (init): Updated accordingly. (e_task_widget_construct): Use GtkImage, not GtkPixmap. * e-storage.c (class_init): Converted from gtk_signal_new() to g_signal_new(). * e-storage-set.c (class_init): Converted from gtk_signal_new() to g_signal_new(). * e-storage-set-view.c (setup_folder_properties_items_if_corba_storage_clicked): g_string_append_printf() instead of g_string_sprintfa(). (class_init): Converted from gtk_signal_new() to g_signal_new(). (setup_folder_changed_callbacks): Use e_signal_connect_while_alive() and e_signal_connect_full_while_alive(). * e-splash.c (e_splash_construct): gtk_window_set_resizable() instead of gtk_window_set_policy(). * e-shortcuts.c: Do not #include e-unicode.h. (shortcut_item_update): Use g_path_get_basename() instead of g_basename(). (e_shortcuts_add_default_shortcuts): No need to de-utfize strings. (e_shortcuts_add_default_group): Same here. * e-shortcuts-view.c: Do not #include <e-unicode.h>. (class_init): Converted from gtk_signal_new() to g_signal_new(). (destroy_group_cb): No need to convert the text from UTF8. * e-shell.c (impl_dispose): g_signal_handlers_disconnect_by_func() instead of gtk_signal_disconnect_by_func(). (class_init): Use g_signal_new() instead of gtk_signal_new(). * e-shell-view.c: Do not #include e-unicode.h. Renamed member offline_toggle_pixmap to offline_toggle_image in EShellViewPrivate. (update_folder_title_bar): No need to de-UTF8-ize the title. (cleanup_delayed_selection): g_signal_handlers_disconnect_by_func() instead of gtk_signal_disconnect_by_func(). (ui_engine_add_hint_callback): gtk_label_set_text(), not gtk_label_set(). (class_init): Use g_signal_new() instead of gtk_signal_new(). (update_for_current_uri): No need to convert the title from UTF8. (setup_offline_toggle): Changed to set up a GtkImage, not a GtkPixmap. (update_offline_toggle_status): Accordingly (gtk_image* instead of gtk_pixmap*). (set_current_notebook_page): gtk_notebook_set_current_page(), not gtk_notebook_set_page(). (get_storage_set_path_from_uri): g_ascii_strncasecmp() instead of g_strncasecmp(). (folder_bar_popup_map_callback): Use e_signal_connect_while_alive() instead of gtk_signal_connect_while_alive(). (e_shell_view_construct): Likewise. (display_uri): g_signal_connect_after() instead of gtk_signal_connect_full(). (update_for_current_uri): g_signal_handlers_{block,unblock}_by_func() instead of gtk_signal_handler_{block,unblock}_by_func(). * e-shell-view-menu.c (command_about_box): gtk_window_set_resizable() instead of gtk_window_set_policy(). * e-shell-utils.c (get_mini_name): g_path_get_basename() instead of g_basename(). * e-shell-user-creatable-items-handler.c (append_xml_for_menu_item): g_string_append_printf() instead of g_string_sprintfa(). (create_menu_xml): Likewise. * e-shell-shared-folder-picker-dialog.c: Do not #include e-unicode.h. (progress_bar_timeout_callback): Expect a GtkProgressBar data and just use gtk_progress_bar_pulse(). (create_progress_dialog): gtk_window_set_resizable() instead of gtk_window_set_policy(). No gtk_progress_set_activity_mode(). (setup_server_option_menu): Just use gtk_menu_item_new_with_label() instead of e_utf8_gtk_menu_item_new_with_label(). * e-shell-settings-dialog.c (set_dialog_size): Ported to Pango. * e-shell-offline-sync.c (impl_SyncFolderProgressListener_updateProgress): gtk_progress_set_fraction() instead of gtk_progress_bar_set_percentage(). (sync_folder): Likewise. (setup_dialog): gtk_window_set_resizable() instead of gtk_window_set_policy(). (sync_folder): No gtk_progress_set_activity_mode(). * e-shell-offline-handler.c: #undef {G,GTK}_DISABLE_DEPRECATED here for now (need to port from GtkCList). (class_init): Use g_signal_new() * e-shell-importer.c: #undef {G,GTK}_DISABLE_DEPRECATED here for now (need to port from GtkCList). * e-shell-folder-title-bar.c (create_image_widget_from_xpm): Renamed from create_pixmap_widget_from_xpm(); handle GtkImage instead of GtkPixmap. (new_empty_image_widget): Renamed from new_empty_pixmap_widget(); return a GtkImage instead of a GtkPixmap. (add_navigation_buttons): Updated accordingly; so use GtkImages instead of GtkPixmaps. (e_shell_folder_title_bar_construct): Likewise. (e_shell_folder_title_bar_set_icon): Use gtk_image_* instead of gtk_pixmap_* on the image widgets. (class_init): Converted to use g_signal_new() instead of gtk_signal_new(). (e_shell_folder_title_bar_new): No need for pushing the GdkRGB visual/cmap anymore. * e-shell-folder-selection-dialog.c (check_folder_type_valid): Just use strcmp instead of strcasecmp(). (class_init): g_signal_new() instead of gtk_signal_new(). (e_shell_folder_selection_dialog_construct): Removed call to gtk_window_set_policy(). * e-shell-folder-creation-dialog.c (type_with_display_name_compare_func): Changed to use g_utf8_casefold(). * e-shell-folder-commands.c: Do not #include e-unicode.h. (e_shell_command_rename_folder): g_path_get_dirname() instead of g_dirname(). (folder_selection_dialog_folder_selected_callback): g_path_get_basename() instead of g_basename(). (rename_cb): Likewise. (delete_dialog): Do not convert from UTF8 for display purposes. (e_shell_command_rename_folder): Likewise. * e-shell-about-box.c (timeout_callback): Ported to Pango and use gdk_window_invalidate_rect() instead of gtk_widget_draw(). * e-setup.c (check_evolution_directory): Use gtk_window_set_resizable() instead of gtk_window_set_policy(). * e-local-storage.c: Do not include e-unicode.h. (create_folder): Use g_path_get_basename() instead of g_basename(). (create_folder_directory): Likewise. (remove_folder_directory): Likewise. (append_xfer_item_list): Likewise. * e-local-folder.c (construct_loading_metadata): Use g_path_get_basename() instead of g_basename(). * e-folder-dnd-bridge.c (handle_evolution_path_drag_motion): Use g_path_get_basename() instead of g_basename(). * e-corba-storage-registry.c: G_STRUCT_OFFSET instead of GTK_STRUCT_OFFSET. * e-corba-shortcuts.c: G_STRUCT_OFFSET instead of GTK_STRUCT_OFFSET. * e-component-registry.c (sleep_with_g_main_loop_timeout_callback): g_main_loop_* instead of g_main_*. (sleep_with_g_main_loop): Likewise. * e-activity-handler.c: G_STRUCT_OFFSET instead of GTK_STRUCT_OFFSET. * evolution-wizard.c (evolution_wizard_class_init): g_signal_new() instead of gtk_signal_new(). * evolution-storage-listener.c (class_init): g_signal_new() instead of gtk_signal_new(). * evolution-shell-view.c (class_init): g_signal_new() instead of gtk_signal_new(). * evolution-shell-component-dnd.c: G_STRUCT_OFFSET instead of GTK_STRUCT_OFFSET. * evolution-shell-component.c: G_STRUCT_OFFSET instead of GTK_STRUCT_OFFSET. * evolution-session.c: G_STRUCT_OFFSET instead of GTK_STRUCT_OFFSET. * evolution-folder-selector-button.c: Do not include e-unicode.h. (set_folder): No need to convert from UTF8 to locale encoding anymore. Removed unused variable. * evolution-config-control.c (class_init): Use g_signal_new() instead of gtk_signal_new(). * evolution-activity-client.c (class_init): Use g_signal_new() instead of gtk_signal_new(). * e-folder-list.c: Do not include e-unicode.h. Use E_MAKE_TYPE(). (e_folder_list_get_type): Removed explicit implementation of this. (e_folder_list_set_arg): Removed. (e_folder_list_get_arg): Removed. (e_folder_list_set_property): New. (e_folder_list_get_property): New. (e_folder_list_destroy): Removed. (e_folder_list_dispose): New. (e_folder_list_class_init): Updated accordingly. * e-folder.c (impl_save_info): Use G_OBJECT_TYPE_NAME() instead of gtk_type_name(). (impl_load_info): Likewise. (impl_remove): Likewise. (class_init): Use g_signal_new() instead of gtk_signal_new(). svn path=/trunk/; revision=19170
889 lines
22 KiB
C
889 lines
22 KiB
C
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
|
||
/* e-storage-set.c
|
||
*
|
||
* Copyright (C) 2000 Ximian, Inc.
|
||
*
|
||
* This program is free software; you can redistribute it and/or
|
||
* modify it under the terms of version 2 of the GNU General Public
|
||
* License as published by the Free Software Foundation.
|
||
*
|
||
* 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: Ettore Perazzoli
|
||
*/
|
||
|
||
#ifdef HAVE_CONFIG_H
|
||
#include <config.h>
|
||
#endif
|
||
|
||
#include "e-storage-set.h"
|
||
|
||
#include "e-storage-set-view.h"
|
||
#include "e-shell-constants.h"
|
||
#include "e-shell-marshal.h"
|
||
|
||
#include <glib.h>
|
||
#include <gtk/gtkobject.h>
|
||
#include <gtk/gtksignal.h>
|
||
#include <gtk/gtktypeutils.h>
|
||
|
||
#include <gal/util/e-util.h>
|
||
|
||
#include <string.h>
|
||
|
||
|
||
#define PARENT_TYPE GTK_TYPE_OBJECT
|
||
|
||
static GtkObjectClass *parent_class = NULL;
|
||
|
||
/* This is just to make GHashTable happy. */
|
||
struct _NamedStorage {
|
||
char *name;
|
||
EStorage *storage;
|
||
};
|
||
typedef struct _NamedStorage NamedStorage;
|
||
|
||
struct _EStorageSetPrivate {
|
||
GList *storages; /* EStorage */
|
||
GHashTable *name_to_named_storage;
|
||
|
||
EFolderTypeRegistry *folder_type_registry;
|
||
};
|
||
|
||
enum {
|
||
NEW_STORAGE,
|
||
REMOVED_STORAGE,
|
||
NEW_FOLDER,
|
||
UPDATED_FOLDER,
|
||
REMOVED_FOLDER,
|
||
MOVED_FOLDER,
|
||
CLOSE_FOLDER,
|
||
LAST_SIGNAL
|
||
};
|
||
|
||
static guint signals[LAST_SIGNAL] = { 0 };
|
||
|
||
|
||
static NamedStorage *
|
||
named_storage_new (EStorage *storage)
|
||
{
|
||
NamedStorage *new;
|
||
|
||
new = g_new (NamedStorage, 1);
|
||
new->name = g_strdup (e_storage_get_name (storage));
|
||
new->storage = storage;
|
||
|
||
return new;
|
||
}
|
||
|
||
static void
|
||
named_storage_destroy (NamedStorage *named_storage)
|
||
{
|
||
g_free (named_storage->name);
|
||
g_free (named_storage);
|
||
}
|
||
|
||
static gboolean
|
||
name_to_named_storage_foreach_destroy (void *key,
|
||
void *value,
|
||
void *user_data)
|
||
{
|
||
NamedStorage *named_storage;
|
||
|
||
named_storage = (NamedStorage *) value;
|
||
named_storage_destroy (named_storage);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
|
||
/* "Callback converter", from `EStorageResultCallback' to
|
||
`EStorageSetResultCallback'. */
|
||
|
||
enum _StorageOperation {
|
||
OPERATION_COPY,
|
||
OPERATION_MOVE,
|
||
OPERATION_REMOVE,
|
||
OPERATION_CREATE
|
||
};
|
||
typedef enum _StorageOperation StorageOperation;
|
||
|
||
struct _StorageCallbackData {
|
||
EStorageSet *storage_set;
|
||
EStorageSetResultCallback storage_set_result_callback;
|
||
char *source_path;
|
||
char *destination_path;
|
||
StorageOperation operation;
|
||
void *data;
|
||
};
|
||
typedef struct _StorageCallbackData StorageCallbackData;
|
||
|
||
static StorageCallbackData *
|
||
storage_callback_data_new (EStorageSet *storage_set,
|
||
EStorageSetResultCallback callback,
|
||
const char *source_path,
|
||
const char *destination_path,
|
||
StorageOperation operation,
|
||
void *data)
|
||
{
|
||
StorageCallbackData *new;
|
||
|
||
new = g_new (StorageCallbackData, 1);
|
||
new->storage_set = storage_set;
|
||
new->storage_set_result_callback = callback;
|
||
new->source_path = g_strdup (source_path);
|
||
new->destination_path = g_strdup (destination_path);
|
||
new->operation = operation;
|
||
new->data = data;
|
||
|
||
return new;
|
||
}
|
||
|
||
static void
|
||
storage_callback_data_free (StorageCallbackData *data)
|
||
{
|
||
g_free (data->source_path);
|
||
g_free (data->destination_path);
|
||
|
||
g_free (data);
|
||
}
|
||
|
||
static void
|
||
storage_callback (EStorage *storage,
|
||
EStorageResult result,
|
||
void *data)
|
||
{
|
||
StorageCallbackData *storage_callback_data;
|
||
|
||
storage_callback_data = (StorageCallbackData *) data;
|
||
|
||
(* storage_callback_data->storage_set_result_callback) (storage_callback_data->storage_set,
|
||
result,
|
||
storage_callback_data->data);
|
||
|
||
if (storage_callback_data->operation == OPERATION_MOVE)
|
||
g_signal_emit (storage_callback_data->storage_set, signals[MOVED_FOLDER], 0,
|
||
storage_callback_data->source_path, storage_callback_data->destination_path);
|
||
|
||
storage_callback_data_free (storage_callback_data);
|
||
}
|
||
|
||
|
||
/* Handling for signals coming from the EStorages. */
|
||
|
||
static char *
|
||
make_full_path (EStorage *storage,
|
||
const char *path)
|
||
{
|
||
const char *storage_name;
|
||
char *full_path;
|
||
|
||
storage_name = e_storage_get_name (storage);
|
||
|
||
if (strcmp (path, E_PATH_SEPARATOR_S) == 0)
|
||
full_path = g_strconcat (E_PATH_SEPARATOR_S, storage_name,
|
||
NULL);
|
||
else if (! g_path_is_absolute (path))
|
||
full_path = g_strconcat (E_PATH_SEPARATOR_S, storage_name,
|
||
E_PATH_SEPARATOR_S, path, NULL);
|
||
else
|
||
full_path = g_strconcat (E_PATH_SEPARATOR_S, storage_name,
|
||
path, NULL);
|
||
|
||
return full_path;
|
||
}
|
||
|
||
static void
|
||
storage_new_folder_cb (EStorage *storage,
|
||
const char *path,
|
||
void *data)
|
||
{
|
||
EStorageSet *storage_set;
|
||
char *full_path;
|
||
|
||
storage_set = E_STORAGE_SET (data);
|
||
|
||
full_path = make_full_path (storage, path);
|
||
g_signal_emit (storage_set, signals[NEW_FOLDER], 0, full_path);
|
||
g_free (full_path);
|
||
}
|
||
|
||
static void
|
||
storage_updated_folder_cb (EStorage *storage,
|
||
const char *path,
|
||
void *data)
|
||
{
|
||
EStorageSet *storage_set;
|
||
char *full_path;
|
||
|
||
storage_set = E_STORAGE_SET (data);
|
||
|
||
full_path = make_full_path (storage, path);
|
||
g_signal_emit (storage_set, signals[UPDATED_FOLDER], 0, full_path);
|
||
g_free (full_path);
|
||
}
|
||
|
||
static void
|
||
storage_removed_folder_cb (EStorage *storage,
|
||
const char *path,
|
||
void *data)
|
||
{
|
||
EStorageSet *storage_set;
|
||
char *full_path;
|
||
|
||
storage_set = E_STORAGE_SET (data);
|
||
|
||
full_path = make_full_path (storage, path);
|
||
g_signal_emit (storage_set, signals[REMOVED_FOLDER], 0, full_path);
|
||
g_free (full_path);
|
||
}
|
||
|
||
static void
|
||
storage_close_folder_cb (EStorage *storage,
|
||
const char *path,
|
||
void *data)
|
||
{
|
||
EStorageSet *storage_set;
|
||
char *full_path;
|
||
|
||
storage_set = E_STORAGE_SET (data);
|
||
|
||
full_path = make_full_path (storage, path);
|
||
g_signal_emit (storage_set, signals[CLOSE_FOLDER], 0, full_path);
|
||
g_free (full_path);
|
||
}
|
||
|
||
|
||
static EStorage *
|
||
get_storage_for_path (EStorageSet *storage_set,
|
||
const char *path,
|
||
const char **subpath_return)
|
||
{
|
||
EStorage *storage;
|
||
char *storage_name;
|
||
const char *first_separator;
|
||
|
||
g_return_val_if_fail (g_path_is_absolute (path), NULL);
|
||
g_return_val_if_fail (path[1] != E_PATH_SEPARATOR, NULL);
|
||
|
||
/* Skip initial separator. */
|
||
path++;
|
||
|
||
first_separator = strchr (path, E_PATH_SEPARATOR);
|
||
|
||
if (first_separator == NULL || first_separator[1] == 0) {
|
||
storage = e_storage_set_get_storage (storage_set, path);
|
||
*subpath_return = E_PATH_SEPARATOR_S;
|
||
} else {
|
||
storage_name = g_strndup (path, first_separator - path);
|
||
storage = e_storage_set_get_storage (storage_set, storage_name);
|
||
g_free (storage_name);
|
||
|
||
*subpath_return = first_separator;
|
||
}
|
||
|
||
return storage;
|
||
}
|
||
|
||
static void
|
||
signal_new_folder_for_all_folders_under_paths (EStorageSet *storage_set,
|
||
EStorage *storage,
|
||
GList *path_list)
|
||
{
|
||
GList *p;
|
||
|
||
for (p = path_list; p != NULL; p = p->next) {
|
||
GList *sub_path_list;
|
||
const char *path;
|
||
char *path_with_storage;
|
||
|
||
path = (const char *) p->data;
|
||
|
||
path_with_storage = g_strconcat (E_PATH_SEPARATOR_S, e_storage_get_name (storage), path, NULL);
|
||
g_signal_emit (storage_set, signals[NEW_FOLDER], 0, path_with_storage);
|
||
g_free (path_with_storage);
|
||
|
||
sub_path_list = e_storage_get_subfolder_paths (storage, path);
|
||
|
||
signal_new_folder_for_all_folders_under_paths (storage_set, storage, sub_path_list);
|
||
|
||
e_free_string_list (sub_path_list);
|
||
}
|
||
}
|
||
|
||
static void
|
||
signal_new_folder_for_all_folders_in_storage (EStorageSet *storage_set,
|
||
EStorage *storage)
|
||
{
|
||
GList *path_list;
|
||
|
||
path_list = e_storage_get_subfolder_paths (storage, E_PATH_SEPARATOR_S);
|
||
|
||
signal_new_folder_for_all_folders_under_paths (storage_set, storage, path_list);
|
||
|
||
e_free_string_list (path_list);
|
||
}
|
||
|
||
|
||
/* GtkObject methods. */
|
||
|
||
static void
|
||
impl_dispose (GObject *object)
|
||
{
|
||
EStorageSet *storage_set;
|
||
EStorageSetPrivate *priv;
|
||
|
||
storage_set = E_STORAGE_SET (object);
|
||
priv = storage_set->priv;
|
||
|
||
if (priv->storages != NULL) {
|
||
e_free_object_list (priv->storages);
|
||
priv->storages = NULL;
|
||
}
|
||
|
||
if (priv->folder_type_registry != NULL) {
|
||
g_object_unref (priv->folder_type_registry);
|
||
priv->folder_type_registry = NULL;
|
||
}
|
||
|
||
(* G_OBJECT_CLASS (parent_class)->dispose) (object);
|
||
}
|
||
|
||
static void
|
||
impl_finalize (GObject *object)
|
||
{
|
||
EStorageSet *storage_set;
|
||
EStorageSetPrivate *priv;
|
||
|
||
storage_set = E_STORAGE_SET (object);
|
||
priv = storage_set->priv;
|
||
|
||
g_hash_table_foreach (priv->name_to_named_storage, (GHFunc) name_to_named_storage_foreach_destroy, NULL);
|
||
g_hash_table_destroy (priv->name_to_named_storage);
|
||
|
||
g_free (priv);
|
||
|
||
(* G_OBJECT_CLASS (parent_class)->finalize) (object);
|
||
}
|
||
|
||
|
||
static void
|
||
class_init (EStorageSetClass *klass)
|
||
{
|
||
GObjectClass *object_class;
|
||
|
||
parent_class = g_type_class_ref(gtk_object_get_type ());
|
||
object_class = G_OBJECT_CLASS (klass);
|
||
|
||
object_class->dispose = impl_dispose;
|
||
object_class->finalize = impl_finalize;
|
||
|
||
signals[NEW_STORAGE] =
|
||
g_signal_new ("new_storage",
|
||
G_OBJECT_CLASS_TYPE (object_class),
|
||
G_SIGNAL_RUN_FIRST,
|
||
G_STRUCT_OFFSET (EStorageSetClass, new_storage),
|
||
NULL, NULL,
|
||
e_shell_marshal_NONE__POINTER,
|
||
G_TYPE_NONE, 1,
|
||
G_TYPE_POINTER);
|
||
signals[REMOVED_STORAGE] =
|
||
g_signal_new ("removed_storage",
|
||
G_OBJECT_CLASS_TYPE (object_class),
|
||
G_SIGNAL_RUN_FIRST,
|
||
G_STRUCT_OFFSET (EStorageSetClass, removed_storage),
|
||
NULL, NULL,
|
||
e_shell_marshal_NONE__POINTER,
|
||
G_TYPE_NONE, 1,
|
||
G_TYPE_POINTER);
|
||
signals[NEW_FOLDER] =
|
||
g_signal_new ("new_folder",
|
||
G_OBJECT_CLASS_TYPE (object_class),
|
||
G_SIGNAL_RUN_FIRST,
|
||
G_STRUCT_OFFSET (EStorageSetClass, new_folder),
|
||
NULL, NULL,
|
||
e_shell_marshal_NONE__STRING,
|
||
G_TYPE_NONE, 1,
|
||
G_TYPE_STRING);
|
||
signals[UPDATED_FOLDER] =
|
||
g_signal_new ("updated_folder",
|
||
G_OBJECT_CLASS_TYPE (object_class),
|
||
G_SIGNAL_RUN_FIRST,
|
||
G_STRUCT_OFFSET (EStorageSetClass, updated_folder),
|
||
NULL, NULL,
|
||
e_shell_marshal_NONE__STRING,
|
||
G_TYPE_NONE, 1,
|
||
G_TYPE_STRING);
|
||
signals[REMOVED_FOLDER] =
|
||
g_signal_new ("removed_folder",
|
||
G_OBJECT_CLASS_TYPE (object_class),
|
||
G_SIGNAL_RUN_FIRST,
|
||
G_STRUCT_OFFSET (EStorageSetClass, removed_folder),
|
||
NULL, NULL,
|
||
e_shell_marshal_NONE__STRING,
|
||
G_TYPE_NONE, 1,
|
||
G_TYPE_STRING);
|
||
signals[MOVED_FOLDER] =
|
||
g_signal_new ("moved_folder",
|
||
G_OBJECT_CLASS_TYPE (object_class),
|
||
G_SIGNAL_RUN_FIRST,
|
||
G_STRUCT_OFFSET (EStorageSetClass, moved_folder),
|
||
NULL, NULL,
|
||
e_shell_marshal_NONE__STRING_STRING,
|
||
G_TYPE_NONE, 2,
|
||
G_TYPE_STRING,
|
||
G_TYPE_STRING);
|
||
signals[CLOSE_FOLDER] =
|
||
g_signal_new ("close_folder",
|
||
G_OBJECT_CLASS_TYPE (object_class),
|
||
G_SIGNAL_RUN_FIRST,
|
||
G_STRUCT_OFFSET (EStorageSetClass, close_folder),
|
||
NULL, NULL,
|
||
e_shell_marshal_NONE__STRING,
|
||
G_TYPE_NONE, 1,
|
||
G_TYPE_STRING);
|
||
}
|
||
|
||
static void
|
||
init (EStorageSet *storage_set)
|
||
{
|
||
EStorageSetPrivate *priv;
|
||
|
||
g_return_if_fail (E_IS_STORAGE_SET (storage_set));
|
||
|
||
priv = g_new (EStorageSetPrivate, 1);
|
||
priv->storages = NULL;
|
||
priv->name_to_named_storage = g_hash_table_new (g_str_hash, g_str_equal);
|
||
priv->folder_type_registry = NULL;
|
||
|
||
storage_set->priv = priv;
|
||
}
|
||
|
||
|
||
void
|
||
e_storage_set_construct (EStorageSet *storage_set,
|
||
EFolderTypeRegistry *folder_type_registry)
|
||
{
|
||
g_return_if_fail (storage_set != NULL);
|
||
g_return_if_fail (E_IS_STORAGE_SET (storage_set));
|
||
|
||
GTK_OBJECT_UNSET_FLAGS (storage_set, GTK_FLOATING);
|
||
|
||
g_object_ref (folder_type_registry);
|
||
storage_set->priv->folder_type_registry = folder_type_registry;
|
||
}
|
||
|
||
EStorageSet *
|
||
e_storage_set_new (EFolderTypeRegistry *folder_type_registry)
|
||
{
|
||
EStorageSet *new;
|
||
|
||
new = g_object_new (e_storage_set_get_type (), NULL);
|
||
|
||
e_storage_set_construct (new, folder_type_registry);
|
||
|
||
return new;
|
||
}
|
||
|
||
|
||
GList *
|
||
e_storage_set_get_storage_list (EStorageSet *storage_set)
|
||
{
|
||
EStorageSetPrivate *priv;
|
||
GList *list;
|
||
GList *p;
|
||
|
||
g_return_val_if_fail (storage_set != NULL, NULL);
|
||
g_return_val_if_fail (E_IS_STORAGE_SET (storage_set), NULL);
|
||
|
||
priv = storage_set->priv;
|
||
|
||
list = NULL;
|
||
for (p = priv->storages; p != NULL; p = p->next) {
|
||
g_object_ref (p->data);
|
||
list = g_list_prepend (list, p->data);
|
||
}
|
||
|
||
return g_list_reverse (list); /* Lame. */
|
||
}
|
||
|
||
/**
|
||
* e_storage_set_add_storage:
|
||
* @storage_set:
|
||
* @storage:
|
||
*
|
||
* Add @storage to @storage_set. Notice that will ref the storage.
|
||
**/
|
||
gboolean
|
||
e_storage_set_add_storage (EStorageSet *storage_set,
|
||
EStorage *storage)
|
||
{
|
||
EStorageSetPrivate *priv;
|
||
const char *storage_name;
|
||
NamedStorage *named_storage;
|
||
|
||
g_return_val_if_fail (storage_set != NULL, FALSE);
|
||
g_return_val_if_fail (E_IS_STORAGE_SET (storage_set), FALSE);
|
||
g_return_val_if_fail (storage != NULL, FALSE);
|
||
g_return_val_if_fail (E_IS_STORAGE (storage), FALSE);
|
||
|
||
priv = storage_set->priv;
|
||
|
||
storage_name = e_storage_get_name (storage);
|
||
if (g_hash_table_lookup (priv->name_to_named_storage, storage_name) != NULL)
|
||
return FALSE;
|
||
|
||
g_object_ref (storage);
|
||
|
||
g_signal_connect (storage, "new_folder",
|
||
G_CALLBACK (storage_new_folder_cb), storage_set);
|
||
g_signal_connect (storage, "updated_folder",
|
||
G_CALLBACK (storage_updated_folder_cb), storage_set);
|
||
g_signal_connect (storage, "removed_folder",
|
||
G_CALLBACK (storage_removed_folder_cb), storage_set);
|
||
g_signal_connect (storage, "close_folder",
|
||
G_CALLBACK (storage_close_folder_cb), storage_set);
|
||
|
||
priv->storages = g_list_append (priv->storages, storage);
|
||
|
||
named_storage = named_storage_new (storage);
|
||
g_hash_table_insert (priv->name_to_named_storage, named_storage->name, named_storage);
|
||
|
||
g_signal_emit (storage_set, signals[NEW_STORAGE], 0, storage);
|
||
|
||
signal_new_folder_for_all_folders_in_storage (storage_set, storage);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
gboolean
|
||
e_storage_set_remove_storage (EStorageSet *storage_set,
|
||
EStorage *storage)
|
||
{
|
||
EStorageSetPrivate *priv;
|
||
NamedStorage *named_storage;
|
||
|
||
g_return_val_if_fail (storage_set != NULL, FALSE);
|
||
g_return_val_if_fail (E_IS_STORAGE_SET (storage_set), FALSE);
|
||
g_return_val_if_fail (storage != NULL, FALSE);
|
||
g_return_val_if_fail (E_IS_STORAGE (storage), FALSE);
|
||
|
||
priv = storage_set->priv;
|
||
|
||
named_storage = g_hash_table_lookup (priv->name_to_named_storage,
|
||
e_storage_get_name (storage));
|
||
if (named_storage == NULL)
|
||
return FALSE;
|
||
|
||
g_hash_table_remove (priv->name_to_named_storage, named_storage->name);
|
||
named_storage_destroy (named_storage);
|
||
|
||
priv->storages = g_list_remove (priv->storages, storage);
|
||
|
||
g_signal_emit (storage_set, signals[REMOVED_STORAGE], 0, storage);
|
||
g_object_unref (storage);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
void
|
||
e_storage_set_remove_all_storages (EStorageSet *storage_set)
|
||
{
|
||
EStorageSetPrivate *priv;
|
||
GList *p;
|
||
|
||
g_return_if_fail (storage_set != NULL);
|
||
g_return_if_fail (E_IS_STORAGE_SET (storage_set));
|
||
|
||
priv = storage_set->priv;
|
||
|
||
for (p = priv->storages; p != NULL; p = p->next) {
|
||
EStorage *storage;
|
||
|
||
storage = E_STORAGE (p->data);
|
||
|
||
g_signal_emit (storage_set, signals[REMOVED_STORAGE], 0, storage);
|
||
g_object_unref (storage);
|
||
}
|
||
|
||
g_hash_table_foreach_remove (priv->name_to_named_storage,
|
||
name_to_named_storage_foreach_destroy,
|
||
NULL);
|
||
|
||
g_list_free (priv->storages);
|
||
priv->storages = NULL;
|
||
}
|
||
|
||
|
||
EStorage *
|
||
e_storage_set_get_storage (EStorageSet *storage_set,
|
||
const char *name)
|
||
{
|
||
EStorageSetPrivate *priv;
|
||
NamedStorage *named_storage;
|
||
|
||
g_return_val_if_fail (storage_set != NULL, NULL);
|
||
g_return_val_if_fail (E_IS_STORAGE_SET (storage_set), NULL);
|
||
g_return_val_if_fail (name != NULL, NULL);
|
||
|
||
priv = storage_set->priv;
|
||
|
||
named_storage = g_hash_table_lookup (priv->name_to_named_storage, name);
|
||
if (named_storage == NULL)
|
||
return NULL;
|
||
else
|
||
return named_storage->storage;
|
||
}
|
||
|
||
EFolder *
|
||
e_storage_set_get_folder (EStorageSet *storage_set,
|
||
const char *path)
|
||
{
|
||
EStorage *storage;
|
||
const char *subpath;
|
||
|
||
g_return_val_if_fail (storage_set != NULL, NULL);
|
||
g_return_val_if_fail (E_IS_STORAGE_SET (storage_set), NULL);
|
||
g_return_val_if_fail (path != NULL, NULL);
|
||
g_return_val_if_fail (g_path_is_absolute (path), NULL);
|
||
|
||
storage = get_storage_for_path (storage_set, path, &subpath);
|
||
if (storage == NULL)
|
||
return NULL;
|
||
|
||
return e_storage_get_folder (storage, subpath);
|
||
}
|
||
|
||
|
||
static void
|
||
storage_set_view_folder_opened (EStorageSetView *storage_set_view,
|
||
const char *path,
|
||
EStorageSet *storage_set)
|
||
{
|
||
EStorage *storage;
|
||
const char *subpath;
|
||
|
||
storage = get_storage_for_path (storage_set, path, &subpath);
|
||
if (storage == NULL)
|
||
return;
|
||
|
||
e_storage_async_open_folder (storage, subpath);
|
||
}
|
||
|
||
GtkWidget *
|
||
e_storage_set_create_new_view (EStorageSet *storage_set,
|
||
BonoboUIContainer *ui_container)
|
||
{
|
||
GtkWidget *storage_set_view;
|
||
|
||
g_return_val_if_fail (storage_set != NULL, NULL);
|
||
g_return_val_if_fail (E_IS_STORAGE_SET (storage_set), NULL);
|
||
|
||
storage_set_view = e_storage_set_view_new (storage_set, ui_container);
|
||
g_signal_connect (storage_set_view, "folder_opened",
|
||
G_CALLBACK (storage_set_view_folder_opened),
|
||
storage_set);
|
||
|
||
return storage_set_view;
|
||
}
|
||
|
||
|
||
void
|
||
e_storage_set_async_create_folder (EStorageSet *storage_set,
|
||
const char *path,
|
||
const char *type,
|
||
const char *description,
|
||
EStorageSetResultCallback callback,
|
||
void *data)
|
||
{
|
||
EStorage *storage;
|
||
const char *subpath;
|
||
StorageCallbackData *storage_callback_data;
|
||
|
||
g_return_if_fail (storage_set != NULL);
|
||
g_return_if_fail (E_IS_STORAGE_SET (storage_set));
|
||
g_return_if_fail (path != NULL);
|
||
g_return_if_fail (g_path_is_absolute (path));
|
||
g_return_if_fail (type != NULL);
|
||
g_return_if_fail (description != NULL);
|
||
g_return_if_fail (callback != NULL);
|
||
|
||
storage = get_storage_for_path (storage_set, path, &subpath);
|
||
|
||
storage_callback_data = storage_callback_data_new (storage_set, callback,
|
||
path, NULL, OPERATION_CREATE,
|
||
data);
|
||
|
||
e_storage_async_create_folder (storage, subpath, type, description,
|
||
storage_callback, storage_callback_data);
|
||
}
|
||
|
||
void
|
||
e_storage_set_async_remove_folder (EStorageSet *storage_set,
|
||
const char *path,
|
||
EStorageSetResultCallback callback,
|
||
void *data)
|
||
{
|
||
EStorage *storage;
|
||
const char *subpath;
|
||
StorageCallbackData *storage_callback_data;
|
||
|
||
g_return_if_fail (storage_set != NULL);
|
||
g_return_if_fail (E_IS_STORAGE_SET (storage_set));
|
||
g_return_if_fail (path != NULL);
|
||
g_return_if_fail (g_path_is_absolute (path));
|
||
g_return_if_fail (callback != NULL);
|
||
|
||
storage = get_storage_for_path (storage_set, path, &subpath);
|
||
|
||
storage_callback_data = storage_callback_data_new (storage_set, callback,
|
||
path, NULL, OPERATION_REMOVE,
|
||
data);
|
||
|
||
e_storage_async_remove_folder (storage, subpath,
|
||
storage_callback, storage_callback_data);
|
||
}
|
||
|
||
void
|
||
e_storage_set_async_xfer_folder (EStorageSet *storage_set,
|
||
const char *source_path,
|
||
const char *destination_path,
|
||
gboolean remove_source,
|
||
EStorageSetResultCallback callback,
|
||
void *data)
|
||
{
|
||
EStorage *source_storage;
|
||
EStorage *destination_storage;
|
||
const char *source_subpath;
|
||
const char *destination_subpath;
|
||
StorageCallbackData *storage_callback_data;
|
||
|
||
g_return_if_fail (storage_set != NULL);
|
||
g_return_if_fail (E_IS_STORAGE_SET (storage_set));
|
||
g_return_if_fail (source_path != NULL);
|
||
g_return_if_fail (g_path_is_absolute (source_path));
|
||
g_return_if_fail (destination_path != NULL);
|
||
g_return_if_fail (g_path_is_absolute (destination_path));
|
||
g_return_if_fail (callback != NULL);
|
||
|
||
source_storage = get_storage_for_path (storage_set, source_path, &source_subpath);
|
||
destination_storage = get_storage_for_path (storage_set, destination_path, &destination_subpath);
|
||
|
||
if (source_storage != destination_storage) {
|
||
g_warning ("e_storage_set_async_xfer_folder(): "
|
||
"Attempt to xfer folders between different storages -- not supported yet.");
|
||
(* callback) (storage_set, E_STORAGE_UNSUPPORTEDOPERATION, data);
|
||
return;
|
||
}
|
||
|
||
storage_callback_data = storage_callback_data_new (storage_set,
|
||
callback,
|
||
source_path,
|
||
destination_path,
|
||
remove_source ? OPERATION_MOVE : OPERATION_COPY,
|
||
data);
|
||
|
||
e_storage_async_xfer_folder (source_storage,
|
||
source_subpath, destination_subpath, remove_source,
|
||
storage_callback, storage_callback_data);
|
||
}
|
||
|
||
void
|
||
e_storage_set_async_remove_shared_folder (EStorageSet *storage_set,
|
||
const char *path,
|
||
EStorageSetResultCallback callback,
|
||
void *data)
|
||
{
|
||
EStorage *storage;
|
||
const char *subpath;
|
||
StorageCallbackData *storage_callback_data;
|
||
|
||
g_return_if_fail (storage_set != NULL);
|
||
g_return_if_fail (E_IS_STORAGE_SET (storage_set));
|
||
g_return_if_fail (path != NULL);
|
||
g_return_if_fail (g_path_is_absolute (path));
|
||
g_return_if_fail (callback != NULL);
|
||
|
||
storage = get_storage_for_path (storage_set, path, &subpath);
|
||
|
||
if (!e_storage_supports_shared_folders (storage)) {
|
||
(* callback) (storage_set, E_STORAGE_NOTIMPLEMENTED, data);
|
||
return;
|
||
}
|
||
|
||
storage_callback_data = storage_callback_data_new (storage_set, callback,
|
||
path, NULL, OPERATION_REMOVE,
|
||
data);
|
||
|
||
e_storage_async_remove_shared_folder (storage, subpath,
|
||
storage_callback,
|
||
storage_callback_data);
|
||
}
|
||
|
||
|
||
EFolderTypeRegistry *
|
||
e_storage_set_get_folder_type_registry (EStorageSet *storage_set)
|
||
{
|
||
g_return_val_if_fail (storage_set != NULL, NULL);
|
||
g_return_val_if_fail (E_IS_STORAGE_SET (storage_set), NULL);
|
||
|
||
return storage_set->priv->folder_type_registry;
|
||
}
|
||
|
||
|
||
/**
|
||
* e_storage_set_get_path_for_physical_uri:
|
||
* @storage_set: A storage set
|
||
* @physical_uri: A physical URI
|
||
*
|
||
* Retrieve the path of the folder whose physical URI matches @physical_uri.
|
||
*
|
||
* Return value:
|
||
**/
|
||
char *
|
||
e_storage_set_get_path_for_physical_uri (EStorageSet *storage_set,
|
||
const char *physical_uri)
|
||
{
|
||
EStorageSetPrivate *priv;
|
||
GList *p;
|
||
|
||
g_return_val_if_fail (storage_set != NULL, NULL);
|
||
g_return_val_if_fail (E_IS_STORAGE_SET (storage_set), NULL);
|
||
g_return_val_if_fail (physical_uri != NULL, NULL);
|
||
|
||
priv = storage_set->priv;
|
||
|
||
for (p = priv->storages; p != NULL; p = p->next) {
|
||
EStorage *storage;
|
||
char *storage_path;
|
||
|
||
storage = E_STORAGE (p->data);
|
||
|
||
storage_path = e_storage_get_path_for_physical_uri (storage, physical_uri);
|
||
if (storage_path != NULL) {
|
||
char *storage_set_path;
|
||
|
||
storage_set_path = g_strconcat (E_PATH_SEPARATOR_S,
|
||
e_storage_get_name (storage),
|
||
storage_path,
|
||
NULL);
|
||
g_free (storage_path);
|
||
|
||
return storage_set_path;
|
||
}
|
||
}
|
||
|
||
return NULL;
|
||
}
|
||
|
||
|
||
E_MAKE_TYPE (e_storage_set, "EStorageSet", EStorageSet, class_init, init, PARENT_TYPE)
|