aboutsummaryrefslogtreecommitdiffstats
path: root/calendar/pcs/cal-factory.c
diff options
context:
space:
mode:
Diffstat (limited to 'calendar/pcs/cal-factory.c')
-rw-r--r--calendar/pcs/cal-factory.c150
1 files changed, 112 insertions, 38 deletions
diff --git a/calendar/pcs/cal-factory.c b/calendar/pcs/cal-factory.c
index 75c281d235..2f31355e73 100644
--- a/calendar/pcs/cal-factory.c
+++ b/calendar/pcs/cal-factory.c
@@ -20,6 +20,7 @@
*/
#include <config.h>
+#include "cal.h"
#include "cal-backend.h"
#include "cal-factory.h"
#include "job.h"
@@ -29,7 +30,7 @@
/* Private part of the CalFactory structure */
typedef struct {
/* Hash table from GnomeVFSURI structures to CalBackend objects */
- GHashTable *calendars;
+ GHashTable *backends;
} CalFactoryPrivate;
@@ -108,7 +109,21 @@ cal_factory_init (CalFactory *factory)
priv = g_new0 (CalFactoryPrivate, 1);
factory->priv = priv;
- priv->calendars = g_hash_table_new (gnome_vfs_uri_hash, gnome_vfs_uri_hequal);
+ priv->backends = g_hash_table_new (gnome_vfs_uri_hash, gnome_vfs_uri_hequal);
+}
+
+/* Frees a uri/backend pair from the backends hash table */
+static void
+free_backend (gpointer key, gpointer value, gpointer data)
+{
+ GnomeVFSURI *uri;
+ CalBackend *backend;
+
+ uri = key;
+ backend = value;
+
+ gnome_vfs_uri_unref (uri);
+ gtk_object_unref (GTK_OBJECT (backend));
}
/* Destroy handler for the calendar */
@@ -124,7 +139,9 @@ cal_factory_destroy (GtkObject *object)
factory = CAL_FACTORY (object);
priv = factory->priv;
- /* FIXME: free the calendar hash table */
+ g_hash_table_foreach (priv->backends, free_backend, NULL);
+ g_hash_table_destroy (priv->backends);
+ priv->backends = NULL;
g_free (priv);
@@ -199,57 +216,94 @@ typedef struct {
GNOME_Calendar_Listener listener;
} LoadCreateJobData;
-/* Looks up a calendar in a factory's hash table of uri->cal */
-static Cal *
-lookup_calendar (CalFactory *factory, GnomeVFSURI *uri)
+/* Looks up a calendar backend in a factory's hash table of uri->cal */
+static CalBackend *
+lookup_backend (CalFactory *factory, GnomeVFSURI *uri)
{
CalFactoryPrivate *priv;
- Cal *cal;
+ CalBackend *backend;
priv = factory->priv;
- cal = g_hash_table_lookup (priv->calendars, uri);
+ backend = g_hash_table_lookup (priv->backends, uri);
return cal;
}
-/* Loads a calendar and puts it in the factory's hash table */
-static void
-load_calendar (CalFactory *factory, GnomeVFSURI *uri, GNOME_Calendar_Listener listener)
+/* Loads a calendar backend and puts it in the factory's backend hash table */
+static CalBackend *
+load_backend (CalFactory *factory, GnomeVFSURI *uri, GNOME_Calendar_Listener listener)
{
CalFactoryPrivate *priv;
CalBackend *backend;
CalBackendLoadStatus status;
+ CORBA_Environment ev;
priv = factory->priv;
backend = cal_backend_new ();
if (!backend) {
- CORBA_Environment ev;
+ g_message ("load_backend(): could not create the backend");
+ return NULL;
+ }
+
+ status = cal_backend_load (backend, uri);
- g_message ("load_calendar(): could not create the backend");
+ switch (status) {
+ case CAL_BACKEND_LOAD_SUCCESS:
+ gnome_vfs_uri_ref (uri);
+ g_hash_table_insert (priv->backends, uri, backend);
+
+ return backend;
+
+ case CAL_BACKEND_LOAD_ERROR:
+ gtk_object_unref (backend);
+ return NULL;
+
+ default:
+ g_assert_not_reached ();
+ return NULL;
+ }
+}
+
+/* Adds a listener to a calendar backend by creating a calendar client interface
+ * object.
+ */
+static void
+add_calendar_client (CalFactory *factory, CalBackend *backend, GNOME_Calendar_Listener listener)
+{
+ Cal *cal;
+ CORBA_Environment ev;
+
+ cal = cal_new (backend, listener);
+ if (!cal) {
+ g_message ("add_calendar_client(): could not create the calendar client interface");
CORBA_exception_init (&ev);
GNOME_Calendar_Listener_cal_loaded (listener,
GNOME_Calendar_Listener_ERROR,
CORBA_OBJECT_NIL,
&ev);
- if (ev._major != CORBA_NO_EXCEPTION) {
- g_message ("load_calendar(): could not notify the listener");
- CORBA_exception_free (&ev);
- gtk_object_unref (backend);
- return;
- }
+ if (ev._major != CORBA_NO_EXCEPTION)
+ g_message ("add_calendar_client(): could not notify the listener");
+
CORBA_exception_free (&ev);
+ return;
}
- status = cal_backend_load (backend, uri);
-}
+ cal_backend_add_cal (backend, cal);
-/* Adds a listener to a calendar */
-static void
-add_calendar_listener (CalFactory *factory, Cal *cal, GNOME_Calendar_Listener listener)
-{
- /* FIXME */
+ CORBA_exception_init (&ev);
+ GNOME_Calendar_Listener_cal_loaded (listener,
+ GNOME_Calendar_Listener_SUCESSS,
+ gnome_object_corba_objref (GNOME_OBJECT (cal)),
+ &ev);
+
+ if (ev._major != CORBA_NO_EXCEPTION) {
+ g_message ("add_calendar_client(): could not notify the listener");
+ cal_backend_remove_cal (backend, cal);
+ }
+
+ gtk_object_unref (GTK_OBJECT (cal));
}
/* Job handler for the load calendar command */
@@ -257,8 +311,10 @@ static void
load_fn (gpointer data)
{
LoadCreateJobData jd;
+ CalFactory *factory;
GnomeVFSURI *uri;
- Cal *cal;
+ GNOME_Calendar_Listener listener;
+ CalBackend *backend;
CORBA_Environment ev;
jd = data;
@@ -266,27 +322,45 @@ load_fn (gpointer data)
/* Look up the calendar */
uri = gnome_vfs_uri_new (jd->uri);
- cal = lookup_calendar (jd->factory, uri);
+ g_free (jd->uri);
+
+ factory = jd->factory;
+ listener = jd->listener;
+ g_free (jd);
+
+ backend = lookup_backend (factory, uri);
- if (!cal)
- load_calendar (factory, uri, jd->listener);
- else
- add_calendar_listener (factory, cal, jd->listener);
+ if (!backend)
+ backend = load_backend (factory, uri, listener);
gnome_vfs_uri_unref (uri);
- g_free (jd->uri);
+
+ if (!backend) {
+ g_message ("load_fn(): could not load the backend");
+ CORBA_exception_init (&ev);
+ GNOME_Calendar_Listener_cal_loaded (listener,
+ GNOME_Calendar_Listener_ERROR,
+ CORBA_OBJECT_NIL,
+ &ev);
+
+ if (ev._major != CORBA_NO_EXCEPTION)
+ g_message ("load_fn(): could not notify the listener");
+
+ CORBA_exception_free (&ev);
+ goto out;
+ }
+
+ add_calendar_client (factory, backend, listener);
+
+ out:
CORBA_exception_init (&ev);
- CORBA_Object_release (jd->listener, &ev);
+ CORBA_Object_release (listener, &ev);
if (ev._major != CORBA_NO_EXCEPTION)
g_message ("load_fn(): could not release the listener");
CORBA_exception_free (&ev);
-
- /* Done */
-
- g_free (jd);
}