aboutsummaryrefslogblamecommitdiffstats
path: root/composer/e-msg-composer-address-dialog.c
blob: 868bafb4142ab279cde187435bef354f90f216d1 (plain) (tree)
























                                                                           
                             





























                                                                                
                                                         

 









                                                                             







                                                                             
                            
                       





                                                                       
 


                                                                               


                                                       

                                                                      
                                                  
                       
 
                                                   

                         
                      






































































                                                                       









                                                        

 























                                                                        








































                                                                               











































                                                                              
           



                                                                        
 













































































                                                                               
                               

                                                                          
                               

                                                                                  


                               


















































                                                                                  






                                          



                                                        
                                    











                                                                      
                                             



                                                


                                                        

















                                                                                    
                                  











































                                                                           

                                                                   







                                                                               
                                         















                                                                       



                                            
 


                        
 







                                                                          
                                                      


                               

 






























                                                                               



                                            



                        



                                                                           
                            
 

                                                              






























                                                                               
/* -*- 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");
}