851 lines
18 KiB
C
851 lines
18 KiB
C
|
|
#include <sys/types.h>
|
|
#include <dirent.h>
|
|
#include <string.h>
|
|
|
|
#include <glib/gi18n.h>
|
|
|
|
#include "e-plugin.h"
|
|
|
|
/* plugin debug */
|
|
#define pd(x) x
|
|
/* plugin hook debug */
|
|
#define phd(x) x
|
|
|
|
/*
|
|
<camel-plugin
|
|
class="com.ximian.camel.plugin.provider:1.0"
|
|
id="com.ximian.camel.provider.imap:1.0"
|
|
type="shlib"
|
|
location="/opt/gnome2/lib/camel/1.0/libcamelimap.so"
|
|
factory="camel_imap_provider_new">
|
|
<name>imap</name>
|
|
<description>IMAP4 and IMAP4v1 mail store</description>
|
|
<class-data class="com.ximian.camel.plugin.provider:1.0"
|
|
protocol="imap"
|
|
domain="mail"
|
|
flags="remote,source,storage,ssl"/>
|
|
</camel-plugin>
|
|
|
|
<camel-plugin
|
|
class="com.ximian.camel.plugin.sasl:1.0"
|
|
id="com.ximian.camel.sasl.plain:1.0"
|
|
type="shlib"
|
|
location="/opt/gnome2/lib/camel/1.0/libcamelsasl.so"
|
|
factory="camel_sasl_plain_new">
|
|
<name>PLAIN</name>
|
|
<description>SASL PLAIN authentication mechanism</description>
|
|
</camel-plugin>
|
|
*/
|
|
|
|
static GObjectClass *ep_parent_class;
|
|
static GHashTable *ep_types;
|
|
static GSList *ep_path;
|
|
|
|
static int
|
|
ep_construct(EPlugin *ep, xmlNodePtr root)
|
|
{
|
|
xmlNodePtr node;
|
|
int res = -1;
|
|
|
|
ep->domain = e_plugin_xml_prop(root, "domain");
|
|
ep->name = e_plugin_xml_prop_domain(root, "name", ep->domain);
|
|
|
|
printf("creating plugin '%s'\n", ep->name);
|
|
|
|
node = root->children;
|
|
while (node) {
|
|
if (strcmp(node->name, "hook") == 0) {
|
|
struct _EPluginHook *hook;
|
|
|
|
hook = e_plugin_hook_new(ep, node);
|
|
if (hook)
|
|
ep->hooks = g_slist_prepend(ep->hooks, hook);
|
|
else {
|
|
char *tmp = xmlGetProp(node, "class");
|
|
|
|
g_warning("Plugin '%s' failed to load hook '%s'", ep->name, tmp?tmp:"unknown");
|
|
if (tmp)
|
|
xmlFree(tmp);
|
|
}
|
|
} else if (strcmp(node->name, "description") == 0) {
|
|
ep->description = e_plugin_xml_content_domain(node, ep->domain);
|
|
}
|
|
node = node->next;
|
|
}
|
|
res = 0;
|
|
|
|
return res;
|
|
}
|
|
|
|
static void
|
|
ep_finalise(GObject *o)
|
|
{
|
|
EPlugin *ep = (EPlugin *)o;
|
|
|
|
g_free(ep->description);
|
|
g_free(ep->name);
|
|
g_free(ep->domain);
|
|
|
|
g_slist_foreach(ep->hooks, (GFunc)g_object_unref, NULL);
|
|
g_slist_free(ep->hooks);
|
|
|
|
((GObjectClass *)ep_parent_class)->finalize(o);
|
|
}
|
|
|
|
static void
|
|
ep_init(GObject *o)
|
|
{
|
|
EPlugin *ep = (EPlugin *)o;
|
|
|
|
ep->enabled = TRUE;
|
|
}
|
|
|
|
static void
|
|
ep_class_init(EPluginClass *klass)
|
|
{
|
|
((GObjectClass *)klass)->finalize = ep_finalise;
|
|
klass->construct = ep_construct;
|
|
}
|
|
|
|
/**
|
|
* e_plugin_get_type:
|
|
*
|
|
* Standard GObject type function. This is only an abstract class, so
|
|
* you can only use this to subclass EPlugin.
|
|
*
|
|
* Return value: The type.
|
|
**/
|
|
GType
|
|
e_plugin_get_type(void)
|
|
{
|
|
static GType type = 0;
|
|
|
|
if (!type) {
|
|
char *path, *col, *p;
|
|
|
|
static const GTypeInfo info = {
|
|
sizeof(EPluginClass), NULL, NULL, (GClassInitFunc)ep_class_init, NULL, NULL,
|
|
sizeof(EPlugin), 0, (GInstanceInitFunc)ep_init,
|
|
};
|
|
|
|
ep_parent_class = g_type_class_ref(G_TYPE_OBJECT);
|
|
type = g_type_register_static(G_TYPE_OBJECT, "EPlugin", &info, 0);
|
|
|
|
/* Add paths in the environment variable or default global and user specific paths */
|
|
path = g_strdup(getenv("EVOLUTION_PLUGIN_PATH"));
|
|
if (path == NULL) {
|
|
/* Add the global path */
|
|
e_plugin_add_load_path(EVOLUTION_PLUGINDIR);
|
|
|
|
path = g_build_filename(g_get_home_dir(), ".eplug", NULL);
|
|
}
|
|
|
|
p = path;
|
|
while ((col = strchr(p, ':'))) {
|
|
*col++ = 0;
|
|
e_plugin_add_load_path(p);
|
|
p = col;
|
|
}
|
|
e_plugin_add_load_path(p);
|
|
g_free(path);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
static int
|
|
ep_load(const char *filename)
|
|
{
|
|
xmlDocPtr doc;
|
|
xmlNodePtr root;
|
|
int res = -1;
|
|
EPlugin *ep;
|
|
|
|
doc = xmlParseFile(filename);
|
|
if (doc == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
root = xmlDocGetRootElement(doc);
|
|
if (strcmp(root->name, "e-plugin-list") != 0)
|
|
goto fail;
|
|
|
|
root = root->children;
|
|
while (root) {
|
|
if (strcmp(root->name, "e-plugin") == 0) {
|
|
char *prop;
|
|
EPluginClass *klass;
|
|
|
|
prop = xmlGetProp(root, "type");
|
|
if (prop == NULL)
|
|
goto fail;
|
|
|
|
klass = g_hash_table_lookup(ep_types, prop);
|
|
if (klass == NULL) {
|
|
g_warning("can't find plugin type '%s'\n", prop);
|
|
xmlFree(prop);
|
|
goto fail;
|
|
}
|
|
|
|
xmlFree(prop);
|
|
|
|
ep = g_object_new(G_TYPE_FROM_CLASS(klass), NULL);
|
|
if (e_plugin_construct(ep, root) == -1) {
|
|
g_object_unref(ep);
|
|
} else {
|
|
/* ... */
|
|
}
|
|
}
|
|
root = root->next;
|
|
}
|
|
|
|
res = 0;
|
|
fail:
|
|
xmlFreeDoc(doc);
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* e_plugin_add_load_path:
|
|
* @path: The path to add to search for plugins.
|
|
*
|
|
* Add a path to be searched when e_plugin_load_plugins() is called.
|
|
* By default ~/.eplug is used as the search path unless overriden by
|
|
* the environmental variable %EVOLUTION_PLUGIN_PATH.
|
|
*
|
|
* %EVOLUTION_PLUGIN_PATH is a : separated list of paths to search for
|
|
* plugin definitions in order.
|
|
*
|
|
* Plugin definitions are XML files ending in the extension ".eplug".
|
|
**/
|
|
void
|
|
e_plugin_add_load_path(const char *path)
|
|
{
|
|
ep_path = g_slist_append(ep_path, g_strdup(path));
|
|
}
|
|
|
|
/**
|
|
* e_plugin_load_plugins:
|
|
*
|
|
* Scan the search path, looking for plugin definitions, and load them
|
|
* into memory.
|
|
*
|
|
* Return value: Returns -1 if an error occured.
|
|
**/
|
|
int
|
|
e_plugin_load_plugins(void)
|
|
{
|
|
GSList *l;
|
|
|
|
if (ep_types == NULL) {
|
|
g_warning("no plugin types defined");
|
|
return 0;
|
|
}
|
|
|
|
|
|
for (l = ep_path;l;l = g_slist_next(l)) {
|
|
DIR *dir;
|
|
struct dirent *d;
|
|
char *path = l->data;
|
|
|
|
printf("scanning plugin dir '%s'\n", path);
|
|
|
|
dir = opendir(path);
|
|
if (dir == NULL) {
|
|
g_warning("Could not find plugin path: %s", path);
|
|
continue;
|
|
}
|
|
|
|
while ( (d = readdir(dir)) ) {
|
|
if (strlen(d->d_name) > 6
|
|
&& !strcmp(d->d_name + strlen(d->d_name) - 6, ".eplug")) {
|
|
char * name = g_build_filename(path, d->d_name, NULL);
|
|
|
|
ep_load(name);
|
|
g_free(name);
|
|
}
|
|
}
|
|
|
|
closedir(dir);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* e_plugin_register_type:
|
|
* @type: The GObject type of the plugin loader.
|
|
*
|
|
* Register a new plugin type with the plugin system. Each type must
|
|
* subclass EPlugin and must override the type member of the
|
|
* EPluginClass with a unique name.
|
|
**/
|
|
void
|
|
e_plugin_register_type(GType type)
|
|
{
|
|
EPluginClass *klass;
|
|
|
|
if (ep_types == NULL)
|
|
ep_types = g_hash_table_new(g_str_hash, g_str_equal);
|
|
|
|
klass = g_type_class_ref(type);
|
|
|
|
pd(printf("register plugin type '%s'\n", klass->type));
|
|
|
|
g_hash_table_insert(ep_types, (void *)klass->type, klass);
|
|
}
|
|
|
|
/**
|
|
* e_plugin_construct:
|
|
* @ep: An EPlugin derived object.
|
|
* @root: The XML root node of the sub-tree containing the plugin
|
|
* definition.
|
|
*
|
|
* Helper to invoke the construct virtual method.
|
|
*
|
|
* Return value: The return from the construct virtual method.
|
|
**/
|
|
int
|
|
e_plugin_construct(EPlugin *ep, xmlNodePtr root)
|
|
{
|
|
return ((EPluginClass *)G_OBJECT_GET_CLASS(ep))->construct(ep, root);
|
|
}
|
|
|
|
/**
|
|
* e_plugin_invoke:
|
|
* @ep:
|
|
* @name: The name of the function to invoke. The format of this name
|
|
* will depend on the EPlugin type and its language conventions.
|
|
* @data: The argument to the function. Its actual type depends on
|
|
* the hook on which the function resides. It is up to the called
|
|
* function to get this right.
|
|
*
|
|
* Helper to invoke the invoke virtual method.
|
|
*
|
|
* Return value: The return of the plugin invocation.
|
|
**/
|
|
void *
|
|
e_plugin_invoke(EPlugin *ep, const char *name, void *data)
|
|
{
|
|
if (!ep->enabled)
|
|
g_warning("Invoking method on disabled plugin");
|
|
|
|
return ((EPluginClass *)G_OBJECT_GET_CLASS(ep))->invoke(ep, name, data);
|
|
}
|
|
|
|
/**
|
|
* e_plugin_enable:
|
|
* @ep:
|
|
* @state:
|
|
*
|
|
* Set the enable state of a plugin.
|
|
*
|
|
* THIS IS NOT FULLY IMPLEMENTED YET
|
|
**/
|
|
void
|
|
e_plugin_enable(EPlugin *ep, int state)
|
|
{
|
|
GSList *l;
|
|
|
|
if ((ep->enabled == 0) == (state == 0))
|
|
return;
|
|
|
|
ep->enabled = state;
|
|
for (l=ep->hooks;l;l = g_slist_next(l)) {
|
|
EPluginHook *eph = l->data;
|
|
|
|
e_plugin_hook_enable(eph, state);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* e_plugin_xml_prop:
|
|
* @node: An XML node.
|
|
* @id: The name of the property to retrieve.
|
|
*
|
|
* A static helper function to look up a property on an XML node, and
|
|
* ensure it is allocated in GLib system memory. If GLib isn't using
|
|
* the system malloc then it must copy the property value.
|
|
*
|
|
* Return value: The property, allocated in GLib memory, or NULL if no
|
|
* such property exists.
|
|
**/
|
|
char *
|
|
e_plugin_xml_prop(xmlNodePtr node, const char *id)
|
|
{
|
|
char *p = xmlGetProp(node, id);
|
|
|
|
if (g_mem_is_system_malloc()) {
|
|
return p;
|
|
} else {
|
|
char * out = g_strdup(p);
|
|
|
|
if (p)
|
|
xmlFree(p);
|
|
return out;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* e_plugin_xml_prop_domain:
|
|
* @node: An XML node.
|
|
* @id: The name of the property to retrieve.
|
|
* @domain: The translation domain for this string.
|
|
*
|
|
* A static helper function to look up a property on an XML node, and
|
|
* translate it based on @domain.
|
|
*
|
|
* Return value: The property, allocated in GLib memory, or NULL if no
|
|
* such property exists.
|
|
**/
|
|
char *
|
|
e_plugin_xml_prop_domain(xmlNodePtr node, const char *id, const char *domain)
|
|
{
|
|
char *p, *out;
|
|
|
|
p = xmlGetProp(node, id);
|
|
if (p == NULL)
|
|
return NULL;
|
|
|
|
out = g_strdup(dgettext(domain, p));
|
|
xmlFree(p);
|
|
|
|
return out;
|
|
}
|
|
|
|
/**
|
|
* e_plugin_xml_int:
|
|
* @node: An XML node.
|
|
* @id: The name of the property to retrieve.
|
|
* @def: A default value if the property doesn't exist. Can be used
|
|
* to determine if the property isn't set.
|
|
*
|
|
* A static helper function to look up a property on an XML node as an
|
|
* integer. If the property doesn't exist, then @def is returned as a
|
|
* default value instead.
|
|
*
|
|
* Return value: The value if set, or @def if not.
|
|
**/
|
|
int
|
|
e_plugin_xml_int(xmlNodePtr node, const char *id, int def)
|
|
{
|
|
char *p = xmlGetProp(node, id);
|
|
|
|
if (p)
|
|
return atoi(p);
|
|
else
|
|
return def;
|
|
}
|
|
|
|
/**
|
|
* e_plugin_xml_content:
|
|
* @node:
|
|
*
|
|
* A static helper function to retrieve the entire textual content of
|
|
* an XML node, and ensure it is allocated in GLib system memory. If
|
|
* GLib isn't using the system malloc them it must copy the content.
|
|
*
|
|
* Return value: The node content, allocated in GLib memory.
|
|
**/
|
|
char *
|
|
e_plugin_xml_content(xmlNodePtr node)
|
|
{
|
|
char *p = xmlNodeGetContent(node);
|
|
|
|
if (g_mem_is_system_malloc()) {
|
|
return p;
|
|
} else {
|
|
char * out = g_strdup(p);
|
|
|
|
if (p)
|
|
xmlFree(p);
|
|
return out;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* e_plugin_xml_content_domain:
|
|
* @node:
|
|
* @domain:
|
|
*
|
|
* A static helper function to retrieve the entire textual content of
|
|
* an XML node, and ensure it is allocated in GLib system memory. If
|
|
* GLib isn't using the system malloc them it must copy the content.
|
|
*
|
|
* Return value: The node content, allocated in GLib memory.
|
|
**/
|
|
char *
|
|
e_plugin_xml_content_domain(xmlNodePtr node, const char *domain)
|
|
{
|
|
char *p, *out;
|
|
|
|
p = xmlNodeGetContent(node);
|
|
if (p == NULL)
|
|
return NULL;
|
|
|
|
out = g_strdup(dgettext(domain, p));
|
|
xmlFree(p);
|
|
|
|
return out;
|
|
}
|
|
|
|
/* ********************************************************************** */
|
|
static void *epl_parent_class;
|
|
|
|
#define epl ((EPluginLib *)ep)
|
|
|
|
/* TODO:
|
|
We need some way to manage lifecycle.
|
|
We need some way to manage state.
|
|
|
|
Maybe just the g module init method will do, or we could add
|
|
another which returns context.
|
|
|
|
There is also the question of per-instance context, e.g. for config
|
|
pages.
|
|
*/
|
|
|
|
static void *
|
|
epl_invoke(EPlugin *ep, const char *name, void *data)
|
|
{
|
|
void *(*cb)(EPlugin *ep, void *data);
|
|
|
|
if (epl->module == NULL
|
|
&& (epl->module = g_module_open(epl->location, 0)) == NULL) {
|
|
g_warning("can't load plugin '%s'", g_module_error());
|
|
return NULL;
|
|
}
|
|
|
|
if (!g_module_symbol(epl->module, name, (void *)&cb))
|
|
return NULL;
|
|
|
|
return cb(ep, data);
|
|
}
|
|
|
|
static int
|
|
epl_construct(EPlugin *ep, xmlNodePtr root)
|
|
{
|
|
if (((EPluginClass *)epl_parent_class)->construct(ep, root) == -1)
|
|
return -1;
|
|
|
|
epl->location = e_plugin_xml_prop(root, "location");
|
|
|
|
if (epl->location == NULL)
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
epl_finalise(GObject *o)
|
|
{
|
|
EPlugin *ep = (EPlugin *)o;
|
|
|
|
g_free(epl->location);
|
|
|
|
if (epl->module)
|
|
g_module_close(epl->module);
|
|
|
|
((GObjectClass *)epl_parent_class)->finalize(o);
|
|
}
|
|
|
|
static void
|
|
epl_class_init(EPluginClass *klass)
|
|
{
|
|
((GObjectClass *)klass)->finalize = epl_finalise;
|
|
klass->construct = epl_construct;
|
|
klass->invoke = epl_invoke;
|
|
klass->type = "shlib";
|
|
}
|
|
|
|
/**
|
|
* e_plugin_lib_get_type:
|
|
*
|
|
* Standard GObject function to retrieve the EPluginLib type. Use to
|
|
* register the type with the plugin system if you want to use shared
|
|
* library plugins.
|
|
*
|
|
* Return value: The EPluginLib type.
|
|
**/
|
|
GType
|
|
e_plugin_lib_get_type(void)
|
|
{
|
|
static GType type = 0;
|
|
|
|
if (!type) {
|
|
static const GTypeInfo info = {
|
|
sizeof(EPluginLibClass), NULL, NULL, (GClassInitFunc) epl_class_init, NULL, NULL,
|
|
sizeof(EPluginLib), 0, (GInstanceInitFunc) NULL,
|
|
};
|
|
|
|
epl_parent_class = g_type_class_ref(e_plugin_get_type());
|
|
type = g_type_register_static(e_plugin_get_type(), "EPluginLib", &info, 0);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
/* ********************************************************************** */
|
|
static void *eph_parent_class;
|
|
static GHashTable *eph_types;
|
|
|
|
static int
|
|
eph_construct(EPluginHook *eph, EPlugin *ep, xmlNodePtr root)
|
|
{
|
|
eph->plugin = ep;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
eph_enable(EPluginHook *eph, int state)
|
|
{
|
|
/* NOOP */
|
|
}
|
|
|
|
static void
|
|
eph_finalise(GObject *o)
|
|
{
|
|
((GObjectClass *)eph_parent_class)->finalize((GObject *)o);
|
|
}
|
|
|
|
static void
|
|
eph_class_init(EPluginHookClass *klass)
|
|
{
|
|
((GObjectClass *)klass)->finalize = eph_finalise;
|
|
klass->construct = eph_construct;
|
|
klass->enable = eph_enable;
|
|
}
|
|
|
|
/**
|
|
* e_plugin_hook_get_type:
|
|
*
|
|
* Standard GObject function to retrieve the EPluginHook type. Since
|
|
* EPluginHook is an abstract class, this is only used to subclass it.
|
|
*
|
|
* Return value: The EPluginHook type.
|
|
**/
|
|
GType
|
|
e_plugin_hook_get_type(void)
|
|
{
|
|
static GType type = 0;
|
|
|
|
if (!type) {
|
|
static const GTypeInfo info = {
|
|
sizeof(EPluginHookClass), NULL, NULL, (GClassInitFunc) eph_class_init, NULL, NULL,
|
|
sizeof(EPluginHook), 0, (GInstanceInitFunc) NULL,
|
|
};
|
|
|
|
eph_parent_class = g_type_class_ref(G_TYPE_OBJECT);
|
|
type = g_type_register_static(G_TYPE_OBJECT, "EPluginHook", &info, 0);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
/**
|
|
* e_plugin_hook_new:
|
|
* @ep: The parent EPlugin this hook belongs to.
|
|
* @root: The XML node of the root of the hook definition.
|
|
*
|
|
* This is a static factory method to instantiate a new EPluginHook
|
|
* object to represent a plugin hook.
|
|
*
|
|
* Return value: The EPluginHook appropriate for the XML definition at
|
|
* @root. NULL is returned if a syntax error is encountered.
|
|
**/
|
|
EPluginHook *
|
|
e_plugin_hook_new(EPlugin *ep, xmlNodePtr root)
|
|
{
|
|
EPluginHookClass *type;
|
|
char *class;
|
|
EPluginHook *hook;
|
|
|
|
/* FIXME: Keep a list of all plugin hooks */
|
|
|
|
if (eph_types == NULL)
|
|
return NULL;
|
|
|
|
class = xmlGetProp(root, "class");
|
|
if (class == NULL)
|
|
return NULL;
|
|
|
|
type = g_hash_table_lookup(eph_types, class);
|
|
g_free(class);
|
|
if (type == NULL)
|
|
return NULL;
|
|
|
|
hook = g_object_new(G_OBJECT_CLASS_TYPE(type), NULL);
|
|
if (type->construct(hook, ep, root) == -1) {
|
|
g_object_unref(hook);
|
|
hook = NULL;
|
|
}
|
|
|
|
return hook;
|
|
}
|
|
|
|
/**
|
|
* e_plugin_hook_enable: Set hook enabled state.
|
|
* @eph:
|
|
* @state:
|
|
*
|
|
* Set the enabled state of the plugin hook. This is called by the
|
|
* plugin code.
|
|
*
|
|
* THIS IS NOT FULY IMEPLEMENTED YET
|
|
**/
|
|
void
|
|
e_plugin_hook_enable(EPluginHook *eph, int state)
|
|
{
|
|
((EPluginHookClass *)G_OBJECT_GET_CLASS(eph))->enable(eph, state);
|
|
}
|
|
|
|
/**
|
|
* e_plugin_hook_register_type:
|
|
* @type:
|
|
*
|
|
* Register a new plugin hook type with the plugin system. Each type
|
|
* must subclass EPluginHook and must override the id member of the
|
|
* EPluginHookClass with a unique identification string.
|
|
**/
|
|
void
|
|
e_plugin_hook_register_type(GType type)
|
|
{
|
|
EPluginHookClass *klass;
|
|
|
|
if (eph_types == NULL)
|
|
eph_types = g_hash_table_new(g_str_hash, g_str_equal);
|
|
|
|
klass = g_type_class_ref(type);
|
|
|
|
phd(printf("register plugin hook type '%s'\n", klass->id));
|
|
|
|
g_hash_table_insert(eph_types, (void *)klass->id, klass);
|
|
}
|
|
|
|
/**
|
|
* e_plugin_hook_mask:
|
|
* @root: An XML node.
|
|
* @map: A zero-fill terminated array of EPluginHookTargeKeys used to
|
|
* map a string with a bit value.
|
|
* @prop: The property name.
|
|
*
|
|
* This is a static helper function which looks up a property @prop on
|
|
* the XML node @root, and then uses the @map table to convert it into
|
|
* a bitmask. The property value is a comma separated list of
|
|
* enumeration strings which are indexed into the @map table.
|
|
*
|
|
* Return value: A bitmask representing the inclusive-or of all of the
|
|
* integer values of the corresponding string id's stored in the @map.
|
|
**/
|
|
guint32
|
|
e_plugin_hook_mask(xmlNodePtr root, const struct _EPluginHookTargetKey *map, const char *prop)
|
|
{
|
|
char *val, *p, *start, c;
|
|
guint32 mask = 0;
|
|
|
|
val = xmlGetProp(root, prop);
|
|
if (val == NULL)
|
|
return 0;
|
|
|
|
p = val;
|
|
do {
|
|
start = p;
|
|
while (*p && *p != ',')
|
|
p++;
|
|
c = *p;
|
|
*p = 0;
|
|
if (start != p) {
|
|
int i;
|
|
|
|
for (i=0;map[i].key;i++) {
|
|
if (!strcmp(map[i].key, start)) {
|
|
mask |= map[i].value;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
*p++ = c;
|
|
} while (c);
|
|
|
|
xmlFree(val);
|
|
|
|
return mask;
|
|
}
|
|
|
|
/**
|
|
* e_plugin_hook_id:
|
|
* @root:
|
|
* @map:
|
|
* @prop:
|
|
*
|
|
* This is a static helper function which looks up a property @prop on
|
|
* the XML node @root, and then uses the @map table to convert it into
|
|
* an integer.
|
|
*
|
|
* This is used as a helper wherever you need to represent an
|
|
* enumerated value in the XML.
|
|
*
|
|
* Return value: If the @prop value is in @map, then the corresponding
|
|
* integer value, if not, then ~0.
|
|
**/
|
|
guint32
|
|
e_plugin_hook_id(xmlNodePtr root, const struct _EPluginHookTargetKey *map, const char *prop)
|
|
{
|
|
char *val;
|
|
int i;
|
|
|
|
val = xmlGetProp(root, prop);
|
|
if (val == NULL)
|
|
return ~0;
|
|
|
|
for (i=0;map[i].key;i++) {
|
|
if (!strcmp(map[i].key, val)) {
|
|
xmlFree(val);
|
|
return map[i].value;
|
|
}
|
|
}
|
|
|
|
xmlFree(val);
|
|
|
|
return ~0;
|
|
}
|
|
|
|
#if 0
|
|
/*
|
|
e-mail-format-handler
|
|
mime_type
|
|
target
|
|
*/
|
|
struct _EMFormatPlugin {
|
|
EPlugin plugin;
|
|
|
|
char *target;
|
|
char *mime_type;
|
|
struct _EMFormatHandler *(*get_handler)(void);
|
|
};
|
|
|
|
struct _EMFormatPluginClass {
|
|
EPluginClass plugin_class;
|
|
};
|
|
|
|
#endif
|
|
|
|
#if 0
|
|
void em_setup_plugins(void);
|
|
|
|
void
|
|
em_setup_plugins(void)
|
|
{
|
|
GType *e_plugin_mono_get_type(void);
|
|
|
|
e_plugin_register_type(e_plugin_lib_get_type());
|
|
e_plugin_register_type(e_plugin_mono_get_type());
|
|
|
|
e_plugin_hook_register_type(em_popup_hook_get_type());
|
|
|
|
e_plugin_load_plugins(".");
|
|
}
|
|
#endif
|