aboutsummaryrefslogblamecommitdiffstats
path: root/composer/e-msg-composer-hdrs.c
blob: 414ee2d2065ed32303f4732372f3ae0c6a09fea9 (plain) (tree)





























                                                                           





                                              

                                         

                                  
 

                             
 

                                                                                                 
                                 

                                                             


                                                    


                              
                                
                              



                                 


                               




                                          




                            





                        

                                 
 



























                                                                                              



                                             








                                          
                                                                                                            




                                   








                                                                









                                                    
                                      







                                                                                   

                                                                                


                                          

                                                                          





                                           
 








                                                                           
                                                                                      






                             































                                                                                                                

 
                  

                                   

                                     
                     



                                      
                  
        
                          

                                      
                                                         
                                                             



                                                                                


                                                                           



                                             
        


                                                                   
                                    
                                


                             
                                                              




                                                           




                                                    
         
 
                            
                                        
                
                                                          
                                                                           
                                                           
                                        
                

                                                                                     
        
                         
        








                                      


                                     
                                                                            

                                  



                                                          
                                  




                                                                            
                                  





                                                                            
                                  



                                                    
                                  













                                            







                                                                     
                                  



                                                                      



           
                                         


                                     

                                                

                                                              










                                                                          








                                                  

                                                    
                                   


                                   

                                   

                                             

                           



                                                                 






























                                                                      
                                      
        
                                                             
                         
        



                                                      
        
                            




                                
















































                                                                               




                                        

                    
                                     


                                                                                    
 


                                    



                                                



                                                                        








                                                       


                       


                                                         
                                                       








                                                         


                                                                              
 

 


















                                                                                              
    






                                                         
        
                                                                      
                                                


    
                                                   
                                                 
 


                                                         
                                                                  



                                                   
                                                 
 


                                                         
                                                                  



                                                    
                                                   
 


                                                         
                                                                    

 



                                                        



                                                         


                                                               

 





                                                                   
                                                                                                 

 
                                                 




                                                                   
        
                                
        
                    

 
                                                 




                                                                   
        
                                
        
                    

 
                                                 




                                                                   
        
                                
        
                    

 

                                                                                       

                                                        
                       
        

                                                                   
        



                                                               
 















                                                                   
        



















                                                                   
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
/* msg-composer-hdrs.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
 */

#ifdef _HAVE_CONFIG_H
#include <config.h>
#endif

#include <gnome.h>
#include <camel/camel.h>

#include <bonobo.h>

#include <liboaf/liboaf.h>

#include "Evolution-Addressbook-SelectNames.h"

#include "e-msg-composer-address-entry.h"
#include "e-msg-composer-hdrs.h"
#include <gal/e-text/e-entry.h>
#include <gal/widgets/e-unicode.h>

#include "mail/mail-config.h"


#define SELECT_NAMES_OAFID "OAFIID:addressbook:select-names:39301deb-174b-40d1-8a6e-5edc300f7b61"

struct _EMsgComposerHdrsPrivate {
    Evolution_Addressbook_SelectNames corba_select_names;

    /* Total number of headers that we have.  */
    guint num_hdrs;

    /* The tooltips.  */
    GtkTooltips *tooltips;

    /* Standard headers.  */
    GtkWidget *from_entry;
    GtkWidget *to_entry;
    GtkWidget *cc_entry;
    GtkWidget *bcc_entry;
    GtkWidget *subject_entry;

    /* Unique section ID */
    char *section_id;
};


static GtkTableClass *parent_class = NULL;

enum {
    SHOW_ADDRESS_DIALOG,
    LAST_SIGNAL
};

enum {
    HEADER_ADDRBOOK,
    HEADER_COMBOBOX,
    HEADER_ENTRYBOX
};

static gint signals[LAST_SIGNAL];


static gboolean
setup_corba (EMsgComposerHdrs *hdrs)
{
    EMsgComposerHdrsPrivate *priv;
    CORBA_Environment ev;

    priv = hdrs->priv;

    g_assert (priv->corba_select_names == CORBA_OBJECT_NIL);

    CORBA_exception_init (&ev);

    priv->corba_select_names = oaf_activate_from_id (SELECT_NAMES_OAFID, 0, NULL, &ev);

    /* OAF seems to be broken -- it can return a CORBA_OBJECT_NIL without
           raising an exception in `ev'.  */
    if (ev._major != CORBA_NO_EXCEPTION || priv->corba_select_names == CORBA_OBJECT_NIL) {
        g_warning ("Cannot activate -- %s", SELECT_NAMES_OAFID);
        CORBA_exception_free (&ev);
        return FALSE;
    }

    CORBA_exception_free (&ev);

    return TRUE;
}


