aboutsummaryrefslogtreecommitdiffstats
path: root/widgets/misc/e-attachment-bar.c
diff options
context:
space:
mode:
Diffstat (limited to 'widgets/misc/e-attachment-bar.c')
-rw-r--r--widgets/misc/e-attachment-bar.c390
1 files changed, 269 insertions, 121 deletions
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 <camel/camel-mime-filter-bestenc.h>
#include <camel/camel-mime-part.h>
+#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"
@@ -207,11 +208,39 @@ 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