aboutsummaryrefslogblamecommitdiffstats
path: root/widgets/misc/e-attachment.c
blob: 27729b396faa9d57259c3afbbfff930dc4d49df1 (plain) (tree)



















                                                                           
                                                                                  






                    


                 
                                                         










                                                                    








                                
                    
                       
                        
 

                             
                          
                           
                            
                                  




                         
               

                   
 









                                                        
 


                         
                          
 

                                                         
 



                                                                                 
 




                                                                  




                                                                          
                                                 
         
 
                                       
                                       
 


                                                         
 







                                                        





                                                           
 



                                    
 

                                                        
 

                                          
                                               
 







                                                                                     







                                                                                    
 










                                        
                                    



                                                                 
                                     
                                       





                              
 











                                                    
 

                                                                                       
 


                    


























                                                                      


                                                  

                              
 
                                                  
 
 




                                                                    
 


                                                       
 
                                                         

                 

         
                         
 
                   
                                                          








                                                                           












                                                      
                                                                                     







                                  
                      
 
                                                       
 
                                               




                                                                     
 






                                                                                     
 
                                                                                 




                                                                     
 
                                                                   




                                                                                 
 







                                                                                       
 
                                    
 


                                                                       
 


                                                            
 









                                                                      
 




                                                     
                                
                                       
                                  
 
                                              


                                                      
 



                   
                
                                
                        

                                                 








                                                                           
               
 




































































































                                                                                                                                                                                            
 
















                                                                                                                         
                        
                                                     
                 
 










                                                                                                                                                                                                            
                 

         



                                                                                         
 



                                                   
 



                             

 
             
                                                                                                                                          

                         
                                    

                      
 
                                                 
 


                                               
 




                                                     
                                
                                        

                                                             
 
                      
 
                                                 

                                                             

                                                 




                                                                
 


                   
 
    
                                                                                                                            






                                  
                      
 
                                             
 
                                                 


                                                                     
                                                                                         

                       
 




                                                                                     
                                                                                   

                       
 
                                                                                 




                                                                     
 
                                                                   




                                                                                 
 







                                                                                       
 
                                    
 







                                                                       
 



                                                                       
 
                                                      





                                                                                
 



                                           
                                       
                                         
 
                                              


                                                             
 
 
 



                                   
  





                                                         
 
                                                               
 

                                                     

                                

                                       
                                                            
                                                                       
 






























                                                                       
 











                                                                         
 








                                                           
 

                                             
 


                                                 
 








                                          
 

                                             
 
                                                                
                                           
                                                                     

                                               
 



                                                                        
                                                 

                                                         
 




                                                                                                                        
 














                                                                                                           
 



















                                                              
                          
                   
                       
 
                                                        
 
                                             
                                                                                 


                                                 
 

                                                                                     
                          
 



                                                               
 
                                            
 

                                                                                               
 
                                            

                                                                          



                                                                                                                          
 
                                                                 

                                                                                                                



                                                                                   
 



                                                                                                          


                                                                                     




                                                                        
 
                                                                                                          
         
 
                                                                                               
 

                                                                                           
                                                                                                          
                                                                          
                                                                                                       

                                                                          
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
/*
 *  Authors: Ettore Perazzoli <ettore@ximian.com>
 *           Jeffrey Stedfast <fejj@ximian.com>
 *       Srinivasa Ragavan <sragavan@novell.com>
 *
 *  Copyright 1999-2005 Novell, Inc. (www.novell.com)
 *
 *  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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 *
 */

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

#include <glib.h>

#ifdef G_OS_WIN32
/* Include <windows.h> early (as the gio stuff below will
 * include it anyway, sigh) to workaround the DATADIR problem.
 * <windows.h> (and the headers it includes) stomps all over the
 * namespace like a baboon on crack, and especially the DATADIR enum
 * in objidl.h causes problems.
 */
#undef DATADIR
#define DATADIR crap_DATADIR
#include <windows.h>
#undef DATADIR
#endif

#include <sys/stat.h>
#include <string.h>
#include <errno.h>

#include <camel/camel.h>
#include <gtk/gtk.h>
#include <gtk/gtknotebook.h>
#include <gtk/gtktogglebutton.h>
#include <gtk/gtkdialog.h>
#include <gio/gio.h>
#include <glib/gi18n.h>
#include <glib/gstdio.h>

