Blame gdata/georss/gdata-georss-where.c

Packit 4b6dd7
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
Packit 4b6dd7
/*
Packit 4b6dd7
 * GData Client
Packit 4b6dd7
 * Copyright (C) Philip Withnall 2009 <philip@tecnocode.co.uk>
Packit 4b6dd7
 * Copyright (C) Richard Schwarting 2009 <aquarichy@gmail.com>
Packit 4b6dd7
 *
Packit 4b6dd7
 * GData Client is free software; you can redistribute it and/or
Packit 4b6dd7
 * modify it under the terms of the GNU Lesser General Public
Packit 4b6dd7
 * License as published by the Free Software Foundation; either
Packit 4b6dd7
 * version 2.1 of the License, or (at your option) any later version.
Packit 4b6dd7
 *
Packit 4b6dd7
 * GData Client is distributed in the hope that it will be useful,
Packit 4b6dd7
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 4b6dd7
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 4b6dd7
 * Lesser General Public License for more details.
Packit 4b6dd7
 *
Packit 4b6dd7
 * You should have received a copy of the GNU Lesser General Public
Packit 4b6dd7
 * License along with GData Client.  If not, see <http://www.gnu.org/licenses/>.
Packit 4b6dd7
 */
Packit 4b6dd7
Packit 4b6dd7
/*
Packit 4b6dd7
 * SECTION:gdata-georss-where
Packit 4b6dd7
 * @short_description: GeoRSS where element
Packit 4b6dd7
 * @stability: Stable
Packit 4b6dd7
 * @include: gdata/georss/gdata-georss-where.h
Packit 4b6dd7
 *
Packit 4b6dd7
 * #GDataGeoRSSWhere represents a "where" element from the
Packit 4b6dd7
 * <ulink type="http" url="http://www.georss.org/georss">GeoRSS specification</ulink>.
Packit 4b6dd7
 * with PicasaWeb usage defined at
Packit 4b6dd7
 * <ulink type="http" url="http://code.google.com/apis/picasaweb/docs/2.0/reference.html#georss_reference">PicasaWeb API reference</ulink>.
Packit 4b6dd7
 *
Packit 4b6dd7
 * It is private API, since implementing classes are likely to proxy the properties and functions
Packit 4b6dd7
 * of #GDataGeoRSSWhere as appropriate; most entry types which implement #GDataGeoRSSWhere have no use
Packit 4b6dd7
 * for most of its properties, and it would be unnecessary and confusing to expose #GDataGeoRSSWhere itself.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Since: 0.5.0
Packit 4b6dd7
 */
