aboutsummaryrefslogtreecommitdiffstats
path: root/addressbook/util
diff options
context:
space:
mode:
Diffstat (limited to 'addressbook/util')
-rw-r--r--addressbook/util/.cvsignore4
-rw-r--r--addressbook/util/Makefile.am41
-rw-r--r--addressbook/util/eab-book-util.c293
-rw-r--r--addressbook/util/eab-book-util.h69
-rw-r--r--addressbook/util/eab-destination.c1569
-rw-r--r--addressbook/util/eab-destination.h128
-rw-r--r--addressbook/util/eab-marshal.list5
7 files changed, 2109 insertions, 0 deletions
diff --git a/addressbook/util/.cvsignore b/addressbook/util/.cvsignore
new file mode 100644
index 0000000000..6205c6f3ab
--- /dev/null
+++ b/addressbook/util/.cvsignore
@@ -0,0 +1,4 @@
+Makefile
+Makefile.in
+eab-marshal.c
+eab-marshal.h
diff --git a/addressbook/util/Makefile.am b/addressbook/util/Makefile.am
new file mode 100644
index 0000000000..51b6e7fc28
--- /dev/null
+++ b/addressbook/util/Makefile.am
@@ -0,0 +1,41 @@
+INCLUDES = \
+ -DPREFIX=\"$(prefix)\" \
+ -DSYSCONFDIR=\"$(sysconfdir)\" \
+ -DDATADIR=\"$(datadir)\" \
+ -DLIBDIR=\"$(libdir)\" \
+ -DG_LOG_DOMAIN=\"EBook\" \
+ -I$(top_srcdir) \
+ -I$(top_srcdir)/camel \
+ -I$(top_srcdir)/addressbook/backend \
+ -I$(top_srcdir)/addressbook/ename \
+ -I$(top_builddir)/addressbook/backend \
+ -I$(top_builddir)/addressbook/ename \
+ -I$(top_builddir)/shell \
+ -I$(top_srcdir)/shell \
+ -DG_DISABLE_DEPRECATED \
+ -DLIBGNOME_DISABLE_DEPRECATED \
+ $(EVOLUTION_ADDRESSBOOK_CFLAGS)
+
+noinst_LTLIBRARIES = libeabutil.la
+
+libeabutil_la_SOURCES = \
+ eab-marshal.c \
+ eab-destination.c \
+ eab-destination.h \
+ eab-book-util.c \
+ eab-book-util.h
+
+libeabutil_la_LIBADD = \
+ $(top_builddir)/addressbook/backend/ebook/libebook.la \
+ $(top_builddir)/camel/libcamel.la \
+ $(top_builddir)/e-util/ename/libename.la \
+ $(top_builddir)/e-util/libeutil.la
+
+MARSHAL_GENERATED = eab-marshal.c eab-marshal.h
+@EVO_MARSHAL_RULE@
+
+BUILT_SOURCES = $(MARSHAL_GENERATED)
+CLEANFILES = $(BUILT_SOURCES)
+
+dist-hook:
+ cd $(distdir); rm -f $(BUILT_SOURCES)
diff --git a/addressbook/util/eab-book-util.c b/addressbook/util/eab-book-util.c
new file mode 100644
index 0000000000..2c80d6f8a1
--- /dev/null
+++ b/addressbook/util/eab-book-util.c
@@ -0,0 +1,293 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+
+/*
+ * eab-util.c
+ *
+ * Copyright (C) 2001-2003 Ximian, Inc.
+ *
+ * Authors: Jon Trowbridge <trow@ximian.com>
+ * Chris Toshok <toshok@ximian.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., 59 Temple Place, Suite 330, Boston, MA 02111-1307
+ * USA.
+ */
+
+#include <config.h>
+#include "eab-book-util.h"
+
+#include <string.h>
+#include <glib.h>
+#include <glib-object.h>
+#include <e-util/e-config-listener.h>
+
+EConfigListener *
+eab_get_config_database ()
+{
+ static EConfigListener *config_db;
+
+ if (config_db == NULL)
+ config_db = e_config_listener_new ();
+
+ return config_db;
+}
+
+/*
+ *
+ * Specialized Queries
+ *
+ */
+
+guint
+eab_name_and_email_query (EBook *book,
+ const gchar *name,
+ const gchar *email,
+ EBookContactsCallback cb,
+ gpointer closure)
+{
+ gchar *email_query=NULL, *name_query=NULL, *query;
+ guint tag;
+
+ g_return_val_if_fail (book && E_IS_BOOK (book), 0);
+ g_return_val_if_fail (cb != NULL, 0);
+
+ if (name && !*name)
+ name = NULL;
+ if (email && !*email)
+ email = NULL;
+
+ if (name == NULL && email == NULL)
+ return 0;
+
+ /* Build our e-mail query.
+ * We only query against the username part of the address, to avoid not matching
+ * fred@foo.com and fred@mail.foo.com. While their may be namespace collisions
+ * in the usernames of everyone out there, it shouldn't be that bad. (Famous last words.)
+ */
+ if (email) {
+ const gchar *t = email;
+ while (*t && *t != '@')
+ ++t;
+ if (*t == '@') {
+ email_query = g_strdup_printf ("(beginswith \"email\" \"%.*s@\")", t-email, email);
+
+ } else {
+ email_query = g_strdup_printf ("(beginswith \"email\" \"%s\")", email);
+ }
+ }
+
+ /* Build our name query.
+ * We only do name-query stuff if we don't have an e-mail address. Our basic assumption
+ * is that the username part of the email is good enough to keep the amount of stuff returned
+ * in the query relatively small.
+ */
+ if (name && !email)
+ name_query = g_strdup_printf ("(or (beginswith \"file_as\" \"%s\") (beginswith \"full_name\" \"%s\"))", name, name);
+
+ /* Assemble our e-mail & name queries */
+ if (email_query && name_query) {
+ query = g_strdup_printf ("(and %s %s)", email_query, name_query);
+ } else if (email_query) {
+ query = email_query;
+ email_query = NULL;
+ } else if (name_query) {
+ query = name_query;
+ name_query = NULL;
+ } else
+ return 0;
+
+ tag = e_book_async_get_contacts (book, query, cb, closure);
+
+ g_free (email_query);
+ g_free (name_query);
+ g_free (query);
+
+ return tag;
+}
+
+/*
+ * Simple nickname query
+ */
+guint
+eab_nickname_query (EBook *book,
+ const char *nickname,
+ EBookContactsCallback cb,
+ gpointer closure)
+{
+ gchar *query;
+ guint retval;
+
+ g_return_val_if_fail (E_IS_BOOK (book), 0);
+ g_return_val_if_fail (nickname != NULL, 0);
+
+ /* The empty-string case shouldn't generate a warning. */
+ if (! *nickname)
+ return 0;
+
+ query = g_strdup_printf ("(is \"nickname\" \"%s\")", nickname);
+
+ retval = e_book_async_get_contacts (book, query, cb, closure);
+
+ g_free (query);
+
+ return retval;
+}
+
+GList*
+eab_contact_list_from_string (const char *str)
+{
+ GList *contacts = NULL;
+ GString *gstr = g_string_new ("");
+ char *p = (char*)str;
+ char *q;
+ char *blank_line;
+
+ while (*p) {
+ if (*p != '\r') g_string_append_c (gstr, *p);
+
+ p++;
+ }
+
+ p = g_string_free (gstr, FALSE);
+ q = p;
+ do {
+ char *temp;
+
+ blank_line = strstr (q, "\n\n");
+ if (blank_line) {
+ temp = g_strndup (q, blank_line - q);
+ }
+ else {
+ temp = g_strdup (q);
+ }
+
+ contacts = g_list_append (contacts, e_contact_new_from_vcard (temp));
+
+ g_free (temp);
+
+ if (blank_line)
+ q = blank_line + 2;
+ else
+ q = NULL;
+ } while (blank_line);
+
+ g_free (p);
+
+ return contacts;
+}
+
+char*
+eab_contact_list_to_string (GList *contacts)
+{
+ GString *str = g_string_new ("");
+ GList *l;
+
+ for (l = contacts; l; l = l->next) {
+ EContact *contact = l->data;
+ char *vcard_str = e_vcard_to_string (E_VCARD (contact), EVC_FORMAT_VCARD_30);
+
+ g_string_append (str, vcard_str);
+ if (l->next)
+ g_string_append (str, "\r\n");
+ }
+
+ return g_string_free (str, FALSE);
+}
+
+#if notyet
+/*
+ * Convenience routine to check for addresses in the local address book.
+ */
+
+typedef struct _HaveAddressInfo HaveAddressInfo;
+struct _HaveAddressInfo {
+ gchar *email;
+ EBookHaveAddressCallback cb;
+ gpointer closure;
+};
+
+static void
+have_address_query_cb (EBook *book, EBookSimpleQueryStatus status, const GList *contacts, gpointer closure)
+{
+ HaveAddressInfo *info = (HaveAddressInfo *) closure;
+
+ info->cb (book,
+ info->email,
+ contacts && (status == E_BOOK_ERROR_OK) ? E_CONTACT (contacts->data) : NULL,
+ info->closure);
+
+ g_free (info->email);
+ g_free (info);
+}
+
+static void
+have_address_book_open_cb (EBook *book, gpointer closure)
+{
+ HaveAddressInfo *info = (HaveAddressInfo *) closure;
+
+ if (book) {
+
+ e_book_name_and_email_query (book, NULL, info->email, have_address_query_cb, info);
+
+ } else {
+
+ info->cb (NULL, info->email, NULL, info->closure);
+
+ g_free (info->email);
+ g_free (info);
+
+ }
+}
+
+void
+eab_query_address_default (const gchar *email,
+ EABHaveAddressCallback cb,
+ gpointer closure)
+{
+ HaveAddressInfo *info;
+
+ g_return_if_fail (email != NULL);
+ g_return_if_fail (cb != NULL);
+
+ info = g_new0 (HaveAddressInfo, 1);
+ info->email = g_strdup (email);
+ info->cb = cb;
+ info->closure = closure;
+
+ e_book_use_default_book (have_address_book_open_cb, info);
+}
+#endif
+
+/* bad place for this i know. */
+int
+e_utf8_casefold_collate_len (const gchar *str1, const gchar *str2, int len)
+{
+ gchar *s1 = g_utf8_casefold(str1, len);
+ gchar *s2 = g_utf8_casefold(str2, len);
+ int rv;
+
+ rv = g_utf8_collate (s1, s2);
+
+ g_free (s1);
+ g_free (s2);
+
+ return rv;
+}
+
+int
+e_utf8_casefold_collate (const gchar *str1, const gchar *str2)
+{
+ return e_utf8_casefold_collate_len (str1, str2, -1);
+}
diff --git a/addressbook/util/eab-book-util.h b/addressbook/util/eab-book-util.h
new file mode 100644
index 0000000000..be5451b795
--- /dev/null
+++ b/addressbook/util/eab-book-util.h
@@ -0,0 +1,69 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+
+/*
+ * e-book-util.h
+ *
+ * Copyright (C) 2001-2003 Ximian, Inc.
+ *
+ * Authors: Jon Trowbridge <trow@ximian.com>
+ * Chris Toshok <toshok@ximian.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., 59 Temple Place, Suite 330, Boston, MA 02111-1307
+ * USA.
+ */
+
+#ifndef __EAB_UTIL_H__
+#define __EAB_UTIL_H__
+
+#include "ebook/e-book-async.h"
+#include "e-util/e-config-listener.h"
+#include <bonobo/bonobo-object.h>
+#include <bonobo/bonobo-moniker-util.h>
+
+G_BEGIN_DECLS
+
+typedef void (*EABHaveAddressCallback) (EBook *book, const gchar *addr, EContact *contact, gpointer closure);
+
+/* config database interface. */
+EConfigListener *eab_get_config_database (void);
+
+/* Specialized Name/Email Queries */
+guint eab_name_and_email_query (EBook *book,
+ const char *name,
+ const char *email,
+ EBookContactsCallback cb,
+ gpointer closure);
+guint eab_nickname_query (EBook *book,
+ const char *nickname,
+ EBookContactsCallback cb,
+ gpointer closure);
+
+GList *eab_contact_list_from_string (const char *str);
+char *eab_contact_list_to_string (GList *contacts);
+
+/* Returns the EContact associated to email in the callback,
+ or NULL if no match is found in the default address book. */
+void eab_query_address_default (const gchar *email,
+ EABHaveAddressCallback cb,
+ gpointer closure);
+
+int e_utf8_casefold_collate_len (const gchar *str1, const gchar *str2, int len);
+int e_utf8_casefold_collate (const gchar *str1, const gchar *str2);
+
+G_END_DECLS
+
+#endif /* __EAB_UTIL_H__ */
+
diff --git a/addressbook/util/eab-destination.c b/addressbook/util/eab-destination.c
new file mode 100644
index 0000000000..d83ad8563a
--- /dev/null
+++ b/addressbook/util/eab-destination.c
@@ -0,0 +1,1569 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+
+/*
+ * eab-destination.c
+ *
+ * Copyright (C) 2001-2003 Ximian, Inc.
+ *
+ * Authors: Jon Trowbridge <trow@ximian.com>
+ * Chris Toshok <toshok@ximian.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., 59 Temple Place, Suite 330, Boston, MA 02111-1307
+ * USA.
+ */
+
+#include <config.h>
+#include "eab-destination.h"
+
+#include <stdlib.h>
+#include <ctype.h>
+#include <string.h>
+#include "ebook/e-book.h"
+#include "eab-marshal.h"
+#include "eab-book-util.h"
+#include <gal/widgets/e-unicode.h>
+
+#include <glib.h>
+#include <libxml/xmlmemory.h>
+#include <camel/camel-internet-address.h>
+
+#define d(x)
+
+enum {
+ CHANGED,
+ CONTACT_LOADED,
+ LAST_SIGNAL
+};
+
+guint eab_destination_signals[LAST_SIGNAL] = { 0 };
+
+struct _EABDestinationPrivate {
+ gchar *raw;
+
+ gchar *book_uri;
+ gchar *uid;
+ EContact *contact;
+ gint email_num;
+
+ gchar *name;
+ gchar *email;
+ gchar *addr;
+ gchar *textrep;
+
+ GList *list_dests;
+
+ guint html_mail_override : 1;
+ guint wants_html_mail : 1;
+
+ guint show_addresses : 1;
+
+ guint contact_loaded : 1;
+ guint cannot_load : 1;
+ guint auto_recipient : 1;
+ guint pending_contact_load;
+
+ guint pending_change : 1;
+
+ EBook *book;
+
+ gint freeze_count;
+};
+
+static void eab_destination_clear_contact (EABDestination *);
+static void eab_destination_clear_strings (EABDestination *);
+
+/* the following prototypes were in e-destination.h, but weren't used
+ by anything in evolution... let's make them private for now. */
+static gboolean eab_destination_is_valid (const EABDestination *);
+static void eab_destination_set_contact_uid (EABDestination *, const gchar *uid, gint email_num);
+static void eab_destination_set_book_uri (EABDestination *, const gchar *uri);
+static gboolean eab_destination_from_contact (const EABDestination *);
+static const gchar *eab_destination_get_book_uri (const EABDestination *);
+static const gchar *eab_destination_get_contact_uid (const EABDestination *);
+static xmlNodePtr eab_destination_xml_encode (const EABDestination *dest);
+static gboolean eab_destination_xml_decode (EABDestination *dest, xmlNodePtr node);
+
+static GObjectClass *parent_class;
+
+static void
+eab_destination_dispose (GObject *obj)
+{
+ EABDestination *dest = EAB_DESTINATION (obj);
+
+ if (dest->priv) {
+ eab_destination_clear (dest);
+
+ if (dest->priv->book)
+ g_object_unref (dest->priv->book);
+
+ g_free (dest->priv);
+ dest->priv = NULL;
+ }
+
+ if (G_OBJECT_CLASS (parent_class)->dispose)
+ (* G_OBJECT_CLASS (parent_class)->dispose) (obj);
+}
+
+static void
+eab_destination_class_init (EABDestinationClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ parent_class = g_type_class_ref (G_TYPE_OBJECT);
+
+ object_class->dispose = eab_destination_dispose;
+
+ eab_destination_signals[CHANGED] =
+ g_signal_new ("changed",
+ G_OBJECT_CLASS_TYPE (object_class),
+ G_SIGNAL_RUN_LAST,
+ G_STRUCT_OFFSET (EABDestinationClass, changed),
+ NULL, NULL,
+ eab_marshal_NONE__NONE,
+ G_TYPE_NONE, 0);
+
+ eab_destination_signals[CONTACT_LOADED] =
+ g_signal_new ("contact_loaded",
+ G_OBJECT_CLASS_TYPE (object_class),
+ G_SIGNAL_RUN_LAST,
+ G_STRUCT_OFFSET (EABDestinationClass, contact_loaded),
+ NULL, NULL,
+ eab_marshal_NONE__NONE,
+ G_TYPE_NONE, 0);
+}
+
+static void
+eab_destination_init (EABDestination *dest)
+{
+ dest->priv = g_new0 (struct _EABDestinationPrivate, 1);
+
+ dest->priv->cannot_load = FALSE;
+ dest->priv->auto_recipient = FALSE;
+ dest->priv->pending_contact_load = 0;
+}
+
+GType
+eab_destination_get_type (void)
+{
+ static GType dest_type = 0;
+
+ if (!dest_type) {
+ GTypeInfo dest_info = {
+ sizeof (EABDestinationClass),
+ NULL, /* base_class_init */
+ NULL, /* base_class_finalize */
+ (GClassInitFunc) eab_destination_class_init,
+ NULL, /* class_finalize */
+ NULL, /* class_data */
+ sizeof (EABDestination),
+ 0, /* n_preallocs */
+ (GInstanceInitFunc) eab_destination_init
+ };
+
+ dest_type = g_type_register_static (G_TYPE_OBJECT, "EABDestination", &dest_info, 0);
+ }
+
+ return dest_type;
+}
+
+EABDestination *
+eab_destination_new (void)
+{
+ return g_object_new (EAB_TYPE_DESTINATION, NULL);
+}
+
+static void
+eab_destination_freeze (EABDestination *dest)
+{
+ g_return_if_fail (EAB_IS_DESTINATION (dest));
+ g_return_if_fail (dest->priv->freeze_count >= 0);
+
+ dest->priv->freeze_count++;
+}
+
+static void
+eab_destination_thaw (EABDestination *dest)
+{
+ g_return_if_fail (EAB_IS_DESTINATION (dest));
+ g_return_if_fail (dest->priv->freeze_count > 0);
+
+ dest->priv->freeze_count--;
+ if (dest->priv->freeze_count == 0 && dest->priv->pending_change)
+ eab_destination_changed (dest);
+}
+
+void
+eab_destination_changed (EABDestination *dest)
+{
+ if (dest->priv->freeze_count == 0) {
+ g_signal_emit (dest, eab_destination_signals[CHANGED], 0);
+ dest->priv->pending_change = FALSE;
+ dest->priv->cannot_load = FALSE;
+
+ } else {
+ dest->priv->pending_change = TRUE;
+ }
+}
+
+EABDestination *
+eab_destination_copy (const EABDestination *dest)
+{
+ EABDestination *new_dest;
+ GList *iter;
+
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ new_dest = eab_destination_new ();
+
+ new_dest->priv->book_uri = g_strdup (dest->priv->book_uri);
+ new_dest->priv->uid = g_strdup (dest->priv->uid);
+ new_dest->priv->name = g_strdup (dest->priv->name);
+ new_dest->priv->email = g_strdup (dest->priv->email);
+ new_dest->priv->addr = g_strdup (dest->priv->addr);
+ new_dest->priv->email_num = dest->priv->email_num;
+
+ new_dest->priv->contact = dest->priv->contact;
+ if (new_dest->priv->contact)
+ g_object_ref (new_dest->priv->contact);
+
+ new_dest->priv->html_mail_override = dest->priv->html_mail_override;
+ new_dest->priv->wants_html_mail = dest->priv->wants_html_mail;
+
+ for (iter = dest->priv->list_dests; iter != NULL; iter = g_list_next (iter)) {
+ new_dest->priv->list_dests = g_list_append (new_dest->priv->list_dests,
+ eab_destination_copy (EAB_DESTINATION (iter->data)));
+ }
+
+ return new_dest;
+}
+
+static void
+eab_destination_clear_contact (EABDestination *dest)
+{
+ g_free (dest->priv->book_uri);
+ dest->priv->book_uri = NULL;
+ g_free (dest->priv->uid);
+ dest->priv->uid = NULL;
+
+ dest->priv->contact = NULL;
+ dest->priv->email_num = -1;
+
+ g_list_foreach (dest->priv->list_dests, (GFunc) g_object_unref, NULL);
+ g_list_free (dest->priv->list_dests);
+ dest->priv->list_dests = NULL;
+
+ dest->priv->cannot_load = FALSE;
+
+ eab_destination_cancel_contact_load (dest);
+
+ eab_destination_changed (dest);
+}
+
+static void
+eab_destination_clear_strings (EABDestination *dest)
+{
+ g_free (dest->priv->raw);
+ dest->priv->raw = NULL;
+
+ g_free (dest->priv->name);
+ dest->priv->name = NULL;
+
+ g_free (dest->priv->email);
+ dest->priv->email = NULL;
+
+ g_free (dest->priv->addr);
+ dest->priv->addr = NULL;
+
+ g_free (dest->priv->textrep);
+ dest->priv->textrep = NULL;
+
+ eab_destination_changed (dest);
+}
+
+void
+eab_destination_clear (EABDestination *dest)
+{
+ g_return_if_fail (dest && EAB_IS_DESTINATION (dest));
+
+ eab_destination_freeze (dest);
+
+ eab_destination_clear_contact (dest);
+ eab_destination_clear_strings (dest);
+
+ eab_destination_thaw (dest);
+}
+
+static gboolean
+nonempty (const gchar *s)
+{
+ gunichar c;
+ while (*s) {
+ c = g_utf8_get_char (s);
+ if (!g_unichar_isspace (c))
+ return TRUE;
+ s = g_utf8_next_char (s);
+ }
+ return FALSE;
+}
+
+gboolean
+eab_destination_is_empty (const EABDestination *dest)
+
+{
+ struct _EABDestinationPrivate *p;
+
+ g_return_val_if_fail (EAB_IS_DESTINATION (dest), TRUE);
+
+ p = dest->priv;
+
+ return !(p->contact != NULL
+ || (p->book_uri && *p->book_uri)
+ || (p->uid && *p->uid)
+ || (p->raw && nonempty (p->raw))
+ || (p->name && nonempty (p->name))
+ || (p->email && nonempty (p->email))
+ || (p->addr && nonempty (p->addr))
+ || (p->list_dests != NULL));
+}
+
+gboolean
+eab_destination_is_valid (const EABDestination *dest)
+{
+ const char *email;
+
+ g_return_val_if_fail (EAB_IS_DESTINATION (dest), FALSE);
+
+ if (eab_destination_from_contact (dest))
+ return TRUE;
+
+ email = eab_destination_get_email (dest);
+
+ /* FIXME: if we really wanted to get fancy here, we could
+ check to make sure that the address was valid according to
+ rfc822's addr-spec grammar. */
+
+ return email && *email && strchr (email, '@');
+}
+
+gboolean
+eab_destination_equal (const EABDestination *a, const EABDestination *b)
+{
+ const struct _EABDestinationPrivate *pa, *pb;
+ const char *na, *nb;
+
+ g_return_val_if_fail (EAB_IS_DESTINATION (a), FALSE);
+ g_return_val_if_fail (EAB_IS_DESTINATION (b), FALSE);
+
+ if (a == b)
+ return TRUE;
+
+ pa = a->priv;
+ pb = b->priv;
+
+ /* Check equality of contacts. */
+ if (pa->contact || pb->contact) {
+ if (! (pa->contact && pb->contact))
+ return FALSE;
+
+ if (pa->contact == pb->contact || !strcmp (e_contact_get_const (pa->contact, E_CONTACT_UID),
+ e_contact_get_const (pb->contact, E_CONTACT_UID)))
+ return TRUE;
+
+ return FALSE;
+ }
+
+ /* Just in case name returns NULL */
+ na = eab_destination_get_name (a);
+ nb = eab_destination_get_name (b);
+ if ((na || nb) && !(na && nb && ! e_utf8_casefold_collate (na, nb)))
+ return FALSE;
+
+ if (!g_ascii_strcasecmp (eab_destination_get_email (a), eab_destination_get_email (b)))
+ return TRUE;
+ else
+ return FALSE;
+}
+
+void
+eab_destination_set_contact (EABDestination *dest, EContact *contact, gint email_num)
+{
+ g_return_if_fail (dest && EAB_IS_DESTINATION (dest));
+ g_return_if_fail (contact && E_IS_CONTACT (contact));
+
+ if (dest->priv->contact != contact || dest->priv->email_num != email_num) {
+ /* We have to freeze/thaw around these operations so that the 'changed'
+ signals don't cause the EABDestination's internal state to be altered
+ before we can finish setting ->contact && ->email_num. */
+ eab_destination_freeze (dest);
+ eab_destination_clear (dest);
+
+ dest->priv->contact = contact;
+ g_object_ref (dest->priv->contact);
+
+ dest->priv->email_num = email_num;
+
+ eab_destination_changed (dest);
+ eab_destination_thaw (dest);
+ }
+}
+
+static void
+eab_destination_set_book_uri (EABDestination *dest, const gchar *uri)
+{
+ g_return_if_fail (dest && EAB_IS_DESTINATION (dest));
+ g_return_if_fail (uri != NULL);
+
+ if (dest->priv->book_uri == NULL || strcmp (dest->priv->book_uri, uri)) {
+ g_free (dest->priv->book_uri);
+ dest->priv->book_uri = g_strdup (uri);
+
+ eab_destination_changed (dest);
+ }
+}
+
+void
+eab_destination_set_contact_uid (EABDestination *dest, const gchar *uid, gint email_num)
+{
+ g_return_if_fail (dest && EAB_IS_DESTINATION (dest));
+ g_return_if_fail (uid != NULL);
+
+ if (dest->priv->uid == NULL
+ || strcmp (dest->priv->uid, uid)
+ || dest->priv->email_num != email_num) {
+
+ g_free (dest->priv->uid);
+ dest->priv->uid = g_strdup (uid);
+ dest->priv->email_num = email_num;
+
+ /* If we already have a contact, remove it unless it's uid matches the one
+ we just set. */
+ if (dest->priv->contact && strcmp (uid,
+ e_contact_get_const (dest->priv->contact, E_CONTACT_UID))) {
+ g_object_unref (dest->priv->contact);
+ dest->priv->contact = NULL;
+ }
+
+ eab_destination_changed (dest);
+ }
+}
+
+void
+eab_destination_set_name (EABDestination *dest, const gchar *name)
+{
+ gboolean changed = FALSE;
+
+ g_return_if_fail (EAB_IS_DESTINATION (dest));
+
+ if (name == NULL) {
+ if (dest->priv->name != NULL) {
+ g_free (dest->priv->name);
+ dest->priv->name = NULL;
+ changed = TRUE;
+ }
+ } else if (dest->priv->name == NULL || strcmp (dest->priv->name, name)) {
+ g_free (dest->priv->name);
+ dest->priv->name = g_strdup (name);
+ changed = TRUE;
+ }
+
+ if (changed) {
+ g_free (dest->priv->addr);
+ dest->priv->addr = NULL;
+ g_free (dest->priv->textrep);
+ dest->priv->textrep = NULL;
+ eab_destination_changed (dest);
+ }
+}
+
+void
+eab_destination_set_email (EABDestination *dest, const gchar *email)
+{
+ gboolean changed = FALSE;
+
+ g_return_if_fail (EAB_IS_DESTINATION (dest));
+
+ if (email == NULL) {
+ if (dest->priv->email != NULL) {
+ g_free (dest->priv->addr);
+ dest->priv->addr = NULL;
+ changed = TRUE;
+ }
+ } else if (dest->priv->email == NULL || strcmp (dest->priv->email, email)) {
+ g_free (dest->priv->email);
+ dest->priv->email = g_strdup (email);
+ changed = TRUE;
+ }
+
+ if (changed) {
+ g_free (dest->priv->addr);
+ dest->priv->addr = NULL;
+ g_free (dest->priv->textrep);
+ dest->priv->textrep = NULL;
+ eab_destination_changed (dest);
+ }
+}
+
+void
+eab_destination_set_html_mail_pref (EABDestination *dest, gboolean x)
+{
+ g_return_if_fail (dest && EAB_IS_DESTINATION (dest));
+
+ dest->priv->html_mail_override = TRUE;
+ if (dest->priv->wants_html_mail != x) {
+ dest->priv->wants_html_mail = x;
+ eab_destination_changed (dest);
+ }
+}
+
+gboolean
+eab_destination_contains_contact (const EABDestination *dest)
+{
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), FALSE);
+ return dest->priv->contact != NULL;
+}
+
+gboolean
+eab_destination_from_contact (const EABDestination *dest)
+{
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), FALSE);
+ return dest->priv->contact != NULL || dest->priv->book_uri != NULL || dest->priv->uid != NULL;
+}
+
+gboolean
+eab_destination_is_auto_recipient (const EABDestination *dest)
+{
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), FALSE);
+
+ return dest->priv->auto_recipient;
+}
+
+void
+eab_destination_set_auto_recipient (EABDestination *dest, gboolean value)
+{
+ g_return_if_fail (dest && EAB_IS_DESTINATION (dest));
+
+ dest->priv->auto_recipient = value;
+}
+
+typedef struct _UseContact UseContact;
+struct _UseContact {
+ EABDestination *dest;
+ EABDestinationContactCallback cb;
+ gpointer closure;
+};
+
+static void
+use_contact_cb (EContact *contact, gpointer closure)
+{
+ UseContact *uc = (UseContact *) closure;
+
+ if (contact != NULL && uc->dest->priv->contact == NULL) {
+ uc->dest->priv->contact = contact;
+ g_object_ref (uc->dest->priv->contact);
+ eab_destination_changed (uc->dest);
+ }
+
+ if (uc->cb) {
+ uc->cb (uc->dest, uc->dest->priv->contact, uc->closure);
+ }
+
+ /* We held a copy of the destination during the callback. */
+ g_object_unref (uc->dest);
+ g_free (uc);
+}
+
+void
+eab_destination_use_contact (EABDestination *dest, EABDestinationContactCallback cb, gpointer closure)
+{
+ g_return_if_fail (dest && EAB_IS_DESTINATION (dest));
+
+ if (dest->priv->contact != NULL) {
+ if (cb)
+ cb (dest, dest->priv->contact, closure);
+ } else if (dest->priv->book_uri != NULL && dest->priv->uid != NULL) {
+ UseContact *uc = g_new (UseContact, 1);
+
+ uc->dest = dest;
+ /* Hold a reference to the destination during the callback. */
+ g_object_ref (uc->dest);
+ uc->cb = cb;
+ uc->closure = closure;
+#if notyet
+ e_contact_load_uri (dest->priv->book_uri, dest->priv->uid, use_contact_cb, uc);
+#endif
+ } else {
+ if (cb)
+ cb (dest, NULL, closure);
+ }
+}
+
+EContact *
+eab_destination_get_contact (const EABDestination *dest)
+{
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ return dest->priv->contact;
+}
+
+const gchar *
+eab_destination_get_contact_uid (const EABDestination *dest)
+{
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ if (dest->priv->uid)
+ return dest->priv->uid;
+
+ if (dest->priv->contact)
+ return e_contact_get_const (dest->priv->contact, E_CONTACT_UID);
+
+ return NULL;
+}
+
+const gchar *
+eab_destination_get_book_uri (const EABDestination *dest)
+{
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ return dest->priv->book_uri;
+}
+
+gint
+eab_destination_get_email_num (const EABDestination *dest)
+{
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), -1);
+
+ if (dest->priv->contact == NULL && (dest->priv->book_uri == NULL || dest->priv->uid == NULL))
+ return -1;
+
+ return dest->priv->email_num;
+}
+
+const gchar *
+eab_destination_get_name (const EABDestination *dest)
+{
+ struct _EABDestinationPrivate *priv;
+
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ priv = (struct _EABDestinationPrivate *)dest->priv; /* cast out const */
+
+ if (priv->name == NULL) {
+ if (priv->contact != NULL) {
+ priv->name = e_contact_get (priv->contact, E_CONTACT_FULL_NAME);
+
+ if (priv->name == NULL || *priv->name == '\0') {
+ g_free (priv->name);
+ priv->name = e_contact_get (priv->contact, E_CONTACT_FILE_AS);
+ }
+
+ if (priv->name == NULL || *priv->name == '\0') {
+ g_free (priv->name);
+ if (e_contact_get (priv->contact, E_CONTACT_IS_LIST))
+ priv->name = g_strdup (_("Unnamed List"));
+ else
+ priv->name = g_strdup (eab_destination_get_email (dest));
+ }
+ } else if (priv->raw != NULL) {
+ CamelInternetAddress *addr = camel_internet_address_new ();
+
+ if (camel_address_unformat (CAMEL_ADDRESS (addr), priv->raw)) {
+ const char *camel_name = NULL;
+
+ camel_internet_address_get (addr, 0, &camel_name, NULL);
+ priv->name = g_strdup (camel_name);
+ }
+
+ camel_object_unref (CAMEL_OBJECT (addr));
+ }
+ }
+
+ return priv->name;
+}
+
+const gchar *
+eab_destination_get_email (const EABDestination *dest)
+{
+ struct _EABDestinationPrivate *priv;
+
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ priv = (struct _EABDestinationPrivate *)dest->priv; /* cast out const */
+
+ if (priv->email == NULL) {
+ if (priv->contact != NULL) {
+ /* Pull the address out of the card. */
+ GList *email = e_contact_get (priv->contact, E_CONTACT_EMAIL);
+ if (email) {
+ char *e = g_list_nth_data (email, priv->email_num);
+
+ if (e)
+ priv->email = g_strdup (e);
+ }
+ if (email) {
+ g_list_foreach (email, (GFunc)g_free, NULL);
+ g_list_free (email);
+ }
+
+ } else if (priv->raw != NULL) {
+ CamelInternetAddress *addr = camel_internet_address_new ();
+
+ if (camel_address_unformat (CAMEL_ADDRESS (addr), priv->raw)) {
+ const gchar *camel_email = NULL;
+ camel_internet_address_get (addr, 0, NULL, &camel_email);
+ priv->email = g_strdup (camel_email);
+ }
+
+ camel_object_unref (CAMEL_OBJECT (addr));
+ }
+
+ /* Force e-mail to be non-null... */
+ if (priv->email == NULL) {
+ priv->email = g_strdup ("");
+ }
+ }
+
+ return priv->email;
+}
+
+const gchar *
+eab_destination_get_address (const EABDestination *dest)
+{
+ struct _EABDestinationPrivate *priv;
+
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ priv = (struct _EABDestinationPrivate *)dest->priv; /* cast out const */
+
+ if (priv->addr == NULL) {
+ CamelInternetAddress *addr = camel_internet_address_new ();
+
+ if (eab_destination_is_evolution_list (dest)) {
+ GList *iter = dest->priv->list_dests;
+
+ while (iter) {
+ EABDestination *list_dest = EAB_DESTINATION (iter->data);
+
+ if (!eab_destination_is_empty (list_dest)) {
+ camel_internet_address_add (addr,
+ eab_destination_get_name (list_dest),
+ eab_destination_get_email (list_dest));
+ }
+ iter = g_list_next (iter);
+ }
+
+ priv->addr = camel_address_encode (CAMEL_ADDRESS (addr));
+ } else if (priv->raw) {
+
+ if (camel_address_unformat (CAMEL_ADDRESS (addr), priv->raw)) {
+ priv->addr = camel_address_encode (CAMEL_ADDRESS (addr));
+ }
+ } else {
+ camel_internet_address_add (addr,
+ eab_destination_get_name (dest),
+ eab_destination_get_email (dest));
+
+ priv->addr = camel_address_encode (CAMEL_ADDRESS (addr));
+ }
+
+ camel_object_unref (CAMEL_OBJECT (addr));
+ }
+
+ return priv->addr;
+}
+
+void
+eab_destination_set_raw (EABDestination *dest, const gchar *raw)
+{
+ g_return_if_fail (EAB_IS_DESTINATION (dest));
+ g_return_if_fail (raw != NULL);
+
+ if (dest->priv->raw == NULL || strcmp (dest->priv->raw, raw)) {
+ eab_destination_freeze (dest);
+
+ eab_destination_clear (dest);
+ dest->priv->raw = g_strdup (raw);
+ eab_destination_changed (dest);
+
+ eab_destination_thaw (dest);
+ }
+}
+
+const gchar *
+eab_destination_get_textrep (const EABDestination *dest, gboolean include_email)
+{
+ const char *name, *email;
+
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ if (dest->priv->raw)
+ return dest->priv->raw;
+
+ name = eab_destination_get_name (dest);
+ email = eab_destination_get_email (dest);
+
+ if (eab_destination_from_contact (dest) && name != NULL && (!include_email || !email || !*email))
+ return name;
+
+ /* Make sure that our address gets quoted properly */
+ if (name && email && dest->priv->textrep == NULL) {
+ CamelInternetAddress *addr = camel_internet_address_new ();
+
+ camel_internet_address_add (addr, name, email);
+ g_free (dest->priv->textrep);
+ dest->priv->textrep = camel_address_format (CAMEL_ADDRESS (addr));
+ camel_object_unref (CAMEL_OBJECT (addr));
+ }
+
+ if (dest->priv->textrep != NULL)
+ return dest->priv->textrep;
+
+ if (email)
+ return email;
+
+ return "";
+}
+
+gboolean
+eab_destination_is_evolution_list (const EABDestination *dest)
+{
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), FALSE);
+
+ if (dest->priv->list_dests == NULL
+ && dest->priv->contact != NULL
+ && e_contact_get (dest->priv->contact, E_CONTACT_IS_LIST)) {
+ GList *email = e_contact_get (dest->priv->contact, E_CONTACT_EMAIL);
+ if (email) {
+ GList *iter;
+ for (iter = email; iter; iter = iter->next) {
+ EABDestination *list_dest = eab_destination_import ((char *) iter->data);
+
+ if (list_dest)
+ dest->priv->list_dests = g_list_append (dest->priv->list_dests, list_dest);
+ }
+ }
+ }
+
+ return dest->priv->list_dests != NULL;
+}
+
+gboolean
+eab_destination_list_show_addresses (const EABDestination *dest)
+{
+ g_return_val_if_fail (EAB_IS_DESTINATION (dest), FALSE);
+
+ if (dest->priv->contact != NULL)
+ return GPOINTER_TO_UINT (e_contact_get (dest->priv->contact, E_CONTACT_LIST_SHOW_ADDRESSES));
+
+ return dest->priv->show_addresses;
+}
+
+gboolean
+eab_destination_get_html_mail_pref (const EABDestination *dest)
+{
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), FALSE);
+
+ if (dest->priv->html_mail_override || dest->priv->contact == NULL)
+ return dest->priv->wants_html_mail;
+
+ return e_contact_get (dest->priv->contact, E_CONTACT_WANTS_HTML) ? TRUE : FALSE;
+}
+
+static void
+set_book (EABDestination *dest, EBook *book)
+{
+ if (dest->priv->book && dest->priv->book != book) {
+ g_object_unref (dest->priv->book);
+ }
+
+ dest->priv->book = book;
+
+ if (book)
+ g_object_ref (book);
+}
+
+static void
+name_and_email_cb (EBook *book, EBookStatus status, GList *contacts, gpointer closure)
+{
+ EABDestination *dest = EAB_DESTINATION (closure);
+
+ if (status == E_BOOK_ERROR_OK && g_list_length ((GList *) contacts) == 1) {
+ EContact *contact = E_CONTACT (contacts->data);
+ const char *email = eab_destination_get_email (dest);
+ int email_num = 0;
+
+#if notyet
+ if (eab_destination_is_valid (dest) && email && *email) {
+ email_num = e_contact_email_find_number (contact, eab_destination_get_email (dest));
+ }
+#endif
+
+ if (email_num >= 0) {
+ const char *book_uri;
+
+ book_uri = e_book_get_uri (book);
+
+ dest->priv->contact_loaded = TRUE;
+ eab_destination_set_contact (dest, contact, email_num);
+ eab_destination_set_book_uri (dest, book_uri);
+ g_signal_emit (dest, eab_destination_signals[CONTACT_LOADED], 0);
+ }
+ }
+
+ if (!dest->priv->contact_loaded)
+ dest->priv->cannot_load = TRUE;
+
+ g_object_unref (dest); /* drop the reference held by the query */
+}
+
+
+static void
+nickname_cb (EBook *book, EBookStatus status, GList *contacts, gpointer closure)
+{
+ EABDestination *dest = EAB_DESTINATION (closure);
+
+ if (status == E_BOOK_ERROR_OK) {
+ if (g_list_length ((GList *) contacts) == 1) {
+ const char *book_uri;
+
+ book_uri = e_book_get_uri (book);
+
+ dest->priv->contact_loaded = TRUE;
+ eab_destination_set_contact (dest, E_CONTACT (contacts->data), 0); /* Uses primary e-mail by default. */
+ eab_destination_set_book_uri (dest, book_uri);
+ g_signal_emit (dest, eab_destination_signals[CONTACT_LOADED], 0);
+
+ g_object_unref (dest); /* drop the reference held by the query */
+
+ } else {
+ /* We can only end up here if we don't look at all like an e-mail address, so
+ we do a name-only query on the textrep */
+
+ eab_name_and_email_query (book,
+ eab_destination_get_textrep (dest, FALSE),
+ NULL,
+ name_and_email_cb,
+ dest);
+ }
+ } else {
+ /* Something went wrong with the query: drop our ref to the destination and return. */
+ g_object_unref (dest);
+ }
+}
+
+static void
+launch_load_contact_query (EABDestination *dest)
+{
+ if (! eab_destination_is_valid (dest)) {
+ /* If it doesn't look like an e-mail address, see if it is a nickname. */
+ eab_nickname_query (dest->priv->book,
+ eab_destination_get_textrep (dest, FALSE),
+ nickname_cb,
+ dest);
+
+ } else {
+ eab_name_and_email_query (dest->priv->book,
+ eab_destination_get_name (dest),
+ eab_destination_get_email (dest),
+ name_and_email_cb,
+ dest);
+ }
+}
+
+void
+eab_destination_load_contact (EABDestination *dest, EBook *book)
+{
+ g_return_if_fail (EAB_IS_DESTINATION (dest));
+ g_return_if_fail (book == NULL || E_IS_BOOK (book));
+
+ if (eab_destination_is_evolution_list (dest))
+ return;
+
+ if (eab_destination_contains_contact (dest))
+ return;
+
+ if (dest->priv->cannot_load)
+ return;
+
+ eab_destination_cancel_contact_load (dest);
+
+ set_book (dest, book);
+
+ /* Handle the case of an EABDestination containing a contact URL */
+ if (eab_destination_contains_contact (dest)) {
+ eab_destination_use_contact (dest, NULL, NULL);
+ return;
+ }
+
+ /* We hold a reference to ourselves until our query is complete. */
+ g_object_ref (dest);
+ launch_load_contact_query (dest);
+}
+
+static int
+do_load_delayed (gpointer ptr)
+{
+ EABDestination *dest = EAB_DESTINATION (ptr);
+
+ eab_destination_load_contact (dest, dest->priv->book);
+ return FALSE;
+}
+
+void
+eab_destination_load_contact_delayed (EABDestination *dest, EBook *book, gint delay)
+{
+ g_return_if_fail (EAB_IS_DESTINATION (dest));
+ g_return_if_fail (book == NULL || E_IS_BOOK (book));
+
+ if (delay < 0)
+ delay = 500;
+
+ eab_destination_cancel_contact_load (dest);
+
+ set_book (dest, book);
+
+ dest->priv->pending_contact_load = g_timeout_add (delay, do_load_delayed, dest);
+}
+
+void
+eab_destination_cancel_contact_load (EABDestination *dest)
+{
+ g_return_if_fail (EAB_IS_DESTINATION (dest));
+
+ if (dest->priv->pending_contact_load) {
+ g_source_remove (dest->priv->pending_contact_load);
+ dest->priv->pending_contact_load = 0;
+ }
+}
+
+gboolean
+eab_destination_unload_contact (EABDestination *dest)
+{
+ char *email;
+
+ g_return_val_if_fail (EAB_IS_DESTINATION (dest), FALSE);
+
+ if (!eab_destination_contains_contact (dest))
+ return FALSE;
+
+ email = g_strdup (eab_destination_get_email (dest));
+
+ if (email == NULL)
+ return FALSE;
+
+ eab_destination_freeze (dest);
+ eab_destination_clear (dest);
+ eab_destination_set_raw (dest, email);
+ g_free (email);
+ eab_destination_thaw (dest);
+
+ return TRUE;
+}
+
+/*
+ * Destination import/export
+ */
+
+gchar *
+eab_destination_get_address_textv (EABDestination **destv)
+{
+ int i, j, len = 0;
+ char **strv;
+ char *str;
+
+ g_return_val_if_fail (destv, NULL);
+
+ /* Q: Please tell me this is only for assertion
+ reasons. If this is considered to be ok behavior then you
+ shouldn't use g_return's. Just a reminder ;-)
+
+ A: Yes, this is just an assertion. (Though it does find the
+ length of the vector in the process...)
+ */
+ while (destv[len]) {
+ g_return_val_if_fail (EAB_IS_DESTINATION (destv[len]), NULL);
+ len++;
+ }
+
+ strv = g_new0 (char *, len + 1);
+ for (i = 0, j = 0; destv[i]; i++) {
+ if (!eab_destination_is_empty (destv[i])) {
+ const char *addr = eab_destination_get_address (destv[i]);
+ strv[j++] = addr ? (char *) addr : "";
+ }
+ }
+
+ str = g_strjoinv (", ", strv);
+
+ g_free (strv);
+
+ return str;
+}
+
+xmlNodePtr
+eab_destination_xml_encode (const EABDestination *dest)
+{
+ xmlNodePtr dest_node;
+ const char *str;
+
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ dest_node = xmlNewNode (NULL, "destination");
+
+ str = eab_destination_get_name (dest);
+ if (str)
+ xmlNewTextChild (dest_node, NULL, "name", str);
+
+ if (!eab_destination_is_evolution_list (dest)) {
+ str = eab_destination_get_email (dest);
+ if (str)
+ xmlNewTextChild (dest_node, NULL, "email", str);
+ } else {
+ GList *iter = dest->priv->list_dests;
+
+ while (iter) {
+ EABDestination *list_dest = EAB_DESTINATION (iter->data);
+ xmlNodePtr list_node = xmlNewNode (NULL, "list_entry");
+
+ str = eab_destination_get_name (list_dest);
+ if (str)
+ xmlNewTextChild (list_node, NULL, "name", str);
+
+ str = eab_destination_get_email (list_dest);
+ if (str)
+ xmlNewTextChild (list_node, NULL, "email", str);
+
+ xmlAddChild (dest_node, list_node);
+
+ iter = g_list_next (iter);
+ }
+
+ xmlNewProp (dest_node, "is_list", "yes");
+ xmlNewProp (dest_node, "show_addresses",
+ eab_destination_list_show_addresses (dest) ? "yes" : "no");
+ }
+
+ str = eab_destination_get_book_uri (dest);
+ if (str) {
+ xmlNewTextChild (dest_node, NULL, "book_uri", str);
+ }
+
+ str = eab_destination_get_contact_uid (dest);
+ if (str) {
+ char buf[16];
+
+ xmlNodePtr uri_node = xmlNewTextChild (dest_node, NULL, "card_uid", str);
+ g_snprintf (buf, 16, "%d", eab_destination_get_email_num (dest));
+ xmlNewProp (uri_node, "email_num", buf);
+ }
+
+ xmlNewProp (dest_node, "html_mail", eab_destination_get_html_mail_pref (dest) ? "yes" : "no");
+
+ xmlNewProp (dest_node, "auto_recipient",
+ eab_destination_is_auto_recipient (dest) ? "yes" : "no");
+
+ return dest_node;
+}
+
+gboolean
+eab_destination_xml_decode (EABDestination *dest, xmlNodePtr node)
+{
+ char *name = NULL, *email = NULL, *book_uri = NULL, *card_uid = NULL;
+ gboolean is_list = FALSE, show_addr = FALSE, auto_recip = FALSE;
+ gboolean html_mail = FALSE;
+ GList *list_dests = NULL;
+ int email_num = -1;
+ char *tmp;
+
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), FALSE);
+ g_return_val_if_fail (node != NULL, FALSE);
+
+ if (strcmp (node->name, "destination"))
+ return FALSE;
+
+ tmp = xmlGetProp (node, "html_mail");
+ if (tmp) {
+ html_mail = !strcmp (tmp, "yes");
+ xmlFree (tmp);
+ }
+
+ tmp = xmlGetProp (node, "is_list");
+ if (tmp) {
+ is_list = !strcmp (tmp, "yes");
+ xmlFree (tmp);
+ }
+
+ tmp = xmlGetProp (node, "show_addresses");
+ if (tmp) {
+ show_addr = !strcmp (tmp, "yes");
+ xmlFree (tmp);
+ }
+
+ tmp = xmlGetProp (node, "auto_recipient");
+ if (tmp) {
+ auto_recip = !strcmp (tmp, "yes");
+ xmlFree (tmp);
+ }
+
+ node = node->xmlChildrenNode;
+ while (node) {
+ if (!strcmp (node->name, "name")) {
+ tmp = xmlNodeGetContent (node);
+ g_free (name);
+ name = g_strdup (tmp);
+ xmlFree (tmp);
+ } else if (!is_list && !strcmp (node->name, "email")) {
+ tmp = xmlNodeGetContent (node);
+ g_free (email);
+ email = g_strdup (tmp);
+ xmlFree (tmp);
+ } else if (is_list && !strcmp (node->name, "list_entry")) {
+ xmlNodePtr subnode = node->xmlChildrenNode;
+ char *list_name = NULL, *list_email = NULL;
+
+ while (subnode) {
+ if (!strcmp (subnode->name, "name")) {
+ tmp = xmlNodeGetContent (subnode);
+ g_free (list_name);
+ list_name = g_strdup (tmp);
+ xmlFree (tmp);
+ } else if (!strcmp (subnode->name, "email")) {
+ tmp = xmlNodeGetContent (subnode);
+ g_free (list_email);
+ list_email = g_strdup (tmp);
+ xmlFree (tmp);
+ }
+
+ subnode = subnode->next;
+ }
+
+ if (list_name || list_email) {
+ EABDestination *list_dest = eab_destination_new ();
+
+ if (list_name)
+ eab_destination_set_name (list_dest, list_name);
+ if (list_email)
+ eab_destination_set_email (list_dest, list_email);
+
+ g_free (list_name);
+ g_free (list_email);
+
+ list_dests = g_list_append (list_dests, list_dest);
+ }
+ } else if (!strcmp (node->name, "book_uri")) {
+ tmp = xmlNodeGetContent (node);
+ g_free (book_uri);
+ book_uri = g_strdup (tmp);
+ xmlFree (tmp);
+ } else if (!strcmp (node->name, "card_uid")) {
+ tmp = xmlNodeGetContent (node);
+ g_free (card_uid);
+ card_uid = g_strdup (tmp);
+ xmlFree (tmp);
+
+ tmp = xmlGetProp (node, "email_num");
+ email_num = atoi (tmp);
+ xmlFree (tmp);
+ }
+
+ node = node->next;
+ }
+
+ eab_destination_freeze (dest);
+
+ eab_destination_clear (dest);
+
+ if (name) {
+ eab_destination_set_name (dest, name);
+ g_free (name);
+ }
+ if (email) {
+ eab_destination_set_email (dest, email);
+ g_free (email);
+ }
+ if (book_uri) {
+ eab_destination_set_book_uri (dest, book_uri);
+ g_free (book_uri);
+ }
+ if (card_uid) {
+ eab_destination_set_contact_uid (dest, card_uid, email_num);
+ g_free (card_uid);
+ }
+ if (list_dests)
+ dest->priv->list_dests = list_dests;
+
+ dest->priv->html_mail_override = TRUE;
+ dest->priv->wants_html_mail = html_mail;
+
+ dest->priv->show_addresses = show_addr;
+
+ dest->priv->auto_recipient = auto_recip;
+
+ eab_destination_thaw (dest);
+
+ return TRUE;
+}
+
+/* FIXME: Make utf-8 safe */
+static gchar *
+null_terminate_and_remove_extra_whitespace (xmlChar *xml_in, gint size)
+{
+ gboolean skip_white = FALSE;
+ char *xml, *r, *w;
+
+ if (xml_in == NULL || size <= 0)
+ return NULL;
+
+ xml = g_strndup (xml_in, size);
+ r = w = xml;
+
+ while (*r) {
+ if (*r == '\n' || *r == '\r') {
+ skip_white = TRUE;
+ } else {
+ gboolean is_space = isspace (*r);
+
+ *w = *r;
+
+ if (!(skip_white && is_space))
+ w++;
+ if (!is_space)
+ skip_white = FALSE;
+ }
+ r++;
+ }
+
+ *w = '\0';
+
+ return xml;
+}
+
+gchar *
+eab_destination_export (const EABDestination *dest)
+{
+ xmlNodePtr dest_node;
+ xmlDocPtr dest_doc;
+ xmlChar *buffer = NULL;
+ int size = -1;
+ char *str;
+
+ g_return_val_if_fail (dest && EAB_IS_DESTINATION (dest), NULL);
+
+ dest_node = eab_destination_xml_encode (dest);
+ if (dest_node == NULL)
+ return NULL;
+
+ dest_doc = xmlNewDoc (XML_DEFAULT_VERSION);
+ xmlDocSetRootElement (dest_doc, dest_node);
+
+ xmlDocDumpMemory (dest_doc, &buffer, &size);
+ xmlFreeDoc (dest_doc);
+
+ str = null_terminate_and_remove_extra_whitespace (buffer, size);
+ xmlFree (buffer);
+
+ return str;
+}
+
+EABDestination *
+eab_destination_import (const gchar *str)
+{
+ EABDestination *dest = NULL;
+ xmlDocPtr dest_doc;
+
+ if (!(str && *str))
+ return NULL;
+
+ dest_doc = xmlParseMemory ((char *) str, strlen (str));
+ if (dest_doc && dest_doc->xmlRootNode) {
+ dest = eab_destination_new ();
+ if (! eab_destination_xml_decode (dest, dest_doc->xmlRootNode)) {
+ g_object_unref (dest);
+ dest = NULL;
+ }
+ }
+ xmlFreeDoc (dest_doc);
+
+ return dest;
+}
+
+gchar *
+eab_destination_exportv (EABDestination **destv)
+{
+ xmlDocPtr destv_doc;
+ xmlNodePtr destv_node;
+ xmlChar *buffer = NULL;
+ int i, size = -1;
+ char *str;
+
+ if (destv == NULL || *destv == NULL)
+ return NULL;
+
+ destv_doc = xmlNewDoc (XML_DEFAULT_VERSION);
+ destv_node = xmlNewNode (NULL, "destinations");
+ xmlDocSetRootElement (destv_doc, destv_node);
+
+ for (i = 0; destv[i]; i++) {
+ if (! eab_destination_is_empty (destv[i])) {
+ xmlNodePtr dest_node = eab_destination_xml_encode (destv[i]);
+ if (dest_node)
+ xmlAddChild (destv_node, dest_node);
+ }
+ }
+
+ xmlDocDumpMemory (destv_doc, &buffer, &size);
+ xmlFreeDoc (destv_doc);
+
+ str = null_terminate_and_remove_extra_whitespace (buffer, size);
+ xmlFree (buffer);
+
+ return str;
+}
+
+EABDestination **
+eab_destination_importv (const gchar *str)
+{
+ GPtrArray *dest_array = NULL;
+ xmlDocPtr destv_doc;
+ xmlNodePtr node;
+ EABDestination **destv = NULL;
+
+ if (!(str && *str))
+ return NULL;
+
+ destv_doc = xmlParseMemory ((char *)str, strlen (str));
+ if (destv_doc == NULL)
+ return NULL;
+
+ node = destv_doc->xmlRootNode;
+
+ if (strcmp (node->name, "destinations"))
+ goto finished;
+
+ node = node->xmlChildrenNode;
+
+ dest_array = g_ptr_array_new ();
+
+ while (node) {
+ EABDestination *dest;
+
+ dest = eab_destination_new ();
+ if (eab_destination_xml_decode (dest, node) && !eab_destination_is_empty (dest)) {
+ g_ptr_array_add (dest_array, dest);
+ } else {
+ g_object_unref (dest);
+ }
+
+ node = node->next;
+ }
+
+ /* we need destv to be NULL terminated */
+ g_ptr_array_add (dest_array, NULL);
+
+ destv = (EABDestination **) dest_array->pdata;
+ g_ptr_array_free (dest_array, FALSE);
+
+ finished:
+ xmlFreeDoc (destv_doc);
+
+ return destv;
+}
+
+EABDestination **
+eab_destination_list_to_vector_sized (GList *list, int n)
+{
+ EABDestination **destv;
+ int i = 0;
+
+ if (n == -1)
+ n = g_list_length (list);
+
+ if (n == 0)
+ return NULL;
+
+ destv = g_new (EABDestination *, n + 1);
+ while (list != NULL && i < n) {
+ destv[i] = EAB_DESTINATION (list->data);
+ list->data = NULL;
+ i++;
+ list = g_list_next (list);
+ }
+ destv[i] = NULL;
+
+ return destv;
+}
+
+EABDestination **
+eab_destination_list_to_vector (GList *list)
+{
+ return eab_destination_list_to_vector_sized (list, -1);
+}
+
+void
+eab_destination_freev (EABDestination **destv)
+{
+ int i;
+
+ if (destv) {
+ for (i = 0; destv[i] != NULL; ++i) {
+ g_object_unref (destv[i]);
+ }
+ g_free (destv);
+ }
+
+}
+
+#if notyet
+static void
+touch_cb (EBook *book, const gchar *addr, ECard *card, gpointer closure)
+{
+ if (book != NULL && card != NULL) {
+ e_card_touch (card);
+ d(g_message ("Use score for \"%s\" is now %f", addr, e_card_get_use_score (card)));
+ e_book_commit_card (book, card, NULL, NULL);
+ }
+}
+#endif
+
+void
+eab_destination_touch (EABDestination *dest)
+{
+#if notyet
+ const char *email;
+
+ g_return_if_fail (dest && EAB_IS_DESTINATION (dest));
+
+ if (!eab_destination_is_auto_recipient (dest)) {
+ email = eab_destination_get_email (dest);
+
+ if (email)
+ e_book_query_address_default (email, touch_cb, NULL);
+ }
+#endif
+}
+
+void
+eab_destination_touchv (EABDestination **destv)
+{
+#if notyet
+ int i;
+
+ g_return_if_fail (destv != NULL);
+
+ for (i = 0; destv[i] != NULL; ++i) {
+ eab_destination_touch (destv[i]);
+ }
+#endif
+}
diff --git a/addressbook/util/eab-destination.h b/addressbook/util/eab-destination.h
new file mode 100644
index 0000000000..452d893e44
--- /dev/null
+++ b/addressbook/util/eab-destination.h
@@ -0,0 +1,128 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+
+/*
+ * eab-destination.h
+ *
+ * Copyright (C) 2001-2003 Ximian, Inc.
+ *
+ * Authors: Jon Trowbridge <trow@ximian.com>
+ * Chris Toshok <toshok@ximian.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., 59 Temple Place, Suite 330, Boston, MA 02111-1307
+ * USA.
+ */
+
+#ifndef __E_DESTINATION_H__
+#define __E_DESTINATION_H__
+
+#include <glib.h>
+#include <glib-object.h>
+#include <ebook/e-contact.h>
+#include <ebook/e-book.h>
+#include <libxml/tree.h>
+
+#define EAB_TYPE_DESTINATION (eab_destination_get_type ())
+#define EAB_DESTINATION(o) (G_TYPE_CHECK_INSTANCE_CAST ((o), EAB_TYPE_DESTINATION, EABDestination))
+#define EAB_DESTINATION_CLASS(k) (G_TYPE_CHECK_CLASS_CAST ((k), EAB_TYPE_DESTINATION, EABDestinationClass))
+#define EAB_IS_DESTINATION(o) (G_TYPE_CHECK_INSTANCE_TYPE ((o), EAB_TYPE_DESTINATION))
+#define EAB_IS_DESTINATION_CLASS(k) (G_TYPE_CHECK_CLASS_TYPE ((k), EAB_TYPE_DESTINATION))
+#define EAB_DESTINATION_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), EAB_TYPE_DESTINATION, EABDestinationClass))
+
+typedef struct _EABDestination EABDestination;
+typedef struct _EABDestinationClass EABDestinationClass;
+
+typedef void (*EABDestinationContactCallback) (EABDestination *dest, EContact *contact, gpointer closure);
+
+struct _EABDestinationPrivate;
+
+struct _EABDestination {
+ GObject object;
+
+ struct _EABDestinationPrivate *priv;
+};
+
+struct _EABDestinationClass {
+ GObjectClass parent_class;
+
+ void (*changed) (EABDestination *dest);
+ void (*contact_loaded) (EABDestination *dest);
+};
+
+GType eab_destination_get_type (void);
+
+
+EABDestination *eab_destination_new (void);
+void eab_destination_changed (EABDestination *);
+EABDestination *eab_destination_copy (const EABDestination *);
+void eab_destination_clear (EABDestination *);
+
+gboolean eab_destination_is_empty (const EABDestination *);
+gboolean eab_destination_equal (const EABDestination *a, const EABDestination *b);
+
+void eab_destination_set_contact (EABDestination *, EContact *contact, gint email_num);
+
+void eab_destination_set_name (EABDestination *, const gchar *name);
+void eab_destination_set_email (EABDestination *, const gchar *email);
+
+void eab_destination_set_html_mail_pref (EABDestination *, gboolean);
+
+gboolean eab_destination_contains_contact (const EABDestination *);
+
+gboolean eab_destination_is_auto_recipient (const EABDestination *);
+void eab_destination_set_auto_recipient (EABDestination *, gboolean value);
+
+void eab_destination_use_contact (EABDestination *, EABDestinationContactCallback cb, gpointer closure);
+
+EContact *eab_destination_get_contact (const EABDestination *);
+gint eab_destination_get_email_num (const EABDestination *);
+
+const gchar *eab_destination_get_name (const EABDestination *); /* "Jane Smith" */
+const gchar *eab_destination_get_email (const EABDestination *); /* "jane@assbarn.com" */
+const gchar *eab_destination_get_address (const EABDestination *);; /* "Jane Smith <jane@assbarn.com>" (or a comma-sep set of such for a list) */
+
+void eab_destination_set_raw (EABDestination *, const gchar *free_form_string);
+const gchar *eab_destination_get_textrep (const EABDestination *, gboolean include_email); /* "Jane Smith" or "jane@assbarn.com" */
+
+gboolean eab_destination_is_evolution_list (const EABDestination *);
+gboolean eab_destination_list_show_addresses (const EABDestination *);
+
+/* If true, they want HTML mail. */
+gboolean eab_destination_get_html_mail_pref (const EABDestination *);
+
+void eab_destination_load_contact (EABDestination *, EBook *);
+void eab_destination_load_contact_delayed (EABDestination *, EBook *, gint delay); /* delay < 0: "default" */
+void eab_destination_cancel_contact_load (EABDestination *);
+gboolean eab_destination_unload_contact (EABDestination *);
+
+gchar *eab_destination_get_address_textv (EABDestination **);
+
+gchar *eab_destination_export (const EABDestination *);
+EABDestination *eab_destination_import (const gchar *str);
+
+gchar *eab_destination_exportv (EABDestination **);
+EABDestination **eab_destination_importv (const gchar *str);
+
+EABDestination **eab_destination_list_to_vector_sized (GList *, int n);
+EABDestination **eab_destination_list_to_vector (GList *);
+
+void eab_destination_freev (EABDestination **);
+
+void eab_destination_touch (EABDestination *);
+void eab_destination_touchv (EABDestination **);
+
+
+#endif /* __EAB_DESTINATION_H__ */
+
diff --git a/addressbook/util/eab-marshal.list b/addressbook/util/eab-marshal.list
new file mode 100644
index 0000000000..680ea039a3
--- /dev/null
+++ b/addressbook/util/eab-marshal.list
@@ -0,0 +1,5 @@
+NONE:NONE
+NONE:BOOL
+NONE:POINTER
+NONE:STRING
+NONE:INT