#include <libebook/e-vcard.h>

#include "e-util/e-util.h"
#include "e-util/e-error.h"
#include "e-util/e-mktemp.h"
#include "e-util/e-util-private.h"

#include "e-attachment.h"

enum {
    CHANGED,
    UPDATE,
    LAST_SIGNAL
};

static guint signals[LAST_SIGNAL] = { 0 };

static GObjectClass *parent_class = NULL;

static void
changed (EAttachment *attachment)
{
    g_signal_emit (attachment, signals[CHANGED], 0);
}


/* GtkObject methods.  */

static void
finalise (GObject *object)
{
    EAttachment *attachment = (EAttachment *) object;
    GtkWidget *dialog;

    if (attachment->editor_gui != NULL) {
        dialog = glade_xml_get_widget (attachment->editor_gui, "dialog");
        g_signal_emit_by_name (dialog, "response", GTK_RESPONSE_CLOSE);
    }

    if (attachment->is_available_local) {
        camel_object_unref (attachment->body);
        if (attachment->pixbuf_cache != NULL)
            g_object_unref (attachment->pixbuf_cache);
    } else {
        if (attachment->cancellable) {
            /* the operation is still running, so cancel it */
            g_cancellable_cancel (attachment->cancellable);
            attachment->cancellable = NULL;
        }
        g_free (attachment->description);
    }

    g_free (attachment->file_name);
    g_free (attachment->store_uri);

    G_OBJECT_CLASS (parent_class)->finalize (object);
}


/* Signals.  */

static void
real_changed (EAttachment *attachment)
{
    g_return_if_fail (E_IS_ATTACHMENT (attachment));
}

static void
real_update_attachment (EAttachment *attachment, char *msg)
{
    g_return_if_fail (E_IS_ATTACHMENT (attachment));
}


static void
class_init (EAttachmentClass *klass)
{
    GObjectClass *object_class;

    object_class = (GObjectClass*) klass;
    parent_class = g_type_class_ref (G_TYPE_OBJECT);

    object_class->finalize = finalise;
    klass->changed = real_changed;
    klass->update = real_update_attachment;

    signals[CHANGED] = g_signal_new ("changed",
                     E_TYPE_ATTACHMENT,
                     G_SIGNAL_RUN_FIRST,
                     G_STRUCT_OFFSET (EAttachmentClass, changed),
                     NULL,
                     NULL,
                     g_cclosure_marshal_VOID__VOID,
                     G_TYPE_NONE, 0);
    signals[UPDATE] = g_signal_new ("update",
                     E_TYPE_ATTACHMENT,
                     G_SIGNAL_RUN_FIRST,
                     G_STRUCT_OFFSET (EAttachmentClass, update),
                     NULL,
                     NULL,
                     g_cclosure_marshal_VOID__VOID,
                     G_TYPE_NONE, 0);

}

static void
init (EAttachment *attachment)
{
    attachment->editor_gui = NULL;
    attachment->body = NULL;
    attachment->size = 0;
    attachment->pixbuf_cache = NULL;
    attachment->index = -1;
    attachment->file_name = NULL;
    attachment->percentage = -1;
    attachment->description = NULL;
    attachment->disposition = FALSE;
    attachment->sign = CAMEL_CIPHER_VALIDITY_SIGN_NONE;
    attachment->encrypt = CAMEL_CIPHER_VALIDITY_ENCRYPT_NONE;
    attachment->store_uri = NULL;
    attachment->cancellable = NULL;
}

GType
e_attachment_get_type (void)
{
    static GType type = 0;

    if (type == 0) {
        static const GTypeInfo info = {
            sizeof (EAttachmentClass),
            NULL,
            NULL,
            (GClassInitFunc) class_init,
            NULL,
            NULL,
            sizeof (EAttachment),
            0,
            (GInstanceInitFunc) init,
        };

        type = g_type_register_static (G_TYPE_OBJECT, "EAttachment", &info, 0);
    }

    return type;
}

/**
 * file_ext_is:
 * @param file_name: path for file
 * @param ext: desired extension, with a dot
 * @return if file_name has extension ext or not
 **/

static gboolean
file_ext_is (const char *file_name, const char *ext)
{
    int i, dot = -1;

    if (!file_name || !ext)
        return FALSE;

    for (i = 0; file_name[i]; i++) {
        if (file_name [i] == '.')
            dot = i;
    }

    if (dot > 0) {
        return 0 == g_ascii_strcasecmp (file_name + dot, ext);
    }

    return FALSE;
}

