/*
* 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
*
*
* Authors:
* Michael Zucchi
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
*/
#ifdef HAVE_CONFIG_H
#include
#endif
#include
#include
#include "em-format-hook.h"
#include
/* class name -> class map for EMFormat and subclasses */
static GHashTable *emfh_types;
/* ********************************************************************** */
/* Mail formatter handler plugin */
/*
*
*
*
*
*
*/
#define emfh ((EMFormatHook *)eph)
#define d(x)
static const EPluginHookTargetKey emfh_flag_map[] = {
{ "inline", EM_FORMAT_HANDLER_INLINE },
{ "inline_disposition", EM_FORMAT_HANDLER_INLINE_DISPOSITION },
{ NULL }
};
G_DEFINE_TYPE (EMFormatHook, em_format_hook, E_TYPE_PLUGIN_HOOK)
static void
emfh_parse_part (EMFormat *emf,
CamelMimePart *part,
GString *part_id,
EMFormatParserInfo *info,
GCancellable *cancellable)
{
struct _EMFormatHookItem *item = (EMFormatHookItem *) info->handler;
if (item->hook->hook.plugin->enabled) {
EMFormatHookTarget target = {
emf, part, part_id, info
};
e_plugin_invoke (item->hook->hook.plugin, item->format, &target);
} else if (info->handler->old) {
info->handler->old->parse_func (
emf, part, part_id, info, cancellable);
}
}
static void
emfh_free_item (struct _EMFormatHookItem *item)
{
/* FIXME: remove from formatter class */
g_free (item->handler.mime_type);
g_free (item->format);
g_free (item);
}
static void
emfh_free_group (struct _EMFormatHookGroup *group)
{
g_slist_foreach (group->items, (GFunc) emfh_free_item, NULL);
g_slist_free (group->items);
g_free (group->id);
g_free (group);
}
static struct _EMFormatHookItem *
emfh_construct_item (EPluginHook *eph,
EMFormatHookGroup *group,
xmlNodePtr root)
{
struct _EMFormatHookItem *item;
d(printf(" loading group item\n"));
item = g_malloc0 (sizeof (*item));
item->handler.mime_type = e_plugin_xml_prop(root, "mime_type");
item->handler.flags = e_plugin_hook_mask(root, emfh_flag_map, "flags");
item->format = e_plugin_xml_prop(root, "format");
item->handler.parse_func = emfh_parse_part;
item->hook = emfh;
if (item->handler.mime_type == NULL || item->format == NULL)
goto error;
d(printf(" type='%s' format='%s'\n", item->handler.mime_type, item->format));
return item;
error:
d(printf("error!\n"));
emfh_free_item (item);
return NULL;
}
static struct _EMFormatHookGroup *
emfh_construct_group (EPluginHook *eph,
xmlNodePtr root)
{
struct _EMFormatHookGroup *group;
xmlNodePtr node;
d(printf(" loading group\n"));
group = g_malloc0 (sizeof (*group));
group->id = e_plugin_xml_prop(root, "id");
if (group->id == NULL)
goto error;
node = root->children;
while (node) {
if (0 == strcmp((gchar *)node->name, "item")) {
struct _EMFormatHookItem *item;
item = emfh_construct_item (eph, group, node);
if (item)
group->items = g_slist_append (group->items, item);
}
node = node->next;
}
return group;
error:
emfh_free_group (group);
return NULL;
}
static gint
emfh_construct (EPluginHook *eph,
EPlugin *ep,
xmlNodePtr root)
{
xmlNodePtr node;
d(printf("loading format hook\n"));
if (((EPluginHookClass *) em_format_hook_parent_class)->
construct (eph, ep, root) == -1)
return -1;
node = root->children;
while (node) {
if (strcmp((gchar *)node->name, "group") == 0) {
struct _EMFormatHookGroup *group;
group = emfh_construct_group (eph, node);
if (group) {
EMFormatClass *class;
if (emfh_types
&& (class = g_hash_table_lookup (emfh_types, group->id))) {
GSList *l = group->items;
for (; l; l = g_slist_next (l)) {
EMFormatHookItem *item = l->data;
/* TODO: only add handlers if enabled? */
/* Well, disabling is handled by the callback,
* if we leave as is, then we can enable the
* plugin after startup and it will start
* working automagically */
em_format_class_add_handler (class, &item->handler);
}
}
/* We don't actually need to keep this
* around once its set on the class. */
emfh->groups = g_slist_append (emfh->groups, group);
}
}
node = node->next;
}
eph->plugin = ep;
/* Load the plugin as it does a few thing in the formatter thread. */
((EPluginClass *) G_OBJECT_GET_CLASS (ep))->enable (ep, 1);
return 0;
}
static void
emfh_enable (EPluginHook *eph,
gint state)
{
GSList *g, *l;
EMFormatClass *class;
g = emfh->groups;
if (emfh_types == NULL)
return;
for (; g; g = g_slist_next (g)) {
struct _EMFormatHookGroup *group = g->data;
class = g_hash_table_lookup (emfh_types, group->id);
for (l = group->items; l; l = g_slist_next (l)) {
EMFormatHookItem *item = l->data;
if (state)
em_format_class_add_handler (class, &item->handler);
else
em_format_class_remove_handler (class, &item->handler);
}
}
}
static void
format_hook_finalize (GObject *object)
{
EPluginHook *eph = (EPluginHook *) object;
g_slist_foreach (emfh->groups, (GFunc) emfh_free_group, NULL);
g_slist_free (emfh->groups);
/* Chain up to parent's finalize() method. */
G_OBJECT_CLASS (em_format_hook_parent_class)->finalize (object);
}
static void
em_format_hook_class_init (EMFormatHookClass *class)
{
GObjectClass *object_class;
EPluginHookClass *hook_class;
object_class = G_OBJECT_CLASS (class);
object_class->finalize = format_hook_finalize;
hook_class = E_PLUGIN_HOOK_CLASS (class);
hook_class->construct = emfh_construct;
hook_class->enable = emfh_enable;
hook_class->id = "org.gnome.evolution.mail.format:1.0";
}
static void
em_format_hook_init (EMFormatHook *hook)
{
}
void
em_format_hook_register_type (GType type)
{
EMFormatClass *class;
if (emfh_types == NULL)
emfh_types = g_hash_table_new (g_str_hash, g_str_equal);
d(printf("registering formatter type '%s'\n", g_type_name(type)));
class = g_type_class_ref (type);
g_hash_table_insert (emfh_types, (gpointer) g_type_name (type), class);
}