aboutsummaryrefslogblamecommitdiffstats
path: root/addressbook/gui/merging/eab-contact-merging.c
blob: 37f147eaa44369486bae36e2a8d4e29532af1686 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
















                                                                           

                             










                                                        
                     




                              







































                                                                                                                           




                                           
                                    












                                                                               

                           










                                                                   

                           


































                                                                                              


                                       

















                                                                                                                    
                                                                                                                           














                                                                                         



                                                                                                      





















                                                                 
                             
 
                            










                                                                   







                                                  
                                                      
 
                            















                                                                           
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
 * Code for checking for duplicates when doing EContact work.
 *
 * Authors:
 *   Christopher James Lahey <clahey@ximian.com>
 *   Chris Toshok <toshok@ximian.com>
 *
 * Copyright (C) 2001, 2002, 2003, Ximian, Inc.
 */

#include <config.h>

#include "eab-contact-merging.h"
#include "eab-contact-compare.h"
#include <glade/glade.h>
#include <gtk/gtksignal.h>
#include <gtk/gtkdialog.h>
#include <gtk/gtkcontainer.h>
#include "addressbook/gui/widgets/eab-contact-display.h"

typedef enum {
    E_CONTACT_MERGING_ADD,
    E_CONTACT_MERGING_COMMIT
} EContactMergingOpType;

typedef struct {
    EContactMergingOpType op;
    EBook *book;
    EContact *contact;
    GList *avoid;
    EBookIdCallback id_cb;
    EBookCallback   cb;
    gpointer closure;
} EContactMergingLookup;

static void match_query_callback (EContact *contact, EContact *match, EABContactMatchType type, gpointer closure);

#define SIMULTANEOUS_MERGING_REQUESTS 20

static GList *merging_queue = NULL;
static int running_merge_requests = 0;


static void
add_lookup (EContactMergingLookup *lookup)
{
    if (running_merge_requests < SIMULTANEOUS_MERGING_REQUESTS) {
        running_merge_requests++;
        eab_contact_locate_match_full (lookup->book, lookup->contact, lookup->avoid, match_query_callback, lookup);
    }
    else {
        merging_queue = g_list_append (merging_queue, lookup);
    }
}

static void
finished_lookup (void)
{
    running_merge_requests--;

    while (running_merge_requests < SIMULTANEOUS_MERGING_REQUESTS) {
        EContactMergingLookup *lookup;
        
        if (!merging_queue)
            break;

        lookup = merging_queue->data;

        merging_queue = g_list_remove_link (merging_queue, merging_queue);

        running_merge_requests++;
        eab_contact_locate_match_full (lookup->book, lookup->contact, lookup->avoid, match_query_callback, lookup);
    }
}

static void
free_lookup (EContactMergingLookup *lookup)
{
    g_object_unref (lookup->book);
    g_object_unref (lookup->contact);
    g_list_free (lookup->avoid);

    g_free (lookup);
}

static void
final_id_cb (EBook *book, EBookStatus status, const char *id, gpointer closure)
{
    EContactMergingLookup *lookup = closure;

    if (lookup->id_cb)
        lookup->id_cb (lookup->book, status, id, lookup->closure);

    free_lookup (lookup);

    finished_lookup ();
}

static void
final_cb (EBook *book, EBookStatus status, gpointer closure)
{
    EContactMergingLookup *lookup = closure;

    if (lookup->cb)
        lookup->cb (lookup->book, status, lookup->closure);

    free_lookup (lookup);

    finished_lookup ();
}

static void
doit (EContactMergingLookup *lookup)
{
    if (lookup->op == E_CONTACT_MERGING_ADD)
        e_book_async_add_contact (lookup->book, lookup->contact, final_id_cb, lookup);
    else if (lookup->op == E_CONTACT_MERGING_COMMIT)
        e_book_async_commit_contact (lookup->book, lookup->contact, final_cb, lookup);
}