static void
address_button_clicked_cb (GtkButton *button,
               gpointer data)
{
    EMsgComposerHdrs *hdrs;
    EMsgComposerHdrsPrivate *priv;
    CORBA_Environment ev;

    hdrs = E_MSG_COMPOSER_HDRS (data);
    priv = hdrs->priv;

    CORBA_exception_init (&ev);

    Evolution_Addressbook_SelectNames_activate_dialog (priv->corba_select_names, priv->section_id, &ev);

    CORBA_exception_free (&ev);
}

static GtkWidget *
create_dropdown_entry (EMsgComposerHdrs *hdrs,
               const char *name)
{
    GtkWidget *combo;
    GList *values = NULL;
    
    combo = gtk_combo_new ();
    gtk_combo_set_use_arrows (GTK_COMBO (combo), TRUE);
    gtk_combo_set_case_sensitive (GTK_COMBO (combo), FALSE);
    if (!strcmp (name, _("From:"))) {
        CamelInternetAddress *ciaddr;
        GSList *ids, *stmp;
        GList *tmp;
        MailConfigIdentity *id;
        char *val;
        
        ids = mail_config_get_identities ();
        stmp = ids;
        while (stmp) {
            char *address;
            
            id = stmp->data;
            g_assert (id);          
            g_assert (id->name);
            g_assert (id->address);
            
            ciaddr = camel_internet_address_new ();
            camel_internet_address_add (ciaddr, id->name, id->address);
            address = camel_address_encode (CAMEL_ADDRESS (ciaddr));
            values = g_list_append (values, address);
            stmp = stmp->next;
        }
        
        gtk_combo_set_popdown_strings (GTK_COMBO (combo), values);

        tmp = values;
        while (tmp) {
            g_free (tmp->data);
            tmp = tmp->next;
        }
        g_list_free (values);

        id = mail_config_get_default_identity ();
        g_assert (id);          
        g_assert (id->name);
        g_assert (id->address);
        
        ciaddr = camel_internet_address_new ();
        camel_internet_address_add (ciaddr, id->name, id->address);
        val = camel_address_encode (CAMEL_ADDRESS (ciaddr));
        
        e_utf8_gtk_entry_set_text (GTK_ENTRY (GTK_COMBO (combo)->entry), val);
        g_free (val);
    }
    
    return combo;
}

static GtkWidget *
create_addressbook_entry (EMsgComposerHdrs *hdrs,
              const char *name)
{
    EMsgComposerHdrsPrivate *priv;
    Evolution_Addressbook_SelectNames corba_select_names;
    Bonobo_Control corba_control;
    GtkWidget *control_widget;
    CORBA_Environment ev;

    priv = hdrs->priv;
    corba_select_names = priv->corba_select_names;

    CORBA_exception_init (&ev);

    Evolution_Addressbook_SelectNames_add_section (corba_select_names, name, name, &ev);
    if (ev._major != CORBA_NO_EXCEPTION) {
        CORBA_exception_free (&ev);
        return NULL;
    }

    corba_control = Evolution_Addressbook_SelectNames_get_entry_for_section (corba_select_names, name, &ev);

    if (ev._major != CORBA_NO_EXCEPTION) {
        CORBA_exception_free (&ev);
        return NULL;
    }

    CORBA_exception_free (&ev);

    control_widget = bonobo_widget_new_control_from_objref (corba_control, CORBA_OBJECT_NIL);

    return control_widget;
}

