/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ /* e-addressbook-selector.c * * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) * * 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., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. */ #ifdef HAVE_CONFIG_H #include #endif #include "e-addressbook-selector.h" #include #include #include #include typedef struct _MergeContext MergeContext; struct _EAddressbookSelectorPrivate { EAddressbookView *current_view; }; struct _MergeContext { EBookClient *source_client; EBookClient *target_client; EContact *current_contact; GSList *remaining_contacts; guint pending_removals; gboolean pending_adds; gint remove_from_source : 1; gint copy_done : 1; }; enum { PROP_0, PROP_CURRENT_VIEW }; static GtkTargetEntry drag_types[] = { { (gchar *) "text/x-source-vcard", 0, 0 } }; static gpointer parent_class; static void merge_context_next (MergeContext *merge_context) { GSList *list; merge_context->current_contact = NULL; if (!merge_context->remaining_contacts) return; list = merge_context->remaining_contacts; merge_context->current_contact = list->data; list = g_slist_delete_link (list, list); merge_context->remaining_contacts = list; } static MergeContext * merge_context_new (EBookClient *source_client, EBookClient *target_client, GSList *contact_list) { MergeContext *merge_context; merge_context = g_slice_new0 (MergeContext); merge_context->source_client = source_client; merge_context->target_client = target_client; merge_context->remaining_contacts = contact_list; merge_context_next (merge_context); return merge_context; } static void merge_context_free (MergeContext *merge_context) { if (merge_context->source_client != NULL) g_object_unref (merge_context->source_client); if (merge_context->target_client != NULL) g_object_unref (merge_context->target_client); g_slice_free (MergeContext, merge_context); } static void addressbook_selector_removed_cb (GObject *source_object, GAsyncResult *result, gpointer user_data) { EBookClient *book_client = E_BOOK_CLIENT (source_object); MergeContext *merge_context = user_data; GError *error = NULL; e_book_client_remove_contact_finish (book_client, result, &error); if (error != NULL) { g_warning ( "%s: Failed to remove contact: %s", G_STRFUNC, error->message); g_error_free (error); } merge_context->pending_removals--; if (merge_context->pending_adds) return; if (merge_context->pending_removals > 0) return; merge_context_free (merge_context); } static void addressbook_selector_merge_next_cb (EBookClient *book_client, const GError *error, const gchar *id, gpointer closure) { MergeContext *merge_context = closure; if (merge_context->remove_from_source && !error) { /* Remove previous contact from source. */ e_book_client_remove_contact ( merge_context->source_client, merge_context->current_contact, NULL, addressbook_selector_removed_cb, merge_context); merge_context->pending_removals++; } g_object_unref (merge_context->current_contact); if (merge_context->remaining_contacts != NULL) { merge_context_next (merge_context); eab_merging_book_add_contact ( merge_context->target_client, merge_context->current_contact, addressbook_selector_merge_next_cb, merge_context); } else if (merge_context->pending_removals == 0) { merge_context_free (merge_context); } else merge_context->pending_adds = FALSE; } static void addressbook_selector_load_primary_source (ESourceSelector *selector) { ESourceList *source_list; ESource *source = NULL; GSList *groups; source_list = e_source_selector_get_source_list (selector); /* Dig up the first source in the source list. * XXX libedataserver should provide API for this. */ groups = e_source_list_peek_groups (source_list); while (groups != NULL) { ESourceGroup *source_group = groups->data; GSList *sources; sources = e_source_group_peek_sources (source_group); if (sources != NULL) { source = sources->data; break; } groups = g_slist_next (groups); } if (source != NULL) e_source_selector_set_primary_selection (selector, source); } static void addressbook_selector_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) { switch (property_id) { case PROP_CURRENT_VIEW: e_addressbook_selector_set_current_view ( E_ADDRESSBOOK_SELECTOR (object), g_value_get_object (value)); return; } G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } static void addressbook_selector_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) { switch (property_id) { case PROP_CURRENT_VIEW: g_value_set_object ( value, e_addressbook_selector_get_current_view ( E_ADDRESSBOOK_SELECTOR (object))); return; } G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } static void addressbook_selector_dispose (GObject *object) { EAddressbookSelectorPrivate *priv; priv = E_ADDRESSBOOK_SELECTOR (object)->priv; if (priv->current_view != NULL) { g_object_unref (priv->current_view); priv->current_view = NULL; } /* Chain up to parent's dispose() method. */ G_OBJECT_CLASS (parent_class)->dispose (object); } static void addressbook_selector_constructed (GObject *object) { ESourceSelector *selector; selector = E_SOURCE_SELECTOR (object); addressbook_selector_load_primary_source (selector); /* Chain up to parent's constructed() method. */ G_OBJECT_CLASS (parent_class)->constructed (object); } static void target_client_open_ready_cb (GObject *source_object, GAsyncResult *result, gpointer user_data) { ESource *source = E_SOURCE (source_object); MergeContext *merge_context = user_data; EClient *client = NULL; GError *error = NULL; g_return_if_fail (merge_context != NULL); e_client_utils_open_new_finish (source, result, &client, &error); if (error != NULL) { g_warn_if_fail (client == NULL); g_warning ( "%s: Failed to open targer client: %s", G_STRFUNC, error->message); g_error_free (error); } g_return_if_fail (E_IS_CLIENT (client)); merge_context->target_client = client ? E_BOOK_CLIENT (client) : NULL; if (!merge_context->target_client) { g_slist_foreach ( merge_context->remaining_contacts, (GFunc) g_object_unref, NULL); g_slist_free (merge_context->remaining_contacts); merge_context_free (merge_context); return; } eab_merging_book_add_contact ( merge_context->target_client, merge_context->current_contact, addressbook_selector_merge_next_cb, merge_context); } static gboolean addressbook_selector_data_dropped (ESourceSelector *selector, GtkSelectionData *selection_data, ESource *destination, GdkDragAction action, guint info) { EAddressbookSelectorPrivate *priv; MergeContext *merge_context; EAddressbookModel *model; EBookClient *source_client = NULL; GSList *list; const gchar *string; gboolean remove_from_source; priv = E_ADDRESSBOOK_SELECTOR (selector)->priv; g_return_val_if_fail (priv->current_view != NULL, FALSE); string = (const gchar *) gtk_selection_data_get_data (selection_data); remove_from_source = (action == GDK_ACTION_MOVE); /* XXX Function assumes both out arguments are provided. All we * care about is the contact list; source_client will be NULL. */ eab_book_and_contact_list_from_string (string, &source_client, &list); if (source_client) g_object_unref (source_client); if (list == NULL) return FALSE; model = e_addressbook_view_get_model (priv->current_view); source_client = e_addressbook_model_get_client (model); g_return_val_if_fail (E_IS_BOOK_CLIENT (source_client), FALSE); merge_context = merge_context_new (g_object_ref (source_client), NULL, list); merge_context->remove_from_source = remove_from_source; merge_context->pending_adds = TRUE; e_client_utils_open_new ( destination, E_CLIENT_SOURCE_TYPE_CONTACTS, FALSE, NULL, e_client_utils_authenticate_handler, NULL, target_client_open_ready_cb, merge_context); return TRUE; } static void addressbook_selector_class_init (EAddressbookSelectorClass *class) { GObjectClass *object_class; ESourceSelectorClass *selector_class; parent_class = g_type_class_peek_parent (class); g_type_class_add_private (class, sizeof (EAddressbookSelectorPrivate)); object_class = G_OBJECT_CLASS (class); object_class->set_property = addressbook_selector_set_property; object_class->get_property = addressbook_selector_get_property; object_class->dispose = addressbook_selector_dispose; object_class->constructed = addressbook_selector_constructed; selector_class = E_SOURCE_SELECTOR_CLASS (class); selector_class->data_dropped = addressbook_selector_data_dropped; g_object_class_install_property ( object_class, PROP_CURRENT_VIEW, g_param_spec_object ( "current-view", NULL, NULL, E_TYPE_ADDRESSBOOK_VIEW, G_PARAM_READWRITE)); } static void addressbook_selector_init (EAddressbookSelector *selector) { selector->priv = G_TYPE_INSTANCE_GET_PRIVATE ( selector, E_TYPE_ADDRESSBOOK_SELECTOR, EAddressbookSelectorPrivate); gtk_drag_dest_set ( GTK_WIDGET (selector), GTK_DEST_DEFAULT_ALL, drag_types, G_N_ELEMENTS (drag_types), GDK_ACTION_COPY | GDK_ACTION_MOVE); e_drag_dest_add_directory_targets (GTK_WIDGET (selector)); } GType e_addressbook_selector_get_type (void) { static GType type = 0; if (G_UNLIKELY (type == 0)) { const GTypeInfo type_info = { sizeof (EAddressbookSelectorClass), (GBaseInitFunc) NULL, (GBaseFinalizeFunc) NULL, (GClassInitFunc) addressbook_selector_class_init, (GClassFinalizeFunc) NULL, NULL, /* class_data */ sizeof (EAddressbookSelector), 0, /* n_preallocs */ (GInstanceInitFunc) addressbook_selector_init, NULL /* value_table */ }; type = g_type_register_static ( E_TYPE_SOURCE_SELECTOR, "EAddressbookSelector", &type_info, 0); } return type; } GtkWidget * e_addressbook_selector_new (ESourceList *source_list) { g_return_val_if_fail (E_IS_SOURCE_LIST (source_list), NULL); return g_object_new ( E_TYPE_ADDRESSBOOK_SELECTOR, "source-list", source_list, NULL); } EAddressbookView * e_addressbook_selector_get_current_view (EAddressbookSelector *selector) { g_return_val_if_fail (E_IS_ADDRESSBOOK_SELECTOR (selector), NULL); return selector->priv->current_view; } void e_addressbook_selector_set_current_view (EAddressbookSelector *selector, EAddressbookView *current_view) { /* XXX This is only needed for moving contacts via drag-and-drop. * The selection data doesn't include the source of the data * (the model for the currently selected address book view), * so we have to rely on it being provided to us. I would * be happy to see this function go away. */ g_return_if_fail (E_IS_ADDRESSBOOK_SELECTOR (selector)); if (current_view != NULL) g_return_if_fail (E_IS_ADDRESSBOOK_VIEW (current_view)); if (selector->priv->current_view != NULL) { g_object_unref (selector->priv->current_view); selector->priv->current_view = NULL; } if (current_view != NULL) g_object_ref (current_view); selector->priv->current_view = current_view; g_object_notify (G_OBJECT (selector), "current-view"); }