EAttachment is now used from worker threads by EMailFormatterAttachment,
so add some thread-safe accessor functions to eliminate potential races.
Added thread-safe functions:
e_attachment_dup_disposition()
e_attachment_ref_file()
e_attachment_ref_file_info()
e_attachment_ref_icon()
e_attachment_ref_mime_part()
e_attachment_dup_description()
e_attachment_dup_thumbnail_path()
Renamed functions:
e_attachment_get_mime_type() -> e_attachment_dup_mime_type()
Removed non-thread-safe functions:
e_attachment_get_file()
e_attachment_get_file_info()
e_attachment_get_icon()
e_attachment_get_mime_part()
e_attachment_get_description()
e_attachment_get_thumbnail_path()
578 lines
15 KiB
C
578 lines
15 KiB
C
/*
|
|
* e-cal-attachment-handler.c
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) version 3.
|
|
*
|
|
* 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with the program; if not, see <http://www.gnu.org/licenses/>
|
|
*
|
|
*
|
|
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include "e-cal-attachment-handler.h"
|
|
|
|
#include <glib/gi18n.h>
|
|
#include <libical/ical.h>
|
|
#include <camel/camel.h>
|
|
#include <libecal/libecal.h>
|
|
|
|
#include <shell/e-shell.h>
|
|
|
|
#define E_CAL_ATTACHMENT_HANDLER_GET_PRIVATE(obj) \
|
|
(G_TYPE_INSTANCE_GET_PRIVATE \
|
|
((obj), E_TYPE_CAL_ATTACHMENT_HANDLER, ECalAttachmentHandlerPrivate))
|
|
|
|
typedef struct _ImportContext ImportContext;
|
|
|
|
struct _ECalAttachmentHandlerPrivate {
|
|
gint placeholder;
|
|
};
|
|
|
|
struct _ImportContext {
|
|
ECalClient *client;
|
|
icalcomponent *component;
|
|
ECalClientSourceType source_type;
|
|
};
|
|
|
|
static gpointer parent_class;
|
|
static GType cal_attachment_handler_type;
|
|
|
|
static const gchar *ui =
|
|
"<ui>"
|
|
" <popup name='context'>"
|
|
" <placeholder name='custom-actions'>"
|
|
" <menuitem action='import-to-calendar'/>"
|
|
" <menuitem action='import-to-tasks'/>"
|
|
" </placeholder>"
|
|
" </popup>"
|
|
"</ui>";
|
|
|
|
static icalcomponent *
|
|
attachment_handler_get_component (EAttachment *attachment)
|
|
{
|
|
CamelDataWrapper *wrapper;
|
|
CamelMimePart *mime_part;
|
|
CamelStream *stream;
|
|
GByteArray *buffer;
|
|
icalcomponent *component;
|
|
const gchar *key = "__icalcomponent__";
|
|
|
|
component = g_object_get_data (G_OBJECT (attachment), key);
|
|
if (component != NULL)
|
|
return component;
|
|
|
|
if (e_attachment_get_loading (attachment) ||
|
|
e_attachment_get_saving (attachment))
|
|
return NULL;
|
|
|
|
mime_part = e_attachment_ref_mime_part (attachment);
|
|
if (mime_part == NULL)
|
|
return NULL;
|
|
|
|
buffer = g_byte_array_new ();
|
|
stream = camel_stream_mem_new ();
|
|
camel_stream_mem_set_byte_array (CAMEL_STREAM_MEM (stream), buffer);
|
|
wrapper = camel_medium_get_content (CAMEL_MEDIUM (mime_part));
|
|
camel_data_wrapper_decode_to_stream_sync (wrapper, stream, NULL, NULL);
|
|
g_object_unref (stream);
|
|
|
|
g_object_unref (mime_part);
|
|
|
|
if (buffer->len > 0) {
|
|
const gchar *str;
|
|
|
|
/* ensure string being null-terminated */
|
|
g_byte_array_append (buffer, (const guint8 *) "", 1);
|
|
|
|
str = (const gchar *) buffer->data;
|
|
while (*str && g_ascii_isspace (*str))
|
|
str++;
|
|
|
|
if (g_ascii_strncasecmp (str, "BEGIN:", 6) == 0)
|
|
component = e_cal_util_parse_ics_string (str);
|
|
}
|
|
|
|
g_byte_array_free (buffer, TRUE);
|
|
|
|
if (component == NULL)
|
|
return NULL;
|
|
|
|
g_object_set_data_full (
|
|
G_OBJECT (attachment), key, component,
|
|
(GDestroyNotify) icalcomponent_free);
|
|
|
|
return component;
|
|
}
|
|
|
|
static gboolean
|
|
attachment_handler_update_objects (ECalClient *client,
|
|
icalcomponent *component)
|
|
{
|
|
icalcomponent_kind kind;
|
|
icalcomponent *vcalendar;
|
|
gboolean success;
|
|
GError *error = NULL;
|
|
|
|
kind = icalcomponent_isa (component);
|
|
|
|
switch (kind) {
|
|
case ICAL_VTODO_COMPONENT:
|
|
case ICAL_VEVENT_COMPONENT:
|
|
vcalendar = e_cal_util_new_top_level ();
|
|
if (icalcomponent_get_method (component) == ICAL_METHOD_CANCEL)
|
|
icalcomponent_set_method (vcalendar, ICAL_METHOD_CANCEL);
|
|
else
|
|
icalcomponent_set_method (vcalendar, ICAL_METHOD_PUBLISH);
|
|
icalcomponent_add_component (
|
|
vcalendar, icalcomponent_new_clone (component));
|
|
break;
|
|
|
|
case ICAL_VCALENDAR_COMPONENT:
|
|
vcalendar = icalcomponent_new_clone (component);
|
|
if (!icalcomponent_get_first_property (vcalendar, ICAL_METHOD_PROPERTY))
|
|
icalcomponent_set_method (vcalendar, ICAL_METHOD_PUBLISH);
|
|
break;
|
|
|
|
default:
|
|
return FALSE;
|
|
}
|
|
|
|
success = e_cal_client_receive_objects_sync (
|
|
client, vcalendar, NULL, &error);
|
|
|
|
if (error != NULL) {
|
|
g_warning (
|
|
"%s: Failed to receive objects: %s",
|
|
G_STRFUNC, error->message);
|
|
g_error_free (error);
|
|
}
|
|
|
|
icalcomponent_free (vcalendar);
|
|
|
|
return success;
|
|
}
|
|
|
|
static void
|
|
attachment_handler_import_event (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
EAttachment *attachment = user_data;
|
|
EClient *client;
|
|
icalcomponent *component;
|
|
icalcomponent *subcomponent;
|
|
icalcompiter iter;
|
|
GError *error = NULL;
|
|
|
|
client = e_cal_client_connect_finish (result, &error);
|
|
|
|
/* Sanity check. */
|
|
g_return_if_fail (
|
|
((client != NULL) && (error == NULL)) ||
|
|
((client == NULL) && (error != NULL)));
|
|
|
|
if (error != NULL) {
|
|
g_warning ("%s: %s", G_STRFUNC, error->message);
|
|
g_object_unref (attachment);
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
|
|
component = attachment_handler_get_component (attachment);
|
|
g_return_if_fail (component != NULL);
|
|
|
|
iter = icalcomponent_begin_component (component, ICAL_ANY_COMPONENT);
|
|
|
|
while ((subcomponent = icalcompiter_deref (&iter)) != NULL) {
|
|
icalcomponent_kind kind;
|
|
|
|
kind = icalcomponent_isa (subcomponent);
|
|
icalcompiter_next (&iter);
|
|
|
|
if (kind == ICAL_VEVENT_COMPONENT)
|
|
continue;
|
|
|
|
if (kind == ICAL_VTIMEZONE_COMPONENT)
|
|
continue;
|
|
|
|
icalcomponent_remove_component (component, subcomponent);
|
|
icalcomponent_free (subcomponent);
|
|
}
|
|
|
|
/* XXX Do something with the return value. */
|
|
attachment_handler_update_objects (E_CAL_CLIENT (client), component);
|
|
|
|
g_object_unref (attachment);
|
|
g_object_unref (client);
|
|
}
|
|
|
|
static void
|
|
attachment_handler_import_todo (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
EAttachment *attachment = user_data;
|
|
EClient *client;
|
|
icalcomponent *component;
|
|
icalcomponent *subcomponent;
|
|
icalcompiter iter;
|
|
GError *error = NULL;
|
|
|
|
client = e_cal_client_connect_finish (result, &error);
|
|
|
|
/* Sanity check. */
|
|
g_return_if_fail (
|
|
((client != NULL) && (error == NULL)) ||
|
|
((client == NULL) && (error != NULL)));
|
|
|
|
if (error != NULL) {
|
|
g_warning ("%s: %s", G_STRFUNC, error->message);
|
|
g_object_unref (attachment);
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
|
|
component = attachment_handler_get_component (attachment);
|
|
g_return_if_fail (component != NULL);
|
|
|
|
iter = icalcomponent_begin_component (component, ICAL_ANY_COMPONENT);
|
|
|
|
while ((subcomponent = icalcompiter_deref (&iter)) != NULL) {
|
|
icalcomponent_kind kind;
|
|
|
|
kind = icalcomponent_isa (subcomponent);
|
|
icalcompiter_next (&iter);
|
|
|
|
if (kind == ICAL_VTODO_COMPONENT)
|
|
continue;
|
|
|
|
if (kind == ICAL_VTIMEZONE_COMPONENT)
|
|
continue;
|
|
|
|
icalcomponent_remove_component (component, subcomponent);
|
|
icalcomponent_free (subcomponent);
|
|
}
|
|
|
|
/* XXX Do something with the return value. */
|
|
attachment_handler_update_objects (E_CAL_CLIENT (client), component);
|
|
|
|
g_object_unref (attachment);
|
|
g_object_unref (client);
|
|
}
|
|
|
|
static void
|
|
attachment_handler_row_activated_cb (GtkDialog *dialog)
|
|
{
|
|
gtk_dialog_response (dialog, GTK_RESPONSE_OK);
|
|
}
|
|
|
|
static void
|
|
attachment_handler_run_dialog (GtkWindow *parent,
|
|
EAttachment *attachment,
|
|
ECalClientSourceType source_type,
|
|
const gchar *title)
|
|
{
|
|
EShell *shell;
|
|
GtkWidget *dialog;
|
|
GtkWidget *container;
|
|
GtkWidget *widget;
|
|
ESourceRegistry *registry;
|
|
ESourceSelector *selector;
|
|
ESource *source;
|
|
const gchar *extension_name;
|
|
icalcomponent *component;
|
|
|
|
switch (source_type) {
|
|
case E_CAL_CLIENT_SOURCE_TYPE_EVENTS:
|
|
extension_name = E_SOURCE_EXTENSION_CALENDAR;
|
|
break;
|
|
case E_CAL_CLIENT_SOURCE_TYPE_TASKS:
|
|
extension_name = E_SOURCE_EXTENSION_TASK_LIST;
|
|
break;
|
|
case E_CAL_CLIENT_SOURCE_TYPE_MEMOS:
|
|
extension_name = E_SOURCE_EXTENSION_MEMO_LIST;
|
|
break;
|
|
default:
|
|
g_return_if_reached ();
|
|
}
|
|
|
|
component = attachment_handler_get_component (attachment);
|
|
g_return_if_fail (component != NULL);
|
|
|
|
dialog = gtk_dialog_new_with_buttons (
|
|
title, parent, GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, NULL);
|
|
|
|
widget = gtk_button_new_with_mnemonic (_("I_mport"));
|
|
gtk_button_set_image (
|
|
GTK_BUTTON (widget), gtk_image_new_from_icon_name (
|
|
"stock_mail-import", GTK_ICON_SIZE_MENU));
|
|
gtk_dialog_add_action_widget (
|
|
GTK_DIALOG (dialog), widget, GTK_RESPONSE_OK);
|
|
gtk_widget_show (widget);
|
|
|
|
gtk_window_set_default_size (GTK_WINDOW (dialog), 300, 400);
|
|
|
|
container = gtk_dialog_get_content_area (GTK_DIALOG (dialog));
|
|
|
|
widget = gtk_scrolled_window_new (NULL, NULL);
|
|
gtk_scrolled_window_set_policy (
|
|
GTK_SCROLLED_WINDOW (widget),
|
|
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
|
gtk_scrolled_window_set_shadow_type (
|
|
GTK_SCROLLED_WINDOW (widget), GTK_SHADOW_IN);
|
|
gtk_box_pack_start (GTK_BOX (container), widget, TRUE, TRUE, 0);
|
|
gtk_widget_show (widget);
|
|
|
|
container = widget;
|
|
|
|
shell = e_shell_get_default ();
|
|
registry = e_shell_get_registry (shell);
|
|
widget = e_source_selector_new (registry, extension_name);
|
|
selector = E_SOURCE_SELECTOR (widget);
|
|
e_source_selector_set_show_toggles (selector, FALSE);
|
|
gtk_container_add (GTK_CONTAINER (container), widget);
|
|
gtk_widget_show (widget);
|
|
|
|
g_signal_connect_swapped (
|
|
widget, "row-activated",
|
|
G_CALLBACK (attachment_handler_row_activated_cb), dialog);
|
|
|
|
if (gtk_dialog_run (GTK_DIALOG (dialog)) != GTK_RESPONSE_OK)
|
|
goto exit;
|
|
|
|
source = e_source_selector_ref_primary_selection (selector);
|
|
if (source == NULL)
|
|
goto exit;
|
|
|
|
switch (source_type) {
|
|
case E_CAL_CLIENT_SOURCE_TYPE_EVENTS:
|
|
e_cal_client_connect (
|
|
source, source_type, NULL,
|
|
attachment_handler_import_event,
|
|
g_object_ref (attachment));
|
|
break;
|
|
case E_CAL_CLIENT_SOURCE_TYPE_TASKS:
|
|
e_cal_client_connect (
|
|
source, source_type, NULL,
|
|
attachment_handler_import_todo,
|
|
g_object_ref (attachment));
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
g_object_unref (source);
|
|
|
|
exit:
|
|
gtk_widget_destroy (dialog);
|
|
}
|
|
|
|
static void
|
|
attachment_handler_import_to_calendar (GtkAction *action,
|
|
EAttachmentHandler *handler)
|
|
{
|
|
EAttachment *attachment;
|
|
EAttachmentView *view;
|
|
GList *selected;
|
|
gpointer parent;
|
|
|
|
view = e_attachment_handler_get_view (handler);
|
|
|
|
parent = gtk_widget_get_toplevel (GTK_WIDGET (view));
|
|
parent = gtk_widget_is_toplevel (parent) ? parent : NULL;
|
|
|
|
selected = e_attachment_view_get_selected_attachments (view);
|
|
g_return_if_fail (g_list_length (selected) == 1);
|
|
attachment = E_ATTACHMENT (selected->data);
|
|
|
|
attachment_handler_run_dialog (
|
|
parent, attachment,
|
|
E_CAL_CLIENT_SOURCE_TYPE_EVENTS,
|
|
_("Select a Calendar"));
|
|
|
|
g_object_unref (attachment);
|
|
g_list_free (selected);
|
|
}
|
|
|
|
static void
|
|
attachment_handler_import_to_tasks (GtkAction *action,
|
|
EAttachmentHandler *handler)
|
|
{
|
|
EAttachment *attachment;
|
|
EAttachmentView *view;
|
|
GList *selected;
|
|
gpointer parent;
|
|
|
|
view = e_attachment_handler_get_view (handler);
|
|
|
|
parent = gtk_widget_get_toplevel (GTK_WIDGET (view));
|
|
parent = gtk_widget_is_toplevel (parent) ? parent : NULL;
|
|
|
|
selected = e_attachment_view_get_selected_attachments (view);
|
|
g_return_if_fail (g_list_length (selected) == 1);
|
|
attachment = E_ATTACHMENT (selected->data);
|
|
|
|
attachment_handler_run_dialog (
|
|
parent, attachment,
|
|
E_CAL_CLIENT_SOURCE_TYPE_TASKS,
|
|
_("Select a Task List"));
|
|
|
|
g_object_unref (attachment);
|
|
g_list_free (selected);
|
|
}
|
|
|
|
static GtkActionEntry standard_entries[] = {
|
|
|
|
{ "import-to-calendar",
|
|
"stock_mail-import",
|
|
N_("I_mport to Calendar"),
|
|
NULL,
|
|
NULL, /* XXX Add a tooltip! */
|
|
G_CALLBACK (attachment_handler_import_to_calendar) },
|
|
|
|
{ "import-to-tasks",
|
|
"stock_mail-import",
|
|
N_("I_mport to Tasks"),
|
|
NULL,
|
|
NULL, /* XXX Add a tooltip! */
|
|
G_CALLBACK (attachment_handler_import_to_tasks) }
|
|
};
|
|
|
|
static void
|
|
cal_attachment_handler_update_actions (EAttachmentView *view)
|
|
{
|
|
EAttachment *attachment;
|
|
GtkAction *action;
|
|
GList *selected;
|
|
icalcomponent *component;
|
|
icalcomponent *subcomponent;
|
|
icalcomponent_kind kind;
|
|
gboolean is_vevent = FALSE;
|
|
gboolean is_vtodo = FALSE;
|
|
|
|
selected = e_attachment_view_get_selected_attachments (view);
|
|
|
|
if (g_list_length (selected) != 1)
|
|
goto exit;
|
|
|
|
attachment = E_ATTACHMENT (selected->data);
|
|
component = attachment_handler_get_component (attachment);
|
|
|
|
if (component == NULL)
|
|
goto exit;
|
|
|
|
subcomponent = icalcomponent_get_inner (component);
|
|
|
|
if (subcomponent == NULL)
|
|
goto exit;
|
|
|
|
kind = icalcomponent_isa (subcomponent);
|
|
is_vevent = (kind == ICAL_VEVENT_COMPONENT);
|
|
is_vtodo = (kind == ICAL_VTODO_COMPONENT);
|
|
|
|
exit:
|
|
action = e_attachment_view_get_action (view, "import-to-calendar");
|
|
gtk_action_set_visible (action, is_vevent);
|
|
|
|
action = e_attachment_view_get_action (view, "import-to-tasks");
|
|
gtk_action_set_visible (action, is_vtodo);
|
|
|
|
g_list_foreach (selected, (GFunc) g_object_unref, NULL);
|
|
g_list_free (selected);
|
|
}
|
|
|
|
static void
|
|
cal_attachment_handler_constructed (GObject *object)
|
|
{
|
|
EAttachmentHandler *handler;
|
|
EAttachmentView *view;
|
|
GtkActionGroup *action_group;
|
|
GtkUIManager *ui_manager;
|
|
GError *error = NULL;
|
|
|
|
handler = E_ATTACHMENT_HANDLER (object);
|
|
|
|
/* Chain up to parent's constructed() method. */
|
|
G_OBJECT_CLASS (parent_class)->constructed (object);
|
|
|
|
view = e_attachment_handler_get_view (handler);
|
|
|
|
action_group = e_attachment_view_add_action_group (view, "calendar");
|
|
gtk_action_group_add_actions (
|
|
action_group, standard_entries,
|
|
G_N_ELEMENTS (standard_entries), handler);
|
|
|
|
ui_manager = e_attachment_view_get_ui_manager (view);
|
|
gtk_ui_manager_add_ui_from_string (ui_manager, ui, -1, &error);
|
|
|
|
if (error != NULL) {
|
|
g_warning ("%s", error->message);
|
|
g_error_free (error);
|
|
}
|
|
|
|
g_signal_connect (
|
|
view, "update_actions",
|
|
G_CALLBACK (cal_attachment_handler_update_actions),
|
|
NULL);
|
|
}
|
|
|
|
static void
|
|
cal_attachment_handler_class_init (ECalAttachmentHandlerClass *class)
|
|
{
|
|
GObjectClass *object_class;
|
|
|
|
parent_class = g_type_class_peek_parent (class);
|
|
g_type_class_add_private (class, sizeof (ECalAttachmentHandlerPrivate));
|
|
|
|
object_class = G_OBJECT_CLASS (class);
|
|
object_class->constructed = cal_attachment_handler_constructed;
|
|
}
|
|
|
|
static void
|
|
cal_attachment_handler_init (ECalAttachmentHandler *handler)
|
|
{
|
|
handler->priv = E_CAL_ATTACHMENT_HANDLER_GET_PRIVATE (handler);
|
|
}
|
|
|
|
GType
|
|
e_cal_attachment_handler_get_type (void)
|
|
{
|
|
return cal_attachment_handler_type;
|
|
}
|
|
|
|
void
|
|
e_cal_attachment_handler_register_type (GTypeModule *type_module)
|
|
{
|
|
static const GTypeInfo type_info = {
|
|
sizeof (ECalAttachmentHandlerClass),
|
|
(GBaseInitFunc) NULL,
|
|
(GBaseFinalizeFunc) NULL,
|
|
(GClassInitFunc) cal_attachment_handler_class_init,
|
|
(GClassFinalizeFunc) NULL,
|
|
NULL, /* class_data */
|
|
sizeof (ECalAttachmentHandler),
|
|
0, /* n_preallocs */
|
|
(GInstanceInitFunc) cal_attachment_handler_init,
|
|
NULL /* value_table */
|
|
};
|
|
|
|
cal_attachment_handler_type = g_type_module_register_type (
|
|
type_module, E_TYPE_ATTACHMENT_HANDLER,
|
|
"ECalAttachmentHandler", &type_info, 0);
|
|
}
|