Packit 4b6dd7
Packit 4b6dd7
#include <glib.h>
Packit 4b6dd7
#include <libxml/parser.h>
Packit 4b6dd7
#include <string.h>
Packit 4b6dd7
Packit 4b6dd7
#include "gdata-georss-where.h"
Packit 4b6dd7
#include "gdata-parsable.h"
Packit 4b6dd7
#include "gdata-parser.h"
Packit 4b6dd7
#include "gdata-private.h"
Packit 4b6dd7
Packit 4b6dd7
static gboolean parse_xml (GDataParsable *parsable, xmlDoc *doc, xmlNode *node, gpointer user_data, GError **error);
Packit 4b6dd7
static void get_namespaces (GDataParsable *parsable, GHashTable *namespaces);
Packit 4b6dd7
static void get_xml (GDataParsable *parsable, GString *xml_string);
Packit 4b6dd7
Packit 4b6dd7
struct _GDataGeoRSSWherePrivate {
Packit 4b6dd7
	gdouble latitude;
Packit 4b6dd7
	gdouble longitude;
Packit 4b6dd7
};
Packit 4b6dd7
Packit 4b6dd7
G_DEFINE_TYPE (GDataGeoRSSWhere, gdata_georss_where, GDATA_TYPE_PARSABLE)
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
gdata_georss_where_class_init (GDataGeoRSSWhereClass *klass)
Packit 4b6dd7
{
Packit 4b6dd7
	GDataParsableClass *parsable_class = GDATA_PARSABLE_CLASS (klass);
Packit 4b6dd7
Packit 4b6dd7
	g_type_class_add_private (klass, sizeof (GDataGeoRSSWherePrivate));
Packit 4b6dd7
Packit 4b6dd7
	parsable_class->get_xml = get_xml;
Packit 4b6dd7
	parsable_class->parse_xml = parse_xml;
Packit 4b6dd7
	parsable_class->get_namespaces = get_namespaces;
Packit 4b6dd7
	parsable_class->element_name = "where";
Packit 4b6dd7
	parsable_class->element_namespace = "georss";
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
gdata_georss_where_init (GDataGeoRSSWhere *self)
Packit 4b6dd7
{
Packit 4b6dd7
	self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, GDATA_TYPE_GEORSS_WHERE, GDataGeoRSSWherePrivate);
Packit 4b6dd7
Packit 4b6dd7
	self->priv->latitude = G_MAXDOUBLE;
Packit 4b6dd7
	self->priv->longitude = G_MAXDOUBLE;
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
static gboolean
Packit 4b6dd7
parse_xml (GDataParsable *parsable, xmlDoc *doc, xmlNode *node, gpointer user_data, GError **error)
Packit 4b6dd7
{
Packit 4b6dd7
	GDataGeoRSSWhere *self = GDATA_GEORSS_WHERE (parsable);
Packit 4b6dd7
Packit 4b6dd7
	if (gdata_parser_is_namespace (node, "http://www.opengis.net/gml") == TRUE &&
Packit 4b6dd7
	    xmlStrcmp (node->name, (xmlChar*) "Point") == 0) {
Packit 4b6dd7
		/* gml:Point */
Packit 4b6dd7
		gboolean found_pos = FALSE;
Packit 4b6dd7
		xmlNode *child;
Packit 4b6dd7
Packit 4b6dd7
		for (child = node->children; child != NULL; child = child->next) {
Packit 4b6dd7
			if (xmlStrcmp (child->name, (xmlChar*) "pos") == 0) {
Packit 4b6dd7
				xmlChar *pos = xmlNodeListGetString (doc, child->children, TRUE);
Packit 4b6dd7
				gchar *endptr;
Packit 4b6dd7
Packit 4b6dd7
				self->priv->latitude = g_ascii_strtod ((gchar*) pos, &endptr);
Packit 4b6dd7
				self->priv->longitude = g_ascii_strtod (endptr, NULL);
Packit 4b6dd7
Packit 4b6dd7
				xmlFree (pos);
Packit 4b6dd7
				found_pos = TRUE;
Packit 4b6dd7
			} else {
Packit 4b6dd7
				/* TODO: this logic copied from gdata-parsable.c.  Re-evaluate this at some point in the future.
Packit 4b6dd7
				   If GeoRSS and GML support were to be used more widely, it might due to implement GML objects. */
Packit 4b6dd7
				xmlBuffer *buffer;
Packit 4b6dd7
Packit 4b6dd7
				/* Unhandled XML */
Packit 4b6dd7
				buffer = xmlBufferCreate ();
Packit 4b6dd7
				xmlNodeDump (buffer, doc, child, 0, 0);
Packit 4b6dd7
				g_debug ("Unhandled XML in <gml:Point>: %s", (gchar*) xmlBufferContent (buffer));
Packit 4b6dd7
				xmlBufferFree (buffer);
Packit 4b6dd7
			}
Packit 4b6dd7
		}
Packit 4b6dd7
Packit 4b6dd7
		if (found_pos == FALSE)
Packit 4b6dd7
			return gdata_parser_error_required_element_missing ("pos", "gml:Point", error);
Packit 4b6dd7
		return TRUE;
Packit 4b6dd7
	}
Packit 4b6dd7
Packit 4b6dd7
	return GDATA_PARSABLE_CLASS (gdata_georss_where_parent_class)->parse_xml (parsable, doc, node, user_data, error);
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
get_xml (GDataParsable *parsable, GString *xml_string)
Packit 4b6dd7
{
Packit 4b6dd7
	GDataGeoRSSWherePrivate *priv = GDATA_GEORSS_WHERE (parsable)->priv;
Packit 4b6dd7
	gchar latitude_str[G_ASCII_DTOSTR_BUF_SIZE];
Packit 4b6dd7
	gchar longitude_str[G_ASCII_DTOSTR_BUF_SIZE];
Packit 4b6dd7
Packit 4b6dd7
	if (priv->latitude != G_MAXDOUBLE && priv->longitude != G_MAXDOUBLE) {
Packit 4b6dd7
		g_string_append_printf (xml_string, "<gml:Point><gml:pos>%s %s</gml:pos></gml:Point>",
Packit 4b6dd7
		                        g_ascii_dtostr (latitude_str, sizeof (latitude_str), priv->latitude),
Packit 4b6dd7
		                        g_ascii_dtostr (longitude_str, sizeof (longitude_str), priv->longitude));
Packit 4b6dd7
	}
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
static void
Packit 4b6dd7
get_namespaces (GDataParsable *parsable, GHashTable *namespaces)
Packit 4b6dd7
{
Packit 4b6dd7
	g_hash_table_insert (namespaces, (gchar*) "georss", (gchar*) "http://www.georss.org/georss");
Packit 4b6dd7
	g_hash_table_insert (namespaces, (gchar*) "gml", (gchar*) "http://www.opengis.net/gml");
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
/**
Packit 4b6dd7
 * gdata_georss_where_get_latitude:
Packit 4b6dd7
 * @self: a #GDataGeoRSSWhere
Packit 4b6dd7
 *
Packit 4b6dd7
 * Gets the #GDataGeoRSSWhere:latitude property.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Return value: the latitude of this position, or %G_MAXDOUBLE if unknown
Packit 4b6dd7
 *
Packit 4b6dd7
 * Since: 0.5.0
Packit 4b6dd7
 */
Packit 4b6dd7
gdouble
Packit 4b6dd7
gdata_georss_where_get_latitude (GDataGeoRSSWhere *self)
Packit 4b6dd7
{
Packit 4b6dd7
	g_return_val_if_fail (GDATA_IS_GEORSS_WHERE (self), G_MAXDOUBLE);
Packit 4b6dd7
	return self->priv->latitude;
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
/**
Packit 4b6dd7
 * gdata_georss_where_get_longitude:
Packit 4b6dd7
 * @self: a #GDataGeoRSSWhere
Packit 4b6dd7
 *
Packit 4b6dd7
 * Gets the #GDataGeoRSSWhere:longitude property.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Return value: the longitude of this position, or %G_MAXDOUBLE if unknown
Packit 4b6dd7
 *
Packit 4b6dd7
 * Since: 0.5.0
Packit 4b6dd7
 */
Packit 4b6dd7
gdouble
Packit 4b6dd7
gdata_georss_where_get_longitude (GDataGeoRSSWhere *self)
Packit 4b6dd7
{
Packit 4b6dd7
	g_return_val_if_fail (GDATA_IS_GEORSS_WHERE (self), G_MAXDOUBLE);
Packit 4b6dd7
	return self->priv->longitude;
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
/**
Packit 4b6dd7
 * gdata_georss_where_set_latitude:
Packit 4b6dd7
 * @self: a #GDataGeoRSSWhere
Packit 4b6dd7
 * @latitude: the new latitude coordinate, or %G_MAXDOUBLE
Packit 4b6dd7
 *
Packit 4b6dd7
 * Sets the #GDataGeoRSSWhere:latitude property to @latitude.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Valid values range from -90.0 to 90.0 inclusive.
Packit 4b6dd7
 * Set @latitude to %G_MAXDOUBLE to unset it.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Since: 0.5.0
Packit 4b6dd7
 */
Packit 4b6dd7
void
Packit 4b6dd7
gdata_georss_where_set_latitude (GDataGeoRSSWhere *self, gdouble latitude)
Packit 4b6dd7
{
Packit 4b6dd7
	g_return_if_fail (GDATA_IS_GEORSS_WHERE (self));
Packit 4b6dd7
Packit 4b6dd7
	if (latitude < -90.0 || latitude > 90.0)
Packit 4b6dd7
		self->priv->latitude = G_MAXDOUBLE;
Packit 4b6dd7
	else
Packit 4b6dd7
		self->priv->latitude = latitude;
Packit 4b6dd7
}
Packit 4b6dd7
Packit 4b6dd7
/**
Packit 4b6dd7
 * gdata_georss_where_set_longitude:
Packit 4b6dd7
 * @self: a #GDataGeoRSSWhere
Packit 4b6dd7
 * @longitude: the new longitude coordinate, or %G_MAXDOUBLE
Packit 4b6dd7
 *
Packit 4b6dd7
 * Sets the #GDataGeoRSSWhere:longitude property to @longitude.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Valid values range from -180.0 to 180.0 inclusive.
Packit 4b6dd7
 * Set @longitude to %G_MAXDOUBLE to unset it.
Packit 4b6dd7
 *
Packit 4b6dd7
 * Since: 0.5.0
Packit 4b6dd7
 */
Packit 4b6dd7
void
Packit 4b6dd7
gdata_georss_where_set_longitude (GDataGeoRSSWhere *self, gdouble longitude)
Packit 4b6dd7
{
Packit 4b6dd7
	g_return_if_fail (GDATA_IS_GEORSS_WHERE (self));
Packit 4b6dd7
Packit 4b6dd7
	if (longitude < -180.0 || longitude > 180.0)
Packit 4b6dd7
		self->priv->longitude = G_MAXDOUBLE;
Packit 4b6dd7
	else
Packit 4b6dd7
		self->priv->longitude = longitude;
Packit 4b6dd7
}