static GtkWidget *
add_header (EMsgComposerHdrs *hdrs,
        const gchar *name,
        const gchar *tip,
        const gchar *tip_private,
        int type)
{
    EMsgComposerHdrsPrivate *priv;
    GtkWidget *label;
    GtkWidget *entry;
    guint pad;
    
    priv = hdrs->priv;
    
    if (type == HEADER_ADDRBOOK) {
        label = gtk_button_new_with_label (name);
        GTK_OBJECT_UNSET_FLAGS(label, GTK_CAN_FOCUS);
        gtk_signal_connect (GTK_OBJECT (label), "clicked",
                    GTK_SIGNAL_FUNC (address_button_clicked_cb),
                    hdrs);
        pad = 2;
        gtk_tooltips_set_tip (hdrs->priv->tooltips, label,
                      _("Click here for the address book"),
                      NULL);
    } else {
        label = gtk_label_new (name);
        pad = GNOME_PAD;
    }
    
    gtk_table_attach (GTK_TABLE (hdrs), label,
              0, 1, priv->num_hdrs, priv->num_hdrs + 1,
              GTK_FILL, GTK_FILL,
              pad, pad);
    gtk_widget_show (label);
    
    switch (type) {
    case HEADER_ADDRBOOK:
        entry = create_addressbook_entry (hdrs, name);
        break;
    case HEADER_COMBOBOX:
        entry = create_dropdown_entry (hdrs, name);
        break;
    default:
        entry = e_entry_new ();
        gtk_object_set(GTK_OBJECT(entry),
                   "editable", TRUE,
                   "use_ellipsis", TRUE,
                   NULL);
    }

    if (entry != NULL) {
        gtk_widget_show (entry);
        
        gtk_table_attach (GTK_TABLE (hdrs), entry,
                  1, 2, priv->num_hdrs, priv->num_hdrs + 1,
                  GTK_FILL | GTK_EXPAND, 0,
                  2, 2);
        
        gtk_tooltips_set_tip (hdrs->priv->tooltips, entry, tip, tip_private);
    }
    
    priv->num_hdrs++;
    
    return entry;
}

static void
setup_headers (EMsgComposerHdrs *hdrs)
{
    EMsgComposerHdrsPrivate *priv;

    priv = hdrs->priv;
    
    priv->from_entry = add_header
        (hdrs, _("From:"), 
         _("Enter the identity you wish to send this message from"),
         NULL,
         HEADER_COMBOBOX);
    priv->to_entry = add_header
        (hdrs, _("To:"), 
         _("Enter the recipients of the message"),
         NULL,
         HEADER_ADDRBOOK);
    priv->cc_entry = add_header
        (hdrs, _("Cc:"),
         _("Enter the addresses that will receive a carbon copy of "
           "the message"),
         NULL,
         HEADER_ADDRBOOK);
    priv->bcc_entry = add_header
        (hdrs, _("Bcc:"),
         _("Enter the addresses that will receive a carbon copy of "
           "the message without appearing in the recipient list of "
           "the message."),
         NULL,
         HEADER_ADDRBOOK);
    priv->subject_entry = add_header
        (hdrs, _("Subject:"),
         _("Enter the subject of the mail"),
         NULL,
         HEADER_ENTRYBOX);
}


/* GtkObject methods.  */

static void
destroy (GtkObject *object)
{
    EMsgComposerHdrs *hdrs;
    EMsgComposerHdrsPrivate *priv;

    hdrs = E_MSG_COMPOSER_HDRS (object);
    priv = hdrs->priv;

    if (priv->corba_select_names != CORBA_OBJECT_NIL) {
        CORBA_Environment ev;

        CORBA_exception_init (&ev);
        CORBA_Object_release (priv->corba_select_names, &ev);
        CORBA_exception_free (&ev);
    }

    g_free (priv->section_id);
    gtk_object_destroy (GTK_OBJECT (priv->tooltips));

    if (GTK_OBJECT_CLASS (parent_class)->destroy != NULL)
        (* GTK_OBJECT_CLASS (parent_class)->destroy) (object);
}


static void
class_init (EMsgComposerHdrsClass *class)
{
    GtkObjectClass *object_class;

    object_class = GTK_OBJECT_CLASS (class);
    object_class->destroy = destroy;

    parent_class = gtk_type_class (gtk_table_get_type ());

    signals[SHOW_ADDRESS_DIALOG] =
        gtk_signal_new ("show_address_dialog",
                GTK_RUN_LAST,
                object_class->type,
                GTK_SIGNAL_OFFSET (EMsgComposerHdrsClass,
                           show_address_dialog),
                gtk_marshal_NONE__NONE,
                GTK_TYPE_NONE, 0);

    gtk_object_class_add_signals (object_class, signals, LAST_SIGNAL);
}

