1 /* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
4 * Copyright (C) 2003 Ximian, Inc.
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of version 2 of the GNU General Public
8 * License as published by the Free Software Foundation.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * General Public License for more details.
15 * You should have received a copy of the GNU General Public
16 * License along with this program; if not, write to the
17 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 * Boston, MA 02111-1307, USA.
20 * Author: Ettore Perazzoli <ettore@ximian.com>
28 #include "e-data-server-marshal.h"
30 #include "e-source-group.h"
32 static GObjectClass *parent_class = NULL;
34 /* Private members. */
36 struct _ESourceGroupPrivate {
43 gboolean ignore_source_changed;
55 static unsigned int signals[LAST_SIGNAL] = { 0 };
61 source_changed_callback (ESource *source,
64 if (! group->priv->ignore_source_changed)
65 g_signal_emit (group, signals[CHANGED], 0);
69 /* GObject methods. */
72 impl_dispose (GObject *object)
74 ESourceGroupPrivate *priv = E_SOURCE_GROUP (object)->priv;
76 if (priv->sources != NULL) {
79 for (p = priv->sources; p != NULL; p = p->next) {
80 ESource *source = E_SOURCE (p->data);
82 g_signal_handlers_disconnect_by_func (source,
83 G_CALLBACK (source_changed_callback),
85 g_object_unref (source);
88 g_slist_free (priv->sources);
92 (* G_OBJECT_CLASS (parent_class)->dispose) (object);
96 impl_finalize (GObject *object)
98 ESourceGroupPrivate *priv = E_SOURCE_GROUP (object)->priv;
102 g_free (priv->base_uri);
105 (* G_OBJECT_CLASS (parent_class)->finalize) (object);
109 /* Initialization. */
112 class_init (ESourceGroupClass *class)
114 GObjectClass *object_class = G_OBJECT_CLASS (class);
116 object_class->dispose = impl_dispose;
117 object_class->finalize = impl_finalize;
119 parent_class = g_type_class_peek_parent (class);
122 g_signal_new ("changed",
123 G_OBJECT_CLASS_TYPE (object_class),
125 G_STRUCT_OFFSET (ESourceGroupClass, changed),
127 e_data_server_marshal_VOID__VOID,
130 signals[SOURCE_ADDED] =
131 g_signal_new ("source_added",
132 G_OBJECT_CLASS_TYPE (object_class),
134 G_STRUCT_OFFSET (ESourceGroupClass, source_added),
136 e_data_server_marshal_VOID__OBJECT,
139 signals[SOURCE_REMOVED] =
140 g_signal_new ("source_removed",
141 G_OBJECT_CLASS_TYPE (object_class),
143 G_STRUCT_OFFSET (ESourceGroupClass, source_removed),
145 e_data_server_marshal_VOID__OBJECT,
151 init (ESourceGroup *source_group)
153 ESourceGroupPrivate *priv;
155 priv = g_new0 (ESourceGroupPrivate, 1);
156 source_group->priv = priv;
160 e_source_group_get_type (void)
162 static GType e_source_group_type = 0;
164 if (!e_source_group_type) {
165 static GTypeInfo info = {
166 sizeof (ESourceGroupClass),
167 (GBaseInitFunc) NULL,
168 (GBaseFinalizeFunc) NULL,
169 (GClassInitFunc) class_init,
171 sizeof (ESourceGroup),
173 (GInstanceInitFunc) init
175 e_source_group_type = g_type_register_static (G_TYPE_OBJECT, "ESourceGroup", &info, 0);
178 return e_source_group_type;
181 /* Public methods. */
184 e_source_group_new (const char *name,
185 const char *base_uri)
189 g_return_val_if_fail (name != NULL, NULL);
190 g_return_val_if_fail (base_uri != NULL, NULL);
192 new = g_object_new (e_source_group_get_type (), NULL);
193 new->priv->uid = e_uid_new ();
195 e_source_group_set_name (new, name);
196 e_source_group_set_base_uri (new, base_uri);
202 e_source_group_new_from_xml (const char *xml)
207 doc = xmlParseDoc ((char *) xml);
211 group = e_source_group_new_from_xmldoc (doc);
218 e_source_group_new_from_xmldoc (xmlDocPtr doc)
224 ESourceGroup *new = NULL;
226 g_return_val_if_fail (doc != NULL, NULL);
228 root = doc->children;
229 if (strcmp (root->name, "group") != 0)
232 uid = xmlGetProp (root, "uid");
233 name = xmlGetProp (root, "name");
234 base_uri = xmlGetProp (root, "base_uri");
236 if (uid == NULL || name == NULL || base_uri == NULL)
239 new = g_object_new (e_source_group_get_type (), NULL);
240 new->priv->uid = g_strdup (uid);
242 e_source_group_set_name (new, name);
243 e_source_group_set_base_uri (new, base_uri);
245 for (p = root->children; p != NULL; p = p->next) {
246 ESource *new_source = e_source_new_from_xml_node (p);
247 e_source_group_add_source (new, new_source, -1);
253 if (base_uri != NULL)
259 e_source_group_update_from_xml (ESourceGroup *group,
261 gboolean *changed_return)
266 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
267 g_return_val_if_fail (xml != NULL, FALSE);
269 xmldoc = xmlParseDoc ((char *) xml);
271 success = e_source_group_update_from_xmldoc (group, xmldoc, changed_return);
279 e_source_group_update_from_xmldoc (ESourceGroup *group,
281 gboolean *changed_return)
283 GHashTable *new_sources_hash;
284 GSList *new_sources_list = NULL;
285 xmlNodePtr root, nodep;
286 xmlChar *name, *base_uri;
287 gboolean changed = FALSE;
290 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
291 g_return_val_if_fail (doc != NULL, FALSE);
293 *changed_return = FALSE;
295 root = doc->children;
296 if (strcmp (root->name, "group") != 0)
299 name = xmlGetProp (root, "name");
303 base_uri = xmlGetProp (root, "base_uri");
304 if (base_uri == NULL) {
309 if (strcmp (group->priv->name, name) != 0) {
310 g_free (group->priv->name);
311 group->priv->name = g_strdup (name);
316 if (strcmp (group->priv->base_uri, base_uri) != 0) {
317 g_free (group->priv->base_uri);
318 group->priv->base_uri = g_strdup (base_uri);
323 new_sources_hash = g_hash_table_new (g_direct_hash, g_direct_equal);
325 for (nodep = root->children; nodep != NULL; nodep = nodep->next) {
326 ESource *existing_source;
327 char *uid = e_source_uid_from_xml_node (nodep);
332 existing_source = e_source_group_peek_source_by_uid (group, uid);
333 if (g_hash_table_lookup (new_sources_hash, existing_source) != NULL)
336 if (existing_source == NULL) {
337 ESource *new_source = e_source_new_from_xml_node (nodep);
339 if (new_source != NULL) {
340 e_source_set_group (new_source, group);
341 g_signal_connect (new_source, "changed", G_CALLBACK (source_changed_callback), group);
342 new_sources_list = g_slist_prepend (new_sources_list, new_source);
344 g_hash_table_insert (new_sources_hash, new_source, new_source);
346 g_signal_emit (group, signals[SOURCE_ADDED], 0, new_source);
350 gboolean source_changed;
352 group->priv->ignore_source_changed ++;
354 if (e_source_update_from_xml_node (existing_source, nodep, &source_changed)) {
355 new_sources_list = g_slist_prepend (new_sources_list, existing_source);
356 g_object_ref (existing_source);
357 g_hash_table_insert (new_sources_hash, existing_source, existing_source);
363 group->priv->ignore_source_changed --;
369 new_sources_list = g_slist_reverse (new_sources_list);
371 /* Emit "group_removed" and disconnect the "changed" signal for all the
372 groups that we haven't found in the new list. */
373 q = new_sources_list;
374 for (p = group->priv->sources; p != NULL; p = p->next) {
375 ESource *source = E_SOURCE (p->data);
377 if (g_hash_table_lookup (new_sources_hash, source) == NULL) {
380 g_signal_emit (group, signals[SOURCE_REMOVED], 0, source);
381 g_signal_handlers_disconnect_by_func (source, source_changed_callback, group);
384 if (! changed && q != NULL) {
385 if (q->data != p->data)
391 g_hash_table_destroy (new_sources_hash);
393 /* Replace the original group list with the new one. */
394 g_slist_foreach (group->priv->sources, (GFunc) g_object_unref, NULL);
395 g_slist_free (group->priv->sources);
397 group->priv->sources = new_sources_list;
399 /* FIXME if the order changes, the function doesn't notice. */
402 g_signal_emit (group, signals[CHANGED], 0);
403 *changed_return = TRUE;
406 return TRUE; /* Success. */
410 e_source_group_uid_from_xmldoc (xmlDocPtr doc)
412 xmlNodePtr root = doc->children;
416 if (strcmp (root->name, "group") != 0)
419 name = xmlGetProp (root, "uid");
423 retval = g_strdup (name);
429 e_source_group_set_name (ESourceGroup *group,
432 g_return_if_fail (E_IS_SOURCE_GROUP (group));
433 g_return_if_fail (name != NULL);
435 if (group->priv->name == name)
438 g_free (group->priv->name);
439 group->priv->name = g_strdup (name);
441 g_signal_emit (group, signals[CHANGED], 0);
444 void e_source_group_set_base_uri (ESourceGroup *group,
445 const char *base_uri)
447 g_return_if_fail (E_IS_SOURCE_GROUP (group));
448 g_return_if_fail (base_uri != NULL);
450 if (group->priv->base_uri == base_uri)
453 g_free (group->priv->base_uri);
454 group->priv->base_uri = g_strdup (base_uri);
456 g_signal_emit (group, signals[CHANGED], 0);
461 e_source_group_peek_uid (ESourceGroup *group)
463 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), NULL);
465 return group->priv->uid;
469 e_source_group_peek_name (ESourceGroup *group)
471 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), NULL);
473 return group->priv->name;
477 e_source_group_peek_base_uri (ESourceGroup *group)
479 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), NULL);
481 return group->priv->base_uri;
486 e_source_group_peek_sources (ESourceGroup *group)
488 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), NULL);
490 return group->priv->sources;
494 e_source_group_peek_source_by_uid (ESourceGroup *group,
499 for (p = group->priv->sources; p != NULL; p = p->next) {
500 if (strcmp (e_source_peek_uid (E_SOURCE (p->data)), uid) == 0)
501 return E_SOURCE (p->data);
508 e_source_group_peek_source_by_name (ESourceGroup *group,
513 for (p = group->priv->sources; p != NULL; p = p->next) {
514 if (strcmp (e_source_peek_name (E_SOURCE (p->data)), name) == 0)
515 return E_SOURCE (p->data);
522 e_source_group_add_source (ESourceGroup *group,
526 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
528 if (e_source_group_peek_source_by_uid (group, e_source_peek_uid (source)) != NULL)
531 e_source_set_group (source, group);
532 g_object_ref (source);
534 g_signal_connect (source, "changed", G_CALLBACK (source_changed_callback), group);
536 group->priv->sources = g_slist_insert (group->priv->sources, source, position);
537 g_signal_emit (group, signals[SOURCE_ADDED], 0, source);
538 g_signal_emit (group, signals[CHANGED], 0);
544 e_source_group_remove_source (ESourceGroup *group,
549 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
550 g_return_val_if_fail (E_IS_SOURCE (source), FALSE);
552 for (p = group->priv->sources; p != NULL; p = p->next) {
553 if (E_SOURCE (p->data) == source) {
554 group->priv->sources = g_slist_remove_link (group->priv->sources, p);
555 g_signal_emit (group, signals[SOURCE_REMOVED], 0, source);
556 g_signal_emit (group, signals[CHANGED], 0);
565 e_source_group_remove_source_by_uid (ESourceGroup *group,
570 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
571 g_return_val_if_fail (uid != NULL, FALSE);
573 for (p = group->priv->sources; p != NULL; p = p->next) {
574 ESource *source = E_SOURCE (p->data);
576 if (strcmp (e_source_peek_uid (source), uid) == 0) {
577 group->priv->sources = g_slist_remove_link (group->priv->sources, p);
578 g_signal_emit (group, signals[SOURCE_REMOVED], 0, source);
579 g_signal_emit (group, signals[CHANGED], 0);
589 e_source_group_to_xml (ESourceGroup *group)
594 char *returned_buffer;
598 doc = xmlNewDoc ("1.0");
600 root = xmlNewDocNode (doc, NULL, "group", NULL);
601 xmlSetProp (root, "uid", e_source_group_peek_uid (group));
602 xmlSetProp (root, "name", e_source_group_peek_name (group));
603 xmlSetProp (root, "base_uri", e_source_group_peek_base_uri (group));
605 xmlDocSetRootElement (doc, root);
607 for (p = group->priv->sources; p != NULL; p = p->next)
608 e_source_dump_to_xml_node (E_SOURCE (p->data), root);
610 xmlDocDumpMemory (doc, &xml_buffer, &xml_buffer_size);
613 returned_buffer = g_malloc (xml_buffer_size + 1);
614 memcpy (returned_buffer, xml_buffer, xml_buffer_size);
615 returned_buffer [xml_buffer_size] = '\0';
616 xmlFree (xml_buffer);
618 return returned_buffer;