aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--calendar/ChangeLog14
-rw-r--r--calendar/gui/migration.c327
2 files changed, 271 insertions, 70 deletions
diff --git a/calendar/ChangeLog b/calendar/ChangeLog
index 6366f30cc2..875ff3223a 100644
--- a/calendar/ChangeLog
+++ b/calendar/ChangeLog
@@ -1,3 +1,17 @@
+2004-01-24 JP Rosevear <jpr@ximian.com>
+
+ * gui/migration.c (setup_progress_dialog): display a progress
+ dialog
+ (dialog_close): destroy it
+ (dialog_set_folder_name): display the folder being migrated
+ (dialog_set_progress): update the progress
+ (check_for_conflict): make sure we don't name anything the same
+ (get_source_name): create a source name
+ (migrate_ical): migrate the events/tasks from one ecal to another
+ (migrate_ical_folder): migrate a specific folder
+ (migrate_calendars): use above
+ (migrate_tasks): ditto
+
2004-01-23 Rodney Dawes <dobey@ximian.com>
* gui/e-itip-control.c: Use PUBLISH if no METHOD defined
diff --git a/calendar/gui/migration.c b/calendar/gui/migration.c
index c91f356261..2ba2739ebd 100644
--- a/calendar/gui/migration.c
+++ b/calendar/gui/migration.c
@@ -20,11 +20,19 @@
* Author: Rodrigo Moya <rodrigo@ximian.com>
*/
+#include <string.h>
#include <bonobo/bonobo-i18n.h>
#include <libgnomevfs/gnome-vfs-uri.h>
#include <libgnomevfs/gnome-vfs-xfer.h>
+#include <gtk/gtkwidget.h>
+#include <gtk/gtkvbox.h>
+#include <gtk/gtkmain.h>
+#include <gtk/gtklabel.h>
+#include <gtk/gtkprogressbar.h>
#include <gal/util/e-util.h>
+#include <libecal/e-cal.h>
#include <e-util/e-bconf-map.h>
+#include <e-util/e-folder-map.h>
#include "migration.h"
static e_gconf_map_t calendar_display_map[] = {
@@ -106,78 +114,223 @@ e_gconf_map_list_t task_remap_list[] = {
{ 0 },
};
+static GtkWidget *window;
+static GtkLabel *label;
+static GtkProgressBar *progress;
+
+static void
+setup_progress_dialog (gboolean tasks)
+{
+ GtkWidget *vbox, *hbox, *w;
+
+ window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
+ gtk_window_set_title ((GtkWindow *) window, _("Migrating..."));
+ gtk_window_set_modal ((GtkWindow *) window, TRUE);
+ gtk_container_set_border_width ((GtkContainer *) window, 6);
+
+ vbox = gtk_vbox_new (FALSE, 6);
+ gtk_widget_show (vbox);
+ gtk_container_add ((GtkContainer *) window, vbox);
+
+ if (tasks)
+ w = gtk_label_new (_("The location and hierarchy of the Evolution task "
+ "folders has changed since Evolution 1.x.\n\nPlease be "
+ "patient while Evolution migrates your folders..."));
+ else
+ w = gtk_label_new (_("The location and hierarchy of the Evolution calendar "
+ "folders has changed since Evolution 1.x.\n\nPlease be "
+ "patient while Evolution migrates your folders..."));
+
+ gtk_label_set_line_wrap ((GtkLabel *) w, TRUE);
+ gtk_widget_show (w);
+ gtk_box_pack_start_defaults ((GtkBox *) vbox, w);
+
+ hbox = gtk_hbox_new (FALSE, 6);
+ gtk_widget_show (hbox);
+ gtk_box_pack_start_defaults ((GtkBox *) vbox, hbox);
+
+ label = (GtkLabel *) gtk_label_new ("");
+ gtk_widget_show ((GtkWidget *) label);
+ gtk_box_pack_start_defaults ((GtkBox *) hbox, (GtkWidget *) label);
+
+ progress = (GtkProgressBar *) gtk_progress_bar_new ();
+ gtk_widget_show ((GtkWidget *) progress);
+ gtk_box_pack_start_defaults ((GtkBox *) hbox, (GtkWidget *) progress);
+
+ gtk_widget_show (window);
+}
+
+static void
+dialog_close (void)
+{
+ gtk_widget_destroy ((GtkWidget *) window);
+}
+
+static void
+dialog_set_folder_name (const char *folder_name)
+{
+ char *text;
+
+ text = g_strdup_printf (_("Migrating `%s':"), folder_name);
+ gtk_label_set_text (label, text);
+ g_free (text);
+
+ gtk_progress_bar_set_fraction (progress, 0.0);
+
+ while (gtk_events_pending ())
+ gtk_main_iteration ();
+}
+
+static void
+dialog_set_progress (double percent)
+{
+ char text[5];
+
+ snprintf (text, sizeof (text), "%d%%", (int) (percent * 100.0f));
+
+ gtk_progress_bar_set_fraction (progress, percent);
+ gtk_progress_bar_set_text (progress, text);
+
+ while (gtk_events_pending ())
+ gtk_main_iteration ();
+}
+
static gboolean
-process_old_dir (ESourceGroup *source_group, const char *path,
- const char *filename, const char *name, const char *base_uri)
+check_for_conflict (ESourceGroup *group, char *name)
{
- char *s;
- GnomeVFSURI *from, *to;
- GnomeVFSResult vres;
- ESource *source;
- GDir *dir;
- gboolean retval = TRUE;
+ GSList *sources;
+ GSList *s;
- s = g_build_filename (path, filename, NULL);
- if (!g_file_test (s, G_FILE_TEST_EXISTS)) {
- g_free (s);
- return FALSE;
- }
+ sources = e_source_group_peek_sources (group);
- /* transfer the old file to its new location */
- from = gnome_vfs_uri_new (s);
- g_free (s);
- if (!from)
- return FALSE;
+ for (s = sources; s; s = s->next) {
+ ESource *source = E_SOURCE (s->data);
- to = gnome_vfs_uri_new (s);
- g_free (s);
- if (!to) {
- gnome_vfs_uri_unref (from);
- return FALSE;
+ if (!strcmp (e_source_peek_name (source), name))
+ return TRUE;
}
- vres = gnome_vfs_xfer_uri ((const GnomeVFSURI *) from,
- (const GnomeVFSURI *) to,
- GNOME_VFS_XFER_DEFAULT,
- GNOME_VFS_XFER_ERROR_MODE_ABORT,
- GNOME_VFS_XFER_OVERWRITE_MODE_REPLACE,
- NULL, NULL);
- gnome_vfs_uri_unref (from);
- gnome_vfs_uri_unref (to);
+ return FALSE;
+}
+
+static char *
+get_source_name (ESourceGroup *group, const char *path)
+{
+ char **p = g_strsplit (path, "/", 0);
+ int i, j, starting_index;
+ int num_elements;
+ gboolean conflict;
+ GString *s = g_string_new ("");
+
+ for (i = 0; p[i]; i ++) ;
+
+ num_elements = i;
+ i--;
+
+ /* p[i] is now the last path element */
+
+ /* check if it conflicts */
+ starting_index = i;
+ do {
+ g_string_assign (s, "");
+ for (j = starting_index; j < num_elements; j += 2) {
+ if (j != starting_index)
+ g_string_append_c (s, '_');
+ g_string_append (s, p[j]);
+ }
+
+ conflict = check_for_conflict (group, s->str);
+
+
+ /* if there was a conflict back up 2 levels (skipping the /subfolder/ element) */
+ if (conflict)
+ starting_index -= 2;
+
+ /* we always break out if we can't go any further,
+ regardless of whether or not we conflict. */
+ if (starting_index < 0)
+ break;
+
+ } while (conflict);
- if (vres != GNOME_VFS_OK)
+ return g_string_free (s, FALSE);
+}
+
+static gboolean
+migrate_ical (ECal *old_ecal, ECal *new_ecal)
+{
+ GList *l, *objects;
+ int num_added = 0;
+ int num_objects;
+ gboolean retval = TRUE;
+
+ /* both ecals are loaded, start the actual migration */
+ if (!e_cal_get_object_list (old_ecal, "#t", &objects, NULL))
return FALSE;
- /* Find the default source we create or create a new source */
- source = e_source_group_peek_source_by_name (source_group, name);
- if (!source)
- source = e_source_new (name, base_uri);
- e_source_group_add_source (source_group, source, -1);
-
- /* process subfolders */
- s = g_build_filename (path, "subfolders", NULL);
- dir = g_dir_open (s, 0, NULL);
- if (dir) {
- const char *name;
- char *tmp_s;
-
- while ((name = g_dir_read_name (dir))) {
- tmp_s = g_build_filename (s, name, NULL);
- if (g_file_test (tmp_s, G_FILE_TEST_IS_DIR)) {
- retval = process_old_dir (source_group, tmp_s, filename, name, name);
- }
+ num_objects = g_list_length (objects);
+ for (l = objects; l; l = l->next) {
+ icalcomponent *ical_comp = l->data;
+ GError *error = NULL;
- g_free (tmp_s);
+ if (!e_cal_create_object (new_ecal, ical_comp, NULL, &error)) {
+ g_warning ("Migration of object failed: %s", error->message);
+ retval = FALSE;
}
+
+ g_clear_error (&error);
- g_dir_close (dir);
+ num_added ++;
+ dialog_set_progress ((double)num_added / num_objects);
}
- g_free (s);
-
return retval;
}
+static gboolean
+migrate_ical_folder (char *old_path, ESourceGroup *dest_group, char *source_name, ECalSourceType type)
+{
+ ECal *old_ecal = NULL, *new_ecal = NULL;
+ ESource *old_source;
+ ESource *new_source;
+ ESourceGroup *group;
+ char *old_uri = g_strdup_printf ("file://%s", old_path);
+ GError *error = NULL;
+ gboolean retval = FALSE;
+
+ group = e_source_group_new ("", old_uri);
+ old_source = e_source_new ("", "");
+ e_source_set_group (old_source, group);
+ g_object_unref (group);
+
+ new_source = e_source_new (source_name, source_name);
+ e_source_set_group (new_source, dest_group);
+
+ dialog_set_folder_name (source_name);
+
+ old_ecal = e_cal_new (old_source, type);
+ if (!e_cal_open (old_ecal, TRUE, &error)) {
+ g_warning ("failed to load source ecal for migration: `%s'", error->message);
+ goto finish;
+ }
+
+ new_ecal = e_cal_new (new_source, type);
+ if (!e_cal_open (new_ecal, FALSE, &error)) {
+ g_warning ("failed to load destination ecal for migration: `%s'", error->message);
+ goto finish;
+ }
+
+ retval = migrate_ical (old_ecal, new_ecal);
+
+ finish:
+ g_clear_error (&error);
+ g_object_unref (old_ecal);
+ g_object_unref (new_ecal);
+ g_free (old_uri);
+
+ return retval;
+}
+
static ESourceGroup *
create_calendar_contact_source (ESourceList *source_list)
{
@@ -351,21 +504,38 @@ migrate_calendars (CalendarComponent *component, int major, int minor, int revis
if (minor <= 4) {
ESourceGroup *on_this_computer;
- char *path;
+ GSList *migration_dirs, *l;
+ char *path, *local_cal_folder;
+
+ setup_progress_dialog (FALSE);
if (!create_calendar_sources (component, calendar_component_peek_source_list (component), &on_this_computer, NULL, NULL))
return FALSE;
- /* FIXME Look for all top level calendars */
- path = g_build_filename (g_get_home_dir (), "evolution/local/Calendar", NULL);
- if (!g_file_test (path, G_FILE_TEST_IS_DIR)) {
- g_free (path);
- return FALSE;
- }
- retval = process_old_dir (on_this_computer, path, "calendar.ics", _("Personal"), "Personal");
+ path = g_build_filename (g_get_home_dir (), "evolution", "local", NULL);
+ migration_dirs = e_folder_map_local_folders (path, "calendar");
+ local_cal_folder = g_build_filename (path, "Calendar", NULL);
g_free (path);
+
+ for (l = migration_dirs; l; l = l->next) {
+ char *source_name;
+
+ if (!strcmp (l->data, local_cal_folder))
+ source_name = g_strdup (_("Personal"));
+ else
+ source_name = get_source_name (on_this_computer, (char*)l->data + strlen (path) + 1);
+
+ if (!migrate_ical_folder (l->data, on_this_computer, source_name, E_CAL_SOURCE_TYPE_EVENT))
+ retval = FALSE;
+
+ g_free (source_name);
+ }
+
+ g_free (local_cal_folder);
e_source_list_sync (calendar_component_peek_source_list (component), NULL);
+
+ dialog_close ();
}
if (minor == 5 && revision < 2) {
@@ -421,21 +591,38 @@ migrate_tasks (TasksComponent *component, int major, int minor, int revision)
if (minor <= 4) {
ESourceGroup *on_this_computer;
- char *path;
+ GSList *migration_dirs, *l;
+ char *path, *local_task_folder;
+ setup_progress_dialog (TRUE);
+
if (!create_task_sources (component, tasks_component_peek_source_list (component), &on_this_computer))
return FALSE;
- /* FIXME Look for all top level tasks */
- path = g_build_filename (g_get_home_dir (), "evolution/local/Tasks", NULL);
- if (!g_file_test (path, G_FILE_TEST_IS_DIR)) {
- g_free (path);
- return FALSE;
- }
- retval = process_old_dir (on_this_computer, path, "tasks.ics", _("Personal"), "Personal");
+ path = g_build_filename (g_get_home_dir (), "evolution", "local", NULL);
+ migration_dirs = e_folder_map_local_folders (path, "tasks");
+ local_task_folder = g_build_filename (path, "Tasks", NULL);
g_free (path);
+ for (l = migration_dirs; l; l = l->next) {
+ char *source_name;
+
+ if (!strcmp (l->data, local_task_folder))
+ source_name = g_strdup (_("Personal"));
+ else
+ source_name = get_source_name (on_this_computer, (char*)l->data + strlen (path) + 1);
+
+ if (!migrate_ical_folder (l->data, on_this_computer, source_name, E_CAL_SOURCE_TYPE_TODO))
+ retval = FALSE;
+
+ g_free (source_name);
+ }
+
+ g_free (local_task_folder);
+
e_source_list_sync (tasks_component_peek_source_list (component), NULL);
+
+ dialog_close ();
}
}