static void
init (EMsgComposerHdrs *hdrs)
{
    EMsgComposerHdrsPrivate *priv;

    priv = g_new (EMsgComposerHdrsPrivate, 1);

    priv->corba_select_names = CORBA_OBJECT_NIL;

    priv->from_entry    = NULL;
    priv->to_entry      = NULL;
    priv->cc_entry      = NULL;
    priv->bcc_entry     = NULL;
    priv->subject_entry = NULL;

    priv->tooltips = gtk_tooltips_new ();

    priv->num_hdrs = 0;

    /* Make a unique id from the addresses various things 
       This only needs to be unique as long as hdrs exists */
    priv->section_id = g_strdup_printf ("%p-%p", hdrs, priv);

    hdrs->priv = priv;
}


GtkType
e_msg_composer_hdrs_get_type (void)
{
    static GtkType type = 0;

    if (type == 0) {
        static const GtkTypeInfo info = {
            "EMsgComposerHdrs",
            sizeof (EMsgComposerHdrs),
            sizeof (EMsgComposerHdrsClass),
            (GtkClassInitFunc) class_init,
            (GtkObjectInitFunc) init,
            /* reserved_1 */ NULL,
            /* reserved_2 */ NULL,
            (GtkClassInitFunc) NULL,
        };

        type = gtk_type_unique (gtk_table_get_type (), &info);
    }

    return type;
}

GtkWidget *
e_msg_composer_hdrs_new (void)
{
    EMsgComposerHdrs *new;
    EMsgComposerHdrsPrivate *priv;
    
    new = gtk_type_new (e_msg_composer_hdrs_get_type ());
    priv = new->priv;
    
    if (! setup_corba (new)) {
        gtk_widget_destroy (GTK_WIDGET (new));
        return NULL;
    }
    
    setup_headers (new);

    return GTK_WIDGET (new);
}


static GList *
decode_addresses (const char *s)
{
    const char *p, *oldp;
    gboolean in_quotes;
    GList *list;

    g_print ("Decoding addresses -- %s\n", s ? s : "(null)");

    if (s == NULL)
        return NULL;

    in_quotes = FALSE;
    list = NULL;

    p = s;
    oldp = s;

    while (1) {
        if (*p == '"') {
            in_quotes = ! in_quotes;
            p++;
        } else if ((! in_quotes && *p == ',') || *p == 0) {
            if (p != oldp) {
                char *new_addr;

                new_addr = g_strndup (oldp, p - oldp);
                new_addr = g_strstrip (new_addr);
                if (*new_addr != '\0')
                    list = g_list_prepend (list, new_addr);
                else
                    g_free (new_addr);
            }

            while (*p == ',' || *p == ' ' || *p == '\t')
                p++;

            if (*p == 0)
                break;

            oldp = p;
        } else {
            p++;
        }
    }

    return g_list_reverse (list);
}

static void
set_recipients (CamelMimeMessage *msg,
        GtkWidget *entry_widget,
        const gchar *type)
{
    GList *list;
    GList *p;
    struct _header_address *addr;
    char *s;

    bonobo_widget_get_property (BONOBO_WIDGET (entry_widget), "text", &s, NULL);

    list = decode_addresses (s);

    g_free (s);

    /* FIXME leak?  */

    for (p = list; p != NULL; p = p->next) {
        addr = header_address_decode (p->data);
        camel_mime_message_add_recipient (msg, type, addr->name,
                          addr->v.addr);
        header_address_unref (addr);
    }

    g_list_free (list);
}

void
e_msg_composer_hdrs_to_message (EMsgComposerHdrs *hdrs,
                CamelMimeMessage *msg)
{
    gchar *subject;
    gchar *from;
    
    g_return_if_fail (hdrs != NULL);
    g_return_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs));
    g_return_if_fail (msg != NULL);
    g_return_if_fail (CAMEL_IS_MIME_MESSAGE (msg));
    
    subject = e_msg_composer_hdrs_get_subject (hdrs);
    camel_mime_message_set_subject (msg, subject);
    g_free (subject);
    
    from = e_msg_composer_hdrs_get_from (hdrs);
    camel_mime_message_set_from (msg, from);
    g_free (from);
    
    set_recipients (msg, hdrs->priv->to_entry, CAMEL_RECIPIENT_TYPE_TO);
    set_recipients (msg, hdrs->priv->cc_entry, CAMEL_RECIPIENT_TYPE_CC);
    set_recipients (msg, hdrs->priv->bcc_entry, CAMEL_RECIPIENT_TYPE_BCC);
}


