/*
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) version 3.
 *
 * 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with the program; if not, see <http://www.gnu.org/licenses/>
 *
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 *
 */

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include <string.h>

#include <libxml/tree.h>
#include <libxml/parser.h>
#include <libxml/xmlmemory.h>

#include <glib/gi18n-lib.h>
#include <gconf/gconf-client.h>

#include <libedataserver/e-uid.h>
#include <libedataserver/e-data-server-util.h>

#include "e-signature.h"

struct _ESignaturePrivate {
	gchar *filename;
	gchar *name;
	gchar *uid;

	gboolean autogenerated;
	gboolean is_html;
	gboolean is_script;
};

enum {
	PROP_0,
	PROP_AUTOGENERATED,
	PROP_FILENAME,
	PROP_IS_HTML,
	PROP_IS_SCRIPT,
	PROP_NAME,
	PROP_UID
};

G_DEFINE_TYPE (
	ESignature,
	e_signature,
	G_TYPE_OBJECT)

static gboolean
xml_set_bool (xmlNodePtr node,
              const gchar *name,
              gboolean *val)
{
	gboolean v_boolean;
	gchar *buf;

	if ((buf = (gchar *)xmlGetProp (node, (xmlChar *) name))) {
		v_boolean = (!strcmp (buf, "true") || !strcmp (buf, "yes"));
		xmlFree (buf);

		if (v_boolean != *val) {
			*val = v_boolean;
			return TRUE;
		}
	}

	return FALSE;
}

static gboolean
xml_set_prop (xmlNodePtr node,
              const gchar *name,
              gchar **val)
{
	gchar *buf, *new_val;

	buf = (gchar *)xmlGetProp (node, (xmlChar *) name);
	new_val = g_strdup (buf);
	xmlFree (buf);

	/* We can use strcmp here whether the value is UTF8 or
	 * not, since we only care if the bytes changed.
	 */
	if (!*val || strcmp (*val, new_val)) {
		g_free (*val);
		*val = new_val;
		return TRUE;
	} else {
		g_free (new_val);
		return FALSE;
	}
}

static gboolean
xml_set_content (xmlNodePtr node,
                 gchar **val)
{
	gchar *buf, *new_val;

	buf = (gchar *)xmlNodeGetContent (node);
	new_val = g_strdup (buf);
	xmlFree (buf);

	/* We can use strcmp here whether the value is UTF8 or
	 * not, since we only care if the bytes changed. */
	if (!*val || strcmp (*val, new_val)) {
		g_free (*val);
		*val = new_val;
		return TRUE;
	} else {
		g_free (new_val);
		return FALSE;
	}
}

