aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--calendar/ChangeLog10
-rw-r--r--calendar/gui/dialogs/comp-editor.c385
2 files changed, 394 insertions, 1 deletions
diff --git a/calendar/ChangeLog b/calendar/ChangeLog
index 7b59a7b45c..4b4a7d1ef3 100644
--- a/calendar/ChangeLog
+++ b/calendar/ChangeLog
@@ -1,3 +1,13 @@
+2005-02-05 Harish Krishnaswamy <kharish@novell.com>
+
+ * gui/dialogs/comp-editor.c (attach_message), (drop_action),
+ (drop_popup_copy), (drop_popup_move), (drop_popup_cancel),
+ (drop_popup_free), (drag_data_received), (drag_motion),
+ (comp_editor_class_init), (comp_editor_init), (real_edit_comp):
+ fix the dnd woes on calendar attachments.
+ * gui/dialogs/cal-attachment-bar.c (add_from_file): reuse
+ mail composer's error message.
+
2005-02-04 Harry Lu <harry.lu@sun.com>
Add a11y name for attachment button and bars.
diff --git a/calendar/gui/dialogs/comp-editor.c b/calendar/gui/dialogs/comp-editor.c
index d91c55a348..4ba8900b3a 100644
--- a/calendar/gui/dialogs/comp-editor.c
+++ b/calendar/gui/dialogs/comp-editor.c
@@ -38,6 +38,16 @@
#include <e-util/e-dialog-utils.h>
#include <e-util/e-icon-factory.h>
#include <evolution-shell-component-utils.h>
+
+#include <camel/camel-url.h>
+#include <camel/camel-exception.h>
+#include <camel/camel-folder.h>
+#include <camel/camel-stream-mem.h>
+#include <camel/camel-mime-message.h>
+
+#include "mail/mail-tools.h"
+#include "mail/em-popup.h"
+
#include "../print.h"
#include "../comp-util.h"
#include "save-comp.h"
@@ -50,8 +60,10 @@
#include "cal-attachment-bar.h"
#include "widgets/misc/e-expander.h"
+#include "widgets/misc/e-error.h"
+#define d(x) x
@@ -123,13 +135,378 @@ static void obj_removed_cb (ECal *client, GList *uids, gpointer data);
G_DEFINE_TYPE (CompEditor, comp_editor, GTK_TYPE_DIALOG);
+enum {
+ DND_TYPE_MESSAGE_RFC822,
+ DND_TYPE_X_UID_LIST,
+ DND_TYPE_TEXT_URI_LIST,
+ DND_TYPE_NETSCAPE_URL,
+ DND_TYPE_TEXT_VCARD,
+ DND_TYPE_TEXT_CALENDAR,
+};
+
+static GtkTargetEntry drop_types[] = {
+ { "message/rfc822", 0, DND_TYPE_MESSAGE_RFC822 },
+ { "x-uid-list", 0, DND_TYPE_X_UID_LIST },
+ { "text/uri-list", 0, DND_TYPE_TEXT_URI_LIST },
+ { "_NETSCAPE_URL", 0, DND_TYPE_NETSCAPE_URL },
+ { "text/x-vcard", 0, DND_TYPE_TEXT_VCARD },
+ { "text/calendar", 0, DND_TYPE_TEXT_CALENDAR },
+};
+
+#define num_drop_types (sizeof (drop_types) / sizeof (drop_types[0]))
+
+static struct {
+ char *target;
+ GdkAtom atom;
+ guint32 actions;
+} drag_info[] = {
+ { "message/rfc822", 0, GDK_ACTION_COPY },
+ { "x-uid-list", 0, GDK_ACTION_ASK|GDK_ACTION_MOVE|GDK_ACTION_COPY },
+ { "text/uri-list", 0, GDK_ACTION_COPY },
+ { "_NETSCAPE_URL", 0, GDK_ACTION_COPY },
+ { "text/x-vcard", 0, GDK_ACTION_COPY },
+ { "text/calendar", 0, GDK_ACTION_COPY },
+};
+
+static void
+attach_message(CompEditor *editor, CamelMimeMessage *msg)
+{
+ CamelMimePart *mime_part;
+ const char *subject;
+
+ mime_part = camel_mime_part_new();
+ camel_mime_part_set_disposition(mime_part, "inline");
+ subject = camel_mime_message_get_subject(msg);
+ if (subject) {
+ char *desc = g_strdup_printf(_("Attached message - %s"), subject);
+
+ camel_mime_part_set_description(mime_part, desc);
+ g_free(desc);
+ } else
+ camel_mime_part_set_description(mime_part, _("Attached message"));
+
+ camel_medium_set_content_object((CamelMedium *)mime_part, (CamelDataWrapper *)msg);
+ camel_mime_part_set_content_type(mime_part, "message/rfc822");
+ cal_attachment_bar_attach_mime_part(CAL_ATTACHMENT_BAR(editor->priv->attachment_bar), mime_part);
+ camel_object_unref(mime_part);
+}
+
+struct _drop_data {
+ CompEditor *editor;
+
+ GdkDragContext *context;
+ /* Only selection->data and selection->length are valid */
+ GtkSelectionData *selection;
+
+ guint32 action;
+ guint info;
+ guint time;
+
+ unsigned int move:1;
+ unsigned int moved:1;
+ unsigned int aborted:1;
+};
+
+static void
+drop_action(CompEditor *editor, GdkDragContext *context, guint32 action, GtkSelectionData *selection, guint info, guint time)
+{
+ char *tmp, *str, **urls;
+ CamelMimePart *mime_part;
+ CamelStream *stream;
+ CamelURL *url;
+ CamelMimeMessage *msg;
+ char *content_type;
+ int i, success=FALSE, delete=FALSE;
+
+ switch (info) {
+ case DND_TYPE_MESSAGE_RFC822:
+ d(printf ("dropping a message/rfc822\n"));
+ /* write the message(s) out to a CamelStream so we can use it */
+ stream = camel_stream_mem_new ();
+ camel_stream_write (stream, selection->data, selection->length);
+ camel_stream_reset (stream);
+
+ msg = camel_mime_message_new ();
+ if (camel_data_wrapper_construct_from_stream((CamelDataWrapper *)msg, stream) != -1) {
+ attach_message(editor, msg);
+ success = TRUE;
+ delete = action == GDK_ACTION_MOVE;
+ }
+
+ camel_object_unref(msg);
+ camel_object_unref(stream);
+ break;
+ case DND_TYPE_TEXT_URI_LIST:
+ case DND_TYPE_NETSCAPE_URL:
+ d(printf ("dropping a text/uri-list\n"));
+ tmp = g_strndup (selection->data, selection->length);
+ urls = g_strsplit (tmp, "\n", 0);
+ g_free (tmp);
+
+ for (i = 0; urls[i] != NULL; i++) {
+ str = g_strstrip (urls[i]);
+ if (urls[i][0] == '#') {
+ g_free(str);
+ continue;
+ }
+
+ if (!g_ascii_strncasecmp (str, "mailto:", 7)) {
+ /* TODO does not handle mailto now */
+ g_free (str);
+ } else {
+ url = camel_url_new (str, NULL);
+ g_free (str);
+
+ if (url == NULL)
+ continue;
+
+ if (!g_ascii_strcasecmp (url->protocol, "file"))
+ cal_attachment_bar_attach
+ (CAL_ATTACHMENT_BAR (editor->priv->attachment_bar),
+ url->path);
+
+ camel_url_free (url);
+ }
+ }
+
+ g_free (urls);
+ success = TRUE;
+ break;
+ case DND_TYPE_TEXT_VCARD:
+ case DND_TYPE_TEXT_CALENDAR:
+ content_type = gdk_atom_name (selection->type);
+ d(printf ("dropping a %s\n", content_type));
+
+ mime_part = camel_mime_part_new ();
+ camel_mime_part_set_content (mime_part, selection->data, selection->length, content_type);
+ camel_mime_part_set_disposition (mime_part, "inline");
+
+ cal_attachment_bar_attach_mime_part
+ (CAL_ATTACHMENT_BAR (editor->priv->attachment_bar),
+ mime_part);
+
+ camel_object_unref (mime_part);
+ g_free (content_type);
+
+ success = TRUE;
+ break;
+ case DND_TYPE_X_UID_LIST: {
+ GPtrArray *uids;
+ char *inptr, *inend;
+ CamelFolder *folder;
+ CamelException ex = CAMEL_EXCEPTION_INITIALISER;
+
+ /* NB: This all runs synchronously, could be very slow/hang/block the ui */
+
+ uids = g_ptr_array_new();
+
+ inptr = selection->data;
+ inend = selection->data + selection->length;
+ while (inptr < inend) {
+ char *start = inptr;
+
+ while (inptr < inend && *inptr)
+ inptr++;
+
+ if (start > (char *)selection->data)
+ g_ptr_array_add(uids, g_strndup(start, inptr-start));
+
+ inptr++;
+ }
+
+ if (uids->len > 0) {
+ folder = mail_tool_uri_to_folder(selection->data, 0, &ex);
+ if (folder) {
+ if (uids->len == 1) {
+ msg = camel_folder_get_message(folder, uids->pdata[0], &ex);
+ if (msg == NULL)
+ goto fail;
+ attach_message(editor, msg);
+ } else {
+ CamelMultipart *mp = camel_multipart_new();
+ char *desc;
+
+ camel_data_wrapper_set_mime_type((CamelDataWrapper *)mp, "multipart/digest");
+ camel_multipart_set_boundary(mp, NULL);
+ for (i=0;i<uids->len;i++) {
+ msg = camel_folder_get_message(folder, uids->pdata[i], &ex);
+ if (msg) {
+ mime_part = camel_mime_part_new();
+ camel_mime_part_set_disposition(mime_part, "inline");
+ camel_medium_set_content_object((CamelMedium *)mime_part, (CamelDataWrapper *)msg);
+ camel_mime_part_set_content_type(mime_part, "message/rfc822");
+ camel_multipart_add_part(mp, mime_part);
+ camel_object_unref(mime_part);
+ camel_object_unref(msg);
+ } else {
+ camel_object_unref(mp);
+ goto fail;
+ }
+ }
+ mime_part = camel_mime_part_new();
+ camel_medium_set_content_object((CamelMedium *)mime_part, (CamelDataWrapper *)mp);
+ /* translators, this count will always be >1 */
+ desc = g_strdup_printf(ngettext("Attached message", "%d attached messages", uids->len), uids->len);
+ camel_mime_part_set_description(mime_part, desc);
+ g_free(desc);
+ cal_attachment_bar_attach_mime_part
+ (CAL_ATTACHMENT_BAR(editor->priv->attachment_bar), mime_part);
+ camel_object_unref(mime_part);
+ camel_object_unref(mp);
+ }
+ success = TRUE;
+ delete = action == GDK_ACTION_MOVE;
+ fail:
+ if (camel_exception_is_set(&ex)) {
+ char *name;
+
+ camel_object_get(folder, NULL, CAMEL_FOLDER_NAME, &name, NULL);
+ e_error_run((GtkWindow *)editor, "mail-editor:attach-nomessages",
+ name?name:(char *)selection->data, camel_exception_get_description(&ex), NULL);
+ camel_object_free(folder, CAMEL_FOLDER_NAME, name);
+ }
+ camel_object_unref(folder);
+ } else {
+ e_error_run((GtkWindow *)editor, "mail-editor:attach-nomessages",
+ selection->data, camel_exception_get_description(&ex), NULL);
+ }
+
+ camel_exception_clear(&ex);
+ }
+
+ g_ptr_array_free(uids, TRUE);
+
+ break; }
+ default:
+ d(printf ("dropping an unknown\n"));
+ break;
+ }
+
+ printf("Drag finished, success %d delete %d\n", success, delete);
+
+ gtk_drag_finish(context, success, delete, time);
+}
+
+static void
+drop_popup_copy (EPopup *ep, EPopupItem *item, void *data)
+{
+ struct _drop_data *m = data;
+ drop_action(m->editor, m->context, GDK_ACTION_COPY, m->selection, m->info, m->time);
+}
+
+static void
+drop_popup_move (EPopup *ep, EPopupItem *item, void *data)
+{
+ struct _drop_data *m = data;
+ drop_action(m->editor, m->context, GDK_ACTION_MOVE, m->selection, m->info, m->time);
+}
+
+static void
+drop_popup_cancel(EPopup *ep, EPopupItem *item, void *data)
+{
+ struct _drop_data *m = data;
+ gtk_drag_finish(m->context, FALSE, FALSE, m->time);
+}
+
+static EPopupItem drop_popup_menu[] = {
+ { E_POPUP_ITEM, "00.emc.02", N_("_Copy"), drop_popup_copy, NULL, "stock_mail-copy", 0 },
+ { E_POPUP_ITEM, "00.emc.03", N_("_Move"), drop_popup_move, NULL, "stock_mail-move", 0 },
+ { E_POPUP_BAR, "10.emc" },
+ { E_POPUP_ITEM, "99.emc.00", N_("Cancel _Drag"), drop_popup_cancel, NULL, NULL, 0 },
+};
+
+static void
+drop_popup_free(EPopup *ep, GSList *items, void *data)
+{
+ struct _drop_data *m = data;
+
+ g_slist_free(items);
+
+ g_object_unref(m->context);
+ g_object_unref(m->editor);
+ g_free(m->selection->data);
+ g_free(m->selection);
+ g_free(m);
+}
+
+static void
+drag_data_received (CompEditor *editor, GdkDragContext *context,
+ int x, int y, GtkSelectionData *selection,
+ guint info, guint time)
+{
+ if (selection->data == NULL || selection->length == -1)
+ return;
+
+ if (context->action == GDK_ACTION_ASK) {
+ EMPopup *emp;
+ GSList *menus = NULL;
+ GtkMenu *menu;
+ int i;
+ struct _drop_data *m;
+
+ m = g_malloc0(sizeof(*m));
+ m->context = context;
+ g_object_ref(context);
+ m->editor = editor;
+ g_object_ref(editor);
+ m->action = context->action;
+ m->info = info;
+ m->time = time;
+ m->selection = g_malloc0(sizeof(*m->selection));
+ m->selection->data = g_malloc(selection->length);
+ memcpy(m->selection->data, selection->data, selection->length);
+ m->selection->length = selection->length;
+
+ emp = em_popup_new("org.gnome.evolution.mail.editor.popup.drop");
+ for (i=0;i<sizeof(drop_popup_menu)/sizeof(drop_popup_menu[0]);i++)
+ menus = g_slist_append(menus, &drop_popup_menu[i]);
+
+ e_popup_add_items((EPopup *)emp, menus, NULL, drop_popup_free, m);
+ menu = e_popup_create_menu_once((EPopup *)emp, NULL, 0);
+ gtk_menu_popup(menu, NULL, NULL, NULL, NULL, 0, time);
+ } else {
+ drop_action(editor, context, context->action, selection, info, time);
+ }
+}
+
+static gboolean
+drag_motion(GObject *o, GdkDragContext *context, gint x, gint y, guint time, CompEditor *editor)
+{
+ GList *targets;
+ GdkDragAction action, actions = 0;
+
+ for (targets = context->targets; targets; targets = targets->next) {
+ int i;
+
+ for (i=0;i<sizeof(drag_info)/sizeof(drag_info[0]);i++)
+ if (targets->data == (void *)drag_info[i].atom)
+ actions |= drag_info[i].actions;
+ }
+
+ actions &= context->actions;
+ action = context->suggested_action;
+ /* we default to copy */
+ if (action == GDK_ACTION_ASK && (actions & (GDK_ACTION_MOVE|GDK_ACTION_COPY)) != (GDK_ACTION_MOVE|GDK_ACTION_COPY))
+ action = GDK_ACTION_COPY;
+
+ gdk_drag_status(context, action, time);
+
+ return action != 0;
+}
+
/* Class initialization function for the calendar component editor */
static void
comp_editor_class_init (CompEditorClass *klass)
{
GObjectClass *object_class;
GtkWidgetClass *widget_class;
+ GObjectClass *gobject_class;
+ int i;
+ for (i=0;i<sizeof(drag_info)/sizeof(drag_info[0]);i++)
+ drag_info[i].atom = gdk_atom_intern(drag_info[i].target, FALSE);
+
+ gobject_class = G_OBJECT_CLASS(klass);
object_class = G_OBJECT_CLASS (klass);
widget_class = GTK_WIDGET_CLASS (klass);
@@ -607,6 +984,11 @@ comp_editor_init (CompEditor *editor)
priv->is_group_item = FALSE;
priv->help_section = g_strdup ("usage-calendar");
+ /* DND support */
+ gtk_drag_dest_set (GTK_WIDGET (editor), GTK_DEST_DEFAULT_ALL, drop_types, num_drop_types, GDK_ACTION_COPY|GDK_ACTION_ASK|GDK_ACTION_MOVE);
+ g_signal_connect(editor, "drag_data_received", G_CALLBACK (drag_data_received), NULL);
+ g_signal_connect(editor, "drag-motion", G_CALLBACK(drag_motion), editor);
+
gtk_window_set_type_hint (GTK_WINDOW (editor), GDK_WINDOW_TYPE_HINT_NORMAL);
gtk_dialog_set_has_separator (GTK_DIALOG (editor), FALSE);
@@ -1334,7 +1716,7 @@ real_edit_comp (CompEditor *editor, ECalComponent *comp)
set_title_from_comp (editor);
set_icon_from_comp (editor);
e_cal_component_get_uid (comp, &uid);
- cal_attachment_bar_set_local_attachment_store (priv->attachment_bar,
+ cal_attachment_bar_set_local_attachment_store ((CalAttachmentBar *) priv->attachment_bar,
e_cal_get_local_attachment_store (priv->client));
cal_attachment_bar_set_comp_uid (priv->attachment_bar, g_strdup (uid));
@@ -1717,3 +2099,4 @@ obj_removed_cb (ECal *client, GList *uids, gpointer data)
if (changed_component_dialog ((GtkWindow *) editor, priv->comp, TRUE, priv->changed))
close_dialog (editor);
}
+