static void
set_entry (BonoboWidget *bonobo_widget,
       const GList *list)
{
    GString *string;
    const GList *p;

    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);
    }

    bonobo_widget_set_property (BONOBO_WIDGET (bonobo_widget), "text", string->str, NULL);

    g_string_free (string, TRUE);
}

void
e_msg_composer_hdrs_set_from (EMsgComposerHdrs *hdrs,
                  const char *from)
{
    GtkEntry *entry;
    
    g_return_if_fail (hdrs != NULL);
    g_return_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs));
    
    entry = GTK_ENTRY (GTK_COMBO (hdrs->priv->from_entry)->entry);
    e_utf8_gtk_entry_set_text (entry, from);
}

void
e_msg_composer_hdrs_set_to (EMsgComposerHdrs *hdrs,
                const GList *to_list)
{
    g_return_if_fail (hdrs != NULL);
    g_return_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs));

    set_entry (BONOBO_WIDGET (hdrs->priv->to_entry), to_list);
}

void
e_msg_composer_hdrs_set_cc (EMsgComposerHdrs *hdrs,
                const GList *cc_list)
{
    g_return_if_fail (hdrs != NULL);
    g_return_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs));

    set_entry (BONOBO_WIDGET (hdrs->priv->cc_entry), cc_list);
}

void
e_msg_composer_hdrs_set_bcc (EMsgComposerHdrs *hdrs,
                 const GList *bcc_list)
{
    g_return_if_fail (hdrs != NULL);
    g_return_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs));

    set_entry (BONOBO_WIDGET (hdrs->priv->bcc_entry), bcc_list);
}

void
e_msg_composer_hdrs_set_subject (EMsgComposerHdrs *hdrs,
                 const char *subject)
{
    g_return_if_fail (hdrs != NULL);
    g_return_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs));
    g_return_if_fail (subject != NULL);

    gtk_object_set (GTK_OBJECT (hdrs->priv->subject_entry),
            "text", subject,
            NULL);
}

char *
e_msg_composer_hdrs_get_from (EMsgComposerHdrs *hdrs)
{
    g_return_val_if_fail (hdrs != NULL, NULL);
    g_return_val_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs), NULL);
    
    return e_utf8_gtk_entry_get_text (GTK_ENTRY (GTK_COMBO (hdrs->priv->from_entry)->entry));
}

/* FIXME this is currently unused and broken.  */
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);
    
    g_assert_not_reached ();
    
    return NULL;
}

/* FIXME this is currently unused and broken.  */
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);
    
    g_assert_not_reached ();
    
    return NULL;
}

/* FIXME this is currently unused and broken.  */
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);
    
    g_assert_not_reached ();
    
    return NULL;
}

/* FIXME: This is just changed to return allooc'd mem to be consistant with get_from */
char *
e_msg_composer_hdrs_get_subject (EMsgComposerHdrs *hdrs)
{
    gchar *subject;
    
    g_return_val_if_fail (hdrs != NULL, NULL);
    g_return_val_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs), NULL);
    
    gtk_object_get (GTK_OBJECT (hdrs->priv->subject_entry),
            "text", &subject, NULL);
    
    return g_strdup (subject);
}


GtkWidget *
e_msg_composer_hdrs_get_to_entry (EMsgComposerHdrs *hdrs)
{
    g_return_val_if_fail (hdrs != NULL, NULL);
    g_return_val_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs), NULL);

    return hdrs->priv->to_entry;
}

GtkWidget *
e_msg_composer_hdrs_get_cc_entry (EMsgComposerHdrs *hdrs)
{
    g_return_val_if_fail (hdrs != NULL, NULL);
    g_return_val_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs), NULL);
    
    return hdrs->priv->cc_entry;
}

GtkWidget *
e_msg_composer_hdrs_get_bcc_entry (EMsgComposerHdrs *hdrs)
{
    g_return_val_if_fail (hdrs != NULL, NULL);
    g_return_val_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs), NULL);

    return hdrs->priv->bcc_entry;
}

GtkWidget *
e_msg_composer_hdrs_get_subject_entry (EMsgComposerHdrs *hdrs)
{
    g_return_val_if_fail (hdrs != NULL, NULL);
    g_return_val_if_fail (E_IS_MSG_COMPOSER_HDRS (hdrs), NULL);

    return hdrs->priv->subject_entry;
}