diff options
Diffstat (limited to 'calendar/gui/alarm-notify/alarm-notify.c')
-rw-r--r-- | calendar/gui/alarm-notify/alarm-notify.c | 237 |
1 files changed, 180 insertions, 57 deletions
diff --git a/calendar/gui/alarm-notify/alarm-notify.c b/calendar/gui/alarm-notify/alarm-notify.c index 88d25dafbc..61b8e8ecc5 100644 --- a/calendar/gui/alarm-notify/alarm-notify.c +++ b/calendar/gui/alarm-notify/alarm-notify.c @@ -27,6 +27,7 @@ #include <cal-client/cal-client.h> #include "alarm-notify.h" #include "alarm-queue.h" +#include "save.h" @@ -64,8 +65,6 @@ static void AlarmNotify_addCalendar (PortableServer_Servant servant, static void AlarmNotify_removeCalendar (PortableServer_Servant servant, const CORBA_char *str_uri, CORBA_Environment *ev); -static void AlarmNotify_die (PortableServer_Servant servant, - CORBA_Environment *ev); static BonoboXObjectClass *parent_class; @@ -89,7 +88,6 @@ alarm_notify_class_init (AlarmNotifyClass *class) class->epv.addCalendar = AlarmNotify_addCalendar; class->epv.removeCalendar = AlarmNotify_removeCalendar; - class->epv.die = AlarmNotify_die; object_class->destroy = alarm_notify_destroy; } @@ -148,63 +146,124 @@ alarm_notify_destroy (GtkObject *object) /* CORBA servant implementation */ -/* AlarmNotify::addCalendar method */ +/* Looks for a canonicalized URI inside an array of URIs; returns the index + * within the array or -1 if not found. + */ +static int +find_uri_index (GPtrArray *uris, const char *str_uri) +{ + int i; + + for (i = 0; i < uris->len; i++) { + char *uri; + + uri = uris->pdata[i]; + if (strcmp (uri, str_uri) == 0) + break; + } + + if (i == uris->len) + return -1; + else + return i; +} + +/* Frees an array of URIs and the URIs within it. */ static void -AlarmNotify_addCalendar (PortableServer_Servant servant, - const CORBA_char *str_uri, - CORBA_Environment *ev) +free_uris (GPtrArray *uris) { - AlarmNotify *an; - AlarmNotifyPrivate *priv; - GnomeVFSURI *uri; - CalClient *client; - LoadedClient *lc; + int i; - an = ALARM_NOTIFY (bonobo_object_from_servant (servant)); - priv = an->priv; + for (i = 0; i < uris->len; i++) { + char *uri; - uri = gnome_vfs_uri_new (str_uri); - if (!uri) { - CORBA_exception_set (ev, CORBA_USER_EXCEPTION, - ex_GNOME_Evolution_Calendar_AlarmNotify_InvalidURI, - NULL); - return; + uri = uris->pdata[i]; + g_free (uri); } - lc = g_hash_table_lookup (priv->uri_client_hash, uri); + g_ptr_array_free (uris, TRUE); +} - if (lc) { - gnome_vfs_uri_unref (uri); - g_assert (lc->refcount > 0); - lc->refcount++; +/* Adds an URI to the list of calendars to load on startup */ +static void +add_uri_to_load (GnomeVFSURI *uri) +{ + char *str_uri; + GPtrArray *loaded_uris; + int i; + + /* Canonicalize the URI */ + str_uri = gnome_vfs_uri_to_string (uri, GNOME_VFS_URI_HIDE_NONE); + g_assert (str_uri != NULL); + + loaded_uris = get_calendars_to_load (); + g_assert (loaded_uris != NULL); + + /* Look for the URI in the list of calendars to load */ + + i = find_uri_index (loaded_uris, str_uri); + + /* We only need to add the URI if we didn't find it among the list of + * calendars. + */ + if (i != -1) { + g_free (str_uri); + free_uris (loaded_uris); return; } - client = cal_client_new (); + g_ptr_array_add (loaded_uris, str_uri); + save_calendars_to_load (loaded_uris); - if (client) { - if (cal_client_open_calendar (client, str_uri, FALSE)) { - lc = g_new (LoadedClient, 1); - lc->client = client; - lc->uri = uri; - lc->refcount = 1; - g_hash_table_insert (priv->uri_client_hash, uri, lc); + free_uris (loaded_uris); +} - alarm_queue_add_client (client); - } else { - gtk_object_unref (GTK_OBJECT (client)); - client = NULL; - } - } +/* Removes an URI from the list of calendars to load on startup */ +static void +remove_uri_to_load (GnomeVFSURI *uri) +{ + char *str_uri; + GPtrArray *loaded_uris; + char *loaded_uri; + int i; - if (!client) { - gnome_vfs_uri_unref (uri); + /* Canonicalize the URI */ + str_uri = gnome_vfs_uri_to_string (uri, GNOME_VFS_URI_HIDE_NONE); + g_assert (str_uri != NULL); - CORBA_exception_set (ev, CORBA_USER_EXCEPTION, - ex_GNOME_Evolution_Calendar_AlarmNotify_BackendContactError, - NULL); + loaded_uris = get_calendars_to_load (); + g_assert (loaded_uris != NULL); + + /* Look for the URI in the list of calendars to load */ + + i = find_uri_index (loaded_uris, str_uri); + g_free (str_uri); + + /* If we didn't find it, there is no need to remove it */ + if (i == -1) { + free_uris (loaded_uris); return; } + + loaded_uri = loaded_uris->pdata[i]; + g_free (loaded_uri); + + g_ptr_array_remove_index (loaded_uris, i); + save_calendars_to_load (loaded_uris); + + free_uris (loaded_uris); +} + +/* AlarmNotify::addCalendar method */ +static void +AlarmNotify_addCalendar (PortableServer_Servant servant, + const CORBA_char *str_uri, + CORBA_Environment *ev) +{ + AlarmNotify *an; + + an = ALARM_NOTIFY (bonobo_object_from_servant (servant)); + alarm_notify_add_calendar (an, str_uri, TRUE, ev); } /* AlarmNotify::removeCalendar method */ @@ -229,6 +288,8 @@ AlarmNotify_removeCalendar (PortableServer_Servant servant, return; } + remove_uri_to_load (uri); + lc = g_hash_table_lookup (priv->uri_client_hash, uri); gnome_vfs_uri_unref (uri); @@ -252,19 +313,6 @@ AlarmNotify_removeCalendar (PortableServer_Servant servant, g_free (lc); } -static void -AlarmNotify_die (PortableServer_Servant servant, - CORBA_Environment *ev) -{ - AlarmNotify *an; - AlarmNotifyPrivate *priv; - - an = ALARM_NOTIFY (bonobo_object_from_servant (servant)); - priv = an->priv; - - /* FIXME */ -} - /** @@ -283,3 +331,78 @@ alarm_notify_new (void) an = gtk_type_new (TYPE_ALARM_NOTIFY); return an; } + +/** + * alarm_notify_add_calendar: + * @an: An alarm notification service. + * @uri: URI of the calendar to load. + * @load_afterwards: Whether this calendar should be loaded in the future + * when the alarm daemon starts up. + * @ev: CORBA environment for exceptions. + * + * Tells the alarm notification service to load a calendar and start monitoring + * its alarms. It can optionally be made to save the URI of this calendar so + * that it can be loaded in the future when the alarm daemon starts up. + **/ +void +alarm_notify_add_calendar (AlarmNotify *an, const char *str_uri, gboolean load_afterwards, + CORBA_Environment *ev) +{ + AlarmNotifyPrivate *priv; + GnomeVFSURI *uri; + CalClient *client; + LoadedClient *lc; + + g_return_if_fail (an != NULL); + g_return_if_fail (IS_ALARM_NOTIFY (an)); + g_return_if_fail (str_uri != NULL); + g_return_if_fail (ev != NULL); + + priv = an->priv; + + uri = gnome_vfs_uri_new (str_uri); + if (!uri) { + CORBA_exception_set (ev, CORBA_USER_EXCEPTION, + ex_GNOME_Evolution_Calendar_AlarmNotify_InvalidURI, + NULL); + return; + } + + if (load_afterwards) + add_uri_to_load (uri); + + lc = g_hash_table_lookup (priv->uri_client_hash, uri); + + if (lc) { + gnome_vfs_uri_unref (uri); + g_assert (lc->refcount > 0); + lc->refcount++; + return; + } + + client = cal_client_new (); + + if (client) { + if (cal_client_open_calendar (client, str_uri, FALSE)) { + lc = g_new (LoadedClient, 1); + lc->client = client; + lc->uri = uri; + lc->refcount = 1; + g_hash_table_insert (priv->uri_client_hash, uri, lc); + + alarm_queue_add_client (client); + } else { + gtk_object_unref (GTK_OBJECT (client)); + client = NULL; + } + } + + if (!client) { + gnome_vfs_uri_unref (uri); + + CORBA_exception_set (ev, CORBA_USER_EXCEPTION, + ex_GNOME_Evolution_Calendar_AlarmNotify_BackendContactError, + NULL); + return; + } +} |