1899 lines
49 KiB
C
1899 lines
49 KiB
C
/*
|
|
* 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)
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* ETaskTable - displays the ECalComponent objects in a table (an ETable).
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include "e-task-table.h"
|
|
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include <glib/gi18n.h>
|
|
#include <glib/gstdio.h>
|
|
#include <gtk/gtk.h>
|
|
#include <gdk/gdkkeysyms.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-view.h"
|
|
#include "e-cell-date-edit-text.h"
|
|
#include "print.h"
|
|
#include "misc.h"
|
|
|
|
#define E_TASK_TABLE_GET_PRIVATE(obj) \
|
|
(G_TYPE_INSTANCE_GET_PRIVATE \
|
|
((obj), E_TYPE_TASK_TABLE, ETaskTablePrivate))
|
|
|
|
struct _ETaskTablePrivate {
|
|
gpointer shell_view; /* weak pointer */
|
|
ECalModel *model;
|
|
GCancellable *completed_cancellable; /* when processing completed tasks */
|
|
|
|
GtkTargetList *copy_target_list;
|
|
GtkTargetList *paste_target_list;
|
|
};
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_COPY_TARGET_LIST,
|
|
PROP_MODEL,
|
|
PROP_PASTE_TARGET_LIST,
|
|
PROP_SHELL_VIEW
|
|
};
|
|
|
|
enum {
|
|
OPEN_COMPONENT,
|
|
POPUP_EVENT,
|
|
STATUS_MESSAGE,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static struct tm e_task_table_get_current_time (ECellDateEdit *ecde, gpointer data);
|
|
|
|
static guint signals[LAST_SIGNAL];
|
|
|
|
/* The icons to represent the task. */
|
|
static const gchar *icon_names[] = {
|
|
"stock_task",
|
|
"stock_task-recurring",
|
|
"stock_task-assigned",
|
|
"stock_task-assigned-to"
|
|
};
|
|
|
|
/* Forward Declarations */
|
|
static void e_task_table_selectable_init
|
|
(ESelectableInterface *interface);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (
|
|
ETaskTable,
|
|
e_task_table,
|
|
E_TYPE_TABLE,
|
|
G_IMPLEMENT_INTERFACE (
|
|
E_TYPE_SELECTABLE,
|
|
e_task_table_selectable_init))
|
|
|
|
static void
|
|
task_table_emit_open_component (ETaskTable *task_table,
|
|
ECalModelComponent *comp_data)
|
|
{
|
|
guint signal_id = signals[OPEN_COMPONENT];
|
|
|
|
g_signal_emit (task_table, signal_id, 0, comp_data);
|
|
}
|
|
|
|
static void
|
|
task_table_emit_popup_event (ETaskTable *task_table,
|
|
GdkEvent *event)
|
|
{
|
|
guint signal_id = signals[POPUP_EVENT];
|
|
|
|
g_signal_emit (task_table, signal_id, 0, event);
|
|
}
|
|
|
|
static void
|
|
task_table_emit_status_message (ETaskTable *task_table,
|
|
const gchar *message,
|
|
gdouble percent)
|
|
{
|
|
guint signal_id = signals[STATUS_MESSAGE];
|
|
|
|
g_signal_emit (task_table, signal_id, 0, message, percent);
|
|
}
|
|
|
|
static gint
|
|
task_table_percent_compare_cb (gconstpointer a,
|
|
gconstpointer b,
|
|
gpointer cmp_cache)
|
|
{
|
|
gint percent1 = GPOINTER_TO_INT (a);
|
|
gint percent2 = GPOINTER_TO_INT (b);
|
|
|
|
return (percent1 < percent2) ? -1 : (percent1 > percent2);
|
|
}
|
|
|
|
static gint
|
|
task_table_priority_compare_cb (gconstpointer a,
|
|
gconstpointer b,
|
|
gpointer cmp_cache)
|
|
{
|
|
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. */
|
|
return (priority1 < priority2) ? -1 : (priority1 > priority2);
|
|
}
|
|
|
|
static const gchar *
|
|
get_cache_str (gpointer cmp_cache,
|
|
const gchar *str)
|
|
{
|
|
const gchar *value;
|
|
|
|
if (!cmp_cache || !str)
|
|
return str;
|
|
|
|
value = e_table_sorting_utils_lookup_cmp_cache (cmp_cache, str);
|
|
if (!value) {
|
|
gchar *ckey;
|
|
|
|
ckey = g_utf8_collate_key (str, -1);
|
|
e_table_sorting_utils_add_to_cmp_cache (cmp_cache, (gchar *) str, ckey);
|
|
value = ckey;
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
static gboolean
|
|
same_cache_string (gpointer cmp_cache,
|
|
const gchar *str_a,
|
|
const gchar *str_b)
|
|
{
|
|
if (!cmp_cache)
|
|
return g_utf8_collate (str_a, str_b) == 0;
|
|
|
|
str_b = get_cache_str (cmp_cache, str_b);
|
|
|
|
g_return_val_if_fail (str_a != NULL, FALSE);
|
|
g_return_val_if_fail (str_b != NULL, FALSE);
|
|
|
|
return strcmp (str_a, str_b) == 0;
|
|
}
|
|
|
|
static gint
|
|
task_table_status_compare_cb (gconstpointer a,
|
|
gconstpointer b,
|
|
gpointer cmp_cache)
|
|
{
|
|
const gchar *string_a = a;
|
|
const gchar *string_b = b;
|
|
gint status_a = -2;
|
|
gint status_b = -2;
|
|
|
|
if (string_a == NULL || *string_a == '\0')
|
|
status_a = -1;
|
|
else {
|
|
const gchar *cache_str = get_cache_str (cmp_cache, string_a);
|
|
|
|
if (same_cache_string (cmp_cache, cache_str, _("Not Started")))
|
|
status_a = 0;
|
|
else if (same_cache_string (cmp_cache, cache_str, _("In Progress")))
|
|
status_a = 1;
|
|
else if (same_cache_string (cmp_cache, cache_str, _("Completed")))
|
|
status_a = 2;
|
|
else if (same_cache_string (cmp_cache, cache_str, _("Canceled")))
|
|
status_a = 3;
|
|
}
|
|
|
|
if (string_b == NULL || *string_b == '\0')
|
|
status_b = -1;
|
|
else {
|
|
const gchar *cache_str = get_cache_str (cmp_cache, string_b);
|
|
|
|
if (same_cache_string (cmp_cache, cache_str, _("Not Started")))
|
|
status_b = 0;
|
|
else if (same_cache_string (cmp_cache, cache_str, _("In Progress")))
|
|
status_b = 1;
|
|
else if (same_cache_string (cmp_cache, cache_str, _("Completed")))
|
|
status_b = 2;
|
|
else if (same_cache_string (cmp_cache, cache_str, _("Canceled")))
|
|
status_b = 3;
|
|
}
|
|
|
|
return (status_a < status_b) ? -1 : (status_a > status_b);
|
|
}
|
|
|
|
static void
|
|
task_table_model_cal_view_progress_cb (ETaskTable *task_table,
|
|
const gchar *message,
|
|
gint progress,
|
|
ECalClientSourceType type)
|
|
{
|
|
gdouble percent = (gdouble) progress;
|
|
|
|
task_table_emit_status_message (task_table, message, percent);
|
|
}
|
|
|
|
static void
|
|
task_table_model_cal_view_complete_cb (ETaskTable *task_table,
|
|
const GError *error,
|
|
ECalClientSourceType type)
|
|
{
|
|
task_table_emit_status_message (task_table, NULL, -1.0);
|
|
}
|
|
|
|
/* Deletes all of the selected components in the table */
|
|
static void
|
|
delete_selected_components (ETaskTable *task_table)
|
|
{
|
|
GSList *objs, *l;
|
|
const gchar *status_message;
|
|
|
|
objs = e_task_table_get_selected (task_table);
|
|
|
|
status_message = _("Deleting selected objects");
|
|
task_table_emit_status_message (task_table, status_message, -1.0);
|
|
|
|
for (l = objs; l; l = l->next) {
|
|
ECalModelComponent *comp_data = (ECalModelComponent *) l->data;
|
|
GError *error = NULL;
|
|
|
|
e_cal_client_remove_object_sync (
|
|
comp_data->client,
|
|
icalcomponent_get_uid (comp_data->icalcomp),
|
|
NULL, CALOBJ_MOD_THIS, NULL, &error);
|
|
delete_error_dialog (error, E_CAL_COMPONENT_TODO);
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
task_table_emit_status_message (task_table, NULL, -1.0);
|
|
|
|
g_slist_free (objs);
|
|
}
|
|
|
|
static void
|
|
task_table_queue_draw_cb (ECalModelTasks *tasks_model,
|
|
GParamSpec *param,
|
|
GtkWidget *task_table)
|
|
{
|
|
g_return_if_fail (task_table != NULL);
|
|
|
|
gtk_widget_queue_draw (task_table);
|
|
}
|
|
|
|
static void
|
|
task_table_set_model (ETaskTable *task_table,
|
|
ECalModel *model)
|
|
{
|
|
g_return_if_fail (task_table->priv->model == NULL);
|
|
|
|
task_table->priv->model = g_object_ref (model);
|
|
|
|
g_signal_connect_swapped (
|
|
model, "cal-view-progress",
|
|
G_CALLBACK (task_table_model_cal_view_progress_cb),
|
|
task_table);
|
|
|
|
g_signal_connect_swapped (
|
|
model, "cal-view-complete",
|
|
G_CALLBACK (task_table_model_cal_view_complete_cb),
|
|
task_table);
|
|
|
|
/* redraw on drawing options change */
|
|
g_signal_connect (
|
|
model, "notify::highlight-due-today",
|
|
G_CALLBACK (task_table_queue_draw_cb),
|
|
task_table);
|
|
|
|
g_signal_connect (
|
|
model, "notify::color-due-today",
|
|
G_CALLBACK (task_table_queue_draw_cb),
|
|
task_table);
|
|
|
|
g_signal_connect (
|
|
model, "notify::highlight-overdue",
|
|
G_CALLBACK (task_table_queue_draw_cb),
|
|
task_table);
|
|
|
|
g_signal_connect (
|
|
model, "notify::color-overdue",
|
|
G_CALLBACK (task_table_queue_draw_cb),
|
|
task_table);
|
|
|
|
}
|
|
|
|
static void
|
|
task_table_set_shell_view (ETaskTable *task_table,
|
|
EShellView *shell_view)
|
|
{
|
|
g_return_if_fail (task_table->priv->shell_view == NULL);
|
|
|
|
task_table->priv->shell_view = shell_view;
|
|
|
|
g_object_add_weak_pointer (
|
|
G_OBJECT (shell_view),
|
|
&task_table->priv->shell_view);
|
|
}
|
|
|
|
static void
|
|
task_table_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id) {
|
|
case PROP_MODEL:
|
|
task_table_set_model (
|
|
E_TASK_TABLE (object),
|
|
g_value_get_object (value));
|
|
return;
|
|
|
|
case PROP_SHELL_VIEW:
|
|
task_table_set_shell_view (
|
|
E_TASK_TABLE (object),
|
|
g_value_get_object (value));
|
|
return;
|
|
}
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
}
|
|
|
|
static void
|
|
task_table_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id) {
|
|
case PROP_COPY_TARGET_LIST:
|
|
g_value_set_boxed (
|
|
value, e_task_table_get_copy_target_list (
|
|
E_TASK_TABLE (object)));
|
|
return;
|
|
|
|
case PROP_MODEL:
|
|
g_value_set_object (
|
|
value, e_task_table_get_model (
|
|
E_TASK_TABLE (object)));
|
|
return;
|
|
|
|
case PROP_PASTE_TARGET_LIST:
|
|
g_value_set_boxed (
|
|
value, e_task_table_get_paste_target_list (
|
|
E_TASK_TABLE (object)));
|
|
return;
|
|
|
|
case PROP_SHELL_VIEW:
|
|
g_value_set_object (
|
|
value, e_task_table_get_shell_view (
|
|
E_TASK_TABLE (object)));
|
|
return;
|
|
}
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
}
|
|
|
|
static void
|
|
task_table_dispose (GObject *object)
|
|
{
|
|
ETaskTablePrivate *priv;
|
|
|
|
priv = E_TASK_TABLE_GET_PRIVATE (object);
|
|
|
|
if (priv->completed_cancellable) {
|
|
g_cancellable_cancel (priv->completed_cancellable);
|
|
g_object_unref (priv->completed_cancellable);
|
|
priv->completed_cancellable = NULL;
|
|
}
|
|
|
|
if (priv->shell_view != NULL) {
|
|
g_object_remove_weak_pointer (
|
|
G_OBJECT (priv->shell_view), &priv->shell_view);
|
|
priv->shell_view = NULL;
|
|
}
|
|
|
|
if (priv->model != NULL) {
|
|
g_signal_handlers_disconnect_by_func (priv->model, task_table_queue_draw_cb, object);
|
|
g_object_unref (priv->model);
|
|
priv->model = NULL;
|
|
}
|
|
|
|
if (priv->copy_target_list != NULL) {
|
|
gtk_target_list_unref (priv->copy_target_list);
|
|
priv->copy_target_list = NULL;
|
|
}
|
|
|
|
if (priv->paste_target_list != NULL) {
|
|
gtk_target_list_unref (priv->paste_target_list);
|
|
priv->paste_target_list = NULL;
|
|
}
|
|
|
|
/* Chain up to parent's dispose() method. */
|
|
G_OBJECT_CLASS (e_task_table_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
task_table_constructed (GObject *object)
|
|
{
|
|
ETaskTable *task_table;
|
|
ECalModel *model;
|
|
ECell *cell, *popup_cell;
|
|
ETableExtras *extras;
|
|
GList *strings;
|
|
AtkObject *a11y;
|
|
gchar *etspecfile;
|
|
gint percent;
|
|
|
|
task_table = E_TASK_TABLE (object);
|
|
model = e_task_table_get_model (task_table);
|
|
|
|
/* Create the header columns */
|
|
|
|
extras = e_table_extras_new ();
|
|
|
|
/*
|
|
* Normal string fields.
|
|
*/
|
|
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
|
|
g_object_set (
|
|
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);
|
|
g_object_unref (cell);
|
|
|
|
/*
|
|
* Date fields.
|
|
*/
|
|
cell = e_cell_date_edit_text_new (NULL, GTK_JUSTIFY_LEFT);
|
|
g_object_set (
|
|
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);
|
|
|
|
g_object_bind_property (
|
|
model, "timezone",
|
|
cell, "timezone",
|
|
G_BINDING_BIDIRECTIONAL |
|
|
G_BINDING_SYNC_CREATE);
|
|
|
|
g_object_bind_property (
|
|
model, "use-24-hour-format",
|
|
cell, "use-24-hour-format",
|
|
G_BINDING_BIDIRECTIONAL |
|
|
G_BINDING_SYNC_CREATE);
|
|
|
|
popup_cell = e_cell_date_edit_new ();
|
|
e_cell_popup_set_child (E_CELL_POPUP (popup_cell), cell);
|
|
g_object_unref (cell);
|
|
|
|
g_object_bind_property (
|
|
model, "use-24-hour-format",
|
|
popup_cell, "use-24-hour-format",
|
|
G_BINDING_BIDIRECTIONAL |
|
|
G_BINDING_SYNC_CREATE);
|
|
|
|
e_table_extras_add_cell (extras, "dateedit", popup_cell);
|
|
g_object_unref (popup_cell);
|
|
|
|
task_table->dates_cell = E_CELL_DATE_EDIT (popup_cell);
|
|
|
|
e_cell_date_edit_set_get_time_callback (
|
|
E_CELL_DATE_EDIT (popup_cell),
|
|
e_task_table_get_current_time, task_table, NULL);
|
|
|
|
/*
|
|
* Combo fields.
|
|
*/
|
|
|
|
/* Classification field. */
|
|
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
|
|
g_object_set (
|
|
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);
|
|
g_list_free (strings);
|
|
|
|
e_table_extras_add_cell (extras, "classification", popup_cell);
|
|
g_object_unref (popup_cell);
|
|
|
|
/* Priority field. */
|
|
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
|
|
g_object_set (
|
|
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);
|
|
g_list_free (strings);
|
|
|
|
e_table_extras_add_cell (extras, "priority", popup_cell);
|
|
g_object_unref (popup_cell);
|
|
|
|
/* Percent field. */
|
|
cell = e_cell_percent_new (NULL, GTK_JUSTIFY_LEFT);
|
|
g_object_set (
|
|
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;
|
|
for (percent = 0; percent <= 100; percent += 10) {
|
|
/* Translators: "%d%%" is the percentage of a task done.
|
|
* %d is the actual value, %% is replaced with a percent sign.
|
|
* Result values will be 0%, 10%, 20%, ... 100%
|
|
*/
|
|
strings = g_list_append (strings, g_strdup_printf (_("%d%%"), percent));
|
|
}
|
|
e_cell_combo_set_popdown_strings (
|
|
E_CELL_COMBO (popup_cell),
|
|
strings);
|
|
|
|
g_list_foreach (strings, (GFunc) g_free, NULL);
|
|
g_list_free (strings);
|
|
|
|
e_table_extras_add_cell (extras, "percent", popup_cell);
|
|
g_object_unref (popup_cell);
|
|
|
|
/* Transparency field. */
|
|
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
|
|
g_object_set (
|
|
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);
|
|
g_list_free (strings);
|
|
|
|
e_table_extras_add_cell (extras, "transparency", popup_cell);
|
|
g_object_unref (popup_cell);
|
|
|
|
/* Status field. */
|
|
cell = e_cell_text_new (NULL, GTK_JUSTIFY_LEFT);
|
|
g_object_set (
|
|
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);
|
|
g_list_free (strings);
|
|
|
|
e_table_extras_add_cell (extras, "calstatus", popup_cell);
|
|
g_object_unref (popup_cell);
|
|
|
|
e_table_extras_add_compare (
|
|
extras, "date-compare",
|
|
e_cell_date_edit_compare_cb);
|
|
e_table_extras_add_compare (
|
|
extras, "percent-compare",
|
|
task_table_percent_compare_cb);
|
|
e_table_extras_add_compare (
|
|
extras, "priority-compare",
|
|
task_table_priority_compare_cb);
|
|
e_table_extras_add_compare (
|
|
extras, "status-compare",
|
|
task_table_status_compare_cb);
|
|
|
|
/* Create pixmaps */
|
|
|
|
cell = e_cell_toggle_new (icon_names, G_N_ELEMENTS (icon_names));
|
|
e_table_extras_add_cell (extras, "icon", cell);
|
|
g_object_unref (cell);
|
|
|
|
e_table_extras_add_icon_name (extras, "icon", "stock_task");
|
|
|
|
e_table_extras_add_icon_name (extras, "complete", "stock_check-filled");
|
|
|
|
/* set proper format component for a default 'date' cell renderer */
|
|
cell = e_table_extras_get_cell (extras, "date");
|
|
e_cell_date_set_format_component (E_CELL_DATE (cell), "calendar");
|
|
|
|
/* Create the table */
|
|
|
|
etspecfile = g_build_filename (
|
|
EVOLUTION_ETSPECDIR, "e-calendar-table.etspec", NULL);
|
|
e_table_construct_from_spec_file (
|
|
E_TABLE (task_table), E_TABLE_MODEL (model),
|
|
extras, etspecfile, NULL);
|
|
g_free (etspecfile);
|
|
|
|
gtk_widget_set_has_tooltip (GTK_WIDGET (task_table), TRUE);
|
|
|
|
g_object_unref (extras);
|
|
|
|
a11y = gtk_widget_get_accessible (GTK_WIDGET (task_table));
|
|
if (a11y)
|
|
atk_object_set_name (a11y, _("Tasks"));
|
|
|
|
/* Chain up to parent's constructed() method. */
|
|
G_OBJECT_CLASS (e_task_table_parent_class)->constructed (object);
|
|
}
|
|
|
|
static gboolean
|
|
task_table_popup_menu (GtkWidget *widget)
|
|
{
|
|
ETaskTable *task_table;
|
|
|
|
task_table = E_TASK_TABLE (widget);
|
|
task_table_emit_popup_event (task_table, NULL);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
task_table_query_tooltip (GtkWidget *widget,
|
|
gint x,
|
|
gint y,
|
|
gboolean keyboard_mode,
|
|
GtkTooltip *tooltip)
|
|
{
|
|
ETaskTable *task_table;
|
|
ECalModel *model;
|
|
ECalModelComponent *comp_data;
|
|
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;
|
|
gboolean use_24_hour_format;
|
|
ECalComponent *new_comp;
|
|
ECalComponentOrganizer organizer;
|
|
ECalComponentDateTime dtstart, dtdue;
|
|
icalcomponent *clone;
|
|
icaltimezone *zone, *default_zone;
|
|
GSList *desc, *p;
|
|
gint len;
|
|
ESelectionModel *esm;
|
|
struct tm tmp_tm;
|
|
|
|
if (keyboard_mode)
|
|
return FALSE;
|
|
|
|
task_table = E_TASK_TABLE (widget);
|
|
|
|
e_table_get_mouse_over_cell (E_TABLE (task_table), &row, &col);
|
|
if (row == -1)
|
|
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 (E_TABLE (task_table));
|
|
if (esm && esm->sorter && e_sorter_needs_sorting (esm->sorter))
|
|
row = e_sorter_sorted_to_model (esm->sorter, row);
|
|
|
|
model = e_task_table_get_model (task_table);
|
|
comp_data = e_cal_model_get_component_at (model, row);
|
|
|
|
if (!comp_data || !comp_data->icalcomp)
|
|
return FALSE;
|
|
|
|
new_comp = e_cal_component_new ();
|
|
clone = icalcomponent_new_clone (comp_data->icalcomp);
|
|
if (!e_cal_component_set_icalcomponent (new_comp, clone)) {
|
|
g_object_unref (new_comp);
|
|
return FALSE;
|
|
}
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
|
|
str = e_calendar_view_get_icalcomponent_summary (
|
|
comp_data->client, comp_data->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_box_new (GTK_ORIENTATION_VERTICAL, 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
|
|
* "Organizer: NameOfTheUser <email@ofuser.com>" */
|
|
tmp = g_strdup_printf (_("Organizer: %s <%s>"), organizer.cn, ptr);
|
|
} else {
|
|
/* With SunOne accounts, there may be no ':' in
|
|
* organizer.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);
|
|
|
|
gtk_widget_modify_fg (l, GTK_STATE_NORMAL, &(style->text[GTK_STATE_NORMAL]));
|
|
}
|
|
|
|
e_cal_component_get_dtstart (new_comp, &dtstart);
|
|
e_cal_component_get_due (new_comp, &dtdue);
|
|
|
|
default_zone = e_cal_model_get_timezone (model);
|
|
use_24_hour_format = e_cal_model_get_use_24_hour_format (model);
|
|
|
|
if (dtstart.tzid) {
|
|
zone = icalcomponent_get_timezone (
|
|
e_cal_component_get_icalcomponent (new_comp),
|
|
dtstart.tzid);
|
|
if (!zone)
|
|
e_cal_client_get_timezone_sync (
|
|
comp_data->client, dtstart.tzid, &zone, NULL, NULL);
|
|
if (!zone)
|
|
zone = default_zone;
|
|
} else {
|
|
zone = NULL;
|
|
}
|
|
|
|
tmp2 = g_string_new ("");
|
|
|
|
if (dtstart.value) {
|
|
buff[0] = 0;
|
|
|
|
tmp_tm = icaltimetype_to_tm_with_zone (
|
|
dtstart.value, zone, default_zone);
|
|
e_time_format_date_and_time (
|
|
&tmp_tm, use_24_hour_format,
|
|
FALSE, FALSE, buff, 1000);
|
|
|
|
if (buff[0]) {
|
|
g_string_append (tmp2, _("Start: "));
|
|
g_string_append (tmp2, buff);
|
|
}
|
|
}
|
|
|
|
if (dtdue.value) {
|
|
buff[0] = 0;
|
|
|
|
tmp_tm = icaltimetype_to_tm_with_zone (
|
|
dtdue.value, zone, default_zone);
|
|
e_time_format_date_and_time (
|
|
&tmp_tm, use_24_hour_format,
|
|
FALSE, FALSE, 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);
|
|
|
|
gtk_widget_modify_fg (l, GTK_STATE_NORMAL, &(style->text[GTK_STATE_NORMAL]));
|
|
}
|
|
|
|
g_string_free (tmp2, TRUE);
|
|
|
|
e_cal_component_free_datetime (&dtstart);
|
|
e_cal_component_free_datetime (&dtdue);
|
|
|
|
tmp = e_cal_model_get_attendees_status_info (
|
|
model, new_comp, comp_data->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;
|
|
|
|
gtk_widget_modify_fg (l, GTK_STATE_NORMAL, &(style->text[GTK_STATE_NORMAL]));
|
|
}
|
|
|
|
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 (w), l, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_modify_fg (l, GTK_STATE_NORMAL, &(style->text[GTK_STATE_NORMAL]));
|
|
}
|
|
|
|
g_string_free (tmp2, TRUE);
|
|
|
|
gtk_widget_show_all (box);
|
|
gtk_tooltip_set_custom (tooltip, box);
|
|
|
|
g_object_unref (new_comp);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
task_table_double_click (ETable *table,
|
|
gint row,
|
|
gint col,
|
|
GdkEvent *event)
|
|
{
|
|
ETaskTable *task_table;
|
|
ECalModel *model;
|
|
ECalModelComponent *comp_data;
|
|
|
|
task_table = E_TASK_TABLE (table);
|
|
model = e_task_table_get_model (task_table);
|
|
comp_data = e_cal_model_get_component_at (model, row);
|
|
task_table_emit_open_component (task_table, comp_data);
|
|
}
|
|
|
|
static gint
|
|
task_table_right_click (ETable *table,
|
|
gint row,
|
|
gint col,
|
|
GdkEvent *event)
|
|
{
|
|
ETaskTable *task_table;
|
|
|
|
task_table = E_TASK_TABLE (table);
|
|
task_table_emit_popup_event (task_table, event);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
task_table_update_actions (ESelectable *selectable,
|
|
EFocusTracker *focus_tracker,
|
|
GdkAtom *clipboard_targets,
|
|
gint n_clipboard_targets)
|
|
{
|
|
ETaskTable *task_table;
|
|
GtkAction *action;
|
|
GtkTargetList *target_list;
|
|
GSList *list, *iter;
|
|
gboolean can_paste = FALSE;
|
|
gboolean sources_are_editable = TRUE;
|
|
gboolean sensitive;
|
|
const gchar *tooltip;
|
|
gint n_selected;
|
|
gint ii;
|
|
|
|
task_table = E_TASK_TABLE (selectable);
|
|
n_selected = e_table_selected_count (E_TABLE (task_table));
|
|
|
|
list = e_task_table_get_selected (task_table);
|
|
for (iter = list; iter != NULL && sources_are_editable; iter = iter->next) {
|
|
ECalModelComponent *comp_data = iter->data;
|
|
|
|
sources_are_editable = sources_are_editable &&
|
|
!e_client_is_readonly (E_CLIENT (comp_data->client));
|
|
}
|
|
g_slist_free (list);
|
|
|
|
target_list = e_selectable_get_paste_target_list (selectable);
|
|
for (ii = 0; ii < n_clipboard_targets && !can_paste; ii++)
|
|
can_paste = gtk_target_list_find (
|
|
target_list, clipboard_targets[ii], NULL);
|
|
|
|
action = e_focus_tracker_get_cut_clipboard_action (focus_tracker);
|
|
sensitive = (n_selected > 0) && sources_are_editable;
|
|
tooltip = _("Cut selected tasks to the clipboard");
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
gtk_action_set_tooltip (action, tooltip);
|
|
|
|
action = e_focus_tracker_get_copy_clipboard_action (focus_tracker);
|
|
sensitive = (n_selected > 0);
|
|
tooltip = _("Copy selected tasks to the clipboard");
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
gtk_action_set_tooltip (action, tooltip);
|
|
|
|
action = e_focus_tracker_get_paste_clipboard_action (focus_tracker);
|
|
sensitive = sources_are_editable && can_paste;
|
|
tooltip = _("Paste tasks from the clipboard");
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
gtk_action_set_tooltip (action, tooltip);
|
|
|
|
action = e_focus_tracker_get_delete_selection_action (focus_tracker);
|
|
sensitive = (n_selected > 0) && sources_are_editable;
|
|
tooltip = _("Delete selected tasks");
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
gtk_action_set_tooltip (action, tooltip);
|
|
|
|
action = e_focus_tracker_get_select_all_action (focus_tracker);
|
|
sensitive = TRUE;
|
|
tooltip = _("Select all visible tasks");
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
gtk_action_set_tooltip (action, tooltip);
|
|
}
|
|
|
|
static void
|
|
task_table_cut_clipboard (ESelectable *selectable)
|
|
{
|
|
ETaskTable *task_table;
|
|
|
|
task_table = E_TASK_TABLE (selectable);
|
|
|
|
e_selectable_copy_clipboard (selectable);
|
|
delete_selected_components (task_table);
|
|
}
|
|
|
|
/* Helper for task_table_copy_clipboard() */
|
|
static void
|
|
copy_row_cb (gint model_row,
|
|
gpointer data)
|
|
{
|
|
ETaskTable *task_table;
|
|
ECalModelComponent *comp_data;
|
|
ECalModel *model;
|
|
gchar *comp_str;
|
|
icalcomponent *child;
|
|
|
|
task_table = E_TASK_TABLE (data);
|
|
|
|
g_return_if_fail (task_table->tmp_vcal != NULL);
|
|
|
|
model = e_task_table_get_model (task_table);
|
|
comp_data = e_cal_model_get_component_at (model, model_row);
|
|
if (!comp_data)
|
|
return;
|
|
|
|
/* Add timezones to the VCALENDAR component. */
|
|
e_cal_util_add_timezones_from_component (
|
|
task_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 (
|
|
task_table->tmp_vcal,
|
|
icalcomponent_new_clone (child));
|
|
icalcomponent_free (child);
|
|
}
|
|
g_free (comp_str);
|
|
}
|
|
|
|
static void
|
|
task_table_copy_clipboard (ESelectable *selectable)
|
|
{
|
|
ETaskTable *task_table;
|
|
GtkClipboard *clipboard;
|
|
gchar *comp_str;
|
|
|
|
task_table = E_TASK_TABLE (selectable);
|
|
|
|
/* Create a temporary VCALENDAR object. */
|
|
task_table->tmp_vcal = e_cal_util_new_top_level ();
|
|
|
|
e_table_selected_row_foreach (
|
|
E_TABLE (task_table), copy_row_cb, task_table);
|
|
comp_str = icalcomponent_as_ical_string_r (task_table->tmp_vcal);
|
|
|
|
clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
|
|
e_clipboard_set_calendar (clipboard, comp_str, -1);
|
|
gtk_clipboard_store (clipboard);
|
|
|
|
g_free (comp_str);
|
|
|
|
icalcomponent_free (task_table->tmp_vcal);
|
|
task_table->tmp_vcal = NULL;
|
|
}
|
|
|
|
/* Helper for calenable_table_paste_clipboard() */
|
|
static void
|
|
clipboard_get_calendar_data (ETaskTable *task_table,
|
|
const gchar *text)
|
|
{
|
|
icalcomponent *icalcomp;
|
|
gchar *uid;
|
|
ECalComponent *comp;
|
|
ECalModel *model;
|
|
ECalClient *client;
|
|
icalcomponent_kind kind;
|
|
const gchar *status_message;
|
|
|
|
g_return_if_fail (E_IS_TASK_TABLE (task_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;
|
|
}
|
|
|
|
model = e_task_table_get_model (task_table);
|
|
client = e_cal_model_ref_default_client (model);
|
|
|
|
status_message = _("Updating objects");
|
|
task_table_emit_status_message (task_table, status_message, -1.0);
|
|
|
|
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;
|
|
GError *error = NULL;
|
|
|
|
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);
|
|
g_free (uid);
|
|
uid = NULL;
|
|
|
|
/* FIXME should we convert start/due/complete
|
|
* times? Also, need error handling. */
|
|
if (!e_cal_client_create_object_sync (client, e_cal_component_get_icalcomponent (tmp_comp), &uid, NULL, &error))
|
|
uid = NULL;
|
|
|
|
if (error != NULL) {
|
|
g_warning (
|
|
"%s: Failed to create object: %s",
|
|
G_STRFUNC, error->message);
|
|
g_error_free (error);
|
|
}
|
|
|
|
g_object_unref (tmp_comp);
|
|
g_free (uid);
|
|
}
|
|
subcomp = icalcomponent_get_next_component (
|
|
vcal_comp, ICAL_ANY_COMPONENT);
|
|
}
|
|
} else {
|
|
GError *error = NULL;
|
|
|
|
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);
|
|
g_free (uid);
|
|
uid = NULL;
|
|
|
|
if (!e_cal_client_create_object_sync (client, e_cal_component_get_icalcomponent (comp), &uid, NULL, &error))
|
|
uid = NULL;
|
|
|
|
if (error != NULL) {
|
|
g_warning (
|
|
"%s: Failed to create object: %s",
|
|
G_STRFUNC, error->message);
|
|
g_error_free (error);
|
|
}
|
|
|
|
g_object_unref (comp);
|
|
g_free (uid);
|
|
}
|
|
|
|
task_table_emit_status_message (task_table, NULL, -1.0);
|
|
|
|
g_object_unref (client);
|
|
}
|
|
|
|
static void
|
|
task_table_paste_clipboard (ESelectable *selectable)
|
|
{
|
|
ETaskTable *task_table;
|
|
GtkClipboard *clipboard;
|
|
GnomeCanvasItem *item;
|
|
GnomeCanvas *table_canvas;
|
|
|
|
task_table = E_TASK_TABLE (selectable);
|
|
|
|
clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
|
|
|
|
table_canvas = E_TABLE (task_table)->table_canvas;
|
|
item = table_canvas->focused_item;
|
|
|
|
/* XXX Should ECellText implement GtkEditable? */
|
|
|
|
/* Paste text into a cell being edited. */
|
|
if (gtk_clipboard_wait_is_text_available (clipboard) &&
|
|
gtk_widget_has_focus (GTK_WIDGET (table_canvas)) &&
|
|
E_IS_TABLE_ITEM (item) &&
|
|
E_TABLE_ITEM (item)->editing_col >= 0 &&
|
|
E_TABLE_ITEM (item)->editing_row >= 0) {
|
|
|
|
ETableItem *etable_item = E_TABLE_ITEM (item);
|
|
|
|
e_cell_text_paste_clipboard (
|
|
etable_item->cell_views[etable_item->editing_col],
|
|
etable_item->editing_col,
|
|
etable_item->editing_row);
|
|
|
|
/* Paste iCalendar data into the table. */
|
|
} else if (e_clipboard_wait_is_calendar_available (clipboard)) {
|
|
gchar *calendar_source;
|
|
|
|
calendar_source = e_clipboard_wait_for_calendar (clipboard);
|
|
clipboard_get_calendar_data (task_table, calendar_source);
|
|
g_free (calendar_source);
|
|
}
|
|
}
|
|
|
|
/* 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.
|
|
*/
|
|
static ECalModelComponent *
|
|
get_selected_comp (ETaskTable *task_table)
|
|
{
|
|
ECalModel *model;
|
|
gint row;
|
|
|
|
model = e_task_table_get_model (task_table);
|
|
if (e_table_selected_count (E_TABLE (task_table)) != 1)
|
|
return NULL;
|
|
|
|
row = -1;
|
|
e_table_selected_row_foreach (
|
|
E_TABLE (task_table), get_selected_row_cb, &row);
|
|
g_return_val_if_fail (row != -1, NULL);
|
|
|
|
return e_cal_model_get_component_at (model, row);
|
|
}
|
|
|
|
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,
|
|
ECalClient *client)
|
|
{
|
|
ECalComponentOrganizer org;
|
|
gchar *email = NULL;
|
|
const gchar *strip = NULL;
|
|
gboolean ret_val;
|
|
|
|
if (!e_cal_component_has_attendees (comp))
|
|
return FALSE;
|
|
|
|
if (!e_cal_client_check_save_schedules (client))
|
|
return FALSE;
|
|
|
|
e_cal_component_get_organizer (comp, &org);
|
|
strip = itip_strip_mailto (org.value);
|
|
|
|
ret_val = e_client_get_backend_property_sync (
|
|
E_CLIENT (client),
|
|
CAL_BACKEND_PROPERTY_CAL_EMAIL_ADDRESS,
|
|
&email, NULL, NULL) && email != NULL &&
|
|
g_ascii_strcasecmp (email, strip) == 0;
|
|
|
|
g_free (email);
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
static void
|
|
task_table_delete_selection (ESelectable *selectable)
|
|
{
|
|
ECalModel *model;
|
|
ETaskTable *task_table;
|
|
ECalModelComponent *comp_data;
|
|
ECalComponent *comp = NULL;
|
|
gboolean delete = TRUE;
|
|
gint n_selected;
|
|
GError *error = NULL;
|
|
|
|
task_table = E_TASK_TABLE (selectable);
|
|
model = e_task_table_get_model (task_table);
|
|
|
|
n_selected = e_table_selected_count (E_TABLE (task_table));
|
|
if (n_selected <= 0)
|
|
return;
|
|
|
|
if (n_selected == 1)
|
|
comp_data = get_selected_comp (task_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 (task_table), &retract);
|
|
if (retract) {
|
|
GSList *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_client_send_objects_sync (comp_data->client, icalcomp, &users, &mod_comp, NULL, &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_slist_foreach (users, (GFunc) g_free, NULL);
|
|
g_slist_free (users);
|
|
}
|
|
}
|
|
|
|
}
|
|
} else if (e_cal_model_get_confirm_delete (model))
|
|
delete = delete_component_dialog (
|
|
comp, FALSE, n_selected,
|
|
E_CAL_COMPONENT_TODO,
|
|
GTK_WIDGET (task_table));
|
|
|
|
if (delete)
|
|
delete_selected_components (task_table);
|
|
|
|
/* free memory */
|
|
if (comp)
|
|
g_object_unref (comp);
|
|
}
|
|
|
|
static void
|
|
task_table_select_all (ESelectable *selectable)
|
|
{
|
|
e_table_select_all (E_TABLE (selectable));
|
|
}
|
|
|
|
static void
|
|
e_task_table_class_init (ETaskTableClass *class)
|
|
{
|
|
GObjectClass *object_class;
|
|
GtkWidgetClass *widget_class;
|
|
ETableClass *table_class;
|
|
|
|
g_type_class_add_private (class, sizeof (ETaskTablePrivate));
|
|
|
|
object_class = G_OBJECT_CLASS (class);
|
|
object_class->set_property = task_table_set_property;
|
|
object_class->get_property = task_table_get_property;
|
|
object_class->dispose = task_table_dispose;
|
|
object_class->constructed = task_table_constructed;
|
|
|
|
widget_class = GTK_WIDGET_CLASS (class);
|
|
widget_class->popup_menu = task_table_popup_menu;
|
|
widget_class->query_tooltip = task_table_query_tooltip;
|
|
|
|
table_class = E_TABLE_CLASS (class);
|
|
table_class->double_click = task_table_double_click;
|
|
table_class->right_click = task_table_right_click;
|
|
|
|
/* Inherited from ESelectableInterface */
|
|
g_object_class_override_property (
|
|
object_class,
|
|
PROP_COPY_TARGET_LIST,
|
|
"copy-target-list");
|
|
|
|
g_object_class_install_property (
|
|
object_class,
|
|
PROP_MODEL,
|
|
g_param_spec_object (
|
|
"model",
|
|
"Model",
|
|
NULL,
|
|
E_TYPE_CAL_MODEL,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
/* Inherited from ESelectableInterface */
|
|
g_object_class_override_property (
|
|
object_class,
|
|
PROP_PASTE_TARGET_LIST,
|
|
"paste-target-list");
|
|
|
|
g_object_class_install_property (
|
|
object_class,
|
|
PROP_SHELL_VIEW,
|
|
g_param_spec_object (
|
|
"shell-view",
|
|
"Shell View",
|
|
NULL,
|
|
E_TYPE_SHELL_VIEW,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
signals[OPEN_COMPONENT] = g_signal_new (
|
|
"open-component",
|
|
G_TYPE_FROM_CLASS (class),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (ETaskTableClass, open_component),
|
|
NULL, NULL,
|
|
g_cclosure_marshal_VOID__OBJECT,
|
|
G_TYPE_NONE, 1,
|
|
E_TYPE_CAL_MODEL_COMPONENT);
|
|
|
|
signals[POPUP_EVENT] = g_signal_new (
|
|
"popup-event",
|
|
G_TYPE_FROM_CLASS (class),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (ETaskTableClass, popup_event),
|
|
NULL, NULL,
|
|
g_cclosure_marshal_VOID__BOXED,
|
|
G_TYPE_NONE, 1,
|
|
GDK_TYPE_EVENT | G_SIGNAL_TYPE_STATIC_SCOPE);
|
|
|
|
signals[STATUS_MESSAGE] = g_signal_new (
|
|
"status-message",
|
|
G_TYPE_FROM_CLASS (class),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (ETaskTableClass, status_message),
|
|
NULL, NULL,
|
|
e_marshal_VOID__STRING_DOUBLE,
|
|
G_TYPE_NONE, 2,
|
|
G_TYPE_STRING, G_TYPE_DOUBLE);
|
|
}
|
|
|
|
static void
|
|
e_task_table_init (ETaskTable *task_table)
|
|
{
|
|
GtkTargetList *target_list;
|
|
|
|
task_table->priv = E_TASK_TABLE_GET_PRIVATE (task_table);
|
|
|
|
task_table->priv->completed_cancellable = NULL;
|
|
|
|
target_list = gtk_target_list_new (NULL, 0);
|
|
e_target_list_add_calendar_targets (target_list, 0);
|
|
task_table->priv->copy_target_list = target_list;
|
|
|
|
target_list = gtk_target_list_new (NULL, 0);
|
|
e_target_list_add_calendar_targets (target_list, 0);
|
|
task_table->priv->paste_target_list = target_list;
|
|
}
|
|
|
|
static void
|
|
e_task_table_selectable_init (ESelectableInterface *interface)
|
|
{
|
|
interface->update_actions = task_table_update_actions;
|
|
interface->cut_clipboard = task_table_cut_clipboard;
|
|
interface->copy_clipboard = task_table_copy_clipboard;
|
|
interface->paste_clipboard = task_table_paste_clipboard;
|
|
interface->delete_selection = task_table_delete_selection;
|
|
interface->select_all = task_table_select_all;
|
|
}
|
|
|
|
/**
|
|
* e_task_table_new:
|
|
* @shell_view: an #EShellView
|
|
* @model: an #ECalModel for the table
|
|
*
|
|
* Returns a new #ETaskTable.
|
|
*
|
|
* Returns: a new #ETaskTable
|
|
**/
|
|
GtkWidget *
|
|
e_task_table_new (EShellView *shell_view,
|
|
ECalModel *model)
|
|
{
|
|
g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);
|
|
g_return_val_if_fail (E_IS_CAL_MODEL (model), NULL);
|
|
|
|
return g_object_new (
|
|
E_TYPE_TASK_TABLE,
|
|
"model", model, "shell-view", shell_view, NULL);
|
|
}
|
|
|
|
/**
|
|
* e_task_table_get_model:
|
|
* @task_table: A calendar table.
|
|
*
|
|
* Queries the calendar data model that a calendar table is using.
|
|
*
|
|
* Return value: A calendar model.
|
|
**/
|
|
ECalModel *
|
|
e_task_table_get_model (ETaskTable *task_table)
|
|
{
|
|
g_return_val_if_fail (E_IS_TASK_TABLE (task_table), NULL);
|
|
|
|
return task_table->priv->model;
|
|
}
|
|
|
|
EShellView *
|
|
e_task_table_get_shell_view (ETaskTable *task_table)
|
|
{
|
|
g_return_val_if_fail (E_IS_TASK_TABLE (task_table), NULL);
|
|
|
|
return task_table->priv->shell_view;
|
|
}
|
|
|
|
struct get_selected_uids_closure {
|
|
ETaskTable *task_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 = data;
|
|
ECalModelComponent *comp_data;
|
|
ECalModel *model;
|
|
|
|
model = e_task_table_get_model (closure->task_table);
|
|
comp_data = e_cal_model_get_component_at (model, model_row);
|
|
|
|
closure->objects = g_slist_prepend (closure->objects, comp_data);
|
|
}
|
|
|
|
/**
|
|
* e_task_table_get_selected:
|
|
* @task_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_task_table_get_selected (ETaskTable *task_table)
|
|
{
|
|
struct get_selected_uids_closure closure;
|
|
|
|
closure.task_table = task_table;
|
|
closure.objects = NULL;
|
|
|
|
e_table_selected_row_foreach (
|
|
E_TABLE (task_table), add_uid_cb, &closure);
|
|
|
|
return closure.objects;
|
|
}
|
|
|
|
GtkTargetList *
|
|
e_task_table_get_copy_target_list (ETaskTable *task_table)
|
|
{
|
|
g_return_val_if_fail (E_IS_TASK_TABLE (task_table), NULL);
|
|
|
|
return task_table->priv->copy_target_list;
|
|
}
|
|
|
|
GtkTargetList *
|
|
e_task_table_get_paste_target_list (ETaskTable *task_table)
|
|
{
|
|
g_return_val_if_fail (E_IS_TASK_TABLE (task_table), NULL);
|
|
|
|
return task_table->priv->paste_target_list;
|
|
}
|
|
|
|
static void
|
|
task_table_get_object_list_async (GList *clients_list,
|
|
const gchar *sexp,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer callback_data)
|
|
{
|
|
GList *l;
|
|
|
|
for (l = clients_list; l != NULL; l = l->next) {
|
|
ECalClient *client = l->data;
|
|
|
|
e_cal_client_get_object_list (
|
|
client, sexp, cancellable,
|
|
callback, callback_data);
|
|
}
|
|
}
|
|
|
|
static void
|
|
hide_completed_rows_ready (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
ECalModel *model = user_data;
|
|
GSList *m, *objects;
|
|
gboolean changed = FALSE;
|
|
gint pos;
|
|
GPtrArray *comp_objects;
|
|
GError *error = NULL;
|
|
|
|
e_cal_client_get_object_list_finish (
|
|
E_CAL_CLIENT (source_object), result, &objects, &error);
|
|
|
|
if (error != NULL) {
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED) &&
|
|
!g_error_matches (error, E_CLIENT_ERROR, E_CLIENT_ERROR_CANCELLED)) {
|
|
ESource *source = e_client_get_source (E_CLIENT (source_object));
|
|
|
|
g_debug (
|
|
"%s: Could not get the objects from '%s': %s",
|
|
G_STRFUNC,
|
|
e_source_get_display_name (source),
|
|
error->message);
|
|
}
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
|
|
comp_objects = e_cal_model_get_object_array (model);
|
|
g_return_if_fail (comp_objects != NULL);
|
|
|
|
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);
|
|
|
|
comp_data = e_cal_model_get_component_for_uid (model, id);
|
|
if (comp_data != NULL) {
|
|
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);
|
|
changed = TRUE;
|
|
|
|
if (g_ptr_array_remove (comp_objects, comp_data))
|
|
g_object_unref (comp_data);
|
|
}
|
|
e_cal_component_free_id (id);
|
|
g_object_unref (comp);
|
|
}
|
|
|
|
e_cal_client_free_icalcomp_slist (objects);
|
|
|
|
if (changed) {
|
|
/* 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_ready (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
ECalClient *client;
|
|
ECalModel *model = user_data;
|
|
GSList *m, *objects;
|
|
GPtrArray *comp_objects;
|
|
GError *error = NULL;
|
|
|
|
e_cal_client_get_object_list_finish (
|
|
E_CAL_CLIENT (source_object), result, &objects, &error);
|
|
|
|
if (error != NULL) {
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED) &&
|
|
!g_error_matches (error, E_CLIENT_ERROR, E_CLIENT_ERROR_CANCELLED)) {
|
|
ESource *source = e_client_get_source (E_CLIENT (source_object));
|
|
|
|
g_debug (
|
|
"%s: Could not get the objects from '%s': %s",
|
|
G_STRFUNC,
|
|
e_source_get_display_name (source),
|
|
error->message);
|
|
}
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
|
|
client = E_CAL_CLIENT (source_object);
|
|
g_return_if_fail (client != NULL);
|
|
|
|
comp_objects = e_cal_model_get_object_array (model);
|
|
g_return_if_fail (comp_objects != NULL);
|
|
|
|
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_object_new (
|
|
E_TYPE_CAL_MODEL_COMPONENT, NULL);
|
|
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 = NULL;
|
|
comp_data->dtend = NULL;
|
|
comp_data->due = NULL;
|
|
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);
|
|
}
|
|
|
|
e_cal_client_free_icalcomp_slist (objects);
|
|
}
|
|
|
|
/* 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_task_table_get_current_time (ECellDateEdit *ecde,
|
|
gpointer data)
|
|
{
|
|
ETaskTable *task_table = data;
|
|
ECalModel *model;
|
|
icaltimezone *zone;
|
|
struct tm tmp_tm = { 0 };
|
|
struct icaltimetype tt;
|
|
|
|
/* Get the current timezone. */
|
|
model = e_task_table_get_model (task_table);
|
|
zone = e_cal_model_get_timezone (model);
|
|
|
|
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;
|
|
}
|
|
|
|
/**
|
|
* e_task_table_process_completed_tasks:
|
|
* @table: A calendar table model.
|
|
*
|
|
* Process completed tasks.
|
|
*/
|
|
void
|
|
e_task_table_process_completed_tasks (ETaskTable *task_table,
|
|
gboolean config_changed)
|
|
{
|
|
ECalModel *model;
|
|
GList *client_list;
|
|
GCancellable *cancellable;
|
|
gchar *hide_sexp, *show_sexp;
|
|
|
|
if (task_table->priv->completed_cancellable) {
|
|
g_cancellable_cancel (task_table->priv->completed_cancellable);
|
|
g_object_unref (task_table->priv->completed_cancellable);
|
|
}
|
|
|
|
task_table->priv->completed_cancellable = g_cancellable_new ();
|
|
cancellable = task_table->priv->completed_cancellable;
|
|
|
|
model = e_task_table_get_model (task_table);
|
|
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?)");
|
|
|
|
client_list = e_cal_model_list_clients (model);
|
|
|
|
/* Delete rows from model */
|
|
if (hide_sexp) {
|
|
task_table_get_object_list_async (
|
|
client_list, hide_sexp, cancellable,
|
|
hide_completed_rows_ready, model);
|
|
}
|
|
|
|
/* Insert rows into model */
|
|
if (config_changed) {
|
|
task_table_get_object_list_async (
|
|
client_list, show_sexp, cancellable,
|
|
show_completed_rows_ready, model);
|
|
}
|
|
|
|
g_list_free_full (client_list, (GDestroyNotify) g_object_unref);
|
|
|
|
g_free (hide_sexp);
|
|
g_free (show_sexp);
|
|
}
|