static void
signature_set_property (GObject *object,
                        guint property_id,
                        const GValue *value,
                        GParamSpec *pspec)
{
	switch (property_id) {
		case PROP_AUTOGENERATED:
			e_signature_set_autogenerated (
				E_SIGNATURE (object),
				g_value_get_boolean (value));
			return;

		case PROP_FILENAME:
			e_signature_set_filename (
				E_SIGNATURE (object),
				g_value_get_string (value));
			return;

		case PROP_IS_HTML:
			e_signature_set_is_html (
				E_SIGNATURE (object),
				g_value_get_boolean (value));
			return;

		case PROP_IS_SCRIPT:
			e_signature_set_is_script (
				E_SIGNATURE (object),
				g_value_get_boolean (value));
			return;

		case PROP_NAME:
			e_signature_set_name (
				E_SIGNATURE (object),
				g_value_get_string (value));
			return;

		case PROP_UID:
			e_signature_set_uid (
				E_SIGNATURE (object),
				g_value_get_string (value));
			return;
	}

	G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
signature_get_property (GObject *object,
                        guint property_id,
                        GValue *value,
                        GParamSpec *pspec)
{
	switch (property_id) {
		case PROP_AUTOGENERATED:
			g_value_set_boolean (
				value, e_signature_get_autogenerated (
				E_SIGNATURE (object)));
			return;

		case PROP_FILENAME:
			g_value_set_string (
				value, e_signature_get_filename (
				E_SIGNATURE (object)));
			return;

		case PROP_IS_HTML:
			g_value_set_boolean (
				value, e_signature_get_is_html (
				E_SIGNATURE (object)));
			return;

		case PROP_IS_SCRIPT:
			g_value_set_boolean (
				value, e_signature_get_is_script (
				E_SIGNATURE (object)));
			return;

		case PROP_NAME:
			g_value_set_string (
				value, e_signature_get_name (
				E_SIGNATURE (object)));
			return;

		case PROP_UID:
			g_value_set_string (
				value, e_signature_get_uid (
				E_SIGNATURE (object)));
			return;
	}

	G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
signature_finalize (GObject *object)
{
	ESignaturePrivate *priv;

	priv = E_SIGNATURE (object)->priv;

	g_free (priv->filename);
	g_free (priv->name);
	g_free (priv->uid);

	/* Chain up to parent's finalize() method. */
	G_OBJECT_CLASS (e_signature_parent_class)->finalize (object);
}

static void
e_signature_class_init (ESignatureClass *class)
{
	GObjectClass *object_class;

	g_type_class_add_private (class, sizeof (ESignaturePrivate));

	object_class = G_OBJECT_CLASS (class);
	object_class->set_property = signature_set_property;
	object_class->get_property = signature_get_property;
	object_class->finalize = signature_finalize;

	g_object_class_install_property (
		object_class,
		PROP_AUTOGENERATED,
		g_param_spec_boolean (
			"autogenerated",
			"Autogenerated",
			NULL,
			FALSE,
			G_PARAM_READWRITE |
			G_PARAM_CONSTRUCT));

	g_object_class_install_property (
		object_class,
		PROP_FILENAME,
		g_param_spec_string (
			"filename",
			"Filename",
			NULL,
			NULL,
			G_PARAM_READWRITE |
			G_PARAM_CONSTRUCT));

	g_object_class_install_property (
		object_class,
		PROP_IS_HTML,
		g_param_spec_boolean (
			"is-html",
			"Is HTML",
			NULL,
			FALSE,
			G_PARAM_READWRITE |
			G_PARAM_CONSTRUCT));

	g_object_class_install_property (
		object_class,
		PROP_IS_SCRIPT,
		g_param_spec_boolean (
			"is-script",
			"Is Script",
			NULL,
			FALSE,
			G_PARAM_READWRITE |
			G_PARAM_CONSTRUCT));

	g_object_class_install_property (
		object_class,
		PROP_NAME,
		g_param_spec_string (
			"name",
			"Name",
			NULL,
			NULL,
			G_PARAM_READWRITE |
			G_PARAM_CONSTRUCT));

	g_object_class_install_property (
		object_class,
		PROP_UID,
		g_param_spec_string (
			"uid",
			"UID",
			NULL,
			NULL,
			G_PARAM_READWRITE |
			G_PARAM_CONSTRUCT));
}

static void
e_signature_init (ESignature *signature)
{
	signature->priv = G_TYPE_INSTANCE_GET_PRIVATE (
		signature, E_TYPE_SIGNATURE, ESignaturePrivate);
}

/**
 * e_signature_new:
 *
 * Returns a new signature which can be filled in and
 * added to an #ESignatureList.
 *
 * Returns: a new #ESignature
 **/
ESignature *
e_signature_new (void)
{
	ESignature *signature;

	signature = g_object_new (E_TYPE_SIGNATURE, NULL);
	signature->priv->uid = e_uid_new ();

	return signature;
}

/**
 * e_signature_new_from_xml:
 * @xml: an XML signature description
 *
 * Return value: a new #ESignature based on the data in @xml, or %NULL
 * if @xml could not be parsed as valid signature data.
 **/
ESignature *
e_signature_new_from_xml (const gchar *xml)
{
	ESignature *signature;

	signature = g_object_new (E_TYPE_SIGNATURE, NULL);

	if (!e_signature_set_from_xml (signature, xml)) {
		g_object_unref (signature);
		return NULL;
	}

	return signature;
}

/**
 * e_signature_uid_from_xml:
 * @xml: an XML signature description
 *
 * Return value: the permanent UID of the signature described by @xml
 * (or %NULL if @xml could not be parsed or did not contain a uid).
 * The caller must free this string.
 **/
gchar *
e_signature_uid_from_xml (const gchar *xml)
{
	xmlNodePtr node;
	xmlDocPtr doc;
	gchar *uid = NULL;

	if (!(doc = xmlParseDoc ((xmlChar *) xml)))
		return NULL;

	node = doc->children;
	if (strcmp ((gchar *)node->name, "signature") != 0) {
		xmlFreeDoc (doc);
		return NULL;
	}

	xml_set_prop (node, "uid", &uid);
	xmlFreeDoc (doc);

	return uid;
}

/**
 * e_signature_set_from_xml:
 * @signature: an #ESignature
 * @xml: an XML signature description.
 *
 * Changes @signature to match @xml.
 *
 * Returns: %TRUE if the signature was loaded or %FALSE otherwise
 **/
gboolean
e_signature_set_from_xml (ESignature *signature, const gchar *xml)
{
	gboolean changed = FALSE;
	xmlNodePtr node, cur;
	xmlDocPtr doc;
	gboolean bool;
	gchar *buf;

	if (!(doc = xmlParseDoc ((xmlChar *) xml)))
		return FALSE;

	node = doc->children;
	if (strcmp ((gchar *)node->name, "signature") != 0) {
		xmlFreeDoc (doc);
		return FALSE;
	}

	buf = NULL;
	xml_set_prop (node, "uid", &buf);

	if (buf && *buf) {
		g_free (signature->priv->uid);
		signature->priv->uid = buf;
	}

	changed |= xml_set_prop (node, "name", &signature->priv->name);
	changed |= xml_set_bool (node, "auto", &signature->priv->autogenerated);

	if (e_signature_get_autogenerated (signature)) {
		xmlFreeDoc (doc);

		return changed;
	}

	buf = NULL;
	xml_set_prop (node, "format", &buf);
	if (buf && !strcmp (buf, "text/html"))
		bool = TRUE;
	else
		bool = FALSE;
	g_free (buf);

	if (e_signature_get_is_html (signature) != bool) {
		e_signature_set_is_html (signature, bool);
		changed = TRUE;
	}

	cur = node->children;
	while (cur) {
		if (!strcmp ((gchar *)cur->name, "filename")) {
			changed |= xml_set_content (
				cur, &signature->priv->filename);
			changed |= xml_set_bool (
				cur, "script", &signature->priv->is_script);
			break;
		} else if (!strcmp ((gchar *)cur->name, "script")) {
			/* this is for handling 1.4 signature script definitions */
			changed |= xml_set_content (
				cur, &signature->priv->filename);
			if (!e_signature_get_is_script (signature)) {
				e_signature_set_is_script (signature, TRUE);
				changed = TRUE;
			}
			break;
		}
		cur = cur->next;
	}

	/* If the signature is not a script, replace the directory
	 * part with the current signatures directory.  This makes
	 * moving the signatures directory transparent. */
	if (!e_signature_get_is_script (signature)) {
		const gchar *user_data_dir;
		gchar *basename;
		gchar *filename;

		user_data_dir = e_get_user_data_dir ();

		filename = signature->priv->filename;
		basename = g_path_get_basename (filename);
		signature->priv->filename = g_build_filename (
			user_data_dir, "signatures", basename, NULL);
		g_free (basename);
		g_free (filename);
	}

	xmlFreeDoc (doc);

	return changed;
}

/**
 * e_signature_to_xml:
 * @signature: an #ESignature
 *
 * Return value: an XML representation of @signature, which the caller
 * must free.
 **/
gchar *
e_signature_to_xml (ESignature *signature)
{
	xmlChar *xmlbuf;
	gchar *tmp;
	xmlNodePtr root, node;
	xmlDocPtr doc;
	const gchar *string;
	gint n;

	doc = xmlNewDoc ((xmlChar *) "1.0");

	root = xmlNewDocNode (doc, NULL, (xmlChar *) "signature", NULL);
	xmlDocSetRootElement (doc, root);

	string = e_signature_get_name (signature);
	xmlSetProp (root, (xmlChar *) "name", (xmlChar *) string);

	string = e_signature_get_uid (signature);
	xmlSetProp (root, (xmlChar *) "uid", (xmlChar *) string);

	if (e_signature_get_autogenerated (signature))
		string = "true";
	else
		string = "false";
	xmlSetProp (root, (xmlChar *) "auto", (xmlChar *) string);

	if (!e_signature_get_autogenerated (signature)) {
		if (e_signature_get_is_html (signature))
			string = "text/html";
		else
			string = "text/plain";
		xmlSetProp (root, (xmlChar *) "format", (xmlChar *) string);

		string = e_signature_get_filename (signature);
		if (string != NULL) {

			/* For scripts we save the full filename,
			 * for normal signatures just the basename. */
			if (e_signature_get_is_script (signature)) {
				node = xmlNewTextChild (
					root, NULL, (xmlChar *) "filename",
					(xmlChar *) string);
				xmlSetProp (
					node, (xmlChar *) "script",
					(xmlChar *) "true");
			} else {
				gchar *basename;

				basename = g_path_get_basename (string);
				node = xmlNewTextChild (
					root, NULL, (xmlChar *) "filename",
					(xmlChar *) basename);
				g_free (basename);
			}
		}
	} else {
		/* this is to make Evolution-1.4 and older 1.5 versions happy */
		xmlSetProp (root, (xmlChar *) "format", (xmlChar *) "text/html");
	}

	xmlDocDumpMemory (doc, &xmlbuf, &n);
	xmlFreeDoc (doc);

	/* remap to glib memory */
	tmp = g_malloc (n + 1);
	memcpy (tmp, xmlbuf, n);
	tmp[n] = '\0';
	xmlFree (xmlbuf);

	return tmp;
}

gboolean
e_signature_is_equal (ESignature *signature1,
                      ESignature *signature2)
{
	const gchar *uid1;
	const gchar *uid2;

	g_return_val_if_fail (E_IS_SIGNATURE (signature1), FALSE);
	g_return_val_if_fail (E_IS_SIGNATURE (signature2), FALSE);

	/* XXX Simply compares the UIDs.  Not fool-proof. */
	uid1 = e_signature_get_uid (signature1);
	uid2 = e_signature_get_uid (signature2);

	return (g_strcmp0 (uid1, uid2) == 0);
}

gboolean
e_signature_get_autogenerated (ESignature *signature)
{
	g_return_val_if_fail (E_IS_SIGNATURE (signature), FALSE);

	return signature->priv->autogenerated;
}

void
e_signature_set_autogenerated (ESignature *signature,
                               gboolean autogenerated)
{
	g_return_if_fail (E_IS_SIGNATURE (signature));

	if (signature->priv->autogenerated == autogenerated)
		return;

	signature->priv->autogenerated = autogenerated;

	/* Autogenerated flags overrides several properties. */
	g_object_freeze_notify (G_OBJECT (signature));
	g_object_notify (G_OBJECT (signature), "autogenerated");
	g_object_notify (G_OBJECT (signature), "filename");
	g_object_notify (G_OBJECT (signature), "is-html");
	g_object_notify (G_OBJECT (signature), "is-script");
	g_object_notify (G_OBJECT (signature), "name");
	g_object_thaw_notify (G_OBJECT (signature));
}

const gchar *
e_signature_get_filename (ESignature *signature)
{
	g_return_val_if_fail (E_IS_SIGNATURE (signature), NULL);

	/* Autogenerated flags overrides the filename property. */
	if (e_signature_get_autogenerated (signature))
		return NULL;

	return signature->priv->filename;
}

void
e_signature_set_filename (ESignature *signature,
                          const gchar *filename)
{
	g_return_if_fail (E_IS_SIGNATURE (signature));

	g_free (signature->priv->filename);
	signature->priv->filename = g_strdup (filename);

	g_object_notify (G_OBJECT (signature), "filename");
}

gboolean
e_signature_get_is_html (ESignature *signature)
{
	g_return_val_if_fail (E_IS_SIGNATURE (signature), FALSE);

	/* Autogenerated flag overrides the is-html property. */
	if (e_signature_get_autogenerated (signature))
		return FALSE;

	return signature->priv->is_html;
}

void
e_signature_set_is_html (ESignature *signature,
                         gboolean is_html)
{
	g_return_if_fail (E_IS_SIGNATURE (signature));

	if (signature->priv->is_html == is_html)
		return;

	signature->priv->is_html = is_html;

	g_object_notify (G_OBJECT (signature), "is-html");
}

gboolean
e_signature_get_is_script (ESignature *signature)
{
	g_return_val_if_fail (E_IS_SIGNATURE (signature), FALSE);

	/* Autogenerated flags overrides the is-script property. */
	if (e_signature_get_autogenerated (signature))
		return FALSE;

	return signature->priv->is_script;
}

void
e_signature_set_is_script (ESignature *signature,
                           gboolean is_script)
{
	g_return_if_fail (E_IS_SIGNATURE (signature));

	if (signature->priv->is_script == is_script)
		return;

	signature->priv->is_script = is_script;

	g_object_notify (G_OBJECT (signature), "is-script");
}

const gchar *
e_signature_get_name (ESignature *signature)
{
	g_return_val_if_fail (E_IS_SIGNATURE (signature), NULL);

	/* Autogenerated flag overrides the name property. */
	if (e_signature_get_autogenerated (signature))
		return _("Autogenerated");

	return signature->priv->name;
}

void
e_signature_set_name (ESignature *signature,
                      const gchar *name)
{
	g_return_if_fail (E_IS_SIGNATURE (signature));

	g_free (signature->priv->name);
	signature->priv->name = g_strdup (name);

	g_object_notify (G_OBJECT (signature), "name");
}

const gchar *
e_signature_get_uid (ESignature *signature)
{
	g_return_val_if_fail (E_IS_SIGNATURE (signature), NULL);

	return signature->priv->uid;
}

void
e_signature_set_uid (ESignature *signature,
                     const gchar *uid)
{
	g_return_if_fail (E_IS_SIGNATURE (signature));

	g_free (signature->priv->uid);

	if (uid == NULL)
		signature->priv->uid = e_uid_new ();
	else
		signature->priv->uid = g_strdup (uid);

	g_object_notify (G_OBJECT (signature), "uid");
}