/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ /* e-msg-composer-address-dialog.c * * Copyright (C) 1999 Helix Code, Inc. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * 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. * * Author: Ettore Perazzoli */ #include <gnome.h> #include "e-msg-composer-address-dialog.h" #include <e-util/e-unicode.h> enum { APPLY, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; static GnomeDialogClass *parent_class = NULL; /* This function should load the addresses we know of into the dialog. We don't have a precise setup for the addressbook yet, so we will just put some fake entries in. */ static void load_addresses (EMsgComposerAddressDialog *dialog) { gchar *text[][3] = { { "Bertrand Guiheneuf", "Bertrand.Guiheneuf@aful.org", NULL }, { "Ettore Perazzoli", "ettore@gnu.org", NULL }, { "Miguel de Icaza", "miguel@gnu.org", NULL }, { "Nat Friedman", "nat@nat.org", NULL }, { NULL, NULL, NULL } }; GtkCList *clist; guint i; clist = GTK_CLIST (glade_xml_get_widget (dialog->gui, "address_clist")); for (i = 0; text[i][0] != NULL; i++) e_utf8_gtk_clist_append (clist, text[i]); } /* Combine name and email into an address, e.g. "Ettore Perazzoli <ettore@gnu.org>". FIXME FIXME FIXME this does not handle quoting (commas will cause troubles), but it should. */ static gchar * make_full_address (const gchar *name, const gchar *email) { return g_strconcat (name, " <", email, ">", NULL); } /* This loads the selected address in the address GtkCList into the requested GtkList. */ static void add_address (EMsgComposerAddressDialog *dialog, const gchar *list_name) { GtkCList *src_clist; GtkCList *dest_clist; gchar *name, *email; gchar *text[2]; guint row; src_clist = GTK_CLIST (glade_xml_get_widget (dialog->gui, "address_clist")); if (src_clist->selection == NULL) return; dest_clist = GTK_CLIST (glade_xml_get_widget (dialog->gui, list_name)); row = GPOINTER_TO_INT (src_clist->selection->data); gtk_clist_get_text (src_clist, row, 0, &name); gtk_clist_get_text (src_clist, row, 1, &email); name = e_utf8_from_gtk_string ((GtkWidget *) src_clist, name); text[0] = make_full_address (name, email); text[1] = NULL; e_utf8_gtk_clist_append (dest_clist, text); g_free (text[0]); g_free (name); } static void apply (EMsgComposerAddressDialog *dialog) { gtk_signal_emit (GTK_OBJECT (dialog), signals[APPLY]); } /* Recipient list popup menu. */ struct _RecipientListInfo { EMsgComposerAddressDialog *dialog; GtkCList *clist; gint row; /* -1 if menu was popped up in an empty area. */ }; typedef struct _RecipientListInfo RecipientListInfo; static void copy_recipient (RecipientListInfo *info, gboolean remove) { gchar *text; gint row; if (info->clist->selection == NULL) return; row = GPOINTER_TO_INT (info->clist->selection->data); gtk_clist_get_text (info->clist, row, 0, &text); g_free (info->dialog->cut_buffer); info->dialog->cut_buffer = g_strdup (text); if (remove) gtk_clist_remove (info->clist, row); gtk_selection_owner_set (GTK_WIDGET (info->clist), GDK_SELECTION_PRIMARY, GDK_CURRENT_TIME); } static void copy_recipient_cb (GtkWidget *widget, gpointer data) { RecipientListInfo *info; info = (RecipientListInfo *) data; copy_recipient (info, FALSE); g_free (info); } static void cut_recipient_cb (GtkWidget *widget, gpointer data) { RecipientListInfo *info; info = (RecipientListInfo *) data; copy_recipient (info, TRUE); g_free (info); } static void paste_recipient_cb (GtkWidget *widget, gpointer data) { RecipientListInfo *info; GdkAtom atom; info = (RecipientListInfo *) data; atom = gdk_atom_intern ("STRING", FALSE); gtk_selection_convert (GTK_WIDGET (info->clist), GDK_SELECTION_PRIMARY, atom, GDK_CURRENT_TIME); g_free (info); } static GnomeUIInfo recipient_list_item_popup_info[] = { GNOMEUIINFO_ITEM_STOCK (N_("Cut"), N_("Cut selected item into clipboard"), cut_recipient_cb, GNOME_STOCK_MENU_CUT), GNOMEUIINFO_ITEM_STOCK (N_("Copy"), N_("Copy selected item into clipboard"), copy_recipient_cb, GNOME_STOCK_MENU_COPY), GNOMEUIINFO_ITEM_STOCK (N_("Paste"), N_("Paste item from clipboard"), paste_recipient_cb, GNOME_STOCK_MENU_PASTE), GNOMEUIINFO_END }; static GnomeUIInfo recipient_list_popup_info[] = { GNOMEUIINFO_ITEM_STOCK (N_("Paste"), N_("Paste item from clipboard"), paste_recipient_cb, GNOME_STOCK_MENU_PASTE), GNOMEUIINFO_END }; /* Signals. */ static void add_to_cb (GtkWidget *widget, gpointer data) { add_address (E_MSG_COMPOSER_ADDRESS_DIALOG (data), "to_clist"); } static void add_cc_cb (GtkWidget *widget, gpointer data) { add_address (E_MSG_COMPOSER_ADDRESS_DIALOG (data), "cc_clist"); } static void add_bcc_cb (GtkWidget *widget, gpointer data) { add_address (E_MSG_COMPOSER_ADDRESS_DIALOG (data), "bcc_clist"); } static void glade_connect (GladeXML *gui, const gchar *widget_name, const gchar *signal_name, GtkSignalFunc callback, gpointer callback_data) { GtkWidget *widget; widget = glade_xml_get_widget (gui, widget_name); if (widget == NULL) g_warning ("Widget `%s' was not found.", widget_name); else gtk_signal_connect (GTK_OBJECT (widget), signal_name, GTK_SIGNAL_FUNC (callback), callback_data); } static gint recipient_clist_button_press_cb (GtkWidget *widget, GdkEventButton *event, gpointer data) { EMsgComposerAddressDialog *dialog; RecipientListInfo *info; GtkWidget *popup; GtkCList *clist; gboolean on_row; gint row, column; dialog = E_MSG_COMPOSER_ADDRESS_DIALOG (data); clist = GTK_CLIST (widget); if (event->window != clist->clist_window || event->button != 3) return FALSE; on_row = gtk_clist_get_selection_info (clist, event->x, event->y, &row, &column); info = g_new (RecipientListInfo, 1); info->dialog = dialog; info->clist = clist; if (on_row) { gtk_clist_unselect_all (clist); gtk_clist_select_row (clist, row, 0); info->row = row; popup = gnome_popup_menu_new (recipient_list_item_popup_info); } else { info->row = -1; popup = gnome_popup_menu_new (recipient_list_popup_info); } gnome_popup_menu_do_popup_modal (popup, NULL, NULL, event, info); gtk_widget_destroy (popup); return TRUE; } /* FIXME needs more work. */ static void recipient_clist_selection_received_cb (GtkWidget *widget, GtkSelectionData *selection_data, guint time, gpointer data) { GtkCList *clist; gchar *text[2]; gchar *p; puts (__FUNCTION__); if (selection_data->length < 0) return; clist = GTK_CLIST (widget); /* FIXME quoting. */ text[0] = g_strdup (selection_data->data); text[1] = NULL; /* It is a common mistake to paste `\n's, let's work around that. */ for (p = text[0]; *p != '\0'; p++) { if (*p == '\n') { *p = '\0'; break; } } if (clist->selection != NULL) { gint row; row = GPOINTER_TO_INT (clist->selection->data); gtk_clist_insert (clist, row, text); } else { gtk_clist_append (clist, text); } g_free (text[0]); } static void recipient_clist_selection_get_cb (GtkWidget *widget, GtkSelectionData *selection_data, guint info, guint time, gpointer data) { EMsgComposerAddressDialog *dialog; GdkAtom atom; puts (__FUNCTION__); dialog = E_MSG_COMPOSER_ADDRESS_DIALOG (data); if (dialog->cut_buffer == NULL) return; /* FIXME should I do something special? */ atom = gdk_atom_intern ("STRING", FALSE); gtk_selection_data_set (selection_data, atom, 8, dialog->cut_buffer, strlen (dialog->cut_buffer)); } static void recipient_clist_selection_clear_event_cb (GtkWidget *widget, GdkEventSelection *selection, gpointer data) { EMsgComposerAddressDialog *dialog; dialog = E_MSG_COMPOSER_ADDRESS_DIALOG (data); g_free (dialog->cut_buffer); dialog->cut_buffer = NULL; } static void setup_recipient_list_signals (EMsgComposerAddressDialog *dialog, const gchar *name) { glade_connect (dialog->gui, name, "button_press_event", GTK_SIGNAL_FUNC (recipient_clist_button_press_cb), dialog); glade_connect (dialog->gui, name, "selection_received", GTK_SIGNAL_FUNC (recipient_clist_selection_received_cb), dialog); glade_connect (dialog->gui, name, "selection_get", GTK_SIGNAL_FUNC (recipient_clist_selection_get_cb), dialog); glade_connect (dialog->gui, name, "selection_clear_event", GTK_SIGNAL_FUNC (recipient_clist_selection_clear_event_cb), dialog); } static void setup_signals (EMsgComposerAddressDialog *dialog) { glade_connect (dialog->gui, "to_add_button", "clicked", GTK_SIGNAL_FUNC (add_to_cb), dialog); glade_connect (dialog->gui, "cc_add_button", "clicked", GTK_SIGNAL_FUNC (add_cc_cb), dialog); glade_connect (dialog->gui, "bcc_add_button", "clicked", GTK_SIGNAL_FUNC (add_bcc_cb), dialog); setup_recipient_list_signals (dialog, "to_clist"); setup_recipient_list_signals (dialog, "cc_clist"); setup_recipient_list_signals (dialog, "bcc_clist"); } static void setup_selection_targets (EMsgComposerAddressDialog *dialog) { gtk_selection_add_target (glade_xml_get_widget (dialog->gui, "to_clist"), GDK_SELECTION_PRIMARY, GDK_SELECTION_TYPE_STRING, 0); gtk_selection_add_target (glade_xml_get_widget (dialog->gui, "cc_clist"), GDK_SELECTION_PRIMARY, GDK_SELECTION_TYPE_STRING, 0); gtk_selection_add_target (glade_xml_get_widget (dialog->gui, "bcc_clist"), GDK_SELECTION_PRIMARY, GDK_SELECTION_TYPE_STRING, 0); } /* GnomeDialog methods. */ static void clicked (GnomeDialog *dialog, gint button_number) { switch (button_number) { case 0: /* OK */ apply (E_MSG_COMPOSER_ADDRESS_DIALOG (dialog)); gnome_dialog_close (dialog); break; case 1: /* Apply */ apply (E_MSG_COMPOSER_ADDRESS_DIALOG (dialog)); break; case 2: /* Cancel */ gnome_dialog_close (dialog); break; } } /* GtkObject methods. */ static void destroy (GtkObject *object) { EMsgComposerAddressDialog *dialog; dialog = E_MSG_COMPOSER_ADDRESS_DIALOG (object); gtk_object_unref (GTK_OBJECT (dialog->gui)); g_free (dialog->cut_buffer); if (GTK_OBJECT_CLASS (parent_class)->destroy != NULL) (* GTK_OBJECT_CLASS (parent_class)->destroy) (object); } /* Initialization. */ static void class_init (EMsgComposerAddressDialogClass *class) { GtkObjectClass *object_class; GnomeDialogClass *gnome_dialog_class; object_class = GTK_OBJECT_CLASS (class); object_class->destroy = destroy; gnome_dialog_class = GNOME_DIALOG_CLASS (class); gnome_dialog_class->clicked = clicked; parent_class = gtk_type_class (gnome_dialog_get_type ()); signals[APPLY] = gtk_signal_new ("apply", GTK_RUN_FIRST, object_class->type, GTK_SIGNAL_OFFSET (EMsgComposerAddressDialogClass, apply), gtk_marshal_NONE__NONE, GTK_TYPE_NONE, 0); gtk_object_class_add_signals (object_class, signals, LAST_SIGNAL); } static void init (EMsgComposerAddressDialog *dialog) { dialog->gui = NULL; dialog->cut_buffer = NULL; } GtkType e_msg_composer_address_dialog_get_type (void) { static GtkType type = 0; if (type == 0) { static const GtkTypeInfo info = { "EMsgComposerAddressDialog", sizeof (EMsgComposerAddressDialog), sizeof (EMsgComposerAddressDialogClass), (GtkClassInitFunc) class_init, (GtkObjectInitFunc) init, /* reserved_1 */ NULL, /* reserved_2 */ NULL, (GtkClassInitFunc) NULL, }; type = gtk_type_unique (gnome_dialog_get_type (), &info); } return type; } void e_msg_composer_address_dialog_construct (EMsgComposerAddressDialog *dialog) { static const gchar *buttons[] = { GNOME_STOCK_BUTTON_OK, GNOME_STOCK_BUTTON_APPLY, GNOME_STOCK_BUTTON_CANCEL, NULL }; g_return_if_fail (dialog != NULL); g_return_if_fail (E_IS_MSG_COMPOSER_ADDRESS_DIALOG (dialog)); gnome_dialog_constructv (GNOME_DIALOG (dialog), _("Select recipients' addresses"), buttons); dialog->gui = glade_xml_new (E_GLADEDIR "/e-msg-composer-address-dialog.glade", "main_table"); if (dialog->gui == NULL) { g_warning ("Cannot load `e-msg-composer-address-dialog.glade"); return; } gtk_container_add (GTK_CONTAINER (GNOME_DIALOG (dialog)->vbox), glade_xml_get_widget (dialog->gui, "main_table")); setup_selection_targets (dialog); load_addresses (dialog); setup_signals (dialog); } GtkWidget * e_msg_composer_address_dialog_new (void) { EMsgComposerAddressDialog *new; new = gtk_type_new (e_msg_composer_address_dialog_get_type ()); e_msg_composer_address_dialog_construct (new); return GTK_WIDGET (new); } static void set_list (EMsgComposerAddressDialog *dialog, const gchar *list_name, GList *list) { GtkCList *clist; GList *p; gchar *text[2]; clist = GTK_CLIST (glade_xml_get_widget (dialog->gui, list_name)); gtk_clist_freeze (clist); gtk_clist_clear (clist); text[1] = NULL; for (p = list; p != NULL; p = p->next) { text[0] = (gchar *) p->data; e_utf8_gtk_clist_append (clist, text); } gtk_clist_thaw (clist); } void e_msg_composer_address_dialog_set_to_list (EMsgComposerAddressDialog *dialog, GList *to_list) { g_return_if_fail (dialog != NULL); g_return_if_fail (E_IS_MSG_COMPOSER_ADDRESS_DIALOG (dialog)); set_list (dialog, "to_clist", to_list); } void e_msg_composer_address_dialog_set_cc_list (EMsgComposerAddressDialog *dialog, GList *cc_list) { g_return_if_fail (dialog != NULL); g_return_if_fail (E_IS_MSG_COMPOSER_ADDRESS_DIALOG (dialog)); set_list (dialog, "cc_clist", cc_list); } void e_msg_composer_address_dialog_set_bcc_list (EMsgComposerAddressDialog *dialog, GList *bcc_list) { g_return_if_fail (dialog != NULL); g_return_if_fail (E_IS_MSG_COMPOSER_ADDRESS_DIALOG (dialog)); set_list (dialog, "bcc_clist", bcc_list); } static GList * get_list (EMsgComposerAddressDialog *dialog, const gchar *clist_name) { GtkCList *clist; GList *list; guint i; clist = GTK_CLIST (glade_xml_get_widget (dialog->gui, clist_name)); list = NULL; for (i = 0; i < clist->rows; i++) { gchar *addr; gtk_clist_get_text (clist, i, 0, &addr); list = g_list_prepend (list, g_strdup (addr)); } return g_list_reverse (list); } GList * e_msg_composer_address_dialog_get_to_list (EMsgComposerAddressDialog *dialog) { g_return_val_if_fail (dialog != NULL, NULL); g_return_val_if_fail (E_IS_MSG_COMPOSER_ADDRESS_DIALOG (dialog), NULL); return get_list (dialog, "to_clist"); } GList * e_msg_composer_address_dialog_get_cc_list (EMsgComposerAddressDialog *dialog) { g_return_val_if_fail (dialog != NULL, NULL); g_return_val_if_fail (E_IS_MSG_COMPOSER_ADDRESS_DIALOG (dialog), NULL); return get_list (dialog, "cc_clist"); } GList * e_msg_composer_address_dialog_get_bcc_list (EMsgComposerAddressDialog *dialog) { g_return_val_if_fail (dialog != NULL, NULL); g_return_val_if_fail (E_IS_MSG_COMPOSER_ADDRESS_DIALOG (dialog), NULL); return get_list (dialog, "bcc_clist"); }