static void
cancelit (EContactMergingLookup *lookup)
{
    if (lookup->op == E_CONTACT_MERGING_ADD) {
        if (lookup->id_cb)
            final_id_cb (lookup->book, E_BOOK_ERROR_CANCELLED, NULL, lookup);
    } else if (lookup->op == E_CONTACT_MERGING_COMMIT) {
        if (lookup->cb)
            final_cb (lookup->book, E_BOOK_ERROR_CANCELLED, lookup);
    }
}

static void
response (GtkWidget *dialog, int response, EContactMergingLookup *lookup)
{
    gtk_widget_destroy (dialog);

    switch (response) {
    case 0:
        doit (lookup);
        break;
    case 1:
        cancelit (lookup);
        break;
    case GTK_RESPONSE_DELETE_EVENT:
        cancelit (lookup);
        break;
    }
}

static void
match_query_callback (EContact *contact, EContact *match, EABContactMatchType type, gpointer closure)
{
    EContactMergingLookup *lookup = closure;

    if ((gint) type <= (gint) EAB_CONTACT_MATCH_VAGUE) {
        doit (lookup);
    } else {
        GladeXML *ui;
        
        GtkWidget *widget;

        if (lookup->op == E_CONTACT_MERGING_ADD)
            ui = glade_xml_new (EVOLUTION_GLADEDIR "/eab-contact-duplicate-detected.glade", NULL, NULL);
        else if (lookup->op == E_CONTACT_MERGING_COMMIT)
            ui = glade_xml_new (EVOLUTION_GLADEDIR "/eab-contact-commit-duplicate-detected.glade", NULL, NULL);
        else {
            doit (lookup);
            return;
        }

        widget = glade_xml_get_widget (ui, "custom-old-contact");
        eab_contact_display_render (EAB_CONTACT_DISPLAY (widget),
                        match, EAB_CONTACT_DISPLAY_RENDER_COMPACT);

        widget = glade_xml_get_widget (ui, "custom-new-contact");
        eab_contact_display_render (EAB_CONTACT_DISPLAY (widget),
                        contact, EAB_CONTACT_DISPLAY_RENDER_COMPACT);

        widget = glade_xml_get_widget (ui, "dialog-duplicate-contact");

        gtk_widget_ensure_style (widget);
        gtk_container_set_border_width (GTK_CONTAINER (GTK_DIALOG (widget)->vbox), 0);
        gtk_container_set_border_width (GTK_CONTAINER (GTK_DIALOG (widget)->action_area), 12);

        g_signal_connect (widget, "response",
                  G_CALLBACK (response), lookup);

        gtk_widget_show_all (widget);
    }
}

gboolean
eab_merging_book_add_contact (EBook           *book,
                  EContact           *contact,
                  EBookIdCallback  cb,
                  gpointer         closure)
{
    EContactMergingLookup *lookup;

    lookup = g_new (EContactMergingLookup, 1);

    lookup->op = E_CONTACT_MERGING_ADD;
    lookup->book = g_object_ref (book);
    lookup->contact = g_object_ref (contact);
    lookup->id_cb = cb;
    lookup->closure = closure;
    lookup->avoid = NULL;

    add_lookup (lookup);

    return TRUE;
}

gboolean
eab_merging_book_commit_contact (EBook                 *book,
                 EContact                 *contact,
                 EBookCallback          cb,
                 gpointer               closure)
{
    EContactMergingLookup *lookup;
    
    lookup = g_new (EContactMergingLookup, 1);

    lookup->op = E_CONTACT_MERGING_COMMIT;
    lookup->book = g_object_ref (book);
    lookup->contact = g_object_ref (contact);
    lookup->cb = cb;
    lookup->closure = closure;
    lookup->avoid = g_list_append (NULL, contact);

    add_lookup (lookup);

    return TRUE;
}

GtkWidget *
_eab_contact_merging_create_contact_display(gchar *name,
                        gchar *string1, gchar *string2,
                        gint int1, gint int2);

GtkWidget *
_eab_contact_merging_create_contact_display(gchar *name,
                        gchar *string1, gchar *string2,
                        gint int1, gint int2)
{
    return eab_contact_display_new();
}