/* * e-summary-calendar.c: * * Copyright (C) 2001 Ximian, Inc. * * Authors: Iain Holmes <iain@ximian.com> */ #ifdef HAVE_CONFIG_H #include <config.h> #endif #include <gnome.h> #include <gal/widgets/e-unicode.h> #include "e-summary-calendar.h" #include "e-summary.h" #include <cal-client/cal-client.h> #include <cal-util/timeutil.h> struct _ESummaryCalendar { CalClient *client; char *html; }; const char * e_summary_calendar_get_html (ESummary *summary) { if (summary->calendar == NULL) { return NULL; } return summary->calendar->html; } /* list_sort_merge, and list_sort are copied from GNOME-VFS. Author: Sven Oliver <sven.over@ob.kamp.net> Modified by Ettore Perazzoli <ettore@comm2000.it> to let the compare functions get an additional gpointer parameter. Included here as using gnome-vfs for 1 20 line function seems a bit of overkill. */ typedef gint (* CalSummaryListCompareFunc) (gconstpointer a, gconstpointer b, gpointer data); static GList * cal_list_sort_merge (GList *l1, GList *l2, CalSummaryListCompareFunc compare_func, gpointer data) { GList list, *l, *lprev; l = &list; lprev = NULL; while (l1 && l2) { if (compare_func (l1->data, l2->data, data) < 0) { l->next = l1; l = l->next; l->prev = lprev; lprev = l; l1 = l1->next; } else { l->next = l2; l = l->next; l->prev = lprev; lprev = l; l2 = l2->next; } } l->next = l1 ? l1 : l2; l->next->prev = l; return list.next; } static GList * cal_list_sort (GList *list, CalSummaryListCompareFunc compare_func, gpointer data) { GList *l1, *l2; if (!list) return NULL; if (!list->next) return list; l1 = list; l2 = list->next; while ((l2 = l2->next) != NULL) { if ((l2 = l2->next) == NULL) break; l1 = l1->next; } l2 = l1->next; l1->next = NULL; return cal_list_sort_merge (cal_list_sort (list, compare_func, data), cal_list_sort (l2, compare_func, data), compare_func, data); } static int sort_uids (gconstpointer a, gconstpointer b, gpointer user_data) { CalComponent *comp_a, *comp_b; ESummary *summary = user_data; ESummaryCalendar *calendar = summary->calendar; CalClientGetStatus status; CalComponentDateTime start_a, start_b; /* a after b then return > 0 */ status = cal_client_get_object (calendar->client, a, &comp_a); if (status != CAL_CLIENT_GET_SUCCESS) return -1; status = cal_client_get_object (calendar->client, b, &comp_b); if (status != CAL_CLIENT_GET_SUCCESS) return 1; cal_component_get_dtstart (comp_a, &start_a); cal_component_get_dtstart (comp_b, &start_b); return icaltime_compare (*start_a.value, *start_b.value); } static gboolean generate_html (gpointer data) { ESummary *summary = data; ESummaryCalendar *calendar = summary->calendar; GList *uids, *l; GString *string; char *tmp; time_t t, begin, end, f; t = time (NULL); begin = time_day_begin (t); switch (summary->preferences->days) { case E_SUMMARY_CALENDAR_ONE_DAY: end = time_day_end (t); break; case E_SUMMARY_CALENDAR_FIVE_DAYS: f = time_add_day (t, 5); end = time_day_end (f); break; case E_SUMMARY_CALENDAR_ONE_WEEK: f = time_add_week (t, 1); end = time_day_end (f); break; case E_SUMMARY_CALENDAR_ONE_MONTH: f = time_add_month (t, 1); end = time_day_end (f); break; } uids = cal_client_get_objects_in_range (calendar->client, CALOBJ_TYPE_EVENT, begin, end); if (uids == NULL) { char *s1, *s2; s1 = e_utf8_from_locale_string (_("Appointments")); s2 = e_utf8_from_locale_string (_("No appointments")); g_free (calendar->html); calendar->html = g_strconcat ("<dl><dt><img src=\"ico-calendar.png\" align=\"middle\" " "alt=\"\" width=\"48\" height=\"48\"> <b><a href=\"evolution:/Local Folders/Calendar\">", s1, "</a></b></dt><dd><b>", s2, "</b></dd></dl>", NULL); g_free (s1); g_free (s2); e_summary_draw (summary); return FALSE; } else { char *s; uids = cal_list_sort (uids, sort_uids, summary); string = g_string_new ("<dl><dt><img src=\"ico-calendar.png\" align=\"middle\" " "alt=\"\" width=\"48\" height=\"48\"> <b><a href=\"evolution:/Local Folders/Calendar\">"); s = e_utf8_from_locale_string (_("Appointments")); g_string_append (string, s); g_free (s); g_string_append (string, "</a></b></dt><dd>"); for (l = uids; l; l = l->next) { char *uid, *start_str; CalComponent *comp; CalComponentText text; CalClientGetStatus status; CalComponentDateTime start, end; time_t start_t, dt; struct tm *start_tm; uid = l->data; status = cal_client_get_object (calendar->client, uid, &comp); if (status != CAL_CLIENT_GET_SUCCESS) { continue; } cal_component_get_summary (comp, &text); cal_component_get_dtstart (comp, &start); cal_component_get_dtend (comp, &end); start_t = icaltime_as_timet (*start.value); start_str = g_new (char, 20); start_tm = localtime (&start_t); dt = start_t - t; /* 86400 == 1 day 604800 == 1 week Otherwise: Month */ if (dt < 86400) { strftime (start_str, 19, _("%l:%M%p"), start_tm); } else if (dt < 604800) { strftime (start_str, 19, _("%a %l:%M%p"), start_tm); } else { strftime (start_str, 19, _("%d %B"), start_tm); } tmp = g_strdup_printf ("<img align=\"middle\" src=\"es-appointments.png\" " "alt=\"\" width=\"16\" height=\"16\">   " "<font size=\"-1\"><a href=\"#\">%s, %s</a></font><br>", start_str, text.value); g_free (start_str); g_string_append (string, tmp); g_free (tmp); } cal_obj_uid_list_free (uids); g_string_append (string, "</dd></dl>"); } if (calendar->html) { g_free (calendar->html); } calendar->html = string->str; g_string_free (string, FALSE); e_summary_draw (summary); return FALSE; } static void cal_opened_cb (CalClient *client, CalClientOpenStatus status, ESummary *summary) { if (status == CAL_CLIENT_OPEN_SUCCESS) { g_idle_add (generate_html, summary); } else { /* Need to work out what to do if there's an error */ } } static void obj_changed_cb (CalClient *client, const char *uid, gpointer data) { g_idle_add (generate_html, data); } static void e_summary_calendar_protocol (ESummary *summary, const char *uri, void *closure) { } void e_summary_calendar_init (ESummary *summary) { ESummaryCalendar *calendar; gboolean result; char *uri; g_return_if_fail (summary != NULL); calendar = g_new (ESummaryCalendar, 1); summary->calendar = calendar; calendar->html = NULL; calendar->client = cal_client_new (); if (calendar->client == NULL) { g_warning ("Error making the client"); return; } gtk_signal_connect (GTK_OBJECT (calendar->client), "cal-opened", GTK_SIGNAL_FUNC (cal_opened_cb), summary); gtk_signal_connect (GTK_OBJECT (calendar->client), "obj-updated", GTK_SIGNAL_FUNC (obj_changed_cb), summary); gtk_signal_connect (GTK_OBJECT (calendar->client), "obj-removed", GTK_SIGNAL_FUNC (obj_changed_cb), summary); uri = gnome_util_prepend_user_home ("evolution/local/Calendar/calendar.ics"); result = cal_client_open_calendar (calendar->client, uri, FALSE); g_free (uri); if (result == FALSE) { g_message ("Open calendar failed"); } e_summary_add_protocol_listener (summary, "calendar", e_summary_calendar_protocol, calendar); } void e_summary_calendar_reconfigure (ESummary *summary) { generate_html (summary); } void e_summary_calendar_free (ESummary *summary) { ESummaryCalendar *calendar; g_return_if_fail (summary != NULL); g_return_if_fail (IS_E_SUMMARY (summary)); calendar = summary->calendar; gtk_object_unref (GTK_OBJECT (calendar->client)); g_free (calendar->html); g_free (calendar); summary->calendar = NULL; }