2 * position-publisher.c - Source for PositionPublisher
3 * Copyright (C) 2010 Guillaume Desmottes
4 * @author Guillaume Desmottes <gdesmott@gnome.org>
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
25 #include <telepathy-glib/dbus.h>
26 #include <telepathy-glib/interfaces.h>
28 #include <location/location-gps-device.h>
30 #include "connection-watcher.h"
31 #include "position-publisher.h"
33 G_DEFINE_TYPE(PositionPublisher, position_publisher, G_TYPE_OBJECT)
35 /* Minimum time before 2 publishing (in seconds) */
36 #define PUBLISH_THROTTLE 10
38 /* private structure */
39 typedef struct _PositionPublisherPrivate PositionPublisherPrivate;
41 struct _PositionPublisherPrivate
43 ConnectionWatcher *watcher;
44 LocationGPSDevice *gps_device;
47 /* If not 0, we are waiting before publishing again */
48 guint throttle_timeout;
49 /* TRUE if location has been modified while we were waiting */
52 gboolean dispose_has_run;
55 #define POSITION_PUBLISHER_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), POSITION_PUBLISHER_TYPE, PositionPublisherPrivate))
57 static void conn_invalidated_cb (TpProxy *conn,
61 PositionPublisher *self);
63 static void publish_to_all (PositionPublisher *self);
66 remove_connection (PositionPublisher *self,
69 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (self);
71 g_signal_handlers_disconnect_by_func (conn, G_CALLBACK (conn_invalidated_cb),
73 priv->connections = g_slist_remove (priv->connections, conn);
74 g_object_unref (conn);
78 set_location_cb (TpConnection *conn,
83 PositionPublisher *self = POSITION_PUBLISHER (weak_object);
87 g_print ("SetLocation failed (%s): %s\n", tp_proxy_get_object_path (conn),
90 if (error->code == TP_ERROR_NOT_IMPLEMENTED)
92 g_print ("remove connection\n");
93 remove_connection (self, conn);
99 g_print ("SetLocation succeed\n");
103 publish_throttle_timeout_cb (gpointer data)
105 PositionPublisher *self = data;
106 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (self);
108 priv->throttle_timeout = 0;
112 publish_to_all (self);
113 priv->modified = FALSE;
120 publish_to_conn (PositionPublisher *self,
123 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (self);
125 if (priv->location == NULL)
128 if (priv->throttle_timeout != 0)
132 tp_cli_connection_interface_location_call_set_location (conn, -1,
133 priv->location, set_location_cb, NULL, NULL, G_OBJECT (self));
135 /* We won't publish during the next PUBLISH_THROTTLE seconds */
136 priv->throttle_timeout = g_timeout_add_seconds (PUBLISH_THROTTLE,
137 publish_throttle_timeout_cb, self);
141 publish_to_all (PositionPublisher *self)
143 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (self);
146 for (l = priv->connections; l != NULL; l = g_slist_next (l))
148 TpConnection *conn = l->data;
150 publish_to_conn (self, conn);
155 update_position (PositionPublisher *self,
161 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (self);
163 g_print ("update position: lat: %f lon: %f alt: %f accuracy: %f\n",
164 lat, lon, alt, accuracy);
166 if (priv->location != NULL)
167 g_hash_table_unref (priv->location);
169 priv->location = tp_asv_new (
170 "timestamp", G_TYPE_INT64, (gint64) time (NULL),
171 "lat", G_TYPE_DOUBLE, lat,
172 "lon", G_TYPE_DOUBLE, lon,
173 "alt", G_TYPE_DOUBLE, alt,
174 "accuracy", G_TYPE_DOUBLE, accuracy,
177 priv->modified = TRUE;
179 publish_to_all (self);
183 location_changed_cb (LocationGPSDevice *device,
184 PositionPublisher *self)
189 if (device->fix == NULL)
192 if (!(device->fix->fields & LOCATION_GPS_DEVICE_LATLONG_SET))
195 update_position (self, device->fix->latitude, device->fix->longitude,
196 device->fix->altitude, device->fix->eph / 100.0);
200 conn_invalidated_cb (TpProxy *conn,
204 PositionPublisher *self)
206 g_print ("connection %s invalidated; removing\n", tp_proxy_get_object_path (
209 remove_connection (self, TP_CONNECTION (conn));
213 connection_added_cb (ConnectionWatcher *watcher,
215 PositionPublisher *self)
217 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (self);
219 if (g_slist_find (priv->connections, conn) != NULL)
222 if (!tp_proxy_has_interface_by_id (conn,
223 TP_IFACE_QUARK_CONNECTION_INTERFACE_LOCATION))
226 priv->connections = g_slist_prepend (priv->connections, g_object_ref (conn));
228 g_signal_connect (conn, "invalidated",
229 G_CALLBACK (conn_invalidated_cb), self);
231 publish_to_conn (self, conn);
235 position_publisher_init (PositionPublisher *obj)
237 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (obj);
239 priv->watcher = connection_watcher_new ();
241 g_signal_connect (priv->watcher, "connection-added",
242 G_CALLBACK (connection_added_cb), obj);
244 priv->gps_device = g_object_new (LOCATION_TYPE_GPS_DEVICE, NULL);
246 g_signal_connect (priv->gps_device, "changed",
247 G_CALLBACK (location_changed_cb), obj);
249 priv->connections = NULL;
253 position_publisher_constructed (GObject *object)
255 PositionPublisher *self = POSITION_PUBLISHER (object);
256 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (self);
258 connection_watcher_start (priv->watcher);
260 if (G_OBJECT_CLASS (position_publisher_parent_class)->constructed)
261 G_OBJECT_CLASS (position_publisher_parent_class)->constructed (object);
264 static void position_publisher_dispose (GObject *object);
265 static void position_publisher_finalize (GObject *object);
268 position_publisher_class_init (PositionPublisherClass *position_publisher_class)
270 GObjectClass *object_class = G_OBJECT_CLASS (position_publisher_class);
272 g_type_class_add_private (position_publisher_class, sizeof (PositionPublisherPrivate));
274 object_class->dispose = position_publisher_dispose;
275 object_class->finalize = position_publisher_finalize;
277 object_class->constructed = position_publisher_constructed;
281 position_publisher_dispose (GObject *object)
283 PositionPublisher *self = POSITION_PUBLISHER (object);
284 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (self);
287 if (priv->dispose_has_run)
290 priv->dispose_has_run = TRUE;
292 g_object_unref (priv->watcher);
293 g_object_unref (priv->gps_device);
295 for (l = priv->connections; l != NULL; l = g_slist_next (l))
297 g_object_unref (l->data);
300 g_hash_table_unref (priv->location);
302 if (priv->throttle_timeout != 0)
303 g_source_remove (priv->throttle_timeout);
305 if (G_OBJECT_CLASS (position_publisher_parent_class)->dispose)
306 G_OBJECT_CLASS (position_publisher_parent_class)->dispose (object);
310 position_publisher_finalize (GObject *object)
312 PositionPublisher *self = POSITION_PUBLISHER (object);
313 PositionPublisherPrivate *priv = POSITION_PUBLISHER_GET_PRIVATE (self);
315 g_slist_free (priv->connections);
317 G_OBJECT_CLASS (position_publisher_parent_class)->finalize (object);
321 position_publisher_new (void)
323 return g_object_new (POSITION_PUBLISHER_TYPE,