Blame gtk/gtktextbufferserialize.c

Packit 98cdb6
/* gtktextbufferserialize.c
Packit 98cdb6
 *
Packit 98cdb6
 * Copyright (C) 2001 Havoc Pennington
Packit 98cdb6
 * Copyright (C) 2004 Nokia Corporation
Packit 98cdb6
 *
Packit 98cdb6
 * This library is free software; you can redistribute it and/or
Packit 98cdb6
 * modify it under the terms of the GNU Library General Public
Packit 98cdb6
 * License as published by the Free Software Foundation; either
Packit 98cdb6
 * version 2 of the License, or (at your option) any later version.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is distributed in the hope that it will be useful,
Packit 98cdb6
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 98cdb6
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 98cdb6
 * Library General Public License for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * You should have received a copy of the GNU Library General Public
Packit 98cdb6
 * License along with this library; if not, write to the
Packit 98cdb6
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit 98cdb6
 * Boston, MA 02111-1307, USA.
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
/* FIXME: We should use other error codes for the
Packit 98cdb6
 * parts that deal with the format errors
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
#include "config.h"
Packit 98cdb6
Packit 98cdb6
#include <stdio.h>
Packit 98cdb6
#include <string.h>
Packit 98cdb6
#include <stdlib.h>
Packit 98cdb6
Packit 98cdb6
#undef GDK_PIXBUF_DISABLE_DEPRECATED
Packit 98cdb6
#include "gdk-pixbuf/gdk-pixdata.h"
Packit 98cdb6
#include "gtktextbufferserialize.h"
Packit 98cdb6
#include "gtkintl.h"
Packit 98cdb6
#include "gtkalias.h"
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
typedef struct
Packit 98cdb6
{
Packit 98cdb6
  GString *tag_table_str;
Packit 98cdb6
  GString *text_str;
Packit 98cdb6
  GHashTable *tags;
Packit 98cdb6
  GtkTextIter start, end;
Packit 98cdb6
Packit 98cdb6
  gint n_pixbufs;
Packit 98cdb6
  GList *pixbufs;
Packit 98cdb6
  gint tag_id;
Packit 98cdb6
  GHashTable *tag_id_tags;
Packit 98cdb6
} SerializationContext;
Packit 98cdb6
Packit 98cdb6
static gchar *
Packit 98cdb6
serialize_value (GValue *value)
Packit 98cdb6
{
Packit 98cdb6
  if (g_value_type_transformable (value->g_type, G_TYPE_STRING))
Packit 98cdb6
    {
Packit 98cdb6
      GValue text_value = { 0 };
Packit 98cdb6
      gchar *tmp;
Packit 98cdb6
Packit 98cdb6
      g_value_init (&text_value, G_TYPE_STRING);
Packit 98cdb6
      g_value_transform (value, &text_value);
Packit 98cdb6
Packit 98cdb6
      tmp = g_markup_escape_text (g_value_get_string (&text_value), -1);
Packit 98cdb6
      g_value_unset (&text_value);
Packit 98cdb6
Packit 98cdb6
      return tmp;
Packit 98cdb6
    }
Packit 98cdb6
  else if (value->g_type == GDK_TYPE_COLOR)
Packit 98cdb6
    {
Packit 98cdb6
      GdkColor *color = g_value_get_boxed (value);
Packit 98cdb6
Packit 98cdb6
      return g_strdup_printf ("%x:%x:%x", color->red, color->green, color->blue);
Packit 98cdb6
    }
Packit 98cdb6
  else if (g_type_is_a (value->g_type, GDK_TYPE_DRAWABLE))
Packit 98cdb6
    {
Packit 98cdb6
      /* Don't do anything */
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      g_warning ("Type %s is not serializable\n", g_type_name (value->g_type));
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return NULL;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
deserialize_value (const gchar *str,
Packit 98cdb6
                   GValue      *value)
