/*
* EDateTimeList - list of calendar dates/times with GtkTreeModel interface.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) version 3.
*
* 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with the program; if not, see <http://www.gnu.org/licenses/>
*
*
* Authors:
* Hans Petter Jansson <hpj@ximian.com>
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <string.h>
#include <libedataserver/e-time-utils.h>
#include "e-date-time-list.h"
#include <libecal/e-cal-time-util.h>
/* XXX Was it really necessary to implement a custom GtkTreeModel for a
* one-column list store? There's no mention of why this was done. */
#define G_LIST(x) ((GList *) x)
#define E_DATE_TIME_LIST_IS_SORTED(list) \
(E_DATE_TIME_LIST (list)->sort_column_id != -2)
#define IS_VALID_ITER(dt_list, iter) \
(iter != NULL && iter->user_data != NULL && \
dt_list->stamp == iter->stamp)
enum {
PROP_0,
PROP_USE_24_HOUR_FORMAT
};
static GType column_types[E_DATE_TIME_LIST_NUM_COLUMNS];
static void e_date_time_list_tree_model_init (GtkTreeModelIface *iface);
G_DEFINE_TYPE_WITH_CODE (
EDateTimeList, e_date_time_list, G_TYPE_OBJECT,
G_IMPLEMENT_INTERFACE (
GTK_TYPE_TREE_MODEL, e_date_time_list_tree_model_init))
static void
free_datetime (ECalComponentDateTime *datetime)
{
g_free (datetime->value);
if (datetime->tzid)
g_free ((gchar *) datetime->tzid);
g_free (datetime);
}
static ECalComponentDateTime *
copy_datetime (const ECalComponentDateTime *datetime)
{
ECalComponentDateTime *datetime_copy;
datetime_copy = g_new0 (ECalComponentDateTime, 1);
datetime_copy->value = g_new (struct icaltimetype, 1);
*datetime_copy->value = *datetime->value;
if (datetime->tzid)
datetime_copy->tzid = g_strdup (datetime->tzid);
return datetime_copy;
}
static gint
compare_datetime (const ECalComponentDateTime *datetime1,
const ECalComponentDateTime *datetime2)
{
return icaltime_compare (*datetime1->value, *datetime2->value);
}
static void
all_rows_deleted (EDateTimeList *date_time_list)
{
GtkTreePath *path;
gint i;
if (!date_time_list->list)
return;
path = gtk_tree_path_new ();
i = g_list_length (date_time_list->list);
gtk_tree_path_append_index (path, i);
for (; i >= 0; i--) {
gtk_tree_model_row_deleted (GTK_TREE_MODEL (date_time_list), path);
gtk_tree_path_prev (path);
}
gtk_tree_path_free (path);
}
static void
row_deleted (EDateTimeList *date_time_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 (date_time_list), path);
gtk_tree_path_free (path);
}
static void
row_added (EDateTimeList *date_time_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 (date_time_list), &iter, path))
gtk_tree_model_row_inserted (GTK_TREE_MODEL (date_time_list), path, &iter);
gtk_tree_path_free (path);
}
static void
row_updated (EDateTimeList *date_time_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 (date_time_list), &iter, path))
gtk_tree_model_row_changed (GTK_TREE_MODEL (date_time_list), path, &iter);
gtk_tree_path_free (path);
}
/* Builds a static string out of an exception date */
static gchar *
get_exception_string (EDateTimeList *date_time_list,
ECalComponentDateTime *dt)
{
static gchar buf[256];
struct tm tmp_tm;
gboolean use_24_hour_format;
use_24_hour_format =
e_date_time_list_get_use_24_hour_format (date_time_list);
tmp_tm.tm_year = dt->value->year - 1900;
tmp_tm.tm_mon = dt->value->month - 1;
tmp_tm.tm_mday = dt->value->day;
tmp_tm.tm_hour = dt->value->hour;
tmp_tm.tm_min = dt->value->minute;
tmp_tm.tm_sec = dt->value->second;
tmp_tm.tm_isdst = -1;
tmp_tm.tm_wday = time_day_of_week (dt->value->day,
dt->value->month - 1,
dt->value->year);
e_time_format_date_and_time (
&tmp_tm, use_24_hour_format,
FALSE, FALSE, buf, sizeof (buf));
return buf;
}
static void
date_time_list_set_property (GObject *object,
guint property_id,
const GValue *value,
GParamSpec *pspec)
{
switch (property_id) {
case PROP_USE_24_HOUR_FORMAT:
e_date_time_list_set_use_24_hour_format (
E_DATE_TIME_LIST (object),
g_value_get_boolean (value));
return;
}
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}
static void
date_time_list_get_property (GObject *object,
guint property_id,
GValue *value,
GParamSpec *pspec)
{
switch (property_id) {
case PROP_USE_24_HOUR_FORMAT:
g_value_set_boolean (
value,
e_date_time_list_get_use_24_hour_format (
E_DATE_TIME_LIST (object)));
return;
}
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}
static GtkTreeModelFlags
date_time_list_get_flags (GtkTreeModel *tree_model)
{
g_return_val_if_fail (E_IS_DATE_TIME_LIST (tree_model), 0);
return GTK_TREE_MODEL_LIST_ONLY;
}
static gint
date_time_list_get_n_columns (GtkTreeModel *tree_model)
{
EDateTimeList *date_time_list = (EDateTimeList *) tree_model;
g_return_val_if_fail (E_IS_DATE_TIME_LIST (tree_model), 0);
date_time_list->columns_dirty = TRUE;
return E_DATE_TIME_LIST_NUM_COLUMNS;
}
static GType
date_time_list_get_column_type (GtkTreeModel *tree_model,
gint index)
{
EDateTimeList *date_time_list = (EDateTimeList *) tree_model;
g_return_val_if_fail (E_IS_DATE_TIME_LIST (tree_model), G_TYPE_INVALID);
g_return_val_if_fail (index < E_DATE_TIME_LIST_NUM_COLUMNS &&
index >= 0, G_TYPE_INVALID);
date_time_list->columns_dirty = TRUE;
return column_types[index];
}
static gboolean
date_time_list_get_iter (GtkTreeModel *tree_model,
GtkTreeIter *iter,
GtkTreePath *path)
{
EDateTimeList *date_time_list = (EDateTimeList *) tree_model;
GList *l;
gint i;
g_return_val_if_fail (E_IS_DATE_TIME_LIST (tree_model), FALSE);
g_return_val_if_fail (gtk_tree_path_get_depth (path) > 0, FALSE);
if (!date_time_list->list)
return FALSE;
date_time_list->columns_dirty = TRUE;
i = gtk_tree_path_get_indices (path)[0];
l = g_list_nth (date_time_list->list, i);
if (!l)
return FALSE;
iter->user_data = l;
iter->stamp = date_time_list->stamp;
return TRUE;
}
static GtkTreePath *
date_time_list_get_path (GtkTreeModel *tree_model,
GtkTreeIter *iter)
{
EDateTimeList *date_time_list = (EDateTimeList *) tree_model;
GtkTreePath *retval;
GList *l;
g_return_val_if_fail (E_IS_DATE_TIME_LIST (tree_model), NULL);
g_return_val_if_fail (iter->stamp == E_DATE_TIME_LIST (tree_model)->stamp, NULL);
l = iter->user_data;
retval = gtk_tree_path_new ();
gtk_tree_path_append_index (retval, g_list_position (date_time_list->list, l));
return retval;
}
static void
date_time_list_get_value (GtkTreeModel *tree_model,
GtkTreeIter *iter,
gint column,
GValue *value)
{
EDateTimeList *date_time_list = E_DATE_TIME_LIST (tree_model);
ECalComponentDateTime *datetime;
GList *l;
const gchar *str;
g_return_if_fail (E_IS_DATE_TIME_LIST (tree_model));
g_return_if_fail (column < E_DATE_TIME_LIST_NUM_COLUMNS);
g_return_if_fail (E_DATE_TIME_LIST (tree_model)->stamp == iter->stamp);
g_return_if_fail (IS_VALID_ITER (date_time_list, iter));
g_value_init (value, column_types[column]);
if (!date_time_list->list)
return;
l = iter->user_data;
datetime = l->data;
if (!datetime)
return;
switch (column) {
case E_DATE_TIME_LIST_COLUMN_DESCRIPTION:
str = get_exception_string (date_time_list, datetime);
g_value_set_string (value, str);
break;
}
}
static gboolean
date_time_list_iter_next (GtkTreeModel *tree_model,
GtkTreeIter *iter)
{
GList *l;
g_return_val_if_fail (E_IS_DATE_TIME_LIST (tree_model), FALSE);
g_return_val_if_fail (IS_VALID_ITER (E_DATE_TIME_LIST (tree_model), iter), FALSE);
if (!E_DATE_TIME_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
date_time_list_iter_children (GtkTreeModel *tree_model,
GtkTreeIter *iter,
GtkTreeIter *parent)
{
EDateTimeList *date_time_list = E_DATE_TIME_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 (!date_time_list->list)
return FALSE;
iter->stamp = E_DATE_TIME_LIST (tree_model)->stamp;
iter->user_data = date_time_list->list;
return TRUE;
}
static gboolean
date_time_list_iter_has_child (GtkTreeModel *tree_model,
GtkTreeIter *iter)
{
g_return_val_if_fail (IS_VALID_ITER (E_DATE_TIME_LIST (tree_model), iter), FALSE);
return FALSE;
}
static gint
date_time_list_iter_n_children (GtkTreeModel *tree_model,
GtkTreeIter *iter)
{
EDateTimeList *date_time_list = E_DATE_TIME_LIST (tree_model);
g_return_val_if_fail (E_IS_DATE_TIME_LIST (tree_model), -1);
if (iter == NULL)
return g_list_length (date_time_list->list);
g_return_val_if_fail (E_DATE_TIME_LIST (tree_model)->stamp == iter->stamp, -1);
return 0;
}
static gboolean
date_time_list_iter_nth_child (GtkTreeModel *tree_model,
GtkTreeIter *iter,
GtkTreeIter *parent,
gint n)
{
EDateTimeList *date_time_list = E_DATE_TIME_LIST (tree_model);
g_return_val_if_fail (E_IS_DATE_TIME_LIST (tree_model), FALSE);
if (parent)
return FALSE;
if (date_time_list->list) {
GList *l;
l = g_list_nth (date_time_list->list, n);
if (!l)
return FALSE;
iter->stamp = date_time_list->stamp;
iter->user_data = l;
return TRUE;
}
return FALSE;
}
static gboolean
date_time_list_iter_parent (GtkTreeModel *tree_model,
GtkTreeIter *iter,
GtkTreeIter *child)
{
return FALSE;
}
static void
e_date_time_list_class_init (EDateTimeListClass *class)
{
GObjectClass *object_class;
object_class = G_OBJECT_CLASS (class);
object_class->set_property = date_time_list_set_property;
object_class->get_property = date_time_list_get_property;
g_object_class_install_property (
object_class,
PROP_USE_24_HOUR_FORMAT,
g_param_spec_boolean (
"use-24-hour-format",
"Use 24-hour Format",
NULL,
FALSE,
G_PARAM_READWRITE));
column_types[E_DATE_TIME_LIST_COLUMN_DESCRIPTION] = G_TYPE_STRING;
}
static void
e_date_time_list_init (EDateTimeList *date_time_list)
{
date_time_list->stamp = g_random_int ();
date_time_list->columns_dirty = FALSE;
date_time_list->list = NULL;
}
static void
e_date_time_list_tree_model_init (GtkTreeModelIface *iface)
{
iface->get_flags = date_time_list_get_flags;
iface->get_n_columns = date_time_list_get_n_columns;
iface->get_column_type = date_time_list_get_column_type;
iface->get_iter = date_time_list_get_iter;
iface->get_path = date_time_list_get_path;
iface->get_value = date_time_list_get_value;
iface->iter_next = date_time_list_iter_next;
iface->iter_children = date_time_list_iter_children;
iface->iter_has_child = date_time_list_iter_has_child;
iface->iter_n_children = date_time_list_iter_n_children;
iface->iter_nth_child = date_time_list_iter_nth_child;
iface->iter_parent = date_time_list_iter_parent;
}
EDateTimeList *
e_date_time_list_new (void)
{
return g_object_new (E_TYPE_DATE_TIME_LIST, NULL);
}
const ECalComponentDateTime *
e_date_time_list_get_date_time (EDateTimeList *date_time_list, GtkTreeIter *iter)
{
g_return_val_if_fail (IS_VALID_ITER (date_time_list, iter), NULL);
return G_LIST (iter->user_data)->data;
}
void
e_date_time_list_set_date_time (EDateTimeList *date_time_list, GtkTreeIter *iter,
const ECalComponentDateTime *datetime)
{
ECalComponentDateTime *datetime_old;
g_return_if_fail (IS_VALID_ITER (date_time_list, iter));
datetime_old = G_LIST (iter->user_data)->data;
free_datetime (datetime_old);
G_LIST (iter->user_data)->data = copy_datetime (datetime);
row_updated (
date_time_list, g_list_position (
date_time_list->list, G_LIST (iter->user_data)));
}
gboolean
e_date_time_list_get_use_24_hour_format (EDateTimeList *date_time_list)
{
g_return_val_if_fail (E_IS_DATE_TIME_LIST (date_time_list), FALSE);
return date_time_list->use_24_hour_format;
}
void
e_date_time_list_set_use_24_hour_format (EDateTimeList *date_time_list,
gboolean use_24_hour_format)
{
g_return_if_fail (E_IS_DATE_TIME_LIST (date_time_list));
date_time_list->use_24_hour_format = use_24_hour_format;
g_object_notify (G_OBJECT (date_time_list), "use-24-hour-format");
}
void
e_date_time_list_append (EDateTimeList *date_time_list, GtkTreeIter *iter,
const ECalComponentDateTime *datetime)
{
g_return_if_fail (datetime != NULL);
if (g_list_find_custom (
date_time_list->list, datetime,
(GCompareFunc) compare_datetime) == NULL) {
date_time_list->list = g_list_append (
date_time_list->list, copy_datetime (datetime));
row_added (date_time_list, g_list_length (date_time_list->list) - 1);
}
if (iter) {
iter->user_data = g_list_last (date_time_list->list);
iter->stamp = date_time_list->stamp;
}
}
void
e_date_time_list_remove (EDateTimeList *date_time_list, GtkTreeIter *iter)
{
gint n;
g_return_if_fail (IS_VALID_ITER (date_time_list, iter));
n = g_list_position (date_time_list->list, G_LIST (iter->user_data));
free_datetime ((ECalComponentDateTime *) G_LIST (iter->user_data)->data);
date_time_list->list = g_list_delete_link (
date_time_list->list, G_LIST (iter->user_data));
row_deleted (date_time_list, n);
}
void
e_date_time_list_clear (EDateTimeList *date_time_list)
{
GList *l;
all_rows_deleted (date_time_list);
for (l = date_time_list->list; l; l = g_list_next (l)) {
free_datetime ((ECalComponentDateTime *) l->data);
}
g_list_free (date_time_list->list);
date_time_list->list = NULL;
}