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 Lesser 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 Lesser 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>
29 #include "e-source-group.h"
31 /* Private members. */
33 struct _ESourceGroupPrivate {
40 gboolean ignore_source_changed;
53 static unsigned int signals[LAST_SIGNAL] = { 0 };
59 source_changed_callback (ESource *source,
62 if (! group->priv->ignore_source_changed)
63 g_signal_emit (group, signals[CHANGED], 0);
67 /* GObject methods. */
69 G_DEFINE_TYPE (ESourceGroup, e_source_group, G_TYPE_OBJECT);
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 (e_source_group_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 (e_source_group_parent_class)->finalize) (object);
109 /* Initialization. */
112 e_source_group_class_init (ESourceGroupClass *class)
114 GObjectClass *object_class = G_OBJECT_CLASS (class);
116 object_class->dispose = impl_dispose;
117 object_class->finalize = impl_finalize;
120 g_signal_new ("changed",
121 G_OBJECT_CLASS_TYPE (object_class),
123 G_STRUCT_OFFSET (ESourceGroupClass, changed),
125 g_cclosure_marshal_VOID__VOID,
128 signals[SOURCE_ADDED] =
129 g_signal_new ("source_added",
130 G_OBJECT_CLASS_TYPE (object_class),
132 G_STRUCT_OFFSET (ESourceGroupClass, source_added),
134 g_cclosure_marshal_VOID__OBJECT,
137 signals[SOURCE_REMOVED] =
138 g_signal_new ("source_removed",
139 G_OBJECT_CLASS_TYPE (object_class),
141 G_STRUCT_OFFSET (ESourceGroupClass, source_removed),
143 g_cclosure_marshal_VOID__OBJECT,
149 e_source_group_init (ESourceGroup *source_group)
151 ESourceGroupPrivate *priv;
153 priv = g_new0 (ESourceGroupPrivate, 1);
154 source_group->priv = priv;
157 /* Public methods. */
160 e_source_group_new (const char *name,
161 const char *base_uri)
165 g_return_val_if_fail (name != NULL, NULL);
166 g_return_val_if_fail (base_uri != NULL, NULL);
168 new = g_object_new (e_source_group_get_type (), NULL);
169 new->priv->uid = e_uid_new ();
171 e_source_group_set_name (new, name);
172 e_source_group_set_base_uri (new, base_uri);
178 e_source_group_new_from_xml (const char *xml)
183 doc = xmlParseDoc ((char *) xml);
187 group = e_source_group_new_from_xmldoc (doc);
194 e_source_group_new_from_xmldoc (xmlDocPtr doc)
200 xmlChar *readonly_str;
201 ESourceGroup *new = NULL;
203 g_return_val_if_fail (doc != NULL, NULL);
205 root = doc->children;
206 if (strcmp (root->name, "group") != 0)
209 uid = xmlGetProp (root, "uid");
210 name = xmlGetProp (root, "name");
211 base_uri = xmlGetProp (root, "base_uri");
212 readonly_str = xmlGetProp (root, "readonly");
214 if (uid == NULL || name == NULL || base_uri == NULL)
217 new = g_object_new (e_source_group_get_type (), NULL);
222 new->priv->uid = g_strdup (uid);
224 e_source_group_set_name (new, name);
225 e_source_group_set_base_uri (new, base_uri);
227 for (p = root->children; p != NULL; p = p->next) {
228 ESource *new_source = e_source_new_from_xml_node (p);
230 if (new_source == NULL) {
231 g_object_unref (new);
235 e_source_group_add_source (new, new_source, -1);
238 e_source_group_set_readonly (new, readonly_str && !strcmp (readonly_str, "yes"));
246 if (base_uri != NULL)
248 if (readonly_str != NULL)
249 xmlFree (readonly_str);
254 e_source_group_update_from_xml (ESourceGroup *group,
256 gboolean *changed_return)
261 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
262 g_return_val_if_fail (xml != NULL, FALSE);
264 xmldoc = xmlParseDoc ((char *) xml);
266 success = e_source_group_update_from_xmldoc (group, xmldoc, changed_return);
274 e_source_group_update_from_xmldoc (ESourceGroup *group,
276 gboolean *changed_return)
278 GHashTable *new_sources_hash;
279 GSList *new_sources_list = NULL;
280 xmlNodePtr root, nodep;
281 xmlChar *name, *base_uri, *readonly_str;
283 gboolean changed = FALSE;
286 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
287 g_return_val_if_fail (doc != NULL, FALSE);
289 *changed_return = FALSE;
291 root = doc->children;
292 if (strcmp (root->name, "group") != 0)
295 name = xmlGetProp (root, "name");
299 base_uri = xmlGetProp (root, "base_uri");
300 if (base_uri == NULL) {
305 if (strcmp (group->priv->name, name) != 0) {
306 g_free (group->priv->name);
307 group->priv->name = g_strdup (name);
312 if (strcmp (group->priv->base_uri, base_uri) != 0) {
313 g_free (group->priv->base_uri);
314 group->priv->base_uri = g_strdup (base_uri);
319 readonly_str = xmlGetProp (root, "readonly");
320 readonly = readonly_str && !strcmp (readonly_str, "yes");
321 if (readonly != group->priv->readonly) {
322 group->priv->readonly = readonly;
325 xmlFree (readonly_str);
327 new_sources_hash = g_hash_table_new (g_direct_hash, g_direct_equal);
329 for (nodep = root->children; nodep != NULL; nodep = nodep->next) {
330 ESource *existing_source;
331 char *uid = e_source_uid_from_xml_node (nodep);
336 existing_source = e_source_group_peek_source_by_uid (group, uid);
337 if (g_hash_table_lookup (new_sources_hash, existing_source) != NULL)
340 if (existing_source == NULL) {
341 ESource *new_source = e_source_new_from_xml_node (nodep);
343 if (new_source != NULL) {
344 e_source_set_group (new_source, group);
345 g_signal_connect (new_source, "changed", G_CALLBACK (source_changed_callback), group);
346 new_sources_list = g_slist_prepend (new_sources_list, new_source);
348 g_hash_table_insert (new_sources_hash, new_source, new_source);
350 g_signal_emit (group, signals[SOURCE_ADDED], 0, new_source);
354 gboolean source_changed;
356 group->priv->ignore_source_changed ++;
358 if (e_source_update_from_xml_node (existing_source, nodep, &source_changed)) {
359 new_sources_list = g_slist_prepend (new_sources_list, existing_source);
360 g_object_ref (existing_source);
361 g_hash_table_insert (new_sources_hash, existing_source, existing_source);
367 group->priv->ignore_source_changed --;
373 new_sources_list = g_slist_reverse (new_sources_list);
375 /* Emit "group_removed" and disconnect the "changed" signal for all the
376 groups that we haven't found in the new list. */
377 q = new_sources_list;
378 for (p = group->priv->sources; p != NULL; p = p->next) {
379 ESource *source = E_SOURCE (p->data);
381 if (g_hash_table_lookup (new_sources_hash, source) == NULL) {
384 g_signal_emit (group, signals[SOURCE_REMOVED], 0, source);
385 g_signal_handlers_disconnect_by_func (source, source_changed_callback, group);
388 if (! changed && q != NULL) {
389 if (q->data != p->data)
395 g_hash_table_destroy (new_sources_hash);
397 /* Replace the original group list with the new one. */
398 g_slist_foreach (group->priv->sources, (GFunc) g_object_unref, NULL);
399 g_slist_free (group->priv->sources);
401 group->priv->sources = new_sources_list;
403 /* FIXME if the order changes, the function doesn't notice. */
406 g_signal_emit (group, signals[CHANGED], 0);
407 *changed_return = TRUE;
410 return TRUE; /* Success. */
414 e_source_group_uid_from_xmldoc (xmlDocPtr doc)
416 xmlNodePtr root = doc->children;
420 if (root && root->name) {
421 if (strcmp (root->name, "group") != 0)
427 name = xmlGetProp (root, "uid");
431 retval = g_strdup (name);
437 e_source_group_set_name (ESourceGroup *group,
440 g_return_if_fail (E_IS_SOURCE_GROUP (group));
441 g_return_if_fail (name != NULL);
443 if (group->priv->readonly)
446 if (group->priv->name != NULL &&
447 strcmp (group->priv->name, name) == 0)
450 g_free (group->priv->name);
451 group->priv->name = g_strdup (name);
453 g_signal_emit (group, signals[CHANGED], 0);
456 void e_source_group_set_base_uri (ESourceGroup *group,
457 const char *base_uri)
459 g_return_if_fail (E_IS_SOURCE_GROUP (group));
460 g_return_if_fail (base_uri != NULL);
462 if (group->priv->readonly)
465 if (group->priv->base_uri == base_uri)
468 g_free (group->priv->base_uri);
469 group->priv->base_uri = g_strdup (base_uri);
471 g_signal_emit (group, signals[CHANGED], 0);
474 void e_source_group_set_readonly (ESourceGroup *group,
479 g_return_if_fail (E_IS_SOURCE_GROUP (group));
481 if (group->priv->readonly)
484 if (group->priv->readonly == readonly)
487 group->priv->readonly = readonly;
488 for (i = group->priv->sources; i != NULL; i = i->next)
489 e_source_set_readonly (E_SOURCE (i->data), readonly);
491 g_signal_emit (group, signals[CHANGED], 0);
495 e_source_group_peek_uid (ESourceGroup *group)
497 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), NULL);
499 return group->priv->uid;
503 e_source_group_peek_name (ESourceGroup *group)
505 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), NULL);
507 return group->priv->name;
511 e_source_group_peek_base_uri (ESourceGroup *group)
513 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), NULL);
515 return group->priv->base_uri;
519 e_source_group_get_readonly (ESourceGroup *group)
521 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
523 return group->priv->readonly;
527 e_source_group_peek_sources (ESourceGroup *group)
529 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), NULL);
531 return group->priv->sources;
535 e_source_group_peek_source_by_uid (ESourceGroup *group,
540 for (p = group->priv->sources; p != NULL; p = p->next) {
541 if (strcmp (e_source_peek_uid (E_SOURCE (p->data)), uid) == 0)
542 return E_SOURCE (p->data);
549 e_source_group_peek_source_by_name (ESourceGroup *group,
554 for (p = group->priv->sources; p != NULL; p = p->next) {
555 if (strcmp (e_source_peek_name (E_SOURCE (p->data)), name) == 0)
556 return E_SOURCE (p->data);
563 e_source_group_add_source (ESourceGroup *group,
567 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
569 if (group->priv->readonly)
572 if (e_source_group_peek_source_by_uid (group, e_source_peek_uid (source)) != NULL)
575 e_source_set_group (source, group);
576 e_source_set_readonly (source, group->priv->readonly);
577 g_object_ref (source);
579 g_signal_connect (source, "changed", G_CALLBACK (source_changed_callback), group);
581 group->priv->sources = g_slist_insert (group->priv->sources, source, position);
582 g_signal_emit (group, signals[SOURCE_ADDED], 0, source);
583 g_signal_emit (group, signals[CHANGED], 0);
589 e_source_group_remove_source (ESourceGroup *group,
594 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
595 g_return_val_if_fail (E_IS_SOURCE (source), FALSE);
597 if (group->priv->readonly)
600 for (p = group->priv->sources; p != NULL; p = p->next) {
601 if (E_SOURCE (p->data) == source) {
602 group->priv->sources = g_slist_remove_link (group->priv->sources, p);
603 g_signal_handlers_disconnect_by_func (source,
604 G_CALLBACK (source_changed_callback),
606 g_signal_emit (group, signals[SOURCE_REMOVED], 0, source);
607 g_signal_emit (group, signals[CHANGED], 0);
616 e_source_group_remove_source_by_uid (ESourceGroup *group,
621 g_return_val_if_fail (E_IS_SOURCE_GROUP (group), FALSE);
622 g_return_val_if_fail (uid != NULL, FALSE);
624 if (group->priv->readonly)
627 for (p = group->priv->sources; p != NULL; p = p->next) {
628 ESource *source = E_SOURCE (p->data);
630 if (strcmp (e_source_peek_uid (source), uid) == 0) {
631 group->priv->sources = g_slist_remove_link (group->priv->sources, p);
632 g_signal_handlers_disconnect_by_func (source,
633 G_CALLBACK (source_changed_callback),
635 g_signal_emit (group, signals[SOURCE_REMOVED], 0, source);
636 g_signal_emit (group, signals[CHANGED], 0);
646 e_source_group_to_xml (ESourceGroup *group)
651 char *returned_buffer;
655 doc = xmlNewDoc ("1.0");
657 root = xmlNewDocNode (doc, NULL, "group", NULL);
658 xmlSetProp (root, "uid", e_source_group_peek_uid (group));
659 xmlSetProp (root, "name", e_source_group_peek_name (group));
660 xmlSetProp (root, "base_uri", e_source_group_peek_base_uri (group));
661 xmlSetProp (root, "readonly", group->priv->readonly ? "yes" : "no");
663 xmlDocSetRootElement (doc, root);
665 for (p = group->priv->sources; p != NULL; p = p->next)
666 e_source_dump_to_xml_node (E_SOURCE (p->data), root);
668 xmlDocDumpMemory (doc, &xml_buffer, &xml_buffer_size);
671 returned_buffer = g_malloc (xml_buffer_size + 1);
672 memcpy (returned_buffer, xml_buffer, xml_buffer_size);
673 returned_buffer [xml_buffer_size] = '\0';
674 xmlFree (xml_buffer);
676 return returned_buffer;