From a4e4725d360cb603c6c122227c0f781992fe9b52 Mon Sep 17 00:00:00 2001 From: Srinivasa Ragavan Date: Wed, 20 Jul 2005 11:17:42 +0000 Subject: Cleaned up most of it. Added the DnD. 2005-07-20 Srinivasa Ragavan * e-attachment-bar.[ch]:(remove_attachment) (update) (e_attachment_bar_get_selector) (e_attachment_bar_get_attachment) (destroy) (init) (e_attachment_bar_new) (e_attachment_bar_add_attachment) (e_attachment_bar_attach_remote_file) Cleaned up most of it. Added the DnD. * e-attachment.[ch]: (real_changed) (class_init) (init) (e_attachment_new) (e_attachment_new_remote_file) (e_attachment_build_remote_file): Moved the remote download code from bar to here and few cleanups. svn path=/trunk/; revision=29813 --- widgets/misc/ChangeLog | 10 ++ widgets/misc/e-attachment-bar.c | 390 +++++++++++++++++++++++++++------------- widgets/misc/e-attachment-bar.h | 6 +- widgets/misc/e-attachment.c | 127 ++++++++++++- widgets/misc/e-attachment.h | 5 +- 5 files changed, 407 insertions(+), 131 deletions(-) (limited to 'widgets') diff --git a/widgets/misc/ChangeLog b/widgets/misc/ChangeLog index a16b0c4265..cdfc8195ec 100644 --- a/widgets/misc/ChangeLog +++ b/widgets/misc/ChangeLog @@ -1,3 +1,13 @@ +2005-07-20 Srinivasa Ragavan + + * e-attachment-bar.[ch]:(remove_attachment) (update) (e_attachment_bar_get_selector) + (e_attachment_bar_get_attachment) (destroy) (init) (e_attachment_bar_new) + (e_attachment_bar_add_attachment) (e_attachment_bar_attach_remote_file) + Cleaned up most of it. Added the DnD. + * e-attachment.[ch]: (real_changed) (class_init) (init) (e_attachment_new) + (e_attachment_new_remote_file) (e_attachment_build_remote_file): Moved the + remote download code from bar to here and few cleanups. + 2005-07-11 Srinivasa Ragavan * Makefile.am: Added e-attachment-bar.[ch] e-attachment.[ch] diff --git a/widgets/misc/e-attachment-bar.c b/widgets/misc/e-attachment-bar.c index 35aefed59d..202702016a 100644 --- a/widgets/misc/e-attachment-bar.c +++ b/widgets/misc/e-attachment-bar.c @@ -49,6 +49,7 @@ #include #include +#include "e-util/e-util.h" #include "e-util/e-gui-utils.h" #include "e-util/e-icon-factory.h" #include "e-util/e-error.h" @@ -206,12 +207,40 @@ remove_attachment (EAttachmentBar *bar, /* Icon list contents handling. */ +static void +calculate_height_width(EAttachmentBar *bar, int *new_width, int *new_height) +{ + int width, height, icon_width; + PangoFontMetrics *metrics; + PangoContext *context; + GnomeIconList *icon_list; + + icon_list = GNOME_ICON_LIST (bar); + + context = gtk_widget_get_pango_context ((GtkWidget *) bar); + metrics = pango_context_get_metrics (context, ((GtkWidget *) bar)->style->font_desc, pango_context_get_language (context)); + width = PANGO_PIXELS (pango_font_metrics_get_approximate_char_width (metrics)) * 15; + /* This should be *2, but the icon list creates too much space above ... */ + height = PANGO_PIXELS (pango_font_metrics_get_ascent (metrics) + pango_font_metrics_get_descent (metrics)) * 3; + pango_font_metrics_unref (metrics); + icon_width = ICON_WIDTH + ICON_SPACING + ICON_BORDER + ICON_TEXT_SPACING; + + if (new_width) + *new_width = MAX (icon_width, width); + + if (new_height) + *new_height = ICON_WIDTH + ICON_SPACING + ICON_BORDER + ICON_TEXT_SPACING + height; + + return; +} + static void update (EAttachmentBar *bar) { EAttachmentBarPrivate *priv; GnomeIconList *icon_list; GList *p; + int bar_width, bar_height; priv = bar->priv; icon_list = GNOME_ICON_LIST (bar); @@ -365,16 +394,38 @@ update (EAttachmentBar *bar) } gnome_icon_list_thaw (icon_list); + + /* Resize */ + if (bar->expand) { + gtk_widget_get_size_request ((GtkWidget *)bar, &bar_width, &bar_height); + + if (bar->priv->num_attachments) { + int per_col, rows, height, width; + + calculate_height_width(bar, &width, &height); + per_col = bar_width / width; + rows = bar->priv->num_attachments / per_col; + gtk_widget_set_size_request ((GtkWidget *)bar, bar_width, (rows+1) * height); + } + } } static void -update_remote_file (EAttachmentBar *bar, EAttachment *attachment, char *msg) +update_remote_file (EAttachment *attachment, EAttachmentBar *bar) { EAttachmentBarPrivate *priv; GnomeIconList *icon_list; GnomeIconTextItem *item; - + char *msg, *base; priv = bar->priv; + + if (attachment->percentage == -1) + return update (bar); + + base = g_path_get_basename(attachment->file_name); + msg = g_strdup_printf("%s (%d\%)", base, attachment->percentage); + g_free(base); + icon_list = GNOME_ICON_LIST (bar); gnome_icon_list_freeze (icon_list); @@ -455,6 +506,28 @@ e_attachment_bar_get_selector(EAttachmentBar *bar) return &bar->priv->attach; } +GSList * +e_attachment_bar_get_selected (EAttachmentBar *bar) +{ + GSList *attachments = NULL; + GList *p; + + p = gnome_icon_list_get_selection((GnomeIconList *)bar); + for ( ; p != NULL; p = p->next) { + int num = GPOINTER_TO_INT(p->data); + EAttachment *attachment = g_list_nth_data(bar->priv->attachments, num); + + if (attachment && g_slist_find(attachments, attachment) == NULL) { + g_object_ref(attachment); + attachments = g_slist_prepend(attachments, attachment); + } + } + attachments = g_slist_reverse(attachments); + + return attachments; +} + +/* FIXME: Cleanup this, since there is a api to get selected attachments */ /* if id != -1, then use it as an index for target of the popup */ GSList * e_attachment_bar_get_attachment (EAttachmentBar *bar, int id) @@ -490,18 +563,18 @@ e_attachment_bar_get_attachment (EAttachmentBar *bar, int id) /* Just the GSList has to be freed by the caller */ GSList * -e_attachment_bar_get_attachment_part_list (EAttachmentBar *bar) +e_attachment_bar_get_parts (EAttachmentBar *bar) { EAttachment *attachment; - GSList *p = NULL, *part_list = NULL; + GList *p = NULL; + GSList *part_list = NULL; for ( p = bar->priv->attachments; p!= NULL; p = p->next) { attachment = p->data; - - if (attachment && attachment->is_available_local) { + if (attachment && attachment->is_available_local) part_list = g_slist_prepend(part_list, attachment->body); - } } + return part_list; } @@ -531,6 +604,182 @@ destroy (GtkObject *object) (* GTK_OBJECT_CLASS (parent_class)->destroy) (object); } +static char * +temp_save_part(CamelMimePart *part) +{ + const char *filename; + char *tmpdir, *path, *mfilename = NULL; + CamelStream *stream; + CamelDataWrapper *wrapper; + + tmpdir = e_mkdtemp("evolution-tmp-XXXXXX"); + if (tmpdir == NULL) { + return NULL; + } + + filename = camel_mime_part_get_filename (part); + if (filename == NULL) { + /* This is the default filename used for temporary file creation */ + filename = _("Unknown"); + } else { + mfilename = g_strdup(filename); + e_filename_make_safe(mfilename); + filename = mfilename; + } + + path = g_build_filename(tmpdir, filename, NULL); + g_free(tmpdir); + g_free(mfilename); + + wrapper = camel_medium_get_content_object (CAMEL_MEDIUM (part)); + stream = camel_stream_fs_new_with_name (path, O_RDWR|O_CREAT|O_TRUNC, 0600); + + if (!stream) { + /* TODO handle error conditions */ + g_message ("DEBUG: could not open the file to write\n"); + g_free (path); + return NULL; + } + + if (camel_data_wrapper_decode_to_stream (wrapper, (CamelStream *) stream) == -1) { + g_free (path); + camel_stream_close (stream); + camel_object_unref (stream); + g_message ("DEBUG: could not write to file\n"); + return NULL; + } + + camel_stream_close(stream); + camel_object_unref(stream); + + return path; +} + +static void +eab_drag_data_get(EAttachmentBar *bar, GdkDragContext *drag, GtkSelectionData *data, guint info, guint time) +{ + char *path; + GList *tmp; + gchar **uris; + int length, i=0; + + if (info) + return; + + tmp = gnome_icon_list_get_selection (GNOME_ICON_LIST(bar)); + length = g_list_length (tmp); + + uris = g_malloc0(sizeof(bar) * (length+1)); + + for (; tmp; tmp = tmp->next) { + int num = GPOINTER_TO_INT(tmp->data); + EAttachment *attachment = g_list_nth_data(bar->priv->attachments, num); + char *uri; + + if (!attachment->is_available_local) + continue; + + uri = g_object_get_data((GObject *)attachment, "e-drag-uri"); + if (uri) { + uris[i] = uri; + i++; + continue; + } + path = temp_save_part(attachment->body); + /* If we are not able to save, ignore it*/ + if (path == NULL) + continue; + + uri = g_strdup_printf("file://%s\r\n", path); + g_free(path); + g_object_set_data_full((GObject *)attachment, "e-drag-uri", uri, g_free); + uris[i] = uri; + i++; + } + uris[i]=0; + gtk_selection_data_set_uris(data, uris); + + g_free (uris); + + return; +} + +static gboolean +eab_button_release_event(EAttachmentBar *bar, GdkEventButton *event, gpointer dummy) +{ + GnomeIconList *icon_list = GNOME_ICON_LIST(bar); + GList *selected; + int length; + GtkTargetEntry drag_types[] = { + { "text/uri-list", 0, 0 }, + }; + + if (event && event->button == 1) { + selected = gnome_icon_list_get_selection(icon_list); + length = g_list_length (selected); + if (length) + gtk_drag_source_set((GtkWidget *)bar, GDK_BUTTON1_MASK, drag_types, G_N_ELEMENTS(drag_types), GDK_ACTION_COPY); + else + gtk_drag_source_unset((GtkWidget *)bar); + } + + return FALSE; +} + +static gboolean +eab_button_press_event(EAttachmentBar *bar, GdkEventButton *event, gpointer dummy) +{ + GnomeIconList *icon_list = GNOME_ICON_LIST(bar); + GList *selected = NULL, *tmp; + int length, icon_number; + gboolean take_selected = FALSE; + GtkTargetEntry drag_types[] = { + { "text/uri-list", 0, 0 }, + }; + + selected = gnome_icon_list_get_selection(icon_list); + length = g_list_length(selected); + + if (event) { + icon_number = gnome_icon_list_get_icon_at(icon_list, event->x, event->y); + if (icon_number < 0) { + /* When nothing is selected deselect all*/ + gnome_icon_list_unselect_all(icon_list); + length = 0; + selected = NULL; + } + + if (event->button == 1) { + /* If something is selected, then allow drag or else help to select */ + if (length) + gtk_drag_source_set((GtkWidget *)bar, GDK_BUTTON1_MASK, drag_types, G_N_ELEMENTS(drag_types), GDK_ACTION_COPY); + else + gtk_drag_source_unset((GtkWidget *)bar); + return FALSE; + } + + /* If not r-click dont progress any more.*/ + if (event->button != 3) + return FALSE; + + /* When a r-click on something, if it is in the already selected list, consider a r-click of multiple things + * or deselect all and select only this for r-click + */ + if (icon_number >= 0) { + for (tmp = selected; tmp; tmp = tmp->next) { + if (GPOINTER_TO_INT(tmp->data) == icon_number) + take_selected = TRUE; + } + + if (!take_selected) { + gnome_icon_list_unselect_all(icon_list); + gnome_icon_list_select_icon(icon_list, icon_number); + } + } + } + return FALSE; +} + /* Initialization. */ static void @@ -573,6 +822,7 @@ init (EAttachmentBar *bar) priv->path = NULL; bar->priv = priv; + bar->expand = FALSE; } @@ -603,28 +853,19 @@ e_attachment_bar_new (GtkAdjustment *adj) { EAttachmentBar *new; GnomeIconList *icon_list; - int width, height, icon_width, window_height; - PangoFontMetrics *metrics; - PangoContext *context; + int icon_width, window_height; new = g_object_new (e_attachment_bar_get_type (), NULL); icon_list = GNOME_ICON_LIST (new); - context = gtk_widget_get_pango_context ((GtkWidget *) new); - metrics = pango_context_get_metrics (context, ((GtkWidget *) new)->style->font_desc, pango_context_get_language (context)); - width = PANGO_PIXELS (pango_font_metrics_get_approximate_char_width (metrics)) * 15; - /* This should be *2, but the icon list creates too much space above ... */ - height = PANGO_PIXELS (pango_font_metrics_get_ascent (metrics) + pango_font_metrics_get_descent (metrics)) * 3; - pango_font_metrics_unref (metrics); - - icon_width = ICON_WIDTH + ICON_SPACING + ICON_BORDER + ICON_TEXT_SPACING; - icon_width = MAX (icon_width, width); + calculate_height_width (new, &icon_width, &window_height); gnome_icon_list_construct (icon_list, icon_width, adj, 0); - window_height = ICON_WIDTH + ICON_SPACING + ICON_BORDER + ICON_TEXT_SPACING + height; gtk_widget_set_size_request (GTK_WIDGET (new), icon_width * 4, window_height); + + GTK_WIDGET_SET_FLAGS (new, GTK_CAN_FOCUS); gnome_icon_list_set_separators (icon_list, ICON_SEPARATORS); gnome_icon_list_set_row_spacing (icon_list, ICON_ROW_SPACING); @@ -635,6 +876,10 @@ e_attachment_bar_new (GtkAdjustment *adj) atk_object_set_name (gtk_widget_get_accessible (GTK_WIDGET (new)), _("Attachment Bar")); + + g_signal_connect (new, "button_release_event", G_CALLBACK(eab_button_release_event), NULL); + g_signal_connect (new, "button_press_event", G_CALLBACK(eab_button_press_event), NULL); + g_signal_connect (new, "drag-data-get", G_CALLBACK(eab_drag_data_get), NULL); return GTK_WIDGET (new); } @@ -783,88 +1028,6 @@ e_attachment_bar_add_attachment (EAttachmentBar *bar, add_common (bar, attachment); } -typedef struct DownloadInfo { - EAttachment *attachment; - EAttachmentBar *bar; - gchar *file_name; -}DownloadInfo; - -static int -async_progress_update_cb (GnomeVFSAsyncHandle *handle, - GnomeVFSXferProgressInfo *info, - DownloadInfo *download_info) -{ - int percent=0; - switch (info->status) { - case GNOME_VFS_XFER_PROGRESS_STATUS_OK: - { - gchar *base_path = g_path_get_basename(download_info->attachment->file_name); - if (info->file_size) { - percent = info->bytes_copied*100/info->file_size; - update_remote_file (download_info->bar, - download_info->attachment, - g_strdup_printf("%s (%d\%)", base_path, percent)); - } else { - update_remote_file (download_info->bar, - download_info->attachment, - g_strdup_printf("%s (%d\%)", base_path, percent)); - } - g_free (base_path); - - if (info->phase == GNOME_VFS_XFER_PHASE_COMPLETED) { - CamelException ex; - - download_info->attachment->is_available_local = TRUE; - download_info->attachment->handle = NULL; - camel_exception_init (&ex); - e_attachment_build_remote_file (download_info->file_name, download_info->attachment, "attachment", &ex); - update(download_info->bar); - g_free (download_info->file_name); - g_free (download_info); - } - return TRUE; - break; - } - case GNOME_VFS_XFER_PROGRESS_STATUS_VFSERROR: - gnome_vfs_async_cancel (handle); - g_free (download_info->file_name); - g_free (download_info); - return FALSE; - break; - - default: - break; - } - - return TRUE; -} - -static void -download_to_local_path (GnomeVFSURI *source_uri, GnomeVFSURI *target_uri, DownloadInfo *download_info) - -{ - GnomeVFSResult result; - GList *source_uri_list = NULL; - GList *target_uri_list = NULL; - - source_uri_list = g_list_prepend (source_uri_list, source_uri); - target_uri_list = g_list_prepend (target_uri_list, target_uri); - - /* Callback info */ - result = gnome_vfs_async_xfer (&download_info->attachment->handle, /* handle_return */ - source_uri_list, /* source_uri_list */ - target_uri_list, /* target_uri_list */ - GNOME_VFS_XFER_DEFAULT, /* xfer_options */ - GNOME_VFS_XFER_ERROR_MODE_ABORT, /* error_mode */ - GNOME_VFS_XFER_OVERWRITE_MODE_REPLACE, /* overwrite_mode */ - GNOME_VFS_PRIORITY_DEFAULT, /* priority */ - async_progress_update_cb, /* progress_update_callback */ - download_info, /* update_callback_data */ - NULL, /* progress_sync_callback */ - NULL); /* sync_callback_data */ -} - - int e_attachment_bar_get_download_count (EAttachmentBar *bar) { @@ -891,37 +1054,22 @@ e_attachment_bar_attach_remote_file (EAttachmentBar *bar, { EAttachment *attachment; CamelException ex; - gchar *tmpfile; - gchar *base; - if (!bar->priv->path) - bar->priv->path = e_mkdtemp("attach-XXXXXX"); - base = g_path_get_basename (url); - g_return_if_fail (E_IS_ATTACHMENT_BAR (bar)); - tmpfile = g_build_filename (bar->priv->path, base, NULL); - g_free (base); + if (!bar->priv->path) + bar->priv->path = e_mkdtemp("attach-XXXXXX"); camel_exception_init (&ex); - attachment = e_attachment_new_remote_file (tmpfile, "attachment", &ex); + attachment = e_attachment_new_remote_file (url, "attachment", bar->priv->path, &ex); + g_signal_connect (attachment, "update", G_CALLBACK(update_remote_file), bar); if (attachment) { - DownloadInfo *download_info; - download_info = g_new (DownloadInfo, 1); - download_info->attachment = attachment; - download_info->bar =bar; - download_info->file_name = g_strdup (tmpfile); add_common (bar, attachment); - download_to_local_path (gnome_vfs_uri_new(url), gnome_vfs_uri_new(tmpfile), download_info); - } else { e_error_run((GtkWindow *)gtk_widget_get_toplevel((GtkWidget *)bar), "mail-composer:no-attach", attachment->file_name, camel_exception_get_description(&ex), NULL); camel_exception_clear (&ex); } - - g_free (tmpfile); - } void diff --git a/widgets/misc/e-attachment-bar.h b/widgets/misc/e-attachment-bar.h index 947376339e..c9ad12061c 100644 --- a/widgets/misc/e-attachment-bar.h +++ b/widgets/misc/e-attachment-bar.h @@ -52,7 +52,8 @@ typedef struct _EAttachmentBarPrivate EAttachmentBarPrivate; struct _EAttachmentBar { GnomeIconList parent; - + gboolean expand; + EAttachmentBarPrivate *priv; }; @@ -78,7 +79,8 @@ void e_attachment_bar_add_attachment (EAttachmentBar *bar, EAttachment *attachme void e_attachment_bar_edit_selected (EAttachmentBar *bar); void e_attachment_bar_remove_selected (EAttachmentBar *bar); GtkWidget ** e_attachment_bar_get_selector(EAttachmentBar *bar); -GSList *e_attachment_bar_get_attachment_part_list (EAttachmentBar *bar); +GSList *e_attachment_bar_get_parts (EAttachmentBar *bar); +GSList *e_attachment_bar_get_selected (EAttachmentBar *bar); #ifdef __cplusplus } #endif /* __cplusplus */ diff --git a/widgets/misc/e-attachment.c b/widgets/misc/e-attachment.c index 3e06467578..c813e9965a 100644 --- a/widgets/misc/e-attachment.c +++ b/widgets/misc/e-attachment.c @@ -44,6 +44,7 @@ enum { CHANGED, + UPDATE, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; @@ -91,6 +92,12 @@ 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) @@ -102,6 +109,7 @@ class_init (EAttachmentClass *klass) object_class->finalize = finalise; klass->changed = real_changed; + klass->update = real_update_attachment; signals[CHANGED] = g_signal_new ("changed", E_TYPE_ATTACHMENT, @@ -111,6 +119,15 @@ class_init (EAttachmentClass *klass) 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 @@ -122,6 +139,7 @@ init (EAttachment *attachment) 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; @@ -193,6 +211,7 @@ e_attachment_new (const char *file_name, struct stat statbuf; char *mime_type; char *filename; + char *uri; g_return_val_if_fail (file_name != NULL, NULL); @@ -265,19 +284,101 @@ e_attachment_new (const char *file_name, new->guessed_type = TRUE; new->handle = NULL; new->is_available_local = TRUE; + uri = g_strdup_printf("file://%s\r\n", file_name); + g_object_set_data_full((GObject *)new, "e-drag-uri", uri, g_free); return new; } +typedef struct DownloadInfo { + EAttachment *attachment; + gchar *file_name; +}DownloadInfo; + +static int +async_progress_update_cb (GnomeVFSAsyncHandle *handle, + GnomeVFSXferProgressInfo *info, + DownloadInfo *download_info) +{ + switch (info->status) { + case GNOME_VFS_XFER_PROGRESS_STATUS_OK: + { + if (info->file_size) { + download_info->attachment->percentage = info->bytes_copied*100/info->file_size; + g_signal_emit (download_info->attachment, signals[UPDATE], 0); + } else { + download_info->attachment->percentage = 0; + g_signal_emit (download_info->attachment, signals[UPDATE], 0); + } + + if (info->phase == GNOME_VFS_XFER_PHASE_COMPLETED) { + CamelException ex; + + download_info->attachment->is_available_local = TRUE; + download_info->attachment->handle = NULL; + camel_exception_init (&ex); + e_attachment_build_remote_file (download_info->file_name, download_info->attachment, "attachment", &ex); + download_info->attachment->percentage = -1; + g_signal_emit (download_info->attachment, signals[UPDATE], 0); + g_free (download_info->file_name); + g_free (download_info); + } + return TRUE; + break; + } + case GNOME_VFS_XFER_PROGRESS_STATUS_VFSERROR: + gnome_vfs_async_cancel (handle); + g_free (download_info->file_name); + g_free (download_info); + return FALSE; + break; + + default: + break; + } + + return TRUE; +} + +static void +download_to_local_path (GnomeVFSURI *source_uri, GnomeVFSURI *target_uri, DownloadInfo *download_info) + +{ + GnomeVFSResult result; + GList *source_uri_list = NULL; + GList *target_uri_list = NULL; + + source_uri_list = g_list_prepend (source_uri_list, source_uri); + target_uri_list = g_list_prepend (target_uri_list, target_uri); + + /* Callback info */ + result = gnome_vfs_async_xfer (&download_info->attachment->handle, /* handle_return */ + source_uri_list, /* source_uri_list */ + target_uri_list, /* target_uri_list */ + GNOME_VFS_XFER_DEFAULT, /* xfer_options */ + GNOME_VFS_XFER_ERROR_MODE_ABORT, /* error_mode */ + GNOME_VFS_XFER_OVERWRITE_MODE_REPLACE, /* overwrite_mode */ + GNOME_VFS_PRIORITY_DEFAULT, /* priority */ + async_progress_update_cb, /* progress_update_callback */ + download_info, /* update_callback_data */ + NULL, /* progress_sync_callback */ + NULL); /* sync_callback_data */ +} + EAttachment * -e_attachment_new_remote_file (const char *file_name, - const char *disposition, - CamelException *ex) +e_attachment_new_remote_file (const char *url, + const char *disposition, + const char *path, + CamelException *ex) { EAttachment *new; - - g_return_val_if_fail (file_name != NULL, NULL); + DownloadInfo *download_info; + gchar *base; + + g_return_val_if_fail (url != NULL, NULL); + + base = g_path_get_basename (url); new = g_object_new (E_TYPE_ATTACHMENT, NULL); new->editor_gui = NULL; @@ -286,8 +387,16 @@ e_attachment_new_remote_file (const char *file_name, new->guessed_type = FALSE; new->handle = NULL; new->is_available_local = FALSE; - new->file_name = g_path_get_basename(file_name); - + new->percentage = 0; + new->file_name = g_build_filename (path, base, NULL); + + g_free(base); + + download_info = g_new (DownloadInfo, 1); + download_info->attachment = new; + download_info->file_name = g_strdup (new->file_name); + download_to_local_path (gnome_vfs_uri_new(url), gnome_vfs_uri_new(new->file_name), download_info); + return new; } @@ -303,6 +412,7 @@ e_attachment_build_remote_file (const char *file_name, struct stat statbuf; char *mime_type; char *filename; + char *uri; g_return_if_fail (file_name != NULL); @@ -379,6 +489,9 @@ e_attachment_build_remote_file (const char *file_name, g_free (attachment->file_name); attachment->file_name = NULL; } + uri = g_strdup_printf("file://%s\r\n", file_name); + g_object_set_data_full((GObject *)attachment, "e-drag-uri", uri, g_free); + } diff --git a/widgets/misc/e-attachment.h b/widgets/misc/e-attachment.h index 1a0ef7d167..92686faec2 100644 --- a/widgets/misc/e-attachment.h +++ b/widgets/misc/e-attachment.h @@ -60,6 +60,7 @@ struct _EAttachment { GnomeVFSAsyncHandle *handle; gboolean is_available_local; + int percentage; char *file_name; char *description; gboolean disposition; @@ -74,14 +75,16 @@ struct _EAttachmentClass { GObjectClass parent_class; void (*changed) (EAttachment *attachment); + void (*update) (EAttachment *attachment, char *msg); }; GType e_attachment_get_type (void); EAttachment *e_attachment_new (const char *file_name, const char *disposition, CamelException *ex); -EAttachment * e_attachment_new_remote_file (const char *file_name, +EAttachment * e_attachment_new_remote_file (const char *url, const char *disposition, + const char *path, CamelException *ex); void e_attachment_build_remote_file (const char *filename, EAttachment *attachment, -- cgit