static char *
attachment_guess_mime_type (const char *file_name)
{
    char *type;
    gchar *content = NULL;

    type = e_util_guess_mime_type (file_name);


    if (type && strcmp (type, "text/directory") == 0 &&
        file_ext_is (file_name, ".vcf") &&
        g_file_get_contents (file_name, &content, NULL, NULL) &&
        content) {
        EVCard *vc = e_vcard_new_from_string (content);

        if (vc) {
            g_free (type);
            g_object_unref (G_OBJECT (vc));

            type = g_strdup ("text/x-vcard");
        }

    }

    g_free (content);

    if (type) {
        /* Check if returned mime_type is valid */
        CamelContentType *ctype = camel_content_type_decode (type);

        if (!ctype) {
            g_free (type);
            type = NULL;
        } else
            camel_content_type_unref (ctype);
    }

    return type;
}


/**
 * e_attachment_new:
 * @file_name: filename to attach
 * @disposition: Content-Disposition of the attachment
 * @ex: exception
 *
 * Return value: the new attachment, or %NULL on error
 **/
EAttachment *
e_attachment_new (const char *file_name, const char *disposition, CamelException *ex)
{
    EAttachment *new;
    CamelMimePart *part;
    CamelDataWrapper *wrapper;
    CamelStream *stream;
    struct stat statbuf;
    char *mime_type;
    char *filename;
    CamelURL *url;

    g_return_val_if_fail (file_name != NULL, NULL);

    if (g_stat (file_name, &statbuf) < 0) {
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      _("Cannot attach file %s: %s"),
                      file_name, g_strerror (errno));
        return NULL;
    }

    /* return if it's not a regular file */
    if (!S_ISREG (statbuf.st_mode)) {
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      _("Cannot attach file %s: not a regular file"),
                      file_name);
        return NULL;
    }

    if (!(stream = camel_stream_fs_new_with_name (file_name, O_RDONLY, 0))) {
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      _("Cannot attach file %s: %s"),
                      file_name, g_strerror (errno));
        return NULL;
    }

    if ((mime_type = attachment_guess_mime_type (file_name))) {
        if (!g_ascii_strcasecmp (mime_type, "message/rfc822")) {
            wrapper = (CamelDataWrapper *) camel_mime_message_new ();
        } else {
            wrapper = camel_data_wrapper_new ();
        }

        camel_data_wrapper_construct_from_stream (wrapper, stream);
        camel_data_wrapper_set_mime_type (wrapper, mime_type);
        g_free (mime_type);
    } else {
        wrapper = camel_data_wrapper_new ();
        camel_data_wrapper_construct_from_stream (wrapper, stream);
        camel_data_wrapper_set_mime_type (wrapper, "application/octet-stream");
    }

    camel_object_unref (stream);

    part = camel_mime_part_new ();
    camel_medium_set_content_object (CAMEL_MEDIUM (part), wrapper);
    camel_object_unref (wrapper);

    camel_mime_part_set_disposition (part, disposition);
    filename = g_path_get_basename (file_name);
    camel_mime_part_set_filename (part, filename);

#if 0
    /* Note: Outlook 2002 is broken with respect to Content-Ids on
           non-multipart/related parts, so as an interoperability
           workaround, don't set a Content-Id on these parts. Fixes
           bug #10032 */
    /* set the Content-Id */
    content_id = camel_header_msgid_generate ();
    camel_mime_part_set_content_id (part, content_id);
    g_free (content_id);
#endif

    new = g_object_new (E_TYPE_ATTACHMENT, NULL);
    new->editor_gui = NULL;
    new->body = part;
    new->size = statbuf.st_size;
    new->guessed_type = TRUE;
    new->cancellable = NULL;
    new->is_available_local = TRUE;
    new->file_name = filename;

    url = camel_url_new ("file://", NULL);
    camel_url_set_path (url, file_name);
    new->store_uri = camel_url_to_string (url, 0);
    camel_url_free (url);

    return new;
}


