summaryrefslogtreecommitdiff
path: root/trunk/gui/greeter/greeter_parser.c
diff options
context:
space:
mode:
Diffstat (limited to 'trunk/gui/greeter/greeter_parser.c')
-rw-r--r--trunk/gui/greeter/greeter_parser.c1997
1 files changed, 1997 insertions, 0 deletions
diff --git a/trunk/gui/greeter/greeter_parser.c b/trunk/gui/greeter/greeter_parser.c
new file mode 100644
index 00000000..2a131fc0
--- /dev/null
+++ b/trunk/gui/greeter/greeter_parser.c
@@ -0,0 +1,1997 @@
+/* GDM - The GNOME Display Manager
+ * Copyright (C) 1998, 1999, 2000 Martin K. Petersen <mkp@mkp.net>
+ *
+ * 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.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ */
+
+#include "config.h"
+
+#include <gtk/gtk.h>
+#include <libxml/parser.h>
+#include <string.h>
+#include <stdlib.h>
+#include <librsvg/rsvg.h>
+#include <math.h>
+#include <locale.h>
+#include <glib/gi18n.h>
+#include <gdk/gdkx.h>
+#include <syslog.h>
+
+#include "gdmwm.h"
+#include "gdmcommon.h"
+#include "gdmconfig.h"
+
+#include "gdm-common.h"
+#include "gdm-daemon-config-keys.h"
+
+#include "greeter_configuration.h"
+#include "greeter_parser.h"
+#include "greeter_events.h"
+#include "gdm.h"
+
+/* FIXME: hack */
+extern GreeterItemInfo *welcome_string_info;
+
+/* evil globals */
+static char *file_search_path = NULL;
+static GList *button_stack = NULL;
+
+static GHashTable *pixbuf_hash = NULL;
+
+GHashTable *item_hash = NULL;
+GList *custom_items = NULL;
+
+static gboolean parse_items (xmlNodePtr node,
+ GList **items_out,
+ GreeterItemInfo *parent,
+ GError **error);
+
+static GdkPixbuf *
+load_pixbuf (const char *fname, GError **error)
+{
+ GdkPixbuf *pb;
+
+ if (pixbuf_hash == NULL)
+ pixbuf_hash = g_hash_table_new_full (g_str_hash,
+ g_str_equal,
+ g_free,
+ (GDestroyNotify)g_object_unref);
+ pb = g_hash_table_lookup (pixbuf_hash, fname);
+ if (pb != NULL)
+ return g_object_ref (pb);
+
+ pb = gdk_pixbuf_new_from_file (fname, error);
+ if G_UNLIKELY (pb == NULL)
+ return NULL;
+
+ g_hash_table_insert (pixbuf_hash, g_strdup (fname), g_object_ref (pb));
+
+ return pb;
+}
+
+GQuark
+greeter_parser_error_quark (void)
+{
+ static GQuark quark = 0;
+ if (!quark)
+ quark = g_quark_from_static_string ("greeter_parser_error");
+
+ return quark;
+}
+
+
+GreeterItemInfo *
+greeter_lookup_id (const char *id)
+{
+ GreeterItemInfo key;
+ GreeterItemInfo *info;
+
+ key.id = (char *)id;
+ info = g_hash_table_lookup (item_hash, &key);
+
+ return info;
+}
+
+static void
+parse_id (xmlNodePtr node,
+ GreeterItemInfo *info)
+{
+ xmlChar *prop;
+
+ prop = xmlGetProp (node, (const xmlChar *) "id");
+
+ if (prop)
+ {
+ info->id = g_strdup ((char *) prop);
+ g_hash_table_insert (item_hash, info, info);
+ xmlFree (prop);
+ }
+}
+
+/* Doesn't set the parts of rect that are not specified.
+ * If you want specific default values you need to fill them out
+ * in rect first
+ */
+static gboolean
+parse_pos (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlChar *prop;
+ char *p;
+
+ prop = xmlGetProp (node, (const xmlChar *) "anchor");
+ if (prop)
+ {
+ if (strcmp ((char *) prop, "center") == 0)
+ info->anchor = GTK_ANCHOR_CENTER;
+ else if (strcmp ((char *) prop, "c") == 0)
+ info->anchor = GTK_ANCHOR_CENTER;
+ else if (strcmp ((char *) prop, "nw") == 0)
+ info->anchor = GTK_ANCHOR_NW;
+ else if (strcmp ((char *) prop, "n") == 0)
+ info->anchor = GTK_ANCHOR_N;
+ else if (strcmp ((char *) prop, "ne") == 0)
+ info->anchor = GTK_ANCHOR_NE;
+ else if (strcmp ((char *) prop, "w") == 0)
+ info->anchor = GTK_ANCHOR_W;
+ else if (strcmp ((char *) prop, "e") == 0)
+ info->anchor = GTK_ANCHOR_E;
+ else if (strcmp ((char *) prop, "sw") == 0)
+ info->anchor = GTK_ANCHOR_SW;
+ else if (strcmp ((char *) prop, "s") == 0)
+ info->anchor = GTK_ANCHOR_S;
+ else if (strcmp ((char *) prop, "se") == 0)
+ info->anchor = GTK_ANCHOR_SE;
+ else
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Unknown anchor type %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+
+
+ prop = xmlGetProp (node,(const xmlChar *) "x");
+ if (prop)
+ {
+ info->x = g_ascii_strtod ((char *) prop, &p);
+
+ if ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad position specifier %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+
+ if (prop[0] == '-' || info->x < 0)
+ info->x_negative = TRUE;
+ else
+ info->x_negative = FALSE;
+
+ if (strchr ((char *) prop, '%') != NULL)
+ info->x_type = GREETER_ITEM_POS_RELATIVE;
+ else
+ info->x_type = GREETER_ITEM_POS_ABSOLUTE;
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "y");
+ if (prop)
+ {
+ info->y = g_ascii_strtod ((char *) prop, &p);
+
+ if ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad position specifier %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+
+ if (prop[0] == '-' || info->y < 0)
+ info->y_negative = TRUE;
+ else
+ info->y_negative = FALSE;
+
+ if (strchr ((char *) prop, '%') != NULL)
+ info->y_type = GREETER_ITEM_POS_RELATIVE;
+ else
+ info->y_type = GREETER_ITEM_POS_ABSOLUTE;
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "width");
+ if (prop)
+ {
+ if (strcmp ((char *) prop, "box") == 0)
+ info->width_type = GREETER_ITEM_SIZE_BOX;
+ else if (strcmp ((char *) prop, "scale") == 0)
+ info->width_type = GREETER_ITEM_SIZE_SCALE;
+ else
+ {
+ info->width = g_ascii_strtod ((char *) prop, &p);
+
+ if ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad size specifier %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+
+ if (strchr ((char *) prop, '%') != NULL)
+ info->width_type = GREETER_ITEM_SIZE_RELATIVE;
+ else
+ info->width_type = GREETER_ITEM_SIZE_ABSOLUTE;
+ }
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "height");
+ if (prop)
+ {
+ if (strcmp ((char *) prop, "box") == 0)
+ info->height_type = GREETER_ITEM_SIZE_BOX;
+ else if (strcmp ((char *) prop, "scale") == 0)
+ info->height_type = GREETER_ITEM_SIZE_SCALE;
+ else
+ {
+ info->height = g_ascii_strtod ((char *) prop, &p);
+
+ if ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad size specifier %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+
+ if (strchr ((char *) prop, '%') != NULL)
+ info->height_type = GREETER_ITEM_SIZE_RELATIVE;
+ else
+ info->height_type = GREETER_ITEM_SIZE_ABSOLUTE;
+ }
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "expand");
+ if (prop)
+ {
+ if (strcmp ((char *) prop, "true") == 0)
+ {
+ info->expand = TRUE;
+ }
+ else if (strcmp ((char *) prop, "false") == 0)
+ {
+ info->expand = FALSE;
+ }
+ else
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad expand spec %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+
+ xmlFree (prop);
+ }
+
+ return TRUE;
+}
+
+/* We pass the same arguments as to translated text, since we'll override it
+ * with translation score */
+static gboolean
+parse_stock (xmlNodePtr node,
+ GreeterItemInfo *info,
+ char **translated_text,
+ gint *translation_score,
+ GError **error)
+{
+ xmlChar *prop;
+
+ prop = xmlGetProp (node,(const xmlChar *) "type");
+ if (prop)
+ {
+ if (g_ascii_strcasecmp ((char *) prop, "language") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("_Language"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "session") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("_Session"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "system") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("_Actions"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "disconnect") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("D_isconnect"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "quit") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("_Quit"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "halt") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("Shut _Down"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "suspend") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("Sus_pend"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "reboot") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("_Restart"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "chooser") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("Remote Login via _XDMCP"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "config") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("Confi_gure"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "options") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("Op_tions"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "caps-lock-warning") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("Caps Lock is on."));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "timed-label") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("User %u will login in %t"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "welcome-label") == 0)
+ {
+ /* FIXME: hack */
+ welcome_string_info = info;
+
+ g_free (*translated_text);
+ *translated_text = gdm_common_get_welcomemsg ();
+ }
+ /* FIXME: is this actually needed? */
+ else if (g_ascii_strcasecmp ((char *) prop, "username-label") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("Username:"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "ok") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("_OK"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "cancel") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("_Cancel"));
+ }
+ else if (g_ascii_strcasecmp ((char *) prop, "startagain") == 0)
+ {
+ g_free (*translated_text);
+ *translated_text = g_strdup (_("_Start Again"));
+ }
+ else
+ {
+ gboolean is_error = TRUE;
+ register int i = 0;
+ for (; i < GDM_CUSTOM_COMMAND_MAX; i++) {
+ gchar * key_string = NULL;
+ key_string = g_strdup_printf ("custom_cmd%d", i);
+ if (g_ascii_strcasecmp ((char *) prop, key_string) == 0) {
+ g_free (*translated_text);
+ g_free (key_string);
+ key_string = g_strdup_printf ("%s%d=", GDM_KEY_CUSTOM_CMD_LABEL_TEMPLATE, i);
+ *translated_text = g_strdup(gdm_config_get_string (key_string));
+ g_free (key_string);
+ is_error = FALSE;
+ break;
+ }
+ g_free (key_string);
+ }
+
+ if (is_error)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad stock label type");
+ xmlFree (prop);
+ return FALSE;
+ }
+ }
+
+ /* This is the very very very best "translation" */
+ *translation_score = -1;
+
+ xmlFree (prop);
+
+ return TRUE;
+ }
+ else
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Stock type not specified");
+ return FALSE;
+ }
+}
+
+static void
+do_font_size_reduction (GreeterItemInfo *info)
+{
+ double size_reduction = 1.0;
+ int i;
+
+ if (gdm_wm_screen.width <= 800 &&
+ gdm_wm_screen.width > 640)
+ size_reduction = PANGO_SCALE_SMALL;
+ else if (gdm_wm_screen.width <= 640)
+ size_reduction = PANGO_SCALE_X_SMALL;
+
+ if (size_reduction < 0.99)
+ {
+ for (i = 0; i < GREETER_ITEM_STATE_MAX; i++)
+ {
+ if (info->data.text.fonts[i] != NULL)
+ {
+ int old_size = pango_font_description_get_size (info->data.text.fonts[i]);
+ pango_font_description_set_size (info->data.text.fonts[i], old_size * size_reduction);
+ }
+ }
+ }
+}
+
+
+static gboolean
+parse_canvasbutton (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlChar *prop;
+
+ prop = xmlGetProp (node,(const xmlChar *) "button");
+ if (prop)
+ {
+ if (strcmp ((char *) prop, "true") == 0)
+ {
+ info->canvasbutton = TRUE;
+ }
+ else if (strcmp ((char *) prop, "false") == 0)
+ {
+ info->canvasbutton = FALSE;
+ }
+ else
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "bad button spec %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+ return TRUE;
+}
+
+static gboolean
+parse_gtkbutton (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlNodePtr child;
+ char *translated_text = NULL;
+ gint translation_score = 1000;
+
+ child = node->children;
+
+ while (child)
+ {
+ if (strcmp ((char *) child->name, "pos") == 0)
+ {
+ if G_UNLIKELY (!parse_pos (child, info, error))
+ return FALSE;
+ }
+ else if (child->type == XML_ELEMENT_NODE &&
+ strcmp ((char *) child->name, "stock") == 0)
+ {
+ if G_UNLIKELY (!parse_stock (child, info, &translated_text, &translation_score, error))
+ return FALSE;
+ }
+
+ child = child->next;
+ }
+
+ if (translated_text == NULL)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "A label must specify the text attribute");
+ return FALSE;
+ }
+
+ /* FIXME: evil hack to use internally translated strings */
+ if (translation_score == 999 &&
+ ! ve_string_empty (translated_text))
+ {
+ char *foo = g_strdup (_(translated_text));
+ g_free (translated_text);
+ translated_text = foo;
+ }
+
+ info->data.text.orig_text = translated_text;
+
+ return TRUE;
+}
+
+static gboolean
+parse_show (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlChar *prop;
+ char **argv = NULL;
+ int i;
+
+ prop = xmlGetProp (node,(const xmlChar *) "type");
+ if (prop != NULL)
+ {
+ g_free (info->show_type);
+ info->show_type = g_strdup ((char *) prop);
+ xmlFree (prop);
+ }
+
+ /* Note: subtype is deprecated, use type only */
+ prop = xmlGetProp (node,(const xmlChar *) "subtype");
+ if G_UNLIKELY (prop != NULL)
+ {
+ /* code for legacy uses of subtype only, are there any such
+ * themes out there? The Bluecurve was the one this was made
+ * for and bluecurve is NOT using it. */
+ if (info->show_type == NULL ||
+ strcmp (info->show_type, "system") == 0) {
+ g_free (info->show_type);
+ info->show_type = g_strdup ((char *) prop);
+ }
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "min-screen-width");
+ if (prop != NULL)
+ {
+ g_warning ("minimum width is %d", info->minimum_required_screen_height);
+ info->minimum_required_screen_width = atoi ((char *) prop);
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "min-screen-height");
+ if (prop != NULL)
+ {
+ info->minimum_required_screen_height = atoi ((char *) prop);
+ g_warning ("minimum height is %d", info->minimum_required_screen_height);
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "modes");
+ if (prop != NULL)
+ {
+ if (strcmp ((char *) prop, "everywhere") == 0)
+ {
+ info->show_modes = GREETER_ITEM_SHOW_EVERYWHERE;
+ xmlFree (prop);
+ return TRUE;
+ }
+ else if (strcmp ((char *) prop, "nowhere") == 0)
+ {
+ info->show_modes = GREETER_ITEM_SHOW_NOWHERE;
+ xmlFree (prop);
+ return TRUE;
+ }
+
+ argv = g_strsplit ((char *) prop, ",", 0);
+ xmlFree (prop);
+ }
+ else
+ {
+ info->show_modes = GREETER_ITEM_SHOW_EVERYWHERE;
+ return TRUE;
+ }
+
+ info->show_modes = GREETER_ITEM_SHOW_NOWHERE;
+
+ if (argv != NULL)
+ {
+ for (i = 0; argv[i] != NULL; i++)
+ {
+ if (strcmp (argv[i], "console") == 0)
+ {
+ info->show_modes |= GREETER_ITEM_SHOW_CONSOLE;
+ }
+ else if (strcmp (argv[i], "console-fixed") == 0)
+ {
+ info->show_modes |= GREETER_ITEM_SHOW_CONSOLE_FIXED;
+ }
+ else if (strcmp (argv[i], "console-flexi") == 0)
+ {
+ info->show_modes |= GREETER_ITEM_SHOW_CONSOLE_FLEXI;
+ }
+ else if (strcmp (argv[i], "remote-flexi") == 0)
+ {
+ info->show_modes |= GREETER_ITEM_SHOW_REMOTE_FLEXI;
+ }
+ else if (strcmp (argv[i], "flexi") == 0)
+ {
+ info->show_modes |= GREETER_ITEM_SHOW_FLEXI;
+ }
+ else if (strcmp (argv[i], "remote") == 0)
+ {
+ info->show_modes |= GREETER_ITEM_SHOW_REMOTE;
+ }
+ }
+ g_strfreev (argv);
+ }
+ return TRUE;
+}
+
+static gboolean
+parse_fixed (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ return parse_items (node,
+ &info->fixed_children,
+ info,
+ error);
+}
+
+static gboolean
+parse_box (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlChar *prop;
+ char *p;
+
+ prop = xmlGetProp (node,(const xmlChar *) "orientation");
+ if (prop)
+ {
+ if (strcmp ((char *) prop, "horizontal") == 0)
+ {
+ info->box_orientation = GTK_ORIENTATION_HORIZONTAL;
+ }
+ else if (strcmp ((char *) prop, "vertical") == 0)
+ {
+ info->box_orientation = GTK_ORIENTATION_VERTICAL;
+ }
+ else
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad orientation %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "homogeneous");
+ if (prop)
+ {
+ if (strcmp ((char *) prop, "true") == 0)
+ {
+ info->box_homogeneous = TRUE;
+ }
+ else if (strcmp ((char *) prop, "false") == 0)
+ {
+ info->box_homogeneous = FALSE;
+ }
+ else
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad homogenous spec %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+
+ xmlFree (prop);
+ }
+
+
+ prop = xmlGetProp (node,(const xmlChar *) "xpadding");
+ if (prop)
+ {
+ info->box_x_padding = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad padding specification %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "ypadding");
+ if (prop)
+ {
+ info->box_y_padding = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad padding specification %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "min-width");
+ if (prop)
+ {
+ info->box_min_width = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad min-width specification %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "min-height");
+ if (prop)
+ {
+ info->box_min_height = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad min-height specification %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "spacing");
+ if (prop)
+ {
+ info->box_spacing = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad spacing specification %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+
+ return parse_items (node,
+ &info->box_children,
+ info,
+ error);
+
+}
+
+
+static gboolean
+parse_color (const char *str,
+ guint32 *col_out,
+ GError **error)
+{
+ guint32 col;
+ int i;
+ if G_UNLIKELY (str[0] != '#')
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "colors must start with #, %s is an invalid color", str);
+ return FALSE;
+ }
+ if G_UNLIKELY (strlen (str) != 7)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Colors must be on the format #xxxxxx, %s is an invalid color", str);
+ return FALSE;
+ }
+
+ col = 0;
+
+ for (i = 0; i < 6; i++)
+ col = (col << 4) | g_ascii_xdigit_value (str[i+1]);
+
+ *col_out = col;
+
+ return TRUE;
+}
+
+static gboolean
+parse_state_file_pixmap (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GreeterItemState state,
+ GError **error)
+{
+ xmlChar *prop;
+ char *p;
+
+ info->have_state |= (1<<state);
+
+ prop = xmlGetProp (node,(const xmlChar *) "file");
+ if (prop)
+ {
+ if (g_path_is_absolute ((char *) prop))
+ info->data.pixmap.files[state] = g_strdup ((char *) prop);
+ else
+ info->data.pixmap.files[state] = g_build_filename (file_search_path,
+ (char *) prop,
+ NULL);
+
+ xmlFree (prop);
+ }
+
+ {
+ int i = 1;
+ char *altfile_prop_name = g_strdup_printf ("altfile%d", i);
+
+ prop = xmlGetProp (node,(const xmlChar *) altfile_prop_name);
+ while (prop)
+ {
+ char *filename = NULL;
+ if (g_path_is_absolute ((char *) prop))
+ filename = g_strdup ((char *) prop);
+ else
+ filename = g_build_filename (file_search_path,
+ (char *) prop,
+ NULL);
+
+ if (g_file_test (filename, G_FILE_TEST_EXISTS))
+ {
+ if (info->data.pixmap.files[state])
+ g_free (info->data.pixmap.files[state]);
+ info->data.pixmap.files[state] = filename;
+ }
+ xmlFree (prop);
+ g_free (altfile_prop_name);
+
+ i++;
+ altfile_prop_name = g_strdup_printf ("altfile%d", i);
+ prop = xmlGetProp (node,(const xmlChar *) altfile_prop_name);
+ }
+ g_free (altfile_prop_name);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "tint");
+ if (prop)
+ {
+ if (!parse_color ((char *) prop, &info->data.pixmap.tints[state], error))
+ return FALSE;
+ info->data.pixmap.have_tint |= (1<<state);
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "alpha");
+ if (prop)
+ {
+ double alpha = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad alpha specifier format %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+
+ if (alpha >= 1.0)
+ info->data.pixmap.alphas[state] = 0xff;
+ else if (alpha < 0)
+ info->data.pixmap.alphas[state] = 0;
+ else
+ info->data.pixmap.alphas[state] = floor (alpha * 0xff);
+ }
+
+ return TRUE;
+}
+
+static gboolean
+parse_state_color_rect (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GreeterItemState state,
+ GError **error)
+{
+ xmlChar *prop;
+ char *p;
+
+ info->have_state |= (1<<state);
+
+ prop = xmlGetProp (node,(const xmlChar *) "color");
+ if (prop)
+ {
+ if G_UNLIKELY (!parse_color ((char *) prop, &info->data.rect.colors[state], error))
+ return FALSE;
+ info->data.rect.have_color |= (1<<state);
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "alpha");
+ if (prop)
+ {
+ double alpha = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad alpha specifier format %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+
+ if (alpha >= 1.0)
+ info->data.rect.alphas[state] = 0xff;
+ else if (alpha < 0)
+ info->data.rect.alphas[state] = 0;
+ else
+ info->data.rect.alphas[state] = floor (alpha * 0xff);
+ }
+
+ return TRUE;
+}
+
+static gboolean
+parse_color_list (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlChar *prop;
+ guint32 color;
+
+ prop = xmlGetProp (node,(const xmlChar *) "iconcolor");
+ if (prop)
+ {
+ if G_UNLIKELY (!parse_color ((char *) prop, &color, error)) {
+ info->data.list.icon_color = NULL;
+ return FALSE;
+ } else {
+ info->data.list.icon_color = g_strdup ((char *) prop);
+ }
+
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "labelcolor");
+ if (prop)
+ {
+ if G_UNLIKELY (!parse_color ((char *) prop, &color, error)) {
+ info->data.list.label_color = NULL;
+ return FALSE;
+ } else {
+ info->data.list.label_color = g_strdup ((char *) prop);
+ }
+
+ xmlFree (prop);
+ }
+
+ return TRUE;
+}
+
+static gboolean
+parse_pixmap (xmlNodePtr node,
+ gboolean svg,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlNodePtr child;
+ int i;
+
+ child = node->children;
+
+ while (child)
+ {
+ if (strcmp ((char *) child->name, "normal") == 0)
+ {
+ if G_UNLIKELY (!parse_state_file_pixmap (child, info, GREETER_ITEM_STATE_NORMAL, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "prelight") == 0)
+ {
+ if G_UNLIKELY (!parse_state_file_pixmap (child, info, GREETER_ITEM_STATE_PRELIGHT, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "active") == 0)
+ {
+ if G_UNLIKELY (!parse_state_file_pixmap (child, info, GREETER_ITEM_STATE_ACTIVE, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "pos") == 0)
+ {
+ if G_UNLIKELY (!parse_pos (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "fixed") == 0)
+ {
+ if G_UNLIKELY (!parse_fixed (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "box") == 0)
+ {
+ if G_UNLIKELY (!parse_box (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "show") == 0)
+ {
+ if G_UNLIKELY (!parse_show (child, info, error))
+ return FALSE;
+ }
+
+ child = child->next;
+ }
+
+ if G_UNLIKELY (!info->data.pixmap.files[GREETER_ITEM_STATE_NORMAL])
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "No filename specified for normal state");
+ return FALSE;
+ }
+
+ if (!svg)
+ {
+ for (i = 0; i < GREETER_ITEM_STATE_MAX; i++)
+ {
+ if (info->data.pixmap.files[i] != NULL)
+ {
+ info->data.pixmap.pixbufs[i] = load_pixbuf (info->data.pixmap.files[i], error);
+
+ if G_UNLIKELY (info->data.pixmap.pixbufs[i] == NULL)
+ return FALSE;
+ }
+ else
+ info->data.pixmap.pixbufs[i] = NULL;
+ }
+ }
+
+ return TRUE;
+}
+
+static gboolean
+parse_rect (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlNodePtr child;
+ int i;
+
+ child = node->children;
+
+ while (child)
+ {
+ if (strcmp ((char *) child->name, "normal") == 0)
+ {
+ if G_UNLIKELY (!parse_state_color_rect (child, info, GREETER_ITEM_STATE_NORMAL, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "prelight") == 0)
+ {
+ if G_UNLIKELY (!parse_state_color_rect (child, info, GREETER_ITEM_STATE_PRELIGHT, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "active") == 0)
+ {
+ if G_UNLIKELY (!parse_state_color_rect (child, info, GREETER_ITEM_STATE_ACTIVE, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "pos") == 0)
+ {
+ if G_UNLIKELY (!parse_pos (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "fixed") == 0)
+ {
+ if G_UNLIKELY (!parse_fixed (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "box") == 0)
+ {
+ if G_UNLIKELY (!parse_box (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "show") == 0)
+ {
+ if G_UNLIKELY (!parse_show (child, info, error))
+ return FALSE;
+ }
+
+ child = child->next;
+ }
+
+ for (i = 0; i < GREETER_ITEM_STATE_MAX; i++)
+ {
+ if ( ! (info->data.rect.have_color & (1<<i)))
+ continue;
+
+ info->data.rect.colors[i] = (info->data.rect.colors[i] << 8) | (guint) info->data.rect.alphas[i];
+ }
+
+ return TRUE;
+}
+
+
+static gboolean
+parse_state_text (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GreeterItemState state,
+ GError **error)
+{
+ xmlChar *prop;
+ char *p;
+
+ info->have_state |= (1<<state);
+
+ prop = xmlGetProp (node,(const xmlChar *) "font");
+ if (prop)
+ {
+ info->data.text.fonts[state] = pango_font_description_from_string ((char *) prop);
+ if G_UNLIKELY (info->data.text.fonts[state] == NULL)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad font specification %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "color");
+ if (prop)
+ {
+ if G_UNLIKELY (!parse_color ((char *) prop, &info->data.text.colors[state], error))
+ return FALSE;
+ info->data.text.have_color |= (1<<state);
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "alpha");
+ if (prop)
+ {
+ double alpha = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad alpha specifier format %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+
+ if (alpha >= 1.0)
+ info->data.rect.alphas[state] = 0xff;
+ else if (alpha < 0)
+ info->data.rect.alphas[state] = 0;
+ else
+ info->data.rect.alphas[state] = floor (alpha * 0xff);
+ }
+
+ return TRUE;
+}
+
+static gint
+is_current_locale (const char *lang)
+{
+ const char * const *langs;
+ int score = 0;
+ int i;
+
+ langs = g_get_language_names ();
+
+ for (i = 0; langs[i] != NULL; i++)
+ {
+ if (strcmp (langs[i], lang) == 0)
+ return score;
+
+ score++;
+ }
+ return 1000;
+}
+
+static gboolean
+parse_translated_text (xmlNodePtr node,
+ char **translated_text,
+ gint *translation_score,
+ GError **error)
+{
+ xmlChar *text;
+ xmlChar *prop;
+ gint score;
+
+ prop = xmlNodeGetLang (node);
+ if (prop)
+ {
+ score = is_current_locale ((char *) prop);
+ xmlFree (prop);
+ } else
+ score = 999;
+
+ if (score >= *translation_score)
+ return TRUE;
+
+ text = xmlNodeGetContent (node);
+ if (text == NULL)
+ {
+ /* This is empty text */
+ *translation_score = score;
+ if (*translated_text)
+ g_free (*translated_text);
+ *translated_text = g_strdup ("");
+
+ return TRUE;
+ }
+
+ *translation_score = score;
+ if (*translated_text)
+ g_free (*translated_text);
+ *translated_text = g_strdup ((char *) text);
+
+ xmlFree (text);
+
+ return TRUE;
+}
+
+static gboolean
+parse_label_pos_extras (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlChar *prop;
+ char *p;
+
+ prop = xmlGetProp (node,(const xmlChar *) "max-width");
+ if (prop)
+ {
+ info->data.text.max_width = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad max-width specification %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+
+ prop = xmlGetProp (node,(const xmlChar *) "max-screen-percent-width");
+ if (prop)
+ {
+ info->data.text.max_screen_percent_width = g_ascii_strtod ((char *) prop, &p);
+
+ if G_UNLIKELY ((char *)prop == p)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad max-screen-percent-width specification %s", prop);
+ xmlFree (prop);
+ return FALSE;
+ }
+ xmlFree (prop);
+ }
+
+ return TRUE;
+}
+
+
+static gboolean
+parse_label (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlNodePtr child;
+ int i;
+ char *translated_text = NULL;
+ gint translation_score = 1000;
+
+ child = node->children;
+ while (child)
+ {
+ if (strcmp ((char *) child->name, "normal") == 0)
+ {
+ if G_UNLIKELY (!parse_state_text (child, info, GREETER_ITEM_STATE_NORMAL, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "prelight") == 0)
+ {
+ if G_UNLIKELY (!parse_state_text (child, info, GREETER_ITEM_STATE_PRELIGHT, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "active") == 0)
+ {
+ if G_UNLIKELY (!parse_state_text (child, info, GREETER_ITEM_STATE_ACTIVE, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "pos") == 0)
+ {
+ if G_UNLIKELY (!parse_pos (child, info, error))
+ return FALSE;
+ if G_UNLIKELY (!parse_label_pos_extras (child, info, error))
+ return FALSE;
+ }
+ else if (child->type == XML_ELEMENT_NODE &&
+ strcmp ((char *) child->name, "text") == 0)
+ {
+ if G_UNLIKELY (!parse_translated_text (child, &translated_text, &translation_score, error))
+ return FALSE;
+ }
+ else if (child->type == XML_ELEMENT_NODE &&
+ strcmp ((char *) child->name, "stock") == 0)
+ {
+ if G_UNLIKELY (!parse_stock (child, info, &translated_text, &translation_score, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "show") == 0)
+ {
+ if G_UNLIKELY (!parse_show (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "fixed") == 0 ||
+ strcmp ((char *) child->name, "box") == 0)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Label items cannot have children");
+ return FALSE;
+ }
+
+ child = child->next;
+ }
+
+ if (translated_text == NULL)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "A label must specify the text attribute");
+ return FALSE;
+ }
+ /* FIXME: evil hack to use internally translated strings */
+ if (translation_score == 999 &&
+ ! ve_string_empty (translated_text))
+ {
+ char *foo = g_strdup (_(translated_text));
+ g_free (translated_text);
+ translated_text = foo;
+ }
+
+ for (i = 0; i < GREETER_ITEM_STATE_MAX; i++)
+ {
+ if ( ! (info->data.text.have_color & (1<<i)))
+ continue;
+
+ info->data.text.colors[i] = (info->data.text.colors[i] << 8) | (guint) info->data.text.alphas[i];
+ }
+
+ if (info->data.text.fonts[GREETER_ITEM_STATE_NORMAL] == NULL)
+ info->data.text.fonts[GREETER_ITEM_STATE_NORMAL] = pango_font_description_from_string ("Sans");
+
+ do_font_size_reduction (info);
+
+ info->data.text.orig_text = translated_text;
+
+ return TRUE;
+}
+
+static gboolean
+parse_listitem (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlNodePtr child;
+ xmlChar *prop;
+ GreeterItemListItem *li;
+ char *translated_text = NULL;
+ gint translation_score = 1000;
+
+ prop = xmlGetProp (node,(const xmlChar *) "id");
+
+ if G_LIKELY (prop)
+ {
+ li = g_new0 (GreeterItemListItem, 1);
+ li->id = g_strdup ((char *) prop);
+ xmlFree (prop);
+ }
+ else
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Listitem id not specified");
+ return FALSE;
+ }
+
+ child = node->children;
+ while (child)
+ {
+ if (child->type == XML_ELEMENT_NODE &&
+ strcmp ((char *) child->name, "text") == 0)
+ {
+ if G_UNLIKELY ( ! parse_translated_text (child, &translated_text, &translation_score, error))
+ {
+ g_free (li->id);
+ g_free (li);
+ return FALSE;
+ }
+ }
+
+ child = child->next;
+ }
+
+ if G_UNLIKELY (translated_text == NULL)
+ {
+ g_free (li->id);
+ g_free (li);
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "A list item must specify the text attribute");
+ return FALSE;
+ }
+ li->text = translated_text;
+
+ info->data.list.items = g_list_append (info->data.list.items, li);
+
+ return TRUE;
+}
+
+static gboolean
+parse_list (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlNodePtr child;
+ xmlChar *prop;
+
+ info->data.list.combo_type = FALSE;
+ prop = xmlGetProp (node,(const xmlChar *) "combo");
+ if (prop)
+ {
+ if (strcmp ((char *) prop, "true") == 0)
+ {
+ info->data.list.combo_type = TRUE;
+ }
+ else if (strcmp ((char *) prop, "false") == 0)
+ {
+ info->data.list.combo_type = FALSE;
+ }
+ xmlFree (prop);
+ }
+
+ child = node->children;
+ while (child)
+ {
+ if (strcmp ((char *) child->name, "color") == 0)
+ {
+ if G_UNLIKELY (!parse_color_list (child, info, error))
+ return FALSE;
+ }
+ if (strcmp ((char *) child->name, "pos") == 0)
+ {
+ if G_UNLIKELY (!parse_pos (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "show") == 0)
+ {
+ if G_UNLIKELY (!parse_show (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "listitem") == 0)
+ {
+ if G_UNLIKELY ( ! parse_listitem (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "fixed") == 0 ||
+ strcmp ((char *) child->name, "box") == 0)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "List items cannot have children");
+ return FALSE;
+ }
+
+ child = child->next;
+ }
+
+ if ((strcmp (info->id, "userlist") == 0) && (info->data.list.combo_type == TRUE)) {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "userlist doest not support combo style");
+ return FALSE;
+ } else if (info->data.list.items != NULL) {
+
+ if G_UNLIKELY (strcmp (info->id, "userlist") == 0 ||
+ strcmp (info->id, "session") == 0 ||
+ strcmp (info->id, "language") == 0) {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "List of id userlist, session, and language cannot have custom list items");
+ return FALSE;
+ }
+ custom_items = g_list_append (custom_items, info);
+
+ } else if (strcmp (info->id, "session") == 0 ||
+ strcmp (info->id, "language") == 0) {
+ custom_items = g_list_append (custom_items, info);
+ }
+
+ return TRUE;
+}
+
+static gboolean
+parse_entry (xmlNodePtr node,
+ GreeterItemInfo *info,
+ GError **error)
+{
+ xmlNodePtr child;
+
+ child = node->children;
+ while (child)
+ {
+ if (strcmp ((char *) child->name, "normal") == 0)
+ {
+ if G_UNLIKELY (!parse_state_text (child, info, GREETER_ITEM_STATE_NORMAL, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "pos") == 0)
+ {
+ if G_UNLIKELY (!parse_pos (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "show") == 0)
+ {
+ if G_UNLIKELY (!parse_show (child, info, error))
+ return FALSE;
+ }
+ else if (strcmp ((char *) child->name, "fixed") == 0 ||
+ strcmp ((char *) child->name, "box") == 0)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Entry items cannot have children");
+ return FALSE;
+ }
+
+ child = child->next;
+ }
+
+ do_font_size_reduction (info);
+
+ return TRUE;
+}
+
+static gboolean
+parse_items (xmlNodePtr node,
+ GList **items_out,
+ GreeterItemInfo *parent,
+ GError **error)
+{
+ xmlNodePtr child;
+ GList *items;
+ gboolean res;
+ xmlChar *type;
+ xmlChar *background;
+ GreeterItemInfo *info;
+ GreeterItemType item_type;
+
+ *items_out = NULL;
+
+ items = NULL;
+
+ child = node->children;
+ while (child)
+ {
+ if (child->type == XML_ELEMENT_NODE)
+ {
+ if G_UNLIKELY (strcmp ((char *) child->name, "item") != 0)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Found tag %s when looking for item", child->name);
+ return FALSE;
+ }
+
+ type = xmlGetProp (child, (const xmlChar *) "type");
+ if G_UNLIKELY (!type)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Items must specify their type");
+ return FALSE;
+ }
+
+ if (strcmp ((char *) type, "svg") == 0)
+ item_type = GREETER_ITEM_TYPE_SVG;
+ else if (strcmp ((char *) type, "pixmap") == 0)
+ item_type = GREETER_ITEM_TYPE_PIXMAP;
+ else if (strcmp ((char *) type, "rect") == 0)
+ item_type = GREETER_ITEM_TYPE_RECT;
+ else if (strcmp ((char *) type, "label") == 0)
+ item_type = GREETER_ITEM_TYPE_LABEL;
+ else if (strcmp ((char *) type, "entry") == 0)
+ item_type = GREETER_ITEM_TYPE_ENTRY;
+ else if (strcmp ((char *) type, "list") == 0)
+ item_type = GREETER_ITEM_TYPE_LIST;
+ else if (strcmp ((char *) type, "button") == 0)
+ item_type = GREETER_ITEM_TYPE_BUTTON;
+ else
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Unknown item type %s", type);
+ xmlFree (type);
+ return FALSE;
+ }
+
+ xmlFree (type);
+
+ info = greeter_item_info_new (parent, item_type);
+
+ parse_id (child, info);
+ if G_UNLIKELY ( ! parse_canvasbutton (child, info, error))
+ return FALSE;
+
+ if (button_stack != NULL)
+ info->my_button = button_stack->data;
+ if (info->canvasbutton)
+ button_stack = g_list_prepend (button_stack, info);
+
+ switch (item_type)
+ {
+ case GREETER_ITEM_TYPE_SVG:
+ res = parse_pixmap (child, TRUE, info, error);
+ break;
+ case GREETER_ITEM_TYPE_PIXMAP:
+ res = parse_pixmap (child, FALSE, info, error);
+ break;
+ case GREETER_ITEM_TYPE_RECT:
+ res = parse_rect (child, info, error);
+ break;
+ case GREETER_ITEM_TYPE_LABEL:
+ res = parse_label (child, info, error);
+ break;
+ case GREETER_ITEM_TYPE_ENTRY:
+ res = parse_entry (child, info, error);
+ break;
+ case GREETER_ITEM_TYPE_LIST:
+ res = parse_list (child, info, error);
+ break;
+ case GREETER_ITEM_TYPE_BUTTON:
+ res = parse_gtkbutton (child, info, error);
+ break;
+ default:
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_SPEC,
+ "Bad item type");
+ res = FALSE;
+ }
+
+ if (info->canvasbutton)
+ button_stack = g_list_remove (button_stack, info);
+
+ background = xmlGetProp (child, (const xmlChar *) "background");
+ if G_UNLIKELY (background)
+ {
+ if (strcmp ((char *) background, "true") == 0)
+ {
+ info->background = TRUE;
+ }
+ else if (strcmp ((char *) background, "false") == 0)
+ {
+ info->background = FALSE;
+ }
+ xmlFree (background);
+ }
+
+ if G_UNLIKELY (!res)
+ return FALSE;
+
+ items = g_list_prepend (items, info);
+
+ }
+ child = child->next;
+ }
+
+ *items_out = g_list_reverse (items);
+ return TRUE;
+}
+
+static gboolean
+greeter_info_id_equal (GreeterItemInfo *a,
+ GreeterItemInfo *b)
+{
+ return g_str_equal (a->id, b->id);
+}
+
+static guint
+greeter_info_id_hash (GreeterItemInfo *key)
+{
+ return g_str_hash (key->id);
+}
+
+GreeterItemInfo *
+greeter_parse (const char *file, const char *datadir,
+ GnomeCanvas *canvas,
+ int width, int height, GError **error)
+{
+ GreeterItemInfo *root;
+ xmlDocPtr doc;
+ xmlNodePtr node;
+ xmlChar *prop;
+ gboolean res;
+ GList *items;
+
+ /* FIXME: EVIL! GLOBAL! */
+ g_free (file_search_path);
+ file_search_path = g_strdup (datadir);
+
+ if G_UNLIKELY (!g_file_test (file, G_FILE_TEST_EXISTS))
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_NO_FILE,
+ "Can't open file %s", file);
+ return NULL;
+ }
+
+
+ doc = xmlParseFile (file);
+ if G_UNLIKELY (doc == NULL)
+ {
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_XML,
+ "XML Parse error reading %s", file);
+ return NULL;
+ }
+
+ node = xmlDocGetRootElement (doc);
+ if G_UNLIKELY (node == NULL)
+ {
+ xmlFreeDoc (doc);
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_BAD_XML,
+ "Can't find the xml root node in file %s", file);
+ return NULL;
+ }
+
+ if G_UNLIKELY (strcmp ((char *) node->name, "greeter") != 0)
+ {
+ xmlFreeDoc (doc);
+ g_set_error (error,
+ GREETER_PARSER_ERROR,
+ GREETER_PARSER_ERROR_WRONG_TYPE,
+ "The file %s has the wrong xml type", file);
+ return NULL;
+ }
+
+ /*
+ * The gtk-theme property specifies a theme specific gtk-theme to use
+ */
+ prop = xmlGetProp (node, (const xmlChar *) "gtk-theme");
+ if (prop)
+ {
+ gchar *theme_dir;
+
+ /*
+ * It might be nice if we allowed this property to also supply a gtkrc file
+ * that could be included in the theme. Perhaps we should check first in
+ * the theme directory for a gtkrc file by the provided name and use that
+ * if found.
+ */
+ theme_dir = g_strdup_printf ("%s/%s", gtk_rc_get_theme_dir (), (char *) prop);
+ if (g_file_test (theme_dir, G_FILE_TEST_IS_DIR))
+ gdm_set_theme ((char *) prop);
+
+ xmlFree (prop);
+ }
+
+ item_hash = g_hash_table_new ((GHashFunc)greeter_info_id_hash,
+ (GEqualFunc)greeter_info_id_equal);
+
+
+ root = greeter_item_info_new (NULL, GREETER_ITEM_TYPE_RECT);
+ res = parse_items (node, &items, root, error);
+
+ /* Now we can whack the hash, we don't want to keep cached
+ pixbufs around anymore */
+ if (pixbuf_hash != NULL) {
+ g_hash_table_destroy (pixbuf_hash);
+ pixbuf_hash = NULL;
+ }
+
+ if G_UNLIKELY (!res)
+ {
+ welcome_string_info = NULL;
+
+ g_hash_table_destroy (item_hash);
+ item_hash = NULL;
+ g_list_free (custom_items);
+ custom_items = NULL;
+
+ g_list_free (button_stack);
+ button_stack = NULL;
+
+ g_list_foreach (items, (GFunc) greeter_item_info_free, NULL);
+ g_list_free (items);
+ items = NULL;
+
+ greeter_item_info_free (root);
+
+ xmlFreeDoc (doc);
+
+ return NULL;
+ }
+
+ xmlFreeDoc (doc);
+
+ root->fixed_children = items;
+
+ root->x = 0;
+ root->y = 0;
+ root->x_type = GREETER_ITEM_POS_ABSOLUTE;
+ root->y_type = GREETER_ITEM_POS_ABSOLUTE;
+
+ root->width = width;
+ root->height = height;
+ root->width_type = GREETER_ITEM_SIZE_ABSOLUTE;
+ root->width_type = GREETER_ITEM_SIZE_ABSOLUTE;
+
+ root->group_item = gnome_canvas_root (canvas);
+
+ return root;
+}
+
+const GList *
+greeter_custom_items (void)
+{
+ return custom_items;
+}
+
+static void
+hide_item (GreeterItemInfo *info, gpointer user_data)
+{
+ GnomeCanvasItem *item;
+ gboolean *found_background;
+
+ found_background = user_data;
+
+ if (info)
+ {
+ if (info->background)
+ {
+ *found_background = TRUE;
+ }
+ else {
+ item = info->item;
+ if (item) {
+ if (GNOME_IS_CANVAS_WIDGET (item)) {
+ gtk_widget_hide (GNOME_CANVAS_WIDGET (item)->widget);
+ }
+ else
+ gnome_canvas_item_hide (item);
+ }
+ if ((info->item_type == GREETER_ITEM_TYPE_ENTRY) &&
+ (info->data.text.menubar != NULL)) {
+ gtk_widget_hide (info->data.text.menubar);
+ }
+ }
+
+ g_list_foreach (info->fixed_children, (GFunc) hide_item, user_data);
+ g_list_foreach (info->box_children, (GFunc) hide_item, user_data);
+ }
+}
+
+gboolean
+greeter_show_only_background (GreeterItemInfo *root_item)
+{
+ gboolean found_background = FALSE;
+
+ hide_item (root_item, &found_background);
+
+ /* ensure root canvas is updated */
+ while (gtk_events_pending ())
+ gtk_main_iteration ();
+
+ return found_background;
+}