/*
* 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:
* Not Zed
* Jeffrey Stedfast
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
*/
#ifdef HAVE_CONFIG_H
#include
#endif
#include
#include
#include
#include
#include "filter-option.h"
#include "filter-part.h"
#include
#define d(x)
static int option_eq (FilterElement *fe, FilterElement *cm);
static void xml_create (FilterElement *fe, xmlNodePtr node);
static xmlNodePtr xml_encode (FilterElement *fe);
static int xml_decode (FilterElement *fe, xmlNodePtr node);
static FilterElement *clone (FilterElement *fe);
static GtkWidget *get_widget (FilterElement *fe);
static void build_code (FilterElement *fe, GString *out, struct _FilterPart *ff);
static void format_sexp (FilterElement *, GString *);
static GSList *get_dynamic_options (FilterOption *fo);
static void filter_option_class_init (FilterOptionClass *klass);
static void filter_option_init (FilterOption *fo);
static void filter_option_finalise (GObject *obj);
static FilterElementClass *parent_class;
GType
filter_option_get_type (void)
{
static GType type = 0;
if (!type) {
static const GTypeInfo info = {
sizeof (FilterOptionClass),
NULL, /* base_class_init */
NULL, /* base_class_finalize */
(GClassInitFunc) filter_option_class_init,
NULL, /* class_finalize */
NULL, /* class_data */
sizeof (FilterOption),
0, /* n_preallocs */
(GInstanceInitFunc) filter_option_init,
};
type = g_type_register_static (FILTER_TYPE_ELEMENT, "FilterOption", &info, 0);
}
return type;
}
static void
filter_option_class_init (FilterOptionClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
FilterElementClass *fe_class = FILTER_ELEMENT_CLASS (klass);
parent_class = g_type_class_ref (FILTER_TYPE_ELEMENT);
object_class->finalize = filter_option_finalise;
/* override methods */
fe_class->eq = option_eq;
fe_class->xml_create = xml_create;
fe_class->xml_encode = xml_encode;
fe_class->xml_decode = xml_decode;
fe_class->clone = clone;
fe_class->get_widget = get_widget;
fe_class->build_code = build_code;
fe_class->format_sexp = format_sexp;
}
static void
filter_option_init (FilterOption *fo)
{
fo->type = "option";
fo->dynamic_func = NULL;
}
static void
free_option (struct _filter_option *o, void *data)
{
g_free (o->title);
g_free (o->value);
g_free (o->code);
g_free (o);
}
static void
filter_option_finalise (GObject *obj)
{
FilterOption *fo = (FilterOption *) obj;
g_list_foreach (fo->options, (GFunc)free_option, NULL);
g_list_free (fo->options);
g_free (fo->dynamic_func);
G_OBJECT_CLASS (parent_class)->finalize (obj);
}
/**
* filter_option_new:
*
* Create a new FilterOption object.
*
* Return value: A new #FilterOption object.
**/
FilterOption *
filter_option_new (void)
{
return (FilterOption *) g_object_new (FILTER_TYPE_OPTION, NULL, NULL);
}
static struct _filter_option *
find_option (FilterOption *fo, const char *name)
{
GList *l = fo->options;
struct _filter_option *op;
while (l) {
op = l->data;
if (!strcmp (name, op->value)) {
return op;
}
l = g_list_next (l);
}
return NULL;
}
void
filter_option_set_current (FilterOption *option, const char *name)
{
g_return_if_fail (IS_FILTER_OPTION(option));
option->current = find_option (option, name);
}
/* used by implementers to add additional options */
struct _filter_option *
filter_option_add(FilterOption *fo, const char *value, const char *title, const char *code, gboolean is_dynamic)
{
struct _filter_option *op;
g_return_val_if_fail (IS_FILTER_OPTION(fo), NULL);
g_return_val_if_fail(find_option(fo, value) == NULL, NULL);
op = g_malloc(sizeof(*op));
op->title = g_strdup(title);
op->value = g_strdup(value);
op->code = g_strdup(code);
op->is_dynamic = is_dynamic;
fo->options = g_list_append(fo->options, op);
if (fo->current == NULL)
fo->current = op;
return op;
}
const char *
filter_option_get_current (FilterOption *option)
{
g_return_val_if_fail (IS_FILTER_OPTION (option), NULL);
if (!option->current)
return NULL;
return option->current->value;
}
void
filter_option_remove_all (FilterOption *fo)
{
g_return_if_fail (IS_FILTER_OPTION (fo));
g_list_foreach (fo->options, (GFunc)free_option, NULL);
g_list_free (fo->options);
fo->options = NULL;
fo->current = NULL;
}
static int
option_eq(FilterElement *fe, FilterElement *cm)
{
FilterOption *fo = (FilterOption *)fe, *co = (FilterOption *)cm;
return FILTER_ELEMENT_CLASS (parent_class)->eq (fe, cm)
&& ((fo->current && co->current && strcmp(fo->current->value, co->current->value) == 0)
|| (fo->current == NULL && co->current == NULL));
}
static void
xml_create (FilterElement *fe, xmlNodePtr node)
{
FilterOption *fo = (FilterOption *)fe;
xmlNodePtr n, work;
/* parent implementation */
FILTER_ELEMENT_CLASS (parent_class)->xml_create (fe, node);
n = node->children;
while (n) {
if (!strcmp ((char *)n->name, "option")) {
char *tmp, *value, *title = NULL, *code = NULL;
value = (char *)xmlGetProp (n, (const unsigned char *)"value");
work = n->children;
while (work) {
if (!strcmp ((char *)work->name, "title") || !strcmp ((char *)work->name, "_title")) {
if (!title) {
if (!(tmp = (char *)xmlNodeGetContent (work)))
tmp = (char *)xmlStrdup ((const unsigned char *)"");
title = g_strdup (tmp);
xmlFree (tmp);
}
} else if (!strcmp ((char *)work->name, "code")) {
if (!code) {
if (!(tmp = (char*)xmlNodeGetContent (work)))
tmp = (char *)xmlStrdup ((const unsigned char *)"");
code = g_strdup (tmp);
xmlFree (tmp);
}
}
work = work->next;
}
filter_option_add (fo, value, title, code, FALSE);
xmlFree (value);
g_free (title);
g_free (code);
} else if (g_str_equal ((char *)n->name, "dynamic")) {
if (fo->dynamic_func) {
g_warning ("Only one 'dynamic' node is acceptable in the optionlist '%s'", fe->name);
} else {
/* Expecting only one in the option list,
The 'cb' should be of this prototype:
GSList *cb (void);
returning GSList of struct _filter_option, all newly allocated, because it'll
be freed with g_free and g_slist_free. 'is_dynamic' member is ignored here.
*/
xmlChar *fn;
fn = xmlGetProp (n, (const unsigned char *)"func");
if (fn && *fn) {
GSList *items, *i;
struct _filter_option *op;
fo->dynamic_func = g_strdup ((const char *)fn);
/* get options now, to have them available when reading saved rules */
items = get_dynamic_options (fo);
for (i = items; i; i = i->next) {
op = i->data;
if (op) {
filter_option_add (fo, op->value, op->title, op->code, TRUE);
free_option (op, NULL);
}
}
g_slist_free (items);
} else {
g_warning ("Missing 'func' attribute within '%s' node in optionlist '%s'", n->name, fe->name);
}
xmlFree (fn);
}
} else if (n->type == XML_ELEMENT_NODE) {
g_warning ("Unknown xml node within optionlist: %s\n", n->name);
}
n = n->next;
}
}
static xmlNodePtr
xml_encode (FilterElement *fe)
{
xmlNodePtr value;
FilterOption *fo = (FilterOption *)fe;
d(printf ("Encoding option as xml\n"));
value = xmlNewNode (NULL, (const unsigned char *)"value");
xmlSetProp (value, (const unsigned char *)"name", (unsigned char *)fe->name);
xmlSetProp (value, (const unsigned char *)"type", (unsigned char *)fo->type);
if (fo->current)
xmlSetProp (value, (const unsigned char *)"value", (unsigned char *)fo->current->value);
return value;
}
static int
xml_decode (FilterElement *fe, xmlNodePtr node)
{
FilterOption *fo = (FilterOption *)fe;
char *value;
d(printf ("Decoding option from xml\n"));
xmlFree (fe->name);
fe->name = (char *)xmlGetProp (node, (const unsigned char *)"name");
value = (char *)xmlGetProp (node, (const unsigned char *)"value");
if (value) {
fo->current = find_option (fo, value);
xmlFree (value);
} else {
fo->current = NULL;
}
return 0;
}
static void
option_changed (GtkWidget *widget, FilterElement *fe)
{
FilterOption *fo = (FilterOption *)fe;
fo->current = g_object_get_data ((GObject *) widget, "option");
}
static GSList *
get_dynamic_options (FilterOption *fo)
{
GModule *module;
GSList *(*get_func)(void);
GSList *res = NULL;
if (!fo || !fo->dynamic_func)
return res;
module = g_module_open (NULL, G_MODULE_BIND_LAZY);
if (g_module_symbol (module, fo->dynamic_func, (gpointer) &get_func)) {
res = get_func ();
} else {
g_warning ("optionlist dynamic fill function '%s' not found", fo->dynamic_func);
}
g_module_close (module);
return res;
}
static GtkWidget *
get_widget (FilterElement *fe)
{
FilterOption *fo = (FilterOption *)fe;
GtkWidget *menu;
GtkWidget *omenu;
GtkWidget *item;
GtkWidget *first = NULL;
GList *l;
struct _filter_option *op;
int index = 0, current = 0;
if (fo->dynamic_func) {
/* it is dynamically filled, thus remove all dynamics and put there the fresh ones */
GSList *items, *i;
GList *old_ops;
struct _filter_option *old_cur;
old_ops = fo->options;
old_cur = fo->current;
l = old_ops;
/* start with an empty list */
fo->current = NULL;
fo->options = NULL;
for (l = fo->options; l; l = l->next) {
op = l->data;
if (op->is_dynamic) {
break;
} else {
filter_option_add (fo, op->value, op->title, op->code, FALSE);
}
}
items = get_dynamic_options (fo);
for (i = items; i; i = i->next) {
op = i->data;
if (op) {
filter_option_add (fo, op->value, op->title, op->code, TRUE);
free_option (op, NULL);
}
}
g_slist_free (items);
/* maybe some static left after those dynamic, add them too */
for (; l; l = l->next) {
op = l->data;
if (!op->is_dynamic)
filter_option_add (fo, op->value, op->title, op->code, FALSE);
}
if (old_cur)
filter_option_set_current (fo, old_cur->value);
/* free old list */
g_list_foreach (old_ops, (GFunc)free_option, NULL);
g_list_free (old_ops);
}
menu = gtk_menu_new ();
l = fo->options;
while (l) {
op = l->data;
item = gtk_menu_item_new_with_label (_(op->title));
g_object_set_data ((GObject *) item, "option", op);
g_signal_connect (item, "activate", G_CALLBACK (option_changed), fe);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
gtk_widget_show (item);
if (op == fo->current) {
current = index;
first = item;
} else if (!first) {
first = item;
}
l = g_list_next (l);
index++;
}
omenu = gtk_option_menu_new ();
gtk_option_menu_set_menu (GTK_OPTION_MENU (omenu), menu);
if (first)
g_signal_emit_by_name (first, "activate", fe);
gtk_option_menu_set_history (GTK_OPTION_MENU (omenu), current);
return omenu;
}
static void
build_code (FilterElement *fe, GString *out, struct _FilterPart *ff)
{
FilterOption *fo = (FilterOption *)fe;
d(printf ("building option code %p, current = %p\n", fo, fo->current));
if (fo->current && fo->current->code)
filter_part_expand_code (ff, fo->current->code, out);
}
static void
format_sexp (FilterElement *fe, GString *out)
{
FilterOption *fo = (FilterOption *)fe;
if (fo->current)
e_sexp_encode_string (out, fo->current->value);
}
static FilterElement *
clone (FilterElement *fe)
{
FilterOption *fo = (FilterOption *)fe, *new;
GList *l;
struct _filter_option *op, *newop;
d(printf ("cloning option\n"));
new = FILTER_OPTION (FILTER_ELEMENT_CLASS (parent_class)->clone (fe));
l = fo->options;
while (l) {
op = l->data;
newop = filter_option_add (new, op->value, op->title, op->code, op->is_dynamic);
if (fo->current == op)
new->current = newop;
l = l->next;
}
new->dynamic_func = g_strdup (fo->dynamic_func);
d(printf ("cloning option code %p, current = %p\n", new, new->current));
return (FilterElement *) new;
}