typedef struct {
    EAttachment *attachment;
    char *file_name;
    char *uri;
    GtkWindow *parent; /* for error dialog */

    guint64 file_size; /* zero indicates unknown size */
    GInputStream *istream; /* read from here ... */
    GOutputStream *ostream; /* ...and write into this. */
    gboolean was_error;
    GCancellable *cancellable;

    void *buffer; /* read into this, not more than buffer_size bytes */
    gsize buffer_size;
} DownloadInfo;

static void
download_info_free (DownloadInfo *download_info)
{
    /* if there was an error, then free attachment too */
    if (download_info->was_error)
        g_object_unref (download_info->attachment);

    if (download_info->ostream)
        g_object_unref (download_info->ostream);

    if (download_info->istream)
        g_object_unref (download_info->istream);

    if (download_info->cancellable)
        g_object_unref (download_info->cancellable);

    g_free (download_info->file_name);
    g_free (download_info->uri);
    g_free (download_info->buffer);
    g_free (download_info);
}

static void
data_ready_cb (GObject *source_object, GAsyncResult *res, gpointer user_data)
{
    DownloadInfo *download_info = (DownloadInfo *)user_data;
    GError *error = NULL;
    gssize read;

    g_return_if_fail (download_info != NULL);

    if (g_cancellable_is_cancelled (download_info->cancellable)) {
        /* finish the operation and close both streams */
        g_input_stream_read_finish (G_INPUT_STREAM (source_object), res, NULL);

        g_output_stream_close (download_info->ostream, NULL, NULL);
        g_input_stream_close  (download_info->istream, NULL, NULL);

        /* The only way how to get this canceled is in EAttachment's finalize method,
           and because the download_info_free free's the attachment on error,
           then do not consider cancellation as an error. */
        download_info_free (download_info);
        return;
    }

    read = g_input_stream_read_finish (G_INPUT_STREAM (source_object), res, &error);

    if (!error)
        g_output_stream_write_all (download_info->ostream, download_info->buffer, read, NULL, download_info->cancellable, &error);

    if (error) {
        download_info->was_error = error->domain != G_IO_ERROR || error->code != G_IO_ERROR_CANCELLED;
        if (download_info->was_error)
            e_error_run (download_info->parent, "mail-composer:no-attach", download_info->uri, error->message, NULL);

        g_error_free (error);

        download_info->attachment->cancellable = NULL;
        download_info_free (download_info);
        return;
    }

    if (read == 0) {
        CamelException ex;

        /* done with reading */
        g_output_stream_close (download_info->ostream, NULL, NULL);
        g_input_stream_close  (download_info->istream, NULL, NULL);

        download_info->attachment->cancellable = NULL;

        camel_exception_init (&ex);
        e_attachment_build_remote_file (download_info->file_name, download_info->attachment, "attachment", &ex);

        if (camel_exception_is_set (&ex)) {
            download_info->was_error = TRUE;
            e_error_run (download_info->parent, "mail-composer:no-attach", download_info->uri, camel_exception_get_description (&ex), NULL);
            camel_exception_clear (&ex);
        }

        download_info->attachment->percentage = -1;
        download_info->attachment->is_available_local = TRUE;
        g_signal_emit (download_info->attachment, signals[UPDATE], 0);

        download_info_free (download_info);
        return;
    } else  if (download_info->file_size) {
        download_info->attachment->percentage = read * 100 / download_info->file_size;
        download_info->file_size -= MIN (download_info->file_size, read);
        g_signal_emit (download_info->attachment, signals[UPDATE], 0);
    } else {
        download_info->attachment->percentage = 0;
        g_signal_emit (download_info->attachment, signals[UPDATE], 0);
    }

    /* read next chunk */
    g_input_stream_read_async (download_info->istream, download_info->buffer, download_info->buffer_size, G_PRIORITY_DEFAULT, download_info->cancellable, data_ready_cb, download_info);
}

