/*
* e-mail-config-import-page.c
*
* This program 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 of the License, or (at your option) version 3.
*
* 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with the program; if not, see
*
*/
#include "e-mail-config-import-page.h"
#include
#include
#include
#define E_MAIL_CONFIG_IMPORT_PAGE_GET_PRIVATE(obj) \
(G_TYPE_INSTANCE_GET_PRIVATE \
((obj), E_TYPE_MAIL_CONFIG_IMPORT_PAGE, EMailConfigImportPagePrivate))
typedef struct _AsyncContext AsyncContext;
struct _EMailConfigImportPagePrivate {
EImport *import;
EImportTarget *import_target;
GSList *available_importers;
};
struct _AsyncContext {
EMailConfigImportPage *page;
GQueue pending_importers;
EActivity *activity;
GCancellable *cancellable;
gulong cancel_id;
};
/* Forward Declarations */
static void e_mail_config_import_page_interface_init
(EMailConfigPageInterface *interface);
static gboolean mail_config_import_page_next (gpointer user_data);
G_DEFINE_DYNAMIC_TYPE_EXTENDED (
EMailConfigImportPage,
e_mail_config_import_page,
GTK_TYPE_BOX,
0,
G_IMPLEMENT_INTERFACE_DYNAMIC (
E_TYPE_MAIL_CONFIG_PAGE,
e_mail_config_import_page_interface_init))
static void
async_context_free (AsyncContext *async_context)
{
if (async_context->page != NULL)
g_object_unref (async_context->page);
if (async_context->activity != NULL)
g_object_unref (async_context->activity);
if (async_context->cancellable != NULL) {
g_cancellable_disconnect (
async_context->cancellable,
async_context->cancel_id);
g_object_unref (async_context->cancellable);
}
g_queue_clear (&async_context->pending_importers);
g_slice_free (AsyncContext, async_context);
}
static void
mail_config_import_page_status (EImport *import,
const gchar *what,
gint percent,
GSimpleAsyncResult *simple)
{
AsyncContext *async_context;
async_context = g_simple_async_result_get_op_res_gpointer (simple);
e_activity_set_text (async_context->activity, what);
e_activity_set_percent (async_context->activity, (gdouble) percent);
}
static void
mail_config_import_page_complete (EImport *import,
GSimpleAsyncResult *simple)
{
/* Schedule the next importer to start. */
g_idle_add (mail_config_import_page_next, simple);
}
static gboolean
mail_config_import_page_next (gpointer user_data)
{
GSimpleAsyncResult *simple;
AsyncContext *async_context;
GCancellable *cancellable;
EImportImporter *next_importer;
GError *error = NULL;
simple = G_SIMPLE_ASYNC_RESULT (user_data);
async_context = g_simple_async_result_get_op_res_gpointer (simple);
cancellable = async_context->cancellable;
/* Pop the completed importer and peek at the next one. */
g_queue_pop_head (&async_context->pending_importers);
next_importer = g_queue_peek_head (&async_context->pending_importers);
if (g_cancellable_set_error_if_cancelled (cancellable, &error)) {
g_simple_async_result_take_error (simple, error);
g_simple_async_result_complete (simple);
g_object_unref (simple);
} else if (next_importer != NULL) {
e_import_import (
async_context->page->priv->import,
async_context->page->priv->import_target,
next_importer,
(EImportStatusFunc) mail_config_import_page_status,
(EImportCompleteFunc) mail_config_import_page_complete,
simple);
} else {
g_simple_async_result_complete (simple);
g_object_unref (simple);
}
return FALSE;
}
static void
mail_config_import_page_cancelled (GCancellable *cancellable,
AsyncContext *async_context)
{
GQueue *pending_importers;
EImportImporter *current_importer;
pending_importers = &async_context->pending_importers;
current_importer = g_queue_peek_head (pending_importers);
g_return_if_fail (current_importer != NULL);
e_import_cancel (
async_context->page->priv->import,
async_context->page->priv->import_target,
current_importer);
}
static void
mail_config_import_page_dispose (GObject *object)
{
EMailConfigImportPagePrivate *priv;
priv = E_MAIL_CONFIG_IMPORT_PAGE_GET_PRIVATE (object);
if (priv->import != NULL) {
e_import_target_free (
priv->import,
priv->import_target);
g_object_unref (priv->import);
priv->import_target = NULL;
priv->import = NULL;
}
g_slist_free (priv->available_importers);
priv->available_importers = NULL;
/* Chain up to parent's dispose() method. */
G_OBJECT_CLASS (e_mail_config_import_page_parent_class)->
dispose (object);
}
static void
mail_config_import_page_constructed (GObject *object)
{
EMailConfigImportPage *page;
GtkWidget *widget;
GtkWidget *container;
GSList *list, *link;
const gchar *text;
gint row = 0;
page = E_MAIL_CONFIG_IMPORT_PAGE (object);
/* Chain up to parent's constructed() method. */
G_OBJECT_CLASS (e_mail_config_import_page_parent_class)->
constructed (object);
gtk_orientable_set_orientation (
GTK_ORIENTABLE (page), GTK_ORIENTATION_VERTICAL);
gtk_box_set_spacing (GTK_BOX (page), 24);
text = _("Please select the information "
"that you would like to import:");
widget = gtk_label_new (text);
gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.5);
gtk_box_pack_start (GTK_BOX (page), widget, FALSE, FALSE, 0);
gtk_widget_show (widget);
widget = gtk_grid_new ();
gtk_grid_set_row_spacing (GTK_GRID (widget), 12);
gtk_grid_set_column_spacing (GTK_GRID (widget), 12);
gtk_box_pack_start (GTK_BOX (page), widget, FALSE, FALSE, 0);
gtk_widget_show (widget);
container = widget;
list = page->priv->available_importers;
for (link = list; link != NULL; link = link->next) {
EImportImporter *importer = link->data;
gchar *from_text;
widget = e_import_get_widget (
page->priv->import,
page->priv->import_target, importer);
if (widget == NULL)
continue;
gtk_grid_attach (GTK_GRID (container), widget, 1, row, 1, 1);
gtk_widget_show (widget);
from_text = g_strdup_printf (_("From %s:"), importer->name);
widget = gtk_label_new (from_text);
gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.0);
gtk_grid_attach (GTK_GRID (container), widget, 0, row, 1, 1);
gtk_widget_show (widget);
row++;
}
}
static void
e_mail_config_import_page_class_init (EMailConfigImportPageClass *class)
{
GObjectClass *object_class;
g_type_class_add_private (
class, sizeof (EMailConfigImportPagePrivate));
object_class = G_OBJECT_CLASS (class);
object_class->dispose = mail_config_import_page_dispose;
object_class->constructed = mail_config_import_page_constructed;
}
static void
e_mail_config_import_page_class_finalize (EMailConfigImportPageClass *class)
{
}
static void
e_mail_config_import_page_interface_init (EMailConfigPageInterface *interface)
{
interface->title = _("Importing Files");
interface->sort_order = E_MAIL_CONFIG_IMPORT_PAGE_SORT_ORDER;
}
static void
e_mail_config_import_page_init (EMailConfigImportPage *page)
{
page->priv = E_MAIL_CONFIG_IMPORT_PAGE_GET_PRIVATE (page);
page->priv->import =
e_import_new ("org.gnome.evolution.shell.importer");
page->priv->import_target = (EImportTarget *)
e_import_target_new_home (page->priv->import);
page->priv->available_importers = e_import_get_importers (
page->priv->import, page->priv->import_target);
}
void
e_mail_config_import_page_type_register (GTypeModule *type_module)
{
/* XXX G_DEFINE_DYNAMIC_TYPE declares a static type registration
* function, so we have to wrap it with a public function in
* order to register types from a separate compilation unit. */
e_mail_config_import_page_register_type (type_module);
}
EMailConfigPage *
e_mail_config_import_page_new (void)
{
return g_object_new (E_TYPE_MAIL_CONFIG_IMPORT_PAGE, NULL);
}
guint
e_mail_config_import_page_get_n_importers (EMailConfigImportPage *page)
{
g_return_val_if_fail (E_IS_MAIL_CONFIG_IMPORT_PAGE (page), 0);
return g_slist_length (page->priv->available_importers);
}
void
e_mail_config_import_page_import (EMailConfigImportPage *page,
EActivity *activity,
GAsyncReadyCallback callback,
gpointer user_data)
{
GSimpleAsyncResult *simple;
AsyncContext *async_context;
GCancellable *cancellable;
EImportImporter *first_importer;
GSList *list, *link;
g_return_if_fail (E_IS_MAIL_CONFIG_IMPORT_PAGE (page));
g_return_if_fail (E_IS_ACTIVITY (activity));
cancellable = e_activity_get_cancellable (activity);
async_context = g_slice_new0 (AsyncContext);
async_context->page = g_object_ref (page);
async_context->activity = g_object_ref (activity);
list = page->priv->available_importers;
for (link = list; link != NULL; link = g_slist_next (link)) {
EImportImporter *importer = link->data;
g_queue_push_tail (&async_context->pending_importers, importer);
}
if (G_IS_CANCELLABLE (cancellable)) {
async_context->cancellable = g_object_ref (cancellable);
async_context->cancel_id = g_cancellable_connect (
cancellable,
G_CALLBACK (mail_config_import_page_cancelled),
async_context, (GDestroyNotify) NULL);
}
simple = g_simple_async_result_new (
G_OBJECT (page), callback, user_data,
e_mail_config_import_page_import);
g_simple_async_result_set_op_res_gpointer (
simple, async_context, (GDestroyNotify) async_context_free);
/* Start the first importer. */
first_importer = g_queue_peek_head (&async_context->pending_importers);
if (first_importer != NULL)
e_import_import (
async_context->page->priv->import,
async_context->page->priv->import_target,
first_importer,
(EImportStatusFunc) mail_config_import_page_status,
(EImportCompleteFunc) mail_config_import_page_complete,
simple);
else
g_simple_async_result_complete_in_idle (simple);
}
gboolean
e_mail_config_import_page_import_finish (EMailConfigImportPage *page,
GAsyncResult *result,
GError **error)
{
GSimpleAsyncResult *simple;
g_return_val_if_fail (
g_simple_async_result_is_valid (
result, G_OBJECT (page),
e_mail_config_import_page_import), FALSE);
simple = G_SIMPLE_ASYNC_RESULT (result);
/* Assume success unless a GError is set. */
return !g_simple_async_result_propagate_error (simple, error);
}