Packit 98cdb6
{
Packit 98cdb6
  if (g_value_type_transformable (G_TYPE_STRING, value->g_type))
Packit 98cdb6
    {
Packit 98cdb6
      GValue text_value = { 0 };
Packit 98cdb6
      gboolean retval;
Packit 98cdb6
Packit 98cdb6
      g_value_init (&text_value, G_TYPE_STRING);
Packit 98cdb6
      g_value_set_static_string (&text_value, str);
Packit 98cdb6
Packit 98cdb6
      retval = g_value_transform (&text_value, value);
Packit 98cdb6
      g_value_unset (&text_value);
Packit 98cdb6
Packit 98cdb6
      return retval;
Packit 98cdb6
    }
Packit 98cdb6
  else if (value->g_type == G_TYPE_BOOLEAN)
Packit 98cdb6
    {
Packit 98cdb6
      gboolean v;
Packit 98cdb6
Packit 98cdb6
      v = strcmp (str, "TRUE") == 0;
Packit 98cdb6
Packit 98cdb6
      g_value_set_boolean (value, v);
Packit 98cdb6
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
  else if (value->g_type == G_TYPE_INT)
Packit 98cdb6
    {
Packit 98cdb6
      gchar *tmp;
Packit 98cdb6
      int v;
Packit 98cdb6
Packit 98cdb6
      v = strtol (str, &tmp, 10);
Packit 98cdb6
Packit 98cdb6
      if (tmp == NULL || tmp == str)
Packit 98cdb6
	return FALSE;
Packit 98cdb6
Packit 98cdb6
      g_value_set_int (value, v);
Packit 98cdb6
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
  else if (value->g_type == G_TYPE_DOUBLE)
Packit 98cdb6
    {
Packit 98cdb6
      gchar *tmp;
Packit 98cdb6
      gdouble v;
Packit 98cdb6
Packit 98cdb6
      v = g_ascii_strtod (str, &tmp);
Packit 98cdb6
Packit 98cdb6
      if (tmp == NULL || tmp == str)
Packit 98cdb6
	return FALSE;
Packit 98cdb6
Packit 98cdb6
      g_value_set_double (value, v);
Packit 98cdb6
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
  else if (value->g_type == GDK_TYPE_COLOR)
Packit 98cdb6
    {
Packit 98cdb6
      GdkColor color;
Packit 98cdb6
      const gchar *old;
Packit 98cdb6
      gchar *tmp;
Packit 98cdb6
Packit 98cdb6
      old = str;
Packit 98cdb6
      color.red = strtol (old, &tmp, 16);
Packit 98cdb6
Packit 98cdb6
      if (tmp == NULL || tmp == old)
Packit 98cdb6
	return FALSE;
Packit 98cdb6
Packit 98cdb6
      old = tmp;
Packit 98cdb6
      if (*old++ != ':')
Packit 98cdb6
	return FALSE;
Packit 98cdb6
Packit 98cdb6
      color.green = strtol (old, &tmp, 16);
Packit 98cdb6
      if (tmp == NULL || tmp == old)
Packit 98cdb6
	return FALSE;
Packit 98cdb6
Packit 98cdb6
      old = tmp;
Packit 98cdb6
      if (*old++ != ':')
Packit 98cdb6
	return FALSE;
Packit 98cdb6
Packit 98cdb6
      color.blue = strtol (old, &tmp, 16);
Packit 98cdb6
Packit 98cdb6
      if (tmp == NULL || tmp == old || *tmp != '\0')
Packit 98cdb6
	return FALSE;
Packit 98cdb6
Packit 98cdb6
      g_value_set_boxed (value, &color;;
Packit 98cdb6
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
  else if (G_VALUE_HOLDS_ENUM (value))
Packit 98cdb6
    {
Packit 98cdb6
      GEnumClass *class = G_ENUM_CLASS (g_type_class_peek (value->g_type));
Packit 98cdb6
      GEnumValue *enum_value;
Packit 98cdb6
Packit 98cdb6
      enum_value = g_enum_get_value_by_name (class, str);
Packit 98cdb6
Packit 98cdb6
      if (enum_value)
Packit 98cdb6
	{
Packit 98cdb6
	  g_value_set_enum (value, enum_value->value);
Packit 98cdb6
	  return TRUE;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      return FALSE;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      g_warning ("Type %s can not be deserialized\n", g_type_name (value->g_type));
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return FALSE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* Checks if a param is set, or if it's the default value */
Packit 98cdb6
static gboolean
Packit 98cdb6
is_param_set (GObject    *object,
Packit 98cdb6
              GParamSpec *pspec,
Packit 98cdb6
              GValue     *value)
Packit 98cdb6
{
Packit 98cdb6
  /* We need to special case some attributes here */
Packit 98cdb6
  if (strcmp (pspec->name, "background-gdk") == 0)
Packit 98cdb6
    {
Packit 98cdb6
      gboolean is_set;
Packit 98cdb6
Packit 98cdb6
      g_object_get (object, "background-set", &is_set, NULL);
Packit 98cdb6
Packit 98cdb6
      if (is_set)
Packit 98cdb6
	{
Packit 98cdb6
	  g_value_init (value, G_PARAM_SPEC_VALUE_TYPE (pspec));
Packit 98cdb6
Packit 98cdb6
	  g_object_get_property (object, pspec->name, value);
Packit 98cdb6
Packit 98cdb6
	  return TRUE;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      return FALSE;
Packit 98cdb6
    }
Packit 98cdb6
  else if (strcmp (pspec->name, "foreground-gdk") == 0)
Packit 98cdb6
    {
Packit 98cdb6
      gboolean is_set;
Packit 98cdb6
Packit 98cdb6
      g_object_get (object, "foreground-set", &is_set, NULL);
Packit 98cdb6
Packit 98cdb6
      if (is_set)
Packit 98cdb6
	{
Packit 98cdb6
	  g_value_init (value, G_PARAM_SPEC_VALUE_TYPE (pspec));
Packit 98cdb6
Packit 98cdb6
	  g_object_get_property (object, pspec->name, value);
Packit 98cdb6
Packit 98cdb6
	  return TRUE;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      return FALSE;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      gboolean is_set;
Packit 98cdb6
      gchar *is_set_name;
Packit 98cdb6
Packit 98cdb6
      is_set_name = g_strdup_printf ("%s-set", pspec->name);
Packit 98cdb6
Packit 98cdb6
      if (g_object_class_find_property (G_OBJECT_GET_CLASS (object), is_set_name) == NULL)
Packit 98cdb6
	{
Packit 98cdb6
	  g_free (is_set_name);
Packit 98cdb6
	  return FALSE;
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	{
Packit 98cdb6
	  g_object_get (object, is_set_name, &is_set, NULL);
Packit 98cdb6
Packit 98cdb6
	  if (!is_set)
Packit 98cdb6
	    {
Packit 98cdb6
	      g_free (is_set_name);
Packit 98cdb6
	      return FALSE;
Packit 98cdb6
	    }
Packit 98cdb6
Packit 98cdb6
	  g_free (is_set_name);
Packit 98cdb6
Packit 98cdb6
	  g_value_init (value, G_PARAM_SPEC_VALUE_TYPE (pspec));
Packit 98cdb6
Packit 98cdb6
	  g_object_get_property (object, pspec->name, value);
Packit 98cdb6
Packit 98cdb6
	  if (g_param_value_defaults (pspec, value))
Packit 98cdb6
	    {
Packit 98cdb6
	      g_value_unset (value);
Packit 98cdb6
Packit 98cdb6
	      return FALSE;
Packit 98cdb6
	    }
Packit 98cdb6
	}
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
serialize_tag (gpointer key,
Packit 98cdb6
               gpointer data,
Packit 98cdb6
               gpointer user_data)
Packit 98cdb6
{
Packit 98cdb6
  SerializationContext *context = user_data;
Packit 98cdb6
  GtkTextTag *tag = data;
Packit 98cdb6
  gchar *tag_name;
Packit 98cdb6
  gint tag_id;
Packit 98cdb6
  GParamSpec **pspecs;
Packit 98cdb6
  guint n_pspecs;
Packit 98cdb6
  int i;
Packit 98cdb6
Packit 98cdb6
  g_string_append (context->tag_table_str, "  
Packit 98cdb6
Packit 98cdb6
  /* Handle anonymous tags */
Packit 98cdb6
  if (tag->name)
Packit 98cdb6
    {
Packit 98cdb6
      tag_name = g_markup_escape_text (tag->name, -1);
Packit 98cdb6
      g_string_append_printf (context->tag_table_str, "name=\"%s\"", tag_name);
Packit 98cdb6
      g_free (tag_name);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      tag_id = GPOINTER_TO_INT (g_hash_table_lookup (context->tag_id_tags, tag));
Packit 98cdb6
Packit 98cdb6
      g_string_append_printf (context->tag_table_str, "id=\"%d\"", tag_id);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  g_string_append_printf (context->tag_table_str, " priority=\"%d\">\n", tag->priority);
Packit 98cdb6
Packit 98cdb6
  /* Serialize properties */
Packit 98cdb6
  pspecs = g_object_class_list_properties (G_OBJECT_GET_CLASS (tag), &n_pspecs);
Packit 98cdb6
Packit 98cdb6
  for (i = 0; i < n_pspecs; i++)
Packit 98cdb6
    {
Packit 98cdb6
      GValue value = { 0 };
Packit 98cdb6
      gchar *tmp, *tmp2;
Packit 98cdb6
Packit 98cdb6
      if (!(pspecs[i]->flags & G_PARAM_READABLE) ||
Packit 98cdb6
	  !(pspecs[i]->flags & G_PARAM_WRITABLE))
Packit 98cdb6
	continue;
Packit 98cdb6
Packit 98cdb6
      if (!is_param_set (G_OBJECT (tag), pspecs[i], &value))
Packit 98cdb6
	continue;
Packit 98cdb6
Packit 98cdb6
      /* Now serialize the attr */
Packit 98cdb6
      tmp2 = serialize_value (&value);
Packit 98cdb6
Packit 98cdb6
      if (tmp2)
Packit 98cdb6
	{
Packit 98cdb6
	  tmp = g_markup_escape_text (pspecs[i]->name, -1);
Packit 98cdb6
	  g_string_append_printf (context->tag_table_str, "   
Packit 98cdb6
	  g_free (tmp);
Packit 98cdb6
Packit 98cdb6
	  tmp = g_markup_escape_text (g_type_name (pspecs[i]->value_type), -1);
Packit 98cdb6
	  g_string_append_printf (context->tag_table_str, "type=\"%s\" value=\"%s\" />\n", tmp, tmp2);
Packit 98cdb6
Packit 98cdb6
	  g_free (tmp);
Packit 98cdb6
	  g_free (tmp2);
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      g_value_unset (&value);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  g_free (pspecs);
Packit 98cdb6
Packit 98cdb6
  g_string_append (context->tag_table_str, "  </tag>\n");
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
serialize_tags (SerializationContext *context)
Packit 98cdb6
{
Packit 98cdb6
  g_string_append (context->tag_table_str, " <text_view_markup>\n");
Packit 98cdb6
  g_string_append (context->tag_table_str, " <tags>\n");
Packit 98cdb6
  g_hash_table_foreach (context->tags, serialize_tag, context);
Packit 98cdb6
  g_string_append (context->tag_table_str, " </tags>\n");
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
#if 0
Packit 98cdb6
static void
Packit 98cdb6
dump_tag_list (const gchar *str,
Packit 98cdb6
               GList       *list)
Packit 98cdb6
{
Packit 98cdb6
  g_print ("%s: ", str);
Packit 98cdb6
Packit 98cdb6
  if (!list)
Packit 98cdb6
    g_print ("(empty)");
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      while (list)
Packit 98cdb6
	{
Packit 98cdb6
	  g_print ("%s ", ((GtkTextTag *)list->data)->name);
Packit 98cdb6
	  list = list->next;
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  g_print ("\n");
Packit 98cdb6
}
Packit 98cdb6
#endif
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
find_list_delta (GSList  *old_list,
Packit 98cdb6
                 GSList  *new_list,
Packit 98cdb6
		 GList  **added,
Packit 98cdb6
                 GList  **removed)
Packit 98cdb6
{
Packit 98cdb6
  GSList *tmp;
Packit 98cdb6
  GList *tmp_added, *tmp_removed;
Packit 98cdb6
Packit 98cdb6
  tmp_added = NULL;
Packit 98cdb6
  tmp_removed = NULL;
Packit 98cdb6
Packit 98cdb6
  /* Find added tags */
Packit 98cdb6
  tmp = new_list;
Packit 98cdb6
  while (tmp)
Packit 98cdb6
    {
Packit 98cdb6
      if (!g_slist_find (old_list, tmp->data))
Packit 98cdb6
	tmp_added = g_list_prepend (tmp_added, tmp->data);
Packit 98cdb6
Packit 98cdb6
      tmp = tmp->next;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  *added = tmp_added;
Packit 98cdb6
Packit 98cdb6
  /* Find removed tags */
Packit 98cdb6
  tmp = old_list;
Packit 98cdb6
  while (tmp)
Packit 98cdb6
    {
Packit 98cdb6
      if (!g_slist_find (new_list, tmp->data))
Packit 98cdb6
	tmp_removed = g_list_prepend (tmp_removed, tmp->data);
Packit 98cdb6
Packit 98cdb6
      tmp = tmp->next;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  /* We reverse the list here to match the xml semantics */
Packit 98cdb6
  *removed = g_list_reverse (tmp_removed);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
serialize_section_header (GString     *str,
Packit 98cdb6
			  const gchar *name,
Packit 98cdb6
			  gint         length)
Packit 98cdb6
{
Packit 98cdb6
  g_return_if_fail (strlen (name) == 26);
Packit 98cdb6
Packit 98cdb6
  g_string_append (str, name);
Packit 98cdb6
Packit 98cdb6
  g_string_append_c (str, length >> 24);
Packit 98cdb6
Packit 98cdb6
  g_string_append_c (str, (length >> 16) & 0xff);
Packit 98cdb6
  g_string_append_c (str, (length >> 8) & 0xff);
Packit 98cdb6
  g_string_append_c (str, length & 0xff);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
serialize_text (GtkTextBuffer        *buffer,
Packit 98cdb6
                SerializationContext *context)
Packit 98cdb6
{
Packit 98cdb6
  GtkTextIter iter, old_iter;
Packit 98cdb6
  GSList *tag_list, *new_tag_list;
Packit 98cdb6
  GSList *active_tags;
Packit 98cdb6
Packit 98cdb6
  g_string_append (context->text_str, "<text>");
Packit 98cdb6
Packit 98cdb6
  iter = context->start;
Packit 98cdb6
  tag_list = NULL;
Packit 98cdb6
  active_tags = NULL;
Packit 98cdb6
Packit 98cdb6
  do
Packit 98cdb6
    {
Packit 98cdb6
      GList *added, *removed;
Packit 98cdb6
      GList *tmp;
Packit 98cdb6
      gchar *tmp_text, *escaped_text;
Packit 98cdb6
Packit 98cdb6
      new_tag_list = gtk_text_iter_get_tags (&iter);
Packit 98cdb6
      find_list_delta (tag_list, new_tag_list, &added, &removed);
Packit 98cdb6
Packit 98cdb6
      /* Handle removed tags */
Packit 98cdb6
      for (tmp = removed; tmp; tmp = tmp->next)
Packit 98cdb6
	{
Packit 98cdb6
	  GtkTextTag *tag = tmp->data;
Packit 98cdb6
Packit 98cdb6
          /* Only close the tag if we didn't close it before (by using
Packit 98cdb6
           * the stack logic in the while() loop below)
Packit 98cdb6
           */
Packit 98cdb6
          if (g_slist_find (active_tags, tag))
Packit 98cdb6
            {
Packit 98cdb6
              g_string_append (context->text_str, "</apply_tag>");
Packit 98cdb6
Packit 98cdb6
              /* Drop all tags that were opened after this one (which are
Packit 98cdb6
               * above this on in the stack)
Packit 98cdb6
               */
Packit 98cdb6
              while (active_tags->data != tag)
Packit 98cdb6
                {
Packit 98cdb6
                  added = g_list_prepend (added, active_tags->data);
Packit 98cdb6
                  active_tags = g_slist_remove (active_tags, active_tags->data);
Packit 98cdb6
                  g_string_append_printf (context->text_str, "</apply_tag>");
Packit 98cdb6
                }
Packit 98cdb6
Packit 98cdb6
              active_tags = g_slist_remove (active_tags, active_tags->data);
Packit 98cdb6
            }
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      /* Handle added tags */
Packit 98cdb6
      for (tmp = added; tmp; tmp = tmp->next)
Packit 98cdb6
	{
Packit 98cdb6
	  GtkTextTag *tag = tmp->data;
Packit 98cdb6
	  gchar *tag_name;
Packit 98cdb6
Packit 98cdb6
	  /* Add it to the tag hash table */
Packit 98cdb6
	  g_hash_table_insert (context->tags, tag, tag);
Packit 98cdb6
Packit 98cdb6
	  if (tag->name)
Packit 98cdb6
	    {
Packit 98cdb6
	      tag_name = g_markup_escape_text (tag->name, -1);
Packit 98cdb6
Packit 98cdb6
	      g_string_append_printf (context->text_str, "<apply_tag name=\"%s\">", tag_name);
Packit 98cdb6
	      g_free (tag_name);
Packit 98cdb6
	    }
Packit 98cdb6
	  else
Packit 98cdb6
	    {
Packit 98cdb6
	      gpointer tag_id;
Packit 98cdb6
Packit 98cdb6
	      /* We've got an anonymous tag, find out if it's been
Packit 98cdb6
		 used before */
Packit 98cdb6
	      if (!g_hash_table_lookup_extended (context->tag_id_tags, tag, NULL, &tag_id))
Packit 98cdb6
		{
Packit 98cdb6
		  tag_id = GINT_TO_POINTER (context->tag_id++);
Packit 98cdb6
Packit 98cdb6
		  g_hash_table_insert (context->tag_id_tags, tag, tag_id);
Packit 98cdb6
		}
Packit 98cdb6
Packit 98cdb6
	      g_string_append_printf (context->text_str, "<apply_tag id=\"%d\">", GPOINTER_TO_INT (tag_id));
Packit 98cdb6
	    }
Packit 98cdb6
Packit 98cdb6
	  active_tags = g_slist_prepend (active_tags, tag);
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      g_slist_free (tag_list);
Packit 98cdb6
      tag_list = new_tag_list;
Packit 98cdb6
Packit 98cdb6
      g_list_free (added);
Packit 98cdb6
      g_list_free (removed);
Packit 98cdb6
Packit 98cdb6
      old_iter = iter;
Packit 98cdb6
Packit 98cdb6
      /* Now try to go to either the next tag toggle, or if a pixbuf appears */
Packit 98cdb6
      while (TRUE)
Packit 98cdb6
	{
Packit 98cdb6
	  gunichar ch = gtk_text_iter_get_char (&iter);
Packit 98cdb6
Packit 98cdb6
	  if (ch == 0xFFFC)
Packit 98cdb6
	    {
Packit 98cdb6
	      GdkPixbuf *pixbuf = gtk_text_iter_get_pixbuf (&iter);
Packit 98cdb6
Packit 98cdb6
	      if (pixbuf)
Packit 98cdb6
		{
Packit 98cdb6
		  /* Append the text before the pixbuf */
Packit 98cdb6
		  tmp_text = gtk_text_iter_get_slice (&old_iter, &iter);
Packit 98cdb6
		  escaped_text = g_markup_escape_text (tmp_text, -1);
Packit 98cdb6
		  g_free (tmp_text);
Packit 98cdb6
Packit 98cdb6
		  /* Forward so we don't get the 0xfffc char */
Packit 98cdb6
		  gtk_text_iter_forward_char (&iter);
Packit 98cdb6
		  old_iter = iter;
Packit 98cdb6
Packit 98cdb6
		  g_string_append (context->text_str, escaped_text);
Packit 98cdb6
		  g_free (escaped_text);
Packit 98cdb6
Packit 98cdb6
		  g_string_append_printf (context->text_str, "<pixbuf index=\"%d\" />", context->n_pixbufs);
Packit 98cdb6
Packit 98cdb6
		  context->n_pixbufs++;
Packit 98cdb6
		  context->pixbufs = g_list_prepend (context->pixbufs, pixbuf);
Packit 98cdb6
		}
Packit 98cdb6
	    }
Packit 98cdb6
          else if (ch == 0)
Packit 98cdb6
            {
Packit 98cdb6
                break;
Packit 98cdb6
            }
Packit 98cdb6
	  else
Packit 98cdb6
	    gtk_text_iter_forward_char (&iter);
Packit 98cdb6
Packit 98cdb6
	  if (gtk_text_iter_toggles_tag (&iter, NULL))
Packit 98cdb6
	    break;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      /* We might have moved too far */
Packit 98cdb6
      if (gtk_text_iter_compare (&iter, &context->end) > 0)
Packit 98cdb6
	iter = context->end;
Packit 98cdb6
Packit 98cdb6
      /* Append the text */
Packit 98cdb6
      tmp_text = gtk_text_iter_get_slice (&old_iter, &iter);
Packit 98cdb6
      escaped_text = g_markup_escape_text (tmp_text, -1);
Packit 98cdb6
      g_free (tmp_text);
Packit 98cdb6
Packit 98cdb6
      g_string_append (context->text_str, escaped_text);
Packit 98cdb6
      g_free (escaped_text);
Packit 98cdb6
    }
Packit 98cdb6
  while (!gtk_text_iter_equal (&iter, &context->end));
Packit 98cdb6
Packit 98cdb6
  /* Close any open tags */
Packit 98cdb6
  for (tag_list = active_tags; tag_list; tag_list = tag_list->next)
Packit 98cdb6
    g_string_append (context->text_str, "</apply_tag>");
Packit 98cdb6
Packit 98cdb6
  g_slist_free (active_tags);
Packit 98cdb6
  g_string_append (context->text_str, "</text>\n</text_view_markup>\n");
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
serialize_pixbufs (SerializationContext *context,
Packit 98cdb6
		   GString              *text)
Packit 98cdb6
{
Packit 98cdb6
  GList *list;
Packit 98cdb6
Packit 98cdb6
  for (list = context->pixbufs; list != NULL; list = list->next)
Packit 98cdb6
    {
Packit 98cdb6
      GdkPixbuf *pixbuf = list->data;
Packit 98cdb6
      GdkPixdata pixdata;
Packit 98cdb6
      guint8 *tmp;
Packit 98cdb6
      guint len;
Packit 98cdb6
Packit 98cdb6
      gdk_pixdata_from_pixbuf (&pixdata, pixbuf, FALSE);
Packit 98cdb6
      tmp = gdk_pixdata_serialize (&pixdata, &len;;
Packit 98cdb6
Packit 98cdb6
      serialize_section_header (text, "GTKTEXTBUFFERPIXBDATA-0001", len);
Packit 98cdb6
      g_string_append_len (text, (gchar *) tmp, len);
Packit 98cdb6
      g_free (tmp);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
guint8 *
Packit 98cdb6
_gtk_text_buffer_serialize_rich_text (GtkTextBuffer     *register_buffer,
Packit 98cdb6
                                      GtkTextBuffer     *content_buffer,
Packit 98cdb6
                                      const GtkTextIter *start,
Packit 98cdb6
                                      const GtkTextIter *end,
Packit 98cdb6
                                      gsize             *length,
Packit 98cdb6
                                      gpointer           user_data)
Packit 98cdb6
{
Packit 98cdb6
  SerializationContext context;
Packit 98cdb6
  GString *text;
Packit 98cdb6
Packit 98cdb6
  context.tags = g_hash_table_new (NULL, NULL);
Packit 98cdb6
  context.text_str = g_string_new (NULL);
Packit 98cdb6
  context.tag_table_str = g_string_new (NULL);
Packit 98cdb6
  context.start = *start;
Packit 98cdb6
  context.end = *end;
Packit 98cdb6
  context.n_pixbufs = 0;
Packit 98cdb6
  context.pixbufs = NULL;
Packit 98cdb6
  context.tag_id = 0;
Packit 98cdb6
  context.tag_id_tags = g_hash_table_new (NULL, NULL);
Packit 98cdb6
Packit 98cdb6
  /* We need to serialize the text before the tag table so we know
Packit 98cdb6
     what tags are used */
Packit 98cdb6
  serialize_text (content_buffer, &context);
Packit 98cdb6
  serialize_tags (&context);
Packit 98cdb6
Packit 98cdb6
  text = g_string_new (NULL);
Packit 98cdb6
  serialize_section_header (text, "GTKTEXTBUFFERCONTENTS-0001",
Packit 98cdb6
                            context.tag_table_str->len + context.text_str->len);
Packit 98cdb6
Packit 98cdb6
  g_string_append_len (text, context.tag_table_str->str, context.tag_table_str->len);
Packit 98cdb6
  g_string_append_len (text, context.text_str->str, context.text_str->len);
Packit 98cdb6
Packit 98cdb6
  context.pixbufs = g_list_reverse (context.pixbufs);
Packit 98cdb6
  serialize_pixbufs (&context, text);
Packit 98cdb6
Packit 98cdb6
  g_hash_table_destroy (context.tags);
Packit 98cdb6
  g_list_free (context.pixbufs);
Packit 98cdb6
  g_string_free (context.text_str, TRUE);
Packit 98cdb6
  g_string_free (context.tag_table_str, TRUE);
Packit 98cdb6
  g_hash_table_destroy (context.tag_id_tags);
Packit 98cdb6
Packit 98cdb6
  *length = text->len;
Packit 98cdb6
Packit 98cdb6
  return (guint8 *) g_string_free (text, FALSE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
typedef enum
Packit 98cdb6
{
Packit 98cdb6
  STATE_START,
Packit 98cdb6
  STATE_TEXT_VIEW_MARKUP,
Packit 98cdb6
  STATE_TAGS,
Packit 98cdb6
  STATE_TAG,
Packit 98cdb6
  STATE_ATTR,
Packit 98cdb6
  STATE_TEXT,
Packit 98cdb6
  STATE_APPLY_TAG,
Packit 98cdb6
  STATE_PIXBUF
Packit 98cdb6
} ParseState;
Packit 98cdb6
Packit 98cdb6
typedef struct
Packit 98cdb6
{
Packit 98cdb6
  gchar *text;
Packit 98cdb6
  GdkPixbuf *pixbuf;
Packit 98cdb6
  GSList *tags;
Packit 98cdb6
} TextSpan;
Packit 98cdb6
Packit 98cdb6
typedef struct
Packit 98cdb6
{
Packit 98cdb6
  GtkTextTag *tag;
Packit 98cdb6
  gint prio;
Packit 98cdb6
} TextTagPrio;
Packit 98cdb6
Packit 98cdb6
typedef struct
Packit 98cdb6
{
Packit 98cdb6
  GSList *states;
Packit 98cdb6
Packit 98cdb6
  GList *headers;
Packit 98cdb6
Packit 98cdb6
  GtkTextBuffer *buffer;
Packit 98cdb6
Packit 98cdb6
  /* Tags that are defined in <tag> elements */
Packit 98cdb6
  GHashTable *defined_tags;
Packit 98cdb6
Packit 98cdb6
  /* Tags that are anonymous */
Packit 98cdb6
  GHashTable *anonymous_tags;
Packit 98cdb6
Packit 98cdb6
  /* Tag name substitutions */
Packit 98cdb6
  GHashTable *substitutions;
Packit 98cdb6
Packit 98cdb6
  /* Current tag */
Packit 98cdb6
  GtkTextTag *current_tag;
Packit 98cdb6
Packit 98cdb6
  /* Priority of current tag */
Packit 98cdb6
  gint current_tag_prio;
Packit 98cdb6
Packit 98cdb6
  /* Id of current tag */
Packit 98cdb6
  gint current_tag_id;
Packit 98cdb6
Packit 98cdb6
  /* Tags and their priorities */
Packit 98cdb6
  GList *tag_priorities;
Packit 98cdb6
Packit 98cdb6
  GSList *tag_stack;
Packit 98cdb6
Packit 98cdb6
  GList *spans;
Packit 98cdb6
Packit 98cdb6
  gboolean create_tags;
Packit 98cdb6
Packit 98cdb6
  gboolean parsed_text;
Packit 98cdb6
  gboolean parsed_tags;
Packit 98cdb6
} ParseInfo;
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
set_error (GError              **err,
Packit 98cdb6
           GMarkupParseContext  *context,
Packit 98cdb6
           int                   error_domain,
Packit 98cdb6
           int                   error_code,
Packit 98cdb6
           const char           *format,
Packit 98cdb6
           ...)
Packit 98cdb6
{
Packit 98cdb6
  int line, ch;
Packit 98cdb6
  va_list args;
Packit 98cdb6
  char *str;
Packit 98cdb6
Packit 98cdb6
  g_markup_parse_context_get_position (context, &line, &ch);
Packit 98cdb6
Packit 98cdb6
  va_start (args, format);
Packit 98cdb6
  str = g_strdup_vprintf (format, args);
Packit 98cdb6
  va_end (args);
Packit 98cdb6
Packit 98cdb6
  g_set_error (err, error_domain, error_code,
Packit 98cdb6
               ("Line %d character %d: %s"),
Packit 98cdb6
               line, ch, str);
Packit 98cdb6
Packit 98cdb6
  g_free (str);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
push_state (ParseInfo  *info,
Packit 98cdb6
            ParseState  state)
Packit 98cdb6
{
Packit 98cdb6
  info->states = g_slist_prepend (info->states, GINT_TO_POINTER (state));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
pop_state (ParseInfo *info)
Packit 98cdb6
{
Packit 98cdb6
  g_return_if_fail (info->states != NULL);
Packit 98cdb6
Packit 98cdb6
  info->states = g_slist_remove (info->states, info->states->data);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static ParseState
Packit 98cdb6
peek_state (ParseInfo *info)
Packit 98cdb6
{
Packit 98cdb6
  g_return_val_if_fail (info->states != NULL, STATE_START);
Packit 98cdb6
Packit 98cdb6
  return GPOINTER_TO_INT (info->states->data);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
#define ELEMENT_IS(name) (strcmp (element_name, (name)) == 0)
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
check_id_or_name (GMarkupParseContext  *context,
Packit 98cdb6
		  const gchar          *element_name,
Packit 98cdb6
		  const gchar         **attribute_names,
Packit 98cdb6
		  const gchar         **attribute_values,
Packit 98cdb6
		  gint                 *id,
Packit 98cdb6
		  const gchar         **name,
Packit 98cdb6
		  GError              **error)
Packit 98cdb6
{
Packit 98cdb6
  gboolean has_id = FALSE;
Packit 98cdb6
  gboolean has_name = FALSE;
Packit 98cdb6
  int i;
Packit 98cdb6
Packit 98cdb6
  *id = 0;
Packit 98cdb6
  *name = NULL;
Packit 98cdb6
Packit 98cdb6
  for (i = 0; attribute_names[i] != NULL; i++)
Packit 98cdb6
    {
Packit 98cdb6
      if (strcmp (attribute_names[i], "name") == 0)
Packit 98cdb6
	{
Packit 98cdb6
	  *name = attribute_values[i];
Packit 98cdb6
Packit 98cdb6
	  if (has_id)
Packit 98cdb6
	    {
Packit 98cdb6
	      set_error (error, context,
Packit 98cdb6
			 G_MARKUP_ERROR,
Packit 98cdb6
			 G_MARKUP_ERROR_PARSE,
Packit 98cdb6
			 _("Both \"id\" and \"name\" were found on the <%s> element"),
Packit 98cdb6
			 element_name);
Packit 98cdb6
	      return FALSE;
Packit 98cdb6
	    }
Packit 98cdb6
Packit 98cdb6
	  if (has_name)
Packit 98cdb6
	    {
Packit 98cdb6
	      set_error (error, context,
Packit 98cdb6
			 G_MARKUP_ERROR,
Packit 98cdb6
			 G_MARKUP_ERROR_PARSE,
Packit 98cdb6
			 _("The attribute \"%s\" was found twice on the <%s> element"),
Packit 98cdb6
			 "name", element_name);
Packit 98cdb6
	      return FALSE;
Packit 98cdb6
	    }
Packit 98cdb6
Packit 98cdb6
	  has_name = TRUE;
Packit 98cdb6
	}
Packit 98cdb6
      else if (strcmp (attribute_names[i], "id") == 0)
Packit 98cdb6
	{
Packit 98cdb6
	  gchar *tmp;
Packit 98cdb6
Packit 98cdb6
	  if (has_name)
Packit 98cdb6
	    {
Packit 98cdb6
	      set_error (error, context,
Packit 98cdb6
			 G_MARKUP_ERROR,
Packit 98cdb6
			 G_MARKUP_ERROR_PARSE,
Packit 98cdb6
			 _("Both \"id\" and \"name\" were found on the <%s> element"),
Packit 98cdb6
			 element_name);
Packit 98cdb6
	      return FALSE;
Packit 98cdb6
	    }
Packit 98cdb6
Packit 98cdb6
	  if (has_id)
Packit 98cdb6
	    {
Packit 98cdb6
	      set_error (error, context,
Packit 98cdb6
			 G_MARKUP_ERROR,
Packit 98cdb6
			 G_MARKUP_ERROR_PARSE,
Packit 98cdb6
			 _("The attribute \"%s\" was found twice on the <%s> element"),
Packit 98cdb6
			 "id", element_name);
Packit 98cdb6
	      return FALSE;
Packit 98cdb6
	    }
Packit 98cdb6
Packit 98cdb6
	  has_id = TRUE;
Packit 98cdb6
Packit 98cdb6
	  /* Try parsing the integer */
Packit 98cdb6
	  *id = strtol (attribute_values[i], &tmp, 10);
Packit 98cdb6
Packit 98cdb6
	  if (tmp == NULL || tmp == attribute_values[i])
Packit 98cdb6
	    {
Packit 98cdb6
	      set_error (error, context,
Packit 98cdb6
			 G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
			 _("<%s> element has invalid ID \"%s\""), attribute_values[i]);
Packit 98cdb6
	      return FALSE;
Packit 98cdb6
	    }
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (!has_id && !has_name)
Packit 98cdb6
    {
Packit 98cdb6
      set_error (error, context,
Packit 98cdb6
		 G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
		 _("<%s> element has neither a \"name\" nor an \"id\" attribute"), element_name);
Packit 98cdb6
      return FALSE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
typedef struct
Packit 98cdb6
{
Packit 98cdb6
  const char  *name;
Packit 98cdb6
  const char **retloc;
Packit 98cdb6
} LocateAttr;
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
locate_attributes (GMarkupParseContext  *context,
Packit 98cdb6
                   const char           *element_name,
Packit 98cdb6
                   const char          **attribute_names,
Packit 98cdb6
                   const char          **attribute_values,
Packit 98cdb6
		   gboolean              allow_unknown_attrs,
Packit 98cdb6
                   GError              **error,
Packit 98cdb6
                   const char           *first_attribute_name,
Packit 98cdb6
                   const char          **first_attribute_retloc,
Packit 98cdb6
                   ...)
Packit 98cdb6
{
Packit 98cdb6
  va_list args;
Packit 98cdb6
  const char *name;
Packit 98cdb6
  const char **retloc;
Packit 98cdb6
  int n_attrs;
Packit 98cdb6
#define MAX_ATTRS 24
Packit 98cdb6
  LocateAttr attrs[MAX_ATTRS];
Packit 98cdb6
  gboolean retval;
Packit 98cdb6
  int i;
Packit 98cdb6
Packit 98cdb6
  g_return_val_if_fail (first_attribute_name != NULL, FALSE);
Packit 98cdb6
  g_return_val_if_fail (first_attribute_retloc != NULL, FALSE);
Packit 98cdb6
Packit 98cdb6
  retval = TRUE;
Packit 98cdb6
Packit 98cdb6
  n_attrs = 1;
Packit 98cdb6
  attrs[0].name = first_attribute_name;
Packit 98cdb6
  attrs[0].retloc = first_attribute_retloc;
Packit 98cdb6
  *first_attribute_retloc = NULL;
Packit 98cdb6
Packit 98cdb6
  va_start (args, first_attribute_retloc);
Packit 98cdb6
Packit 98cdb6
  name = va_arg (args, const char*);
Packit 98cdb6
  retloc = va_arg (args, const char**);
Packit 98cdb6
Packit 98cdb6
  while (name != NULL)
Packit 98cdb6
    {
Packit 98cdb6
      g_return_val_if_fail (retloc != NULL, FALSE);
Packit 98cdb6
Packit 98cdb6
      g_assert (n_attrs < MAX_ATTRS);
Packit 98cdb6
Packit 98cdb6
      attrs[n_attrs].name = name;
Packit 98cdb6
      attrs[n_attrs].retloc = retloc;
Packit 98cdb6
      n_attrs += 1;
Packit 98cdb6
      *retloc = NULL;
Packit 98cdb6
Packit 98cdb6
      name = va_arg (args, const char*);
Packit 98cdb6
      retloc = va_arg (args, const char**);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  va_end (args);
Packit 98cdb6
Packit 98cdb6
  if (!retval)
Packit 98cdb6
    return retval;
Packit 98cdb6
Packit 98cdb6
  i = 0;
Packit 98cdb6
  while (attribute_names[i])
Packit 98cdb6
    {
Packit 98cdb6
      int j;
Packit 98cdb6
      gboolean found;
Packit 98cdb6
Packit 98cdb6
      found = FALSE;
Packit 98cdb6
      j = 0;
Packit 98cdb6
      while (j < n_attrs)
Packit 98cdb6
        {
Packit 98cdb6
          if (strcmp (attrs[j].name, attribute_names[i]) == 0)
Packit 98cdb6
            {
Packit 98cdb6
              retloc = attrs[j].retloc;
Packit 98cdb6
Packit 98cdb6
              if (*retloc != NULL)
Packit 98cdb6
                {
Packit 98cdb6
                  set_error (error, context,
Packit 98cdb6
                             G_MARKUP_ERROR,
Packit 98cdb6
                             G_MARKUP_ERROR_PARSE,
Packit 98cdb6
                             _("Attribute \"%s\" repeated twice on the same <%s> element"),
Packit 98cdb6
                             attrs[j].name, element_name);
Packit 98cdb6
                  retval = FALSE;
Packit 98cdb6
                  goto out;
Packit 98cdb6
                }
Packit 98cdb6
Packit 98cdb6
              *retloc = attribute_values[i];
Packit 98cdb6
              found = TRUE;
Packit 98cdb6
            }
Packit 98cdb6
Packit 98cdb6
          ++j;
Packit 98cdb6
        }
Packit 98cdb6
Packit 98cdb6
      if (!found && !allow_unknown_attrs)
Packit 98cdb6
        {
Packit 98cdb6
          set_error (error, context,
Packit 98cdb6
                     G_MARKUP_ERROR,
Packit 98cdb6
                     G_MARKUP_ERROR_PARSE,
Packit 98cdb6
                     _("Attribute \"%s\" is invalid on <%s> element in this context"),
Packit 98cdb6
                     attribute_names[i], element_name);
Packit 98cdb6
          retval = FALSE;
Packit 98cdb6
          goto out;
Packit 98cdb6
        }
Packit 98cdb6
Packit 98cdb6
      ++i;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
 out:
Packit 98cdb6
  return retval;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
check_no_attributes (GMarkupParseContext  *context,
Packit 98cdb6
                     const char           *element_name,
Packit 98cdb6
                     const char          **attribute_names,
Packit 98cdb6
                     const char          **attribute_values,
Packit 98cdb6
                     GError              **error)
Packit 98cdb6
{
Packit 98cdb6
  if (attribute_names[0] != NULL)
Packit 98cdb6
    {
Packit 98cdb6
      set_error (error, context,
Packit 98cdb6
                 G_MARKUP_ERROR,
Packit 98cdb6
                 G_MARKUP_ERROR_PARSE,
Packit 98cdb6
                 _("Attribute \"%s\" is invalid on <%s> element in this context"),
Packit 98cdb6
                 attribute_names[0], element_name);
Packit 98cdb6
      return FALSE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GtkTextTag *
Packit 98cdb6
tag_exists (GMarkupParseContext *context,
Packit 98cdb6
	    const gchar         *name,
Packit 98cdb6
	    gint                 id,
Packit 98cdb6
	    ParseInfo           *info,
Packit 98cdb6
	    GError             **error)
Packit 98cdb6
{
Packit 98cdb6
  const gchar *real_name;
Packit 98cdb6
Packit 98cdb6
  if (info->create_tags)
Packit 98cdb6
    {
Packit 98cdb6
      /* If we have an anonymous tag, just return it directly */
Packit 98cdb6
      if (!name)
Packit 98cdb6
	return g_hash_table_lookup (info->anonymous_tags,
Packit 98cdb6
				    GINT_TO_POINTER (id));
Packit 98cdb6
Packit 98cdb6
      /* First, try the substitutions */
Packit 98cdb6
      real_name = g_hash_table_lookup (info->substitutions, name);
Packit 98cdb6
Packit 98cdb6
      if (real_name)
Packit 98cdb6
	return gtk_text_tag_table_lookup (info->buffer->tag_table, real_name);
Packit 98cdb6
Packit 98cdb6
      /* Next, try the list of defined tags */
Packit 98cdb6
      if (g_hash_table_lookup (info->defined_tags, name) != NULL)
Packit 98cdb6
	return gtk_text_tag_table_lookup (info->buffer->tag_table, name);
Packit 98cdb6
Packit 98cdb6
      set_error (error, context,
Packit 98cdb6
		 G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
		 _("Tag \"%s\" has not been defined."), name);
Packit 98cdb6
Packit 98cdb6
      return NULL;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      GtkTextTag *tag;
Packit 98cdb6
Packit 98cdb6
      if (!name)
Packit 98cdb6
	{
Packit 98cdb6
	  set_error (error, context,
Packit 98cdb6
		     G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
		     _("Anonymous tag found and tags can not be created."));
Packit 98cdb6
	  return NULL;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      tag = gtk_text_tag_table_lookup (info->buffer->tag_table, name);
Packit 98cdb6
Packit 98cdb6
      if (tag)
Packit 98cdb6
	return tag;
Packit 98cdb6
Packit 98cdb6
      set_error (error, context,
Packit 98cdb6
		 G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
		 _("Tag \"%s\" does not exist in buffer and tags can not be created."), name);
Packit 98cdb6
Packit 98cdb6
      return NULL;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
typedef struct
Packit 98cdb6
{
Packit 98cdb6
  const gchar *id;
Packit 98cdb6
  gint length;
Packit 98cdb6
  const gchar *start;
Packit 98cdb6
} Header;
Packit 98cdb6
Packit 98cdb6
static GdkPixbuf *
Packit 98cdb6
get_pixbuf_from_headers (GList   *headers,
Packit 98cdb6
                         int      id,
Packit 98cdb6
                         GError **error)
Packit 98cdb6
{
Packit 98cdb6
  Header *header;
Packit 98cdb6
  GdkPixdata pixdata;
Packit 98cdb6
  GdkPixbuf *pixbuf;
Packit 98cdb6
Packit 98cdb6
  header = g_list_nth_data (headers, id);
Packit 98cdb6
Packit 98cdb6
  if (!header)
Packit 98cdb6
    return NULL;
Packit 98cdb6
Packit 98cdb6
  if (!gdk_pixdata_deserialize (&pixdata, header->length,
Packit 98cdb6
                                (const guint8 *) header->start, error))
Packit 98cdb6
    return NULL;
Packit 98cdb6
Packit 98cdb6
  pixbuf = gdk_pixbuf_from_pixdata (&pixdata, TRUE, error);
Packit 98cdb6
Packit 98cdb6
  return pixbuf;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
parse_apply_tag_element (GMarkupParseContext  *context,
Packit 98cdb6
			 const gchar          *element_name,
Packit 98cdb6
			 const gchar         **attribute_names,
Packit 98cdb6
			 const gchar         **attribute_values,
Packit 98cdb6
			 ParseInfo            *info,
Packit 98cdb6
			 GError              **error)
Packit 98cdb6
{
Packit 98cdb6
  const gchar *name, *priority;
Packit 98cdb6
  gint id;
Packit 98cdb6
  GtkTextTag *tag;
Packit 98cdb6
Packit 98cdb6
  g_assert (peek_state (info) == STATE_TEXT ||
Packit 98cdb6
	    peek_state (info) == STATE_APPLY_TAG);
Packit 98cdb6
Packit 98cdb6
  if (ELEMENT_IS ("apply_tag"))
Packit 98cdb6
    {
Packit 98cdb6
      if (!locate_attributes (context, element_name, attribute_names, attribute_values, TRUE, error,
Packit 98cdb6
			      "priority", &priority, NULL))
Packit 98cdb6
	return;
Packit 98cdb6
Packit 98cdb6
      if (!check_id_or_name (context, element_name, attribute_names, attribute_values,
Packit 98cdb6
			     &id, &name, error))
Packit 98cdb6
	return;
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
      tag = tag_exists (context, name, id, info, error);
Packit 98cdb6
Packit 98cdb6
      if (!tag)
Packit 98cdb6
	return;
Packit 98cdb6
Packit 98cdb6
      info->tag_stack = g_slist_prepend (info->tag_stack, tag);
Packit 98cdb6
Packit 98cdb6
      push_state (info, STATE_APPLY_TAG);
Packit 98cdb6
    }
Packit 98cdb6
  else if (ELEMENT_IS ("pixbuf"))
Packit 98cdb6
    {
Packit 98cdb6
      int int_id;
Packit 98cdb6
      GdkPixbuf *pixbuf;
Packit 98cdb6
      TextSpan *span;
Packit 98cdb6
      const gchar *pixbuf_id;
Packit 98cdb6
Packit 98cdb6
      if (!locate_attributes (context, element_name, attribute_names, attribute_values, FALSE, error,
Packit 98cdb6
			      "index", &pixbuf_id, NULL))
Packit 98cdb6
	return;
Packit 98cdb6
Packit 98cdb6
      int_id = atoi (pixbuf_id);
Packit 98cdb6
      pixbuf = get_pixbuf_from_headers (info->headers, int_id, error);
Packit 98cdb6
Packit 98cdb6
      span = g_new0 (TextSpan, 1);
Packit 98cdb6
      span->pixbuf = pixbuf;
Packit 98cdb6
      span->tags = NULL;
Packit 98cdb6
Packit 98cdb6
      info->spans = g_list_prepend (info->spans, span);
Packit 98cdb6
Packit 98cdb6
      if (!pixbuf)
Packit 98cdb6
	return;
Packit 98cdb6
Packit 98cdb6
      push_state (info, STATE_PIXBUF);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    set_error (error, context,
Packit 98cdb6
	       G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
	       _("Element <%s> is not allowed below <%s>"),
Packit 98cdb6
	       element_name, peek_state(info) == STATE_TEXT ? "text" : "apply_tag");
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
parse_attr_element (GMarkupParseContext  *context,
Packit 98cdb6
		    const gchar          *element_name,
Packit 98cdb6
		    const gchar         **attribute_names,
Packit 98cdb6
		    const gchar         **attribute_values,
Packit 98cdb6
		    ParseInfo            *info,
Packit 98cdb6
		    GError              **error)
Packit 98cdb6
{
Packit 98cdb6
  const gchar *name, *type, *value;
Packit 98cdb6
  GType gtype;
Packit 98cdb6
  GValue gvalue = { 0 };
Packit 98cdb6
  GParamSpec *pspec;
Packit 98cdb6
Packit 98cdb6
  g_assert (peek_state (info) == STATE_TAG);
Packit 98cdb6
Packit 98cdb6
  if (ELEMENT_IS ("attr"))
Packit 98cdb6
    {
Packit 98cdb6
      if (!locate_attributes (context, element_name, attribute_names, attribute_values, FALSE, error,
Packit 98cdb6
			      "name", &name, "type", &type, "value", &value, NULL))
Packit 98cdb6
	return;
Packit 98cdb6
Packit 98cdb6
      gtype = g_type_from_name (type);
Packit 98cdb6
Packit 98cdb6
      if (gtype == G_TYPE_INVALID)
Packit 98cdb6
	{
Packit 98cdb6
	  set_error (error, context,
Packit 98cdb6
		     G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
		     _("\"%s\" is not a valid attribute type"), type);
Packit 98cdb6
	  return;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      if (!(pspec = g_object_class_find_property (G_OBJECT_GET_CLASS (info->current_tag), name)))
Packit 98cdb6
	{
Packit 98cdb6
	  set_error (error, context,
Packit 98cdb6
		     G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
		     _("\"%s\" is not a valid attribute name"), name);
Packit 98cdb6
	  return;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      g_value_init (&gvalue, gtype);
Packit 98cdb6
Packit 98cdb6
      if (!deserialize_value (value, &gvalue))
Packit 98cdb6
	{
Packit 98cdb6
	  set_error (error, context,
Packit 98cdb6
		     G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
		     _("\"%s\" could not be converted to a value of type \"%s\" for attribute \"%s\""),
Packit 98cdb6
		     value, type, name);
Packit 98cdb6
	  return;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      if (g_param_value_validate (pspec, &gvalue))
Packit 98cdb6
	{
Packit 98cdb6
	  set_error (error, context,
Packit 98cdb6
		     G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
		     _("\"%s\" is not a valid value for attribute \"%s\""),
Packit 98cdb6
		     value, name);
Packit 98cdb6
	  g_value_unset (&gvalue);
Packit 98cdb6
	  return;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      g_object_set_property (G_OBJECT (info->current_tag),
Packit 98cdb6
			     name, &gvalue);
Packit 98cdb6
Packit 98cdb6
      g_value_unset (&gvalue);
Packit 98cdb6
Packit 98cdb6
      push_state (info, STATE_ATTR);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      set_error (error, context,
Packit 98cdb6
                 G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
                 _("Element <%s> is not allowed below <%s>"),
Packit 98cdb6
                 element_name, "tag");
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
static gchar *
Packit 98cdb6
get_tag_name (ParseInfo   *info,
Packit 98cdb6
	      const gchar *tag_name)
Packit 98cdb6
{
Packit 98cdb6
  gchar *name;
Packit 98cdb6
  gint i;
Packit 98cdb6
Packit 98cdb6
  name = g_strdup (tag_name);
Packit 98cdb6
Packit 98cdb6
  if (!info->create_tags)
Packit 98cdb6
    return name;
Packit 98cdb6
Packit 98cdb6
  i = 0;
Packit 98cdb6
Packit 98cdb6
  while (gtk_text_tag_table_lookup (info->buffer->tag_table, name) != NULL)
Packit 98cdb6
    {
Packit 98cdb6
      g_free (name);
Packit 98cdb6
      name = g_strdup_printf ("%s-%d", tag_name, ++i);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (i != 0)
Packit 98cdb6
    {
Packit 98cdb6
      g_hash_table_insert (info->substitutions, g_strdup (tag_name), g_strdup (name));
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return name;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
parse_tag_element (GMarkupParseContext  *context,
Packit 98cdb6
		   const gchar          *element_name,
Packit 98cdb6
		   const gchar         **attribute_names,
Packit 98cdb6
		   const gchar         **attribute_values,
Packit 98cdb6
		   ParseInfo            *info,
Packit 98cdb6
		   GError              **error)
Packit 98cdb6
{
Packit 98cdb6
  const gchar *name, *priority;
Packit 98cdb6
  gchar *tag_name;
Packit 98cdb6
  gint id;
Packit 98cdb6
  gint prio;
Packit 98cdb6
  gchar *tmp;
Packit 98cdb6
Packit 98cdb6
  g_assert (peek_state (info) == STATE_TAGS);
Packit 98cdb6
Packit 98cdb6
  if (ELEMENT_IS ("tag"))
Packit 98cdb6
    {
Packit 98cdb6
      if (!locate_attributes (context, element_name, attribute_names, attribute_values, TRUE, error,
Packit 98cdb6
			      "priority", &priority, NULL))
Packit 98cdb6
	return;
Packit 98cdb6
Packit 98cdb6
      if (!check_id_or_name (context, element_name, attribute_names, attribute_values,
Packit 98cdb6
			     &id, &name, error))
Packit 98cdb6
	return;
Packit 98cdb6
Packit 98cdb6
      if (name)
Packit 98cdb6
	{
Packit 98cdb6
	  if (g_hash_table_lookup (info->defined_tags, name) != NULL)
Packit 98cdb6
	    {
Packit 98cdb6
	      set_error (error, context,
Packit 98cdb6
			 G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
			 _("Tag \"%s\" already defined"), name);
Packit 98cdb6
	      return;
Packit 98cdb6
	    }
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      prio = strtol (priority, &tmp, 10);
Packit 98cdb6
Packit 98cdb6
      if (tmp == NULL || tmp == priority)
Packit 98cdb6
	{
Packit 98cdb6
	  set_error (error, context,
Packit 98cdb6
		     G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
		     _("Tag \"%s\" has invalid priority \"%s\""), name, priority);
Packit 98cdb6
	  return;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      if (name)
Packit 98cdb6
	{
Packit 98cdb6
	  tag_name = get_tag_name (info, name);
Packit 98cdb6
	  info->current_tag = gtk_text_tag_new (tag_name);
Packit 98cdb6
	  g_free (tag_name);
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	{
Packit 98cdb6
	  info->current_tag = gtk_text_tag_new (NULL);
Packit 98cdb6
	  info->current_tag_id = id;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      info->current_tag_prio = prio;
Packit 98cdb6
Packit 98cdb6
      push_state (info, STATE_TAG);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      set_error (error, context,
Packit 98cdb6
                 G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
                 _("Element <%s> is not allowed below <%s>"),
Packit 98cdb6
                 element_name, "tags");
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
start_element_handler (GMarkupParseContext  *context,
Packit 98cdb6
		       const gchar          *element_name,
Packit 98cdb6
		       const gchar         **attribute_names,
Packit 98cdb6
		       const gchar         **attribute_values,
Packit 98cdb6
		       gpointer              user_data,
Packit 98cdb6
		       GError              **error)
Packit 98cdb6
{
Packit 98cdb6
  ParseInfo *info = user_data;
Packit 98cdb6
Packit 98cdb6
  switch (peek_state (info))
Packit 98cdb6
    {
Packit 98cdb6
    case STATE_START:
Packit 98cdb6
      if (ELEMENT_IS ("text_view_markup"))
Packit 98cdb6
	{
Packit 98cdb6
	  if (!check_no_attributes (context, element_name,
Packit 98cdb6
				    attribute_names, attribute_values, error))
Packit 98cdb6
	    return;
Packit 98cdb6
Packit 98cdb6
	  push_state (info, STATE_TEXT_VIEW_MARKUP);
Packit 98cdb6
	  break;
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
        set_error (error, context, G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
                   _("Outermost element in text must be <text_view_markup> not <%s>"),
Packit 98cdb6
                   element_name);
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_TEXT_VIEW_MARKUP:
Packit 98cdb6
      if (ELEMENT_IS ("tags"))
Packit 98cdb6
	{
Packit 98cdb6
	  if (info->parsed_tags)
Packit 98cdb6
	    {
Packit 98cdb6
	      set_error (error, context, G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
			 _("A <%s> element has already been specified"), "tags");
Packit 98cdb6
	      return;
Packit 98cdb6
	    }
Packit 98cdb6
Packit 98cdb6
	  if (!check_no_attributes (context, element_name,
Packit 98cdb6
				    attribute_names, attribute_values, error))
Packit 98cdb6
	    return;
Packit 98cdb6
Packit 98cdb6
	  push_state (info, STATE_TAGS);
Packit 98cdb6
	  break;
Packit 98cdb6
	}
Packit 98cdb6
      else if (ELEMENT_IS ("text"))
Packit 98cdb6
	{
Packit 98cdb6
	  if (info->parsed_text)
Packit 98cdb6
	    {
Packit 98cdb6
	      set_error (error, context, G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
			 _("A <%s> element has already been specified"), "text");
Packit 98cdb6
	      return;
Packit 98cdb6
	    }
Packit 98cdb6
	  else if (!info->parsed_tags)
Packit 98cdb6
	    {
Packit 98cdb6
	      set_error (error, context, G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
			 _("A <text> element can't occur before a <tags> element"));
Packit 98cdb6
	      return;
Packit 98cdb6
	    }
Packit 98cdb6
Packit 98cdb6
	  if (!check_no_attributes (context, element_name,
Packit 98cdb6
				    attribute_names, attribute_values, error))
Packit 98cdb6
	    return;
Packit 98cdb6
Packit 98cdb6
	  push_state (info, STATE_TEXT);
Packit 98cdb6
	  break;
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	set_error (error, context,
Packit 98cdb6
		   G_MARKUP_ERROR, G_MARKUP_ERROR_PARSE,
Packit 98cdb6
                 _("Element <%s> is not allowed below <%s>"),
Packit 98cdb6
		   element_name, "text_view_markup");
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_TAGS:
Packit 98cdb6
      parse_tag_element (context, element_name,
Packit 98cdb6
			 attribute_names, attribute_values,
Packit 98cdb6
			 info, error);
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_TAG:
Packit 98cdb6
      parse_attr_element (context, element_name,
Packit 98cdb6
			  attribute_names, attribute_values,
Packit 98cdb6
			  info, error);
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_TEXT:
Packit 98cdb6
    case STATE_APPLY_TAG:
Packit 98cdb6
      parse_apply_tag_element (context, element_name,
Packit 98cdb6
			       attribute_names, attribute_values,
Packit 98cdb6
			       info, error);
Packit 98cdb6
      break;
Packit 98cdb6
    default:
Packit 98cdb6
      g_assert_not_reached ();
Packit 98cdb6
      break;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gint
Packit 98cdb6
sort_tag_prio (TextTagPrio *a,
Packit 98cdb6
	       TextTagPrio *b)
Packit 98cdb6
{
Packit 98cdb6
  if (a->prio < b->prio)
Packit 98cdb6
    return -1;
Packit 98cdb6
  else if (a->prio > b->prio)
Packit 98cdb6
    return 1;
Packit 98cdb6
  else
Packit 98cdb6
    return 0;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
end_element_handler (GMarkupParseContext  *context,
Packit 98cdb6
		     const gchar          *element_name,
Packit 98cdb6
		     gpointer              user_data,
Packit 98cdb6
		     GError              **error)
Packit 98cdb6
{
Packit 98cdb6
  ParseInfo *info = user_data;
Packit 98cdb6
  gchar *tmp;
Packit 98cdb6
  GList *list;
Packit 98cdb6
Packit 98cdb6
  switch (peek_state (info))
Packit 98cdb6
    {
Packit 98cdb6
    case STATE_TAGS:
Packit 98cdb6
      pop_state (info);
Packit 98cdb6
      g_assert (peek_state (info) == STATE_TEXT_VIEW_MARKUP);
Packit 98cdb6
Packit 98cdb6
      info->parsed_tags = TRUE;
Packit 98cdb6
Packit 98cdb6
      /* Sort list and add the tags */
Packit 98cdb6
      info->tag_priorities = g_list_sort (info->tag_priorities,
Packit 98cdb6
					  (GCompareFunc)sort_tag_prio);
Packit 98cdb6
      list = info->tag_priorities;
Packit 98cdb6
      while (list)
Packit 98cdb6
	{
Packit 98cdb6
	  TextTagPrio *prio = list->data;
Packit 98cdb6
Packit 98cdb6
	  if (info->create_tags)
Packit 98cdb6
	    gtk_text_tag_table_add (info->buffer->tag_table, prio->tag);
Packit 98cdb6
Packit 98cdb6
	  g_object_unref (prio->tag);
Packit 98cdb6
	  prio->tag = NULL;
Packit 98cdb6
Packit 98cdb6
	  list = list->next;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_TAG:
Packit 98cdb6
      pop_state (info);
Packit 98cdb6
      g_assert (peek_state (info) == STATE_TAGS);
Packit 98cdb6
Packit 98cdb6
      if (info->current_tag->name)
Packit 98cdb6
	{
Packit 98cdb6
	  /* Add tag to defined tags hash */
Packit 98cdb6
	  tmp = g_strdup (info->current_tag->name);
Packit 98cdb6
	  g_hash_table_insert (info->defined_tags,
Packit 98cdb6
			       tmp, tmp);
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	{
Packit 98cdb6
	  g_hash_table_insert (info->anonymous_tags,
Packit 98cdb6
			       GINT_TO_POINTER (info->current_tag_id),
Packit 98cdb6
			       info->current_tag);
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      if (info->create_tags)
Packit 98cdb6
	{
Packit 98cdb6
	  TextTagPrio *prio;
Packit 98cdb6
Packit 98cdb6
	  /* add the tag to the list */
Packit 98cdb6
	  prio = g_new0 (TextTagPrio, 1);
Packit 98cdb6
	  prio->prio = info->current_tag_prio;
Packit 98cdb6
	  prio->tag = info->current_tag;
Packit 98cdb6
Packit 98cdb6
 	  info->tag_priorities = g_list_prepend (info->tag_priorities, prio);
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      info->current_tag = NULL;
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_ATTR:
Packit 98cdb6
      pop_state (info);
Packit 98cdb6
      g_assert (peek_state (info) == STATE_TAG);
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_APPLY_TAG:
Packit 98cdb6
      pop_state (info);
Packit 98cdb6
      g_assert (peek_state (info) == STATE_APPLY_TAG ||
Packit 98cdb6
		peek_state (info) == STATE_TEXT);
Packit 98cdb6
Packit 98cdb6
      /* Pop tag */
Packit 98cdb6
      info->tag_stack = g_slist_delete_link (info->tag_stack,
Packit 98cdb6
					     info->tag_stack);
Packit 98cdb6
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_TEXT:
Packit 98cdb6
      pop_state (info);
Packit 98cdb6
      g_assert (peek_state (info) == STATE_TEXT_VIEW_MARKUP);
Packit 98cdb6
Packit 98cdb6
      info->spans = g_list_reverse (info->spans);
Packit 98cdb6
      info->parsed_text = TRUE;
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_TEXT_VIEW_MARKUP:
Packit 98cdb6
      pop_state (info);
Packit 98cdb6
      g_assert (peek_state (info) == STATE_START);
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_PIXBUF:
Packit 98cdb6
      pop_state (info);
Packit 98cdb6
      g_assert (peek_state (info) == STATE_APPLY_TAG ||
Packit 98cdb6
		peek_state (info) == STATE_TEXT);
Packit 98cdb6
      break;
Packit 98cdb6
    default:
Packit 98cdb6
      g_assert_not_reached ();
Packit 98cdb6
      break;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
all_whitespace (const char *text,
Packit 98cdb6
                int         text_len)
Packit 98cdb6
{
Packit 98cdb6
  const char *p;
Packit 98cdb6
  const char *end;
Packit 98cdb6
Packit 98cdb6
  p = text;
Packit 98cdb6
  end = text + text_len;
Packit 98cdb6
Packit 98cdb6
  while (p != end)
Packit 98cdb6
    {
Packit 98cdb6
      if (!g_ascii_isspace (*p))
Packit 98cdb6
        return FALSE;
Packit 98cdb6
Packit 98cdb6
      p = g_utf8_next_char (p);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
text_handler (GMarkupParseContext  *context,
Packit 98cdb6
	      const gchar          *text,
Packit 98cdb6
	      gsize                 text_len,
Packit 98cdb6
	      gpointer              user_data,
Packit 98cdb6
	      GError              **error)
Packit 98cdb6
{
Packit 98cdb6
  ParseInfo *info = user_data;
Packit 98cdb6
  TextSpan *span;
Packit 98cdb6
Packit 98cdb6
  if (all_whitespace (text, text_len) &&
Packit 98cdb6
      peek_state (info) != STATE_TEXT &&
Packit 98cdb6
      peek_state (info) != STATE_APPLY_TAG)
Packit 98cdb6
    return;
Packit 98cdb6
Packit 98cdb6
  switch (peek_state (info))
Packit 98cdb6
    {
Packit 98cdb6
    case STATE_START:
Packit 98cdb6
      g_assert_not_reached (); /* gmarkup shouldn't do this */
Packit 98cdb6
      break;
Packit 98cdb6
    case STATE_TEXT:
Packit 98cdb6
    case STATE_APPLY_TAG:
Packit 98cdb6
      if (text_len == 0)
Packit 98cdb6
	return;
Packit 98cdb6
Packit 98cdb6
      span = g_new0 (TextSpan, 1);
Packit 98cdb6
      span->text = g_strndup (text, text_len);
Packit 98cdb6
      span->tags = g_slist_copy (info->tag_stack);
Packit 98cdb6
Packit 98cdb6
      info->spans = g_list_prepend (info->spans, span);
Packit 98cdb6
      break;
Packit 98cdb6
    default:
Packit 98cdb6
      g_assert_not_reached ();
Packit 98cdb6
      break;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
parse_info_init (ParseInfo     *info,
Packit 98cdb6
		 GtkTextBuffer *buffer,
Packit 98cdb6
		 gboolean       create_tags,
Packit 98cdb6
		 GList         *headers)
Packit 98cdb6
{
Packit 98cdb6
  info->states = g_slist_prepend (NULL, GINT_TO_POINTER (STATE_START));
Packit 98cdb6
Packit 98cdb6
  info->create_tags = create_tags;
Packit 98cdb6
  info->headers = headers;
Packit 98cdb6
  info->defined_tags = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
Packit 98cdb6
  info->substitutions = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
Packit 98cdb6
  info->anonymous_tags = g_hash_table_new_full (NULL, NULL, NULL, NULL);
Packit 98cdb6
  info->tag_stack = NULL;
Packit 98cdb6
  info->spans = NULL;
Packit 98cdb6
  info->parsed_text = FALSE;
Packit 98cdb6
  info->parsed_tags = FALSE;
Packit 98cdb6
  info->current_tag = NULL;
Packit 98cdb6
  info->current_tag_prio = -1;
Packit 98cdb6
  info->tag_priorities = NULL;
Packit 98cdb6
Packit 98cdb6
  info->buffer = buffer;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
text_span_free (TextSpan *span)
Packit 98cdb6
{
Packit 98cdb6
  g_free (span->text);
Packit 98cdb6
  g_slist_free (span->tags);
Packit 98cdb6
  g_free (span);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
parse_info_free (ParseInfo *info)
Packit 98cdb6
{
Packit 98cdb6
  GList *list;
Packit 98cdb6
Packit 98cdb6
  g_slist_free (info->tag_stack);
Packit 98cdb6
  g_slist_free (info->states);
Packit 98cdb6
Packit 98cdb6
  g_hash_table_destroy (info->substitutions);
Packit 98cdb6
  g_hash_table_destroy (info->defined_tags);
Packit 98cdb6
Packit 98cdb6
  if (info->current_tag)
Packit 98cdb6
    g_object_unref (info->current_tag);
Packit 98cdb6
Packit 98cdb6
  list = info->spans;
Packit 98cdb6
  while (list)
Packit 98cdb6
    {
Packit 98cdb6
      text_span_free (list->data);
Packit 98cdb6
Packit 98cdb6
      list = list->next;
Packit 98cdb6
    }
Packit 98cdb6
  g_list_free (info->spans);
Packit 98cdb6
Packit 98cdb6
  list = info->tag_priorities;
Packit 98cdb6
  while (list)
Packit 98cdb6
    {
Packit 98cdb6
      TextTagPrio *prio = list->data;
Packit 98cdb6
Packit 98cdb6
      if (prio->tag)
Packit 98cdb6
	g_object_unref (prio->tag);
Packit 98cdb6
      g_free (prio);
Packit 98cdb6
Packit 98cdb6
      list = list->next;
Packit 98cdb6
    }
Packit 98cdb6
  g_list_free (info->tag_priorities);
Packit 98cdb6
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
insert_text (ParseInfo   *info,
Packit 98cdb6
	     GtkTextIter *iter)
Packit 98cdb6
{
Packit 98cdb6
  GtkTextIter start_iter;
Packit 98cdb6
  GtkTextMark *mark;
Packit 98cdb6
  GList *tmp;
Packit 98cdb6
  GSList *tags;
Packit 98cdb6
Packit 98cdb6
  start_iter = *iter;
Packit 98cdb6
Packit 98cdb6
  mark = gtk_text_buffer_create_mark (info->buffer, "deserialize_insert_point",
Packit 98cdb6
  				      &start_iter, TRUE);
Packit 98cdb6
Packit 98cdb6
  tmp = info->spans;
Packit 98cdb6
  while (tmp)
Packit 98cdb6
    {
Packit 98cdb6
      TextSpan *span = tmp->data;
Packit 98cdb6
Packit 98cdb6
      if (span->text)
Packit 98cdb6
	gtk_text_buffer_insert (info->buffer, iter, span->text, -1);
Packit 98cdb6
      else
Packit 98cdb6
	{
Packit 98cdb6
	  gtk_text_buffer_insert_pixbuf (info->buffer, iter, span->pixbuf);
Packit 98cdb6
	  g_object_unref (span->pixbuf);
Packit 98cdb6
	}
Packit 98cdb6
      gtk_text_buffer_get_iter_at_mark (info->buffer, &start_iter, mark);
Packit 98cdb6
Packit 98cdb6
      /* Apply tags */
Packit 98cdb6
      tags = span->tags;
Packit 98cdb6
      while (tags)
Packit 98cdb6
	{
Packit 98cdb6
	  GtkTextTag *tag = tags->data;
Packit 98cdb6
Packit 98cdb6
	  gtk_text_buffer_apply_tag (info->buffer, tag,
Packit 98cdb6
				     &start_iter, iter);
Packit 98cdb6
Packit 98cdb6
	  tags = tags->next;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      gtk_text_buffer_move_mark (info->buffer, mark, iter);
Packit 98cdb6
Packit 98cdb6
      tmp = tmp->next;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  gtk_text_buffer_delete_mark (info->buffer, mark);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
static int
Packit 98cdb6
read_int (const guchar *start)
Packit 98cdb6
{
Packit 98cdb6
  int result;
Packit 98cdb6
Packit 98cdb6
  result =
Packit 98cdb6
    start[0] << 24 |
Packit 98cdb6
    start[1] << 16 |
Packit 98cdb6
    start[2] << 8 |
Packit 98cdb6
    start[3];
Packit 98cdb6
Packit 98cdb6
  return result;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
header_is (Header      *header,
Packit 98cdb6
           const gchar *id)
Packit 98cdb6
{
Packit 98cdb6
  return (strncmp (header->id, id, strlen (id)) == 0);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GList *
Packit 98cdb6
read_headers (const gchar *start,
Packit 98cdb6
	      gint         len,
Packit 98cdb6
	      GError     **error)
Packit 98cdb6
{
Packit 98cdb6
  int i = 0;
Packit 98cdb6
  int section_len;
Packit 98cdb6
  Header *header;
Packit 98cdb6
  GList *headers = NULL;
Packit 98cdb6
Packit 98cdb6
  while (i < len)
Packit 98cdb6
    {
Packit 98cdb6
      if (i + 30 >= len)
Packit 98cdb6
	goto error;
Packit 98cdb6
Packit 98cdb6
      if (strncmp (start + i, "GTKTEXTBUFFERCONTENTS-0001", 26) == 0 ||
Packit 98cdb6
	  strncmp (start + i, "GTKTEXTBUFFERPIXBDATA-0001", 26) == 0)
Packit 98cdb6
	{
Packit 98cdb6
	  section_len = read_int ((const guchar *) start + i + 26);
Packit 98cdb6
Packit 98cdb6
	  if (i + 30 + section_len > len)
Packit 98cdb6
	    goto error;
Packit 98cdb6
Packit 98cdb6
	  header = g_new0 (Header, 1);
Packit 98cdb6
	  header->id = start + i;
Packit 98cdb6
	  header->length = section_len;
Packit 98cdb6
	  header->start = start + i + 30;
Packit 98cdb6
Packit 98cdb6
	  i += 30 + section_len;
Packit 98cdb6
Packit 98cdb6
	  headers = g_list_prepend (headers, header);
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	break;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return g_list_reverse (headers);
Packit 98cdb6
Packit 98cdb6
 error:
Packit 98cdb6
  g_list_foreach (headers, (GFunc) g_free, NULL);
Packit 98cdb6
  g_list_free (headers);
Packit 98cdb6
Packit 98cdb6
  g_set_error_literal (error,
Packit 98cdb6
                       G_MARKUP_ERROR,
Packit 98cdb6
                       G_MARKUP_ERROR_PARSE,
Packit 98cdb6
                       _("Serialized data is malformed"));
Packit 98cdb6
Packit 98cdb6
  return NULL;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
deserialize_text (GtkTextBuffer *buffer,
Packit 98cdb6
		  GtkTextIter   *iter,
Packit 98cdb6
		  const gchar   *text,
Packit 98cdb6
		  gint           len,
Packit 98cdb6
		  gboolean       create_tags,
Packit 98cdb6
		  GError       **error,
Packit 98cdb6
		  GList         *headers)
Packit 98cdb6
{
Packit 98cdb6
  GMarkupParseContext *context;
Packit 98cdb6
  ParseInfo info;
Packit 98cdb6
  gboolean retval = FALSE;
Packit 98cdb6
Packit 98cdb6
  static const GMarkupParser rich_text_parser = {
Packit 98cdb6
    start_element_handler,
Packit 98cdb6
    end_element_handler,
Packit 98cdb6
    text_handler,
Packit 98cdb6
    NULL,
Packit 98cdb6
    NULL
Packit 98cdb6
  };
Packit 98cdb6
Packit 98cdb6
  parse_info_init (&info, buffer, create_tags, headers);
Packit 98cdb6
Packit 98cdb6
  context = g_markup_parse_context_new (&rich_text_parser,
Packit 98cdb6
                                        0, &info, NULL);
Packit 98cdb6
Packit 98cdb6
  if (!g_markup_parse_context_parse (context,
Packit 98cdb6
                                     text,
Packit 98cdb6
                                     len,
Packit 98cdb6
                                     error))
Packit 98cdb6
    goto out;
Packit 98cdb6
Packit 98cdb6
  if (!g_markup_parse_context_end_parse (context, error))
Packit 98cdb6
    goto out;
Packit 98cdb6
Packit 98cdb6
  retval = TRUE;
Packit 98cdb6
Packit 98cdb6
  /* Now insert the text */
Packit 98cdb6
  insert_text (&info, iter);
Packit 98cdb6
Packit 98cdb6
 out:
Packit 98cdb6
  parse_info_free (&info;;
Packit 98cdb6
Packit 98cdb6
  g_markup_parse_context_free (context);
Packit 98cdb6
Packit 98cdb6
  return retval;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
gboolean
Packit 98cdb6
_gtk_text_buffer_deserialize_rich_text (GtkTextBuffer *register_buffer,
Packit 98cdb6
                                        GtkTextBuffer *content_buffer,
Packit 98cdb6
                                        GtkTextIter   *iter,
Packit 98cdb6
                                        const guint8  *text,
Packit 98cdb6
                                        gsize          length,
Packit 98cdb6
                                        gboolean       create_tags,
Packit 98cdb6
                                        gpointer       user_data,
Packit 98cdb6
                                        GError       **error)
Packit 98cdb6
{
Packit 98cdb6
  GList *headers;
Packit 98cdb6
  Header *header;
Packit 98cdb6
  gboolean retval;
Packit 98cdb6
Packit 98cdb6
  headers = read_headers ((gchar *) text, length, error);
Packit 98cdb6
Packit 98cdb6
  if (!headers)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  header = headers->data;
Packit 98cdb6
  if (!header_is (header, "GTKTEXTBUFFERCONTENTS-0001"))
Packit 98cdb6
    {
Packit 98cdb6
      g_set_error_literal (error,
Packit 98cdb6
                           G_MARKUP_ERROR,
Packit 98cdb6
                           G_MARKUP_ERROR_PARSE,
Packit 98cdb6
                           _("Serialized data is malformed. First section isn't GTKTEXTBUFFERCONTENTS-0001"));
Packit 98cdb6
Packit 98cdb6
      retval = FALSE;
Packit 98cdb6
      goto out;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  retval = deserialize_text (content_buffer, iter,
Packit 98cdb6
			     header->start, header->length,
Packit 98cdb6
			     create_tags, error, headers->next);
Packit 98cdb6
Packit 98cdb6
 out:
Packit 98cdb6
  g_list_foreach (headers, (GFunc)g_free, NULL);
Packit 98cdb6
  g_list_free (headers);
Packit 98cdb6
Packit 98cdb6
  return retval;
Packit 98cdb6
}