static gboolean
download_to_local_path (DownloadInfo *download_info, CamelException *ex)
{
    GError *error = NULL;
    GFile *src = g_file_new_for_uri (download_info->uri);
    GFile *des = g_file_new_for_path (download_info->file_name);
    gboolean res = FALSE;

    g_return_val_if_fail (src != NULL && des != NULL, FALSE);

    download_info->ostream = G_OUTPUT_STREAM (g_file_replace (des, NULL, FALSE, G_FILE_CREATE_NONE, NULL, &error));

    if (download_info->ostream && !error) {
        GFileInfo *fi;

        fi = g_file_query_info (src, G_FILE_ATTRIBUTE_STANDARD_SIZE, G_FILE_QUERY_INFO_NONE, NULL, NULL);

        if (fi) {
            download_info->file_size = g_file_info_get_attribute_uint64 (fi, G_FILE_ATTRIBUTE_STANDARD_SIZE);
            g_object_unref (fi);
        } else {
            download_info->file_size = 0;
        }

        download_info->istream = G_INPUT_STREAM (g_file_read (src, NULL, &error));

        if (download_info->istream && !error) {
            download_info->cancellable = g_cancellable_new ();
            download_info->attachment->cancellable = download_info->cancellable;
            download_info->buffer_size = 10240; /* max 10KB chunk */
            download_info->buffer = g_malloc (sizeof (char) * download_info->buffer_size);

            g_input_stream_read_async (download_info->istream, download_info->buffer, download_info->buffer_size, G_PRIORITY_DEFAULT, download_info->cancellable, data_ready_cb, download_info);

            res = TRUE;
        }
    }

    if (error) {
        /* propagate error */
        if (ex)
            camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM, error->message);

        g_error_free (error);
        download_info->was_error = TRUE;
        download_info_free (download_info);
    }

    g_object_unref (src);
    g_object_unref (des);

    return res;
}

EAttachment *
e_attachment_new_remote_file (GtkWindow *error_dlg_parent, const char *uri, const char *disposition, const char *path, CamelException *ex)
{
    EAttachment *new;
    DownloadInfo *download_info;
    CamelURL *url;
    char *base;

    g_return_val_if_fail (uri != NULL, NULL);

    url = camel_url_new (uri, NULL);
    base = g_path_get_basename (url->path);
    camel_url_free (url);

    new = g_object_new (E_TYPE_ATTACHMENT, NULL);
    new->editor_gui = NULL;
    new->body = NULL;
    new->size = 0;
    new->guessed_type = FALSE;
    new->cancellable = NULL;
    new->is_available_local = FALSE;
    new->percentage = 0;
    new->file_name = g_build_filename (path, base, NULL);

    g_free (base);

    download_info = g_new0 (DownloadInfo, 1);
    download_info->attachment = new;
    download_info->file_name = g_strdup (new->file_name);
    download_info->uri = g_strdup (uri);
    download_info->parent = error_dlg_parent;
    download_info->was_error = FALSE;

    /* it frees all on the error, so do not free it twice */
    if (!download_to_local_path (download_info, ex))
        return NULL;

    return new;
}


void
e_attachment_build_remote_file (const char *file_name, EAttachment *attachment, const char *disposition, CamelException *ex)
{
    CamelMimePart *part;
    CamelDataWrapper *wrapper;
    CamelStream *stream;
    struct stat statbuf;
    char *mime_type;
    char *filename;
    CamelURL *url;

    g_return_if_fail (file_name != NULL);

    if (g_stat (file_name, &statbuf) == -1) {
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      _("Cannot attach file %s: %s"),
                      file_name, g_strerror (errno));
        g_message ("Cannot attach file %s: %s\n", file_name, g_strerror (errno));
        return;
    }

    /* return if it's not a regular file */
    if (!S_ISREG (statbuf.st_mode)) {
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      _("Cannot attach file %s: not a regular file"),
                      file_name);
        g_message ("Cannot attach file %s: not a regular file", file_name);
        return;
    }

    if (!(stream = camel_stream_fs_new_with_name (file_name, O_RDONLY, 0))) {
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      _("Cannot attach file %s: %s"),
                      file_name, g_strerror (errno));
        return;
    }

    if ((mime_type = attachment_guess_mime_type (file_name))) {
        if (!g_ascii_strcasecmp (mime_type, "message/rfc822")) {
            wrapper = (CamelDataWrapper *) camel_mime_message_new ();
        } else {
            wrapper = camel_data_wrapper_new ();
        }

        camel_data_wrapper_construct_from_stream (wrapper, stream);
        camel_data_wrapper_set_mime_type (wrapper, mime_type);
        g_free (mime_type);
    } else {
        wrapper = camel_data_wrapper_new ();
        camel_data_wrapper_construct_from_stream (wrapper, stream);
        camel_data_wrapper_set_mime_type (wrapper, "application/octet-stream");
    }

    camel_object_unref (stream);

    part = camel_mime_part_new ();
    camel_medium_set_content_object (CAMEL_MEDIUM (part), wrapper);
    camel_object_unref (wrapper);

    if (attachment->disposition)
        camel_mime_part_set_disposition (part, "inline");
    else
        camel_mime_part_set_disposition (part, "attachment");

    if (!attachment->file_name)
        filename = g_path_get_basename (file_name);
    else
        filename = g_path_get_basename (attachment->file_name);

    camel_mime_part_set_filename (part, filename);

    if (attachment->description) {
        camel_mime_part_set_description (part, attachment->description);
        g_free (attachment->description);
        attachment->description = NULL;
    }

    attachment->editor_gui = NULL;
    attachment->body = part;
    attachment->size = statbuf.st_size;
    attachment->guessed_type = TRUE;
    g_free (attachment->file_name);
    attachment->file_name = filename;

    url = camel_url_new ("file://", NULL);
    camel_url_set_path (url, file_name);
    attachment->store_uri = camel_url_to_string (url, 0);
    camel_url_free (url);

}


