/*
* 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:
* Damon Chaplin <damon@ximian.com>
* Rodrigo Moya <rodrigo@ximian.com>
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
*/
/*
* ECalendarTable - displays the ECalComponent objects in a table (an ETable).
* Used for calendar events and tasks.
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <sys/stat.h>
#include <unistd.h>
#include <glib.h>
#include <glib/gi18n.h>
#include <glib/gstdio.h>
#include <gtk/gtk.h>
#include <gdk/gdkkeysyms.h>
#include <misc/e-gui-utils.h>
#include <table/e-cell-checkbox.h>
#include <table/e-cell-toggle.h>
#include <table/e-cell-text.h>
#include <table/e-cell-combo.h>
#include <e-util/e-dialog-utils.h>
#include <e-util/e-util-private.h>
#include <misc/e-cell-date-edit.h>
#include <misc/e-cell-percent.h>
#include <libecal/e-cal-time-util.h>
#include <libedataserver/e-time-utils.h>
#include "calendar-component.h"
#include "calendar-config.h"
#include "dialogs/delete-comp.h"
#include "dialogs/delete-error.h"
#include "dialogs/task-editor.h"
#include "e-cal-model-tasks.h"
#include "e-calendar-table.h"
#include "e-calendar-view.h"
#include "e-cell-date-edit-text.h"
#include "e-comp-editor-registry.h"
#include "print.h"
#include <e-util/e-icon-factory.h>
#include "e-cal-popup.h"
#include "e-tasks.h"
#include "misc.h"
enum TargetType{
TARGET_TYPE_VCALENDAR
};
static GtkTargetEntry target_types[] = {
{ (gchar *) "text/x-calendar", 0, TARGET_TYPE_VCALENDAR },
{ (gchar *) "text/calendar", 0, TARGET_TYPE_VCALENDAR }
};
static guint n_target_types = G_N_ELEMENTS (target_types);
extern ECompEditorRegistry *comp_editor_registry;
static void e_calendar_table_destroy (GtkObject *object);
static void e_calendar_table_on_double_click (ETable *table,
gint row,
gint col,
GdkEvent *event,
ECalendarTable *cal_table);
static gint e_calendar_table_show_popup_menu (ETable *table,
GdkEvent *gdk_event,
ECalendarTable *cal_table);
static gint e_calendar_table_on_right_click (ETable *table,
gint row,
gint col,
GdkEvent *event,
ECalendarTable *cal_table);
static gboolean e_calendar_table_on_popup_menu (GtkWidget *widget,
gpointer data);
static gint e_calendar_table_on_key_press (ETable *table,
gint row,
gint col,
GdkEventKey *event,
ECalendarTable *cal_table);
static struct tm e_calendar_table_get_current_time (ECellDateEdit *ecde,
gpointer data);
static void mark_as_complete_cb (EPopup *ep, EPopupItem *pitem, gpointer data);
static void hide_completed_rows (ECalModel *model, GList *clients_list, gchar *hide_sexp, GPtrArray *comp_objects);
static void show_completed_rows (ECalModel *model, GList *clients_list, gchar *show_sexp, GPtrArray *comp_objects);
/* Signal IDs */
enum {
USER_CREATED,
LAST_SIGNAL
};
static guint signals[LAST_SIGNAL] = { 0 };
/* The icons to represent the task. */
#define E_CALENDAR_MODEL_NUM_ICONS 4
static const gchar * icon_names[E_CALENDAR_MODEL_NUM_ICONS] = {
"stock_task", "stock_task-recurring", "stock_task-assigned", "stock_task-assigned-to"
};
static GdkPixbuf* icon_pixbufs[E_CALENDAR_MODEL_NUM_ICONS] = { NULL };
static GdkAtom clipboard_atom = GDK_NONE;
G_DEFINE_TYPE (ECalendarTable, e_calendar_table, GTK_TYPE_TABLE)
static void
e_calendar_table_class_init (ECalendarTableClass *class)
{
GtkObjectClass *object_class;
object_class = (GtkObjectClass *) class;
/* Method override */
object_class->destroy = e_calendar_table_destroy;
signals[USER_CREATED] =
g_signal_new ("user_created",
G_TYPE_FROM_CLASS (class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (ECalendarTableClass, user_created),
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
/* clipboard atom */
if (!clipboard_atom)
clipboard_atom = gdk_atom_intern ("CLIPBOARD", FALSE);
}
static gint
date_compare_cb (gconstpointer a, gconstpointer b)
{
ECellDateEditValue *dv1 = (ECellDateEditValue *) a;
ECellDateEditValue *dv2 = (ECellDateEditValue *) b;
struct icaltimetype tt;
/* First check if either is NULL. NULL dates sort last. */
if (!dv1 || !dv2) {
if (dv1 == dv2)
return 0;
else if (dv1)
return -1;
else
return 1;
}
/* Copy the 2nd value and convert it to the same timezone as the
first. */
tt = dv2->tt;
icaltimezone_convert_time (&tt, dv2->zone, dv1->zone);
/* Now we can compare them. */
return icaltime_compare (dv1->tt, tt);
}
static gint
percent_compare_cb (gconstpointer a, gconstpointer b)
{
gint percent1 = GPOINTER_TO_INT (a);
gint percent2 = GPOINTER_TO_INT (b);
gint retval;
if (percent1 > percent2)
retval = 1;
else if (percent1 < percent2)
retval = -1;
else
retval = 0;
return retval;
}
static gint
priority_compare_cb (gconstpointer a, gconstpointer b)
{
gint priority1, priority2;
priority1 = e_cal_util_priority_from_string ((const gchar *) a);
priority2 = e_cal_util_priority_from_string ((const gchar *) b);
/* We change undefined priorities so they appear after 'Low'. */
if (priority1 <= 0)
priority1 = 10;
if (priority2 <= 0)
priority2 = 10;
/* We'll just use the ordering of the priority values. */
if (priority1 < priority2)
return -1;
else if (priority1 > priority2)
return 1;
else
return 0;
}
static gint
status_from_string (const gchar *str)
{
gint status = -2;
if (!str || !str[0])
status = -1;
else if (!g_utf8_collate (str, _("Not Started")))
status = 0;
else if (!g_utf8_collate (str, _("In Progress")))
status = 1;
else if (!g_utf8_collate (str, _("Completed")))
status = 2;
else if (!g_utf8_collate (str, _("Canceled")))
status = 3;
return status;
}
static gint
status_compare_cb (gconstpointer a, gconstpointer b)
{
gint sa = status_from_string ((const gchar *)a);
gint sb = status_from_string ((const gchar *)b);
if (sa < sb)
return -1;
else if (sa > sb)
return 1;
return 0;
}
static void
row_appended_cb (ECalModel *model, ECalendarTable *cal_table)
{
g_signal_emit (cal_table, signals[USER_CREATED], 0);
}
static void
get_time_as_text (struct icaltimetype *tt, icaltimezone *f_zone, icaltimezone *t_zone, gchar *buff, gint buff_len)
{
struct tm tmp_tm;
buff [0] = 0;
tmp_tm = icaltimetype_to_tm_with_zone (tt, f_zone, t_zone);
e_time_format_date_and_time (&tmp_tm,
calendar_config_get_24_hour_format (),
FALSE, FALSE,
buff, buff_len);
}
gboolean
ec_query_tooltip (GtkWidget *widget, gint x, gint y, gboolean keyboard_mode, GtkTooltip *tooltip, GtkWidget *etable_wgt, ECalModel *model)
{
ECalModelComponent *comp;
gint row = -1, col = -1;
GtkWidget *box, *l, *w;
GtkStyle *style = gtk_widget_get_default_style ();
gchar *tmp;
const gchar *str;
GString *tmp2;
gchar buff[1001];
gboolean free_text = FALSE;
ECalComponent *new_comp;
ECalComponentOrganizer organizer;
ECalComponentDateTime dtstart, dtdue;
icaltimezone *zone, *default_zone;
GSList *desc, *p;
gint len;
ETable *etable;
ESelectionModel *esm;
if (keyboard_mode)
return FALSE;
g_return_val_if_fail (widget != NULL, FALSE);
g_return_val_if_fail (tooltip != NULL, FALSE);
g_return_val_if_fail (E_IS_TABLE (etable_wgt), FALSE);
g_return_val_if_fail (E_IS_CAL_MODEL (model), FALSE);
etable = E_TABLE (etable_wgt);
e_table_get_mouse_over_cell (etable, &row, &col);
if (row == -1 || !etable)
return FALSE;
/* respect sorting option, the 'e_table_get_mouse_over_cell' returns sorted row, not the model one */
esm = e_table_get_selection_model (etable);
if (esm && esm->sorter && e_sorter_needs_sorting (esm->sorter))
row = e_sorter_sorted_to_model (esm->sorter, row);
comp = e_cal_model_get_component_at (model, row);
if (!comp || !comp->icalcomp)
return FALSE;
new_comp = e_cal_component_new ();
if (!e_cal_component_set_icalcomponent (new_comp, icalcomponent_new_clone (comp->icalcomp))) {
g_object_unref (new_comp);
return FALSE;
}
box = gtk_vbox_new (FALSE, 0);
str = e_calendar_view_get_icalcomponent_summary (comp->client, comp->icalcomp, &free_text);
if (!(str && *str)) {
if (free_text)
g_free ((gchar *)str);
free_text = FALSE;
str = _("* No Summary *");
}
l = gtk_label_new (NULL);
tmp = g_markup_printf_escaped ("<b>%s</b>", str);
gtk_label_set_line_wrap (GTK_LABEL (l), TRUE);
gtk_label_set_markup (GTK_LABEL (l), tmp);
gtk_misc_set_alignment (GTK_MISC (l), 0.0, 0.5);
w = gtk_event_box_new ();
gtk_widget_modify_bg (w, GTK_STATE_NORMAL, &(style->bg[GTK_STATE_SELECTED]));
gtk_widget_modify_fg (l, GTK_STATE_NORMAL, &(style->text[GTK_STATE_SELECTED]));
gtk_container_add (GTK_CONTAINER (w), l);
gtk_box_pack_start (GTK_BOX (box), w, TRUE, TRUE, 0);
g_free (tmp);
if (free_text)
g_free ((gchar *)str);
free_text = FALSE;
w = gtk_event_box_new ();
gtk_widget_modify_bg (w, GTK_STATE_NORMAL, &(style->bg[GTK_STATE_NORMAL]));
l = gtk_vbox_new (FALSE, 0);
gtk_container_add (GTK_CONTAINER (w), l);
gtk_box_pack_start (GTK_BOX (box), w, FALSE, FALSE, 0);
w = l;
e_cal_component_get_organizer (new_comp, &organizer);
if (organizer.cn) {
gchar *ptr;
ptr = strchr( organizer.value, ':');
if (ptr) {
ptr++;
/* To Translators: It will display "Organiser: NameOfTheUser <email@ofuser.com>" */
tmp = g_strdup_printf (_("Organizer: %s <%s>"), organizer.cn, ptr);
} else {
/* With SunOne accounts, there may be no ':' in organiser.value */
tmp = g_strdup_printf (_("Organizer: %s"), organizer.cn);
}
l = gtk_label_new (tmp);
gtk_label_set_line_wrap (GTK_LABEL (l), FALSE);
gtk_misc_set_alignment (GTK_MISC (l), 0.0, 0.5);
gtk_box_pack_start (GTK_BOX (w), l, FALSE, FALSE, 0);
g_free (tmp);
}
e_cal_component_get_dtstart (new_comp, &dtstart);
e_cal_component_get_due (new_comp, &dtdue);
default_zone = e_cal_model_get_timezone (model);
if (dtstart.tzid) {
zone = icalcomponent_get_timezone (e_cal_component_get_icalcomponent (new_comp), dtstart.tzid);
if (!zone)
e_cal_get_timezone (comp->client, dtstart.tzid, &zone, NULL);
if (!zone)
zone = default_zone;
} else {
zone = NULL;
}
tmp2 = g_string_new ("");
if (dtstart.value) {
get_time_as_text (dtstart.value, zone, default_zone, buff, 1000);
if (buff [0]) {
g_string_append (tmp2, _("Start: "));
g_string_append (tmp2, buff);
}
}
if (dtdue.value) {
get_time_as_text (dtdue.value, zone, default_zone, buff, 1000);
if (buff [0]) {
if (tmp2->len)
g_string_append (tmp2, "; ");
g_string_append (tmp2, _("Due: "));
g_string_append (tmp2, buff);
}
}
if (tmp2->len) {
l = gtk_label_new (tmp2->str);
gtk_misc_set_alignment (GTK_MISC (l), 0.0, 0.5);
gtk_box_pack_start (GTK_BOX (w), l, FALSE, FALSE, 0);
}
g_string_free (tmp2, TRUE);
e_cal_component_free_datetime (&dtstart);
e_cal_component_free_datetime (&dtdue);
tmp = e_calendar_view_get_attendees_status_info (new_comp, comp->client);
if (tmp) {
l = gtk_label_new (tmp);
gtk_misc_set_alignment (GTK_MISC (l), 0.0, 0.5);
gtk_box_pack_start (GTK_BOX (w), l, FALSE, FALSE, 0);
g_free (tmp);
tmp = NULL;
}
tmp2 = g_string_new ("");
e_cal_component_get_description_list (new_comp, &desc);
for (len = 0, p = desc; p != NULL; p = p->next) {
ECalComponentText *text = p->data;
if (text->value != NULL) {
len += strlen (text->value);
g_string_append (tmp2, text->value);
if (len > 1024) {
g_string_set_size (tmp2, 1020);
g_string_append (tmp2, "...");
break;
}
}
}
e_cal_component_free_text_list (desc);
if (tmp2->len) {
l = gtk_label_new (tmp2->str);
gtk_label_set_line_wrap (GTK_LABEL (l), TRUE);
gtk_misc_set_alignment (GTK_MISC (l), 0.0, 0.5);
gtk_box_pack_start (GTK_BOX (box), l, FALSE, FALSE, 0);
}
g_string_free (tmp2, TRUE);
gtk_widget_show_all (box);
gtk_tooltip_set_custom (tooltip, box);
g_object_unref (new_comp);
return TRUE;
}
static gboolean
query_tooltip_cb (GtkWidget *widget, gint x, gint y, gboolean keyboard_mode, GtkTooltip *tooltip, gpointer user_data)
{
ECalendarTable *cal_table;
g_return_val_if_fail (E_IS_CALENDAR_TABLE (user_data), FALSE);
cal_table = E_CALENDAR_TABLE (user_data);
return ec_query_tooltip (widget, x, y, keyboard_mode, tooltip, GTK_WIDGET (e_calendar_table_get_table (cal_table)), cal_table->model);
}
static void
e_calendar_table_init (ECalendarTable *cal_table)
{
GtkWidget *table;
ETable *e_table;
ECell *cell, *popup_cell;
ETableExtras *extras;
gint i;
GdkPixbuf *pixbuf;
GList *strings;
AtkObject *a11y;
gchar *etspecfile;
/* Create the model */
cal_table->model = (ECalModel *) e_cal_model_tasks_new ();
g_signal_connect (cal_table->model, "row_appended", G_CALLBACK (row_appended_cb), cal_table);
cal_table->user_created_cal = NULL;
/* Create the header columns */
extras = e_table_extras_new();
/*
* Normal string fields.
*/
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
g_object_set (G_OBJECT (cell),
"strikeout_column", E_CAL_MODEL_TASKS_FIELD_STRIKEOUT,
"bold_column", E_CAL_MODEL_TASKS_FIELD_OVERDUE,
"bg_color_column", E_CAL_MODEL_FIELD_COLOR,
NULL);
e_table_extras_add_cell (extras, "calstring", cell);
/*
* Date fields.
*/
cell = e_cell_date_edit_text_new (NULL, GTK_JUSTIFY_LEFT);
g_object_set (G_OBJECT (cell),
"strikeout_column", E_CAL_MODEL_TASKS_FIELD_STRIKEOUT,
"bold_column", E_CAL_MODEL_TASKS_FIELD_OVERDUE,
"bg_color_column", E_CAL_MODEL_FIELD_COLOR,
NULL);
popup_cell = e_cell_date_edit_new ();
e_cell_popup_set_child (E_CELL_POPUP (popup_cell), cell);
g_object_unref (cell);
e_table_extras_add_cell (extras, "dateedit", popup_cell);
cal_table->dates_cell = E_CELL_DATE_EDIT (popup_cell);
e_cell_date_edit_set_get_time_callback (E_CELL_DATE_EDIT (popup_cell),
e_calendar_table_get_current_time,
cal_table, NULL);
/*
* Combo fields.
*/
/* Classification field. */
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
g_object_set (G_OBJECT (cell),
"strikeout_column", E_CAL_MODEL_TASKS_FIELD_STRIKEOUT,
"bold_column", E_CAL_MODEL_TASKS_FIELD_OVERDUE,
"bg_color_column", E_CAL_MODEL_FIELD_COLOR,
"editable", FALSE,
NULL);
popup_cell = e_cell_combo_new ();
e_cell_popup_set_child (E_CELL_POPUP (popup_cell), cell);
g_object_unref (cell);
strings = NULL;
strings = g_list_append (strings, (gchar *) _("Public"));
strings = g_list_append (strings, (gchar *) _("Private"));
strings = g_list_append (strings, (gchar *) _("Confidential"));
e_cell_combo_set_popdown_strings (E_CELL_COMBO (popup_cell),
strings);
e_table_extras_add_cell (extras, "classification", popup_cell);
/* Priority field. */
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
g_object_set (G_OBJECT (cell),
"strikeout_column", E_CAL_MODEL_TASKS_FIELD_STRIKEOUT,
"bold_column", E_CAL_MODEL_TASKS_FIELD_OVERDUE,
"bg_color_column", E_CAL_MODEL_FIELD_COLOR,
"editable", FALSE,
NULL);
popup_cell = e_cell_combo_new ();
e_cell_popup_set_child (E_CELL_POPUP (popup_cell), cell);
g_object_unref (cell);
strings = NULL;
strings = g_list_append (strings, (gchar *) _("High"));
strings = g_list_append (strings, (gchar *) _("Normal"));
strings = g_list_append (strings, (gchar *) _("Low"));
strings = g_list_append (strings, (gchar *) _("Undefined"));
e_cell_combo_set_popdown_strings (E_CELL_COMBO (popup_cell),
strings);
e_table_extras_add_cell (extras, "priority", popup_cell);
/* Percent field. */
cell = e_cell_percent_new (NULL, GTK_JUSTIFY_LEFT);
g_object_set (G_OBJECT (cell),
"strikeout_column", E_CAL_MODEL_TASKS_FIELD_STRIKEOUT,
"bold_column", E_CAL_MODEL_TASKS_FIELD_OVERDUE,
"bg_color_column", E_CAL_MODEL_FIELD_COLOR,
NULL);
popup_cell = e_cell_combo_new ();
e_cell_popup_set_child (E_CELL_POPUP (popup_cell), cell);
g_object_unref (cell);
strings = NULL;
strings = g_list_append (strings, (gchar *) _("0%"));
strings = g_list_append (strings, (gchar *) _("10%"));
strings = g_list_append (strings, (gchar *) _("20%"));
strings = g_list_append (strings, (gchar *) _("30%"));
strings = g_list_append (strings, (gchar *) _("40%"));
strings = g_list_append (strings, (gchar *) _("50%"));
strings = g_list_append (strings, (gchar *) _("60%"));
strings = g_list_append (strings, (gchar *) _("70%"));
strings = g_list_append (strings, (gchar *) _("80%"));
strings = g_list_append (strings, (gchar *) _("90%"));
strings = g_list_append (strings, (gchar *) _("100%"));
e_cell_combo_set_popdown_strings (E_CELL_COMBO (popup_cell),
strings);
e_table_extras_add_cell (extras, "percent", popup_cell);
/* Transparency field. */
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
g_object_set (G_OBJECT (cell),
"strikeout_column", E_CAL_MODEL_TASKS_FIELD_STRIKEOUT,
"bold_column", E_CAL_MODEL_TASKS_FIELD_OVERDUE,
"bg_color_column", E_CAL_MODEL_FIELD_COLOR,
"editable", FALSE,
NULL);
popup_cell = e_cell_combo_new ();
e_cell_popup_set_child (E_CELL_POPUP (popup_cell), cell);
g_object_unref (cell);
strings = NULL;
strings = g_list_append (strings, (gchar *) _("Free"));
strings = g_list_append (strings, (gchar *) _("Busy"));
e_cell_combo_set_popdown_strings (E_CELL_COMBO (popup_cell),
strings);
e_table_extras_add_cell (extras, "transparency", popup_cell);
/* Status field. */
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
g_object_set (G_OBJECT (cell),
"strikeout_column", E_CAL_MODEL_TASKS_FIELD_STRIKEOUT,
"bold_column", E_CAL_MODEL_TASKS_FIELD_OVERDUE,
"bg_color_column", E_CAL_MODEL_FIELD_COLOR,
"editable", FALSE,
NULL);
popup_cell = e_cell_combo_new ();
e_cell_popup_set_child (E_CELL_POPUP (popup_cell), cell);
g_object_unref (cell);
strings = NULL;
strings = g_list_append (strings, (gchar *) _("Not Started"));
strings = g_list_append (strings, (gchar *) _("In Progress"));
strings = g_list_append (strings, (gchar *) _("Completed"));
strings = g_list_append (strings, (gchar *) _("Canceled"));
e_cell_combo_set_popdown_strings (E_CELL_COMBO (popup_cell),
strings);
e_table_extras_add_cell (extras, "calstatus", popup_cell);
e_table_extras_add_compare (extras, "date-compare",
date_compare_cb);
e_table_extras_add_compare (extras, "percent-compare",
percent_compare_cb);
e_table_extras_add_compare (extras, "priority-compare",
priority_compare_cb);
e_table_extras_add_compare (extras, "status-compare",
status_compare_cb);
/* Create pixmaps */
if (!icon_pixbufs[0])
for (i = 0; i < E_CALENDAR_MODEL_NUM_ICONS; i++) {
icon_pixbufs[i] = e_icon_factory_get_icon (icon_names[i], GTK_ICON_SIZE_MENU);
}
cell = e_cell_toggle_new (0, E_CALENDAR_MODEL_NUM_ICONS, icon_pixbufs);
e_table_extras_add_cell(extras, "icon", cell);
e_table_extras_add_pixbuf(extras, "icon", icon_pixbufs[0]);
pixbuf = e_icon_factory_get_icon ("stock_check-filled", GTK_ICON_SIZE_MENU);
e_table_extras_add_pixbuf(extras, "complete", pixbuf);
g_object_unref(pixbuf);
/* Create the table */
etspecfile = g_build_filename (EVOLUTION_ETSPECDIR,
"e-calendar-table.etspec",
NULL);
table = e_table_scrolled_new_from_spec_file (E_TABLE_MODEL (cal_table->model),
extras,
etspecfile,
NULL);
g_free (etspecfile);
/* FIXME: this causes a message from GLib about 'extras' having only a floating
reference */
/* g_object_unref (extras); */
cal_table->etable = table;
gtk_table_attach (GTK_TABLE (cal_table), table, 0, 1, 0, 1,
GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
gtk_widget_show (table);
e_table = e_table_scrolled_get_table (E_TABLE_SCROLLED (table));
g_signal_connect (e_table, "double_click", G_CALLBACK (e_calendar_table_on_double_click), cal_table);
g_signal_connect (e_table, "right_click", G_CALLBACK (e_calendar_table_on_right_click), cal_table);
g_signal_connect (e_table, "key_press", G_CALLBACK (e_calendar_table_on_key_press), cal_table);
g_signal_connect (e_table, "popup_menu", G_CALLBACK (e_calendar_table_on_popup_menu), cal_table);
g_signal_connect (e_table, "query-tooltip", G_CALLBACK (query_tooltip_cb), cal_table);
gtk_widget_set_has_tooltip (GTK_WIDGET (e_table), TRUE);
a11y = gtk_widget_get_accessible ((GtkWidget *)e_table);
if (a11y)
atk_object_set_name (a11y, _("Tasks"));
}
/**
* e_calendar_table_new:
* @Returns: a new #ECalendarTable.
*
* Creates a new #ECalendarTable.
**/
GtkWidget *
e_calendar_table_new (void)
{
GtkWidget *cal_table;
cal_table = GTK_WIDGET (g_object_new (e_calendar_table_get_type (), NULL));
return cal_table;
}
/**
* e_calendar_table_get_model:
* @cal_table: A calendar table.
*
* Queries the calendar data model that a calendar table is using.
*
* Return value: A calendar model.
**/
ECalModel *
e_calendar_table_get_model (ECalendarTable *cal_table)
{
g_return_val_if_fail (cal_table != NULL, NULL);
g_return_val_if_fail (E_IS_CALENDAR_TABLE (cal_table), NULL);
return cal_table->model;
}
static void
e_calendar_table_destroy (GtkObject *object)
{
ECalendarTable *cal_table;
cal_table = E_CALENDAR_TABLE (object);
if (cal_table->model) {
g_object_unref (cal_table->model);
cal_table->model = NULL;
}
GTK_OBJECT_CLASS (e_calendar_table_parent_class)->destroy (object);
}
/**
* e_calendar_table_get_table:
* @cal_table: A calendar table.
*
* Queries the #ETable widget that the calendar table is using.
*
* Return value: The #ETable widget that the calendar table uses to display its
* data.
**/
ETable *
e_calendar_table_get_table (ECalendarTable *cal_table)
{
g_return_val_if_fail (cal_table != NULL, NULL);
g_return_val_if_fail (E_IS_CALENDAR_TABLE (cal_table), NULL);
return e_table_scrolled_get_table (E_TABLE_SCROLLED (cal_table->etable));
}
void
e_calendar_table_open_selected (ECalendarTable *cal_table)
{
ECalModelComponent *comp_data;
icalproperty *prop;
comp_data = e_calendar_table_get_selected_comp (cal_table);
prop = icalcomponent_get_first_property (comp_data->icalcomp, ICAL_ATTENDEE_PROPERTY);
if (comp_data != NULL)
e_calendar_table_open_task (cal_table, comp_data->client, comp_data->icalcomp, prop ? TRUE : FALSE);
}
/**
* e_calendar_table_complete_selected:
* @cal_table: A calendar table
*
* Marks the selected items as completed
**/
void
e_calendar_table_complete_selected (ECalendarTable *cal_table)
{
g_return_if_fail (cal_table != NULL);
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
mark_as_complete_cb (NULL, NULL, cal_table);
}
/* Used from e_table_selected_row_foreach(); puts the selected row number in an
* gint pointed to by the closure data.
*/
static void
get_selected_row_cb (gint model_row, gpointer data)
{
gint *row;
row = data;
*row = model_row;
}
/* Returns the component that is selected in the table; only works if there is
* one and only one selected row.
*/
ECalModelComponent *
e_calendar_table_get_selected_comp (ECalendarTable *cal_table)
{
ETable *etable;
gint row;
etable = e_table_scrolled_get_table (E_TABLE_SCROLLED (cal_table->etable));
if (e_table_selected_count (etable) != 1)
return NULL;
row = -1;
e_table_selected_row_foreach (etable,
get_selected_row_cb,
&row);
g_return_val_if_fail (row != -1, NULL);
return e_cal_model_get_component_at (cal_table->model, row);
}
struct get_selected_uids_closure {
ECalendarTable *cal_table;
GSList *objects;
};
/* Used from e_table_selected_row_foreach(), builds a list of the selected UIDs */
static void
add_uid_cb (gint model_row, gpointer data)
{
struct get_selected_uids_closure *closure;
ECalModelComponent *comp_data;
closure = data;
comp_data = e_cal_model_get_component_at (closure->cal_table->model, model_row);
closure->objects = g_slist_prepend (closure->objects, comp_data);
}
static GSList *
get_selected_objects (ECalendarTable *cal_table)
{
struct get_selected_uids_closure closure;
ETable *etable;
closure.cal_table = cal_table;
closure.objects = NULL;
etable = e_table_scrolled_get_table (E_TABLE_SCROLLED (cal_table->etable));
e_table_selected_row_foreach (etable, add_uid_cb, &closure);
return closure.objects;
}
/* Deletes all of the selected components in the table */
static void
delete_selected_components (ECalendarTable *cal_table)
{
GSList *objs, *l;
objs = get_selected_objects (cal_table);
e_calendar_table_set_status_message (cal_table, _("Deleting selected objects"), -1);
for (l = objs; l; l = l->next) {
ECalModelComponent *comp_data = (ECalModelComponent *) l->data;
GError *error = NULL;
e_cal_remove_object (comp_data->client,
icalcomponent_get_uid (comp_data->icalcomp), &error);
delete_error_dialog (error, E_CAL_COMPONENT_TODO);
g_clear_error (&error);
}
e_calendar_table_set_status_message (cal_table, NULL, -1);
g_slist_free (objs);
}
static void
add_retract_data (ECalComponent *comp, const gchar *retract_comment)
{
icalcomponent *icalcomp = NULL;
icalproperty *icalprop = NULL;
icalcomp = e_cal_component_get_icalcomponent (comp);
if (retract_comment && *retract_comment)
icalprop = icalproperty_new_x (retract_comment);
else
icalprop = icalproperty_new_x ("0");
icalproperty_set_x_name (icalprop, "X-EVOLUTION-RETRACT-COMMENT");
icalcomponent_add_property (icalcomp, icalprop);
}
static gboolean
check_for_retract (ECalComponent *comp, ECal *client)
{
ECalComponentOrganizer org;
gchar *email = NULL;
const gchar *strip = NULL;
gboolean ret_val = FALSE;
if (!(e_cal_component_has_attendees (comp) &&
e_cal_get_save_schedules (client)))
return ret_val;
e_cal_component_get_organizer (comp, &org);
strip = itip_strip_mailto (org.value);
if (e_cal_get_cal_address (client, &email, NULL) && !g_ascii_strcasecmp (email, strip)) {
ret_val = TRUE;
}
g_free (email);
return ret_val;
}
/**
* e_calendar_table_delete_selected:
* @cal_table: A calendar table.
*
* Deletes the selected components in the table; asks the user first.
**/
void
e_calendar_table_delete_selected (ECalendarTable *cal_table)
{
ETable *etable;
gint n_selected;
ECalModelComponent *comp_data;
ECalComponent *comp = NULL;
gboolean delete = FALSE;
GError *error = NULL;
g_return_if_fail (cal_table != NULL);
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
etable = e_table_scrolled_get_table (E_TABLE_SCROLLED (cal_table->etable));
n_selected = e_table_selected_count (etable);
if (n_selected <= 0)
return;
if (n_selected == 1)
comp_data = e_calendar_table_get_selected_comp (cal_table);
else
comp_data = NULL;
/* FIXME: this may be something other than a TODO component */
if (comp_data) {
comp = e_cal_component_new ();
e_cal_component_set_icalcomponent (comp, icalcomponent_new_clone (comp_data->icalcomp));
}
if ((n_selected == 1) && comp && check_for_retract (comp, comp_data->client)) {
gchar *retract_comment = NULL;
gboolean retract = FALSE;
delete = prompt_retract_dialog (comp, &retract_comment, GTK_WIDGET (cal_table), &retract);
if (retract) {
GList *users = NULL;
icalcomponent *icalcomp = NULL, *mod_comp = NULL;
add_retract_data (comp, retract_comment);
icalcomp = e_cal_component_get_icalcomponent (comp);
icalcomponent_set_method (icalcomp, ICAL_METHOD_CANCEL);
if (!e_cal_send_objects (comp_data->client, icalcomp, &users,
&mod_comp, &error)) {
delete_error_dialog (error, E_CAL_COMPONENT_TODO);
g_clear_error (&error);
error = NULL;
} else {
if (mod_comp)
icalcomponent_free (mod_comp);
if (users) {
g_list_foreach (users, (GFunc) g_free, NULL);
g_list_free (users);
}
}
}
} else {
delete = delete_component_dialog (comp, FALSE, n_selected, E_CAL_COMPONENT_TODO, GTK_WIDGET (cal_table));
}
if (delete)
delete_selected_components (cal_table);
/* free memory */
if (comp)
g_object_unref (comp);
}
/**
* e_calendar_table_get_selected:
* @cal_table:
*
* Get the currently selected ECalModelComponent's on the table.
*
* Return value: A GSList of the components, which should be
* g_slist_free'd when finished with.
**/
GSList *
e_calendar_table_get_selected (ECalendarTable *cal_table)
{
return get_selected_objects(cal_table);
}
/**
* e_calendar_table_cut_clipboard:
* @cal_table: A calendar table.
*
* Cuts selected tasks in the given calendar table
*/
void
e_calendar_table_cut_clipboard (ECalendarTable *cal_table)
{
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
e_calendar_table_copy_clipboard (cal_table);
delete_selected_components (cal_table);
}
static void
clipboard_get_calendar_cb (GtkClipboard *clipboard,
GtkSelectionData *selection_data,
guint info,
gpointer data)
{
gchar *comp_str = (gchar *) data;
switch (info) {
case TARGET_TYPE_VCALENDAR:
gtk_selection_data_set (selection_data,
gdk_atom_intern (target_types[info].target, FALSE), 8,
(const guchar *) comp_str,
(gint) strlen (comp_str));
break;
default:
break;
}
}
/* callback for e_table_selected_row_foreach */
static void
copy_row_cb (gint model_row, gpointer data)
{
ECalendarTable *cal_table;
ECalModelComponent *comp_data;
gchar *comp_str;
icalcomponent *child;
cal_table = E_CALENDAR_TABLE (data);
g_return_if_fail (cal_table->tmp_vcal != NULL);
comp_data = e_cal_model_get_component_at (cal_table->model, model_row);
if (!comp_data)
return;
/* add timezones to the VCALENDAR component */
e_cal_util_add_timezones_from_component (cal_table->tmp_vcal, comp_data->icalcomp);
/* add the new component to the VCALENDAR component */
comp_str = icalcomponent_as_ical_string_r (comp_data->icalcomp);
child = icalparser_parse_string (comp_str);
if (child) {
icalcomponent_add_component (cal_table->tmp_vcal,
icalcomponent_new_clone (child));
icalcomponent_free (child);
}
g_free (comp_str);
}
/**
* e_calendar_table_copy_clipboard:
* @cal_table: A calendar table.
*
* Copies selected tasks into the clipboard
*/
void
e_calendar_table_copy_clipboard (ECalendarTable *cal_table)
{
ETable *etable;
GtkClipboard *clipboard;
gchar *comp_str;
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
/* create temporary VCALENDAR object */
cal_table->tmp_vcal = e_cal_util_new_top_level ();
etable = e_table_scrolled_get_table (E_TABLE_SCROLLED (cal_table->etable));
e_table_selected_row_foreach (etable, copy_row_cb, cal_table);
comp_str = icalcomponent_as_ical_string_r (cal_table->tmp_vcal);
clipboard = gtk_widget_get_clipboard (GTK_WIDGET (cal_table), clipboard_atom);
if (!gtk_clipboard_set_with_data(clipboard, target_types, n_target_types,
clipboard_get_calendar_cb,
NULL, comp_str)) {
/* no-op */
} else {
gtk_clipboard_set_can_store (clipboard, target_types + 1, n_target_types - 1);
}
/* free memory */
icalcomponent_free (cal_table->tmp_vcal);
g_free (comp_str);
cal_table->tmp_vcal = NULL;
}
static void
clipboard_get_calendar_data (ECalendarTable *cal_table, const gchar *text)
{
icalcomponent *icalcomp;
gchar *uid;
ECalComponent *comp;
ECal *client;
icalcomponent_kind kind;
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
if (!text || !*text)
return;
icalcomp = icalparser_parse_string (text);
if (!icalcomp)
return;
/* check the type of the component */
kind = icalcomponent_isa (icalcomp);
if (kind != ICAL_VCALENDAR_COMPONENT &&
kind != ICAL_VEVENT_COMPONENT &&
kind != ICAL_VTODO_COMPONENT &&
kind != ICAL_VJOURNAL_COMPONENT) {
return;
}
client = e_cal_model_get_default_client (cal_table->model);
e_calendar_table_set_status_message (cal_table, _("Updating objects"), -1);
if (kind == ICAL_VCALENDAR_COMPONENT) {
icalcomponent_kind child_kind;
icalcomponent *subcomp;
icalcomponent *vcal_comp;
vcal_comp = icalcomp;
subcomp = icalcomponent_get_first_component (
vcal_comp, ICAL_ANY_COMPONENT);
while (subcomp) {
child_kind = icalcomponent_isa (subcomp);
if (child_kind == ICAL_VEVENT_COMPONENT ||
child_kind == ICAL_VTODO_COMPONENT ||
child_kind == ICAL_VJOURNAL_COMPONENT) {
ECalComponent *tmp_comp;
uid = e_cal_component_gen_uid ();
tmp_comp = e_cal_component_new ();
e_cal_component_set_icalcomponent (
tmp_comp, icalcomponent_new_clone (subcomp));
e_cal_component_set_uid (tmp_comp, uid);
free (uid);
/* FIXME should we convert start/due/complete times? */
/* FIXME Error handling */
e_cal_create_object (client, e_cal_component_get_icalcomponent (tmp_comp), NULL, NULL);
g_object_unref (tmp_comp);
}
subcomp = icalcomponent_get_next_component (
vcal_comp, ICAL_ANY_COMPONENT);
}
} else {
comp = e_cal_component_new ();
e_cal_component_set_icalcomponent (comp, icalcomp);
uid = e_cal_component_gen_uid ();
e_cal_component_set_uid (comp, (const gchar *) uid);
free (uid);
e_cal_create_object (client, e_cal_component_get_icalcomponent (comp), NULL, NULL);
g_object_unref (comp);
}
e_calendar_table_set_status_message (cal_table, NULL, -1);
}
static void
clipboard_paste_received_cb (GtkClipboard *clipboard,
GtkSelectionData *selection_data,
gpointer data)
{
ECalendarTable *cal_table = E_CALENDAR_TABLE (data);
ETable *e_table = e_table_scrolled_get_table (E_TABLE_SCROLLED (cal_table->etable));
GnomeCanvas *canvas = e_table->table_canvas;
GnomeCanvasItem *item = GNOME_CANVAS (canvas)->focused_item;
if (gtk_clipboard_wait_is_text_available (clipboard) &&
GTK_WIDGET_HAS_FOCUS (canvas) &&
E_IS_TABLE_ITEM (item) &&
E_TABLE_ITEM (item)->editing_col >= 0 &&
E_TABLE_ITEM (item)->editing_row >= 0) {
ETableItem *eti = E_TABLE_ITEM (item);
ECellView *cell_view = eti->cell_views[eti->editing_col];
e_cell_text_paste_clipboard (cell_view, eti->editing_col, eti->editing_row);
} else {
GdkAtom type = selection_data->type;
if (type == gdk_atom_intern (target_types[TARGET_TYPE_VCALENDAR].target, TRUE)) {
gchar *result = NULL;
result = g_strndup ((const gchar *) selection_data->data,
selection_data->length);
clipboard_get_calendar_data (cal_table, result);
g_free (result);
}
}
g_object_unref (cal_table);
}
/**
* e_calendar_table_paste_clipboard:
* @cal_table: A calendar table.
*
* Pastes tasks currently in the clipboard into the given calendar table
*/
void
e_calendar_table_paste_clipboard (ECalendarTable *cal_table)
{
GtkClipboard *clipboard;
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
clipboard = gtk_widget_get_clipboard (GTK_WIDGET (cal_table), clipboard_atom);
g_object_ref (cal_table);
gtk_clipboard_request_contents (clipboard,
gdk_atom_intern (target_types[0].target, FALSE),
clipboard_paste_received_cb, cal_table);
}
/* Opens a task in the task editor */
void
e_calendar_table_open_task (ECalendarTable *cal_table, ECal *client, icalcomponent *icalcomp, gboolean assign)
{
CompEditor *tedit;
const gchar *uid;
guint32 flags = 0;
uid = icalcomponent_get_uid (icalcomp);
tedit = e_comp_editor_registry_find (comp_editor_registry, uid);
if (tedit == NULL) {
ECalComponent *comp;
comp = e_cal_component_new ();
e_cal_component_set_icalcomponent (comp, icalcomponent_new_clone (icalcomp));
if (assign) {
flags |= COMP_EDITOR_IS_ASSIGNED;
if (itip_organizer_is_user (comp, client) ||
!e_cal_component_has_attendees (comp))
flags |= COMP_EDITOR_USER_ORG;
}
tedit = task_editor_new (client, flags);
comp_editor_edit_comp (tedit, comp);
g_object_unref (comp);
if (flags & COMP_EDITOR_IS_ASSIGNED)
task_editor_show_assignment (TASK_EDITOR (tedit));
e_comp_editor_registry_add (comp_editor_registry, tedit, FALSE);
}
gtk_window_present (GTK_WINDOW (tedit));
}
/* Opens the task in the specified row */
static void
open_task_by_row (ECalendarTable *cal_table, gint row)
{
ECalModelComponent *comp_data;
icalproperty *prop;
comp_data = e_cal_model_get_component_at (cal_table->model, row);
prop = icalcomponent_get_first_property (comp_data->icalcomp, ICAL_ATTENDEE_PROPERTY);
e_calendar_table_open_task (cal_table, comp_data->client, comp_data->icalcomp, prop ? TRUE : FALSE);
}
static void
e_calendar_table_on_double_click (ETable *table,
gint row,
gint col,
GdkEvent *event,
ECalendarTable *cal_table)
{
open_task_by_row (cal_table, row);
}
/* popup menu callbacks */
static void
e_calendar_table_on_open_task (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
ECalModelComponent *comp_data;
icalproperty *prop;
comp_data = e_calendar_table_get_selected_comp (cal_table);
if (!comp_data)
return;
prop = icalcomponent_get_first_property (comp_data->icalcomp, ICAL_ATTENDEE_PROPERTY);
e_calendar_table_open_task (cal_table, comp_data->client, comp_data->icalcomp, prop ? TRUE : FALSE);
}
static void
e_calendar_table_on_save_as (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
ECalModelComponent *comp_data;
gchar *filename;
gchar *ical_string;
comp_data = e_calendar_table_get_selected_comp (cal_table);
if (comp_data == NULL)
return;
filename = e_file_dialog_save (_("Save as..."), NULL);
if (filename == NULL)
return;
ical_string = e_cal_get_component_as_string (comp_data->client, comp_data->icalcomp);
if (ical_string == NULL) {
g_warning ("Couldn't convert item to a string");
return;
}
e_write_file_uri (filename, ical_string);
g_free (ical_string);
}
static void
e_calendar_table_on_print_task (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
ECalModelComponent *comp_data;
ECalComponent *comp;
comp_data = e_calendar_table_get_selected_comp (cal_table);
if (comp_data == NULL)
return;
comp = e_cal_component_new ();
e_cal_component_set_icalcomponent (comp, icalcomponent_new_clone (comp_data->icalcomp));
print_comp (comp, comp_data->client, GTK_PRINT_OPERATION_ACTION_PRINT_DIALOG);
g_object_unref (comp);
}
static void
e_calendar_table_on_cut (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
e_calendar_table_cut_clipboard (cal_table);
}
static void
e_calendar_table_on_copy (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
e_calendar_table_copy_clipboard (cal_table);
}
static void
e_calendar_table_on_paste (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
e_calendar_table_paste_clipboard (cal_table);
}
static void
e_calendar_table_on_assign (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
ECalModelComponent *comp_data;
comp_data = e_calendar_table_get_selected_comp (cal_table);
if (comp_data)
e_calendar_table_open_task (cal_table, comp_data->client, comp_data->icalcomp, TRUE);
}
static void
e_calendar_table_on_forward (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
ECalModelComponent *comp_data;
comp_data = e_calendar_table_get_selected_comp (cal_table);
if (comp_data) {
ECalComponent *comp;
comp = e_cal_component_new ();
e_cal_component_set_icalcomponent (comp, icalcomponent_new_clone (comp_data->icalcomp));
itip_send_comp (E_CAL_COMPONENT_METHOD_PUBLISH, comp, comp_data->client, NULL, NULL, NULL, TRUE, FALSE);
g_object_unref (comp);
}
}
struct AffectedComponents {
ECalendarTable *cal_table;
GSList *components; /* contains pointers to ECalModelComponent */
};
/**
* get_selected_components_cb
* Helper function to fill list of selected components in ECalendarTable.
* This function is called from e_table_selected_row_foreach.
**/
static void
get_selected_components_cb (gint model_row, gpointer data)
{
struct AffectedComponents *ac = (struct AffectedComponents *) data;
if (!ac || !ac->cal_table)
return;
ac->components = g_slist_prepend (ac->components, e_cal_model_get_component_at (E_CAL_MODEL (ac->cal_table->model), model_row));
}
/**
* do_for_selected_components
* Calls function func for all selected components in cal_table.
*
* @param cal_table Table with selected components of our interest
* @param func Function to be called on each selected component from cal_table.
* The first parameter of this function is a pointer to ECalModelComponent and
* the second parameter of this function is pointer to cal_table
**/
static void
do_for_selected_components (ECalendarTable *cal_table, GFunc func)
{
ETable *etable;
struct AffectedComponents ac;
g_return_if_fail (cal_table != NULL);
ac.cal_table = cal_table;
ac.components = NULL;
etable = e_table_scrolled_get_table (E_TABLE_SCROLLED (cal_table->etable));
e_table_selected_row_foreach (etable, get_selected_components_cb, &ac);
g_slist_foreach (ac.components, func, cal_table);
g_slist_free (ac.components);
}
/**
* mark_comp_complete_cb
* Function used in call to @ref do_for_selected_components to mark each component as complete
**/
static void
mark_comp_complete_cb (gpointer data, gpointer user_data)
{
ECalendarTable *cal_table;
ECalModelComponent *comp_data;
comp_data = (ECalModelComponent *) data;
cal_table = E_CALENDAR_TABLE (user_data);
e_cal_model_tasks_mark_comp_complete (E_CAL_MODEL_TASKS (cal_table->model), comp_data);
}
/**
* mark_comp_incomplete_cb
* Function used in call to @ref do_for_selected_components to mark each component as incomplete
**/
static void
mark_comp_incomplete_cb (gpointer data, gpointer user_data)
{
ECalendarTable *cal_table;
ECalModelComponent *comp_data;
comp_data = (ECalModelComponent *) data;
cal_table = E_CALENDAR_TABLE (user_data);
e_cal_model_tasks_mark_comp_incomplete (E_CAL_MODEL_TASKS (cal_table->model), comp_data);
}
/* Callback used for the "mark tasks as incomplete" menu item */
static void
mark_as_incomplete_cb (EPopup *ep, EPopupItem *pitem, gpointer data)
{
do_for_selected_components (data, mark_comp_incomplete_cb);
}
/* Callback used for the "mark tasks as complete" menu item */
static void
mark_as_complete_cb (EPopup *ep, EPopupItem *pitem, gpointer data)
{
do_for_selected_components (data, mark_comp_complete_cb);
}
/* Opens the URL of the task */
static void
open_url_cb (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
ECalModelComponent *comp_data;
icalproperty *prop;
comp_data = e_calendar_table_get_selected_comp (cal_table);
if (!comp_data)
return;
prop = icalcomponent_get_first_property (comp_data->icalcomp, ICAL_URL_PROPERTY);
if (!prop)
return;
/* FIXME Pass a parent window. */
e_show_uri (NULL, icalproperty_get_url (prop));
}
/* Opens a new task editor */
static void
on_new_task (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
ETasks *tasks = g_object_get_data (G_OBJECT (cal_table), "tasks");
if (!tasks)
return;
e_tasks_new_task (tasks);
}
/* Callback for the "delete tasks" menu item */
static void
delete_cb (EPopup *ep, EPopupItem *pitem, gpointer data)
{
ECalendarTable *cal_table = data;
e_calendar_table_delete_selected (cal_table);
}
static EPopupItem tasks_popup_items [] = {
{ E_POPUP_ITEM, (gchar *) "00.newtask", (gchar *) N_("New _Task"), on_new_task, NULL, (gchar *) "stock_task", 0, 0},
{ E_POPUP_BAR, (gchar *) "01.bar" },
{ E_POPUP_ITEM, (gchar *) "03.open", (gchar *) N_("_Open"), e_calendar_table_on_open_task, NULL, (gchar *) GTK_STOCK_OPEN, E_CAL_POPUP_SELECT_ONE },
{ E_POPUP_ITEM, (gchar *) "05.openweb", (gchar *) N_("Open _Web Page"), open_url_cb, NULL, NULL, E_CAL_POPUP_SELECT_ONE, E_CAL_POPUP_SELECT_HASURL },
{ E_POPUP_ITEM, (gchar *) "10.saveas", (gchar *) N_("_Save As..."), e_calendar_table_on_save_as, NULL, (gchar *) GTK_STOCK_SAVE_AS, E_CAL_POPUP_SELECT_ONE },
{ E_POPUP_ITEM, (gchar *) "20.print", (gchar *) N_("P_rint..."), e_calendar_table_on_print_task, NULL, (gchar *) GTK_STOCK_PRINT, E_CAL_POPUP_SELECT_ONE },
{ E_POPUP_BAR, (gchar *) "30.bar" },
{ E_POPUP_ITEM, (gchar *) "40.cut", (gchar *) N_("C_ut"), e_calendar_table_on_cut, NULL, (gchar *) GTK_STOCK_CUT, 0, E_CAL_POPUP_SELECT_EDITABLE },
{ E_POPUP_ITEM, (gchar *) "50.copy", (gchar *) N_("_Copy"), e_calendar_table_on_copy, NULL, (gchar *) GTK_STOCK_COPY, 0, 0 },
{ E_POPUP_ITEM, (gchar *) "60.paste", (gchar *) N_("_Paste"), e_calendar_table_on_paste, NULL, (gchar *) GTK_STOCK_PASTE, 0, E_CAL_POPUP_SELECT_EDITABLE },
{ E_POPUP_BAR, (gchar *) "70.bar" },
{ E_POPUP_ITEM, (gchar *) "80.assign", (gchar *) N_("_Assign Task"), e_calendar_table_on_assign, NULL, NULL, E_CAL_POPUP_SELECT_ONE, E_CAL_POPUP_SELECT_EDITABLE|E_CAL_POPUP_SELECT_ASSIGNABLE },
{ E_POPUP_ITEM, (gchar *) "90.forward", (gchar *) N_("_Forward as iCalendar"), e_calendar_table_on_forward, NULL, (gchar *) "mail-forward", E_CAL_POPUP_SELECT_ONE },
{ E_POPUP_ITEM, (gchar *) "a0.markonecomplete", (gchar *) N_("_Mark as Complete"), mark_as_complete_cb, NULL, NULL, E_CAL_POPUP_SELECT_ONE, E_CAL_POPUP_SELECT_EDITABLE | E_CAL_POPUP_SELECT_NOTCOMPLETE},
{ E_POPUP_ITEM, (gchar *) "b0.markmanycomplete", (gchar *) N_("_Mark Selected Tasks as Complete"), mark_as_complete_cb, NULL, NULL, E_CAL_POPUP_SELECT_MANY, E_CAL_POPUP_SELECT_EDITABLE | E_CAL_POPUP_SELECT_NOTCOMPLETE },
{ E_POPUP_ITEM, (gchar *) "c0.markoneincomplete", (gchar *) N_("_Mark as Incomplete"), mark_as_incomplete_cb, NULL, NULL, E_CAL_POPUP_SELECT_ONE, E_CAL_POPUP_SELECT_EDITABLE|E_CAL_POPUP_SELECT_COMPLETE},
{ E_POPUP_ITEM, (gchar *) "d0.markmanyincomplete", (gchar *) N_("_Mark Selected Tasks as Incomplete"), mark_as_incomplete_cb, NULL, NULL, E_CAL_POPUP_SELECT_MANY, E_CAL_POPUP_SELECT_EDITABLE | E_CAL_POPUP_SELECT_COMPLETE },
{ E_POPUP_BAR, (gchar *) "e0.bar" },
{ E_POPUP_ITEM, (gchar *) "f0.delete", (gchar *) N_("_Delete"), delete_cb, NULL, (gchar *) GTK_STOCK_DELETE, E_CAL_POPUP_SELECT_ONE, E_CAL_POPUP_SELECT_EDITABLE },
{ E_POPUP_ITEM, (gchar *) "g0.deletemany", (gchar *) N_("_Delete Selected Tasks"), delete_cb, NULL, (gchar *) GTK_STOCK_DELETE, E_CAL_POPUP_SELECT_MANY, E_CAL_POPUP_SELECT_EDITABLE },
};
static void
ect_popup_free(EPopup *ep, GSList *items, gpointer data)
{
g_slist_free(items);
}
static gint
e_calendar_table_show_popup_menu (ETable *table,
GdkEvent *gdk_event,
ECalendarTable *cal_table)
{
GtkMenu *menu;
GSList *selection, *l, *menus = NULL;
GPtrArray *events;
ECalPopup *ep;
ECalPopupTargetSelect *t;
gint i;
selection = get_selected_objects (cal_table);
if (!selection)
return TRUE;
/** @HookPoint-ECalPopup: Tasks Table Context Menu
* @Id: org.gnome.evolution.tasks.table.popup
* @Class: org.gnome.evolution.calendar.popup:1.0
* @Target: ECalPopupTargetSelect
*
* The context menu on the tasks table.
*/
ep = e_cal_popup_new("org.gnome.evolution.tasks.table.popup");
events = g_ptr_array_new();
for (l=selection;l;l=g_slist_next(l))
g_ptr_array_add(events, e_cal_model_copy_component_data((ECalModelComponent *)l->data));
g_slist_free(selection);
t = e_cal_popup_target_new_select(ep, cal_table->model, events);
t->target.widget = (GtkWidget *)cal_table;
for (i=0;i<sizeof(tasks_popup_items)/sizeof(tasks_popup_items[0]);i++)
menus = g_slist_prepend(menus, &tasks_popup_items[i]);
e_popup_add_items((EPopup *)ep, menus, NULL, ect_popup_free, cal_table);
menu = e_popup_create_menu_once((EPopup *)ep, (EPopupTarget *)t, 0);
gtk_menu_popup(menu, NULL, NULL, NULL, NULL, gdk_event?gdk_event->button.button:0,
gdk_event?gdk_event->button.time:gtk_get_current_event_time());
return TRUE;
}
static gint
e_calendar_table_on_right_click (ETable *table,
gint row,
gint col,
GdkEvent *event,
ECalendarTable *cal_table)
{
return e_calendar_table_show_popup_menu (table, event, cal_table);
}
static gboolean
e_calendar_table_on_popup_menu (GtkWidget *widget, gpointer data)
{
ETable *table = E_TABLE(widget);
g_return_val_if_fail(table, FALSE);
return e_calendar_table_show_popup_menu (table, NULL,
E_CALENDAR_TABLE(data));
}
static gint
e_calendar_table_on_key_press (ETable *table,
gint row,
gint col,
GdkEventKey *event,
ECalendarTable *cal_table)
{
if (event->keyval == GDK_Delete) {
delete_cb (NULL, NULL, cal_table);
return TRUE;
} else if ((event->keyval == GDK_o)
&&(event->state & GDK_CONTROL_MASK)) {
open_task_by_row (cal_table, row);
return TRUE;
}
return FALSE;
}
static void
hide_completed_rows (ECalModel *model, GList *clients_list, gchar *hide_sexp, GPtrArray *comp_objects)
{
GList *l, *m, *objects;
ECal *client;
gint pos;
for (l = clients_list; l != NULL; l = l->next) {
client = l->data;
if (!e_cal_get_object_list (client, hide_sexp, &objects, NULL)) {
g_warning (G_STRLOC ": Could not get the objects");
continue;
}
for (m = objects; m; m = m->next) {
ECalModelComponent *comp_data;
ECalComponentId *id;
ECalComponent *comp = e_cal_component_new ();
e_cal_component_set_icalcomponent (comp, icalcomponent_new_clone (m->data));
id = e_cal_component_get_id (comp);
if ((comp_data = e_cal_model_get_component_for_uid (model, id))) {
e_table_model_pre_change (E_TABLE_MODEL (model));
pos = get_position_in_array (comp_objects, comp_data);
e_table_model_row_deleted (E_TABLE_MODEL (model), pos);
if (g_ptr_array_remove (comp_objects, comp_data))
e_cal_model_free_component_data (comp_data);
}
e_cal_component_free_id (id);
g_object_unref (comp);
}
g_list_foreach (objects, (GFunc) icalcomponent_free, NULL);
g_list_free (objects);
/* to notify about changes, because in call of row_deleted there are still all events */
e_table_model_changed (E_TABLE_MODEL (model));
}
}
static void
show_completed_rows (ECalModel *model, GList *clients_list, gchar *show_sexp, GPtrArray *comp_objects)
{
GList *l, *m, *objects;
ECal *client;
for (l = clients_list; l != NULL; l = l->next) {
client = l->data;
if (!e_cal_get_object_list (client, show_sexp, &objects, NULL)) {
g_warning (G_STRLOC ": Could not get the objects");
continue;
}
for (m = objects; m; m = m->next) {
ECalModelComponent *comp_data;
ECalComponentId *id;
ECalComponent *comp = e_cal_component_new ();
e_cal_component_set_icalcomponent (comp, icalcomponent_new_clone (m->data));
id = e_cal_component_get_id (comp);
if (!(e_cal_model_get_component_for_uid (model, id))) {
e_table_model_pre_change (E_TABLE_MODEL (model));
comp_data = g_new0 (ECalModelComponent, 1);
comp_data->client = g_object_ref (client);
comp_data->icalcomp = icalcomponent_new_clone (m->data);
e_cal_model_set_instance_times (comp_data,
e_cal_model_get_timezone (model));
comp_data->dtstart = comp_data->dtend = comp_data->due = comp_data->completed = NULL;
comp_data->color = NULL;
g_ptr_array_add (comp_objects, comp_data);
e_table_model_row_inserted (E_TABLE_MODEL (model), comp_objects->len - 1);
}
e_cal_component_free_id (id);
g_object_unref (comp);
}
}
}
/* Loads the state of the table (headers shown etc.) from the given file. */
void
e_calendar_table_load_state (ECalendarTable *cal_table,
gchar *filename)
{
struct stat st;
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
if (g_stat (filename, &st) == 0 && st.st_size > 0
&& S_ISREG (st.st_mode)) {
e_table_load_state (e_table_scrolled_get_table (E_TABLE_SCROLLED (cal_table->etable)), filename);
}
}
/* Saves the state of the table (headers shown etc.) to the given file. */
void
e_calendar_table_save_state (ECalendarTable *cal_table,
gchar *filename)
{
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
e_table_save_state (e_table_scrolled_get_table (E_TABLE_SCROLLED (cal_table->etable)),
filename);
}
/* Returns the current time, for the ECellDateEdit items.
FIXME: Should probably use the timezone of the item rather than the
current timezone, though that may be difficult to get from here. */
static struct tm
e_calendar_table_get_current_time (ECellDateEdit *ecde, gpointer data)
{
icaltimezone *zone;
struct tm tmp_tm = { 0 };
struct icaltimetype tt;
/* Get the current timezone. */
zone = calendar_config_get_icaltimezone ();
tt = icaltime_from_timet_with_zone (time (NULL), FALSE, zone);
/* Now copy it to the struct tm and return it. */
tmp_tm.tm_year = tt.year - 1900;
tmp_tm.tm_mon = tt.month - 1;
tmp_tm.tm_mday = tt.day;
tmp_tm.tm_hour = tt.hour;
tmp_tm.tm_min = tt.minute;
tmp_tm.tm_sec = tt.second;
tmp_tm.tm_isdst = -1;
return tmp_tm;
}
#ifdef TRANSLATORS_ONLY
static gchar *test[] = {
N_("Click to add a task")
};
#endif
void
e_calendar_table_set_activity_handler (ECalendarTable *cal_table, EActivityHandler *activity_handler)
{
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
cal_table->activity_handler = activity_handler;
}
void
e_calendar_table_set_status_message (ECalendarTable *cal_table, const gchar *message, gint percent)
{
g_return_if_fail (E_IS_CALENDAR_TABLE (cal_table));
if (!cal_table->activity_handler)
return;
if (!message || !*message) {
if (cal_table->activity_id != 0) {
e_activity_handler_operation_finished (cal_table->activity_handler, cal_table->activity_id);
cal_table->activity_id = 0;
}
} else if (cal_table->activity_id == 0) {
gchar *client_id = g_strdup_printf ("%p", (gpointer) cal_table);
cal_table->activity_id = e_activity_handler_operation_started (
cal_table->activity_handler, client_id, message, TRUE);
g_free (client_id);
} else {
double progress;
if (percent < 0)
progress = -1.0;
else {
progress = ((double) percent / 100);
}
e_activity_handler_operation_progressing (cal_table->activity_handler, cal_table->activity_id, message, progress);
}
}
/**
* e_calendar_table_hide_completed_tasks:
* @table: A calendar table model.
* @client_list: Clients List
*
* Hide completed tasks.
*/
void
e_calendar_table_process_completed_tasks (ECalendarTable *table, GList *clients_list, gboolean config_changed)
{
ECalModel *model;
static GMutex *mutex = NULL;
gchar *hide_sexp, *show_sexp;
GPtrArray *comp_objects = NULL;
if (!mutex)
mutex = g_mutex_new ();
g_mutex_lock (mutex);
model = e_calendar_table_get_model (table);
comp_objects = e_cal_model_get_object_array (model);
hide_sexp = calendar_config_get_hide_completed_tasks_sexp (TRUE);
show_sexp = calendar_config_get_hide_completed_tasks_sexp (FALSE);
/* If hide option is unchecked */
if (!(hide_sexp && show_sexp))
show_sexp = g_strdup ("(is-completed?)");
/* Delete rows from model*/
if (hide_sexp) {
hide_completed_rows (model, clients_list, hide_sexp, comp_objects);
}
/* Insert rows into model */
if (config_changed) {
show_completed_rows (model, clients_list, show_sexp, comp_objects);
}
g_free (hide_sexp);
g_free (show_sexp);
g_mutex_unlock (mutex);
}