aboutsummaryrefslogblamecommitdiffstats
path: root/calendar/gui/e-alarm-list.c
blob: 9eb5f2db344ded9dab9a85d737efddbf818a9cf8 (plain) (tree)





















                                                                            
                   





                                
                                    
                                        


























































































































































                                                                                      







































                                                                                       

                                                                     



































                                                                            
                          







                                                                      
                                      
 
                                           

 

                                            
 
                                                                          



                                                                  
                                                        
 
                                      










                                                                                               
                                                     
 


















                                                                                
                                                                           











                                                                                           
                                                            

























































                                                                                            
                                  
                                                                                                             
                                                                                                         


                                      
                                   
                                                                                                              
                                                                                                            


                                      
                                   
                                                                                                              
                                                                                                             


                                      
                                     
                                                                                                                
                                                                                                                     


                                      
                                     
                                                                                                                
                                                                                                                     













                                              
                                            
 

                                          




                                      

                                                            

                         
                                         


                                         
                                           
                                            

                      
                                         


                                          
                                             


                                          

                                           







                                                                        
                                                          



                                                                          


                                                                                                             

                                                                  

                                                                                                             




                                                                                                     

                                                                                             



                                                                                              
                                                        



                                                                          

                                                                                                              


                                                                                                    

                                                                                                              




                                                                                                   

                                                                                              



                                                                                            
                                                      

                                                      







                                                                                          
                                                                   





                                                                                        

                                                                                            



                                                                 
                                                
                

                                                                                      













                                                                                  
                                  





















































































                                                                                      










































                                                                                   
/* -*- 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;
}