/**
 * e_attachment_new_from_mime_part:
 * @part: a CamelMimePart
 *
 * Return value: a new EAttachment based on the mime part
 **/
EAttachment *
e_attachment_new_from_mime_part (CamelMimePart *part)
{
    EAttachment *new;

    g_return_val_if_fail (CAMEL_IS_MIME_PART (part), NULL);

    new = g_object_new (E_TYPE_ATTACHMENT, NULL);
    new->editor_gui = NULL;
    camel_object_ref (part);
    new->body = part;
    new->guessed_type = FALSE;
    new->is_available_local = TRUE;
    new->size = camel_mime_part_get_content_size (part);
    new->file_name = g_strdup (camel_mime_part_get_filename(part));

    return new;
}


/* The attachment property dialog.  */

typedef struct {
    GtkWidget *dialog;
    GtkEntry *file_name_entry;
    GtkEntry *description_entry;
    GtkEntry *mime_type_entry;
    GtkToggleButton *disposition_checkbox;
    EAttachment *attachment;
} DialogData;

static void
destroy_dialog_data (DialogData *data)
{
    g_free (data);
}

/*
 * fixme: I am converting EVERYTHING to/from UTF-8, although mime types
 * are in ASCII. This is not strictly necessary, but we want to be
 * consistent and possibly check for errors somewhere.
 */

static void
set_entry (GladeXML *xml, const char *widget_name, const char *value)
{
    GtkEntry *entry;

    entry = GTK_ENTRY (glade_xml_get_widget (xml, widget_name));
    if (entry == NULL)
        g_warning ("Entry for `%s' not found.", widget_name);
    else
        gtk_entry_set_text (entry, value ? value : "");
}

static void
connect_widget (GladeXML *gui, const char *name, const char *signal_name,
        GCallback func, gpointer data)
{
    GtkWidget *widget;

    widget = glade_xml_get_widget (gui, name);
    g_signal_connect (widget, signal_name, func, data);
}

static void
close_cb (GtkWidget *widget, gpointer data)
{
    EAttachment *attachment;
    DialogData *dialog_data;

    dialog_data = (DialogData *) data;
    attachment = dialog_data->attachment;

    gtk_widget_destroy (dialog_data->dialog);
    g_object_unref (attachment->editor_gui);
    attachment->editor_gui = NULL;

    destroy_dialog_data (dialog_data);
}

static void
ok_cb (GtkWidget *widget, gpointer data)
{
    DialogData *dialog_data;
    EAttachment *attachment;
    const char *str;

    dialog_data = (DialogData *) data;
    attachment = dialog_data->attachment;

    str = gtk_entry_get_text (dialog_data->file_name_entry);
    if (attachment->is_available_local)
        camel_mime_part_set_filename (attachment->body, str);
    g_free (attachment->file_name);
    attachment->file_name = g_strdup (str);

    str = gtk_entry_get_text (dialog_data->description_entry);
    if (attachment->is_available_local) {
        camel_mime_part_set_description (attachment->body, str);
    } else {
        g_free (attachment->description);
        attachment->description = g_strdup (str);
    }

    str = gtk_entry_get_text (dialog_data->mime_type_entry);
    if (attachment->is_available_local) {
        camel_mime_part_set_content_type (attachment->body, str);
        camel_data_wrapper_set_mime_type(camel_medium_get_content_object(CAMEL_MEDIUM (attachment->body)), str);
    }

    if (attachment->is_available_local) {
        switch (gtk_toggle_button_get_active (dialog_data->disposition_checkbox)) {
        case 0:
            camel_mime_part_set_disposition (attachment->body, "attachment");
            break;
        case 1:
            camel_mime_part_set_disposition (attachment->body, "inline");
            break;
        default:
            /* Hmmmm? */
            break;
        }
    } else {
        attachment->disposition = gtk_toggle_button_get_active (dialog_data->disposition_checkbox);
    }

    changed (attachment);
    close_cb (widget, data);
}

