
2003-10-31 Hans Petter Jansson <hpj@ximian.com> * e-source.c (e_source_get_uri): Don't compress trailing slashes in URI elements. svn path=/trunk/; revision=23155
471 lines
10 KiB
C
471 lines
10 KiB
C
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
|
|
/* e-source.c
|
|
*
|
|
* Copyright (C) 2003 Ximian, Inc.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of version 2 of the GNU General Public
|
|
* License as published by the Free Software Foundation.
|
|
*
|
|
* 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.
|
|
*
|
|
* Author: Ettore Perazzoli <ettore@ximian.com>
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "e-source.h"
|
|
|
|
#include "e-util-marshal.h"
|
|
#include "e-uid.h"
|
|
|
|
#include <string.h>
|
|
#include <gal/util/e-util.h>
|
|
|
|
|
|
#define PARENT_TYPE G_TYPE_OBJECT
|
|
static GObjectClass *parent_class = NULL;
|
|
|
|
#define ES_CLASS(obj) E_SOURCE_CLASS (G_OBJECT_GET_CLASS (obj))
|
|
|
|
|
|
/* String used to put the color in the XML. */
|
|
#define COLOR_FORMAT_STRING "%06x"
|
|
|
|
|
|
/* Private members. */
|
|
|
|
struct _ESourcePrivate {
|
|
ESourceGroup *group;
|
|
|
|
char *uid;
|
|
char *name;
|
|
char *relative_uri;
|
|
|
|
gboolean has_color;
|
|
guint32 color;
|
|
};
|
|
|
|
|
|
/* Signals. */
|
|
|
|
enum {
|
|
CHANGED,
|
|
LAST_SIGNAL
|
|
};
|
|
static unsigned int signals[LAST_SIGNAL] = { 0 };
|
|
|
|
|
|
/* Callbacks. */
|
|
|
|
static void
|
|
group_weak_notify (ESource *source,
|
|
GObject **where_the_object_was)
|
|
{
|
|
source->priv->group = NULL;
|
|
|
|
g_signal_emit (source, signals[CHANGED], 0);
|
|
}
|
|
|
|
|
|
/* GObject methods. */
|
|
|
|
static void
|
|
impl_finalize (GObject *object)
|
|
{
|
|
ESourcePrivate *priv = E_SOURCE (object)->priv;
|
|
|
|
g_free (priv->uid);
|
|
g_free (priv->name);
|
|
g_free (priv->relative_uri);
|
|
g_free (priv);
|
|
|
|
(* G_OBJECT_CLASS (parent_class)->finalize) (object);
|
|
}
|
|
|
|
static void
|
|
impl_dispose (GObject *object)
|
|
{
|
|
ESourcePrivate *priv = E_SOURCE (object)->priv;
|
|
|
|
if (priv->group != NULL) {
|
|
g_object_weak_unref (G_OBJECT (priv->group), (GWeakNotify) group_weak_notify, object);
|
|
priv->group = NULL;
|
|
}
|
|
|
|
(* G_OBJECT_CLASS (parent_class)->dispose) (object);
|
|
}
|
|
|
|
|
|
/* Initialization. */
|
|
|
|
static void
|
|
class_init (ESourceClass *class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
|
|
object_class->dispose = impl_dispose;
|
|
object_class->finalize = impl_finalize;
|
|
|
|
parent_class = g_type_class_peek_parent (class);
|
|
|
|
signals[CHANGED] =
|
|
g_signal_new ("changed",
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (ESourceClass, changed),
|
|
NULL, NULL,
|
|
e_util_marshal_VOID__VOID,
|
|
G_TYPE_NONE, 0);
|
|
}
|
|
|
|
static void
|
|
init (ESource *source)
|
|
{
|
|
ESourcePrivate *priv;
|
|
|
|
priv = g_new0 (ESourcePrivate, 1);
|
|
source->priv = priv;
|
|
}
|
|
|
|
|
|
/* Public methods. */
|
|
|
|
ESource *
|
|
e_source_new (const char *name,
|
|
const char *relative_uri)
|
|
{
|
|
ESource *source;
|
|
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
g_return_val_if_fail (relative_uri != NULL, NULL);
|
|
|
|
source = g_object_new (e_source_get_type (), NULL);
|
|
source->priv->uid = e_uid_new ();
|
|
|
|
e_source_set_name (source, name);
|
|
e_source_set_relative_uri (source, relative_uri);
|
|
return source;
|
|
}
|
|
|
|
ESource *
|
|
e_source_new_from_xml_node (xmlNodePtr node)
|
|
{
|
|
ESource *source;
|
|
xmlChar *uid;
|
|
|
|
uid = xmlGetProp (node, "uid");
|
|
if (uid == NULL)
|
|
return NULL;
|
|
|
|
source = g_object_new (e_source_get_type (), NULL);
|
|
|
|
source->priv->uid = g_strdup (uid);
|
|
xmlFree (uid);
|
|
|
|
if (e_source_update_from_xml_node (source, node, NULL))
|
|
return source;
|
|
|
|
g_object_unref (source);
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* e_source_update_from_xml_node:
|
|
* @source: An ESource.
|
|
* @node: A pointer to the node to parse.
|
|
*
|
|
* Update the ESource properties from @node.
|
|
*
|
|
* Return value: %TRUE if the data in @node was recognized and parsed into
|
|
* acceptable values for @source, %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
e_source_update_from_xml_node (ESource *source,
|
|
xmlNodePtr node,
|
|
gboolean *changed_return)
|
|
{
|
|
xmlChar *name;
|
|
xmlChar *relative_uri;
|
|
xmlChar *color_string;
|
|
gboolean retval;
|
|
gboolean changed = FALSE;
|
|
|
|
name = xmlGetProp (node, "name");
|
|
relative_uri = xmlGetProp (node, "relative_uri");
|
|
color_string = xmlGetProp (node, "color");
|
|
|
|
if (name == NULL || relative_uri == NULL) {
|
|
retval = FALSE;
|
|
goto done;
|
|
}
|
|
|
|
if (source->priv->name == NULL
|
|
|| strcmp (name, source->priv->name) != 0
|
|
|| source->priv->relative_uri == NULL
|
|
|| strcmp (relative_uri, source->priv->relative_uri) != 0) {
|
|
g_free (source->priv->name);
|
|
source->priv->name = g_strdup (name);
|
|
|
|
g_free (source->priv->relative_uri);
|
|
source->priv->relative_uri = g_strdup (relative_uri);
|
|
|
|
changed = TRUE;
|
|
}
|
|
|
|
if (color_string == NULL) {
|
|
if (source->priv->has_color) {
|
|
source->priv->has_color = FALSE;
|
|
changed = TRUE;
|
|
}
|
|
} else {
|
|
guint32 color = 0;
|
|
|
|
sscanf (color_string, COLOR_FORMAT_STRING, &color);
|
|
if (! source->priv->has_color || source->priv->color != color) {
|
|
source->priv->has_color = TRUE;
|
|
source->priv->color = color;
|
|
changed = TRUE;
|
|
}
|
|
}
|
|
|
|
retval = TRUE;
|
|
|
|
done:
|
|
if (changed)
|
|
g_signal_emit (source, signals[CHANGED], 0);
|
|
|
|
if (changed_return != NULL)
|
|
*changed_return = changed;
|
|
|
|
if (name != NULL)
|
|
xmlFree (name);
|
|
if (relative_uri != NULL)
|
|
xmlFree (relative_uri);
|
|
if (color_string != NULL)
|
|
xmlFree (color_string);
|
|
|
|
return retval;
|
|
}
|
|
|
|
/**
|
|
* e_source_name_from_xml_node:
|
|
* @node: A pointer to an XML node.
|
|
*
|
|
* Assuming that @node is a valid ESource specification, retrieve the name of
|
|
* the source from it.
|
|
*
|
|
* Return value: Name of the source in the specified @node. The caller must
|
|
* free the string.
|
|
**/
|
|
char *
|
|
e_source_uid_from_xml_node (xmlNodePtr node)
|
|
{
|
|
xmlChar *uid = xmlGetProp (node, "uid");
|
|
char *retval;
|
|
|
|
if (uid == NULL)
|
|
return NULL;
|
|
|
|
retval = g_strdup (uid);
|
|
xmlFree (uid);
|
|
return retval;
|
|
}
|
|
|
|
void
|
|
e_source_set_group (ESource *source,
|
|
ESourceGroup *group)
|
|
{
|
|
g_return_if_fail (E_IS_SOURCE (source));
|
|
g_return_if_fail (group == NULL || E_IS_SOURCE_GROUP (group));
|
|
|
|
if (source->priv->group == group)
|
|
return;
|
|
|
|
if (source->priv->group != NULL)
|
|
g_object_weak_unref (G_OBJECT (source->priv->group), (GWeakNotify) group_weak_notify, source);
|
|
|
|
source->priv->group = group;
|
|
if (group != NULL)
|
|
g_object_weak_ref (G_OBJECT (group), (GWeakNotify) group_weak_notify, source);
|
|
|
|
g_signal_emit (source, signals[CHANGED], 0);
|
|
}
|
|
|
|
void
|
|
e_source_set_name (ESource *source,
|
|
const char *name)
|
|
{
|
|
g_return_if_fail (E_IS_SOURCE (source));
|
|
|
|
if (source->priv->name == name)
|
|
return;
|
|
|
|
g_free (source->priv->name);
|
|
source->priv->name = g_strdup (name);
|
|
|
|
g_signal_emit (source, signals[CHANGED], 0);
|
|
}
|
|
|
|
void
|
|
e_source_set_relative_uri (ESource *source,
|
|
const char *relative_uri)
|
|
{
|
|
g_return_if_fail (E_IS_SOURCE (source));
|
|
|
|
if (source->priv->relative_uri == relative_uri)
|
|
return;
|
|
|
|
g_free (source->priv->relative_uri);
|
|
source->priv->relative_uri = g_strdup (relative_uri);
|
|
|
|
g_signal_emit (source, signals[CHANGED], 0);
|
|
}
|
|
|
|
void
|
|
e_source_set_color (ESource *source,
|
|
guint32 color)
|
|
{
|
|
g_return_if_fail (E_IS_SOURCE (source));
|
|
|
|
if (source->priv->has_color && source->priv->color == color)
|
|
return;
|
|
|
|
source->priv->has_color = TRUE;
|
|
source->priv->color = color;
|
|
|
|
g_signal_emit (source, signals[CHANGED], 0);
|
|
}
|
|
|
|
void
|
|
e_source_unset_color (ESource *source)
|
|
{
|
|
g_return_if_fail (E_IS_SOURCE (source));
|
|
|
|
if (! source->priv->has_color)
|
|
return;
|
|
|
|
source->priv->has_color = FALSE;
|
|
g_signal_emit (source, signals[CHANGED], 0);
|
|
}
|
|
|
|
|
|
ESourceGroup *
|
|
e_source_peek_group (ESource *source)
|
|
{
|
|
g_return_val_if_fail (E_IS_SOURCE (source), NULL);
|
|
|
|
return source->priv->group;
|
|
}
|
|
|
|
const char *
|
|
e_source_peek_uid (ESource *source)
|
|
{
|
|
g_return_val_if_fail (E_IS_SOURCE (source), NULL);
|
|
|
|
return source->priv->uid;
|
|
}
|
|
|
|
const char *
|
|
e_source_peek_name (ESource *source)
|
|
{
|
|
g_return_val_if_fail (E_IS_SOURCE (source), NULL);
|
|
|
|
return source->priv->name;
|
|
}
|
|
|
|
const char *
|
|
e_source_peek_relative_uri (ESource *source)
|
|
{
|
|
g_return_val_if_fail (E_IS_SOURCE (source), NULL);
|
|
|
|
return source->priv->relative_uri;
|
|
}
|
|
|
|
/**
|
|
* e_source_get_color:
|
|
* @source: An ESource
|
|
* @color_return: Pointer to a variable where the returned color will be
|
|
* stored.
|
|
*
|
|
* If @source has an associated color, return it in *@color_return.
|
|
*
|
|
* Return value: %TRUE if the @source has a defined color (and hence
|
|
* *@color_return was set), %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
e_source_get_color (ESource *source,
|
|
guint32 *color_return)
|
|
{
|
|
g_return_val_if_fail (E_IS_SOURCE (source), FALSE);
|
|
|
|
if (! source->priv->has_color)
|
|
return FALSE;
|
|
|
|
if (color_return != NULL)
|
|
*color_return = source->priv->color;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
char *
|
|
e_source_get_uri (ESource *source)
|
|
{
|
|
const gchar *base_uri_str;
|
|
gchar *uri_str;
|
|
|
|
g_return_val_if_fail (E_IS_SOURCE (source), NULL);
|
|
|
|
if (source->priv->group == NULL)
|
|
return NULL;
|
|
|
|
base_uri_str = e_source_group_peek_base_uri (source->priv->group);
|
|
|
|
/* If last character in base URI is a dir separator, just concat the strings.
|
|
* We don't want to compress e.g. the trailing :// in a protocol specification */
|
|
if (*base_uri_str && *(base_uri_str + strlen (base_uri_str) - 1) == G_DIR_SEPARATOR)
|
|
uri_str = g_strconcat (base_uri_str, source->priv->relative_uri, NULL);
|
|
else
|
|
uri_str = g_build_filename (e_source_group_peek_base_uri (source->priv->group),
|
|
source->priv->relative_uri,
|
|
NULL);
|
|
|
|
return uri_str;
|
|
}
|
|
|
|
|
|
void
|
|
e_source_dump_to_xml_node (ESource *source,
|
|
xmlNodePtr parent_node)
|
|
{
|
|
gboolean has_color;
|
|
guint32 color;
|
|
xmlNodePtr node = xmlNewChild (parent_node, NULL, "source", NULL);
|
|
|
|
g_return_if_fail (E_IS_SOURCE (source));
|
|
|
|
|
|
xmlSetProp (node, "uid", e_source_peek_uid (source));
|
|
xmlSetProp (node, "name", e_source_peek_name (source));
|
|
xmlSetProp (node, "relative_uri", e_source_peek_relative_uri (source));
|
|
|
|
has_color = e_source_get_color (source, &color);
|
|
if (has_color) {
|
|
char *color_string = g_strdup_printf (COLOR_FORMAT_STRING, color);
|
|
xmlSetProp (node, "color", color_string);
|
|
g_free (color_string);
|
|
}
|
|
}
|
|
|
|
|
|
E_MAKE_TYPE (e_source, "ESource", ESource, class_init, init, PARENT_TYPE)
|