/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* EAlarmList - list of calendar alarms with GtkTreeModel interface. * * Copyright (C) 2003 Ximian, Inc. * * This program is free software; you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA. * * Authors: Hans Petter Jansson <hpj@ximian.com> */ #include <config.h> #include <string.h> #include <gtk/gtktreemodel.h> #include <gtk/gtksignal.h> #include <gtk/gtktreednd.h> #include <libgnome/gnome-i18n.h> #include <glib.h> #include <libecal/e-cal-time-util.h> #include <libedataserver/e-time-utils.h> #include "calendar-config.h" #include "e-alarm-list.h" #define G_LIST(x) ((GList *) x) #define E_ALARM_LIST_IS_SORTED(list) (E_ALARM_LIST (list)->sort_column_id != -2) #define IS_VALID_ITER(dt_list, iter) (iter!= NULL && iter->user_data != NULL && \ dt_list->stamp == iter->stamp) static GType column_types [E_ALARM_LIST_NUM_COLUMNS]; static void e_alarm_list_init (EAlarmList *file_list); static void e_alarm_list_class_init (EAlarmListClass *class); static void e_alarm_list_tree_model_init (GtkTreeModelIface *iface); static void e_alarm_list_finalize (GObject *object); static guint e_alarm_list_get_flags (GtkTreeModel *tree_model); static gint e_alarm_list_get_n_columns (GtkTreeModel *tree_model); static GType e_alarm_list_get_column_type (GtkTreeModel *tree_model, gint index); static gboolean e_alarm_list_get_iter (GtkTreeModel *tree_model, GtkTreeIter *iter, GtkTreePath *path); static GtkTreePath *e_alarm_list_get_path (GtkTreeModel *tree_model, GtkTreeIter *iter); static void e_alarm_list_get_value (GtkTreeModel *tree_model, GtkTreeIter *iter, gint column, GValue *value); static gboolean e_alarm_list_iter_next (GtkTreeModel *tree_model, GtkTreeIter *iter); static gboolean e_alarm_list_iter_children (GtkTreeModel *tree_model, GtkTreeIter *iter, GtkTreeIter *parent); static gboolean e_alarm_list_iter_has_child (GtkTreeModel *tree_model, GtkTreeIter *iter); static gint e_alarm_list_iter_n_children (GtkTreeModel *tree_model, GtkTreeIter *iter); static gboolean e_alarm_list_iter_nth_child (GtkTreeModel *tree_model, GtkTreeIter *iter, GtkTreeIter *parent, gint n); static gboolean e_alarm_list_iter_parent (GtkTreeModel *tree_model, GtkTreeIter *iter, GtkTreeIter *child); static GObjectClass *parent_class = NULL; GtkType e_alarm_list_get_type (void) { static GType alarm_list_type = 0; if (!alarm_list_type) { static const GTypeInfo alarm_list_info = { sizeof (EAlarmListClass), NULL, /* base_init */ NULL, /* base_finalize */ (GClassInitFunc) e_alarm_list_class_init, NULL, /* class_finalize */ NULL, /* class_data */ sizeof (EAlarmList), 0, (GInstanceInitFunc) e_alarm_list_init, }; static const GInterfaceInfo tree_model_info = { (GInterfaceInitFunc) e_alarm_list_tree_model_init, NULL, NULL }; column_types [E_ALARM_LIST_COLUMN_DESCRIPTION] = G_TYPE_STRING; alarm_list_type = g_type_register_static (G_TYPE_OBJECT, "EAlarmList", &alarm_list_info, 0); g_type_add_interface_static (alarm_list_type, GTK_TYPE_TREE_MODEL, &tree_model_info); } return alarm_list_type; } static void e_alarm_list_class_init (EAlarmListClass *class) { GObjectClass *object_class; parent_class = g_type_class_peek_parent (class); object_class = (GObjectClass *) class; object_class->finalize = e_alarm_list_finalize; } static void e_alarm_list_tree_model_init (GtkTreeModelIface *iface) { iface->get_flags = e_alarm_list_get_flags; iface->get_n_columns = e_alarm_list_get_n_columns; iface->get_column_type = e_alarm_list_get_column_type; iface->get_iter = e_alarm_list_get_iter; iface->get_path = e_alarm_list_get_path; iface->get_value = e_alarm_list_get_value; iface->iter_next = e_alarm_list_iter_next; iface->iter_children = e_alarm_list_iter_children; iface->iter_has_child = e_alarm_list_iter_has_child; iface->iter_n_children = e_alarm_list_iter_n_children; iface->iter_nth_child = e_alarm_list_iter_nth_child; iface->iter_parent = e_alarm_list_iter_parent; } static void e_alarm_list_init (EAlarmList *alarm_list) { alarm_list->stamp = g_random_int (); alarm_list->columns_dirty = FALSE; alarm_list->list = NULL; } EAlarmList * e_alarm_list_new (void) { EAlarmList *alarm_list; alarm_list = E_ALARM_LIST (g_object_new (e_alarm_list_get_type (), NULL)); return alarm_list; } static void all_rows_deleted (EAlarmList *alarm_list) { GtkTreePath *path; gint i; if (!alarm_list->list) return; path = gtk_tree_path_new (); i = g_list_length (alarm_list->list); gtk_tree_path_append_index (path, i); for ( ; i >= 0; i--) { gtk_tree_model_row_deleted (GTK_TREE_MODEL (alarm_list), path); gtk_tree_path_prev (path); } gtk_tree_path_free (path); } static void row_deleted (EAlarmList *alarm_list, gint n) { GtkTreePath *path; path = gtk_tree_path_new (); gtk_tree_path_append_index (path, n); gtk_tree_model_row_deleted (GTK_TREE_MODEL (alarm_list), path); gtk_tree_path_free (path); } static void row_added (EAlarmList *alarm_list, gint n) { GtkTreePath *path; GtkTreeIter iter; path = gtk_tree_path_new (); gtk_tree_path_append_index (path, n); if (gtk_tree_model_get_iter (GTK_TREE_MODEL (alarm_list), &iter, path)) gtk_tree_model_row_inserted (GTK_TREE_MODEL (alarm_list), path, &iter); gtk_tree_path_free (path); } static void row_updated (EAlarmList *alarm_list, gint n) { GtkTreePath *path; GtkTreeIter iter; path = gtk_tree_path_new (); gtk_tree_path_append_index (path, n); if (gtk_tree_model_get_iter (GTK_TREE_MODEL (alarm_list), &iter, path)) gtk_tree_model_row_changed (GTK_TREE_MODEL (alarm_list), path, &iter); gtk_tree_path_free (path); } static void e_alarm_list_finalize (GObject *object) { if (G_OBJECT_CLASS (parent_class)->finalize) (* G_OBJECT_CLASS (parent_class)->finalize) (object); } /* Fulfill the GtkTreeModel requirements */ static guint e_alarm_list_get_flags (GtkTreeModel *tree_model) { g_return_val_if_fail (E_IS_ALARM_LIST (tree_model), 0); return GTK_TREE_MODEL_LIST_ONLY; } static gint e_alarm_list_get_n_columns (GtkTreeModel *tree_model) { EAlarmList *alarm_list = (EAlarmList *) tree_model; g_return_val_if_fail (E_IS_ALARM_LIST (tree_model), 0); alarm_list->columns_dirty = TRUE; return E_ALARM_LIST_NUM_COLUMNS; } static GType e_alarm_list_get_column_type (GtkTreeModel *tree_model, gint index) { EAlarmList *alarm_list = (EAlarmList *) tree_model; g_return_val_if_fail (E_IS_ALARM_LIST (tree_model), G_TYPE_INVALID); g_return_val_if_fail (index < E_ALARM_LIST_NUM_COLUMNS && index >= 0, G_TYPE_INVALID); alarm_list->columns_dirty = TRUE; return column_types [index]; } const ECalComponentAlarm * e_alarm_list_get_alarm (EAlarmList *alarm_list, GtkTreeIter *iter) { g_return_val_if_fail (IS_VALID_ITER (alarm_list, iter), NULL); return G_LIST (iter->user_data)->data; } static void free_alarm (ECalComponentAlarm *alarm) { e_cal_component_alarm_free (alarm); } static ECalComponentAlarm * copy_alarm (const ECalComponentAlarm *alarm) { return e_cal_component_alarm_clone ((ECalComponentAlarm *) alarm); } void e_alarm_list_set_alarm (EAlarmList *alarm_list, GtkTreeIter *iter, const ECalComponentAlarm *alarm) { ECalComponentAlarm *alarm_old; g_return_if_fail (IS_VALID_ITER (alarm_list, iter)); alarm_old = G_LIST (iter->user_data)->data; free_alarm (alarm_old); G_LIST (iter->user_data)->data = copy_alarm (alarm); row_updated (alarm_list, g_list_position (alarm_list->list, G_LIST (iter->user_data))); } void e_alarm_list_append (EAlarmList *alarm_list, GtkTreeIter *iter, const ECalComponentAlarm *alarm) { g_return_if_fail (alarm != NULL); alarm_list->list = g_list_append (alarm_list->list, copy_alarm (alarm)); row_added (alarm_list, g_list_length (alarm_list->list) - 1); if (iter) { iter->user_data = g_list_last (alarm_list->list); iter->stamp = alarm_list->stamp; } } void e_alarm_list_remove (EAlarmList *alarm_list, GtkTreeIter *iter) { gint n; g_return_if_fail (IS_VALID_ITER (alarm_list, iter)); n = g_list_position (alarm_list->list, G_LIST (iter->user_data)); free_alarm ((ECalComponentAlarm *) G_LIST (iter->user_data)->data); alarm_list->list = g_list_delete_link (alarm_list->list, G_LIST (iter->user_data)); row_deleted (alarm_list, n); } void e_alarm_list_clear (EAlarmList *alarm_list) { GList *l; all_rows_deleted (alarm_list); for (l = alarm_list->list; l; l = g_list_next (l)) { free_alarm ((ECalComponentAlarm *) l->data); } g_list_free (alarm_list->list); alarm_list->list = NULL; } static gboolean e_alarm_list_get_iter (GtkTreeModel *tree_model, GtkTreeIter *iter, GtkTreePath *path) { EAlarmList *alarm_list = (EAlarmList *) tree_model; GList *l; gint i; g_return_val_if_fail (E_IS_ALARM_LIST (tree_model), FALSE); g_return_val_if_fail (gtk_tree_path_get_depth (path) > 0, FALSE); if (!alarm_list->list) return FALSE; alarm_list->columns_dirty = TRUE; i = gtk_tree_path_get_indices (path)[0]; l = g_list_nth (alarm_list->list, i); if (!l) return FALSE; iter->user_data = l; iter->stamp = alarm_list->stamp; return TRUE; } static GtkTreePath * e_alarm_list_get_path (GtkTreeModel *tree_model, GtkTreeIter *iter) { EAlarmList *alarm_list = (EAlarmList *) tree_model; GtkTreePath *retval; GList *l; g_return_val_if_fail (E_IS_ALARM_LIST (tree_model), NULL); g_return_val_if_fail (iter->stamp == E_ALARM_LIST (tree_model)->stamp, NULL); l = iter->user_data; retval = gtk_tree_path_new (); gtk_tree_path_append_index (retval, g_list_position (alarm_list->list, l)); return retval; } /* Builds a string for the duration of the alarm. If the duration is zero, returns NULL. */ static char * get_alarm_duration_string (struct icaldurationtype *duration) { GString *string = g_string_new (NULL); char *ret; gboolean have_something; have_something = FALSE; if (duration->days >= 1) { /* Translator: Entire string is like "Pop up an alert %d days before start of appointment" */ g_string_sprintf (string, ngettext("%d day", "%d days", duration->days), duration->days); have_something = TRUE; } if (duration->weeks >= 1) { /* Translator: Entire string is like "Pop up an alert %d weeks before start of appointment" */ g_string_sprintf (string, ngettext("%d week","%d weeks", duration->weeks), duration->weeks); have_something = TRUE; } if (duration->hours >= 1) { /* Translator: Entire string is like "Pop up an alert %d hours before start of appointment" */ g_string_sprintf (string, ngettext("%d hour", "%d hours", duration->hours), duration->hours); have_something = TRUE; } if (duration->minutes >= 1) { /* Translator: Entire string is like "Pop up an alert %d minutes before start of appointment" */ g_string_sprintf (string, ngettext("%d minute", "%d minutes", duration->minutes), duration->minutes); have_something = TRUE; } if (duration->seconds >= 1) { /* Translator: Entire string is like "Pop up an alert %d seconds before start of appointment" */ g_string_sprintf (string, ngettext("%d second", "%d seconds", duration->seconds), duration->seconds); have_something = TRUE; } if (have_something) { ret = string->str; g_string_free (string, FALSE); return ret; } else { g_string_free (string, TRUE); return NULL; } } static char * get_alarm_string (ECalComponentAlarm *alarm) { ECalComponentAlarmAction action; ECalComponentAlarmTrigger trigger; char string[256]; char *base, *str = NULL, *dur; string [0] = '\0'; e_cal_component_alarm_get_action (alarm, &action); e_cal_component_alarm_get_trigger (alarm, &trigger); switch (action) { case E_CAL_COMPONENT_ALARM_AUDIO: base = _("Play a sound"); break; case E_CAL_COMPONENT_ALARM_DISPLAY: base = _("Pop up an alert"); break; case E_CAL_COMPONENT_ALARM_EMAIL: base = _("Send an email"); break; case E_CAL_COMPONENT_ALARM_PROCEDURE: base = _("Run a program"); break; case E_CAL_COMPONENT_ALARM_NONE: case E_CAL_COMPONENT_ALARM_UNKNOWN: default: base = _("Unknown action to be performed"); break; } /* FIXME: This does not look like it will localize correctly. */ switch (trigger.type) { case E_CAL_COMPONENT_ALARM_TRIGGER_RELATIVE_START: dur = get_alarm_duration_string (&trigger.u.rel_duration); if (dur) { if (trigger.u.rel_duration.is_neg) /*Translator: The first %s refers to the base, which would be actions like * "Play a Sound". Second %s refers to the duration string e.g:"15 minutes"*/ str = g_strdup_printf (_("%s %s before the start of the appointment"), base, dur); else /*Translator: The first %s refers to the base, which would be actions like * "Play a Sound". Second %s refers to the duration string e.g:"15 minutes"*/ str = g_strdup_printf (_("%s %s after the start of the appointment"), base, dur); g_free (dur); } else /*Translator: The %s refers to the base, which would be actions like * "Play a sound" */ str = g_strdup_printf (_("%s at the start of the appointment"), base); break; case E_CAL_COMPONENT_ALARM_TRIGGER_RELATIVE_END: dur = get_alarm_duration_string (&trigger.u.rel_duration); if (dur) { if (trigger.u.rel_duration.is_neg) /* Translator: The first %s refers to the base, which would be actions like * "Play a Sound". Second %s refers to the duration string e.g:"15 minutes" */ str = g_strdup_printf (_("%s %s before the end of the appointment"), base, dur); else /* Translator: The first %s refers to the base, which would be actions like * "Play a Sound". Second %s refers to the duration string e.g:"15 minutes" */ str = g_strdup_printf (_("%s %s after the end of the appointment"), base, dur); g_free (dur); } else /* Translator: The %s refers to the base, which would be actions like * "Play a sound" */ str = g_strdup_printf (_("%s at the end of the appointment"), base); break; case E_CAL_COMPONENT_ALARM_TRIGGER_ABSOLUTE: { struct icaltimetype itt; icaltimezone *utc_zone, *current_zone; struct tm tm; char buf[256]; /* Absolute triggers come in UTC, so convert them to the local timezone */ itt = trigger.u.abs_time; utc_zone = icaltimezone_get_utc_timezone (); current_zone = calendar_config_get_icaltimezone (); tm = icaltimetype_to_tm_with_zone (&itt, utc_zone, current_zone); e_time_format_date_and_time (&tm, calendar_config_get_24_hour_format (), FALSE, FALSE, buf, sizeof (buf)); /* Translator: The first %s refers to the base, which would be actions like * "Play a Sound". Second %s is an absolute time, e.g. "10:00AM" */ str = g_strdup_printf (_("%s at %s"), base, buf); break; } case E_CAL_COMPONENT_ALARM_TRIGGER_NONE: default: /* Translator: The %s refers to the base, which would be actions like * "Play a sound". "Trigger types" are absolute or relative dates */ str = g_strdup_printf (_("%s for an unknown trigger type"), base); break; } return str; } static void e_alarm_list_get_value (GtkTreeModel *tree_model, GtkTreeIter *iter, gint column, GValue *value) { EAlarmList *alarm_list = E_ALARM_LIST (tree_model); ECalComponentAlarm *alarm; GList *l; const gchar *str; g_return_if_fail (E_IS_ALARM_LIST (tree_model)); g_return_if_fail (column < E_ALARM_LIST_NUM_COLUMNS); g_return_if_fail (E_ALARM_LIST (tree_model)->stamp == iter->stamp); g_return_if_fail (IS_VALID_ITER (alarm_list, iter)); g_value_init (value, column_types [column]); if (!alarm_list->list) return; l = iter->user_data; alarm = l->data; if (!alarm) return; switch (column) { case E_ALARM_LIST_COLUMN_DESCRIPTION: str = get_alarm_string (alarm); g_value_set_string (value, str); break; } } static gboolean e_alarm_list_iter_next (GtkTreeModel *tree_model, GtkTreeIter *iter) { GList *l; g_return_val_if_fail (E_IS_ALARM_LIST (tree_model), FALSE); g_return_val_if_fail (IS_VALID_ITER (E_ALARM_LIST (tree_model), iter), FALSE); if (!E_ALARM_LIST (tree_model)->list) return FALSE; l = iter->user_data; l = g_list_next (l); if (l) { iter->user_data = l; return TRUE; } return FALSE; } static gboolean e_alarm_list_iter_children (GtkTreeModel *tree_model, GtkTreeIter *iter, GtkTreeIter *parent) { EAlarmList *alarm_list = E_ALARM_LIST (tree_model); /* this is a list, nodes have no children */ if (parent) return FALSE; /* but if parent == NULL we return the list itself as children of the * "root" */ if (!alarm_list->list) return FALSE; iter->stamp = E_ALARM_LIST (tree_model)->stamp; iter->user_data = alarm_list->list; return TRUE; } static gboolean e_alarm_list_iter_has_child (GtkTreeModel *tree_model, GtkTreeIter *iter) { g_return_val_if_fail (IS_VALID_ITER (E_ALARM_LIST (tree_model), iter), FALSE); return FALSE; } static gint e_alarm_list_iter_n_children (GtkTreeModel *tree_model, GtkTreeIter *iter) { EAlarmList *alarm_list = E_ALARM_LIST (tree_model); g_return_val_if_fail (E_IS_ALARM_LIST (tree_model), -1); if (iter == NULL) return g_list_length (alarm_list->list); g_return_val_if_fail (E_ALARM_LIST (tree_model)->stamp == iter->stamp, -1); return 0; } static gboolean e_alarm_list_iter_nth_child (GtkTreeModel *tree_model, GtkTreeIter *iter, GtkTreeIter *parent, gint n) { EAlarmList *alarm_list = E_ALARM_LIST (tree_model); g_return_val_if_fail (E_IS_ALARM_LIST (tree_model), FALSE); if (parent) return FALSE; if (alarm_list->list) { GList *l; l = g_list_nth (alarm_list->list, n); if (!l) return FALSE; iter->stamp = alarm_list->stamp; iter->user_data = l; return TRUE; } return FALSE; } static gboolean e_alarm_list_iter_parent (GtkTreeModel *tree_model, GtkTreeIter *iter, GtkTreeIter *child) { return FALSE; }