/* * Copyright (C) 2002 Marco Pesenti Gritti * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "egg-toolbars-group.h" #include static void egg_toolbars_group_class_init (EggToolbarsGroupClass *klass); static void egg_toolbars_group_init (EggToolbarsGroup *t); static void egg_toolbars_group_finalize (GObject *object); enum { CHANGED, LAST_SIGNAL }; static guint egg_toolbars_group_signals[LAST_SIGNAL] = { 0 }; static GObjectClass *parent_class = NULL; struct EggToolbarsGroupPrivate { GNode *available_actions; GNode *toolbars; char *defaults; char *user; }; GType egg_toolbars_group_get_type (void) { static GType egg_toolbars_group_type = 0; if (egg_toolbars_group_type == 0) { static const GTypeInfo our_info = { sizeof (EggToolbarsGroupClass), NULL, /* base_init */ NULL, /* base_finalize */ (GClassInitFunc) egg_toolbars_group_class_init, NULL, NULL, /* class_data */ sizeof (EggToolbarsGroup), 0, /* n_preallocs */ (GInstanceInitFunc) egg_toolbars_group_init }; egg_toolbars_group_type = g_type_register_static (G_TYPE_OBJECT, "EggToolbarsGroup", &our_info, 0); } return egg_toolbars_group_type; } static xmlDocPtr egg_toolbars_group_to_xml (EggToolbarsGroup *t) { GNode *l1, *l2, *tl; xmlDocPtr doc; g_return_val_if_fail (IS_EGG_TOOLBARS_GROUP (t), NULL); tl = t->priv->toolbars; xmlIndentTreeOutput = TRUE; doc = xmlNewDoc ("1.0"); doc->children = xmlNewDocNode (doc, NULL, "toolbars", NULL); for (l1 = tl->children; l1 != NULL; l1 = l1->next) { xmlNodePtr tnode; tnode = xmlNewChild (doc->children, NULL, "toolbar", NULL); for (l2 = l1->children; l2 != NULL; l2 = l2->next) { xmlNodePtr node; EggToolbarsItem *item = l2->data; if (item->separator) { node = xmlNewChild (tnode, NULL, "separator", NULL); } else { node = xmlNewChild (tnode, NULL, "toolitem", NULL); xmlSetProp (node, "verb", item->action); } } } return doc; } static void toolbars_group_save (EggToolbarsGroup *t) { xmlDocPtr doc; g_return_if_fail (IS_EGG_TOOLBARS_GROUP (t)); doc = egg_toolbars_group_to_xml (t); xmlSaveFormatFile (t->priv->user, doc, 1); xmlFreeDoc (doc); } static EggToolbarsToolbar * toolbars_toolbar_new (void) { EggToolbarsToolbar *toolbar; static int id = 0; toolbar = g_new0 (EggToolbarsToolbar, 1); toolbar->id = g_strdup_printf ("Toolbar%d", id); id++; return toolbar; } static EggToolbarsItem * toolbars_item_new (const char *action, gboolean separator) { EggToolbarsItem *item; static int id = 0; g_return_val_if_fail (action != NULL, NULL); item = g_new0 (EggToolbarsItem, 1); item->action = g_strdup (action); item->separator = separator; item->id = g_strdup_printf ("TI%d", id); id++; return item; } static void free_toolbar_node (EggToolbarsToolbar *toolbar) { g_return_if_fail (toolbar != NULL); g_free (toolbar->id); g_free (toolbar); } static void free_item_node (EggToolbarsItem *item) { g_return_if_fail (item != NULL); g_free (item->action); g_free (item->id); g_free (item); } static void add_action (EggToolbarsGroup *t, GNode *parent, int pos, const char *name) { EggToolbarsItem *item; GNode *node; gboolean separator; separator = (strcmp (name, "separator") == 0); item = toolbars_item_new (name, separator); item->parent = parent->data; node = g_node_new (item); g_node_insert (parent, pos, node); } void egg_toolbars_group_add_item (EggToolbarsGroup *t, EggToolbarsToolbar *parent, int pos, const char *name) { GNode *parent_node; g_return_if_fail (IS_EGG_TOOLBARS_GROUP (t)); g_return_if_fail (parent != NULL); g_return_if_fail (name != NULL); parent_node = g_node_find (t->priv->toolbars, G_IN_ORDER, G_TRAVERSE_ALL, parent); add_action (t, parent_node, pos, name); toolbars_group_save (t); g_signal_emit (G_OBJECT (t), egg_toolbars_group_signals[CHANGED], 0); } static void parse_item_list (EggToolbarsGroup *t, xmlNodePtr child, GNode *parent) { while (child) { if (xmlStrEqual (child->name, "toolitem")) { xmlChar *verb; verb = xmlGetProp (child, "verb"); add_action (t, parent, -1, verb); xmlFree (verb); } else if (xmlStrEqual (child->name, "separator")) { add_action (t, parent, -1, "separator"); } child = child->next; } } static GNode * add_toolbar (EggToolbarsGroup *t) { EggToolbarsToolbar *toolbar; GNode *node; toolbar = toolbars_toolbar_new (); node = g_node_new (toolbar); g_node_append (t->priv->toolbars, node); return node; } EggToolbarsToolbar * egg_toolbars_group_add_toolbar (EggToolbarsGroup *t) { GNode *node; g_return_val_if_fail (IS_EGG_TOOLBARS_GROUP (t), NULL); node = add_toolbar (t); g_return_val_if_fail (node != NULL, NULL); toolbars_group_save (t); g_signal_emit (G_OBJECT (t), egg_toolbars_group_signals[CHANGED], 0); return node->data; } static void parse_toolbars (EggToolbarsGroup *t, xmlNodePtr child) { while (child) { if (xmlStrEqual (child->name, "toolbar")) { GNode *node; node = add_toolbar (t); parse_item_list (t, child->children, node); } child = child->next; } } static void load_defaults (EggToolbarsGroup *t) { xmlDocPtr doc; xmlNodePtr child; xmlNodePtr root; const char *xml_filepath; g_return_if_fail (IS_EGG_TOOLBARS_GROUP (t)); xml_filepath = t->priv->defaults; doc = xmlParseFile (xml_filepath); root = xmlDocGetRootElement (doc); child = root->children; while (child) { if (xmlStrEqual (child->name, "available")) { t->priv->available_actions = g_node_new (NULL); parse_item_list (t, child->children, t->priv->available_actions); } else if (xmlStrEqual (child->name, "default") && t->priv->toolbars == NULL) { t->priv->toolbars = g_node_new (NULL); parse_toolbars (t, child->children); } child = child->next; } xmlFreeDoc (doc); } static void load_toolbar (EggToolbarsGroup *t) { xmlDocPtr doc; xmlNodePtr root; const char *xml_filepath = t->priv->user; g_return_if_fail (IS_EGG_TOOLBARS_GROUP (t)); if (!g_file_test (xml_filepath, G_FILE_TEST_EXISTS)) return; doc = xmlParseFile (xml_filepath); root = xmlDocGetRootElement (doc); t->priv->toolbars = g_node_new (NULL); parse_toolbars (t, root->children); xmlFreeDoc (doc); } char * egg_toolbars_group_to_string (EggToolbarsGroup *t) { GString *s; GNode *l1, *l2, *tl; char *result; int k = 0; g_return_val_if_fail (IS_EGG_TOOLBARS_GROUP (t), NULL); tl = t->priv->toolbars; g_return_val_if_fail (tl != NULL, NULL); s = g_string_new (NULL); g_string_append (s, ""); for (l1 = tl->children; l1 != NULL; l1 = l1->next) { int i = 0; EggToolbarsToolbar *toolbar = l1->data; g_string_append_printf (s, "\n", toolbar->id); for (l2 = l1->children; l2 != NULL; l2 = l2->next) { EggToolbarsItem *item = l2->data; if (item->separator) { g_string_append_printf (s, "" "" "\n", i, k, item->id); } else { g_string_append_printf (s, "" "" "\n", i, k, item->id, item->action); } i++; } g_string_append (s, "\n"); k++; } g_string_append (s, ""); result = g_string_free (s, FALSE); return result; } static void egg_toolbars_group_class_init (EggToolbarsGroupClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); parent_class = g_type_class_peek_parent (klass); object_class->finalize = egg_toolbars_group_finalize; egg_toolbars_group_signals[CHANGED] = g_signal_new ("changed", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (EggToolbarsGroupClass, changed), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); } static void egg_toolbars_group_init (EggToolbarsGroup *t) { t->priv = g_new0 (EggToolbarsGroupPrivate, 1); t->priv->available_actions = NULL; t->priv->toolbars = NULL; t->priv->user = NULL; t->priv->defaults = NULL; } static void egg_toolbars_group_finalize (GObject *object) { EggToolbarsGroup *t = EGG_TOOLBARS_GROUP (object); g_return_if_fail (object != NULL); g_return_if_fail (IS_EGG_TOOLBARS_GROUP (object)); g_node_children_foreach (t->priv->available_actions, G_IN_ORDER, (GNodeForeachFunc) free_item_node, NULL); egg_toolbars_group_foreach_toolbar (t, (EggToolbarsGroupForeachToolbarFunc) free_toolbar_node, NULL); egg_toolbars_group_foreach_item (t, (EggToolbarsGroupForeachItemFunc) free_item_node, NULL); g_node_destroy (t->priv->available_actions); g_node_destroy (t->priv->toolbars); g_free (t->priv->user); g_free (t->priv->defaults); g_free (t->priv); G_OBJECT_CLASS (parent_class)->finalize (object); } EggToolbarsGroup * egg_toolbars_group_new (void) { EggToolbarsGroup *t; t = EGG_TOOLBARS_GROUP (g_object_new (EGG_TOOLBARS_GROUP_TYPE, NULL)); g_return_val_if_fail (t->priv != NULL, NULL); return t; } static void remove_action (EggToolbarsItem *item, gpointer *data) { char *action = data[0]; EggToolbarsGroup *group = EGG_TOOLBARS_GROUP (data[1]); if (strcmp (item->action, action) == 0) { egg_toolbars_group_remove_item (group, item); } } void egg_toolbars_group_remove_action (EggToolbarsGroup *group, const char *action) { gpointer data[2]; data[0] = (char *)action; data[1] = group; egg_toolbars_group_foreach_item (group, (EggToolbarsGroupForeachItemFunc) remove_action, data); g_signal_emit (G_OBJECT (group), egg_toolbars_group_signals[CHANGED], 0); } void egg_toolbars_group_remove_toolbar (EggToolbarsGroup *t, EggToolbarsToolbar *toolbar) { GNode *node; g_return_if_fail (IS_EGG_TOOLBARS_GROUP (t)); g_return_if_fail (toolbar != NULL); node = g_node_find (t->priv->toolbars, G_IN_ORDER, G_TRAVERSE_ALL, toolbar); g_return_if_fail (node != NULL); free_toolbar_node (node->data); g_node_destroy (node); toolbars_group_save (t); g_signal_emit (G_OBJECT (t), egg_toolbars_group_signals[CHANGED], 0); } void egg_toolbars_group_remove_item (EggToolbarsGroup *t, EggToolbarsItem *item) { GNode *node; GNode *toolbar; g_return_if_fail (IS_EGG_TOOLBARS_GROUP (t)); g_return_if_fail (item != NULL); node = g_node_find (t->priv->toolbars, G_IN_ORDER, G_TRAVERSE_ALL, item); g_return_if_fail (node != NULL); toolbar = node->parent; free_item_node (node->data); g_node_destroy (node); if (g_node_n_children (toolbar) == 0) { free_toolbar_node (toolbar->data); g_node_destroy (toolbar); } toolbars_group_save (t); g_signal_emit (G_OBJECT (t), egg_toolbars_group_signals[CHANGED], 0); } void egg_toolbars_group_set_source (EggToolbarsGroup *group, const char *defaults, const char *user) { g_return_if_fail (IS_EGG_TOOLBARS_GROUP (group)); g_return_if_fail (defaults != NULL); group->priv->defaults = g_strdup (defaults); group->priv->user = g_strdup (user); load_toolbar (group); load_defaults (group); } static gboolean is_item_in_toolbars (EggToolbarsGroup *group, const char *action) { GNode *l1, *l2; g_return_val_if_fail (IS_EGG_TOOLBARS_GROUP (group), FALSE); g_return_val_if_fail (action != NULL, FALSE); for (l1 = group->priv->toolbars->children; l1 != NULL; l1 = l1->next) { for (l2 = l1->children; l2 != NULL; l2 = l2->next) { EggToolbarsItem *item; item = (EggToolbarsItem *) l2->data; if (strcmp (action, item->action) == 0) return TRUE; } } return FALSE; } void egg_toolbars_group_foreach_available (EggToolbarsGroup *group, EggToolbarsGroupForeachItemFunc func, gpointer data) { GNode *l1; g_return_if_fail (IS_EGG_TOOLBARS_GROUP (group)); for (l1 = group->priv->available_actions->children; l1 != NULL; l1 = l1->next) { EggToolbarsItem *item; item = (EggToolbarsItem *) l1->data; if (!is_item_in_toolbars (group, item->action)) { func (item, data); } } } void egg_toolbars_group_foreach_toolbar (EggToolbarsGroup *group, EggToolbarsGroupForeachToolbarFunc func, gpointer data) { GNode *l1; g_return_if_fail (IS_EGG_TOOLBARS_GROUP (group)); for (l1 = group->priv->toolbars->children; l1 != NULL; l1 = l1->next) { func (l1->data, data); } } void egg_toolbars_group_foreach_item (EggToolbarsGroup *group, EggToolbarsGroupForeachItemFunc func, gpointer data) { GNode *l1, *l2; g_return_if_fail (IS_EGG_TOOLBARS_GROUP (group)); for (l1 = group->priv->toolbars->children; l1 != NULL; l1 = l1->next) { for (l2 = l1->children; l2 != NULL; l2 = l2->next) { func (l2->data, data); } } } char * egg_toolbars_group_get_path (EggToolbarsGroup *t, gpointer item) { GNode *node; char *path = NULL; EggToolbarsItem *titem; EggToolbarsToolbar *toolbar; g_return_val_if_fail (IS_EGG_TOOLBARS_GROUP (t), NULL); node = g_node_find (t->priv->toolbars, G_IN_ORDER, G_TRAVERSE_ALL, item); g_return_val_if_fail (node != NULL, NULL); titem = (EggToolbarsItem *) node->data; toolbar = (EggToolbarsToolbar *) node->data; switch (g_node_depth (node)) { case 2: path = g_strdup_printf ("/%s", toolbar->id); break; case 3: path = g_strdup_printf ("/Toolbar%d/PlaceHolder%d-%d/%s", g_node_child_position (node->parent->parent, node->parent), g_node_child_position (node->parent, node), g_node_child_position (node->parent->parent, node->parent), titem->id); break; default: g_assert_not_reached (); } return path; }