/* * mooui/moouixml.c * * Copyright (C) 2004-2006 by Yevgen Muntyan * * 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 of the License, or * (at your option) any later version. * * See COPYING file that comes with this distribution. */ #include "mooutils/mooaction-private.h" #include "mooutils/moouixml.h" #include "mooutils/moocompat.h" #include "mooutils/moomarshals.h" #include "mooutils/moomenutoolbutton.h" #include "mooutils/mooutils-misc.h" #include "mooutils/mooi18n.h" #include #include #define REPORT_UNKNOWN_ACTIONS 0 /* XXX weak ref actions */ typedef MooUINodeType NodeType; typedef MooUINode Node; typedef MooUIWidgetNode Widget; typedef MooUIItemNode Item; typedef MooUISeparatorNode Separator; typedef MooUIPlaceholderNode Placeholder; #define CONTAINER MOO_UI_NODE_CONTAINER #define WIDGET MOO_UI_NODE_WIDGET #define TOOLBAR MOO_UI_NODE_TOOLBAR #define ITEM MOO_UI_NODE_ITEM #define SEPARATOR MOO_UI_NODE_SEPARATOR #define PLACEHOLDER MOO_UI_NODE_PLACEHOLDER static const char *NODE_TYPE_NAME[] = { NULL, "MOO_UI_NODE_CONTAINER", "MOO_UI_NODE_WIDGET", "MOO_UI_NODE_ITEM", "MOO_UI_NODE_SEPARATOR", "MOO_UI_NODE_PLACEHOLDER" }; struct _MooUIXMLPrivate { Node *ui; GSList *toplevels; /* Toplevel* */ guint last_merge_id; GSList *merged_ui; /* Merge* */ }; typedef struct { Node *node; GtkWidget *widget; GHashTable *children; /* Node* -> GtkWidget* */ MooActionCollection *actions; GtkAccelGroup *accel_group; gboolean in_creation; } Toplevel; typedef struct { guint id; GSList *nodes; } Merge; typedef enum { UPDATE_ADD_NODE, UPDATE_REMOVE_NODE, UPDATE_CHANGE_NODE } UpdateType; #define TOPLEVEL_QUARK (toplevel_quark ()) #define NODE_QUARK (node_quark ()) #define SLIST_FOREACH(list,var) \ G_STMT_START { \ GSList *var; \ for (var = list; var != NULL; var = var->next) \ #define SLIST_FOREACH_END \ } G_STMT_END /* walking nodes stops when func returns TRUE */ typedef gboolean (*NodeForeachFunc) (Node *node, gpointer data); static void moo_ui_xml_finalize (GObject *object); static void xml_add_markup (MooUIXML *xml, MooMarkupNode *mnode); static void update_widgets (MooUIXML *xml, UpdateType type, Node *node); static Node *parse_markup (MooMarkupNode *mnode); static Node *parse_object (MooMarkupNode *mnode); static Node *parse_widget (MooMarkupNode *mnode); static Node *parse_placeholder (MooMarkupNode *mnode); static Node *parse_item (MooMarkupNode *mnode); static Node *parse_separator (MooMarkupNode *mnode); static gboolean node_check (Node *node); static gboolean placeholder_check (Node *node); static gboolean item_check (Node *node); static gboolean widget_check (Node *node); static gboolean container_check (Node *node); static Merge *lookup_merge (MooUIXML *xml, guint id); static Node *node_new (NodeType type, const char *name); static gboolean node_is_ancestor (Node *node, Node *ancestor); static gboolean node_is_empty (Node *node); static GSList *node_list_children (Node *ndoe); static GSList *node_list_all_children (Node *ndoe); static void node_free (Node *node); static void node_foreach (Node *node, NodeForeachFunc func, gpointer data); static void merge_add_node (Merge *merge, Node *node); static void merge_remove_node (Merge *merge, Node *node); static Toplevel *toplevel_new (Node *node, MooActionCollection *actions, GtkAccelGroup *accel_group); static void toplevel_free (Toplevel *toplevel); static GtkWidget *toplevel_get_widget (Toplevel *toplevel, Node *node); static GQuark toplevel_quark (void); static GQuark node_quark (void); static void xml_add_item_widget (MooUIXML *xml, GtkWidget *widget); static void xml_add_widget (MooUIXML *xml, GtkWidget *widget, Toplevel *toplevel, Node *node); static void xml_remove_widget (MooUIXML *xml, GtkWidget *widget); static void xml_delete_toplevel (MooUIXML *xml, Toplevel *toplevel); static void xml_connect_toplevel (MooUIXML *xml, Toplevel *toplevel); static gboolean create_menu_separator (MooUIXML *xml, Toplevel *toplevel, GtkMenuShell *menu, Node *node, int index); static void create_menu_item (MooUIXML *xml, Toplevel *toplevel, GtkMenuShell *menu, Node *node, int index); static gboolean create_menu_shell (MooUIXML *xml, Toplevel *toplevel, MooUIWidgetType type); static gboolean fill_menu_shell (MooUIXML *xml, Toplevel *toplevel, Node *menu_node, GtkMenuShell *menu); static void check_separators (Node *parent, Toplevel *toplevel); static void check_empty (Node *parent, GtkWidget *widget, Toplevel *toplevel); static gboolean create_tool_separator (MooUIXML *xml, Toplevel *toplevel, GtkToolbar *toolbar, Node *node, int index); static gboolean create_tool_item (MooUIXML *xml, Toplevel *toplevel, GtkToolbar *toolbar, Node *node, int index); static gboolean fill_toolbar (MooUIXML *xml, Toplevel *toplevel, Node *toolbar_node, GtkToolbar *toolbar); static gboolean create_toolbar (MooUIXML *xml, Toplevel *toplevel); static MooUINode *moo_ui_xml_find_node (MooUIXML *xml, const char *path_or_placeholder); /* MOO_TYPE_UI_XML */ G_DEFINE_TYPE (MooUIXML, moo_ui_xml, G_TYPE_OBJECT) static void moo_ui_xml_class_init (MooUIXMLClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); gobject_class->finalize = moo_ui_xml_finalize; } static Node* node_new (NodeType type, const char *name) { Node *node; gsize size = 0; switch (type) { case MOO_UI_NODE_CONTAINER: size = sizeof (MooUINode); break; case MOO_UI_NODE_WIDGET: size = sizeof (MooUIWidgetNode); break; case MOO_UI_NODE_ITEM: size = sizeof (MooUIItemNode); break; case MOO_UI_NODE_SEPARATOR: size = sizeof (MooUISeparatorNode); break; case MOO_UI_NODE_PLACEHOLDER: size = sizeof (MooUIPlaceholderNode); break; } node = g_malloc0 (size); node->type = type; node->name = name ? g_strdup (name) : g_strdup (""); return node; } static void moo_ui_xml_init (MooUIXML *xml) { xml->priv = g_new0 (MooUIXMLPrivate, 1); xml->priv->ui = node_new (CONTAINER, "ui"); } MooUIXML* moo_ui_xml_new (void) { return g_object_new (MOO_TYPE_UI_XML, NULL); } static Node* parse_object (MooMarkupNode *mnode) { Node *node; const char *name; MooMarkupNode *mchild; name = moo_markup_get_prop (mnode, "name"); if (!name || !name[0]) { g_warning ("%s: object name missing", G_STRLOC); return NULL; } node = node_new (CONTAINER, name); for (mchild = mnode->children; mchild != NULL; mchild = mchild->next) { if (MOO_MARKUP_IS_ELEMENT (mchild)) { Node *child = parse_markup (mchild); if (!child) { node_free (node); return NULL; } else { child->parent = node; node->children = g_slist_append (node->children, child); } } } return node; } static Node* parse_widget (MooMarkupNode *mnode) { Node *node; const char *name; MooMarkupNode *mchild; name = moo_markup_get_prop (mnode, "name"); if (!name || !name[0]) { g_warning ("%s: widget name missing", G_STRLOC); return NULL; } node = node_new (WIDGET, name); for (mchild = mnode->children; mchild != NULL; mchild = mchild->next) { if (MOO_MARKUP_IS_ELEMENT (mchild)) { Node *child = parse_markup (mchild); if (!child) { node_free (node); return NULL; } else { child->parent = node; node->children = g_slist_append (node->children, child); } } } return node; } static Node* parse_placeholder (MooMarkupNode *mnode) { Node *node; const char *name; MooMarkupNode *mchild; name = moo_markup_get_prop (mnode, "name"); if (!name || !name[0]) { g_warning ("%s: placeholder name missing", G_STRLOC); return NULL; } node = node_new (PLACEHOLDER, name); for (mchild = mnode->children; mchild != NULL; mchild = mchild->next) { if (MOO_MARKUP_IS_ELEMENT (mchild)) { Node *child = parse_markup (mchild); if (!child) { node_free (node); return NULL; } else { child->parent = node; node->children = g_slist_append (node->children, child); } } } return node; } static Item * item_new (const char *name, const char *action) { Item *item = (Item*) node_new (ITEM, name); item->action = g_strdup (action); return item; } static char * translate_string (const char *string, const char *translation_domain, gboolean translatable) { if (!string || !string[0] || !translatable) return g_strdup (string); if (translation_domain) { const char *translated = D_ (string, translation_domain); if (!strcmp (translated, string)) translated = _(string); return g_strdup (translated); } return g_strdup (_(string)); } static Item * item_new_from_node (MooMarkupNode *node, const char *translation_domain) { Item *item; const char *name; const char *action; const char *stock_id; const char *stock_label; const char *label; const char *tooltip; const char *icon_stock_id; gboolean translatable = FALSE; name = moo_markup_get_prop (node, "name"); if (!name) name = moo_markup_get_prop (node, "action"); label = moo_markup_get_prop (node, "_label"); if (label) translatable = TRUE; else label = moo_markup_get_prop (node, "label"); tooltip = moo_markup_get_prop (node, "_tooltip"); if (tooltip) translatable = TRUE; else tooltip = moo_markup_get_prop (node, "tooltip"); action = moo_markup_get_prop (node, "action"); stock_id = moo_markup_get_prop (node, "stock-id"); stock_label = moo_markup_get_prop (node, "stock-label"); icon_stock_id = moo_markup_get_prop (node, "icon-stock-id"), item = item_new (name, action); item->stock_id = g_strdup (stock_id); item->icon_stock_id = g_strdup (icon_stock_id); if (stock_label) { GtkStockItem stock_item; if (!gtk_stock_lookup (stock_label, &stock_item)) g_warning ("could not find stock item '%s'", stock_label); else if (!stock_item.label) g_warning ("stock item '%s' does not have a label", stock_label); else item->label = g_strdup (stock_item.label); } else { item->label = translate_string (label, translation_domain, translatable); } item->tooltip = translate_string (tooltip, translation_domain, translatable); return item; } static Node* parse_item (MooMarkupNode *mnode) { Item *item; MooMarkupNode *mchild; item = item_new_from_node (mnode, NULL); for (mchild = mnode->children; mchild != NULL; mchild = mchild->next) { if (MOO_MARKUP_IS_ELEMENT (mchild)) { Node *child = parse_markup (mchild); if (!child) { node_free ((Node*) item); return NULL; } else { child->parent = (Node*) item; item->children = g_slist_append (item->children, child); } } } return (Node*) item; } static Node* parse_separator (G_GNUC_UNUSED MooMarkupNode *mnode) { return node_new (SEPARATOR, NULL); } static Node* parse_markup (MooMarkupNode *mnode) { g_return_val_if_fail (MOO_MARKUP_IS_ELEMENT (mnode), NULL); g_return_val_if_fail (mnode->name != NULL, NULL); if (!strcmp (mnode->name, "object")) return parse_object (mnode); else if (!strcmp (mnode->name, "widget")) return parse_widget (mnode); else if (!strcmp (mnode->name, "item")) return parse_item (mnode); else if (!strcmp (mnode->name, "separator")) return parse_separator (mnode); else if (!strcmp (mnode->name, "placeholder")) return parse_placeholder (mnode); g_warning ("%s: unknown element '%s'", G_STRLOC, mnode->name); return NULL; } static void container_free (G_GNUC_UNUSED Node *node) { } static void widget_free (G_GNUC_UNUSED Node *node) { } static void separator_free (G_GNUC_UNUSED Node *node) { } static void placeholder_free (G_GNUC_UNUSED Node *node) { } static void item_free (Item *item) { g_free (item->action); g_free (item->stock_id); g_free (item->label); g_free (item->tooltip); g_free (item->icon_stock_id); } static void node_free (Node *node) { if (node) { g_slist_foreach (node->children, (GFunc) node_free, NULL); g_slist_free (node->children); node->children = NULL; switch (node->type) { case CONTAINER: container_free (node); break; case WIDGET: widget_free (node); break; case ITEM: item_free ((Item*) node); break; case SEPARATOR: separator_free (node); break; case PLACEHOLDER: placeholder_free (node); break; } g_free (node->name); g_free (node); } } void moo_ui_xml_add_ui_from_string (MooUIXML *xml, const char *buffer, gssize length) { MooMarkupDoc *doc; MooMarkupNode *ui_node, *child; GError *error = NULL; g_return_if_fail (MOO_IS_UI_XML (xml)); g_return_if_fail (buffer != NULL); doc = moo_markup_parse_memory (buffer, length, &error); if (!doc) { g_critical ("%s: could not parse markup", G_STRLOC); if (error) { g_critical ("%s: %s", G_STRLOC, error->message); g_error_free (error); } return; } ui_node = moo_markup_get_root_element (doc, "ui"); if (!ui_node) ui_node = MOO_MARKUP_NODE (doc); for (child = ui_node->children; child != NULL; child = child->next) if (MOO_MARKUP_IS_ELEMENT (child)) xml_add_markup (xml, child); moo_markup_doc_unref (doc); } static gboolean node_check (Node *node) { g_return_val_if_fail (node != NULL, FALSE); switch (node->type) { case CONTAINER: return container_check (node); case WIDGET: return widget_check (node); case ITEM: return item_check (node); case SEPARATOR: return TRUE; case PLACEHOLDER: return placeholder_check (node); } g_return_val_if_reached (FALSE); } static gboolean placeholder_check (Node *node) { g_return_val_if_fail (node != NULL, FALSE); g_return_val_if_fail (node->type == PLACEHOLDER, FALSE); g_return_val_if_fail (node->name && node->name[0], FALSE); SLIST_FOREACH (node->children, l) { Node *child = l->data; switch (child->type) { case SEPARATOR: case ITEM: case PLACEHOLDER: if (!node_check (child)) return FALSE; break; default: g_warning ("%s: invalid placeholder child type %s", G_STRLOC, NODE_TYPE_NAME[child->type]); return FALSE; } } SLIST_FOREACH_END; return TRUE; } static gboolean item_check (Node *node) { g_return_val_if_fail (node != NULL, FALSE); g_return_val_if_fail (node->type == ITEM, FALSE); g_return_val_if_fail (node->name != NULL, FALSE); SLIST_FOREACH (node->children, l) { Node *child = l->data; switch (child->type) { case SEPARATOR: case ITEM: case PLACEHOLDER: if (!node_check (child)) return FALSE; break; default: g_warning ("%s: invalid item child type %s", G_STRLOC, NODE_TYPE_NAME[child->type]); return FALSE; } } SLIST_FOREACH_END; return TRUE; } static gboolean widget_check (Node *node) { g_return_val_if_fail (node != NULL, FALSE); g_return_val_if_fail (node->type == WIDGET, FALSE); g_return_val_if_fail (node->name && node->name[0], FALSE); SLIST_FOREACH (node->children, l) { Node *child = l->data; switch (child->type) { case SEPARATOR: case ITEM: case PLACEHOLDER: if (!node_check (child)) return FALSE; break; default: g_warning ("%s: invalid widget child type %s", G_STRLOC, NODE_TYPE_NAME[child->type]); return FALSE; } } SLIST_FOREACH_END; return TRUE; } static gboolean container_check (Node *node) { g_return_val_if_fail (node != NULL, FALSE); g_return_val_if_fail (node->type == CONTAINER, FALSE); g_return_val_if_fail (node->name && node->name[0], FALSE); SLIST_FOREACH (node->children, l) { Node *child = l->data; switch (child->type) { case CONTAINER: case WIDGET: if (!node_check (child)) return FALSE; break; default: g_warning ("%s: invalid widget child type %s", G_STRLOC, NODE_TYPE_NAME[child->type]); return FALSE; } } SLIST_FOREACH_END; return TRUE; } static void xml_add_markup (MooUIXML *xml, MooMarkupNode *mnode) { Node *node = parse_markup (mnode); if (!node) return; switch (node->type) { case CONTAINER: if (!container_check (node)) { node_free (node); return; } break; case WIDGET: if (!widget_check (node)) { node_free (node); return; } break; case ITEM: case SEPARATOR: case PLACEHOLDER: g_warning ("%s: invalid toplevel type %s", G_STRLOC, NODE_TYPE_NAME[node->type]); node_free (node); return; } if (moo_ui_xml_get_node (xml, node->name)) { g_warning ("%s: implement me?", G_STRLOC); node_free (node); return; } node->parent = xml->priv->ui; xml->priv->ui->children = g_slist_append (xml->priv->ui->children, node); } guint moo_ui_xml_new_merge_id (MooUIXML *xml) { Merge *merge; g_return_val_if_fail (MOO_IS_UI_XML (xml), 0); xml->priv->last_merge_id++; merge = g_new0 (Merge, 1); merge->id = xml->priv->last_merge_id; merge->nodes = NULL; xml->priv->merged_ui = g_slist_prepend (xml->priv->merged_ui, merge); return merge->id; } static Merge* lookup_merge (MooUIXML *xml, guint merge_id) { GSList *l; for (l = xml->priv->merged_ui; l != NULL; l = l->next) { Merge *merge = l->data; if (merge->id == merge_id) return merge; } return NULL; } MooUINode* moo_ui_xml_add_item (MooUIXML *xml, guint merge_id, const char *parent_path, const char *name, const char *action, int position) { Merge *merge; MooUINode *parent; Item *item; g_return_val_if_fail (MOO_IS_UI_XML (xml), NULL); g_return_val_if_fail (parent_path != NULL, NULL); if (!name || !name[0]) name = action; merge = lookup_merge (xml, merge_id); g_return_val_if_fail (merge != NULL, NULL); parent = moo_ui_xml_find_node (xml, parent_path); g_return_val_if_fail (parent != NULL, NULL); switch (parent->type) { case MOO_UI_NODE_WIDGET: case MOO_UI_NODE_ITEM: case MOO_UI_NODE_PLACEHOLDER: break; case MOO_UI_NODE_CONTAINER: case MOO_UI_NODE_SEPARATOR: g_warning ("%s: can't add item to node of type %s", G_STRLOC, NODE_TYPE_NAME[parent->type]); } item = item_new (name, action); item->parent = parent; parent->children = g_slist_insert (parent->children, item, position); merge_add_node (merge, (Node*) item); update_widgets (xml, UPDATE_ADD_NODE, (Node*) item); return (Node*) item; } void moo_ui_xml_insert (MooUIXML *xml, guint merge_id, MooUINode *parent, int position, const char *markup) { Merge *merge; GError *error = NULL; MooMarkupDoc *doc; MooMarkupNode *mchild; int children_length; g_return_if_fail (MOO_IS_UI_XML (xml)); g_return_if_fail (markup != NULL); g_return_if_fail (!parent || node_is_ancestor (parent, xml->priv->ui)); merge = lookup_merge (xml, merge_id); g_return_if_fail (merge != NULL); if (!parent) parent = xml->priv->ui; if (parent->type == MOO_UI_NODE_SEPARATOR) { g_warning ("%s: can't add stuff to node of type %s", G_STRLOC, NODE_TYPE_NAME[parent->type]); return; } doc = moo_markup_parse_memory (markup, -1, &error); if (!doc) { g_warning ("%s: could not parse markup", G_STRLOC); if (error) { g_warning ("%s: %s", G_STRLOC, error->message); g_error_free (error); } return; } children_length = g_slist_length (parent->children); if (position < 0 || position > children_length) position = children_length; for (mchild = doc->last; mchild != NULL; mchild = mchild->prev) { Node *node; if (!MOO_MARKUP_IS_ELEMENT (mchild)) continue; node = parse_markup (mchild); if (!node) continue; if (!node_check (node)) { node_free (node); continue; } switch (node->type) { case WIDGET: case CONTAINER: if (parent->type != CONTAINER) { g_warning ("%s: can not add node of type %s to node of type %s", G_STRLOC, NODE_TYPE_NAME[node->type], NODE_TYPE_NAME[parent->type]); node_free (node); continue; } break; case ITEM: case PLACEHOLDER: case SEPARATOR: if (parent->type == SEPARATOR || parent->type == CONTAINER) { g_warning ("%s: can not add node of type %s to node of type %s", G_STRLOC, NODE_TYPE_NAME[node->type], NODE_TYPE_NAME[parent->type]); node_free (node); continue; } break; } /* XXX check names? */ node->parent = parent; parent->children = g_slist_insert (parent->children, node, position); merge_add_node (merge, node); update_widgets (xml, UPDATE_ADD_NODE, node); } moo_markup_doc_unref (doc); } void moo_ui_xml_insert_after (MooUIXML *xml, guint merge_id, MooUINode *parent, MooUINode *after, const char *markup) { int position; g_return_if_fail (MOO_IS_UI_XML (xml)); if (!parent) parent = xml->priv->ui; g_return_if_fail (!after || after->parent == parent); if (!after) position = 0; else position = g_slist_index (parent->children, after) + 1; moo_ui_xml_insert (xml, merge_id, parent, position, markup); } void moo_ui_xml_insert_before (MooUIXML *xml, guint merge_id, MooUINode *parent, MooUINode *before, const char *markup) { int position; g_return_if_fail (MOO_IS_UI_XML (xml)); if (!parent) parent = xml->priv->ui; g_return_if_fail (!before || before->parent == parent); if (!before) position = g_slist_length (parent->children); else position = g_slist_index (parent->children, before); moo_ui_xml_insert (xml, merge_id, parent, position, markup); } void moo_ui_xml_insert_markup_after (MooUIXML *xml, guint merge_id, const char *parent_path, const char *after_name, const char *markup) { Node *parent = NULL, *after = NULL; g_return_if_fail (MOO_IS_UI_XML (xml)); g_return_if_fail (markup != NULL); if (parent_path) { parent = moo_ui_xml_find_node (xml, parent_path); g_return_if_fail (parent != NULL); } else { parent = xml->priv->ui; } if (after_name) { after = moo_ui_node_get_child (parent, after_name); g_return_if_fail (after != NULL); } moo_ui_xml_insert_after (xml, merge_id, parent, after, markup); } void moo_ui_xml_insert_markup_before (MooUIXML *xml, guint merge_id, const char *parent_path, const char *before_name, const char *markup) { Node *parent = NULL, *before = NULL; g_return_if_fail (MOO_IS_UI_XML (xml)); g_return_if_fail (markup != NULL); if (parent_path) { parent = moo_ui_xml_find_node (xml, parent_path); g_return_if_fail (parent != NULL); } else { parent = xml->priv->ui; } if (before_name) { before = moo_ui_node_get_child (parent, before_name); g_return_if_fail (before != NULL); } moo_ui_xml_insert_before (xml, merge_id, parent, before, markup); } void moo_ui_xml_insert_markup (MooUIXML *xml, guint merge_id, const char *parent_path, int position, const char *markup) { Node *parent = NULL; g_return_if_fail (MOO_IS_UI_XML (xml)); g_return_if_fail (markup != NULL); if (parent_path) { parent = moo_ui_xml_find_node (xml, parent_path); g_return_if_fail (parent != NULL); } else { parent = xml->priv->ui; } moo_ui_xml_insert (xml, merge_id, parent, position, markup); } static gboolean node_is_ancestor (Node *node, Node *ancestor) { Node *n; g_return_val_if_fail (node != NULL, FALSE); g_return_val_if_fail (ancestor != NULL, FALSE); for (n = node; n != NULL; n = n->parent) if (n == ancestor) return TRUE; return FALSE; } void moo_ui_xml_remove_ui (MooUIXML *xml, guint merge_id) { Merge *merge; GSList *nodes; g_return_if_fail (MOO_IS_UI_XML (xml)); merge = lookup_merge (xml, merge_id); g_return_if_fail (merge != NULL); nodes = g_slist_copy (merge->nodes); while (nodes) { moo_ui_xml_remove_node (xml, nodes->data); nodes = g_slist_delete_link (nodes, nodes); } g_return_if_fail (merge->nodes == NULL); xml->priv->merged_ui = g_slist_remove (xml->priv->merged_ui, merge); g_free (merge); } void moo_ui_xml_remove_node (MooUIXML *xml, MooUINode *node) { Node *parent; g_return_if_fail (MOO_IS_UI_XML (xml)); g_return_if_fail (node != NULL); g_return_if_fail (node_is_ancestor (node, xml->priv->ui)); SLIST_FOREACH (xml->priv->merged_ui, l) { Merge *merge = l->data; GSList *merge_nodes = g_slist_copy (merge->nodes); SLIST_FOREACH (merge_nodes, n) { Node *merge_node = n->data; if (node_is_ancestor (merge_node, node)) merge_remove_node (merge, merge_node); } SLIST_FOREACH_END; g_slist_free (merge_nodes); } SLIST_FOREACH_END; update_widgets (xml, UPDATE_REMOVE_NODE, node); parent = node->parent; parent->children = g_slist_remove (parent->children, node); node->parent = NULL; while (parent && parent->type == MOO_UI_NODE_PLACEHOLDER) parent = parent->parent; SLIST_FOREACH (xml->priv->toplevels, l) { Toplevel *toplevel = l->data; if (node_is_ancestor (parent, toplevel->node)) check_separators (parent, toplevel); } SLIST_FOREACH_END; node_free (node); } static void merge_add_node (Merge *merge, Node *added) { g_return_if_fail (merge != NULL); g_return_if_fail (added != NULL); SLIST_FOREACH (merge->nodes, l) { Node *node = l->data; if (node_is_ancestor (added, node)) return; } SLIST_FOREACH_END; merge->nodes = g_slist_prepend (merge->nodes, added); } static void merge_remove_node (Merge *merge, Node *removed) { g_return_if_fail (merge != NULL); g_return_if_fail (removed != NULL); merge->nodes = g_slist_remove (merge->nodes, removed); } MooUINode * moo_ui_xml_get_node (MooUIXML *xml, const char *path) { g_return_val_if_fail (MOO_IS_UI_XML (xml), NULL); g_return_val_if_fail (path != NULL, NULL); return moo_ui_node_get_child (xml->priv->ui, path); } static MooUINode * moo_ui_xml_find_node (MooUIXML *xml, const char *path) { MooUINode *node; g_return_val_if_fail (MOO_IS_UI_XML (xml), NULL); g_return_val_if_fail (path != NULL, NULL); node = moo_ui_xml_get_node (xml, path); if (!node) node = moo_ui_xml_find_placeholder (xml, path); return node; } static gboolean find_placeholder_func (Node *node, gpointer user_data) { struct { Node *found; const char *name; } *data = user_data; if (node->type != PLACEHOLDER) return FALSE; if (!strcmp (node->name, data->name)) { data->found = node; return TRUE; } return FALSE; } MooUINode* moo_ui_xml_find_placeholder (MooUIXML *xml, const char *name) { struct { Node *found; const char *name; } data; g_return_val_if_fail (MOO_IS_UI_XML (xml), NULL); g_return_val_if_fail (name != NULL, NULL); data.found = NULL; data.name = name; node_foreach (xml->priv->ui, find_placeholder_func, &data); return data.found; } char* moo_ui_node_get_path (MooUINode *node) { GString *path; g_return_val_if_fail (node != NULL, NULL); path = g_string_new (node->name); while (node->parent && node->parent->parent) { node = node->parent; g_string_prepend_c (path, '/'); g_string_prepend (path, node->name); } g_print ("path: %s\n", path->str); return g_string_free (path, FALSE); } MooUINode* moo_ui_node_get_child (MooUINode *node, const char *path) { char **pieces, **p; g_return_val_if_fail (node != NULL, NULL); g_return_val_if_fail (path != NULL, NULL); pieces = g_strsplit (path, "/", 0); if (!pieces) return node; for (p = pieces; *p != NULL; ++p) { Node *child = NULL; if (!**p) continue; SLIST_FOREACH (node->children, l) { child = l->data; if (!strcmp (child->name, *p)) break; else child = NULL; } SLIST_FOREACH_END; if (child) { node = child; } else { node = NULL; goto out; } } out: g_strfreev (pieces); return node; } static Toplevel* toplevel_new (Node *node, MooActionCollection *actions, GtkAccelGroup *accel_group) { Toplevel *top; g_return_val_if_fail (node != NULL, NULL); top = g_new0 (Toplevel, 1); top->node = node; top->widget = NULL; top->children = g_hash_table_new (g_direct_hash, g_direct_equal); top->actions = actions; top->accel_group = accel_group; return top; } static void toplevel_free (Toplevel *toplevel) { if (toplevel) { g_hash_table_destroy (toplevel->children); g_free (toplevel); } } static GQuark toplevel_quark (void) { static GQuark q = 0; if (!q) q = g_quark_from_static_string ("moo-ui-xml-toplevel"); return q; } static GQuark node_quark (void) { static GQuark q = 0; if (!q) q = g_quark_from_static_string ("moo-ui-xml-node"); return q; } static Node * get_effective_parent (Node *node) { Node *parent; g_return_val_if_fail (node != NULL && node->parent != NULL, NULL); for (parent = node->parent; parent && parent->type == PLACEHOLDER; parent = parent->parent) ; return parent; } static void visibility_notify (GtkWidget *widget, G_GNUC_UNUSED gpointer whatever, MooUIXML *xml) { Toplevel *toplevel; Node *node; g_return_if_fail (GTK_IS_WIDGET (widget)); g_return_if_fail (MOO_IS_UI_XML (xml)); toplevel = g_object_get_qdata (G_OBJECT (widget), TOPLEVEL_QUARK); g_return_if_fail (toplevel != NULL); if (toplevel->in_creation) return; node = g_object_get_qdata (G_OBJECT (widget), NODE_QUARK); g_return_if_fail (node != NULL && node->parent != NULL); g_return_if_fail (node->type == ITEM); check_separators (get_effective_parent (node), toplevel); } static void xml_add_item_widget (MooUIXML *xml, GtkWidget *widget) { g_signal_connect (widget, "notify::visible", G_CALLBACK (visibility_notify), xml); } static void widget_destroyed (GtkWidget *widget, MooUIXML *xml) { g_return_if_fail (GTK_IS_WIDGET (widget)); g_return_if_fail (MOO_IS_UI_XML (xml)); xml_remove_widget (xml, widget); } static void xml_remove_widget (MooUIXML *xml, GtkWidget *widget) { Toplevel *toplevel; Node *node; g_return_if_fail (GTK_IS_WIDGET (widget)); g_return_if_fail (MOO_IS_UI_XML (xml)); toplevel = g_object_get_qdata (G_OBJECT (widget), TOPLEVEL_QUARK); g_return_if_fail (toplevel != NULL); g_return_if_fail (toplevel->widget != widget); node = g_object_get_qdata (G_OBJECT (widget), NODE_QUARK); g_hash_table_remove (toplevel->children, node); g_object_set_qdata (G_OBJECT (widget), NODE_QUARK, NULL); g_object_set_qdata (G_OBJECT (widget), TOPLEVEL_QUARK, NULL); g_signal_handlers_disconnect_by_func (widget, (gpointer) widget_destroyed, xml); g_signal_handlers_disconnect_by_func (widget, (gpointer) visibility_notify, xml); } static void xml_add_widget (MooUIXML *xml, GtkWidget *widget, Toplevel *toplevel, Node *node) { g_signal_connect (widget, "destroy", G_CALLBACK (widget_destroyed), xml); g_object_set_qdata (G_OBJECT (widget), TOPLEVEL_QUARK, toplevel); g_object_set_qdata (G_OBJECT (widget), NODE_QUARK, node); g_hash_table_insert (toplevel->children, node, widget); } static void prepend_value (G_GNUC_UNUSED gpointer key, gpointer value, GSList **list) { *list = g_slist_prepend (*list, value); } static GSList* hash_table_list_values (GHashTable *hash_table) { GSList *list = NULL; g_hash_table_foreach (hash_table, (GHFunc) prepend_value, &list); return list; } static void toplevel_destroyed (GtkWidget *widget, MooUIXML *xml) { Toplevel *toplevel; g_return_if_fail (GTK_IS_WIDGET (widget)); g_return_if_fail (MOO_IS_UI_XML (xml)); toplevel = g_object_get_qdata (G_OBJECT (widget), TOPLEVEL_QUARK); g_return_if_fail (toplevel != NULL); g_return_if_fail (toplevel->widget == widget); xml_delete_toplevel (xml, toplevel); } static void xml_delete_toplevel (MooUIXML *xml, Toplevel *toplevel) { GSList *children, *l; children = hash_table_list_values (toplevel->children); for (l = children; l != NULL; l = l->next) { GtkWidget *child = GTK_WIDGET (l->data); if (child != toplevel->widget) xml_remove_widget (xml, child); } g_signal_handlers_disconnect_by_func (toplevel->widget, (gpointer) toplevel_destroyed, xml); g_object_set_qdata (G_OBJECT (toplevel->widget), TOPLEVEL_QUARK, NULL); g_object_set_qdata (G_OBJECT (toplevel->widget), NODE_QUARK, NULL); xml->priv->toplevels = g_slist_remove (xml->priv->toplevels, toplevel); g_slist_free (children); toplevel_free (toplevel); } static void xml_connect_toplevel (MooUIXML *xml, Toplevel *toplevel) { g_signal_connect (toplevel->widget, "destroy", G_CALLBACK (toplevel_destroyed), xml); g_object_set_qdata (G_OBJECT (toplevel->widget), TOPLEVEL_QUARK, toplevel); g_object_set_qdata (G_OBJECT (toplevel->widget), NODE_QUARK, toplevel->node); g_hash_table_insert (toplevel->children, toplevel->node, toplevel->widget); } static gboolean create_menu_separator (MooUIXML *xml, Toplevel *toplevel, GtkMenuShell *menu, Node *node, int index) { GtkWidget *item = gtk_separator_menu_item_new (); gtk_menu_shell_insert (menu, item, index); xml_add_widget (xml, item, toplevel, node); return TRUE; } static gboolean node_is_empty (Node *node) { SLIST_FOREACH (node->children, l) { Node *child = l->data; if (child->type == SEPARATOR) return FALSE; if (child->type == PLACEHOLDER) { if (!node_is_empty (child)) return FALSE; } else { return FALSE; } } SLIST_FOREACH_END; return TRUE; } static void create_menu_item (MooUIXML *xml, Toplevel *toplevel, GtkMenuShell *menu, Node *node, int index) { GtkWidget *menu_item = NULL; Item *item; g_return_if_fail (node != NULL && node->type == ITEM); g_return_if_fail (node->name != NULL); item = (Item*) node; if (item->action) { GtkAction *action; g_return_if_fail (toplevel->actions != NULL); action = moo_action_collection_get_action (toplevel->actions, item->action); if (!action) { #if REPORT_UNKNOWN_ACTIONS g_critical ("%s: could not find action '%s'", G_STRLOC, item->action); #endif return; } if (_moo_action_get_dead (action)) return; gtk_action_set_accel_group (action, toplevel->accel_group); menu_item = gtk_action_create_menu_item (action); } else { if (item->stock_id) { menu_item = gtk_image_menu_item_new_from_stock (item->stock_id, NULL); } else if (item->label) { if (item->icon_stock_id) { GtkWidget *icon = gtk_image_new_from_stock (item->icon_stock_id, GTK_ICON_SIZE_MENU); menu_item = gtk_image_menu_item_new_with_mnemonic (item->label); gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (menu_item), icon); } else { menu_item = gtk_menu_item_new_with_mnemonic (item->label); } } else { g_warning ("item '%s' does not have an associated action, label, or stock id", item->name); } if (menu_item) gtk_widget_show (menu_item); } g_return_if_fail (menu_item != NULL); gtk_menu_shell_insert (menu, menu_item, index); xml_add_widget (xml, menu_item, toplevel, node); xml_add_item_widget (xml, menu_item); if (!node_is_empty (node)) { GtkWidget *submenu = gtk_menu_new (); gtk_widget_show (submenu); gtk_menu_set_accel_group (GTK_MENU (submenu), toplevel->accel_group); gtk_menu_item_set_submenu (GTK_MENU_ITEM (menu_item), submenu); fill_menu_shell (xml, toplevel, node, GTK_MENU_SHELL (submenu)); } /* XXX completely empty menus? */ if (node->children) check_empty (node, menu_item, toplevel); } static GSList* node_list_children (Node *parent) { GSList *list = NULL, *l; for (l = parent->children; l != NULL; l = l->next) { GSList *tmp, *t; Node *node = l->data; switch (node->type) { case MOO_UI_NODE_ITEM: case MOO_UI_NODE_SEPARATOR: list = g_slist_prepend (list, node); break; case MOO_UI_NODE_PLACEHOLDER: tmp = node_list_children (node); for (t = tmp; t != NULL; t = t->next) list = g_slist_prepend (list, t->data); g_slist_free (tmp); break; default: g_return_val_if_reached (g_slist_reverse (list)); } } return g_slist_reverse (list); } static void real_foreach (Node *node, gpointer data) { GSList *l; struct { NodeForeachFunc func; gpointer func_data; gboolean stop; } *foreach_data = data; if (foreach_data->stop) return; if (foreach_data->func (node, foreach_data->func_data)) { foreach_data->stop = TRUE; return; } for (l = node->children; l != NULL; l = l->next) { Node *child = l->data; real_foreach (child, data); if (foreach_data->stop) return; } } static void node_foreach (Node *node, NodeForeachFunc func, gpointer data) { struct { NodeForeachFunc func; gpointer func_data; gboolean stop; } foreach_data; g_return_if_fail (node != NULL); g_return_if_fail (func != NULL); foreach_data.func = func; foreach_data.func_data = data; foreach_data.stop = FALSE; real_foreach (node, &foreach_data); } static GtkWidget* toplevel_get_widget (Toplevel *toplevel, Node *node) { return g_hash_table_lookup (toplevel->children, node); } static void check_empty (Node *parent, GtkWidget *widget, Toplevel *toplevel) { GSList *children, *l; gboolean has_children = FALSE; children = node_list_children (parent); for (l = children; l != NULL; l = l->next) { Node *node = l->data; if (node->type == MOO_UI_NODE_ITEM) { GtkWidget *nw = toplevel_get_widget (toplevel, node); if (nw && GTK_WIDGET_VISIBLE (nw)) { has_children = TRUE; break; } } } /* XXX decide something on this stuff */ #if 0 // if (!(parent->flags & MOO_UI_NODE_ENABLE_EMPTY)) // gtk_widget_set_sensitive (widget, has_children); #endif g_object_set (widget, "visible", has_children, NULL); g_slist_free (children); } static void check_separators (Node *parent, Toplevel *toplevel) { GSList *children, *l; Node *separator = NULL; gboolean first = TRUE; gboolean has_children = FALSE; GtkWidget *widget; if (!toplevel_get_widget (toplevel, parent)) return; children = node_list_children (parent); for (l = children; l != NULL; l = l->next) { Node *node = l->data; switch (node->type) { case MOO_UI_NODE_ITEM: widget = toplevel_get_widget (toplevel, node); if (!widget || !GTK_WIDGET_VISIBLE (widget)) continue; has_children = TRUE; if (!first) { if (separator) { GtkWidget *sep_widget = toplevel_get_widget (toplevel, separator); g_return_if_fail (sep_widget != NULL); gtk_widget_show (sep_widget); } } else { first = FALSE; separator = NULL; } break; case MOO_UI_NODE_SEPARATOR: widget = toplevel_get_widget (toplevel, node); g_return_if_fail (widget != NULL); gtk_widget_hide (widget); if (!first) separator = node; break; default: g_return_if_reached (); } } widget = toplevel_get_widget (toplevel, parent); if (widget) check_empty (parent, widget, toplevel); g_slist_free (children); } static gboolean fill_menu_shell (MooUIXML *xml, Toplevel *toplevel, Node *menu_node, GtkMenuShell *menu) { GSList *children; children = node_list_children (menu_node); SLIST_FOREACH (children, l) { Node *node = l->data; switch (node->type) { case MOO_UI_NODE_ITEM: create_menu_item (xml, toplevel, menu, node, -1); break; case MOO_UI_NODE_SEPARATOR: create_menu_separator (xml, toplevel, menu, node, -1); break; default: g_warning ("%s: invalid menu item type %s", G_STRLOC, NODE_TYPE_NAME[node->type]); return FALSE; } } SLIST_FOREACH_END; check_separators (menu_node, toplevel); g_slist_free (children); return TRUE; } static gboolean create_menu_shell (MooUIXML *xml, Toplevel *toplevel, MooUIWidgetType type) { g_return_val_if_fail (toplevel != NULL, FALSE); g_return_val_if_fail (toplevel->widget == NULL, FALSE); g_return_val_if_fail (toplevel->node != NULL, FALSE); if (type == MOO_UI_MENUBAR) { toplevel->widget = gtk_menu_bar_new (); } else { toplevel->widget = gtk_menu_new (); gtk_menu_set_accel_group (GTK_MENU (toplevel->widget), toplevel->accel_group); } xml_connect_toplevel (xml, toplevel); return fill_menu_shell (xml, toplevel, toplevel->node, GTK_MENU_SHELL (toplevel->widget)); } static gboolean create_tool_separator (MooUIXML *xml, Toplevel *toplevel, GtkToolbar *toolbar, Node *node, int index) { GtkToolItem *item = gtk_separator_tool_item_new (); gtk_toolbar_insert (toolbar, item, index); xml_add_widget (xml, GTK_WIDGET (item), toplevel, node); return TRUE; } #if GTK_CHECK_VERSION(2,6,0) #define IS_MENU_TOOL_BUTTON(wid) (GTK_IS_MENU_TOOL_BUTTON (wid) || \ MOO_IS_MENU_TOOL_BUTTON (wid)) #else #define IS_MENU_TOOL_BUTTON MOO_IS_MENU_TOOL_BUTTON #endif static void menu_tool_button_set_menu (GtkWidget *button, GtkWidget *menu) { if (MOO_IS_MENU_TOOL_BUTTON (button)) moo_menu_tool_button_set_menu (MOO_MENU_TOOL_BUTTON (button), menu); #if GTK_CHECK_VERSION(2,6,0) else if (GTK_IS_MENU_TOOL_BUTTON (button)) gtk_menu_tool_button_set_menu (GTK_MENU_TOOL_BUTTON (button), menu); #endif else g_return_if_reached (); } static GtkWidget* menu_tool_button_get_menu (GtkWidget *button) { if (MOO_IS_MENU_TOOL_BUTTON (button)) return moo_menu_tool_button_get_menu (MOO_MENU_TOOL_BUTTON (button)); #if GTK_CHECK_VERSION(2,6,0) else if (GTK_IS_MENU_TOOL_BUTTON (button)) return gtk_menu_tool_button_get_menu (GTK_MENU_TOOL_BUTTON (button)); #endif else g_return_val_if_reached (NULL); } static gboolean create_tool_item (MooUIXML *xml, Toplevel *toplevel, GtkToolbar *toolbar, Node *node, int index) { GtkWidget *tool_item = NULL; Item *item; g_return_val_if_fail (node != NULL && node->type == ITEM, FALSE); item = (Item*) node; if (item->action) { GtkAction *action; g_return_val_if_fail (toplevel->actions != NULL, FALSE); action = moo_action_collection_get_action (toplevel->actions, item->action); if (!action || _moo_action_get_dead (action)) return TRUE; gtk_action_set_accel_group (action, toplevel->accel_group); if (_moo_action_get_has_submenu (action)) { tool_item = GTK_WIDGET (gtk_menu_tool_button_new (NULL, NULL)); gtk_action_connect_proxy (action, tool_item); } else { tool_item = gtk_action_create_tool_item (action); } if (index > gtk_toolbar_get_n_items (toolbar)) index = -1; gtk_toolbar_insert (toolbar, GTK_TOOL_ITEM (tool_item), index); g_object_notify (G_OBJECT (action), "tooltip"); if (node->children) { if (!IS_MENU_TOOL_BUTTON (tool_item)) { g_critical ("%s: oops", G_STRLOC); } else { GtkWidget *menu = gtk_menu_new (); /* XXX empty menu */ gtk_widget_show (menu); menu_tool_button_set_menu (tool_item, menu); fill_menu_shell (xml, toplevel, node, GTK_MENU_SHELL (menu)); } } } else { GtkWidget *menu; tool_item = moo_menu_tool_button_new (); gtk_widget_show (tool_item); if (item->tooltip) _moo_widget_set_tooltip (tool_item, item->tooltip); if (item->icon_stock_id) gtk_tool_button_set_stock_id (GTK_TOOL_BUTTON (tool_item), item->icon_stock_id); if (item->stock_id) gtk_tool_button_set_stock_id (GTK_TOOL_BUTTON (tool_item), item->stock_id); if (item->label) gtk_tool_button_set_label (GTK_TOOL_BUTTON (tool_item), item->label); gtk_toolbar_insert (toolbar, GTK_TOOL_ITEM (tool_item), index); menu = gtk_menu_new (); /* XXX empty menu */ gtk_widget_show (menu); menu_tool_button_set_menu (tool_item, menu); fill_menu_shell (xml, toplevel, node, GTK_MENU_SHELL (menu)); } g_return_val_if_fail (tool_item != NULL, FALSE); gtk_tool_item_set_homogeneous (GTK_TOOL_ITEM (tool_item), FALSE); xml_add_widget (xml, tool_item, toplevel, node); xml_add_item_widget (xml, tool_item); return TRUE; } static gboolean fill_toolbar (MooUIXML *xml, Toplevel *toplevel, Node *toolbar_node, GtkToolbar *toolbar) { gboolean result = TRUE; GSList *children; children = node_list_children (toolbar_node); SLIST_FOREACH (children, l) { Node *node = l->data; switch (node->type) { case MOO_UI_NODE_ITEM: create_tool_item (xml, toplevel, toolbar, node, -1); break; case MOO_UI_NODE_SEPARATOR: create_tool_separator (xml, toplevel, toolbar, node, -1); break; default: g_warning ("%s: invalid tool item type %s", G_STRLOC, NODE_TYPE_NAME[node->type]); return FALSE; } if (!result) return FALSE; } SLIST_FOREACH_END; check_separators (toolbar_node, toplevel); g_slist_free (children); return TRUE; } static gboolean create_toolbar (MooUIXML *xml, Toplevel *toplevel) { g_return_val_if_fail (toplevel != NULL, FALSE); g_return_val_if_fail (toplevel->widget == NULL, FALSE); g_return_val_if_fail (toplevel->node != NULL, FALSE); toplevel->widget = gtk_toolbar_new (); gtk_toolbar_set_tooltips (GTK_TOOLBAR (toplevel->widget), TRUE); xml_connect_toplevel (xml, toplevel); return fill_toolbar (xml, toplevel, toplevel->node, GTK_TOOLBAR (toplevel->widget)); } gpointer moo_ui_xml_create_widget (MooUIXML *xml, MooUIWidgetType type, const char *path, MooActionCollection *actions, GtkAccelGroup *accel_group) { Node *node; Toplevel *toplevel; gboolean result = FALSE; g_return_val_if_fail (MOO_IS_UI_XML (xml), NULL); g_return_val_if_fail (path != NULL, NULL); g_return_val_if_fail (!actions || MOO_IS_ACTION_COLLECTION (actions), NULL); node = moo_ui_xml_get_node (xml, path); g_return_val_if_fail (node != NULL, NULL); if (node->type != WIDGET) { g_warning ("%s: can create widgets only for nodes of type %s", G_STRLOC, NODE_TYPE_NAME[WIDGET]); return NULL; } if (type < 1 || type > 3) { g_warning ("%s: invalid widget type %u", G_STRLOC, type); return NULL; } toplevel = toplevel_new (node, actions, accel_group); toplevel->in_creation = TRUE; xml->priv->toplevels = g_slist_append (xml->priv->toplevels, toplevel); switch (type) { case MOO_UI_MENUBAR: result = create_menu_shell (xml, toplevel, MOO_UI_MENUBAR); break; case MOO_UI_MENU: result = create_menu_shell (xml, toplevel, MOO_UI_MENU); break; case MOO_UI_TOOLBAR: result = create_toolbar (xml, toplevel); break; } toplevel->in_creation = FALSE; if (!result) { xml_delete_toplevel (xml, toplevel); return NULL; } return toplevel->widget; } GtkWidget* moo_ui_xml_get_widget (MooUIXML *xml, GtkWidget *widget, const char *path) { Toplevel *toplevel; MooUINode *node; g_return_val_if_fail (MOO_IS_UI_XML (xml), NULL); g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL); g_return_val_if_fail (path != NULL, NULL); node = moo_ui_xml_get_node (xml, path); g_return_val_if_fail (node != NULL, NULL); toplevel = g_object_get_qdata (G_OBJECT (widget), TOPLEVEL_QUARK); g_return_val_if_fail (toplevel != NULL, NULL); return toplevel_get_widget (toplevel, node); } static Node* effective_parent (Node *node) { Node *parent; g_return_val_if_fail (node != NULL, NULL); parent = node->parent; while (parent && parent->type == PLACEHOLDER) parent = parent->parent; return parent; } static int effective_index (Node *parent, Node *node) { GSList *children; int index; g_return_val_if_fail (effective_parent (node) == parent, -1); children = node_list_children (parent); index = g_slist_index (children, node); g_slist_free (children); g_return_val_if_fail (index >= 0, -1); return index; } static void toplevel_add_node (MooUIXML *xml, Toplevel *toplevel, Node *node) { g_return_if_fail (GTK_IS_WIDGET (toplevel->widget)); g_return_if_fail (node->type == ITEM || node->type == SEPARATOR); g_return_if_fail (node_is_ancestor (node, toplevel->node)); if (GTK_IS_TOOLBAR (toplevel->widget)) { GtkWidget *parent_widget; Node *parent = effective_parent (node); g_return_if_fail (parent != NULL); parent_widget = toplevel_get_widget (toplevel, parent); g_return_if_fail (parent_widget != NULL); if (GTK_IS_TOOLBAR (parent_widget)) { switch (node->type) { case ITEM: create_tool_item (xml, toplevel, GTK_TOOLBAR (parent_widget), node, effective_index (parent, node)); break; case SEPARATOR: create_tool_separator (xml, toplevel, GTK_TOOLBAR (parent_widget), node, effective_index (parent, node)); break; default: g_return_if_reached (); } check_separators (parent, toplevel); } else if (IS_MENU_TOOL_BUTTON (parent_widget)) { GtkWidget *menu; menu = menu_tool_button_get_menu (parent_widget); if (!menu) { menu = gtk_menu_new (); gtk_widget_show (menu); menu_tool_button_set_menu (parent_widget, menu); } switch (node->type) { case ITEM: create_menu_item (xml, toplevel, GTK_MENU_SHELL (menu), node, effective_index (parent, node)); break; case SEPARATOR: create_menu_separator (xml, toplevel, GTK_MENU_SHELL (menu), node, effective_index (parent, node)); break; default: g_return_if_reached (); } check_separators (parent, toplevel); } else { g_return_if_reached (); } } else if (GTK_IS_MENU_SHELL (toplevel->widget)) { GtkWidget *parent_widget, *menu_shell; Node *parent = effective_parent (node); g_return_if_fail (parent != NULL); parent_widget = toplevel_get_widget (toplevel, parent); g_return_if_fail (parent_widget != NULL); if (GTK_IS_MENU_SHELL (parent_widget)) { menu_shell = parent_widget; } else { g_return_if_fail (GTK_IS_MENU_ITEM (parent_widget)); menu_shell = gtk_menu_item_get_submenu (GTK_MENU_ITEM (parent_widget)); if (!menu_shell) { menu_shell = gtk_menu_new (); gtk_widget_show (menu_shell); gtk_menu_item_set_submenu (GTK_MENU_ITEM (parent_widget), menu_shell); } } switch (node->type) { case ITEM: create_menu_item (xml, toplevel, GTK_MENU_SHELL (menu_shell), node, effective_index (parent, node)); break; case SEPARATOR: create_menu_separator (xml, toplevel, GTK_MENU_SHELL (menu_shell), node, effective_index (parent, node)); break; default: g_return_if_reached (); } check_separators (parent, toplevel); } else { g_return_if_reached (); } } static GSList* node_list_all_children (Node *node) { GSList *list, *l; g_return_val_if_fail (node != NULL, NULL); list = g_slist_append (NULL, node); for (l = node->children; l != NULL; l = l->next) list = g_slist_append (list, node_list_all_children (l->data)); return list; } static void toplevel_remove_node (G_GNUC_UNUSED MooUIXML *xml, Toplevel *toplevel, Node *node) { GSList *children, *l; g_return_if_fail (node != toplevel->node); g_return_if_fail (node_is_ancestor (node, toplevel->node)); children = node_list_all_children (node); for (l = children; l != NULL; l = l->next) { GtkWidget *widget = g_hash_table_lookup (toplevel->children, node); if (widget) gtk_widget_destroy (widget); } g_slist_free (children); } static void update_widgets (MooUIXML *xml, UpdateType type, Node *node) { switch (type) { case UPDATE_ADD_NODE: SLIST_FOREACH (xml->priv->toplevels, l) { Toplevel *toplevel = l->data; if (node_is_ancestor (node, toplevel->node)) toplevel_add_node (xml, toplevel, node); } SLIST_FOREACH_END; break; case UPDATE_REMOVE_NODE: SLIST_FOREACH (xml->priv->toplevels, l) { Toplevel *toplevel = l->data; if (node_is_ancestor (toplevel->node, node)) xml_delete_toplevel (xml, toplevel); else if (node_is_ancestor (node, toplevel->node)) toplevel_remove_node (xml, toplevel, node); } SLIST_FOREACH_END; break; case UPDATE_CHANGE_NODE: g_warning ("%s: implement me", G_STRLOC); break; default: g_return_if_reached (); } } static void moo_ui_xml_finalize (GObject *object) { MooUIXML *xml = MOO_UI_XML (object); SLIST_FOREACH (xml->priv->toplevels, t) { Toplevel *toplevel = t->data; GSList *widgets = hash_table_list_values (toplevel->children); SLIST_FOREACH (widgets, w) { GObject *widget = G_OBJECT (w->data); g_object_set_qdata (widget, NODE_QUARK, NULL); g_object_set_qdata (widget, TOPLEVEL_QUARK, NULL); g_signal_handlers_disconnect_by_func (widget, (gpointer) widget_destroyed, xml); g_signal_handlers_disconnect_by_func (widget, (gpointer) visibility_notify, xml); } SLIST_FOREACH_END; g_slist_free (widgets); toplevel_free (toplevel); } SLIST_FOREACH_END; SLIST_FOREACH (xml->priv->merged_ui, m) { Merge *merge = m->data; g_slist_free (merge->nodes); g_free (merge); } SLIST_FOREACH_END; g_slist_free (xml->priv->toplevels); g_slist_free (xml->priv->merged_ui); node_free (xml->priv->ui); g_free (xml->priv); xml->priv = NULL; G_OBJECT_CLASS(moo_ui_xml_parent_class)->finalize (object); } GType moo_ui_node_get_type (void) { static GType type = 0; if (G_UNLIKELY (!type)) type = g_pointer_type_register_static ("MooUINode"); return type; } GType moo_ui_widget_type_get_type (void) { static GType type = 0; if (G_UNLIKELY (!type)) { static const GEnumValue values[] = { { MOO_UI_MENUBAR, (char*) "MOO_UI_MENUBAR", (char*) "menubar" }, { MOO_UI_MENU, (char*) "MOO_UI_MENU", (char*) "menu" }, { MOO_UI_TOOLBAR, (char*) "MOO_UI_TOOLBAR", (char*) "toolbar" }, { 0, NULL, NULL } }; type = g_enum_register_static ("MooUIWidgetType", values); } return type; }