aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorEttore Perazzoli <ettore@src.gnome.org>1999-11-09 04:48:32 +0800
committerEttore Perazzoli <ettore@src.gnome.org>1999-11-09 04:48:32 +0800
commit643a3d01716e1904d44b2dffca69219286df9d87 (patch)
tree82a6ae88b198c68d8d7747891cf6ad24149417da
parent0c3101c24a3bb76c57d545ba6071acee529ee3a8 (diff)
downloadgsoc2013-evolution-643a3d01716e1904d44b2dffca69219286df9d87.tar.gz
gsoc2013-evolution-643a3d01716e1904d44b2dffca69219286df9d87.tar.zst
gsoc2013-evolution-643a3d01716e1904d44b2dffca69219286df9d87.zip
Added cut/copy/paste support to the address editing dialog.
svn path=/trunk/; revision=1371
-rw-r--r--widgets/ChangeLog68
-rw-r--r--widgets/e-msg-composer-address-dialog.c394
-rw-r--r--widgets/e-msg-composer-address-dialog.glade8
-rw-r--r--widgets/e-msg-composer-address-dialog.h8
-rw-r--r--widgets/e-msg-composer-address-entry.c33
-rw-r--r--widgets/e-msg-composer-address-entry.h2
-rw-r--r--widgets/e-msg-composer-hdrs.c72
-rw-r--r--widgets/e-msg-composer-hdrs.h19
-rw-r--r--widgets/e-msg-composer.c74
-rw-r--r--widgets/e-table/ChangeLog68
10 files changed, 709 insertions, 37 deletions
diff --git a/widgets/ChangeLog b/widgets/ChangeLog
index e8426055fc..430c404172 100644
--- a/widgets/ChangeLog
+++ b/widgets/ChangeLog
@@ -1,3 +1,71 @@
+1999-11-08 Ettore Perazzoli <ettore@gnu.org>
+
+ * e-msg-composer-address-dialog.c: Implemented cut & paste for the
+ recipient lists.
+ (init): Initialize `cut_buffer'.
+ (destroy): Free it.
+ (recipient_clist_selection_get_cb): New function.
+ (recipient_clist_selection_received_cb): New function.
+ (recipient_clist_selection_clear_event_cb): New function.
+ (setup_recipient_list_signals): Install them as signal handlers
+ for "selection_get", "selection_received" and
+ "selection_clear_event" respectively.
+ (copy_recipient_cb): New function implementing the "copy"
+ operation.
+ (cut_recipient_cb): New function implementing the "cut" operation.
+ (paste_recipient_cb): New function implementing the "paste"
+ operation.
+
+ * e-msg-composer-address-dialog.h: New member `cut_buffer' in
+ `struct _EMsgComposerAddressDialog'.
+
+1999-11-07 Ettore Perazzoli <ettore@gnu.org>
+
+ * e-msg-composer-address-dialog.c: New context menu
+ `recipient_list_popup_info' for the recipient CLists.
+ (recipient_clist_button_press_cb): New function.
+ (setup_signals): Install it as the "button_press_event" signal
+ handler for popping up the CList context menu.
+
+ * e-msg-composer.c (free_string_list): New helper function.
+ (setup_address_dialog): Setup the initial values in the address
+ dialog according to the ones in the header widget.
+
+ * e-msg-composer-hdrs.c (e_msg_composer_hdrs_get_to): New function.
+ (e_msg_composer_hdrs_get_cc): New function.
+ (e_msg_composer_hdrs_get_bcc): New function.
+
+ * e-msg-composer.c (setup_address_dialog): New helper function.
+ (address_dialog_cb): Use it.
+
+ * e-msg-composer-address-dialog.c (add_address): Do not set the
+ row data anymore. Instead, put the full address description
+ (i.e. complete with the email address, not just the full name) in
+ the CList.
+ (add_address): Do nothing if no item is selected in the address
+ CList.
+ (get_list): Get the address list from the CList without passing
+ through the address CList.
+ (set_list): New helper function.
+ (e_msg_composer_address_dialog_set_to_list): New function.
+ (e_msg_composer_address_dialog_set_cc_list): New function.
+ (e_msg_composer_address_dialog_set_bcc_list): New function.
+
+ * e-msg-composer.c (address_dialog_apply_cb): Apply values from
+ the address dialog into the composer.
+
+ * e-msg-composer-hdrs.c (e_msg_composer_hdrs_set_to): New function.
+ (e_msg_composer_hdrs_set_cc): New function.
+ (e_msg_composer_hdrs_set_bcc): New function.
+
+ * e-msg-composer-address-entry.c
+ (e_msg_composer_address_entry_set_list): New function.
+
+ * e-msg-composer-address-dialog.c (apply): New helper function.
+ (clicked): New function, `clicked' method for the `GnomeDialog'
+ class.
+ (class_init): Install it.
+
1999-11-06 Ettore Perazzoli <ettore@gnu.org>
* e-msg-composer-attachment-bar.c (destroy): Call the destroy
diff --git a/widgets/e-msg-composer-address-dialog.c b/widgets/e-msg-composer-address-dialog.c
index aed0a61c9d..412ab987df 100644
--- a/widgets/e-msg-composer-address-dialog.c
+++ b/widgets/e-msg-composer-address-dialog.c
@@ -56,6 +56,16 @@ load_addresses (EMsgComposerAddressDialog *dialog)
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
@@ -64,20 +74,135 @@ add_address (EMsgComposerAddressDialog *dialog,
{
GtkCList *src_clist;
GtkCList *dest_clist;
- guint row;
+ 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;
- src_clist = GTK_CLIST (glade_xml_get_widget (dialog->gui, "address_clist"));
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, &text[0]);
+ gtk_clist_get_text (src_clist, row, 0, &name);
+ gtk_clist_get_text (src_clist, row, 1, &email);
+
+ text[0] = make_full_address (name, email);
text[1] = NULL;
+
gtk_clist_append (dest_clist, text);
- gtk_clist_set_row_data (dest_clist, dest_clist->rows - 1,
- GINT_TO_POINTER (row));
+
+ g_free (text[0]);
+}
+
+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;
+ gchar *text[2];
+
+ 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. */
@@ -119,17 +244,194 @@ glade_connect (GladeXML *gui,
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
-setup_signals (EMsgComposerAddressDialog *dialog)
+recipient_clist_selection_received_cb (GtkWidget *widget,
+ GtkSelectionData *selection_data,
+ guint time,
+ gpointer data)
{
- glade_connect (dialog->gui, "to_add_button", "clicked", add_to_cb,
+ 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, "cc_add_button", "clicked", add_cc_cb,
+ glade_connect (dialog->gui, name, "selection_get",
+ GTK_SIGNAL_FUNC (recipient_clist_selection_get_cb),
dialog);
- glade_connect (dialog->gui, "bcc_add_button", "clicked", add_bcc_cb,
+ 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. */
@@ -143,6 +445,7 @@ destroy (GtkObject *object)
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);
@@ -155,10 +458,14 @@ 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]
@@ -177,6 +484,7 @@ static void
init (EMsgComposerAddressDialog *dialog)
{
dialog->gui = NULL;
+ dialog->cut_buffer = NULL;
}
@@ -230,6 +538,7 @@ e_msg_composer_address_dialog_construct (EMsgComposerAddressDialog *dialog)
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);
}
@@ -246,15 +555,60 @@ e_msg_composer_address_dialog_new (void)
}
-static gchar *
-make_full_address (const gchar *name,
- const gchar *email)
+static void
+set_list (EMsgComposerAddressDialog *dialog,
+ const gchar *list_name,
+ GList *list)
{
- /* FIXME handle quoting. */
+ GtkCList *clist;
+ GList *p;
+ gchar *text[2];
- return g_strconcat (name, " <", email, ">", NULL);
+ 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;
+ 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)
@@ -264,20 +618,14 @@ get_list (EMsgComposerAddressDialog *dialog,
GList *list;
guint i;
- address_clist = GTK_CLIST (glade_xml_get_widget (dialog->gui,
- "address_clist"));
clist = GTK_CLIST (glade_xml_get_widget (dialog->gui, clist_name));
list = NULL;
for (i = 0; i < clist->rows; i++) {
- gchar *name, *email;
- guint addr_row;
-
- addr_row = GPOINTER_TO_INT (gtk_clist_get_row_data (clist, i));
- gtk_clist_get_text (clist, addr_row, 0, &name);
- gtk_clist_get_text (clist, addr_row, 0, &email);
+ gchar *addr;
- list = g_list_prepend (list, make_full_address (name, email));
+ gtk_clist_get_text (clist, i, 0, &addr);
+ list = g_list_prepend (list, g_strdup (addr));
}
return g_list_reverse (list);
diff --git a/widgets/e-msg-composer-address-dialog.glade b/widgets/e-msg-composer-address-dialog.glade
index 7a66d9ab8c..03452ea25b 100644
--- a/widgets/e-msg-composer-address-dialog.glade
+++ b/widgets/e-msg-composer-address-dialog.glade
@@ -163,7 +163,7 @@
<can_focus>True</can_focus>
<columns>2</columns>
<column_widths>128,80</column_widths>
- <selection_mode>GTK_SELECTION_SINGLE</selection_mode>
+ <selection_mode>GTK_SELECTION_BROWSE</selection_mode>
<show_titles>True</show_titles>
<shadow_type>GTK_SHADOW_IN</shadow_type>
@@ -316,7 +316,7 @@
<can_focus>True</can_focus>
<columns>1</columns>
<column_widths>80</column_widths>
- <selection_mode>GTK_SELECTION_SINGLE</selection_mode>
+ <selection_mode>GTK_SELECTION_BROWSE</selection_mode>
<show_titles>False</show_titles>
<shadow_type>GTK_SHADOW_IN</shadow_type>
@@ -365,7 +365,7 @@
<can_focus>True</can_focus>
<columns>1</columns>
<column_widths>80</column_widths>
- <selection_mode>GTK_SELECTION_SINGLE</selection_mode>
+ <selection_mode>GTK_SELECTION_BROWSE</selection_mode>
<show_titles>False</show_titles>
<shadow_type>GTK_SHADOW_IN</shadow_type>
@@ -414,7 +414,7 @@
<can_focus>True</can_focus>
<columns>1</columns>
<column_widths>80</column_widths>
- <selection_mode>GTK_SELECTION_SINGLE</selection_mode>
+ <selection_mode>GTK_SELECTION_BROWSE</selection_mode>
<show_titles>False</show_titles>
<shadow_type>GTK_SHADOW_IN</shadow_type>
diff --git a/widgets/e-msg-composer-address-dialog.h b/widgets/e-msg-composer-address-dialog.h
index 5ff4723aa9..19ffeda703 100644
--- a/widgets/e-msg-composer-address-dialog.h
+++ b/widgets/e-msg-composer-address-dialog.h
@@ -46,6 +46,8 @@ struct _EMsgComposerAddressDialog {
GnomeDialog parent;
GladeXML *gui;
+
+ gchar *cut_buffer;
};
struct _EMsgComposerAddressDialogClass {
@@ -58,6 +60,12 @@ struct _EMsgComposerAddressDialogClass {
GtkType e_msg_composer_address_dialog_get_type (void);
GtkWidget *e_msg_composer_address_dialog_new (void);
void e_msg_composer_address_dialog_construct (EMsgComposerAddressDialog *dialog);
+void e_msg_composer_address_dialog_set_to_list (EMsgComposerAddressDialog *dialog, GList *to_list);
+void e_msg_composer_address_dialog_set_cc_list (EMsgComposerAddressDialog *dialog, GList *cc_list);
+void e_msg_composer_address_dialog_set_bcc_list (EMsgComposerAddressDialog *dialog, GList *bcc_list);
+GList *e_msg_composer_address_dialog_get_to_list (EMsgComposerAddressDialog *dialog);
+GList *e_msg_composer_address_dialog_get_cc_list (EMsgComposerAddressDialog *dialog);
+GList *e_msg_composer_address_dialog_get_bcc_list (EMsgComposerAddressDialog *dialog);
#ifdef __cplusplus
}
diff --git a/widgets/e-msg-composer-address-entry.c b/widgets/e-msg-composer-address-entry.c
index 071730eef2..be63c815f5 100644
--- a/widgets/e-msg-composer-address-entry.c
+++ b/widgets/e-msg-composer-address-entry.c
@@ -140,3 +140,36 @@ e_msg_composer_address_entry_get_addresses (EMsgComposerAddressEntry *entry)
return g_list_reverse (list);
}
+
+/**
+ * e_msg_composer_address_entry_set_list:
+ * @entry: An address entry
+ * @list: List of pointers to strings representing the addresses that must
+ * appear in the entry
+ *
+ * Set the address list from @list.
+ **/
+void
+e_msg_composer_address_entry_set_list (EMsgComposerAddressEntry *entry,
+ GList *list)
+{
+ GString *string;
+ GList *p;
+
+ g_return_if_fail (entry != NULL);
+
+ if (list == NULL) {
+ gtk_editable_delete_text (GTK_EDITABLE (entry), -1, -1);
+ return;
+ }
+
+ string = g_string_new (NULL);
+ for (p = list; p != NULL; p = p->next) {
+ if (string->str[0] != '\0')
+ g_string_append (string, ", ");
+ g_string_append (string, p->data);
+ }
+
+ gtk_entry_set_text (GTK_ENTRY (entry), string->str);
+ g_string_free (string, TRUE);
+}
diff --git a/widgets/e-msg-composer-address-entry.h b/widgets/e-msg-composer-address-entry.h
index d4ab46797d..8e926272e4 100644
--- a/widgets/e-msg-composer-address-entry.h
+++ b/widgets/e-msg-composer-address-entry.h
@@ -53,6 +53,8 @@ struct _EMsgComposerAddressEntryClass {
GtkType e_msg_composer_address_entry_get_type (void);
GtkWidget *e_msg_composer_address_entry_new (void);
GList *e_msg_composer_address_entry_get_addresses (EMsgComposerAddressEntry *entry);
+void e_msg_composer_address_entry_set_list (EMsgComposerAddressEntry *entry,
+ GList *list);
#ifdef __cplusplus
}
diff --git a/widgets/e-msg-composer-hdrs.c b/widgets/e-msg-composer-hdrs.c
index 7269b0da8b..2fe537dabf 100644
--- a/widgets/e-msg-composer-hdrs.c
+++ b/widgets/e-msg-composer-hdrs.c
@@ -194,3 +194,75 @@ e_msg_composer_hdrs_to_message (EMsgComposerHdrs *hdrs,
set_recipients (msg, hdrs->priv->cc_entry, RECIPIENT_TYPE_CC);
set_recipients (msg, hdrs->priv->bcc_entry, RECIPIENT_TYPE_BCC);
}
+
+
+void
+e_msg_composer_hdrs_set_to (EMsgComposerHdrs *hdrs,
+ GList *to_list)
+{
+ EMsgComposerAddressEntry *entry;
+
+ g_return_if_fail (hdrs != NULL);
+ g_return_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs));
+
+ entry = E_MSG_COMPOSER_ADDRESS_ENTRY (hdrs->priv->to_entry);
+ e_msg_composer_address_entry_set_list (entry, to_list);
+}
+
+void
+e_msg_composer_hdrs_set_cc (EMsgComposerHdrs *hdrs,
+ GList *cc_list)
+{
+ EMsgComposerAddressEntry *entry;
+
+ g_return_if_fail (hdrs != NULL);
+ g_return_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs));
+
+ entry = E_MSG_COMPOSER_ADDRESS_ENTRY (hdrs->priv->cc_entry);
+ e_msg_composer_address_entry_set_list (entry, cc_list);
+}
+
+void
+e_msg_composer_hdrs_set_bcc (EMsgComposerHdrs *hdrs,
+ GList *bcc_list)
+{
+ EMsgComposerAddressEntry *entry;
+
+ g_return_if_fail (hdrs != NULL);
+ g_return_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs));
+
+ entry = E_MSG_COMPOSER_ADDRESS_ENTRY (hdrs->priv->bcc_entry);
+ e_msg_composer_address_entry_set_list (entry, bcc_list);
+}
+
+
+GList *
+e_msg_composer_hdrs_get_to (EMsgComposerHdrs *hdrs)
+{
+ g_return_val_if_fail (hdrs != NULL, NULL);
+ g_return_val_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs), NULL);
+
+ return e_msg_composer_address_entry_get_addresses
+ (E_MSG_COMPOSER_ADDRESS_ENTRY (hdrs->priv->to_entry));
+}
+
+GList *
+e_msg_composer_hdrs_get_cc (EMsgComposerHdrs *hdrs)
+{
+ g_return_val_if_fail (hdrs != NULL, NULL);
+ g_return_val_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs), NULL);
+
+ return e_msg_composer_address_entry_get_addresses
+ (E_MSG_COMPOSER_ADDRESS_ENTRY (hdrs->priv->cc_entry));
+}
+
+GList *
+e_msg_composer_hdrs_get_bcc (EMsgComposerHdrs *hdrs)
+{
+ g_return_val_if_fail (hdrs != NULL, NULL);
+ g_return_val_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs), NULL);
+
+ return e_msg_composer_address_entry_get_addresses
+ (E_MSG_COMPOSER_ADDRESS_ENTRY (hdrs->priv->bcc_entry));
+}
+
diff --git a/widgets/e-msg-composer-hdrs.h b/widgets/e-msg-composer-hdrs.h
index de8ac4ac13..404a4cb8ca 100644
--- a/widgets/e-msg-composer-hdrs.h
+++ b/widgets/e-msg-composer-hdrs.h
@@ -54,10 +54,21 @@ struct _EMsgComposerHdrsClass {
};
-GtkType e_msg_composer_hdrs_get_type (void);
-GtkWidget *e_msg_composer_hdrs_new (void);
-void e_msg_composer_hdrs_to_message (EMsgComposerHdrs *hdrs,
- CamelMimeMessage *msg);
+GtkType e_msg_composer_hdrs_get_type (void);
+GtkWidget *e_msg_composer_hdrs_new (void);
+void e_msg_composer_hdrs_to_message (EMsgComposerHdrs *hdrs,
+ CamelMimeMessage *msg);
+
+void e_msg_composer_hdrs_set_to (EMsgComposerHdrs *hdrs,
+ GList *to_list);
+void e_msg_composer_hdrs_set_cc (EMsgComposerHdrs *hdrs,
+ GList *cc_list);
+void e_msg_composer_hdrs_set_bcc (EMsgComposerHdrs *hdrs,
+ GList *bcc_list);
+
+GList *e_msg_composer_hdrs_get_to (EMsgComposerHdrs *hdrs);
+GList *e_msg_composer_hdrs_get_cc (EMsgComposerHdrs *hdrs);
+GList *e_msg_composer_hdrs_get_bcc (EMsgComposerHdrs *hdrs);
#ifdef _cplusplus
}
diff --git a/widgets/e-msg-composer.c b/widgets/e-msg-composer.c
index 6654ddfd17..453bed57f0 100644
--- a/widgets/e-msg-composer.c
+++ b/widgets/e-msg-composer.c
@@ -59,6 +59,20 @@ static guint signals[LAST_SIGNAL] = { 0 };
static GnomeAppClass *parent_class = NULL;
+static void
+free_string_list (GList *list)
+{
+ GList *p;
+
+ if (list == NULL)
+ return;
+
+ for (p = list; p != NULL; p = p->next)
+ g_free (p->data);
+
+ g_list_free (list);
+}
+
/* This functions builds a CamelMimeMessage for the message that the user has
composed in `composer'. */
static CamelMimeMessage *
@@ -134,6 +148,25 @@ address_dialog_destroy_cb (GtkWidget *widget,
composer->address_dialog = NULL;
}
+static void
+address_dialog_apply_cb (EMsgComposerAddressDialog *dialog,
+ gpointer data)
+{
+ EMsgComposerHdrs *hdrs;
+ GList *list;
+
+ hdrs = E_MSG_COMPOSER_HDRS (E_MSG_COMPOSER (data)->hdrs);
+
+ list = e_msg_composer_address_dialog_get_to_list (dialog);
+ e_msg_composer_hdrs_set_to (hdrs, list);
+
+ list = e_msg_composer_address_dialog_get_cc_list (dialog);
+ e_msg_composer_hdrs_set_cc (hdrs, list);
+
+ list = e_msg_composer_address_dialog_get_bcc_list (dialog);
+ e_msg_composer_hdrs_set_bcc (hdrs, list);
+}
+
/* Message composer window callbacks. */
@@ -177,19 +210,48 @@ add_attachment_cb (GtkWidget *widget,
NULL);
}
+/* Create the address dialog if not created already. */
+static void
+setup_address_dialog (EMsgComposer *composer)
+{
+ EMsgComposerAddressDialog *dialog;
+ EMsgComposerHdrs *hdrs;
+ GList *list;
+
+ if (composer->address_dialog != NULL)
+ return;
+
+ composer->address_dialog = e_msg_composer_address_dialog_new ();
+ dialog = E_MSG_COMPOSER_ADDRESS_DIALOG (composer->address_dialog);
+ hdrs = E_MSG_COMPOSER_HDRS (composer->hdrs);
+
+ gtk_signal_connect (GTK_OBJECT (dialog),
+ "destroy", address_dialog_destroy_cb, composer);
+ gtk_signal_connect (GTK_OBJECT (dialog),
+ "apply", address_dialog_apply_cb, composer);
+
+ list = e_msg_composer_hdrs_get_to (hdrs);
+ e_msg_composer_address_dialog_set_to_list (dialog, list);
+
+ list = e_msg_composer_hdrs_get_cc (hdrs);
+ e_msg_composer_address_dialog_set_cc_list (dialog, list);
+
+ list = e_msg_composer_hdrs_get_bcc (hdrs);
+ e_msg_composer_address_dialog_set_bcc_list (dialog, list);
+}
+
static void
address_dialog_cb (GtkWidget *widget,
gpointer data)
{
EMsgComposer *composer;
+ /* FIXME maybe we should hide the dialog on Cancel/OK instead of
+ destroying it. */
+
composer = E_MSG_COMPOSER (data);
- if (composer->address_dialog == NULL) {
- composer->address_dialog = e_msg_composer_address_dialog_new ();
- gtk_signal_connect (GTK_OBJECT (composer->address_dialog),
- "destroy", address_dialog_destroy_cb,
- composer);
- }
+
+ setup_address_dialog (composer);
gtk_widget_show (composer->address_dialog);
gdk_window_show (composer->address_dialog->window);
diff --git a/widgets/e-table/ChangeLog b/widgets/e-table/ChangeLog
index e8426055fc..430c404172 100644
--- a/widgets/e-table/ChangeLog
+++ b/widgets/e-table/ChangeLog
@@ -1,3 +1,71 @@
+1999-11-08 Ettore Perazzoli <ettore@gnu.org>
+
+ * e-msg-composer-address-dialog.c: Implemented cut & paste for the
+ recipient lists.
+ (init): Initialize `cut_buffer'.
+ (destroy): Free it.
+ (recipient_clist_selection_get_cb): New function.
+ (recipient_clist_selection_received_cb): New function.
+ (recipient_clist_selection_clear_event_cb): New function.
+ (setup_recipient_list_signals): Install them as signal handlers
+ for "selection_get", "selection_received" and
+ "selection_clear_event" respectively.
+ (copy_recipient_cb): New function implementing the "copy"
+ operation.
+ (cut_recipient_cb): New function implementing the "cut" operation.
+ (paste_recipient_cb): New function implementing the "paste"
+ operation.
+
+ * e-msg-composer-address-dialog.h: New member `cut_buffer' in
+ `struct _EMsgComposerAddressDialog'.
+
+1999-11-07 Ettore Perazzoli <ettore@gnu.org>
+
+ * e-msg-composer-address-dialog.c: New context menu
+ `recipient_list_popup_info' for the recipient CLists.
+ (recipient_clist_button_press_cb): New function.
+ (setup_signals): Install it as the "button_press_event" signal
+ handler for popping up the CList context menu.
+
+ * e-msg-composer.c (free_string_list): New helper function.
+ (setup_address_dialog): Setup the initial values in the address
+ dialog according to the ones in the header widget.
+
+ * e-msg-composer-hdrs.c (e_msg_composer_hdrs_get_to): New function.
+ (e_msg_composer_hdrs_get_cc): New function.
+ (e_msg_composer_hdrs_get_bcc): New function.
+
+ * e-msg-composer.c (setup_address_dialog): New helper function.
+ (address_dialog_cb): Use it.
+
+ * e-msg-composer-address-dialog.c (add_address): Do not set the
+ row data anymore. Instead, put the full address description
+ (i.e. complete with the email address, not just the full name) in
+ the CList.
+ (add_address): Do nothing if no item is selected in the address
+ CList.
+ (get_list): Get the address list from the CList without passing
+ through the address CList.
+ (set_list): New helper function.
+ (e_msg_composer_address_dialog_set_to_list): New function.
+ (e_msg_composer_address_dialog_set_cc_list): New function.
+ (e_msg_composer_address_dialog_set_bcc_list): New function.
+
+ * e-msg-composer.c (address_dialog_apply_cb): Apply values from
+ the address dialog into the composer.
+
+ * e-msg-composer-hdrs.c (e_msg_composer_hdrs_set_to): New function.
+ (e_msg_composer_hdrs_set_cc): New function.
+ (e_msg_composer_hdrs_set_bcc): New function.
+
+ * e-msg-composer-address-entry.c
+ (e_msg_composer_address_entry_set_list): New function.
+
+ * e-msg-composer-address-dialog.c (apply): New helper function.
+ (clicked): New function, `clicked' method for the `GnomeDialog'
+ class.
+ (class_init): Install it.
+
1999-11-06 Ettore Perazzoli <ettore@gnu.org>
* e-msg-composer-attachment-bar.c (destroy): Call the destroy