2987 lines
90 KiB
C
2987 lines
90 KiB
C
/*
|
|
* mooglade.c
|
|
*
|
|
* Copyright (C) 2004-2007 by Yevgen Muntyan <muntyan@math.tamu.edu>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License version 2.1 as published by the Free Software Foundation.
|
|
*
|
|
* See COPYING file that comes with this distribution.
|
|
*/
|
|
|
|
#ifdef GTK_DISABLE_DEPRECATED
|
|
#undef GTK_DISABLE_DEPRECATED
|
|
#include <gtk/gtkcombo.h>
|
|
#include <gtk/gtklist.h>
|
|
#include <gtk/gtklistitem.h>
|
|
#include <gtk/gtkoptionmenu.h>
|
|
#define GTK_DISABLE_DEPRECATED
|
|
#endif
|
|
|
|
#include "mooutils/mooglade.h"
|
|
#include "mooutils/moomarkup.h"
|
|
#include "mooutils/mooutils-gobject.h"
|
|
#include "mooutils/mooutils-misc.h"
|
|
#include "mooutils/mooprefsdialog.h"
|
|
#include "mooutils/mooprefs.h"
|
|
#include "mooutils/mooi18n.h"
|
|
#include <gtk/gtk.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
|
|
G_DEFINE_TYPE (MooGladeXML, moo_glade_xml, G_TYPE_OBJECT)
|
|
|
|
typedef enum {
|
|
MOO_GLADE_XML_ERROR_FAILED
|
|
} MooGladeXMLError;
|
|
|
|
#define MOO_GLADE_XML_ERROR (moo_glade_xml_error_quark ())
|
|
static GQuark moo_glade_xml_error_quark (void) G_GNUC_CONST;
|
|
|
|
|
|
#define FOREACH_ELM_START(parent,elm) \
|
|
G_STMT_START { \
|
|
MooMarkupNode *elm; \
|
|
for (elm = parent->children; elm != NULL; elm = elm->next) \
|
|
{ \
|
|
if (MOO_MARKUP_IS_ELEMENT (elm)) \
|
|
|
|
#define FOREACH_ELM_END \
|
|
} \
|
|
} G_STMT_END
|
|
|
|
|
|
typedef struct _Widget Widget;
|
|
typedef struct _Signal Signal;
|
|
typedef struct _WidgetProps WidgetProps;
|
|
typedef struct _PackingProps PackingProps;
|
|
typedef struct _Child Child;
|
|
|
|
struct _Widget {
|
|
Child *parent_node;
|
|
GtkWidget *widget;
|
|
char *id;
|
|
GType type;
|
|
WidgetProps *props;
|
|
GSList *children; /* Child* */
|
|
GSList *signals; /* Signal* */
|
|
};
|
|
|
|
struct _Signal {
|
|
char *name;
|
|
char *handler;
|
|
char *object;
|
|
};
|
|
|
|
struct _Child {
|
|
Widget *parent_node;
|
|
gboolean empty;
|
|
char *internal_child;
|
|
char *internal_parent_id;
|
|
Widget *widget;
|
|
PackingProps *props;
|
|
};
|
|
|
|
typedef enum {
|
|
PROP_RESPONSE_ID = 1 << 0,
|
|
PROP_HAS_DEFAULT = 1 << 1,
|
|
PROP_HAS_FOCUS = 1 << 2,
|
|
PROP_TOOLTIP = 1 << 3,
|
|
PROP_MNEMONIC_WIDGET = 1 << 4,
|
|
PROP_RADIO_GROUP = 1 << 5,
|
|
PROP_LABEL = 1 << 6,
|
|
PROP_USE_UNDERLINE = 1 << 7,
|
|
PROP_USE_STOCK = 1 << 8,
|
|
PROP_ENABLE_TOOLTIPS = 1 << 9,
|
|
PROP_HISTORY = 1 << 10
|
|
} PropMask;
|
|
|
|
struct _WidgetProps {
|
|
GParameter *params;
|
|
guint n_params;
|
|
|
|
GPtrArray *custom_props;
|
|
|
|
PropMask mask;
|
|
gboolean visible;
|
|
int response_id;
|
|
gboolean has_default;
|
|
gboolean has_focus;
|
|
char *tooltip;
|
|
char *mnemonic_widget;
|
|
char *radio_group;
|
|
char *label;
|
|
int history;
|
|
};
|
|
|
|
typedef enum {
|
|
PACK_PROP_LABEL_ITEM = 1 << 0,
|
|
PACK_PROP_TAB = 1 << 1
|
|
} PackPropMask;
|
|
|
|
struct _PackingProps {
|
|
GParameter *params;
|
|
guint n_params;
|
|
|
|
PackPropMask mask;
|
|
};
|
|
|
|
struct _MooGladeXMLPrivate {
|
|
GHashTable *widgets;
|
|
char *root_id;
|
|
|
|
char *translation_domain;
|
|
|
|
GHashTable *class_to_type;
|
|
GHashTable *id_to_type;
|
|
GHashTable *id_to_func;
|
|
|
|
GHashTable *props; /* char* -> char** */
|
|
|
|
MooGladeSignalFunc signal_func;
|
|
gpointer signal_func_data;
|
|
GDestroyNotify signal_func_notify;
|
|
MooGladePropFunc prop_func;
|
|
gpointer prop_func_data;
|
|
GDestroyNotify prop_func_notify;
|
|
|
|
guint done : 1;
|
|
};
|
|
|
|
typedef struct {
|
|
MooGladeCreateFunc func;
|
|
gpointer data;
|
|
} FuncDataPair;
|
|
|
|
|
|
#define NODE_IS_WIDGET(node) (!strcmp (node->name, "widget"))
|
|
#define NODE_IS_SIGNAL(node) (!strcmp (node->name, "signal"))
|
|
#define NODE_IS_PROPERTY(node) (!strcmp (node->name, "property"))
|
|
#define NODE_IS_CHILD(node) (!strcmp (node->name, "child"))
|
|
#define NODE_IS_PACKING(node) (!strcmp (node->name, "packing"))
|
|
|
|
|
|
static FuncDataPair *func_data_pair_new (MooGladeCreateFunc func,
|
|
gpointer data);
|
|
static void func_data_pair_free (FuncDataPair *pair);
|
|
static Widget *widget_new (MooGladeXML *xml,
|
|
Child *parent,
|
|
MooMarkupNode *node,
|
|
GError **error);
|
|
static void widget_free (Widget *widget);
|
|
static Child *child_new (MooGladeXML *xml,
|
|
Widget *parent,
|
|
MooMarkupNode *node,
|
|
GError **error);
|
|
static void child_free (Child *child);
|
|
static WidgetProps *widget_props_new (MooMarkupNode *node,
|
|
GType widget_type,
|
|
GHashTable *add_props,
|
|
gboolean ignore_errors,
|
|
const char *translation_domain,
|
|
GError **error);
|
|
static void widget_props_free (WidgetProps *props);
|
|
static PackingProps *packing_props_new (MooMarkupNode *node,
|
|
GType container_type,
|
|
GError **error);
|
|
static void packing_props_free (PackingProps *props);
|
|
static Signal *signal_new (const char *id,
|
|
const char *handler,
|
|
const char *object);
|
|
static void signal_free (Signal *signal);
|
|
static void collect_signals (MooMarkupNode *node,
|
|
Widget *widget);
|
|
|
|
static GType get_type_by_name (const char *name);
|
|
static gboolean parse_bool (const char *value);
|
|
static int parse_int (const char *value);
|
|
static GtkObject *parse_adjustment (const char *value);
|
|
static gboolean parse_property (GParamSpec *param_spec,
|
|
const char *value,
|
|
GParameter *param);
|
|
|
|
static gboolean moo_glade_xml_parse_markup (MooGladeXML *xml,
|
|
MooMarkupDoc *doc,
|
|
const char *root,
|
|
GtkWidget *root_widget,
|
|
GError **error);
|
|
static void moo_glade_xml_dispose (GObject *object);
|
|
static gboolean moo_glade_xml_add_widget (MooGladeXML *xml,
|
|
const char *id,
|
|
GtkWidget *widget,
|
|
GError **error);
|
|
static gboolean moo_glade_xml_build (MooGladeXML *xml,
|
|
Widget *widget_node,
|
|
GtkWidget *widget,
|
|
GError **error);
|
|
static void moo_glade_xml_cleanup (MooGladeXML *xml);
|
|
|
|
static gboolean create_child (MooGladeXML *xml,
|
|
GtkWidget *parent,
|
|
Child *child,
|
|
GError **error);
|
|
static void set_special_props (MooGladeXML *xml,
|
|
GtkWidget *widget,
|
|
WidgetProps *props);
|
|
static void set_custom_props (MooGladeXML *xml,
|
|
Widget *node);
|
|
static void set_mnemonics (MooGladeXML *xml,
|
|
Widget *node);
|
|
static gboolean set_default (MooGladeXML *xml,
|
|
Widget *node);
|
|
static gboolean set_focus (MooGladeXML *xml,
|
|
Widget *node);
|
|
static void connect_signals (MooGladeXML *xml,
|
|
Widget *node);
|
|
|
|
static GtkWidget *create_widget (MooGladeXML *xml,
|
|
Widget *widget,
|
|
GError **error);
|
|
static gboolean create_children (MooGladeXML *xml,
|
|
Widget *widget_node,
|
|
GtkWidget *widget,
|
|
GError **error);
|
|
static void pack_children (MooGladeXML *xml,
|
|
Widget *widget_node,
|
|
GtkWidget *widget);
|
|
static GtkWidget *moo_glade_xml_create_widget(MooGladeXML *xml,
|
|
Widget *node,
|
|
GtkWidget *widget,
|
|
GError **error);
|
|
|
|
static void dump_widget (Widget *widget_node);
|
|
|
|
|
|
static gboolean
|
|
moo_glade_xml_build (MooGladeXML *xml,
|
|
Widget *widget_node,
|
|
GtkWidget *widget,
|
|
GError **error)
|
|
{
|
|
if (widget)
|
|
{
|
|
if (!widget_node->type)
|
|
widget_node->type = G_OBJECT_TYPE (widget);
|
|
|
|
if (!g_type_is_a (G_OBJECT_TYPE (widget), widget_node->type))
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"incompatible types %s and %s",
|
|
g_type_name (G_OBJECT_TYPE (widget)),
|
|
g_type_name (widget_node->type));
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
widget = moo_glade_xml_create_widget (xml, widget_node, widget, error);
|
|
|
|
if (!widget || !moo_glade_xml_add_widget (xml, widget_node->id, widget, error))
|
|
goto error;
|
|
|
|
widget_node->widget = widget;
|
|
|
|
if (!create_children (xml, widget_node, widget, error))
|
|
goto error;
|
|
|
|
set_special_props (xml, widget, widget_node->props);
|
|
set_custom_props (xml, widget_node);
|
|
set_mnemonics (xml, widget_node);
|
|
set_default (xml, widget_node);
|
|
set_focus (xml, widget_node);
|
|
connect_signals (xml, widget_node);
|
|
|
|
return TRUE;
|
|
|
|
error:
|
|
gtk_widget_destroy (widget);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void
|
|
set_mnemonics (MooGladeXML *xml,
|
|
Widget *node)
|
|
{
|
|
GSList *l;
|
|
|
|
g_return_if_fail (node != NULL);
|
|
|
|
if (node->props->mask & PROP_MNEMONIC_WIDGET)
|
|
{
|
|
GtkWidget *mnemonic;
|
|
|
|
mnemonic = moo_glade_xml_get_widget (xml, node->props->mnemonic_widget);
|
|
|
|
if (!mnemonic)
|
|
g_warning ("%s: could not find widget '%s'",
|
|
G_STRLOC,
|
|
node->props->mnemonic_widget ?
|
|
node->props->mnemonic_widget : "NULL");
|
|
else if (!GTK_IS_LABEL (node->widget))
|
|
g_warning ("%s: mnemonic widget property specified for widget of class %s",
|
|
G_STRLOC, g_type_name (G_OBJECT_TYPE (node->widget)));
|
|
else
|
|
gtk_label_set_mnemonic_widget (GTK_LABEL (node->widget), mnemonic);
|
|
}
|
|
|
|
for (l = node->children; l != NULL; l = l->next)
|
|
{
|
|
Child *child = l->data;
|
|
|
|
if (!child->empty)
|
|
set_mnemonics (xml, child->widget);
|
|
}
|
|
}
|
|
|
|
|
|
static gboolean
|
|
set_default (MooGladeXML *xml,
|
|
Widget *node)
|
|
{
|
|
GSList *l;
|
|
|
|
g_return_val_if_fail (node != NULL, FALSE);
|
|
|
|
if ((node->props->mask & PROP_HAS_DEFAULT) &&
|
|
node->props->has_default)
|
|
{
|
|
gtk_widget_grab_default (node->widget);
|
|
return TRUE;
|
|
}
|
|
|
|
for (l = node->children; l != NULL; l = l->next)
|
|
{
|
|
Child *child = l->data;
|
|
|
|
if (!child->empty && set_default (xml, child->widget))
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
set_focus (MooGladeXML *xml,
|
|
Widget *node)
|
|
{
|
|
GSList *l;
|
|
|
|
g_return_val_if_fail (node != NULL, FALSE);
|
|
|
|
if ((node->props->mask & PROP_HAS_FOCUS) &&
|
|
node->props->has_focus)
|
|
{
|
|
gtk_widget_grab_focus (node->widget);
|
|
return TRUE;
|
|
}
|
|
|
|
for (l = node->children; l != NULL; l = l->next)
|
|
{
|
|
Child *child = l->data;
|
|
|
|
if (!child->empty && set_focus (xml, child->widget))
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void
|
|
set_moo_sensitive (MooGladeXML *xml,
|
|
GtkWidget *widget,
|
|
const char *value)
|
|
{
|
|
GtkWidget *btn;
|
|
gboolean invert = FALSE;
|
|
|
|
if (value[0] == '!')
|
|
{
|
|
value = value + 1;
|
|
invert = TRUE;
|
|
}
|
|
|
|
btn = moo_glade_xml_get_widget (xml, value);
|
|
g_return_if_fail (GTK_IS_TOGGLE_BUTTON (btn));
|
|
|
|
moo_bind_sensitive (btn, &widget, 1, invert);
|
|
}
|
|
|
|
|
|
static void
|
|
set_custom_props (MooGladeXML *xml,
|
|
Widget *node)
|
|
{
|
|
guint i;
|
|
|
|
if (!node->props->custom_props)
|
|
return;
|
|
|
|
for (i = 0; i < node->props->custom_props->len; i += 2)
|
|
{
|
|
const char *prop, *value;
|
|
|
|
prop = node->props->custom_props->pdata[i];
|
|
value = node->props->custom_props->pdata[i+1];
|
|
|
|
if (!strcmp (prop, "moo-sensitive"))
|
|
{
|
|
set_moo_sensitive (xml, node->widget, value);
|
|
}
|
|
else if (!xml->priv->prop_func ||
|
|
!xml->priv->prop_func (xml, node->id, node->widget,
|
|
prop, value, xml->priv->prop_func_data))
|
|
{
|
|
/* g_message ("%s: unknown property '%s'", G_STRLOC, prop); */
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static gboolean
|
|
connect_special_signal (MooGladeXML *xml,
|
|
Widget *node,
|
|
Signal *signal)
|
|
{
|
|
if (!strcmp (signal->name, "moo-sensitive"))
|
|
{
|
|
GtkWidget *btn;
|
|
gboolean invert = FALSE;
|
|
|
|
btn = moo_glade_xml_get_widget (xml, signal->handler);
|
|
g_return_val_if_fail (GTK_IS_TOGGLE_BUTTON (btn), FALSE);
|
|
|
|
if (signal->object)
|
|
{
|
|
if (!strcmp (signal->object, "invert"))
|
|
invert = TRUE;
|
|
else
|
|
g_warning ("%s: invalid string '%s'",
|
|
G_STRLOC, signal->object);
|
|
}
|
|
|
|
moo_bind_sensitive (btn, &node->widget, 1, invert);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void
|
|
connect_signals (MooGladeXML *xml,
|
|
Widget *node)
|
|
{
|
|
GSList *l;
|
|
|
|
g_return_if_fail (node != NULL);
|
|
|
|
for (l = node->signals; l != NULL; l = l->next)
|
|
{
|
|
Signal *signal = l->data;
|
|
gboolean connected = FALSE;
|
|
gpointer object = NULL;
|
|
|
|
if (!signal->name || !signal->name[0])
|
|
{
|
|
g_warning ("%s: empty signal name in widget %s",
|
|
G_STRLOC, node->id);
|
|
continue;
|
|
}
|
|
|
|
if (!signal->handler || !signal->handler[0])
|
|
{
|
|
g_warning ("%s: empty handler of signal '%s' in widget %s",
|
|
G_STRLOC, signal->name, node->id);
|
|
continue;
|
|
}
|
|
|
|
if (connect_special_signal (xml, node, signal))
|
|
continue;
|
|
|
|
if (xml->priv->signal_func)
|
|
{
|
|
connected = xml->priv->signal_func (xml, node->id,
|
|
node->widget,
|
|
signal->name,
|
|
signal->handler,
|
|
signal->object,
|
|
xml->priv->signal_func_data);
|
|
|
|
if (connected)
|
|
continue;
|
|
}
|
|
|
|
if (signal->object)
|
|
{
|
|
object = moo_glade_xml_get_widget (xml, signal->object);
|
|
|
|
if (!object)
|
|
{
|
|
g_warning ("%s: could not find object '%s' for signal '%s' of widget '%s'",
|
|
G_STRLOC, signal->object, signal->name, node->id);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
g_warning ("%s: unconnected signal '%s' of widget '%s'",
|
|
G_STRLOC, signal->name, node->id);
|
|
}
|
|
|
|
for (l = node->children; l != NULL; l = l->next)
|
|
{
|
|
Child *child = l->data;
|
|
|
|
if (child && child->widget)
|
|
connect_signals (xml, child->widget);
|
|
}
|
|
}
|
|
|
|
|
|
static GtkWidget*
|
|
create_widget (MooGladeXML *xml,
|
|
Widget *widget_node,
|
|
GError **error)
|
|
{
|
|
GtkWidget *widget;
|
|
|
|
widget = moo_glade_xml_create_widget (xml, widget_node, NULL, error);
|
|
|
|
if (!widget ||
|
|
!moo_glade_xml_add_widget (xml, widget_node->id, widget, error) ||
|
|
!create_children (xml, widget_node, widget, error))
|
|
{
|
|
gtk_widget_destroy (widget);
|
|
return NULL;
|
|
}
|
|
|
|
return widget;
|
|
}
|
|
|
|
|
|
/* XXX */
|
|
static void
|
|
set_special_props (MooGladeXML *xml,
|
|
GtkWidget *widget,
|
|
WidgetProps *props)
|
|
{
|
|
if (props->visible)
|
|
gtk_widget_show (widget);
|
|
|
|
if (props->mask & PROP_TOOLTIP)
|
|
_moo_widget_set_tooltip (widget, props->tooltip);
|
|
|
|
if (props->mask & PROP_ENABLE_TOOLTIPS)
|
|
{
|
|
if (GTK_IS_TOOLBAR (widget))
|
|
gtk_toolbar_set_tooltips (GTK_TOOLBAR (widget), TRUE);
|
|
else
|
|
g_warning ("%s: oops", G_STRLOC);
|
|
}
|
|
|
|
if (props->mask & PROP_HISTORY)
|
|
{
|
|
if (GTK_IS_OPTION_MENU (widget))
|
|
gtk_option_menu_set_history (GTK_OPTION_MENU (widget),
|
|
props->history);
|
|
else
|
|
g_warning ("%s: oops", G_STRLOC);
|
|
}
|
|
|
|
if (props->mask & PROP_RADIO_GROUP)
|
|
{
|
|
if (GTK_IS_RADIO_BUTTON (widget))
|
|
{
|
|
GtkWidget *group_button;
|
|
GSList *group;
|
|
|
|
group_button = moo_glade_xml_get_widget (xml, props->radio_group);
|
|
|
|
g_return_if_fail (GTK_IS_RADIO_BUTTON (group_button));
|
|
|
|
group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (group_button));
|
|
gtk_radio_button_set_group (GTK_RADIO_BUTTON (widget), group);
|
|
}
|
|
else if (GTK_IS_RADIO_MENU_ITEM (widget))
|
|
{
|
|
GtkWidget *group_item;
|
|
GSList *group;
|
|
|
|
group_item = moo_glade_xml_get_widget (xml, props->radio_group);
|
|
|
|
g_return_if_fail (GTK_IS_RADIO_MENU_ITEM (group_item));
|
|
|
|
group = gtk_radio_menu_item_get_group (GTK_RADIO_MENU_ITEM (group_item));
|
|
gtk_radio_menu_item_set_group (GTK_RADIO_MENU_ITEM (widget), group);
|
|
}
|
|
else
|
|
{
|
|
g_return_if_reached ();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static GtkWidget*
|
|
moo_glade_xml_create_widget (MooGladeXML *xml,
|
|
Widget *node,
|
|
GtkWidget *widget,
|
|
GError **error)
|
|
{
|
|
WidgetProps *props;
|
|
GType type = 0;
|
|
|
|
g_return_val_if_fail (node != NULL, NULL);
|
|
|
|
if (!widget)
|
|
{
|
|
FuncDataPair *pair;
|
|
pair = g_hash_table_lookup (xml->priv->id_to_func, node->id);
|
|
if (pair)
|
|
widget = pair->func (xml, node->id, pair->data);
|
|
}
|
|
|
|
if (!widget)
|
|
type = node->type;
|
|
|
|
if (!widget && !type)
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"coudl not create widget '%s'",
|
|
node->id);
|
|
return NULL;
|
|
}
|
|
|
|
props = node->props;
|
|
|
|
if (!widget)
|
|
{
|
|
if (props->mask & PROP_LABEL)
|
|
{
|
|
if (!props->label)
|
|
{
|
|
g_warning ("%s: oops", G_STRLOC);
|
|
}
|
|
if (type == GTK_TYPE_MENU_ITEM)
|
|
{
|
|
if (props->mask & PROP_USE_UNDERLINE)
|
|
widget = gtk_menu_item_new_with_mnemonic (props->label);
|
|
else
|
|
widget = gtk_menu_item_new_with_label (props->label);
|
|
}
|
|
else if (type == GTK_TYPE_IMAGE_MENU_ITEM)
|
|
{
|
|
if (props->mask & PROP_USE_STOCK)
|
|
widget = gtk_image_menu_item_new_from_stock (props->label, NULL);
|
|
else if (props->mask & PROP_USE_UNDERLINE)
|
|
widget = gtk_image_menu_item_new_with_mnemonic (props->label);
|
|
else
|
|
widget = gtk_image_menu_item_new_with_label (props->label);
|
|
}
|
|
else if (type == GTK_TYPE_RADIO_MENU_ITEM)
|
|
{
|
|
if (props->mask & PROP_USE_UNDERLINE)
|
|
widget = gtk_radio_menu_item_new_with_mnemonic (NULL, props->label);
|
|
else
|
|
widget = gtk_radio_menu_item_new_with_label (NULL, props->label);
|
|
}
|
|
else if (type == GTK_TYPE_CHECK_BUTTON)
|
|
{
|
|
if (props->mask & PROP_USE_UNDERLINE)
|
|
widget = gtk_check_button_new_with_mnemonic (props->label);
|
|
else
|
|
widget = gtk_check_button_new_with_label (props->label);
|
|
}
|
|
else if (type == GTK_TYPE_RADIO_BUTTON)
|
|
{
|
|
if (props->mask & PROP_USE_UNDERLINE)
|
|
widget = gtk_radio_button_new_with_mnemonic (NULL, props->label);
|
|
else
|
|
widget = gtk_radio_button_new_with_label (NULL, props->label);
|
|
}
|
|
else if (type == GTK_TYPE_LIST_ITEM)
|
|
{
|
|
widget = gtk_list_item_new_with_label (props->label);
|
|
}
|
|
else
|
|
{
|
|
g_warning ("%s: oops", G_STRLOC);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!widget)
|
|
{
|
|
widget = g_object_newv (type, props->n_params,
|
|
props->params);
|
|
}
|
|
else
|
|
{
|
|
guint i;
|
|
for (i = 0; i < props->n_params; i++)
|
|
g_object_set_property (G_OBJECT (widget),
|
|
props->params[i].name,
|
|
&props->params[i].value);
|
|
}
|
|
|
|
g_return_val_if_fail (widget != NULL, NULL);
|
|
|
|
return widget;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
create_children (MooGladeXML *xml,
|
|
Widget *widget_node,
|
|
GtkWidget *widget,
|
|
GError **error)
|
|
{
|
|
GSList *l;
|
|
|
|
for (l = widget_node->children; l != NULL; l = l->next)
|
|
{
|
|
Child *child = l->data;
|
|
if (!create_child (xml, widget, child, error))
|
|
return FALSE;
|
|
}
|
|
|
|
pack_children (xml, widget_node, widget);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
create_child (MooGladeXML *xml,
|
|
GtkWidget *parent,
|
|
Child *child,
|
|
GError **error)
|
|
{
|
|
GtkWidget *widget = NULL;
|
|
|
|
g_return_val_if_fail (child != NULL, FALSE);
|
|
g_return_val_if_fail (GTK_IS_CONTAINER (parent), FALSE);
|
|
|
|
if (child->empty)
|
|
return TRUE;
|
|
|
|
if (child->internal_child)
|
|
{
|
|
gboolean added = FALSE;
|
|
GtkWidget *real_parent;
|
|
|
|
real_parent = moo_glade_xml_get_widget (xml, child->internal_parent_id);
|
|
|
|
if (!real_parent)
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"could not find widget '%s'",
|
|
child->internal_parent_id);
|
|
return FALSE;
|
|
}
|
|
|
|
if (!strcmp (child->internal_child, "vbox") && GTK_IS_DIALOG (real_parent))
|
|
{
|
|
widget = GTK_DIALOG (real_parent)->vbox;
|
|
}
|
|
else if (!strcmp (child->internal_child, "action_area") && GTK_IS_DIALOG (real_parent))
|
|
{
|
|
widget = GTK_DIALOG (real_parent)->action_area;
|
|
}
|
|
else if (!strcmp (child->internal_child, "entry") && GTK_IS_COMBO_BOX_ENTRY (real_parent))
|
|
{
|
|
widget = GTK_BIN (real_parent)->child;
|
|
}
|
|
else if (!strcmp (child->internal_child, "entry") && GTK_IS_COMBO (real_parent))
|
|
{
|
|
widget = GTK_COMBO (real_parent)->entry;
|
|
}
|
|
else if (!strcmp (child->internal_child, "list") && GTK_IS_COMBO (real_parent))
|
|
{
|
|
widget = GTK_COMBO (real_parent)->list;
|
|
}
|
|
else if (!strcmp (child->internal_child, "image") && GTK_IS_IMAGE_MENU_ITEM (real_parent))
|
|
{
|
|
g_object_get (real_parent, "image", &widget, NULL);
|
|
|
|
if (!widget)
|
|
{
|
|
widget = create_widget (xml, child->widget, error);
|
|
|
|
if (!widget)
|
|
return FALSE;
|
|
|
|
gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (real_parent), widget);
|
|
added = TRUE;
|
|
}
|
|
}
|
|
|
|
if (!widget)
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"could not find internal child '%s' for widget '%s'",
|
|
child->internal_child, child->internal_parent_id);
|
|
return FALSE;
|
|
}
|
|
|
|
if (!added && !moo_glade_xml_add_widget (xml, child->widget->id, widget, error))
|
|
return FALSE;
|
|
|
|
if (child->widget->props->n_params)
|
|
{
|
|
guint i;
|
|
for (i = 0; i < child->widget->props->n_params; ++i)
|
|
g_object_set_property (G_OBJECT (widget),
|
|
child->widget->props->params[i].name,
|
|
&child->widget->props->params[i].value);
|
|
}
|
|
|
|
if (!create_children (xml, child->widget, widget, error))
|
|
return FALSE;
|
|
|
|
set_special_props (xml, widget, child->widget->props);
|
|
}
|
|
else
|
|
{
|
|
widget = create_widget (xml, child->widget, error);
|
|
|
|
if (!widget)
|
|
return FALSE;
|
|
}
|
|
|
|
child->widget->widget = widget;
|
|
set_custom_props (xml, child->widget);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void
|
|
pack_children (MooGladeXML *xml,
|
|
Widget *parent_node,
|
|
GtkWidget *parent_widget)
|
|
{
|
|
GSList *l;
|
|
|
|
g_return_if_fail (parent_node != NULL);
|
|
g_return_if_fail (parent_widget != NULL);
|
|
|
|
for (l = parent_node->children; l != NULL; l = l->next)
|
|
{
|
|
GtkWidget *widget;
|
|
Child *child = l->data;
|
|
gboolean packed = FALSE;
|
|
|
|
if (child->empty || child->internal_child)
|
|
continue;
|
|
|
|
widget = child->widget->widget;
|
|
g_return_if_fail (widget != NULL);
|
|
|
|
if (GTK_IS_FRAME (parent_widget) &&
|
|
(child->props->mask & PACK_PROP_LABEL_ITEM))
|
|
{
|
|
gtk_frame_set_label_widget (GTK_FRAME (parent_widget), widget);
|
|
packed = TRUE;
|
|
}
|
|
else if (GTK_IS_NOTEBOOK (parent_widget) &&
|
|
(child->props->mask & PACK_PROP_TAB))
|
|
{
|
|
int index = g_slist_index (parent_node->children, child);
|
|
|
|
if (index <= 0)
|
|
{
|
|
g_warning ("%s: oops", G_STRLOC);
|
|
}
|
|
else
|
|
{
|
|
Child *page_child = g_slist_nth_data (parent_node->children, index - 1);
|
|
|
|
if (!page_child->widget || !page_child->widget->widget)
|
|
g_message ("%s: empty notebook page with non-empty label", G_STRLOC);
|
|
else
|
|
gtk_notebook_set_tab_label (GTK_NOTEBOOK (parent_widget),
|
|
page_child->widget->widget,
|
|
widget);
|
|
|
|
packed = TRUE;
|
|
}
|
|
}
|
|
else if (GTK_IS_MENU_ITEM (parent_widget) && GTK_IS_MENU (widget))
|
|
{
|
|
gtk_menu_item_set_submenu (GTK_MENU_ITEM (parent_widget), widget);
|
|
packed = TRUE;
|
|
}
|
|
else if (GTK_IS_OPTION_MENU (parent_widget) && GTK_IS_MENU (widget))
|
|
{
|
|
gtk_option_menu_set_menu (GTK_OPTION_MENU (parent_widget), widget);
|
|
packed = TRUE;
|
|
}
|
|
else if (child->widget->props->mask & PROP_RESPONSE_ID)
|
|
{
|
|
Child *parent = child->parent_node->parent_node;
|
|
|
|
/* glade saves response_id with all buttons now, so do not complain
|
|
* if response_id found not in a dialog button */
|
|
if (parent && parent->internal_child &&
|
|
!strcmp (parent->internal_child, "action_area"))
|
|
{
|
|
GtkWidget *dialog = moo_glade_xml_get_widget (xml, parent->internal_parent_id);
|
|
|
|
if (!dialog || !GTK_IS_DIALOG (dialog))
|
|
{
|
|
g_warning ("%s: oops", G_STRLOC);
|
|
}
|
|
else
|
|
{
|
|
gtk_dialog_add_action_widget (GTK_DIALOG (dialog), widget,
|
|
child->widget->props->response_id);
|
|
packed = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!packed)
|
|
gtk_container_add (GTK_CONTAINER (parent_widget), widget);
|
|
|
|
if (child->props->n_params)
|
|
{
|
|
guint i;
|
|
for (i = 0; i < child->props->n_params; ++i)
|
|
gtk_container_child_set_property (GTK_CONTAINER (parent_widget),
|
|
widget,
|
|
child->props->params[i].name,
|
|
&child->props->params[i].value);
|
|
}
|
|
|
|
set_special_props (xml, widget, child->widget->props);
|
|
set_custom_props (xml, child->widget);
|
|
}
|
|
}
|
|
|
|
|
|
static Signal*
|
|
signal_new (const char *name,
|
|
const char *handler,
|
|
const char *object)
|
|
{
|
|
Signal *signal = g_new (Signal, 1);
|
|
signal->name = g_strdup (name);
|
|
signal->handler = g_strdup (handler);
|
|
signal->object = g_strdup (object);
|
|
|
|
if (signal->name)
|
|
g_strdelimit (signal->name, "-_", '-');
|
|
|
|
return signal;
|
|
}
|
|
|
|
|
|
static void
|
|
signal_free (Signal *signal)
|
|
{
|
|
if (signal)
|
|
{
|
|
g_free (signal->name);
|
|
g_free (signal->handler);
|
|
g_free (signal->object);
|
|
g_free (signal);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
collect_signals (MooMarkupNode *node,
|
|
Widget *widget)
|
|
{
|
|
g_return_if_fail (node != NULL && NODE_IS_WIDGET (node));
|
|
g_return_if_fail (widget != NULL);
|
|
|
|
FOREACH_ELM_START (node, elm)
|
|
{
|
|
if (NODE_IS_SIGNAL (elm))
|
|
{
|
|
const char *name, *handler, *object;
|
|
name = moo_markup_get_prop (elm, "name");
|
|
handler = moo_markup_get_prop (elm, "handler");
|
|
object = moo_markup_get_prop (elm, "object");
|
|
g_return_if_fail (name != NULL && handler != NULL);
|
|
widget->signals = g_slist_prepend (widget->signals,
|
|
signal_new (name, handler, object));
|
|
}
|
|
}
|
|
FOREACH_ELM_END;
|
|
|
|
widget->signals = g_slist_reverse (widget->signals);
|
|
}
|
|
|
|
|
|
static void
|
|
widget_free (Widget *widget)
|
|
{
|
|
if (widget)
|
|
{
|
|
g_free (widget->id);
|
|
widget_props_free (widget->props);
|
|
g_slist_foreach (widget->children, (GFunc) child_free, NULL);
|
|
g_slist_free (widget->children);
|
|
g_slist_foreach (widget->signals, (GFunc) signal_free, NULL);
|
|
g_slist_free (widget->signals);
|
|
g_free (widget);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
child_free (Child *child)
|
|
{
|
|
if (child)
|
|
{
|
|
g_free (child->internal_child);
|
|
g_free (child->internal_parent_id);
|
|
widget_free (child->widget);
|
|
packing_props_free (child->props);
|
|
g_free (child);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
widget_props_free (WidgetProps *props)
|
|
{
|
|
if (props)
|
|
{
|
|
if (props->params)
|
|
_moo_param_array_free (props->params, props->n_params);
|
|
|
|
if (props->custom_props)
|
|
{
|
|
g_ptr_array_foreach (props->custom_props, (GFunc) g_free, NULL);
|
|
g_ptr_array_free (props->custom_props, TRUE);
|
|
}
|
|
|
|
g_free (props->tooltip);
|
|
g_free (props->mnemonic_widget);
|
|
g_free (props->label);
|
|
g_free (props->radio_group);
|
|
g_free (props);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
packing_props_free (PackingProps *props)
|
|
{
|
|
if (props)
|
|
{
|
|
if (props->params)
|
|
_moo_param_array_free (props->params, props->n_params);
|
|
g_free (props);
|
|
}
|
|
}
|
|
|
|
|
|
static Widget*
|
|
widget_new (MooGladeXML *xml,
|
|
Child *parent,
|
|
MooMarkupNode *node,
|
|
GError **error)
|
|
{
|
|
Widget *widget;
|
|
WidgetProps *props;
|
|
const char *id, *class_name;
|
|
GType type;
|
|
gboolean ignore_errors = FALSE;
|
|
|
|
g_return_val_if_fail (NODE_IS_WIDGET (node), NULL);
|
|
|
|
id = moo_markup_get_prop (node, "id");
|
|
|
|
if (g_hash_table_lookup (xml->priv->id_to_func, id))
|
|
{
|
|
type = GTK_TYPE_WIDGET;
|
|
}
|
|
else
|
|
{
|
|
type = GPOINTER_TO_SIZE (g_hash_table_lookup (xml->priv->id_to_type, id));
|
|
|
|
if (!type)
|
|
{
|
|
class_name = moo_markup_get_prop (node, "class");
|
|
g_return_val_if_fail (id != NULL && class_name != NULL, NULL);
|
|
|
|
type = GPOINTER_TO_SIZE (g_hash_table_lookup (xml->priv->class_to_type, class_name));
|
|
|
|
if (!type)
|
|
{
|
|
ignore_errors = FALSE;
|
|
type = get_type_by_name (class_name);
|
|
}
|
|
}
|
|
|
|
if (!type)
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"could not find type of class '%s'",
|
|
class_name);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
props = widget_props_new (node, type,
|
|
g_hash_table_lookup (xml->priv->props, id),
|
|
ignore_errors, xml->priv->translation_domain,
|
|
error);
|
|
g_return_val_if_fail (props != NULL, NULL);
|
|
|
|
widget = g_new0 (Widget, 1);
|
|
widget->parent_node = parent;
|
|
widget->id = g_strdup (id);
|
|
widget->type = type;
|
|
widget->props = props;
|
|
widget->children = NULL;
|
|
|
|
collect_signals (node, widget);
|
|
|
|
FOREACH_ELM_START (node, elm)
|
|
{
|
|
if (NODE_IS_CHILD (elm))
|
|
{
|
|
Child *child = child_new (xml, widget, elm, error);
|
|
|
|
if (!child)
|
|
{
|
|
widget_free (widget);
|
|
g_return_val_if_fail (child != NULL, NULL);
|
|
}
|
|
|
|
widget->children = g_slist_prepend (widget->children, child);
|
|
}
|
|
}
|
|
FOREACH_ELM_END;
|
|
|
|
widget->children = g_slist_reverse (widget->children);
|
|
return widget;
|
|
}
|
|
|
|
|
|
static Child*
|
|
child_new (MooGladeXML *xml,
|
|
Widget *parent,
|
|
MooMarkupNode *node,
|
|
GError **error)
|
|
{
|
|
Child *child;
|
|
PackingProps *props;
|
|
const char *class_name, *internal_child, *internal_parent_id = NULL;
|
|
MooMarkupNode *widget_node, *packing_node;
|
|
|
|
g_return_val_if_fail (parent != NULL, NULL);
|
|
g_return_val_if_fail (NODE_IS_CHILD (node), NULL);
|
|
|
|
if (moo_markup_get_element (node, "placeholder"))
|
|
{
|
|
child = g_new0 (Child, 1);
|
|
child->empty = TRUE;
|
|
child->parent_node = parent;
|
|
return child;
|
|
}
|
|
|
|
widget_node = moo_markup_get_element (node, "widget");
|
|
|
|
if (!widget_node)
|
|
{
|
|
char *text = moo_markup_node_get_string (node);
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"no widget element in '%s'",
|
|
text);
|
|
g_free (text);
|
|
return NULL;
|
|
}
|
|
|
|
class_name = moo_markup_get_prop (widget_node, "class");
|
|
|
|
if (!class_name)
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"no class attribute in widget element");
|
|
return NULL;
|
|
}
|
|
|
|
#if 0
|
|
if (!strcmp (class_name, "GtkToolbar"))
|
|
{
|
|
g_warning ("Ignoring %s", class_name);
|
|
child = g_new0 (Child, 1);
|
|
child->empty = TRUE;
|
|
child->parent_node = parent;
|
|
return child;
|
|
}
|
|
#endif
|
|
|
|
internal_child = moo_markup_get_prop (node, "internal-child");
|
|
|
|
if (internal_child)
|
|
{
|
|
GType parent_types[2] = {0, 0};
|
|
guint n_parent_types = 1;
|
|
Widget *real_parent;
|
|
|
|
if (!strcmp (internal_child, "vbox") ||
|
|
!strcmp (internal_child, "action_area"))
|
|
{
|
|
parent_types[0] = GTK_TYPE_DIALOG;
|
|
}
|
|
else if (!strcmp (internal_child, "entry"))
|
|
{
|
|
parent_types[0] = GTK_TYPE_COMBO;
|
|
parent_types[1] = GTK_TYPE_COMBO_BOX_ENTRY;
|
|
n_parent_types = 2;
|
|
}
|
|
else if (!strcmp (internal_child, "list"))
|
|
{
|
|
parent_types[0] = GTK_TYPE_COMBO;
|
|
}
|
|
else if (!strcmp (internal_child, "image"))
|
|
{
|
|
parent_types[0] = GTK_TYPE_IMAGE_MENU_ITEM;
|
|
}
|
|
else
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"unknown internal child name '%s'",
|
|
internal_child);
|
|
return NULL;
|
|
}
|
|
|
|
real_parent = parent;
|
|
|
|
while (TRUE)
|
|
{
|
|
guint i;
|
|
GType pt = 0;
|
|
|
|
for (i = 0; i < n_parent_types && pt == 0; ++i)
|
|
if (g_type_is_a (real_parent->type, parent_types[i]))
|
|
pt = parent_types[i];
|
|
|
|
if (pt == 0)
|
|
{
|
|
if (!real_parent->parent_node)
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"could not find parent of '%s'",
|
|
internal_child);
|
|
return NULL;
|
|
}
|
|
else
|
|
{
|
|
real_parent = real_parent->parent_node->parent_node;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
internal_parent_id = real_parent->id;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
packing_node = moo_markup_get_element (node, "packing");
|
|
|
|
if (packing_node)
|
|
props = packing_props_new (packing_node, parent->type, error);
|
|
else
|
|
props = packing_props_new (NULL, 0, error);
|
|
|
|
if (!props)
|
|
return NULL;
|
|
|
|
child = g_new0 (Child, 1);
|
|
child->parent_node = parent;
|
|
child->internal_child = g_strdup (internal_child);
|
|
child->internal_parent_id = g_strdup (internal_parent_id);
|
|
child->props = props;
|
|
|
|
child->widget = widget_new (xml, child, widget_node, error);
|
|
|
|
if (!child->widget)
|
|
{
|
|
child_free (child);
|
|
return NULL;
|
|
}
|
|
|
|
return child;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
widget_props_add (WidgetProps *props,
|
|
GArray *params,
|
|
GObjectClass *klass,
|
|
const char *name,
|
|
const char *value,
|
|
gboolean ignore_errors,
|
|
GError **error)
|
|
{
|
|
GParameter param = {NULL, {0, {{0}, {0}}}};
|
|
gboolean special = TRUE;
|
|
|
|
if (!strcmp (name, "visible"))
|
|
{
|
|
props->visible = parse_bool (value);
|
|
}
|
|
else if (!strcmp (name, "response_id"))
|
|
{
|
|
props->response_id = parse_int (value);
|
|
props->mask |= PROP_RESPONSE_ID;
|
|
}
|
|
else if (!strcmp (name, "has_default"))
|
|
{
|
|
props->has_default = parse_bool (value);
|
|
props->mask |= PROP_HAS_DEFAULT;
|
|
}
|
|
else if (!strcmp (name, "has_focus"))
|
|
{
|
|
props->has_focus = parse_bool (value);
|
|
props->mask |= PROP_HAS_FOCUS;
|
|
}
|
|
else if (!strcmp (name, "tooltip"))
|
|
{
|
|
props->tooltip = g_strdup (value);
|
|
props->mask |= PROP_TOOLTIP;
|
|
}
|
|
else if (!strcmp (name, "mnemonic_widget") &&
|
|
GTK_IS_LABEL_CLASS (klass))
|
|
{
|
|
props->mnemonic_widget = g_strdup (value);
|
|
props->mask |= PROP_MNEMONIC_WIDGET;
|
|
}
|
|
else if (!strcmp (name, "text") &&
|
|
GTK_IS_TEXT_VIEW_CLASS (klass))
|
|
{
|
|
if (value && value[0])
|
|
g_message ("%s: ignoring TextView text property", G_STRLOC);
|
|
}
|
|
else if (!strcmp (name, "group") &&
|
|
(GTK_IS_RADIO_BUTTON_CLASS (klass) ||
|
|
GTK_IS_RADIO_MENU_ITEM_CLASS (klass)))
|
|
{
|
|
props->radio_group = g_strdup (value);
|
|
props->mask |= PROP_RADIO_GROUP;
|
|
}
|
|
else if (!strcmp (name, "label") &&
|
|
(GTK_IS_MENU_ITEM_CLASS (klass) ||
|
|
GTK_IS_CHECK_BUTTON_CLASS (klass) ||
|
|
GTK_IS_LIST_ITEM_CLASS (klass)))
|
|
{
|
|
props->label = g_strdup (value);
|
|
props->mask |= PROP_LABEL;
|
|
}
|
|
else if (!strcmp (name, "use_underline") &&
|
|
GTK_IS_MENU_ITEM_CLASS (klass))
|
|
{
|
|
props->mask |= PROP_USE_UNDERLINE;
|
|
}
|
|
else if (!strcmp (name, "use_stock") &&
|
|
GTK_IS_IMAGE_MENU_ITEM_CLASS (klass))
|
|
{
|
|
props->mask |= PROP_USE_STOCK;
|
|
}
|
|
else if (!strcmp (name, "tooltips") &&
|
|
GTK_IS_TOOLBAR_CLASS (klass))
|
|
{
|
|
props->mask |= PROP_ENABLE_TOOLTIPS;
|
|
}
|
|
else if (!strcmp (name, "history") &&
|
|
GTK_IS_OPTION_MENU_CLASS (klass))
|
|
{
|
|
props->mask |= PROP_HISTORY;
|
|
props->history = parse_int (value);
|
|
}
|
|
else if (!strcmp (name, "items") &&
|
|
GTK_IS_COMBO_BOX_CLASS (klass))
|
|
{
|
|
#if 0
|
|
// if (value && value[0])
|
|
// g_message ("%s: ignoring ComboBox items property", G_STRLOC);
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
special = FALSE;
|
|
}
|
|
|
|
if (!special)
|
|
{
|
|
GParamSpec *param_spec = g_object_class_find_property (klass, name);
|
|
|
|
if (!param_spec)
|
|
{
|
|
if (!props->custom_props)
|
|
props->custom_props = g_ptr_array_new ();
|
|
g_ptr_array_add (props->custom_props, g_strdelimit (g_strdup (name), "_", '-'));
|
|
g_ptr_array_add (props->custom_props, g_strdup (value));
|
|
}
|
|
else if (parse_property (param_spec, value, ¶m))
|
|
{
|
|
g_array_append_val (params, param);
|
|
}
|
|
else if (!ignore_errors)
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"could not convert '%s' to property '%s'",
|
|
value, name);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void
|
|
widget_props_add_one (const char *name,
|
|
const char *value,
|
|
gpointer user_data)
|
|
{
|
|
struct {
|
|
WidgetProps *props;
|
|
GArray *params;
|
|
GObjectClass *klass;
|
|
gboolean result;
|
|
gboolean ignore_errors;
|
|
GError **error;
|
|
} *data = user_data;
|
|
|
|
if (!data->result)
|
|
return;
|
|
|
|
data->result = widget_props_add (data->props, data->params,
|
|
data->klass, name, value,
|
|
data->ignore_errors,
|
|
data->error);
|
|
}
|
|
|
|
|
|
static const char *
|
|
get_property_text (MooMarkupNode *node,
|
|
const char *translation_domain)
|
|
{
|
|
gboolean translatable = FALSE, strip_context = FALSE;
|
|
const char *prop, *translatable_attr;
|
|
|
|
prop = moo_markup_get_content (node);
|
|
|
|
if (!prop || !prop[0])
|
|
return prop;
|
|
|
|
translatable_attr = moo_markup_get_prop (node, "translatable");
|
|
|
|
if (translatable_attr && !strcmp (translatable_attr, "yes"))
|
|
{
|
|
const char *ctx_attr = moo_markup_get_prop (node, "context");
|
|
|
|
if (ctx_attr && !strcmp (ctx_attr, "yes"))
|
|
strip_context = TRUE;
|
|
|
|
translatable = TRUE;
|
|
}
|
|
|
|
if (!translatable)
|
|
return prop;
|
|
|
|
if (translation_domain)
|
|
{
|
|
const char *translated;
|
|
|
|
if (strip_context)
|
|
translated = g_strip_context (prop, dgettext (translation_domain, prop));
|
|
else
|
|
translated = dgettext (translation_domain, prop);
|
|
|
|
if (strcmp (translated, prop) != 0)
|
|
return translated;
|
|
}
|
|
|
|
if (strip_context)
|
|
return g_strip_context (prop, gettext (prop));
|
|
else
|
|
return gettext (prop);
|
|
}
|
|
|
|
|
|
static WidgetProps*
|
|
widget_props_new (MooMarkupNode *node,
|
|
GType type,
|
|
GHashTable *add_props,
|
|
gboolean ignore_errors,
|
|
const char *translation_domain,
|
|
GError **error)
|
|
{
|
|
GArray *params;
|
|
GObjectClass *klass;
|
|
WidgetProps *props;
|
|
|
|
g_return_val_if_fail (node != NULL, NULL);
|
|
|
|
klass = g_type_class_ref (type);
|
|
g_return_val_if_fail (klass != NULL, NULL);
|
|
|
|
props = g_new0 (WidgetProps, 1);
|
|
params = g_array_new (FALSE, FALSE, sizeof (GParameter));
|
|
|
|
props->visible = FALSE;
|
|
|
|
FOREACH_ELM_START(node, elm)
|
|
{
|
|
if (NODE_IS_PROPERTY (elm))
|
|
{
|
|
const char *name = moo_markup_get_prop (elm, "name");
|
|
const char *value = get_property_text (elm, translation_domain);
|
|
gboolean result;
|
|
|
|
result = widget_props_add (props, params, klass,
|
|
name, value,
|
|
ignore_errors,
|
|
error);
|
|
|
|
if (!result)
|
|
goto error;
|
|
}
|
|
}
|
|
FOREACH_ELM_END;
|
|
|
|
if (add_props)
|
|
{
|
|
struct {
|
|
WidgetProps *props;
|
|
GArray *params;
|
|
GObjectClass *klass;
|
|
gboolean result;
|
|
gboolean ignore_errors;
|
|
GError **error;
|
|
} data;
|
|
|
|
data.props = props;
|
|
data.params = params;
|
|
data.klass = klass;
|
|
data.result = TRUE;
|
|
data.ignore_errors = ignore_errors;
|
|
data.error = error;
|
|
g_hash_table_foreach (add_props, (GHFunc) widget_props_add_one, &data);
|
|
|
|
if (!data.result)
|
|
goto error;
|
|
}
|
|
|
|
props->n_params = params->len;
|
|
props->params = (GParameter*) g_array_free (params, FALSE);
|
|
g_type_class_unref (klass);
|
|
return props;
|
|
|
|
error:
|
|
_moo_param_array_free ((GParameter*) params->data, params->len);
|
|
g_array_free (params, FALSE);
|
|
g_type_class_unref (klass);
|
|
widget_props_free (props);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static PackingProps *
|
|
packing_props_new (MooMarkupNode *node,
|
|
GType container_type,
|
|
GError **error)
|
|
{
|
|
GArray *params;
|
|
GObjectClass *klass;
|
|
PackingProps *props;
|
|
|
|
if (!node && !container_type)
|
|
return g_new0 (PackingProps, 1);
|
|
|
|
g_return_val_if_fail (node != NULL, NULL);
|
|
|
|
klass = g_type_class_ref (container_type);
|
|
g_return_val_if_fail (klass != NULL, NULL);
|
|
|
|
props = g_new0 (PackingProps, 1);
|
|
params = g_array_new (FALSE, FALSE, sizeof (GParameter));
|
|
|
|
FOREACH_ELM_START(node, elm)
|
|
{
|
|
if (NODE_IS_PROPERTY (elm))
|
|
{
|
|
GParameter param = {NULL, {0, {{0}, {0}}}};
|
|
const char *name = moo_markup_get_prop (elm, "name");
|
|
const char *value = moo_markup_get_content (elm);
|
|
gboolean special = FALSE;
|
|
|
|
if (!strcmp (name, "type"))
|
|
{
|
|
if (!strcmp (value, "label_item") && GTK_IS_FRAME_CLASS (klass))
|
|
{
|
|
special = TRUE;
|
|
props->mask |= PACK_PROP_LABEL_ITEM;
|
|
}
|
|
else if (!strcmp (value, "tab") && GTK_IS_NOTEBOOK_CLASS (klass))
|
|
{
|
|
special = TRUE;
|
|
props->mask |= PACK_PROP_TAB;
|
|
}
|
|
}
|
|
|
|
if (!special)
|
|
{
|
|
GParamSpec *param_spec = gtk_container_class_find_child_property (klass, name);
|
|
|
|
if (!param_spec)
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"could not find property '%s'",
|
|
name);
|
|
_moo_param_array_free ((GParameter*) params->data, params->len);
|
|
g_array_free (params, FALSE);
|
|
g_type_class_unref (klass);
|
|
packing_props_free (props);
|
|
return NULL;
|
|
}
|
|
|
|
if (parse_property (param_spec, value, ¶m))
|
|
{
|
|
g_array_append_val (params, param);
|
|
}
|
|
else
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"could not convert '%s' to property '%s'",
|
|
value, name);
|
|
_moo_param_array_free ((GParameter*) params->data, params->len);
|
|
g_array_free (params, FALSE);
|
|
g_type_class_unref (klass);
|
|
packing_props_free (props);
|
|
return NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
FOREACH_ELM_END;
|
|
|
|
props->n_params = params->len;
|
|
props->params = (GParameter*) g_array_free (params, FALSE);
|
|
g_type_class_unref (klass);
|
|
return props;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
parse_property (GParamSpec *param_spec,
|
|
const char *value,
|
|
GParameter *param)
|
|
{
|
|
gboolean take_null = FALSE;
|
|
|
|
g_return_val_if_fail (param_spec != NULL, FALSE);
|
|
|
|
if (param_spec->value_type == GTK_TYPE_ADJUSTMENT)
|
|
take_null = TRUE;
|
|
|
|
if (!value && !take_null)
|
|
{
|
|
g_warning ("NULL value for property '%s' of type '%s'",
|
|
param_spec->name,
|
|
g_type_name (param_spec->value_type));
|
|
return FALSE;
|
|
}
|
|
|
|
g_value_init (¶m->value, param_spec->value_type);
|
|
|
|
errno = 0;
|
|
|
|
if (param_spec->value_type == G_TYPE_CHAR)
|
|
{
|
|
g_value_set_char (¶m->value, value[0]);
|
|
}
|
|
else if (param_spec->value_type == G_TYPE_UCHAR)
|
|
{
|
|
g_value_set_uchar (¶m->value, value[0]);
|
|
}
|
|
else if (param_spec->value_type == G_TYPE_BOOLEAN)
|
|
{
|
|
g_value_set_boolean (¶m->value, parse_bool (value));
|
|
}
|
|
else if (param_spec->value_type == G_TYPE_INT ||
|
|
param_spec->value_type == G_TYPE_LONG ||
|
|
param_spec->value_type == G_TYPE_INT64) /* XXX */
|
|
{
|
|
long val = strtol (value, NULL, 0);
|
|
|
|
if (errno)
|
|
{
|
|
g_warning ("%s: could not convert string '%s' to an int",
|
|
G_STRLOC, value);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
if (param_spec->value_type == G_TYPE_INT)
|
|
g_value_set_int (¶m->value, val);
|
|
else if (param_spec->value_type == G_TYPE_LONG)
|
|
g_value_set_long (¶m->value, val);
|
|
else
|
|
g_value_set_int64 (¶m->value, val);
|
|
}
|
|
}
|
|
else if (param_spec->value_type == G_TYPE_UINT ||
|
|
param_spec->value_type == G_TYPE_ULONG ||
|
|
param_spec->value_type == G_TYPE_UINT64) /* XXX */
|
|
{
|
|
guint64 val = g_ascii_strtoull (value, NULL, 0);
|
|
|
|
if (errno)
|
|
{
|
|
g_warning ("%s: could not convert string '%s' to a guint",
|
|
G_STRLOC, value);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
if (param_spec->value_type == G_TYPE_UINT)
|
|
g_value_set_uint (¶m->value, val);
|
|
else if (param_spec->value_type == G_TYPE_ULONG)
|
|
g_value_set_ulong (¶m->value, val);
|
|
else
|
|
g_value_set_uint64 (¶m->value, val);
|
|
}
|
|
}
|
|
else if (param_spec->value_type == G_TYPE_FLOAT ||
|
|
param_spec->value_type == G_TYPE_DOUBLE) /* XXX */
|
|
{
|
|
double val = g_ascii_strtod (value, NULL);
|
|
|
|
if (errno)
|
|
{
|
|
g_warning ("%s: could not convert string '%s' to double",
|
|
G_STRLOC, value);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
if (param_spec->value_type == G_TYPE_FLOAT)
|
|
g_value_set_float (¶m->value, val);
|
|
else
|
|
g_value_set_double (¶m->value, val);
|
|
}
|
|
}
|
|
else if (param_spec->value_type == G_TYPE_STRING)
|
|
{
|
|
g_value_set_string (¶m->value, value);
|
|
}
|
|
else if (G_TYPE_IS_ENUM (param_spec->value_type))
|
|
{
|
|
GEnumClass *enum_class;
|
|
GEnumValue *val;
|
|
|
|
enum_class = g_type_class_ref (param_spec->value_type);
|
|
|
|
val = g_enum_get_value_by_name (enum_class, value);
|
|
|
|
if (!val)
|
|
val = g_enum_get_value_by_nick (enum_class, value);
|
|
|
|
g_type_class_unref (enum_class);
|
|
|
|
if (!val)
|
|
{
|
|
const char *typename = g_type_name (param_spec->value_type);
|
|
g_warning ("%s: can not convert string '%s' to a value of type %s",
|
|
G_STRLOC, value, typename ? typename : "<unknown>");
|
|
return FALSE;
|
|
}
|
|
|
|
g_value_set_enum (¶m->value, val->value);
|
|
}
|
|
else if (G_TYPE_IS_FLAGS (param_spec->value_type))
|
|
{
|
|
if (!value[0])
|
|
{
|
|
g_value_set_flags (¶m->value, 0);
|
|
}
|
|
else
|
|
{
|
|
GFlagsClass *flags_class;
|
|
char **pieces;
|
|
guint i, len;
|
|
int result = 0;
|
|
|
|
pieces = g_strsplit (value, "|", 0);
|
|
len = g_strv_length (pieces);
|
|
|
|
flags_class = g_type_class_ref (param_spec->value_type);
|
|
|
|
for (i = 0; i < len; ++i)
|
|
{
|
|
GFlagsValue *val;
|
|
|
|
val = g_flags_get_value_by_name (flags_class, pieces[i]);
|
|
|
|
if (!val)
|
|
val = g_flags_get_value_by_nick (flags_class, pieces[i]);
|
|
|
|
if (!val)
|
|
{
|
|
const char *typename = g_type_name (param_spec->value_type);
|
|
g_warning ("%s: can not convert string '%s' to a value of type %s",
|
|
G_STRLOC, pieces[i], typename ? typename : "<unknown>");
|
|
g_type_class_unref (flags_class);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
result |= val->value;
|
|
}
|
|
}
|
|
|
|
g_value_set_flags (¶m->value, result);
|
|
g_type_class_unref (flags_class);
|
|
g_strfreev (pieces);
|
|
}
|
|
}
|
|
else if (param_spec->value_type == GDK_TYPE_PIXBUF)
|
|
{
|
|
GdkPixbuf *pixbuf = gdk_pixbuf_new_from_file (value, NULL);
|
|
|
|
if (!pixbuf)
|
|
{
|
|
g_message ("%s: could not find %s", G_STRLOC, value);
|
|
g_value_set_object (¶m->value, NULL);
|
|
}
|
|
else
|
|
{
|
|
g_value_set_object (¶m->value, pixbuf);
|
|
g_object_unref (pixbuf);
|
|
}
|
|
}
|
|
else if (param_spec->value_type == GTK_TYPE_ADJUSTMENT)
|
|
{
|
|
GtkObject *adjustment = parse_adjustment (value);
|
|
|
|
if (!adjustment)
|
|
{
|
|
const char *typename = g_type_name (param_spec->value_type);
|
|
g_warning ("%s: could not convert string '%s' to a value of type %s",
|
|
G_STRLOC, value, typename ? typename : "<unknown>");
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
MOO_OBJECT_REF_SINK (adjustment);
|
|
g_value_set_object (¶m->value, adjustment);
|
|
g_object_unref (adjustment);
|
|
}
|
|
}
|
|
else if (param_spec->value_type == G_TYPE_STRV)
|
|
{
|
|
char **strv = moo_splitlines (value);
|
|
g_value_take_boxed (¶m->value, strv);
|
|
}
|
|
else
|
|
{
|
|
const char *typename = g_type_name (param_spec->value_type);
|
|
g_warning ("%s: could not convert string '%s' to a value of type %s",
|
|
G_STRLOC, value, typename ? typename : "<unknown>");
|
|
return FALSE;
|
|
}
|
|
|
|
param->name = g_strdup (param_spec->name);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
parse_bool (const char *value)
|
|
{
|
|
g_return_val_if_fail (value != NULL, FALSE);
|
|
return !g_ascii_strcasecmp (value, "true");
|
|
}
|
|
|
|
|
|
static int
|
|
parse_int (const char *value)
|
|
{
|
|
long val;
|
|
|
|
g_return_val_if_fail (value != NULL, 0);
|
|
|
|
errno = 0;
|
|
val = strtol (value, NULL, 0);
|
|
|
|
if (errno)
|
|
{
|
|
g_warning ("%s: could not convert string '%s' to an int",
|
|
G_STRLOC, value);
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return val;
|
|
}
|
|
}
|
|
|
|
|
|
static GtkObject*
|
|
parse_adjustment (const char *value)
|
|
{
|
|
char **pieces;
|
|
GtkObject *adj = NULL;
|
|
double vals[6];
|
|
guint i;
|
|
|
|
/* XXX is this correct? */
|
|
if (!value)
|
|
return g_object_new (GTK_TYPE_ADJUSTMENT, NULL);
|
|
|
|
pieces = g_strsplit (value, " ", 0);
|
|
g_return_val_if_fail (pieces != NULL, NULL);
|
|
|
|
if (g_strv_length (pieces) != 6)
|
|
goto out;
|
|
|
|
errno = 0;
|
|
|
|
for (i = 0; i < 6; ++i)
|
|
{
|
|
vals[i] = g_ascii_strtod (pieces[i], NULL);
|
|
|
|
if (errno)
|
|
{
|
|
g_warning ("could not convert '%s' to double", pieces[i]);
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
adj = gtk_adjustment_new (vals[0], vals[1], vals[2],
|
|
vals[3], vals[4], vals[5]);
|
|
|
|
out:
|
|
g_strfreev (pieces);
|
|
return adj;
|
|
}
|
|
|
|
|
|
static void
|
|
moo_glade_xml_class_init (MooGladeXMLClass *klass)
|
|
{
|
|
G_OBJECT_CLASS(klass)->dispose = moo_glade_xml_dispose;
|
|
}
|
|
|
|
|
|
static void
|
|
moo_glade_xml_init (MooGladeXML *xml)
|
|
{
|
|
xml->priv = g_new0 (MooGladeXMLPrivate, 1);
|
|
|
|
xml->priv->widgets = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
g_free, NULL);
|
|
xml->priv->class_to_type = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
g_free, NULL);
|
|
xml->priv->id_to_type = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
g_free, NULL);
|
|
xml->priv->id_to_func = g_hash_table_new_full (g_str_hash, g_str_equal, g_free,
|
|
(GDestroyNotify) func_data_pair_free);
|
|
xml->priv->props = g_hash_table_new_full (g_str_hash, g_str_equal, g_free,
|
|
(GDestroyNotify) g_hash_table_destroy);
|
|
}
|
|
|
|
|
|
static char *
|
|
normalize_prop_name (const char *prop_name)
|
|
{
|
|
char *norm, *p;
|
|
|
|
norm = g_strdup (prop_name);
|
|
p = norm;
|
|
|
|
while ((p = strchr (p, '_')))
|
|
*p = '-';
|
|
|
|
return norm;
|
|
}
|
|
|
|
|
|
void
|
|
moo_glade_xml_set_property (MooGladeXML *xml,
|
|
const char *widget,
|
|
const char *prop_name,
|
|
const char *value)
|
|
{
|
|
GHashTable *props;
|
|
char *norm_prop_name;
|
|
|
|
g_return_if_fail (MOO_IS_GLADE_XML (xml));
|
|
g_return_if_fail (widget != NULL && prop_name != NULL);
|
|
|
|
props = g_hash_table_lookup (xml->priv->props, widget);
|
|
|
|
if (!props && !value)
|
|
return;
|
|
|
|
norm_prop_name = normalize_prop_name (prop_name);
|
|
|
|
if (!props)
|
|
{
|
|
props = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
g_free, g_free);
|
|
g_hash_table_insert (xml->priv->props, g_strdup (widget), props);
|
|
}
|
|
|
|
if (value)
|
|
{
|
|
g_hash_table_insert (props, norm_prop_name, g_strdup (value));
|
|
}
|
|
else
|
|
{
|
|
g_hash_table_remove (props, norm_prop_name);
|
|
g_free (norm_prop_name);
|
|
}
|
|
}
|
|
|
|
|
|
MooGladeXML *
|
|
moo_glade_xml_new_empty (const char *domain)
|
|
{
|
|
MooGladeXML *xml = g_object_new (MOO_TYPE_GLADE_XML, NULL);
|
|
xml->priv->translation_domain = g_strdup (domain);
|
|
return xml;
|
|
}
|
|
|
|
|
|
gboolean
|
|
moo_glade_xml_parse_file (MooGladeXML *xml,
|
|
const char *file,
|
|
const char *root,
|
|
GError **error)
|
|
{
|
|
MooMarkupDoc *doc;
|
|
gboolean result;
|
|
|
|
g_return_val_if_fail (xml != NULL, FALSE);
|
|
g_return_val_if_fail (file != NULL, FALSE);
|
|
|
|
doc = moo_markup_parse_file (file, error);
|
|
|
|
if (!doc)
|
|
return FALSE;
|
|
|
|
result = moo_glade_xml_parse_markup (xml, doc, root, NULL, error);
|
|
|
|
moo_markup_doc_unref (doc);
|
|
return result;
|
|
}
|
|
|
|
|
|
gboolean
|
|
moo_glade_xml_parse_memory (MooGladeXML *xml,
|
|
const char *buffer,
|
|
int size,
|
|
const char *root,
|
|
GError **error)
|
|
{
|
|
MooMarkupDoc *doc;
|
|
gboolean result;
|
|
GError *error_here = NULL;
|
|
|
|
g_return_val_if_fail (xml != NULL, FALSE);
|
|
g_return_val_if_fail (buffer != NULL, FALSE);
|
|
|
|
doc = moo_markup_parse_memory (buffer, size, &error_here);
|
|
|
|
if (!doc)
|
|
{
|
|
if (error)
|
|
{
|
|
g_propagate_error (error, error_here);
|
|
}
|
|
else
|
|
{
|
|
g_warning ("could not parse markup: %s",
|
|
error_here->message);
|
|
g_error_free (error_here);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
result = moo_glade_xml_parse_markup (xml, doc, root, NULL, &error_here);
|
|
|
|
if (!result)
|
|
{
|
|
if (error)
|
|
{
|
|
g_propagate_error (error, error_here);
|
|
}
|
|
else
|
|
{
|
|
g_warning ("%s: %s", G_STRLOC, error_here->message);
|
|
g_error_free (error_here);
|
|
}
|
|
}
|
|
|
|
moo_markup_doc_unref (doc);
|
|
return result;
|
|
}
|
|
|
|
|
|
MooGladeXML *
|
|
moo_glade_xml_new (const char *file,
|
|
const char *root,
|
|
const char *domain,
|
|
GError **error)
|
|
{
|
|
MooGladeXML *xml;
|
|
|
|
g_return_val_if_fail (file != NULL, NULL);
|
|
|
|
xml = moo_glade_xml_new_empty (domain);
|
|
|
|
if (!moo_glade_xml_parse_file (xml, file, root, error))
|
|
{
|
|
g_object_unref (xml);
|
|
return NULL;
|
|
}
|
|
|
|
return xml;
|
|
}
|
|
|
|
|
|
MooGladeXML*
|
|
moo_glade_xml_new_from_buf (const char *buffer,
|
|
int size,
|
|
const char *root,
|
|
const char *domain,
|
|
GError **error)
|
|
{
|
|
MooGladeXML *xml;
|
|
GError *error_here = NULL;
|
|
|
|
g_return_val_if_fail (buffer != NULL, NULL);
|
|
|
|
xml = moo_glade_xml_new_empty (domain);
|
|
|
|
if (!moo_glade_xml_parse_memory (xml, buffer, size, root, &error_here))
|
|
{
|
|
if (error)
|
|
{
|
|
g_propagate_error (error, error_here);
|
|
}
|
|
else
|
|
{
|
|
g_warning ("%s: %s", G_STRLOC, error_here->message);
|
|
g_error_free (error_here);
|
|
}
|
|
|
|
g_object_unref (xml);
|
|
return NULL;
|
|
}
|
|
|
|
return xml;
|
|
}
|
|
|
|
|
|
static MooMarkupNode*
|
|
find_widget (MooMarkupNode *node,
|
|
const char *id)
|
|
{
|
|
const char *wid_id;
|
|
|
|
g_return_val_if_fail (NODE_IS_WIDGET (node), NULL);
|
|
|
|
wid_id = moo_markup_get_prop (node, "id");
|
|
g_return_val_if_fail (wid_id != NULL, NULL);
|
|
|
|
if (!strcmp (wid_id, id))
|
|
return node;
|
|
|
|
FOREACH_ELM_START(node, elm)
|
|
{
|
|
if (NODE_IS_CHILD (elm))
|
|
{
|
|
FOREACH_ELM_START (elm, child)
|
|
{
|
|
if (NODE_IS_WIDGET (child))
|
|
{
|
|
MooMarkupNode *wid = find_widget (child, id);
|
|
if (wid)
|
|
return wid;
|
|
}
|
|
}
|
|
FOREACH_ELM_END;
|
|
}
|
|
}
|
|
FOREACH_ELM_END;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
moo_glade_xml_parse_markup (MooGladeXML *xml,
|
|
MooMarkupDoc *doc,
|
|
const char *root_id,
|
|
GtkWidget *root_widget,
|
|
GError **error)
|
|
{
|
|
MooMarkupNode *glade_elm;
|
|
MooMarkupNode *root = NULL;
|
|
Widget *widget = NULL;
|
|
gboolean result = FALSE;
|
|
|
|
g_return_val_if_fail (doc != NULL, FALSE);
|
|
g_return_val_if_fail (!xml->priv->done, FALSE);
|
|
g_return_val_if_fail (!root_widget || GTK_IS_WIDGET (root_widget), FALSE);
|
|
|
|
glade_elm = moo_markup_get_root_element (doc, "glade-interface");
|
|
|
|
if (!glade_elm)
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"no 'glade-interface' element");
|
|
return FALSE;
|
|
}
|
|
|
|
FOREACH_ELM_START(glade_elm, elm)
|
|
{
|
|
if (NODE_IS_WIDGET (elm))
|
|
{
|
|
if (root_id)
|
|
{
|
|
root = find_widget (elm, root_id);
|
|
|
|
if (root)
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
root = elm;
|
|
break;
|
|
}
|
|
}
|
|
else if (!strcmp (elm->name, "requires"))
|
|
{
|
|
g_message ("%s: ignoring '%s'", G_STRLOC, elm->name);
|
|
}
|
|
else
|
|
{
|
|
g_warning ("%s: invalid element '%s'", G_STRLOC, elm->name);
|
|
}
|
|
}
|
|
FOREACH_ELM_END;
|
|
|
|
if (!root)
|
|
{
|
|
if (root_id)
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"could not find root element '%s'",
|
|
root_id);
|
|
else
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"could not find root element");
|
|
|
|
goto out;
|
|
}
|
|
|
|
widget = widget_new (xml, NULL, root, error);
|
|
|
|
if (!widget)
|
|
goto out;
|
|
|
|
dump_widget (widget);
|
|
|
|
if (!moo_glade_xml_build (xml, widget, root_widget, error))
|
|
goto out;
|
|
|
|
#if 0
|
|
gtk_widget_show_all (widget->widget);
|
|
#endif
|
|
|
|
xml->priv->root_id = g_strdup (widget->id);
|
|
result = TRUE;
|
|
|
|
out:
|
|
if (widget)
|
|
widget_free (widget);
|
|
moo_glade_xml_cleanup (xml);
|
|
xml->priv->done = TRUE;
|
|
return result;
|
|
}
|
|
|
|
|
|
gboolean
|
|
moo_glade_xml_fill_widget (MooGladeXML *xml,
|
|
GtkWidget *target,
|
|
const char *buffer,
|
|
int size,
|
|
const char *target_name,
|
|
GError **error)
|
|
{
|
|
MooMarkupDoc *doc;
|
|
gboolean result;
|
|
|
|
g_return_val_if_fail (xml != NULL, FALSE);
|
|
g_return_val_if_fail (buffer != NULL, FALSE);
|
|
g_return_val_if_fail (GTK_IS_WIDGET (target), FALSE);
|
|
|
|
doc = moo_markup_parse_memory (buffer, size, error);
|
|
|
|
if (!doc)
|
|
return FALSE;
|
|
|
|
result = moo_glade_xml_parse_markup (xml, doc,
|
|
target_name,
|
|
target,
|
|
error);
|
|
|
|
moo_markup_doc_unref (doc);
|
|
return result;
|
|
}
|
|
|
|
|
|
GtkWidget*
|
|
moo_glade_xml_get_root (MooGladeXML *xml)
|
|
{
|
|
g_return_val_if_fail (xml != NULL, NULL);
|
|
g_return_val_if_fail (xml->priv->root_id != NULL, NULL);
|
|
return moo_glade_xml_get_widget (xml, xml->priv->root_id);
|
|
}
|
|
|
|
|
|
static gboolean
|
|
cmp_widget (G_GNUC_UNUSED gpointer key,
|
|
gpointer widget,
|
|
gpointer dead)
|
|
{
|
|
return widget == dead;
|
|
}
|
|
|
|
static void
|
|
widget_destroyed (MooGladeXML *xml,
|
|
gpointer widget)
|
|
{
|
|
guint removed;
|
|
removed = g_hash_table_foreach_remove (xml->priv->widgets,
|
|
(GHRFunc) cmp_widget, widget);
|
|
g_return_if_fail (removed == 1);
|
|
}
|
|
|
|
static gboolean
|
|
moo_glade_xml_add_widget (MooGladeXML *xml,
|
|
const char *id,
|
|
GtkWidget *widget,
|
|
GError **error)
|
|
{
|
|
g_return_val_if_fail (id != NULL, FALSE);
|
|
g_return_val_if_fail (GTK_IS_WIDGET (widget), FALSE);
|
|
|
|
if (g_hash_table_lookup (xml->priv->widgets, id))
|
|
{
|
|
g_set_error (error, MOO_GLADE_XML_ERROR,
|
|
MOO_GLADE_XML_ERROR_FAILED,
|
|
"duplicated id '%s'", id);
|
|
return FALSE;
|
|
}
|
|
|
|
g_assert (!g_hash_table_lookup (xml->priv->widgets, id));
|
|
g_hash_table_insert (xml->priv->widgets, g_strdup (id), widget);
|
|
g_object_weak_ref (G_OBJECT (widget), (GWeakNotify) widget_destroyed, xml);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void
|
|
moo_glade_xml_cleanup (MooGladeXML *xml)
|
|
{
|
|
if (xml->priv->class_to_type)
|
|
g_hash_table_destroy (xml->priv->class_to_type);
|
|
if (xml->priv->id_to_type)
|
|
g_hash_table_destroy (xml->priv->id_to_type);
|
|
if (xml->priv->id_to_func)
|
|
g_hash_table_destroy (xml->priv->id_to_func);
|
|
if (xml->priv->props)
|
|
g_hash_table_destroy (xml->priv->props);
|
|
if (xml->priv->signal_func_data && xml->priv->signal_func_notify)
|
|
xml->priv->signal_func_notify (xml->priv->signal_func_data);
|
|
if (xml->priv->prop_func_data && xml->priv->prop_func_notify)
|
|
xml->priv->prop_func_notify (xml->priv->prop_func_data);
|
|
|
|
xml->priv->class_to_type = NULL;
|
|
xml->priv->id_to_type = NULL;
|
|
xml->priv->id_to_func = NULL;
|
|
xml->priv->props = NULL;
|
|
xml->priv->signal_func_data = NULL;
|
|
xml->priv->prop_func_data = NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
unref_widget (G_GNUC_UNUSED gpointer key,
|
|
GObject *widget,
|
|
MooGladeXML *xml)
|
|
{
|
|
g_object_weak_unref (widget, (GWeakNotify) widget_destroyed, xml);
|
|
}
|
|
|
|
static void
|
|
moo_glade_xml_dispose (GObject *object)
|
|
{
|
|
MooGladeXML *xml = MOO_GLADE_XML (object);
|
|
|
|
if (xml->priv)
|
|
{
|
|
moo_glade_xml_cleanup (xml);
|
|
g_hash_table_foreach (xml->priv->widgets,
|
|
(GHFunc) unref_widget, xml);
|
|
g_hash_table_destroy (xml->priv->widgets);
|
|
g_free (xml->priv->translation_domain);
|
|
g_free (xml->priv->root_id);
|
|
g_free (xml->priv);
|
|
xml->priv = NULL;
|
|
}
|
|
|
|
G_OBJECT_CLASS(moo_glade_xml_parent_class)->dispose (object);
|
|
}
|
|
|
|
|
|
gpointer
|
|
moo_glade_xml_get_widget (MooGladeXML *xml,
|
|
const char *id)
|
|
{
|
|
g_return_val_if_fail (xml != NULL, NULL);
|
|
g_return_val_if_fail (id != NULL, NULL);
|
|
return g_hash_table_lookup (xml->priv->widgets, id);
|
|
}
|
|
|
|
|
|
void
|
|
moo_glade_xml_map_class (MooGladeXML *xml,
|
|
const char *class_name,
|
|
GType type)
|
|
{
|
|
g_return_if_fail (xml != NULL);
|
|
g_return_if_fail (class_name != NULL);
|
|
|
|
if (type)
|
|
g_hash_table_insert (xml->priv->class_to_type,
|
|
g_strdup (class_name),
|
|
GSIZE_TO_POINTER (type));
|
|
else
|
|
g_hash_table_remove (xml->priv->class_to_type,
|
|
class_name);
|
|
}
|
|
|
|
|
|
void
|
|
moo_glade_xml_map_id (MooGladeXML *xml,
|
|
const char *id,
|
|
GType use_type)
|
|
{
|
|
g_return_if_fail (xml != NULL);
|
|
g_return_if_fail (id != NULL);
|
|
|
|
if (use_type)
|
|
g_hash_table_insert (xml->priv->id_to_type,
|
|
g_strdup (id),
|
|
GSIZE_TO_POINTER (use_type));
|
|
else
|
|
g_hash_table_remove (xml->priv->id_to_type, id);
|
|
}
|
|
|
|
|
|
void moo_glade_xml_map_custom (MooGladeXML *xml,
|
|
const char *id,
|
|
MooGladeCreateFunc func,
|
|
gpointer data)
|
|
{
|
|
g_return_if_fail (xml != NULL);
|
|
g_return_if_fail (id != NULL);
|
|
|
|
if (func)
|
|
g_hash_table_insert (xml->priv->id_to_func,
|
|
g_strdup (id),
|
|
func_data_pair_new (func, data));
|
|
else
|
|
g_hash_table_remove (xml->priv->id_to_func, id);
|
|
}
|
|
|
|
|
|
void
|
|
moo_glade_xml_set_signal_func (MooGladeXML *xml,
|
|
MooGladeSignalFunc func,
|
|
gpointer data,
|
|
GDestroyNotify notify)
|
|
{
|
|
g_return_if_fail (xml != NULL);
|
|
|
|
if (xml->priv->signal_func_data && xml->priv->signal_func_notify)
|
|
xml->priv->signal_func_notify (xml->priv->signal_func_data);
|
|
|
|
xml->priv->signal_func = func;
|
|
xml->priv->signal_func_data = data;
|
|
xml->priv->signal_func_notify = notify;
|
|
}
|
|
|
|
|
|
void
|
|
moo_glade_xml_set_prop_func (MooGladeXML *xml,
|
|
MooGladePropFunc func,
|
|
gpointer data,
|
|
GDestroyNotify notify)
|
|
{
|
|
g_return_if_fail (xml != NULL);
|
|
|
|
if (xml->priv->prop_func_data && xml->priv->prop_func_notify)
|
|
xml->priv->prop_func_notify (xml->priv->prop_func_data);
|
|
|
|
xml->priv->prop_func = func;
|
|
xml->priv->prop_func_data = data;
|
|
xml->priv->prop_func_notify = notify;
|
|
}
|
|
|
|
|
|
static FuncDataPair*
|
|
func_data_pair_new (MooGladeCreateFunc func,
|
|
gpointer data)
|
|
{
|
|
FuncDataPair *pair;
|
|
|
|
g_return_val_if_fail (func != NULL, NULL);
|
|
|
|
pair = g_new (FuncDataPair, 1);
|
|
pair->func = func;
|
|
pair->data = data;
|
|
|
|
return pair;
|
|
}
|
|
|
|
|
|
static void
|
|
func_data_pair_free (FuncDataPair *pair)
|
|
{
|
|
g_free (pair);
|
|
}
|
|
|
|
|
|
static GQuark
|
|
moo_glade_xml_error_quark (void)
|
|
{
|
|
return g_quark_from_static_string ("moo-glade-xml-error");
|
|
}
|
|
|
|
|
|
#define REGISTER_TYPE(class_name,type) \
|
|
if (!strcmp (name, class_name)) \
|
|
return type
|
|
|
|
|
|
static GType
|
|
get_type_by_name (const char *name)
|
|
{
|
|
GType type = g_type_from_name (name);
|
|
|
|
if (type)
|
|
return type;
|
|
|
|
REGISTER_TYPE ("GtkAboutDialog", GTK_TYPE_ABOUT_DIALOG);
|
|
REGISTER_TYPE ("GtkCellRendererCombo", GTK_TYPE_CELL_RENDERER_COMBO);
|
|
REGISTER_TYPE ("GtkCellRendererProgress", GTK_TYPE_CELL_RENDERER_PROGRESS);
|
|
REGISTER_TYPE ("GtkCellView", GTK_TYPE_CELL_VIEW);
|
|
REGISTER_TYPE ("GtkIconView", GTK_TYPE_ICON_VIEW);
|
|
REGISTER_TYPE ("GtkMenuToolButton", GTK_TYPE_MENU_TOOL_BUTTON);
|
|
REGISTER_TYPE ("GtkAccelMap", GTK_TYPE_ACCEL_MAP);
|
|
REGISTER_TYPE ("GtkAction", GTK_TYPE_ACTION);
|
|
REGISTER_TYPE ("GtkActionGroup", GTK_TYPE_ACTION_GROUP);
|
|
REGISTER_TYPE ("GtkCellLayout", GTK_TYPE_CELL_LAYOUT);
|
|
REGISTER_TYPE ("GtkComboBox", GTK_TYPE_COMBO_BOX);
|
|
REGISTER_TYPE ("GtkComboBoxEntry", GTK_TYPE_COMBO_BOX_ENTRY);
|
|
REGISTER_TYPE ("GtkEntryCompletion", GTK_TYPE_ENTRY_COMPLETION);
|
|
REGISTER_TYPE ("GtkExpander", GTK_TYPE_EXPANDER);
|
|
REGISTER_TYPE ("GtkFileChooser", GTK_TYPE_FILE_CHOOSER);
|
|
REGISTER_TYPE ("GtkFileChooserAction", GTK_TYPE_FILE_CHOOSER_ACTION);
|
|
REGISTER_TYPE ("GtkFileChooserDialog", GTK_TYPE_FILE_CHOOSER_DIALOG);
|
|
REGISTER_TYPE ("GtkFileChooserError", GTK_TYPE_FILE_CHOOSER_ERROR);
|
|
REGISTER_TYPE ("GtkFileChooserWidget", GTK_TYPE_FILE_CHOOSER_WIDGET);
|
|
REGISTER_TYPE ("GtkFileFilter", GTK_TYPE_FILE_FILTER);
|
|
REGISTER_TYPE ("GtkFileFilterFlags", GTK_TYPE_FILE_FILTER_FLAGS);
|
|
REGISTER_TYPE ("GtkIconInfo", GTK_TYPE_ICON_INFO);
|
|
REGISTER_TYPE ("GtkIconLookupFlags", GTK_TYPE_ICON_LOOKUP_FLAGS);
|
|
REGISTER_TYPE ("GtkIconTheme", GTK_TYPE_ICON_THEME);
|
|
REGISTER_TYPE ("GtkIconThemeError", GTK_TYPE_ICON_THEME_ERROR);
|
|
REGISTER_TYPE ("GtkRadioAction", GTK_TYPE_RADIO_ACTION);
|
|
REGISTER_TYPE ("GtkRadioToolButton", GTK_TYPE_RADIO_TOOL_BUTTON);
|
|
REGISTER_TYPE ("GtkScrollStep", GTK_TYPE_SCROLL_STEP);
|
|
REGISTER_TYPE ("GtkSeparatorToolItem", GTK_TYPE_SEPARATOR_TOOL_ITEM);
|
|
REGISTER_TYPE ("GtkToggleAction", GTK_TYPE_TOGGLE_ACTION);
|
|
REGISTER_TYPE ("GtkToggleToolButton", GTK_TYPE_TOGGLE_TOOL_BUTTON);
|
|
REGISTER_TYPE ("GtkToolButton", GTK_TYPE_TOOL_BUTTON);
|
|
REGISTER_TYPE ("GtkToolItem", GTK_TYPE_TOOL_ITEM);
|
|
REGISTER_TYPE ("GtkTreeModelFilter", GTK_TYPE_TREE_MODEL_FILTER);
|
|
REGISTER_TYPE ("GtkUiManager", GTK_TYPE_UI_MANAGER);
|
|
REGISTER_TYPE ("GtkUiManagerItemType", GTK_TYPE_UI_MANAGER_ITEM_TYPE);
|
|
REGISTER_TYPE ("GtkAccelFlags", GTK_TYPE_ACCEL_FLAGS);
|
|
REGISTER_TYPE ("GtkAccelGroup", GTK_TYPE_ACCEL_GROUP);
|
|
REGISTER_TYPE ("GtkAccelLabel", GTK_TYPE_ACCEL_LABEL);
|
|
REGISTER_TYPE ("GtkAccessible", GTK_TYPE_ACCESSIBLE);
|
|
REGISTER_TYPE ("GtkAdjustment", GTK_TYPE_ADJUSTMENT);
|
|
REGISTER_TYPE ("GtkAlignment", GTK_TYPE_ALIGNMENT);
|
|
REGISTER_TYPE ("GtkAnchorType", GTK_TYPE_ANCHOR_TYPE);
|
|
REGISTER_TYPE ("GtkArgFlags", GTK_TYPE_ARG_FLAGS);
|
|
REGISTER_TYPE ("GtkArrow", GTK_TYPE_ARROW);
|
|
REGISTER_TYPE ("GtkArrowType", GTK_TYPE_ARROW_TYPE);
|
|
REGISTER_TYPE ("GtkAspectFrame", GTK_TYPE_ASPECT_FRAME);
|
|
REGISTER_TYPE ("GtkAttachOptions", GTK_TYPE_ATTACH_OPTIONS);
|
|
REGISTER_TYPE ("GtkBin", GTK_TYPE_BIN);
|
|
REGISTER_TYPE ("GtkBorder", GTK_TYPE_BORDER);
|
|
REGISTER_TYPE ("GtkBox", GTK_TYPE_BOX);
|
|
REGISTER_TYPE ("GtkButton", GTK_TYPE_BUTTON);
|
|
REGISTER_TYPE ("GtkButtonAction", GTK_TYPE_BUTTON_ACTION);
|
|
REGISTER_TYPE ("GtkButtonBox", GTK_TYPE_BUTTON_BOX);
|
|
REGISTER_TYPE ("GtkButtonBoxStyle", GTK_TYPE_BUTTON_BOX_STYLE);
|
|
REGISTER_TYPE ("GtkButtonsType", GTK_TYPE_BUTTONS_TYPE);
|
|
REGISTER_TYPE ("GtkCalendar", GTK_TYPE_CALENDAR);
|
|
REGISTER_TYPE ("GtkCalendarDisplayOptions", GTK_TYPE_CALENDAR_DISPLAY_OPTIONS);
|
|
REGISTER_TYPE ("GtkCellEditable", GTK_TYPE_CELL_EDITABLE);
|
|
REGISTER_TYPE ("GtkCellRenderer", GTK_TYPE_CELL_RENDERER);
|
|
REGISTER_TYPE ("GtkCellRendererMode", GTK_TYPE_CELL_RENDERER_MODE);
|
|
REGISTER_TYPE ("GtkCellRendererPixbuf", GTK_TYPE_CELL_RENDERER_PIXBUF);
|
|
REGISTER_TYPE ("GtkCellRendererState", GTK_TYPE_CELL_RENDERER_STATE);
|
|
REGISTER_TYPE ("GtkCellRendererText", GTK_TYPE_CELL_RENDERER_TEXT);
|
|
REGISTER_TYPE ("GtkCellRendererToggle", GTK_TYPE_CELL_RENDERER_TOGGLE);
|
|
REGISTER_TYPE ("GtkCellType", GTK_TYPE_CELL_TYPE);
|
|
REGISTER_TYPE ("GtkCheckButton", GTK_TYPE_CHECK_BUTTON);
|
|
REGISTER_TYPE ("GtkCheckMenuItem", GTK_TYPE_CHECK_MENU_ITEM);
|
|
REGISTER_TYPE ("GtkClipboard", GTK_TYPE_CLIPBOARD);
|
|
REGISTER_TYPE ("GtkColorButton", GTK_TYPE_COLOR_BUTTON);
|
|
REGISTER_TYPE ("GtkColorSelection", GTK_TYPE_COLOR_SELECTION);
|
|
REGISTER_TYPE ("GtkColorSelectionDialog", GTK_TYPE_COLOR_SELECTION_DIALOG);
|
|
REGISTER_TYPE ("GtkCombo", GTK_TYPE_COMBO);
|
|
REGISTER_TYPE ("GtkContainer", GTK_TYPE_CONTAINER);
|
|
REGISTER_TYPE ("GtkCornerType", GTK_TYPE_CORNER_TYPE);
|
|
REGISTER_TYPE ("GtkCurve", GTK_TYPE_CURVE);
|
|
REGISTER_TYPE ("GtkCurveType", GTK_TYPE_CURVE_TYPE);
|
|
REGISTER_TYPE ("GtkDeleteType", GTK_TYPE_DELETE_TYPE);
|
|
REGISTER_TYPE ("GtkDestDefaults", GTK_TYPE_DEST_DEFAULTS);
|
|
REGISTER_TYPE ("GtkDialog", GTK_TYPE_DIALOG);
|
|
REGISTER_TYPE ("GtkDialogFlags", GTK_TYPE_DIALOG_FLAGS);
|
|
REGISTER_TYPE ("GtkDirectionType", GTK_TYPE_DIRECTION_TYPE);
|
|
REGISTER_TYPE ("GtkDrawingArea", GTK_TYPE_DRAWING_AREA);
|
|
REGISTER_TYPE ("GtkEditable", GTK_TYPE_EDITABLE);
|
|
REGISTER_TYPE ("GtkEntry", GTK_TYPE_ENTRY);
|
|
REGISTER_TYPE ("GtkEventBox", GTK_TYPE_EVENT_BOX);
|
|
REGISTER_TYPE ("GtkExpanderStyle", GTK_TYPE_EXPANDER_STYLE);
|
|
REGISTER_TYPE ("GtkFixed", GTK_TYPE_FIXED);
|
|
REGISTER_TYPE ("GtkFontButton", GTK_TYPE_FONT_BUTTON);
|
|
REGISTER_TYPE ("GtkFontSelection", GTK_TYPE_FONT_SELECTION);
|
|
REGISTER_TYPE ("GtkFontSelectionDialog", GTK_TYPE_FONT_SELECTION_DIALOG);
|
|
REGISTER_TYPE ("GtkFrame", GTK_TYPE_FRAME);
|
|
REGISTER_TYPE ("GtkGammaCurve", GTK_TYPE_GAMMA_CURVE);
|
|
REGISTER_TYPE ("GtkHBox", GTK_TYPE_HBOX);
|
|
REGISTER_TYPE ("GtkHButtonBox", GTK_TYPE_HBUTTON_BOX);
|
|
REGISTER_TYPE ("GtkHPaned", GTK_TYPE_HPANED);
|
|
REGISTER_TYPE ("GtkHRuler", GTK_TYPE_HRULER);
|
|
REGISTER_TYPE ("GtkHScale", GTK_TYPE_HSCALE);
|
|
REGISTER_TYPE ("GtkHScrollbar", GTK_TYPE_HSCROLLBAR);
|
|
REGISTER_TYPE ("GtkHSeparator", GTK_TYPE_HSEPARATOR);
|
|
REGISTER_TYPE ("GtkHandleBox", GTK_TYPE_HANDLE_BOX);
|
|
REGISTER_TYPE ("GtkIconFactory", GTK_TYPE_ICON_FACTORY);
|
|
REGISTER_TYPE ("GtkIconSet", GTK_TYPE_ICON_SET);
|
|
REGISTER_TYPE ("GtkIconSize", GTK_TYPE_ICON_SIZE);
|
|
REGISTER_TYPE ("GtkIconSource", GTK_TYPE_ICON_SOURCE);
|
|
REGISTER_TYPE ("GtkIdentifier", GTK_TYPE_IDENTIFIER);
|
|
REGISTER_TYPE ("GtkImContext", GTK_TYPE_IM_CONTEXT);
|
|
REGISTER_TYPE ("GtkImContextSimple", GTK_TYPE_IM_CONTEXT_SIMPLE);
|
|
REGISTER_TYPE ("GtkImMulticontext", GTK_TYPE_IM_MULTICONTEXT);
|
|
REGISTER_TYPE ("GtkImPreeditStyle", GTK_TYPE_IM_PREEDIT_STYLE);
|
|
REGISTER_TYPE ("GtkImStatusStyle", GTK_TYPE_IM_STATUS_STYLE);
|
|
REGISTER_TYPE ("GtkImage", GTK_TYPE_IMAGE);
|
|
REGISTER_TYPE ("GtkImageMenuItem", GTK_TYPE_IMAGE_MENU_ITEM);
|
|
REGISTER_TYPE ("GtkImageType", GTK_TYPE_IMAGE_TYPE);
|
|
REGISTER_TYPE ("GtkInputDialog", GTK_TYPE_INPUT_DIALOG);
|
|
REGISTER_TYPE ("GtkInvisible", GTK_TYPE_INVISIBLE);
|
|
REGISTER_TYPE ("GtkItem", GTK_TYPE_ITEM);
|
|
REGISTER_TYPE ("GtkJustification", GTK_TYPE_JUSTIFICATION);
|
|
REGISTER_TYPE ("GtkLabel", GTK_TYPE_LABEL);
|
|
REGISTER_TYPE ("GtkLayout", GTK_TYPE_LAYOUT);
|
|
REGISTER_TYPE ("GtkList", GTK_TYPE_LIST);
|
|
REGISTER_TYPE ("GtkListItem", GTK_TYPE_LIST_ITEM);
|
|
REGISTER_TYPE ("GtkListStore", GTK_TYPE_LIST_STORE);
|
|
REGISTER_TYPE ("GtkMatchType", GTK_TYPE_MATCH_TYPE);
|
|
REGISTER_TYPE ("GtkMenu", GTK_TYPE_MENU);
|
|
REGISTER_TYPE ("GtkMenuBar", GTK_TYPE_MENU_BAR);
|
|
REGISTER_TYPE ("GtkMenuDirectionType", GTK_TYPE_MENU_DIRECTION_TYPE);
|
|
REGISTER_TYPE ("GtkMenuItem", GTK_TYPE_MENU_ITEM);
|
|
REGISTER_TYPE ("GtkMenuShell", GTK_TYPE_MENU_SHELL);
|
|
REGISTER_TYPE ("GtkMessageDialog", GTK_TYPE_MESSAGE_DIALOG);
|
|
REGISTER_TYPE ("GtkMessageType", GTK_TYPE_MESSAGE_TYPE);
|
|
REGISTER_TYPE ("GtkMetricType", GTK_TYPE_METRIC_TYPE);
|
|
REGISTER_TYPE ("GtkMisc", GTK_TYPE_MISC);
|
|
REGISTER_TYPE ("GtkMovementStep", GTK_TYPE_MOVEMENT_STEP);
|
|
REGISTER_TYPE ("GtkNotebook", GTK_TYPE_NOTEBOOK);
|
|
REGISTER_TYPE ("GtkNotebookTab", GTK_TYPE_NOTEBOOK_TAB);
|
|
REGISTER_TYPE ("GtkObject", GTK_TYPE_OBJECT);
|
|
REGISTER_TYPE ("GtkObjectFlags", GTK_TYPE_OBJECT_FLAGS);
|
|
REGISTER_TYPE ("GtkOptionMenu", GTK_TYPE_OPTION_MENU);
|
|
REGISTER_TYPE ("GtkOrientation", GTK_TYPE_ORIENTATION);
|
|
REGISTER_TYPE ("GtkPackType", GTK_TYPE_PACK_TYPE);
|
|
REGISTER_TYPE ("GtkPaned", GTK_TYPE_PANED);
|
|
REGISTER_TYPE ("GtkPathPriorityType", GTK_TYPE_PATH_PRIORITY_TYPE);
|
|
REGISTER_TYPE ("GtkPathType", GTK_TYPE_PATH_TYPE);
|
|
REGISTER_TYPE ("GtkPolicyType", GTK_TYPE_POLICY_TYPE);
|
|
REGISTER_TYPE ("GtkPositionType", GTK_TYPE_POSITION_TYPE);
|
|
REGISTER_TYPE ("GtkPrivateFlags", GTK_TYPE_PRIVATE_FLAGS);
|
|
REGISTER_TYPE ("GtkProgressBar", GTK_TYPE_PROGRESS_BAR);
|
|
REGISTER_TYPE ("GtkProgressBarOrientation", GTK_TYPE_PROGRESS_BAR_ORIENTATION);
|
|
REGISTER_TYPE ("GtkProgressBarStyle", GTK_TYPE_PROGRESS_BAR_STYLE);
|
|
REGISTER_TYPE ("GtkRadioButton", GTK_TYPE_RADIO_BUTTON);
|
|
REGISTER_TYPE ("GtkRadioMenuItem", GTK_TYPE_RADIO_MENU_ITEM);
|
|
REGISTER_TYPE ("GtkRange", GTK_TYPE_RANGE);
|
|
REGISTER_TYPE ("GtkRcFlags", GTK_TYPE_RC_FLAGS);
|
|
REGISTER_TYPE ("GtkRcStyle", GTK_TYPE_RC_STYLE);
|
|
REGISTER_TYPE ("GtkRcTokenType", GTK_TYPE_RC_TOKEN_TYPE);
|
|
REGISTER_TYPE ("GtkReliefStyle", GTK_TYPE_RELIEF_STYLE);
|
|
REGISTER_TYPE ("GtkRequisition", GTK_TYPE_REQUISITION);
|
|
REGISTER_TYPE ("GtkResizeMode", GTK_TYPE_RESIZE_MODE);
|
|
REGISTER_TYPE ("GtkResponseType", GTK_TYPE_RESPONSE_TYPE);
|
|
REGISTER_TYPE ("GtkRuler", GTK_TYPE_RULER);
|
|
REGISTER_TYPE ("GtkScale", GTK_TYPE_SCALE);
|
|
REGISTER_TYPE ("GtkScrollType", GTK_TYPE_SCROLL_TYPE);
|
|
REGISTER_TYPE ("GtkScrollbar", GTK_TYPE_SCROLLBAR);
|
|
REGISTER_TYPE ("GtkScrolledWindow", GTK_TYPE_SCROLLED_WINDOW);
|
|
REGISTER_TYPE ("GtkSelectionData", GTK_TYPE_SELECTION_DATA);
|
|
REGISTER_TYPE ("GtkSelectionMode", GTK_TYPE_SELECTION_MODE);
|
|
REGISTER_TYPE ("GtkSeparator", GTK_TYPE_SEPARATOR);
|
|
REGISTER_TYPE ("GtkSeparatorMenuItem", GTK_TYPE_SEPARATOR_MENU_ITEM);
|
|
REGISTER_TYPE ("GtkSettings", GTK_TYPE_SETTINGS);
|
|
REGISTER_TYPE ("GtkShadowType", GTK_TYPE_SHADOW_TYPE);
|
|
REGISTER_TYPE ("GtkSideType", GTK_TYPE_SIDE_TYPE);
|
|
REGISTER_TYPE ("GtkSignalRunType", GTK_TYPE_SIGNAL_RUN_TYPE);
|
|
REGISTER_TYPE ("GtkSizeGroup", GTK_TYPE_SIZE_GROUP);
|
|
REGISTER_TYPE ("GtkSizeGroupMode", GTK_TYPE_SIZE_GROUP_MODE);
|
|
REGISTER_TYPE ("GtkSortType", GTK_TYPE_SORT_TYPE);
|
|
REGISTER_TYPE ("GtkSpinButton", GTK_TYPE_SPIN_BUTTON);
|
|
REGISTER_TYPE ("GtkSpinButtonUpdatePolicy", GTK_TYPE_SPIN_BUTTON_UPDATE_POLICY);
|
|
REGISTER_TYPE ("GtkSpinType", GTK_TYPE_SPIN_TYPE);
|
|
REGISTER_TYPE ("GtkStateType", GTK_TYPE_STATE_TYPE);
|
|
REGISTER_TYPE ("GtkStatusbar", GTK_TYPE_STATUSBAR);
|
|
REGISTER_TYPE ("GtkStyle", GTK_TYPE_STYLE);
|
|
REGISTER_TYPE ("GtkSubmenuDirection", GTK_TYPE_SUBMENU_DIRECTION);
|
|
REGISTER_TYPE ("GtkSubmenuPlacement", GTK_TYPE_SUBMENU_PLACEMENT);
|
|
REGISTER_TYPE ("GtkTable", GTK_TYPE_TABLE);
|
|
REGISTER_TYPE ("GtkTargetFlags", GTK_TYPE_TARGET_FLAGS);
|
|
REGISTER_TYPE ("GtkTearoffMenuItem", GTK_TYPE_TEAROFF_MENU_ITEM);
|
|
REGISTER_TYPE ("GtkTextAttributes", GTK_TYPE_TEXT_ATTRIBUTES);
|
|
REGISTER_TYPE ("GtkTextBuffer", GTK_TYPE_TEXT_BUFFER);
|
|
REGISTER_TYPE ("GtkTextChildAnchor", GTK_TYPE_TEXT_CHILD_ANCHOR);
|
|
REGISTER_TYPE ("GtkTextDirection", GTK_TYPE_TEXT_DIRECTION);
|
|
REGISTER_TYPE ("GtkTextIter", GTK_TYPE_TEXT_ITER);
|
|
REGISTER_TYPE ("GtkTextMark", GTK_TYPE_TEXT_MARK);
|
|
REGISTER_TYPE ("GtkTextSearchFlags", GTK_TYPE_TEXT_SEARCH_FLAGS);
|
|
REGISTER_TYPE ("GtkTextTag", GTK_TYPE_TEXT_TAG);
|
|
REGISTER_TYPE ("GtkTextTagTable", GTK_TYPE_TEXT_TAG_TABLE);
|
|
REGISTER_TYPE ("GtkTextView", GTK_TYPE_TEXT_VIEW);
|
|
REGISTER_TYPE ("GtkTextWindowType", GTK_TYPE_TEXT_WINDOW_TYPE);
|
|
REGISTER_TYPE ("GtkToggleButton", GTK_TYPE_TOGGLE_BUTTON);
|
|
REGISTER_TYPE ("GtkToolbar", GTK_TYPE_TOOLBAR);
|
|
REGISTER_TYPE ("GtkToolbarChildType", GTK_TYPE_TOOLBAR_CHILD_TYPE);
|
|
REGISTER_TYPE ("GtkToolbarSpaceStyle", GTK_TYPE_TOOLBAR_SPACE_STYLE);
|
|
REGISTER_TYPE ("GtkToolbarStyle", GTK_TYPE_TOOLBAR_STYLE);
|
|
REGISTER_TYPE ("GtkTreeDragDest", GTK_TYPE_TREE_DRAG_DEST);
|
|
REGISTER_TYPE ("GtkTreeDragSource", GTK_TYPE_TREE_DRAG_SOURCE);
|
|
REGISTER_TYPE ("GtkTreeIter", GTK_TYPE_TREE_ITER);
|
|
REGISTER_TYPE ("GtkTreeModel", GTK_TYPE_TREE_MODEL);
|
|
REGISTER_TYPE ("GtkTreeModelFlags", GTK_TYPE_TREE_MODEL_FLAGS);
|
|
REGISTER_TYPE ("GtkTreeModelSort", GTK_TYPE_TREE_MODEL_SORT);
|
|
REGISTER_TYPE ("GtkTreePath", GTK_TYPE_TREE_PATH);
|
|
REGISTER_TYPE ("GtkTreeRowReference", GTK_TYPE_TREE_ROW_REFERENCE);
|
|
REGISTER_TYPE ("GtkTreeSelection", GTK_TYPE_TREE_SELECTION);
|
|
REGISTER_TYPE ("GtkTreeSortable", GTK_TYPE_TREE_SORTABLE);
|
|
REGISTER_TYPE ("GtkTreeStore", GTK_TYPE_TREE_STORE);
|
|
REGISTER_TYPE ("GtkTreeView", GTK_TYPE_TREE_VIEW);
|
|
REGISTER_TYPE ("GtkTreeViewColumn", GTK_TYPE_TREE_VIEW_COLUMN);
|
|
REGISTER_TYPE ("GtkTreeViewColumnSizing", GTK_TYPE_TREE_VIEW_COLUMN_SIZING);
|
|
REGISTER_TYPE ("GtkTreeViewDropPosition", GTK_TYPE_TREE_VIEW_DROP_POSITION);
|
|
REGISTER_TYPE ("GtkTreeViewMode", GTK_TYPE_TREE_VIEW_MODE);
|
|
REGISTER_TYPE ("GtkUpdateType", GTK_TYPE_UPDATE_TYPE);
|
|
REGISTER_TYPE ("GtkVBox", GTK_TYPE_VBOX);
|
|
REGISTER_TYPE ("GtkVButtonBox", GTK_TYPE_VBUTTON_BOX);
|
|
REGISTER_TYPE ("GtkViewport", GTK_TYPE_VIEWPORT);
|
|
REGISTER_TYPE ("GtkVisibility", GTK_TYPE_VISIBILITY);
|
|
REGISTER_TYPE ("GtkVPaned", GTK_TYPE_VPANED);
|
|
REGISTER_TYPE ("GtkVRuler", GTK_TYPE_VRULER);
|
|
REGISTER_TYPE ("GtkVScale", GTK_TYPE_VSCALE);
|
|
REGISTER_TYPE ("GtkVScrollbar", GTK_TYPE_VSCROLLBAR);
|
|
REGISTER_TYPE ("GtkVSeparator", GTK_TYPE_VSEPARATOR);
|
|
REGISTER_TYPE ("GtkWidget", GTK_TYPE_WIDGET);
|
|
REGISTER_TYPE ("GtkWidgetFlags", GTK_TYPE_WIDGET_FLAGS);
|
|
REGISTER_TYPE ("GtkWidgetHelpType", GTK_TYPE_WIDGET_HELP_TYPE);
|
|
REGISTER_TYPE ("GtkWindow", GTK_TYPE_WINDOW);
|
|
REGISTER_TYPE ("GtkWindowGroup", GTK_TYPE_WINDOW_GROUP);
|
|
REGISTER_TYPE ("GtkWindowPosition", GTK_TYPE_WINDOW_POSITION);
|
|
REGISTER_TYPE ("GtkWindowType", GTK_TYPE_WINDOW_TYPE);
|
|
REGISTER_TYPE ("GtkWrapMode", GTK_TYPE_WRAP_MODE);
|
|
|
|
#ifndef __WIN32__
|
|
REGISTER_TYPE ("GtkPlug", GTK_TYPE_PLUG);
|
|
REGISTER_TYPE ("GtkSocket", GTK_TYPE_SOCKET);
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
#if 0
|
|
static void print_widget (Widget *widget, guint offset);
|
|
static void print_child (Child *child, guint offset);
|
|
static void print_props (GParameter *params, guint n_params, guint offset);
|
|
|
|
static void
|
|
print_widget (Widget *widget, guint offset)
|
|
{
|
|
GSList *l;
|
|
char *fill = g_strnfill (offset, ' ');
|
|
g_print ("%sWidget id '%s' class '%s'\n",
|
|
fill, widget->id, g_type_name (widget->type));
|
|
print_props (widget->props->params, widget->props->n_params, offset);
|
|
for (l = widget->children; l != NULL; l = l->next)
|
|
print_child (l->data, offset + 3);
|
|
}
|
|
|
|
|
|
static void
|
|
print_child (Child *child, guint offset)
|
|
{
|
|
char *fill = g_strnfill (offset, ' ');
|
|
g_print ("%sChild\n", fill);
|
|
print_props (child->props->params, child->props->n_params, offset);
|
|
print_widget (child->widget, offset + 3);
|
|
}
|
|
|
|
|
|
static void
|
|
print_value (GValue *value)
|
|
{
|
|
if (value->g_type == G_TYPE_CHAR)
|
|
{
|
|
g_print ("%c", g_value_get_char (value));
|
|
}
|
|
else if (value->g_type == G_TYPE_UCHAR)
|
|
{
|
|
g_print ("%c", g_value_get_uchar (value));
|
|
}
|
|
else if (value->g_type == G_TYPE_BOOLEAN)
|
|
{
|
|
g_print ("%s",
|
|
g_value_get_boolean (value) ? "TRUE" : "FALSE");
|
|
}
|
|
else if (value->g_type == G_TYPE_INT ||
|
|
value->g_type == G_TYPE_LONG ||
|
|
value->g_type == G_TYPE_INT64 ||
|
|
value->g_type == G_TYPE_UINT ||
|
|
value->g_type == G_TYPE_ULONG ||
|
|
value->g_type == G_TYPE_UINT64) /* XXX */
|
|
{
|
|
int val;
|
|
|
|
if (value->g_type == G_TYPE_INT)
|
|
val = g_value_get_int (value);
|
|
else if (value->g_type == G_TYPE_LONG)
|
|
val = g_value_get_long (value);
|
|
else if (value->g_type == G_TYPE_INT64)
|
|
val = g_value_get_int64 (value);
|
|
else if (value->g_type == G_TYPE_UINT)
|
|
val = g_value_get_uint (value);
|
|
else if (value->g_type == G_TYPE_ULONG)
|
|
val = g_value_get_ulong (value);
|
|
else if (value->g_type == G_TYPE_UINT64)
|
|
val = g_value_get_uint64 (value);
|
|
|
|
g_print ("%d", val);
|
|
}
|
|
else if (value->g_type == G_TYPE_FLOAT ||
|
|
value->g_type == G_TYPE_DOUBLE) /* XXX */
|
|
{
|
|
double val;
|
|
|
|
if (value->g_type == G_TYPE_FLOAT)
|
|
val = g_value_get_float (value);
|
|
else if (value->g_type == G_TYPE_DOUBLE)
|
|
val = g_value_get_double (value);
|
|
|
|
g_print ("%f", val);
|
|
}
|
|
else if (value->g_type == G_TYPE_STRING)
|
|
{
|
|
g_print ("%s", g_value_get_string (value));
|
|
}
|
|
else if (G_TYPE_IS_ENUM (value->g_type))
|
|
{
|
|
g_print ("%d", g_value_get_enum (value));
|
|
}
|
|
else if (G_TYPE_IS_FLAGS (value->g_type))
|
|
{
|
|
g_print ("%d", g_value_get_flags (value));
|
|
}
|
|
else
|
|
{
|
|
g_print ("<%s>", g_type_name (value->g_type));
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
print_props (GParameter *params, guint n_params, guint offset)
|
|
{
|
|
guint i;
|
|
char *fill = g_strnfill (offset, ' ');
|
|
for (i = 0; i < n_params; ++i)
|
|
{
|
|
GParameter *param = ¶ms[i];
|
|
g_print ("%s%s = '", fill, param->name);
|
|
print_value (¶m->value);
|
|
g_print ("'\n");
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
dump_widget (Widget *widget)
|
|
{
|
|
print_widget (widget, 0);
|
|
}
|
|
|
|
#else
|
|
static void
|
|
dump_widget (G_GNUC_UNUSED Widget *widget)
|
|
{
|
|
}
|
|
#endif
|