static void
response_cb (GtkWidget *widget, gint response, gpointer data)
{
    if (response == GTK_RESPONSE_OK)
        ok_cb (widget, data);
    else
        close_cb (widget, data);
}

void
e_attachment_edit (EAttachment *attachment, GtkWidget *parent)
{
    CamelContentType *content_type;
    const char *disposition;
    DialogData *dialog_data;
    GladeXML *editor_gui;
    GtkWidget *window;
    char *type;
    char *filename;

    g_return_if_fail (E_IS_ATTACHMENT (attachment));

    if (attachment->editor_gui != NULL) {
        window = glade_xml_get_widget (attachment->editor_gui, "dialog");
        gdk_window_show (window->window);
        return;
    }

    filename = g_build_filename (EVOLUTION_GLADEDIR, "e-attachment.glade", NULL);
    editor_gui = glade_xml_new (filename, NULL, NULL);
    g_free (filename);

    if (editor_gui == NULL) {
        g_warning ("Cannot load `e-attachment.glade'");
        return;
    }

    attachment->editor_gui = editor_gui;

    gtk_window_set_transient_for (GTK_WINDOW (glade_xml_get_widget (editor_gui, "dialog")),
                      GTK_WINDOW (gtk_widget_get_toplevel (parent)));

    dialog_data = g_new (DialogData, 1);
    dialog_data->attachment = attachment;
    dialog_data->dialog = glade_xml_get_widget (editor_gui, "dialog");
    dialog_data->file_name_entry = GTK_ENTRY (glade_xml_get_widget (editor_gui, "file_name_entry"));
    dialog_data->description_entry = GTK_ENTRY (glade_xml_get_widget (editor_gui, "description_entry"));
    dialog_data->mime_type_entry = GTK_ENTRY (glade_xml_get_widget (editor_gui, "mime_type_entry"));
    dialog_data->disposition_checkbox = GTK_TOGGLE_BUTTON (glade_xml_get_widget (editor_gui, "disposition_checkbox"));

    if (attachment->is_available_local && attachment->body) {
        set_entry (editor_gui, "file_name_entry", camel_mime_part_get_filename (attachment->body));
        set_entry (editor_gui, "description_entry", camel_mime_part_get_description (attachment->body));
        content_type = camel_mime_part_get_content_type (attachment->body);
        type = camel_content_type_simple (content_type);
        set_entry (editor_gui, "mime_type_entry", type);
        g_free (type);

        disposition = camel_mime_part_get_disposition (attachment->body);
        gtk_toggle_button_set_active (dialog_data->disposition_checkbox,
                          disposition && !g_ascii_strcasecmp (disposition, "inline"));
    } else {
        set_entry (editor_gui, "file_name_entry", attachment->file_name);
        set_entry (editor_gui, "description_entry", attachment->description);
        if ((type = attachment_guess_mime_type (attachment->file_name))) {
            set_entry (editor_gui, "mime_type_entry", type);
            g_free (type);
        } else {
            set_entry (editor_gui, "mime_type_entry", "");
        }

        gtk_toggle_button_set_active (dialog_data->disposition_checkbox, attachment->disposition);
    }

    connect_widget (editor_gui, "dialog", "response", (GCallback)response_cb, dialog_data);

    /* make sure that when the parent gets hidden/closed that our windows also close */
    parent = gtk_widget_get_toplevel (parent);
    gtk_signal_connect_while_alive (GTK_OBJECT (parent), "destroy", (GCallback) close_cb, dialog_data,
                    GTK_OBJECT (dialog_data->dialog));
    gtk_signal_connect_while_alive (GTK_OBJECT (parent), "hide", (GCallback) close_cb, dialog_data,
                    GTK_OBJECT (dialog_data->dialog));
}