diff options
Diffstat (limited to 'calendar/gui/dialogs/task-page.c')
-rw-r--r-- | calendar/gui/dialogs/task-page.c | 2951 |
1 files changed, 2351 insertions, 600 deletions
diff --git a/calendar/gui/dialogs/task-page.c b/calendar/gui/dialogs/task-page.c index 62732c9eb0..b1f68d3381 100644 --- a/calendar/gui/dialogs/task-page.c +++ b/calendar/gui/dialogs/task-page.c @@ -1,24 +1,28 @@ -/* Evolution calendar - Main page of the task editor dialog - * - * Copyright (C) 2001 Ximian, Inc. - * - * Authors: Federico Mena-Quintero <federico@ximian.com> - * Miguel de Icaza <miguel@ximian.com> - * Seth Alves <alves@hungry.com> - * JP Rosevear <jpr@ximian.com> +/* + * Evolution calendar - Main page of the task editor dialog * * This program is free software; you can redistribute it and/or - * modify it under the terms of version 2 of the GNU General Public - * License as published by the Free Software Foundation. + * 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 General Public License for more details. + * 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: + * Federico Mena-Quintero <federico@ximian.com> + * Miguel de Icaza <miguel@ximian.com> + * Seth Alves <alves@hungry.com> + * JP Rosevear <jpr@ximian.com> + * + * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA. */ #ifdef HAVE_CONFIG_H @@ -26,755 +30,1845 @@ #endif #include <string.h> -#include <gtk/gtksignal.h> -#include <gtk/gtktext.h> -#include <gtk/gtktogglebutton.h> -#include <gtk/gtkspinbutton.h> -#include <gtk/gtkoptionmenu.h> -#include <libgnome/gnome-defs.h> -#include <libgnome/gnome-i18n.h> -#include <glade/glade.h> -#include <gal/widgets/e-unicode.h> -#include <gal/widgets/e-categories.h> -#include <widgets/misc/e-dateedit.h> -#include "e-util/e-dialog-widgets.h" -#include "e-util/e-categories-config.h" +#include <gtk/gtk.h> +#include <glib/gi18n.h> +#include <gdk/gdkkeysyms.h> + +#include "../e-meeting-attendee.h" +#include "../e-meeting-list-view.h" +#include "../e-meeting-store.h" #include "../e-timezone-entry.h" -#include "../calendar-config.h" + +#include "comp-editor.h" #include "comp-editor-util.h" +#include "e-send-options-utils.h" #include "task-page.h" - +#define TASK_PAGE_GET_PRIVATE(obj) \ + (G_TYPE_INSTANCE_GET_PRIVATE \ + ((obj), TYPE_TASK_PAGE, TaskPagePrivate)) -/* Private part of the TaskPage structure */ struct _TaskPagePrivate { - /* Glade XML data */ - GladeXML *xml; + GtkBuilder *builder; - /* Widgets from the Glade file */ + /* Widgets from the UI file */ GtkWidget *main; + gchar **address_strings; + gchar *fallback_address; + EMeetingAttendee *ia; + gchar *user_add; + ECalComponent *comp; + + /* For meeting/event */ + GtkWidget *calendar_label; + GtkWidget *org_cal_label; + GtkWidget *attendee_box; + + /* Lists of attendees */ + GPtrArray *deleted_attendees; + + /* Generic informative messages placeholder */ + GtkWidget *info_hbox; + GtkWidget *info_icon; + GtkWidget *info_string; + GtkWidget *summary; + GtkWidget *summary_label; GtkWidget *due_date; GtkWidget *start_date; - GtkWidget *due_timezone; - GtkWidget *start_timezone; + GtkWidget *completed_date; + GtkWidget *timezone; + GtkWidget *timezone_label; - GtkWidget *description; + GtkWidget *status_combo; + GtkWidget *priority_combo; + GtkWidget *percent_complete; + GtkWidget *classification_combo; + GtkWidget *web_page_entry; - GtkWidget *classification_public; - GtkWidget *classification_private; - GtkWidget *classification_confidential; - - GtkWidget *contacts_btn; - GtkWidget *contacts_box; + GtkWidget *description; GtkWidget *categories_btn; GtkWidget *categories; - gboolean updating; + GtkWidget *client_combo_box; + + /* Meeting related items */ + GtkWidget *list_box; + GtkWidget *organizer_table; + GtkWidget *organizer; + GtkWidget *add; + GtkWidget *remove; + GtkWidget *edit; + GtkWidget *invite; + GtkWidget *attendees_label; + + /* ListView stuff */ + ECalClient *client; + EMeetingStore *meeting_store; + EMeetingListView *list_view; + gint row; + + /* For handling who the organizer is */ + gboolean user_org; + gboolean existing; + + gboolean sendoptions_shown; + gboolean is_assignment; + + ESendOptionsDialog *sod; - /* The Corba component for selecting contacts, and the entry field - which we place in the dialog. */ - GNOME_Evolution_Addressbook_SelectNames corba_select_names; - GtkWidget *contacts_entry; + GCancellable *connect_cancellable; }; -static const int classification_map[] = { - CAL_COMPONENT_CLASS_PUBLIC, - CAL_COMPONENT_CLASS_PRIVATE, - CAL_COMPONENT_CLASS_CONFIDENTIAL, +static const gint classification_map[] = { + E_CAL_COMPONENT_CLASS_PUBLIC, + E_CAL_COMPONENT_CLASS_PRIVATE, + E_CAL_COMPONENT_CLASS_CONFIDENTIAL, -1 }; - - -static void task_page_class_init (TaskPageClass *class); -static void task_page_init (TaskPage *tpage); -static void task_page_destroy (GtkObject *object); +/* Note that these two arrays must match. */ +static const gint status_map[] = { + ICAL_STATUS_NONE, + ICAL_STATUS_INPROCESS, + ICAL_STATUS_COMPLETED, + ICAL_STATUS_CANCELLED, + -1 +}; -static GtkWidget *task_page_get_widget (CompEditorPage *page); -static void task_page_focus_main_widget (CompEditorPage *page); -static void task_page_fill_widgets (CompEditorPage *page, CalComponent *comp); -static gboolean task_page_fill_component (CompEditorPage *page, CalComponent *comp); -static void task_page_set_summary (CompEditorPage *page, const char *summary); -static void task_page_set_dates (CompEditorPage *page, CompEditorPageDates *dates); +typedef enum { + PRIORITY_HIGH, + PRIORITY_NORMAL, + PRIORITY_LOW, + PRIORITY_UNDEFINED +} TaskEditorPriority; + +static const gint priority_map[] = { + PRIORITY_HIGH, + PRIORITY_NORMAL, + PRIORITY_LOW, + PRIORITY_UNDEFINED, + -1 +}; -static CompEditorPageClass *parent_class = NULL; +static gboolean task_page_fill_timezones (CompEditorPage *page, GHashTable *timezones); +static void task_page_select_organizer (TaskPage *tpage, const gchar *backend_address); +static void set_subscriber_info_string (TaskPage *tpage, const gchar *backend_address); - +G_DEFINE_TYPE (TaskPage, task_page, TYPE_COMP_EDITOR_PAGE) -/** - * task_page_get_type: - * - * Registers the #TaskPage class if necessary, and returns the type ID - * associated to it. - * - * Return value: The type ID of the #TaskPage class. - **/ -GtkType -task_page_get_type (void) +static TaskEditorPriority +priority_value_to_index (gint priority_value) { - static GtkType task_page_type; + TaskEditorPriority retval; + + if (priority_value == 0) + retval = PRIORITY_UNDEFINED; + else if (priority_value <= 4) + retval = PRIORITY_HIGH; + else if (priority_value == 5) + retval = PRIORITY_NORMAL; + else + retval = PRIORITY_LOW; + + return retval; +} - if (!task_page_type) { - static const GtkTypeInfo task_page_info = { - "TaskPage", - sizeof (TaskPage), - sizeof (TaskPageClass), - (GtkClassInitFunc) task_page_class_init, - (GtkObjectInitFunc) task_page_init, - NULL, /* reserved_1 */ - NULL, /* reserved_2 */ - (GtkClassInitFunc) NULL - }; +static gint +priority_index_to_value (TaskEditorPriority priority) +{ + gint retval; - task_page_type = gtk_type_unique (TYPE_COMP_EDITOR_PAGE, - &task_page_info); + switch (priority) { + case PRIORITY_UNDEFINED: + retval = 0; + break; + case PRIORITY_HIGH: + retval = 3; + break; + case PRIORITY_NORMAL: + retval = 5; + break; + case PRIORITY_LOW: + retval = 7; + break; + default: + retval = 0; + break; } - return task_page_type; + return retval; } -/* Class initialization function for the task page */ -static void -task_page_class_init (TaskPageClass *class) +static gboolean +get_current_identity (TaskPage *page, + gchar **name, + gchar **mailto) { - CompEditorPageClass *editor_page_class; - GtkObjectClass *object_class; + EShell *shell; + CompEditor *editor; + ESourceRegistry *registry; + GList *list, *iter; + GtkWidget *entry; + const gchar *extension_name; + const gchar *text; + gboolean match = FALSE; - editor_page_class = (CompEditorPageClass *) class; - object_class = (GtkObjectClass *) class; + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (page)); + shell = comp_editor_get_shell (editor); - parent_class = gtk_type_class (TYPE_COMP_EDITOR_PAGE); + entry = gtk_bin_get_child (GTK_BIN (page->priv->organizer)); + text = gtk_entry_get_text (GTK_ENTRY (entry)); - editor_page_class->get_widget = task_page_get_widget; - editor_page_class->focus_main_widget = task_page_focus_main_widget; - editor_page_class->fill_widgets = task_page_fill_widgets; - editor_page_class->fill_component = task_page_fill_component; - editor_page_class->set_summary = task_page_set_summary; - editor_page_class->set_dates = task_page_set_dates; + if (text == NULL || *text == '\0') + return FALSE; + + registry = e_shell_get_registry (shell); + extension_name = E_SOURCE_EXTENSION_MAIL_IDENTITY; + + list = e_source_registry_list_sources (registry, extension_name); + + for (iter = list; !match && iter != NULL; iter = g_list_next (iter)) { + ESource *source = E_SOURCE (iter->data); + ESourceMailIdentity *extension; + const gchar *id_name; + const gchar *id_address; + gchar *identity; + + extension = e_source_get_extension (source, extension_name); + + id_name = e_source_mail_identity_get_name (extension); + id_address = e_source_mail_identity_get_address (extension); - object_class->destroy = task_page_destroy; + if (id_name == NULL || id_address == NULL) + continue; + + identity = g_strdup_printf ("%s <%s>", id_name, id_address); + match = (g_ascii_strcasecmp (text, identity) == 0); + g_free (identity); + + if (match && name != NULL) + *name = g_strdup (id_name); + + if (match && mailto != NULL) + *mailto = g_strdup_printf ("MAILTO:%s", id_address); + } + + g_list_free_full (list, (GDestroyNotify) g_object_unref); + + return match; } -/* Object initialization function for the task page */ +/* Fills the widgets with default values */ static void -task_page_init (TaskPage *tpage) +clear_widgets (TaskPage *tpage) { - TaskPagePrivate *priv; + TaskPagePrivate *priv = tpage->priv; + CompEditor *editor; - priv = g_new0 (TaskPagePrivate, 1); - tpage->priv = priv; + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (tpage)); - priv->xml = NULL; + /* Summary, description */ + gtk_entry_set_text (GTK_ENTRY (priv->summary), ""); + gtk_text_buffer_set_text (gtk_text_view_get_buffer (GTK_TEXT_VIEW (priv->description)), "", 0); + e_buffer_tagger_update_tags (GTK_TEXT_VIEW (priv->description)); + + /* Start, due times - both set to None */ + e_date_edit_set_time (E_DATE_EDIT (priv->start_date), -1); + e_date_edit_set_time (E_DATE_EDIT (priv->due_date), -1); - priv->main = NULL; - priv->summary = NULL; - priv->due_date = NULL; - priv->start_date = NULL; - priv->due_timezone = NULL; - priv->start_timezone = NULL; - priv->description = NULL; - priv->classification_public = NULL; - priv->classification_private = NULL; - priv->classification_confidential = NULL; - priv->contacts_btn = NULL; - priv->contacts_box = NULL; - priv->categories_btn = NULL; - priv->categories = NULL; + /* Classification */ + comp_editor_set_classification (editor, E_CAL_COMPONENT_CLASS_PUBLIC); + + /* Categories */ + gtk_entry_set_text (GTK_ENTRY (priv->categories), ""); - priv->updating = FALSE; + e_date_edit_set_time (E_DATE_EDIT (priv->completed_date), -1); + e_dialog_combo_box_set (priv->status_combo, ICAL_STATUS_NONE, status_map); + e_dialog_combo_box_set (priv->priority_combo, PRIORITY_UNDEFINED, priority_map); + gtk_spin_button_set_value (GTK_SPIN_BUTTON (priv->percent_complete), 0); - priv->corba_select_names = CORBA_OBJECT_NIL; - priv->contacts_entry = NULL; + gtk_entry_set_text (GTK_ENTRY (priv->web_page_entry), ""); } -/* Destroy handler for the task page */ -static void -task_page_destroy (GtkObject *object) +static gboolean +date_in_past (TaskPage *tpage, + EDateEdit *date) +{ + struct icaltimetype tt = icaltime_null_time (); + + if (!e_date_edit_get_date (date, &tt.year, &tt.month, &tt.day)) + return FALSE; + + if (e_date_edit_get_time_of_day (date, &tt.hour, &tt.minute)) + tt.zone = e_timezone_entry_get_timezone (E_TIMEZONE_ENTRY (tpage->priv->timezone)); + else + tt.is_date = TRUE; + + return comp_editor_test_time_in_the_past (tt); +} + +/* returns whether changed info text */ +static gboolean +check_starts_in_the_past (TaskPage *tpage) { - TaskPage *tpage; TaskPagePrivate *priv; + gboolean start_in_past, due_in_past; - g_return_if_fail (object != NULL); - g_return_if_fail (IS_TASK_PAGE (object)); + if ((comp_editor_get_flags (comp_editor_page_get_editor (COMP_EDITOR_PAGE (tpage))) & COMP_EDITOR_NEW_ITEM) == 0) + return FALSE; - tpage = TASK_PAGE (object); priv = tpage->priv; + start_in_past = date_in_past (tpage, E_DATE_EDIT (priv->start_date)); + due_in_past = date_in_past (tpage, E_DATE_EDIT (priv->due_date)); + + if (start_in_past || due_in_past) { + gchar *tmp = g_strconcat ( + "<b>", start_in_past ? _("Task's start date is in the past") : "", + start_in_past && due_in_past ? "\n" : "", due_in_past ? _("Task's due date is in the past") : "", "</b>", NULL); + task_page_set_info_string (tpage, GTK_STOCK_DIALOG_WARNING, tmp); + g_free (tmp); + } else { + task_page_set_info_string (tpage, NULL, NULL); + } - if (priv->corba_select_names != CORBA_OBJECT_NIL) { - CORBA_Environment ev; + return TRUE; +} - CORBA_exception_init (&ev); - bonobo_object_release_unref (priv->corba_select_names, &ev); - CORBA_exception_free (&ev); +static void +sensitize_widgets (TaskPage *tpage) +{ + TaskPagePrivate *priv = tpage->priv; + CompEditor *editor; + CompEditorFlags flags; + ECalClient *client; + GtkActionGroup *action_group; + GtkAction *action; + gboolean read_only, sens = TRUE, sensitize; + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (tpage)); + client = comp_editor_get_client (editor); + flags = comp_editor_get_flags (editor); + + read_only = e_client_is_readonly (E_CLIENT (client)); + + if (flags & COMP_EDITOR_IS_ASSIGNED) + sens = flags & COMP_EDITOR_USER_ORG; + + sensitize = (!read_only && sens); + + if (read_only) { + gchar *tmp = g_strconcat ("<b>", _("Task cannot be edited, because the selected task list is read only"), "</b>", NULL); + task_page_set_info_string (tpage, GTK_STOCK_DIALOG_INFO, tmp); + g_free (tmp); + } else if (!sens) { + gchar *tmp = g_strconcat ("<b>", _("Task cannot be fully edited, because you are not the organizer"), "</b>", NULL); + task_page_set_info_string (tpage, GTK_STOCK_DIALOG_INFO, tmp); + g_free (tmp); + } else if ((flags & COMP_EDITOR_IS_ASSIGNED) != 0 && e_client_check_capability (E_CLIENT (client), CAL_STATIC_CAPABILITY_NO_TASK_ASSIGNMENT)) { + gchar *tmp = g_strconcat ("<b>", _("Task cannot be edited, because the selected task list does not support assigned tasks"), "</b>", NULL); + task_page_set_info_string (tpage, GTK_STOCK_DIALOG_INFO, tmp); + g_free (tmp); + sens = FALSE; + read_only = TRUE; + } else if (!check_starts_in_the_past (tpage)) { + task_page_set_info_string (tpage, NULL, NULL); } - if (priv->xml) { - gtk_object_unref (GTK_OBJECT (priv->xml)); - priv->xml = NULL; + /* The list of organizers is set to be non-editable. Otherwise any + * change in the displayed list causes an 'Account not found' error. + */ + gtk_editable_set_editable (GTK_EDITABLE (gtk_bin_get_child (GTK_BIN (priv->organizer))), FALSE); + + gtk_editable_set_editable (GTK_EDITABLE (priv->summary), !read_only); + gtk_widget_set_sensitive (priv->due_date, !read_only); + gtk_widget_set_sensitive (priv->start_date, !read_only); + gtk_widget_set_sensitive (priv->timezone, !read_only); + gtk_widget_set_sensitive (priv->description, !read_only); + gtk_widget_set_sensitive (priv->categories_btn, !read_only); + gtk_editable_set_editable (GTK_EDITABLE (priv->categories), !read_only); + + gtk_widget_set_sensitive (priv->completed_date, !read_only); + gtk_widget_set_sensitive (priv->status_combo, !read_only); + gtk_widget_set_sensitive (priv->priority_combo, !read_only); + gtk_widget_set_sensitive (priv->percent_complete, !read_only); + gtk_widget_set_sensitive (priv->classification_combo, !read_only); + + gtk_editable_set_editable ( + GTK_EDITABLE (priv->web_page_entry), !read_only); + + gtk_widget_set_sensitive (priv->organizer, !read_only); + gtk_widget_set_sensitive (priv->add, (!read_only && sens)); + gtk_widget_set_sensitive (priv->edit, (!read_only && sens)); + e_meeting_list_view_set_editable (priv->list_view, (!read_only && sens)); + gtk_widget_set_sensitive (priv->remove, (!read_only && sens)); + gtk_widget_set_sensitive (priv->invite, (!read_only && sens)); + gtk_widget_set_sensitive (GTK_WIDGET (priv->list_view), !read_only); + + action_group = comp_editor_get_action_group (editor, "editable"); + gtk_action_group_set_sensitive (action_group, !read_only); + + action_group = comp_editor_get_action_group (editor, "individual"); + gtk_action_group_set_sensitive (action_group, sensitize); + + action = comp_editor_get_action (editor, "send-options"); + gtk_action_set_sensitive (action, sensitize); + + if (!priv->is_assignment) { + gtk_widget_hide (priv->calendar_label); + gtk_widget_hide (priv->list_box); + gtk_widget_hide (priv->attendee_box); + gtk_widget_hide (priv->organizer); + gtk_widget_hide (priv->invite); + gtk_label_set_text_with_mnemonic (GTK_LABEL (priv->org_cal_label), _("_List:")); + gtk_label_set_mnemonic_widget (GTK_LABEL (priv->org_cal_label), priv->client_combo_box); + } else { + gtk_widget_show (priv->invite); + gtk_widget_show (priv->calendar_label); + gtk_widget_show (priv->list_box); + gtk_widget_show (priv->attendee_box); + gtk_widget_show (priv->organizer); + gtk_label_set_text_with_mnemonic (GTK_LABEL (priv->org_cal_label), _("Organi_zer:")); + gtk_label_set_mnemonic_widget (GTK_LABEL (priv->org_cal_label), priv->organizer); } +} - g_free (priv); - tpage->priv = NULL; +static void +set_attendees (ECalComponent *comp, + const GPtrArray *attendees) +{ + GSList *comp_attendees = NULL, *l; + gint i; - if (GTK_OBJECT_CLASS (parent_class)->destroy) - (* GTK_OBJECT_CLASS (parent_class)->destroy) (object); -} + for (i = 0; i < attendees->len; i++) { + EMeetingAttendee *ia = g_ptr_array_index (attendees, i); + ECalComponentAttendee *ca; - + ca = e_meeting_attendee_as_e_cal_component_attendee (ia); -/* get_widget handler for the task page */ -static GtkWidget * -task_page_get_widget (CompEditorPage *page) + comp_attendees = g_slist_prepend (comp_attendees, ca); + + } + comp_attendees = g_slist_reverse (comp_attendees); + + e_cal_component_set_attendee_list (comp, comp_attendees); + + for (l = comp_attendees; l != NULL; l = l->next) + g_free (l->data); + g_slist_free (comp_attendees); +} + +static void +organizer_changed_cb (GtkEntry *entry, + TaskPage *tpage) { - TaskPage *tpage; - TaskPagePrivate *priv; + gchar *name; + gchar *mailto; - tpage = TASK_PAGE (page); - priv = tpage->priv; + g_return_if_fail (GTK_IS_ENTRY (entry)); + g_return_if_fail (IS_TASK_PAGE (tpage)); - return priv->main; + if (!tpage->priv->ia) + return; + + if (!get_current_identity (tpage, &name, &mailto)) + return; + + /* XXX EMeetingAttendee takes ownership of the strings. */ + e_meeting_attendee_set_cn (tpage->priv->ia, name); + e_meeting_attendee_set_address (tpage->priv->ia, mailto); } -/* focus_main_widget handler for the task page */ static void -task_page_focus_main_widget (CompEditorPage *page) +task_page_dispose (GObject *object) { - TaskPage *tpage; TaskPagePrivate *priv; - tpage = TASK_PAGE (page); - priv = tpage->priv; + priv = TASK_PAGE_GET_PRIVATE (object); - gtk_widget_grab_focus (priv->summary); + if (priv->connect_cancellable != NULL) { + g_cancellable_cancel (priv->connect_cancellable); + g_object_unref (priv->connect_cancellable); + priv->connect_cancellable = NULL; + } + + if (priv->main != NULL) { + g_object_unref (priv->main); + priv->main = NULL; + } + + if (priv->builder != NULL) { + g_object_unref (priv->builder); + priv->builder = NULL; + } + + if (priv->sod != NULL) { + g_object_unref (priv->sod); + priv->sod = NULL; + } + + if (priv->comp != NULL) { + g_object_unref (priv->comp); + priv->comp = NULL; + } + + /* Chain up to parent's dispose() method. */ + G_OBJECT_CLASS (task_page_parent_class)->dispose (object); } -/* Fills the widgets with default values */ static void -clear_widgets (TaskPage *tpage) +task_page_finalize (GObject *object) { TaskPagePrivate *priv; - priv = tpage->priv; - - /* Summary, description */ - e_dialog_editable_set (priv->summary, NULL); - e_dialog_editable_set (priv->description, NULL); + priv = TASK_PAGE_GET_PRIVATE (object); - /* Start, due times */ - e_date_edit_set_time (E_DATE_EDIT (priv->start_date), 0); - e_date_edit_set_time (E_DATE_EDIT (priv->due_date), 0); + g_strfreev (priv->address_strings); + g_free (priv->fallback_address); - /* Classification */ - e_dialog_radio_set (priv->classification_public, - CAL_COMPONENT_CLASS_PRIVATE, classification_map); + g_ptr_array_foreach ( + priv->deleted_attendees, (GFunc) g_object_unref, NULL); + g_ptr_array_free (priv->deleted_attendees, TRUE); - /* Categories */ - e_dialog_editable_set (priv->categories, NULL); + /* Chain up to parent's finalize() method. */ + G_OBJECT_CLASS (task_page_parent_class)->finalize (object); } -/* Decode the radio button group for classifications */ -static CalComponentClassification -classification_get (GtkWidget *widget) +static GtkWidget * +task_page_get_widget (CompEditorPage *page) { - return e_dialog_radio_get (widget, classification_map); + TaskPage *tpage; + TaskPagePrivate *priv; + + tpage = TASK_PAGE (page); + priv = tpage->priv; + + return priv->main; } static void -contacts_changed_cb (BonoboListener *listener, - char *event_name, - CORBA_any *arg, - CORBA_Environment *ev, - gpointer data) +task_page_focus_main_widget (CompEditorPage *page) { TaskPage *tpage; TaskPagePrivate *priv; - - tpage = TASK_PAGE (data); + + tpage = TASK_PAGE (page); priv = tpage->priv; - - if (!priv->updating) - comp_editor_page_notify_changed (COMP_EDITOR_PAGE (tpage)); + + gtk_widget_grab_focus (priv->summary); } -/* fill_widgets handler for the task page */ -static void -task_page_fill_widgets (CompEditorPage *page, CalComponent *comp) +static gboolean +task_page_fill_widgets (CompEditorPage *page, + ECalComponent *comp) { TaskPage *tpage; TaskPagePrivate *priv; - CalComponentText text; - CalComponentDateTime d; - CalComponentClassification cl; - CalClientGetStatus get_tz_status; + ECalComponentText text; + ECalComponentDateTime d; + ECalComponentClassification cl; + ESourceRegistry *registry; + CompEditor *editor; + CompEditorFlags flags; + GtkAction *action; + ECalClient *client; + EShell *shell; GSList *l; - const char *categories; + icalcomponent *icalcomp; + const gchar *categories, *uid; icaltimezone *zone, *default_zone; - char *location; + gchar *backend_addr = NULL; + gboolean active; + gint *priority_value, *percent = NULL; + TaskEditorPriority priority; + icalproperty_status status; + const gchar *url; + struct icaltimetype *completed = NULL; tpage = TASK_PAGE (page); priv = tpage->priv; - priv->updating = TRUE; - + editor = comp_editor_page_get_editor (page); + client = comp_editor_get_client (editor); + flags = comp_editor_get_flags (editor); + shell = comp_editor_get_shell (editor); + + registry = e_shell_get_registry (shell); + + /* Clean out old data */ + if (priv->comp != NULL) + g_object_unref (priv->comp); + priv->comp = NULL; + + g_ptr_array_foreach ( + priv->deleted_attendees, (GFunc) g_object_unref, NULL); + g_ptr_array_set_size (priv->deleted_attendees, 0); + + /* Component for cancellation */ + priv->comp = e_cal_component_clone (comp); + comp_editor_copy_new_attendees (priv->comp, comp); + /* Clean the screen */ clear_widgets (tpage); - /* Summary, description(s) */ - cal_component_get_summary (comp, &text); - e_dialog_editable_set (priv->summary, text.value); + priv->user_add = itip_get_comp_attendee ( + registry, comp, client); - cal_component_get_description_list (comp, &l); - if (l) { - text = *(CalComponentText *)l->data; - e_dialog_editable_set (priv->description, text.value); + /* Summary, description(s) */ + e_cal_component_get_summary (comp, &text); + if (text.value != NULL) + gtk_entry_set_text (GTK_ENTRY (priv->summary), text.value); + else + gtk_entry_set_text (GTK_ENTRY (priv->summary), ""); + + e_cal_component_get_description_list (comp, &l); + if (l && l->data) { + ECalComponentText *dtext; + + dtext = l->data; + gtk_text_buffer_set_text ( + gtk_text_view_get_buffer (GTK_TEXT_VIEW (priv->description)), + dtext->value ? dtext->value : "", -1); } else { - e_dialog_editable_set (priv->description, NULL); + gtk_text_buffer_set_text ( + gtk_text_view_get_buffer (GTK_TEXT_VIEW (priv->description)), + "", 0); } - cal_component_free_text_list (l); + e_cal_component_free_text_list (l); + e_buffer_tagger_update_tags (GTK_TEXT_VIEW (priv->description)); - location = calendar_config_get_timezone (); - default_zone = icaltimezone_get_builtin_timezone (location); + default_zone = comp_editor_get_timezone (editor); /* Due Date. */ - cal_component_get_due (comp, &d); + e_cal_component_get_due (comp, &d); zone = NULL; if (d.value) { struct icaltimetype *due_tt = d.value; - e_date_edit_set_date (E_DATE_EDIT (priv->due_date), - due_tt->year, due_tt->month, - due_tt->day); - if (due_tt->is_date) { - e_date_edit_set_time_of_day (E_DATE_EDIT (priv->due_date), - -1, -1); - zone = default_zone; - } else { - e_date_edit_set_time_of_day (E_DATE_EDIT (priv->due_date), - due_tt->hour, - due_tt->minute); - } + e_date_edit_set_date ( + E_DATE_EDIT (priv->due_date), + due_tt->year, due_tt->month, + due_tt->day); + e_date_edit_set_time_of_day (E_DATE_EDIT (priv->due_date), -1, -1); } else { e_date_edit_set_time (E_DATE_EDIT (priv->due_date), -1); /* If no time is set, we use the default timezone, so the - user usually doesn't have to set this when they set the - date. */ - zone = default_zone; + * user usually doesn't have to set this when they set the + * date. */ + zone = NULL; } /* Note that if we are creating a new task, the timezones may not be - on the server, so we try to get the builtin timezone with the TZID - first. */ - if (!zone) - zone = icaltimezone_get_builtin_timezone_from_tzid (d.tzid); - if (!zone) { - get_tz_status = cal_client_get_timezone (page->client, d.tzid, - &zone); - /* FIXME: Handle error better. */ - if (get_tz_status != CAL_CLIENT_GET_SUCCESS) - g_warning ("Couldn't get timezone from server: %s", - d.tzid ? d.tzid : ""); + * on the server, so we try to get the builtin timezone with the TZID + * first. */ + if (!zone && d.tzid) { + GError *error = NULL; + + e_cal_client_get_timezone_sync ( + client, d.tzid, &zone, NULL, &error); + + if (error != NULL) { + /* FIXME: Handle error better. */ + g_warning ( + "Couldn't get timezone '%s' from server: %s", + d.tzid ? d.tzid : "", error->message); + g_error_free (error); + } } - e_timezone_entry_set_timezone (E_TIMEZONE_ENTRY (priv->due_timezone), - zone); - cal_component_free_datetime (&d); + e_timezone_entry_set_timezone ( + E_TIMEZONE_ENTRY (priv->timezone), + zone ? zone : default_zone); + + action = comp_editor_get_action (editor, "view-time-zone"); + active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)); + task_page_set_show_timezone (tpage, active); + if (!(flags & COMP_EDITOR_NEW_ITEM) && !zone) { + GtkAction *action; + + task_page_set_show_timezone (tpage, FALSE); + action = comp_editor_get_action (editor, "view-time-zone"); + gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (action), FALSE); + } + + e_cal_component_free_datetime (&d); /* Start Date. */ - cal_component_get_dtstart (comp, &d); + e_cal_component_get_dtstart (comp, &d); zone = NULL; if (d.value) { struct icaltimetype *start_tt = d.value; - e_date_edit_set_date (E_DATE_EDIT (priv->start_date), - start_tt->year, start_tt->month, - start_tt->day); - if (start_tt->is_date) { - e_date_edit_set_time_of_day (E_DATE_EDIT (priv->start_date), - -1, -1); - zone = default_zone; - } else { - e_date_edit_set_time_of_day (E_DATE_EDIT (priv->start_date), - start_tt->hour, - start_tt->minute); - } + e_date_edit_set_date ( + E_DATE_EDIT (priv->start_date), + start_tt->year, start_tt->month, + start_tt->day); + e_date_edit_set_time_of_day (E_DATE_EDIT (priv->start_date), -1, -1); } else { e_date_edit_set_time (E_DATE_EDIT (priv->start_date), -1); + } - /* If no time is set, we use the default timezone, so the - user usually doesn't have to set this when they set the - date. */ - zone = default_zone; + e_cal_component_free_datetime (&d); + + /* Classification. */ + e_cal_component_get_classification (comp, &cl); + comp_editor_set_classification (editor, cl); + + e_cal_component_get_uid (comp, &uid); + if (e_cal_client_get_object_sync (client, uid, NULL, &icalcomp, NULL, NULL)) { + icalcomponent_free (icalcomp); + task_page_hide_options (tpage); } - if (!zone) - zone = icaltimezone_get_builtin_timezone_from_tzid (d.tzid); - if (!zone) { - get_tz_status = cal_client_get_timezone (page->client, d.tzid, - &zone); - /* FIXME: Handle error better. */ - if (get_tz_status != CAL_CLIENT_GET_SUCCESS) - g_warning ("Couldn't get timezone from server: %s", - d.tzid ? d.tzid : ""); + /* Categories */ + e_cal_component_get_categories (comp, &categories); + if (categories != NULL) + gtk_entry_set_text (GTK_ENTRY (priv->categories), categories); + else + gtk_entry_set_text (GTK_ENTRY (priv->categories), ""); + + /* Source */ + e_source_combo_box_set_active ( + E_SOURCE_COMBO_BOX (priv->client_combo_box), + e_client_get_source (E_CLIENT (client))); + + e_client_get_backend_property_sync (E_CLIENT (client), CAL_BACKEND_PROPERTY_CAL_EMAIL_ADDRESS, &backend_addr, NULL, NULL); + set_subscriber_info_string (tpage, backend_addr); + + if (priv->is_assignment) { + ECalComponentOrganizer organizer; + gchar *name = NULL; + gchar *mailto = NULL; + + priv->user_add = itip_get_comp_attendee ( + registry, comp, client); + + /* Organizer strings */ + task_page_select_organizer (tpage, backend_addr); + + /* If there is an existing organizer show it properly */ + if (e_cal_component_has_organizer (comp)) { + e_cal_component_get_organizer (comp, &organizer); + if (organizer.value != NULL) { + const gchar *strip = itip_strip_mailto (organizer.value); + gchar *string; + + if (itip_organizer_is_user (registry, comp, client) || + itip_sentby_is_user (registry, comp, client)) { + if (e_client_check_capability (E_CLIENT (client), CAL_STATIC_CAPABILITY_ORGANIZER_NOT_EMAIL_ADDRESS)) + priv->user_org = TRUE; + } else { + if (e_client_check_capability (E_CLIENT (client), CAL_STATIC_CAPABILITY_ORGANIZER_NOT_EMAIL_ADDRESS)) + gtk_widget_set_sensitive (priv->invite, FALSE); + gtk_widget_set_sensitive (priv->add, FALSE); + gtk_widget_set_sensitive (priv->edit, FALSE); + gtk_widget_set_sensitive (priv->remove, FALSE); + priv->user_org = FALSE; + } + + if (e_client_check_capability (E_CLIENT (client), CAL_STATIC_CAPABILITY_NO_ORGANIZER) && (flags & COMP_EDITOR_DELEGATE)) + string = g_strdup (priv->user_add); + else if (organizer.cn != NULL) + string = g_strdup_printf ("%s <%s>", organizer.cn, strip); + else + string = g_strdup (strip); + + g_signal_handlers_block_by_func (gtk_bin_get_child (GTK_BIN (priv->organizer)), organizer_changed_cb, tpage); + + if (!priv->user_org) { + GtkComboBox *combo_box; + GtkListStore *list_store; + GtkTreeModel *model; + GtkTreeIter iter; + + combo_box = GTK_COMBO_BOX (priv->organizer); + model = gtk_combo_box_get_model (combo_box); + list_store = GTK_LIST_STORE (model); + + gtk_list_store_clear (list_store); + gtk_list_store_append (list_store, &iter); + gtk_list_store_set (list_store, &iter, 0, string, -1); + gtk_combo_box_set_active (combo_box, 0); + gtk_editable_set_editable (GTK_EDITABLE (gtk_bin_get_child (GTK_BIN (priv->organizer))), FALSE); + } else { + gtk_entry_set_text (GTK_ENTRY (gtk_bin_get_child (GTK_BIN (priv->organizer))), string); + } + + g_signal_handlers_unblock_by_func (gtk_bin_get_child (GTK_BIN (priv->organizer)), organizer_changed_cb, tpage); + + g_free (string); + priv->existing = TRUE; + } + } else if (get_current_identity (tpage, &name, &mailto)) { + EMeetingAttendee *attendee; + gchar *backend_mailto = NULL; + + if (backend_addr != NULL && *backend_addr != '\0') { + backend_mailto = g_strdup_printf ( + "MAILTO:%s", backend_addr); + if (g_ascii_strcasecmp (backend_mailto, mailto) == 0) { + g_free (backend_mailto); + backend_mailto = NULL; + } + } + + attendee = + e_meeting_store_add_attendee_with_defaults ( + priv->meeting_store); + priv->ia = g_object_ref (attendee); + + if (backend_mailto == NULL) { + e_meeting_attendee_set_cn (attendee, name); + e_meeting_attendee_set_address (attendee, mailto); + name = mailto = NULL; + } else { + e_meeting_attendee_set_address (attendee, backend_mailto); + e_meeting_attendee_set_sentby (attendee, mailto); + backend_mailto = mailto = NULL; + } + + if (client && e_cal_client_check_organizer_must_accept (client)) + e_meeting_attendee_set_status ( + attendee, ICAL_PARTSTAT_NEEDSACTION); + else + e_meeting_attendee_set_status ( + attendee, ICAL_PARTSTAT_ACCEPTED); + + e_meeting_list_view_add_attendee_to_name_selector ( + E_MEETING_LIST_VIEW (priv->list_view), attendee); + + g_free (backend_mailto); + } + + g_free (mailto); + g_free (name); } - e_timezone_entry_set_timezone (E_TIMEZONE_ENTRY (priv->start_timezone), - zone); - cal_component_free_datetime (&d); + g_free (backend_addr); - /* Classification. */ - cal_component_get_classification (comp, &cl); + /* Percent Complete. */ + e_cal_component_get_percent (comp, &percent); + if (percent) { + gtk_spin_button_set_value ( + GTK_SPIN_BUTTON (priv->percent_complete), *percent); + } else { + /* FIXME: Could check if task is completed and set 100%. */ + gtk_spin_button_set_value ( + GTK_SPIN_BUTTON (priv->percent_complete), 0); + } - switch (cl) { - case CAL_COMPONENT_CLASS_PUBLIC: - e_dialog_radio_set (priv->classification_public, - CAL_COMPONENT_CLASS_PUBLIC, - classification_map); - break; + /* Status. */ + e_cal_component_get_status (comp, &status); + if (status == ICAL_STATUS_NONE || status == ICAL_STATUS_NEEDSACTION) { + /* Try to use the percent value. */ + if (percent) { + if (*percent == 100) + status = ICAL_STATUS_COMPLETED; + else if (*percent > 0) + status = ICAL_STATUS_INPROCESS; + else + status = ICAL_STATUS_NONE; + } else + status = ICAL_STATUS_NONE; + } + e_dialog_combo_box_set (priv->status_combo, status, status_map); - case CAL_COMPONENT_CLASS_PRIVATE: - e_dialog_radio_set (priv->classification_public, - CAL_COMPONENT_CLASS_PRIVATE, - classification_map); - break; + if (percent) + e_cal_component_free_percent (percent); - case CAL_COMPONENT_CLASS_CONFIDENTIAL: - e_dialog_radio_set (priv->classification_public, - CAL_COMPONENT_CLASS_CONFIDENTIAL, - classification_map); - break; + /* Completed Date. */ + e_cal_component_get_completed (comp, &completed); + if (completed) { + icaltimezone *utc_zone, *zone; - default: - /* default to PUBLIC */ - e_dialog_radio_set (priv->classification_public, - CAL_COMPONENT_CLASS_PUBLIC, - classification_map); - break; - } + /* Completed is in UTC, but that would confuse the user, so + * we convert it to local time. */ + utc_zone = icaltimezone_get_utc_timezone (); + zone = comp_editor_get_timezone (editor); - /* Categories */ - cal_component_get_categories (comp, &categories); - e_dialog_editable_set (priv->categories, categories); + icaltimezone_convert_time (completed, utc_zone, zone); + + e_date_edit_set_date ( + E_DATE_EDIT (priv->completed_date), + completed->year, completed->month, + completed->day); + e_date_edit_set_time_of_day ( + E_DATE_EDIT (priv->completed_date), + completed->hour, + completed->minute); + e_cal_component_free_icaltimetype (completed); + } - /* Contacts */ - comp_editor_contacts_to_widget (priv->contacts_entry, comp); + /* Priority. */ + e_cal_component_get_priority (comp, &priority_value); + if (priority_value) { + priority = priority_value_to_index (*priority_value); + e_cal_component_free_priority (priority_value); + } else { + priority = PRIORITY_UNDEFINED; + } + e_dialog_combo_box_set (priv->priority_combo, priority, priority_map); - /* We connect the contacts changed signal here, as we have to be a bit - more careful with it due to the use or Corba. The priv->updating - flag won't work as we won't get the changed event immediately. - FIXME: Unfortunately this doesn't work either. We never get the - changed event now. */ - comp_editor_connect_contacts_changed (priv->contacts_entry, - contacts_changed_cb, tpage); + /* URL */ + e_cal_component_get_url (comp, &url); + gtk_entry_set_text (GTK_ENTRY (priv->web_page_entry), url ? url : ""); + sensitize_widgets (tpage); - priv->updating = FALSE; + return TRUE; } -/* fill_component handler for the task page */ static gboolean -task_page_fill_component (CompEditorPage *page, CalComponent *comp) +task_page_fill_component (CompEditorPage *page, + ECalComponent *comp) { TaskPage *tpage; TaskPagePrivate *priv; - CalComponentDateTime date; - struct icaltimetype icaltime; - char *cat, *str; - gboolean date_set, time_set; + ECalComponentClassification classification; + ECalComponentDateTime date; + CompEditor *editor; + CompEditorFlags flags; + ECalClient *client; + struct icaltimetype start_tt, due_tt; + gchar *cat, *str; + gboolean start_date_set, due_date_set; + GtkTextBuffer *text_buffer; + GtkTextIter text_iter_start, text_iter_end; + struct icaltimetype icalcomplete, icaltoday; + icalproperty_status status; + TaskEditorPriority priority; + gint priority_value, percent; + const gchar *text; + gboolean date_set; icaltimezone *zone; tpage = TASK_PAGE (page); priv = tpage->priv; + text_buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (priv->description)); + + editor = comp_editor_page_get_editor (page); + client = comp_editor_get_client (editor); + flags = comp_editor_get_flags (editor); + zone = comp_editor_get_timezone (editor); /* Summary. */ - str = e_dialog_editable_get (priv->summary); - if (!str || strlen (str) == 0) - cal_component_set_summary (comp, NULL); + str = gtk_editable_get_chars (GTK_EDITABLE (priv->summary), 0, -1); + if (str == NULL || *str == '\0') + e_cal_component_set_summary (comp, NULL); else { - CalComponentText text; + ECalComponentText text; text.value = str; text.altrep = NULL; - cal_component_set_summary (comp, &text); + e_cal_component_set_summary (comp, &text); } - if (str) - g_free (str); + g_free (str); /* Description */ - str = e_dialog_editable_get (priv->description); + gtk_text_buffer_get_start_iter (text_buffer, &text_iter_start); + gtk_text_buffer_get_end_iter (text_buffer, &text_iter_end); + str = gtk_text_buffer_get_text (text_buffer, &text_iter_start, &text_iter_end, FALSE); + if (!str || strlen (str) == 0) - cal_component_set_description_list (comp, NULL); + e_cal_component_set_description_list (comp, NULL); else { GSList l; - CalComponentText text; + ECalComponentText text; text.value = str; text.altrep = NULL; l.data = &text; l.next = NULL; - cal_component_set_description_list (comp, &l); + e_cal_component_set_description_list (comp, &l); } - if (!str) - g_free (str); + g_free (str); /* Dates */ - icaltime = icaltime_null_time (); + due_tt = icaltime_null_time (); - date.value = &icaltime; + date.value = &due_tt; date.tzid = NULL; /* Due Date. */ - if (!e_date_edit_date_is_valid (E_DATE_EDIT (priv->due_date)) || - !e_date_edit_time_is_valid (E_DATE_EDIT (priv->due_date))) { + if (!e_date_edit_date_is_valid (E_DATE_EDIT (priv->due_date))) { comp_editor_page_display_validation_error (page, _("Due date is wrong"), priv->due_date); return FALSE; } - date_set = e_date_edit_get_date (E_DATE_EDIT (priv->due_date), - &icaltime.year, - &icaltime.month, - &icaltime.day); - time_set = e_date_edit_get_time_of_day (E_DATE_EDIT (priv->due_date), - &icaltime.hour, - &icaltime.minute); - if (date_set) { - if (time_set) { - zone = e_timezone_entry_get_timezone (E_TIMEZONE_ENTRY (priv->due_timezone)); - date.tzid = icaltimezone_get_tzid (zone); - } else { - icaltime.is_date = TRUE; - date.tzid = NULL; - } - cal_component_set_due (comp, &date); + due_date_set = e_date_edit_get_date ( + E_DATE_EDIT (priv->due_date), + &due_tt.year, + &due_tt.month, + &due_tt.day); + if (due_date_set) { + due_tt.is_date = TRUE; + date.tzid = NULL; + e_cal_component_set_due (comp, &date); } else { - cal_component_set_due (comp, NULL); + e_cal_component_set_due (comp, NULL); } /* Start Date. */ - if (!e_date_edit_date_is_valid (E_DATE_EDIT (priv->start_date)) || - !e_date_edit_time_is_valid (E_DATE_EDIT (priv->start_date))) { + if (!e_date_edit_date_is_valid (E_DATE_EDIT (priv->start_date))) { comp_editor_page_display_validation_error (page, _("Start date is wrong"), priv->start_date); return FALSE; } - icaltime = icaltime_null_time (); - date_set = e_date_edit_get_date (E_DATE_EDIT (priv->start_date), - &icaltime.year, - &icaltime.month, - &icaltime.day); - time_set = e_date_edit_get_time_of_day (E_DATE_EDIT (priv->start_date), - &icaltime.hour, - &icaltime.minute); - if (date_set) { - if (time_set) { - zone = e_timezone_entry_get_timezone (E_TIMEZONE_ENTRY (priv->start_timezone)); - date.tzid = icaltimezone_get_tzid (zone); - } else { - icaltime.is_date = TRUE; - date.tzid = NULL; - } - cal_component_set_dtstart (comp, &date); + start_tt = icaltime_null_time (); + date.value = &start_tt; + start_date_set = e_date_edit_get_date ( + E_DATE_EDIT (priv->start_date), + &start_tt.year, + &start_tt.month, + &start_tt.day); + if (start_date_set) { + start_tt.is_date = TRUE; + date.tzid = NULL; + e_cal_component_set_dtstart (comp, &date); } else { - cal_component_set_dtstart (comp, NULL); + e_cal_component_set_dtstart (comp, NULL); } /* Classification. */ - cal_component_set_classification (comp, classification_get (priv->classification_public)); + classification = comp_editor_get_classification (editor); + e_cal_component_set_classification (comp, classification); + + /* send options */ + if (priv->sendoptions_shown && priv->sod) { + icaltimezone *zone = comp_editor_get_timezone (editor); + e_send_options_utils_fill_component (priv->sod, comp, zone); + } /* Categories */ - cat = e_dialog_editable_get (priv->categories); + cat = gtk_editable_get_chars (GTK_EDITABLE (priv->categories), 0, -1); str = comp_editor_strip_categories (cat); - if (cat) - g_free (cat); + g_free (cat); - cal_component_set_categories (comp, str); + e_cal_component_set_categories (comp, str); if (str) g_free (str); - /* Contacts */ - comp_editor_contacts_to_component (priv->contacts_entry, comp); + if (priv->is_assignment) { + ECalComponentOrganizer organizer = {NULL, NULL, NULL, NULL}; + + if (!priv->existing) { + gchar *backend_addr = NULL; + gchar *backend_mailto = NULL; + gchar *name; + gchar *mailto; + + e_client_get_backend_property_sync (E_CLIENT (client), CAL_BACKEND_PROPERTY_CAL_EMAIL_ADDRESS, &backend_addr, NULL, NULL); + + /* Find the identity for the organizer or sentby field */ + if (!get_current_identity (tpage, &name, &mailto)) { + e_notice ( + priv->main, GTK_MESSAGE_ERROR, + _("An organizer is required.")); + return FALSE; + } + + /* Prefer the backend address if we have one. */ + if (backend_addr != NULL && *backend_addr != '\0') { + backend_mailto = g_strdup_printf ( + "MAILTO:%s", backend_addr); + if (g_ascii_strcasecmp (backend_mailto, mailto) == 0) { + g_free (backend_mailto); + backend_mailto = NULL; + } + } + + if (backend_mailto == NULL) { + organizer.cn = name; + organizer.value = mailto; + name = mailto = NULL; + } else { + organizer.value = backend_mailto; + organizer.sentby = mailto; + backend_mailto = mailto = NULL; + } + + e_cal_component_set_organizer (comp, &organizer); + + g_free (backend_addr); + g_free (backend_mailto); + g_free (name); + g_free (mailto); + } + + if (e_meeting_store_count_actual_attendees (priv->meeting_store) < 1) { + e_notice ( + priv->main, GTK_MESSAGE_ERROR, + _("At least one attendee is required.")); + return FALSE; + } + + if (flags & COMP_EDITOR_DELEGATE) { + GSList *attendee_list, *l; + gint i; + const GPtrArray *attendees = e_meeting_store_get_attendees (priv->meeting_store); + + e_cal_component_get_attendee_list (priv->comp, &attendee_list); + + for (i = 0; i < attendees->len; i++) { + EMeetingAttendee *ia = g_ptr_array_index (attendees, i); + ECalComponentAttendee *ca; + + /* Remove the duplicate user from the component if present */ + if (e_meeting_attendee_is_set_delfrom (ia) || e_meeting_attendee_is_set_delto (ia)) { + for (l = attendee_list; l; l = l->next) { + ECalComponentAttendee *a = l->data; + + if (g_str_equal (a->value, e_meeting_attendee_get_address (ia))) { + attendee_list = g_slist_remove (attendee_list, l->data); + break; + } + } + } + + ca = e_meeting_attendee_as_e_cal_component_attendee (ia); + + attendee_list = g_slist_append (attendee_list, ca); + } + e_cal_component_set_attendee_list (comp, attendee_list); + e_cal_component_free_attendee_list (attendee_list); + } else + set_attendees (comp, e_meeting_store_get_attendees (priv->meeting_store)); + } + + /* Percent Complete. */ + percent = gtk_spin_button_get_value_as_int ( + GTK_SPIN_BUTTON (priv->percent_complete)); + e_cal_component_set_percent (comp, &percent); + + /* Status. */ + status = e_dialog_combo_box_get (priv->status_combo, status_map); + e_cal_component_set_status (comp, status); + + /* Priority. */ + priority = e_dialog_combo_box_get (priv->priority_combo, priority_map); + priority_value = priority_index_to_value (priority); + e_cal_component_set_priority (comp, &priority_value); + + icalcomplete = icaltime_null_time (); + + /* COMPLETED must be in UTC. */ + icalcomplete.is_utc = 1; + + /* Completed Date. */ + if (!e_date_edit_date_is_valid (E_DATE_EDIT (priv->completed_date)) || + !e_date_edit_time_is_valid (E_DATE_EDIT (priv->completed_date))) { + comp_editor_page_display_validation_error ( + page, _("Completed date is wrong"), + priv->completed_date); + return FALSE; + } + + date_set = e_date_edit_get_date ( + E_DATE_EDIT (priv->completed_date), + &icalcomplete.year, + &icalcomplete.month, + &icalcomplete.day); + + if (date_set) { + e_date_edit_get_time_of_day ( + E_DATE_EDIT (priv->completed_date), + &icalcomplete.hour, + &icalcomplete.minute); + + /* COMPLETED today or before */ + icaltoday = icaltime_current_time_with_zone (zone); + icaltimezone_convert_time ( + &icaltoday, zone, + icaltimezone_get_utc_timezone ()); + + if (icaltime_compare_date_only (icalcomplete, icaltoday) > 0) { + comp_editor_page_display_validation_error ( + page, _("Completed date is wrong"), + priv->completed_date); + return FALSE; + } + + /* COMPLETED must be in UTC, so we assume that the date in the + * dialog is in the current timezone, and we now convert it + * to UTC. FIXME: We should really use one timezone for the + * entire time the dialog is shown. Otherwise if the user + * changes the timezone, the COMPLETED date may get changed + * as well. */ + icaltimezone_convert_time ( + &icalcomplete, zone, + icaltimezone_get_utc_timezone ()); + e_cal_component_set_completed (comp, &icalcomplete); + } else { + e_cal_component_set_completed (comp, NULL); + } + + /* URL. */ + text = gtk_entry_get_text (GTK_ENTRY (priv->web_page_entry)); + e_cal_component_set_url (comp, text); + + return TRUE; +} + +static gboolean +task_page_fill_timezones (CompEditorPage *page, + GHashTable *timezones) +{ + TaskPage *tpage; + TaskPagePrivate *priv; + icaltimezone *zone; + + tpage = TASK_PAGE (page); + priv = tpage->priv; + + /* add start date timezone */ + zone = e_timezone_entry_get_timezone (E_TIMEZONE_ENTRY (priv->timezone)); + if (zone) { + if (!g_hash_table_lookup (timezones, icaltimezone_get_tzid (zone))) + g_hash_table_insert (timezones, (gpointer) icaltimezone_get_tzid (zone), zone); + } + + /* Add UTC timezone, which is the one + * used for the DATE-COMPLETED property. */ + zone = icaltimezone_get_utc_timezone (); + if (zone != NULL) { + gconstpointer tzid = icaltimezone_get_tzid (zone); + + if (!g_hash_table_lookup (timezones, tzid)) + g_hash_table_insert (timezones, (gpointer) tzid, zone); + } return TRUE; } -/* set_summary handler for the task page */ static void -task_page_set_summary (CompEditorPage *page, const char *summary) +task_page_add_attendee (CompEditorPage *page, + EMeetingAttendee *attendee) { - /* nothing */ + CompEditor *editor; + TaskPagePrivate *priv; + + priv = TASK_PAGE_GET_PRIVATE (page); + editor = comp_editor_page_get_editor (page); + + if ((comp_editor_get_flags (editor) & COMP_EDITOR_DELEGATE) != 0) { + gchar *delfrom; + + /* EMeetingAttendee takes ownership of the string. */ + delfrom = g_strdup_printf ("MAILTO:%s", priv->user_add); + e_meeting_attendee_set_delfrom (attendee, delfrom); + } + + e_meeting_store_add_attendee (priv->meeting_store, attendee); + e_meeting_list_view_add_attendee_to_name_selector ( + E_MEETING_LIST_VIEW (priv->list_view), attendee); } static void -task_page_set_dates (CompEditorPage *page, CompEditorPageDates *dates) +task_page_class_init (TaskPageClass *class) +{ + GObjectClass *object_class; + CompEditorPageClass *editor_page_class; + + g_type_class_add_private (class, sizeof (TaskPagePrivate)); + + object_class = G_OBJECT_CLASS (class); + object_class->dispose = task_page_dispose; + object_class->finalize = task_page_finalize; + + editor_page_class = COMP_EDITOR_PAGE_CLASS (class); + editor_page_class->get_widget = task_page_get_widget; + editor_page_class->focus_main_widget = task_page_focus_main_widget; + editor_page_class->fill_widgets = task_page_fill_widgets; + editor_page_class->fill_component = task_page_fill_component; + editor_page_class->fill_timezones = task_page_fill_timezones; + editor_page_class->add_attendee = task_page_add_attendee; +} + +static void +task_page_init (TaskPage *tpage) +{ + tpage->priv = TASK_PAGE_GET_PRIVATE (tpage); + tpage->priv->deleted_attendees = g_ptr_array_new (); +} + +void +task_page_set_view_role (TaskPage *page, + gboolean state) +{ + TaskPagePrivate *priv = page->priv; + + e_meeting_list_view_column_set_visible (priv->list_view, E_MEETING_STORE_ROLE_COL, state); +} + +void +task_page_set_view_status (TaskPage *page, + gboolean state) +{ + TaskPagePrivate *priv = page->priv; + + e_meeting_list_view_column_set_visible (priv->list_view, E_MEETING_STORE_STATUS_COL, state); +} + +void +task_page_set_view_type (TaskPage *page, + gboolean state) +{ + TaskPagePrivate *priv = page->priv; + + e_meeting_list_view_column_set_visible (priv->list_view, E_MEETING_STORE_TYPE_COL, state); +} + +void +task_page_set_view_rsvp (TaskPage *page, + gboolean state) +{ + TaskPagePrivate *priv = page->priv; + + e_meeting_list_view_column_set_visible (priv->list_view, E_MEETING_STORE_RSVP_COL, state); +} + +void +task_page_hide_options (TaskPage *page) +{ + CompEditor *editor; + GtkAction *action; + + g_return_if_fail (IS_TASK_PAGE (page)); + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (page)); + action = comp_editor_get_action (editor, "send-options"); + gtk_action_set_visible (action, FALSE); +} + +void +task_page_show_options (TaskPage *page) +{ + CompEditor *editor; + GtkAction *action; + + g_return_if_fail (IS_TASK_PAGE (page)); + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (page)); + action = comp_editor_get_action (editor, "send-options"); + gtk_action_set_visible (action, TRUE); +} + +void +task_page_set_assignment (TaskPage *page, + gboolean set) +{ + g_return_if_fail (IS_TASK_PAGE (page)); + + page->priv->is_assignment = set; + sensitize_widgets (page); +} + +static void +add_clicked_cb (GtkButton *btn, + TaskPage *page) +{ + EMeetingAttendee *attendee; + CompEditor *editor; + CompEditorFlags flags; + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (page)); + flags = comp_editor_get_flags (editor); + + attendee = e_meeting_store_add_attendee_with_defaults (page->priv->meeting_store); + + if (flags & COMP_EDITOR_DELEGATE) { + e_meeting_attendee_set_delfrom (attendee, g_strdup_printf ("MAILTO:%s", page->priv->user_add)); + } + + e_meeting_list_view_edit (page->priv->list_view, attendee); +} + +static void edit_clicked_cb (GtkButton *btn, TaskPage *tpage) { - TaskPage *tpage; TaskPagePrivate *priv; + GtkTreePath *path = NULL; + GtkTreeViewColumn *focus_col; - tpage = TASK_PAGE (page); priv = tpage->priv; - if (priv->updating) - return; + gtk_tree_view_get_cursor (GTK_TREE_VIEW (priv->list_view), &path, NULL); + g_return_if_fail (path != NULL); + + gtk_tree_view_get_cursor (GTK_TREE_VIEW (priv->list_view), &path, &focus_col); + gtk_tree_view_set_cursor (GTK_TREE_VIEW (priv->list_view), path, focus_col, TRUE); + gtk_tree_path_free (path); +} + +static gboolean +existing_attendee (EMeetingAttendee *ia, + ECalComponent *comp) +{ + GSList *attendees, *l; + const gchar *ia_address; + const gchar *ia_sentby = NULL; + + ia_address = itip_strip_mailto (e_meeting_attendee_get_address (ia)); + if (!ia_address) + return FALSE; + + if (e_meeting_attendee_is_set_sentby (ia)) + ia_sentby = itip_strip_mailto (e_meeting_attendee_get_sentby (ia)); + + e_cal_component_get_attendee_list (comp, &attendees); + + for (l = attendees; l; l = l->next) { + ECalComponentAttendee *attendee = l->data; + const gchar *address; + const gchar *sentby = NULL; + + address = itip_strip_mailto (attendee->value); + if (attendee->sentby) + sentby = itip_strip_mailto (attendee->sentby); + + if ((address && !g_ascii_strcasecmp (ia_address, address)) || (sentby && ia_sentby && !g_ascii_strcasecmp (ia_sentby, sentby))) { + e_cal_component_free_attendee_list (attendees); + return TRUE; + } + } + + e_cal_component_free_attendee_list (attendees); + + return FALSE; +} + +static void +remove_attendee (TaskPage *page, + EMeetingAttendee *ia) +{ + TaskPagePrivate *priv = page->priv; + CompEditor *editor; + CompEditorFlags flags; + gint pos = 0; + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (page)); + flags = comp_editor_get_flags (editor); + + /* If the user deletes the organizer attendee explicitly, + * assume they no longer want the organizer showing up */ + if (ia == priv->ia) { + g_object_unref (priv->ia); + priv->ia = NULL; + } + + /* If this was a delegatee, no longer delegate */ + if (e_meeting_attendee_is_set_delfrom (ia)) { + EMeetingAttendee *ib; + + ib = e_meeting_store_find_attendee (priv->meeting_store, e_meeting_attendee_get_delfrom (ia), &pos); + if (ib != NULL) { + e_meeting_attendee_set_delto (ib, NULL); + + if (!(flags & COMP_EDITOR_DELEGATE)) + e_meeting_attendee_set_edit_level (ib, E_MEETING_ATTENDEE_EDIT_FULL); + } + } + + /* Handle deleting all attendees in the delegation chain */ + while (ia != NULL) { + EMeetingAttendee *ib = NULL; + + if (existing_attendee (ia, priv->comp) && !comp_editor_have_in_new_attendees (priv->comp, ia)) { + g_object_ref (ia); + g_ptr_array_add (priv->deleted_attendees, ia); + } + + if (e_meeting_attendee_get_delto (ia) != NULL) + ib = e_meeting_store_find_attendee (priv->meeting_store, e_meeting_attendee_get_delto (ia), NULL); + + comp_editor_manage_new_attendees (priv->comp, ia, FALSE); + e_meeting_list_view_remove_attendee_from_name_selector (priv->list_view, ia); + e_meeting_store_remove_attendee (priv->meeting_store, ia); + + ia = ib; + } + + sensitize_widgets (page); +} + +static void +remove_clicked_cb (GtkButton *btn, + TaskPage *page) +{ + TaskPagePrivate *priv; + EMeetingAttendee *ia; + GtkTreeSelection *selection; + GList *paths = NULL, *tmp; + GtkTreeIter iter; + GtkTreePath *path = NULL; + GtkTreeModel *model = NULL; + gboolean valid_iter; + gchar *address; + + priv = page->priv; + + selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (priv->list_view)); + model = GTK_TREE_MODEL (priv->meeting_store); + if (!(paths = gtk_tree_selection_get_selected_rows (selection, &model))) { + g_warning ("Could not get a selection to delete."); + return; + } + paths = g_list_reverse (paths); + + for (tmp = paths; tmp; tmp = tmp->next) { + path = tmp->data; + + gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->meeting_store), &iter, path); + + gtk_tree_model_get (GTK_TREE_MODEL (priv->meeting_store), &iter, E_MEETING_STORE_ADDRESS_COL, &address, -1); + ia = e_meeting_store_find_attendee (priv->meeting_store, address, NULL); + g_free (address); + if (!ia) { + g_warning ("Cannot delete attendee\n"); + continue; + } else if (e_meeting_attendee_get_edit_level (ia) != E_MEETING_ATTENDEE_EDIT_FULL) { + g_warning ("Not enough rights to delete attendee: %s\n", e_meeting_attendee_get_address (ia)); + continue; + } + + remove_attendee (page, ia); + } + + /* Select closest item after removal */ + valid_iter = gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->meeting_store), &iter, path); + if (!valid_iter) { + gtk_tree_path_prev (path); + valid_iter = gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->meeting_store), &iter, path); + } + + if (valid_iter) { + gtk_tree_selection_unselect_all (selection); + gtk_tree_selection_select_iter (selection, &iter); + } + + g_list_foreach (paths, (GFunc) gtk_tree_path_free, NULL); + g_list_free (paths); +} + +static void +invite_cb (GtkWidget *widget, + TaskPage *page) +{ + e_meeting_list_view_invite_others_dialog (page->priv->list_view); +} + +static void +attendee_added_cb (EMeetingListView *emlv, + EMeetingAttendee *ia, + TaskPage *page) +{ + TaskPagePrivate *priv = page->priv; + CompEditor *editor; + CompEditorFlags flags; + ECalClient *client; + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (page)); + client = comp_editor_get_client (editor); + flags = comp_editor_get_flags (editor); + + if (!(flags & COMP_EDITOR_DELEGATE)) { + comp_editor_manage_new_attendees (priv->comp, ia, TRUE); + return; + } + + /* do not remove here, it did EMeetingListView already */ + e_meeting_attendee_set_delfrom (ia, g_strdup_printf ("MAILTO:%s", priv->user_add ? priv->user_add : "")); + + if (!e_client_check_capability (E_CLIENT (client), CAL_STATIC_CAPABILITY_DELEGATE_TO_MANY)) { + EMeetingAttendee *delegator; + + gtk_widget_set_sensitive (priv->invite, FALSE); + gtk_widget_set_sensitive (priv->add, FALSE); + gtk_widget_set_sensitive (priv->edit, FALSE); + + delegator = e_meeting_store_find_attendee (priv->meeting_store, priv->user_add, NULL); + g_return_if_fail (delegator != NULL); + + e_meeting_attendee_set_delto (delegator, g_strdup (e_meeting_attendee_get_address (ia))); + } +} + +static gboolean +list_view_event (EMeetingListView *list_view, + GdkEvent *event, + TaskPage *page) +{ + TaskPagePrivate *priv= page->priv; + CompEditor *editor; + CompEditorFlags flags; + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (page)); + flags = comp_editor_get_flags (editor); + + if (event->type == GDK_2BUTTON_PRESS && flags & COMP_EDITOR_USER_ORG) { + EMeetingAttendee *attendee; + + attendee = e_meeting_store_add_attendee_with_defaults (priv->meeting_store); + + if (flags & COMP_EDITOR_DELEGATE) { + e_meeting_attendee_set_delfrom (attendee, g_strdup_printf ("MAILTO:%s", page->priv->user_add)); + } + + e_meeting_list_view_edit (page->priv->list_view, attendee); + return TRUE; + } + + return FALSE; +} + +static gboolean +list_key_press (EMeetingListView *list_view, + GdkEventKey *event, + TaskPage *page) +{ + if (event->keyval == GDK_KEY_Delete) { + + remove_clicked_cb (NULL, page); + + return TRUE; + } else if (event->keyval == GDK_KEY_Insert) { + add_clicked_cb (NULL, page); + + return TRUE; + } + + return FALSE; +} - priv->updating = TRUE; +void +task_page_set_show_timezone (TaskPage *page, + gboolean state) +{ + if (state) { + gtk_widget_show_all (page->priv->timezone); + gtk_widget_show (page->priv->timezone_label); + } else { + gtk_widget_hide (page->priv->timezone); + gtk_widget_hide (page->priv->timezone_label); + } + +} - priv->updating = FALSE; +void +task_page_set_show_categories (TaskPage *page, + gboolean state) +{ + if (state) { + gtk_widget_show (page->priv->categories_btn); + gtk_widget_show (page->priv->categories); + } else { + gtk_widget_hide (page->priv->categories_btn); + gtk_widget_hide (page->priv->categories); + } } - +/*If the msg has some value set, the icon should always be set */ +void +task_page_set_info_string (TaskPage *tpage, + const gchar *icon, + const gchar *msg) +{ + TaskPagePrivate *priv; + + priv = tpage->priv; + + gtk_image_set_from_stock (GTK_IMAGE (priv->info_icon), icon, GTK_ICON_SIZE_BUTTON); + gtk_label_set_markup (GTK_LABEL (priv->info_string), msg); + + if (msg && icon) + gtk_widget_show (priv->info_hbox); + else + gtk_widget_hide (priv->info_hbox); +} /* Gets the widgets from the XML file and returns if they are all available. */ static gboolean get_widgets (TaskPage *tpage) { + EShell *shell; + EClientCache *client_cache; + CompEditor *editor; CompEditorPage *page = COMP_EDITOR_PAGE (tpage); + GtkEntryCompletion *completion; TaskPagePrivate *priv; GSList *accel_groups; GtkWidget *toplevel; + GtkWidget *parent; + GtkWidget *sw; + GtkTreeSelection *selection; priv = tpage->priv; -#define GW(name) glade_xml_get_widget (priv->xml, name) + editor = comp_editor_page_get_editor (page); + shell = comp_editor_get_shell (editor); + client_cache = e_shell_get_client_cache (shell); - priv->main = GW ("task-page"); + priv->main = e_builder_get_widget (priv->builder, "task-page"); if (!priv->main) return FALSE; /* Get the GtkAccelGroup from the toplevel window, so we can install - it when the notebook page is mapped. */ + * it when the notebook page is mapped. */ toplevel = gtk_widget_get_toplevel (priv->main); - accel_groups = gtk_accel_groups_from_object (GTK_OBJECT (toplevel)); - if (accel_groups) { - page->accel_group = accel_groups->data; - gtk_accel_group_ref (page->accel_group); - } - - gtk_widget_ref (priv->main); - gtk_widget_unparent (priv->main); - - priv->summary = GW ("summary"); - - priv->due_date = GW ("due-date"); - priv->start_date = GW ("start-date"); - priv->due_timezone = GW ("due-timezone"); - priv->start_timezone = GW ("start-timezone"); - - priv->description = GW ("description"); - - priv->classification_public = GW ("classification-public"); - priv->classification_private = GW ("classification-private"); - priv->classification_confidential = GW ("classification-confidential"); - - priv->contacts_btn = GW ("contacts-button"); - priv->contacts_box = GW ("contacts-box"); - - priv->categories_btn = GW ("categories-button"); - priv->categories = GW ("categories"); - -#undef GW + accel_groups = gtk_accel_groups_from_object (G_OBJECT (toplevel)); + if (accel_groups) + page->accel_group = g_object_ref (accel_groups->data); + + g_object_ref (priv->main); + parent = gtk_widget_get_parent (priv->main); + gtk_container_remove (GTK_CONTAINER (parent), priv->main); + + priv->info_hbox = e_builder_get_widget (priv->builder, "generic-info"); + priv->info_icon = e_builder_get_widget (priv->builder, "generic-info-image"); + priv->info_string = e_builder_get_widget (priv->builder, "generic-info-msgs"); + + priv->summary = e_builder_get_widget (priv->builder, "summary"); + priv->summary_label = e_builder_get_widget (priv->builder, "summary-label"); + + /* Glade's visibility flag doesn't seem to work for custom widgets */ + priv->due_date = e_builder_get_widget (priv->builder, "due-date"); + gtk_widget_show (priv->due_date); + priv->start_date = e_builder_get_widget (priv->builder, "start-date"); + gtk_widget_show (priv->start_date); + + priv->completed_date = e_builder_get_widget (priv->builder, "completed-date"); + priv->status_combo = e_builder_get_widget (priv->builder, "status-combobox"); + priv->priority_combo = e_builder_get_widget (priv->builder, "priority-combobox"); + priv->percent_complete = e_builder_get_widget (priv->builder, "percent-complete"); + priv->classification_combo = e_builder_get_widget (priv->builder, "classification-combobox"); + priv->web_page_entry = e_builder_get_widget (priv->builder, "web-page-entry"); + + priv->timezone = e_builder_get_widget (priv->builder, "timezone"); + priv->timezone_label = e_builder_get_widget (priv->builder, "timezone-label"); + priv->attendees_label = e_builder_get_widget (priv->builder, "attendees-label"); + priv->description = e_builder_get_widget (priv->builder, "description"); + priv->categories_btn = e_builder_get_widget (priv->builder, "categories-button"); + priv->categories = e_builder_get_widget (priv->builder, "categories"); + + priv->organizer = e_builder_get_widget (priv->builder, "organizer"); + gtk_list_store_clear (GTK_LIST_STORE (gtk_combo_box_get_model (GTK_COMBO_BOX (priv->organizer)))); + gtk_combo_box_set_entry_text_column (GTK_COMBO_BOX (priv->organizer), 0); + + priv->invite = e_builder_get_widget (priv->builder, "invite"); + priv->add = e_builder_get_widget (priv->builder, "add-attendee"); + priv->edit = e_builder_get_widget (priv->builder, "edit-attendee"); + priv->remove = e_builder_get_widget (priv->builder, "remove-attendee"); + priv->list_box = e_builder_get_widget (priv->builder, "list-box"); + priv->calendar_label = e_builder_get_widget (priv->builder, "group-label"); + priv->attendee_box = e_builder_get_widget (priv->builder, "attendee-box"); + priv->org_cal_label = e_builder_get_widget (priv->builder, "org-task-label"); + + priv->list_view = e_meeting_list_view_new (priv->meeting_store); + + selection = gtk_tree_view_get_selection ((GtkTreeView *) priv->list_view); + gtk_tree_selection_set_mode (selection, GTK_SELECTION_MULTIPLE); + gtk_widget_show (GTK_WIDGET (priv->list_view)); + + sw = gtk_scrolled_window_new (NULL, NULL); + gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); + gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (sw), GTK_SHADOW_IN); + gtk_widget_show (sw); + gtk_container_add (GTK_CONTAINER (sw), GTK_WIDGET (priv->list_view)); + gtk_box_pack_start (GTK_BOX (priv->list_box), sw, TRUE, TRUE, 0); + + priv->client_combo_box = e_builder_get_widget ( + priv->builder, "client-combo-box"); + e_client_combo_box_set_client_cache ( + E_CLIENT_COMBO_BOX (priv->client_combo_box), client_cache); + + gtk_label_set_mnemonic_widget (GTK_LABEL (priv->calendar_label), priv->client_combo_box); + + completion = e_category_completion_new (); + gtk_entry_set_completion (GTK_ENTRY (priv->categories), completion); + g_object_unref (completion); return (priv->summary + && priv->summary_label && priv->due_date && priv->start_date - && priv->due_timezone - && priv->start_timezone - && priv->classification_public - && priv->classification_private - && priv->classification_confidential + && priv->timezone && priv->description - && priv->contacts_btn - && priv->contacts_box && priv->categories_btn - && priv->categories); + && priv->categories + && priv->organizer + && priv->completed_date + && priv->status_combo + && priv->priority_combo + && priv->percent_complete + && priv->classification_combo + && priv->web_page_entry + ); } -/* Callback used when the summary changes; we emit the notification signal. */ static void -summary_changed_cb (GtkEditable *editable, gpointer data) +summary_changed_cb (GtkEntry *entry, + CompEditorPage *page) { - TaskPage *tpage; - TaskPagePrivate *priv; - gchar *summary; - - tpage = TASK_PAGE (data); - priv = tpage->priv; - - if (priv->updating) + CompEditor *editor; + const gchar *text; + + if (comp_editor_page_get_updating (page)) return; - - summary = e_dialog_editable_get (GTK_WIDGET (editable)); - comp_editor_page_notify_summary_changed (COMP_EDITOR_PAGE (tpage), - summary); - g_free (summary); + + editor = comp_editor_page_get_editor (page); + text = gtk_entry_get_text (entry); + comp_editor_set_summary (editor, text); } /* Callback used when the start or due date widgets change. We notify the - * other pages in the task editor, so they can update any labels. + * other pages in the task editor, so they can update any labels. */ static void -date_changed_cb (EDateEdit *dedit, gpointer data) +date_changed_cb (EDateEdit *dedit, + TaskPage *tpage) { - TaskPage *tpage; - TaskPagePrivate *priv; + TaskPagePrivate *priv = tpage->priv; CompEditorPageDates dates; - gboolean date_set, time_set; - CalComponentDateTime start_dt, due_dt; - struct icaltimetype start_tt = icaltime_null_time(); - struct icaltimetype due_tt = icaltime_null_time(); - - tpage = TASK_PAGE (data); - priv = tpage->priv; + gboolean date_set; + ECalComponentDateTime start_dt, due_dt; + struct icaltimetype start_tt = icaltime_null_time (); + struct icaltimetype due_tt = icaltime_null_time (); - if (priv->updating) + if (comp_editor_page_get_updating (COMP_EDITOR_PAGE (tpage))) return; - date_set = e_date_edit_get_date (E_DATE_EDIT (priv->start_date), - &start_tt.year, - &start_tt.month, - &start_tt.day); - time_set = e_date_edit_get_time_of_day (E_DATE_EDIT (priv->start_date), - &start_tt.hour, - &start_tt.minute); + date_set = e_date_edit_get_date ( + E_DATE_EDIT (priv->start_date), + &start_tt.year, + &start_tt.month, + &start_tt.day); if (date_set) { - if (time_set) { - icaltimezone *zone = e_timezone_entry_get_timezone (E_TIMEZONE_ENTRY (priv->start_timezone)); - start_dt.tzid = icaltimezone_get_tzid (zone); - } else { - start_tt.is_date = TRUE; - start_dt.tzid = NULL; - } + start_tt.is_date = TRUE; + start_dt.tzid = NULL; } else { start_tt = icaltime_null_time (); start_dt.tzid = NULL; } - date_set = e_date_edit_get_date (E_DATE_EDIT (priv->due_date), - &due_tt.year, - &due_tt.month, - &due_tt.day); - time_set = e_date_edit_get_time_of_day (E_DATE_EDIT (priv->due_date), - &due_tt.hour, - &due_tt.minute); + date_set = e_date_edit_get_date ( + E_DATE_EDIT (priv->due_date), + &due_tt.year, + &due_tt.month, + &due_tt.day); if (date_set) { - if (time_set) { - icaltimezone *zone = e_timezone_entry_get_timezone (E_TIMEZONE_ENTRY (priv->due_timezone)); - due_dt.tzid = icaltimezone_get_tzid (zone); - } else { - due_tt.is_date = TRUE; - due_dt.tzid = NULL; - } + due_tt.is_date = TRUE; + due_dt.tzid = NULL; } else { due_tt = icaltime_null_time (); due_dt.tzid = NULL; @@ -786,184 +1880,833 @@ date_changed_cb (EDateEdit *dedit, gpointer data) due_dt.value = &due_tt; dates.due = &due_dt; dates.complete = NULL; - + /* Notify upstream */ comp_editor_page_notify_dates_changed (COMP_EDITOR_PAGE (tpage), &dates); + + check_starts_in_the_past (tpage); +} + +static void +timezone_changed_cb (EDateEdit *dedit, + TaskPage *tpage) +{ + date_changed_cb ((EDateEdit *) tpage->priv->start_date, tpage); + date_changed_cb ((EDateEdit *) tpage->priv->due_date, tpage); } -/* Callback used when the contacts button is clicked; we must bring up the - * contact list dialog. +/* Callback used when the categories button is clicked; we must bring up the + * category list dialog. */ static void -contacts_clicked_cb (GtkWidget *button, gpointer data) +categories_clicked_cb (GtkWidget *button, + TaskPage *tpage) +{ + GtkEntry *entry; + + entry = GTK_ENTRY (tpage->priv->categories); + e_categories_config_open_dialog_for_entry (entry); +} + +static gboolean +check_start_before_end (struct icaltimetype *start_tt, + icaltimezone *start_zone, + struct icaltimetype *end_tt, + icaltimezone *end_zone, + gboolean adjust_end_time, + gboolean adjust_by_hour) +{ + struct icaltimetype end_tt_copy; + gint cmp; + + /* Convert the end time to the same timezone as the start time. */ + end_tt_copy = *end_tt; + icaltimezone_convert_time (&end_tt_copy, end_zone, start_zone); + + /* Now check if the start time is after the end time. If it is, + * we need to modify one of the times. */ + cmp = icaltime_compare (*start_tt, end_tt_copy); + if (cmp > 0) { + if (adjust_end_time) { + /* Modify the end time, to be the start + 1 hour/day. */ + *end_tt = *start_tt; + icaltime_adjust (end_tt, 0, adjust_by_hour ? 1 : 24, 0, 0); + icaltimezone_convert_time ( + end_tt, start_zone, + end_zone); + } else { + /* Modify the start time, to be the end - 1 hour/day. */ + *start_tt = *end_tt; + icaltime_adjust (start_tt, 0, adjust_by_hour ? -1 : -24, 0, 0); + icaltimezone_convert_time ( + start_tt, end_zone, + start_zone); + } + return TRUE; + } + + return FALSE; +} + +/* + * This is called whenever the start or due dates. + * It makes sure that the start date < end date. It also emits the notification + * signals so the other event editor pages update their labels etc. + * + * If adjust_end_time is TRUE, if the start time < end time it will adjust + * the end time. If FALSE it will adjust the start time. If the user sets the + * start or end time, the other time is adjusted to make it valid. + * + * Time part of the value is changed only when both edits have time set, + * otherwise times will differ one hour. + */ +static void +times_updated (TaskPage *tpage, + gboolean adjust_end_time) { - TaskPage *tpage; TaskPagePrivate *priv; + struct icaltimetype start_tt = icaltime_null_time (); + struct icaltimetype end_tt = icaltime_null_time (); + gboolean date_set; + gboolean set_start_date = FALSE, set_end_date = FALSE; + icaltimezone *zone; - tpage = TASK_PAGE (data); priv = tpage->priv; - comp_editor_show_contacts_dialog (priv->corba_select_names); + if (comp_editor_page_get_updating (COMP_EDITOR_PAGE (tpage))) + return; + + date_set = e_date_edit_get_date ( + E_DATE_EDIT (priv->start_date), + &start_tt.year, + &start_tt.month, + &start_tt.day); + if (!date_set) + return; + + date_set = e_date_edit_get_date ( + E_DATE_EDIT (priv->due_date), + &end_tt.year, + &end_tt.month, + &end_tt.day); + if (!date_set) + return; - /* FIXME: Currently we aren't getting the changed event from the - SelectNames component correctly, so we aren't saving the event - if just the contacts are changed. To work around that, we assume - that if the contacts button is clicked it is changed. */ - comp_editor_page_notify_changed (COMP_EDITOR_PAGE (tpage)); + zone = e_timezone_entry_get_timezone (E_TIMEZONE_ENTRY (priv->timezone)); + + if (check_start_before_end (&start_tt, zone, + &end_tt, zone, + adjust_end_time, + FALSE)) { + if (adjust_end_time) + set_end_date = TRUE; + else + set_start_date = TRUE; + } + + if (set_start_date) { + g_signal_handlers_block_matched (priv->start_date, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, tpage); + e_date_edit_set_date (E_DATE_EDIT (priv->start_date), start_tt.year, start_tt.month, start_tt.day); + g_signal_handlers_unblock_matched (priv->start_date, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, tpage); + } + + if (set_end_date) { + g_signal_handlers_block_matched (priv->due_date, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, tpage); + e_date_edit_set_date (E_DATE_EDIT (priv->due_date), end_tt.year, end_tt.month, end_tt.day); + g_signal_handlers_unblock_matched (priv->due_date, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, tpage); + } + + /* Notify upstream */ + date_changed_cb ((EDateEdit *) priv->start_date, tpage); + date_changed_cb ((EDateEdit *) priv->due_date, tpage); } -/* Callback used when the categories button is clicked; we must bring up the - * category list dialog. - */ static void -categories_clicked_cb (GtkWidget *button, gpointer data) +start_date_changed_cb (TaskPage *tpage) { - TaskPage *tpage; + times_updated (tpage, TRUE); +} + +static void +due_date_changed_cb (TaskPage *tpage) +{ + times_updated (tpage, FALSE); +} + +static void +tpage_get_client_cb (GObject *source_object, + GAsyncResult *result, + gpointer user_data) +{ + EClient *client; + EClientComboBox *combo_box; + TaskPage *tpage = user_data; TaskPagePrivate *priv; - GtkWidget *entry; + CompEditor *editor; + GError *error = NULL; + + combo_box = E_CLIENT_COMBO_BOX (source_object); + + client = e_client_combo_box_get_client_finish ( + combo_box, result, &error); + + /* Sanity check. */ + g_return_if_fail ( + ((client != NULL) && (error == NULL)) || + ((client == NULL) && (error != NULL))); - tpage = TASK_PAGE (data); + if (g_error_matches (error, E_CLIENT_ERROR, E_CLIENT_ERROR_CANCELLED) || + g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) { + g_clear_error (&error); + return; + } + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (tpage)); priv = tpage->priv; + if (error) { + GtkWidget *dialog; + ECalClient *old_client; + + old_client = comp_editor_get_client (editor); + + e_source_combo_box_set_active ( + E_SOURCE_COMBO_BOX (combo_box), + e_client_get_source (E_CLIENT (old_client))); + + dialog = gtk_message_dialog_new ( + NULL, GTK_DIALOG_MODAL, + GTK_MESSAGE_WARNING, GTK_BUTTONS_OK, + "%s", error->message); + gtk_dialog_run (GTK_DIALOG (dialog)); + gtk_widget_destroy (dialog); + + g_clear_error (&error); + } else { + icaltimezone *zone; + ECalClient *cal_client = E_CAL_CLIENT (client); + + g_return_if_fail (cal_client != NULL); + + zone = comp_editor_get_timezone (editor); + e_cal_client_set_default_timezone (cal_client, zone); + + comp_editor_set_client (editor, cal_client); + comp_editor_page_changed (COMP_EDITOR_PAGE (tpage)); + if (e_client_check_capability (client, CAL_STATIC_CAPABILITY_REQ_SEND_OPTIONS) && priv->is_assignment) + task_page_show_options (tpage); + else + task_page_hide_options (tpage); + + if (client) { + gchar *backend_addr = NULL; + + e_client_get_backend_property_sync (client, CAL_BACKEND_PROPERTY_CAL_EMAIL_ADDRESS, &backend_addr, NULL, NULL); + + if (priv->is_assignment) + task_page_select_organizer (tpage, backend_addr); + + set_subscriber_info_string (tpage, backend_addr); + g_free (backend_addr); + } - entry = priv->categories; - e_categories_config_open_dialog_for_entry (GTK_ENTRY (entry)); + sensitize_widgets (tpage); + } } -/* This is called when any field is changed; it notifies upstream. */ static void -field_changed_cb (GtkWidget *widget, gpointer data) +source_changed_cb (ESourceComboBox *combo_box, + TaskPage *tpage) +{ + TaskPagePrivate *priv = tpage->priv; + ESource *source; + + if (comp_editor_page_get_updating (COMP_EDITOR_PAGE (tpage))) + return; + + source = e_source_combo_box_ref_active (combo_box); + g_return_if_fail (source != NULL); + + if (priv->connect_cancellable != NULL) { + g_cancellable_cancel (priv->connect_cancellable); + g_object_unref (priv->connect_cancellable); + } + priv->connect_cancellable = g_cancellable_new (); + + e_client_combo_box_get_client ( + E_CLIENT_COMBO_BOX (combo_box), + source, priv->connect_cancellable, + tpage_get_client_cb, tpage); + + g_object_unref (source); +} + +static void +set_subscriber_info_string (TaskPage *tpage, + const gchar *backend_address) +{ + if (!check_starts_in_the_past (tpage)) + task_page_set_info_string (tpage, NULL, NULL); +} + +void +task_page_send_options_clicked_cb (TaskPage *tpage) +{ + TaskPagePrivate *priv = tpage->priv; + CompEditor *editor; + GtkWidget *toplevel; + ESource *source; + ECalClient *client; + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (tpage)); + client = comp_editor_get_client (editor); + + if (!priv->sod) { + priv->sod = e_send_options_dialog_new (); + priv->sod->data->initialized = TRUE; + source = e_source_combo_box_ref_active ( + E_SOURCE_COMBO_BOX (priv->client_combo_box)); + e_send_options_utils_set_default_data ( + priv->sod, source, "task"); + g_object_unref (source); + } + + if (e_client_check_capability (E_CLIENT (client), CAL_STATIC_CAPABILITY_NO_GEN_OPTIONS)) { + e_send_options_set_need_general_options (priv->sod, FALSE); + } + + toplevel = gtk_widget_get_toplevel (priv->main); + e_send_options_dialog_run (priv->sod, toplevel, E_ITEM_TASK); +} + +static void +complete_date_changed (TaskPage *tpage, + time_t ctime, + gboolean complete) +{ + CompEditorPageDates dates = {NULL, NULL, NULL, NULL}; + icaltimezone *zone; + struct icaltimetype completed_tt = icaltime_null_time (); + + /* Get the current time in UTC. */ + zone = icaltimezone_get_utc_timezone (); + completed_tt = icaltime_from_timet_with_zone (ctime, FALSE, zone); + completed_tt.is_utc = TRUE; + + dates.start = NULL; + dates.end = NULL; + dates.due = NULL; + if (complete) + dates.complete = &completed_tt; + + /* Notify upstream */ + comp_editor_page_notify_dates_changed (COMP_EDITOR_PAGE (tpage), + &dates); +} + +static void +completed_date_changed_cb (EDateEdit *dedit, + TaskPage *tpage) +{ + TaskPagePrivate *priv = tpage->priv; + CompEditorPageDates dates = {NULL, NULL, NULL, NULL}; + struct icaltimetype completed_tt = icaltime_null_time (); + icalproperty_status status; + gboolean date_set; + + if (comp_editor_page_get_updating (COMP_EDITOR_PAGE (tpage))) + return; + + comp_editor_page_set_updating (COMP_EDITOR_PAGE (tpage), TRUE); + + date_set = e_date_edit_get_date ( + E_DATE_EDIT (priv->completed_date), + &completed_tt.year, + &completed_tt.month, + &completed_tt.day); + e_date_edit_get_time_of_day ( + E_DATE_EDIT (priv->completed_date), + &completed_tt.hour, + &completed_tt.minute); + + status = e_dialog_combo_box_get (priv->status_combo, status_map); + + if (!date_set) { + completed_tt = icaltime_null_time (); + if (status == ICAL_STATUS_COMPLETED) { + e_dialog_combo_box_set ( + priv->status_combo, + ICAL_STATUS_NONE, + status_map); + gtk_spin_button_set_value ( + GTK_SPIN_BUTTON (priv->percent_complete), 0); + } + } else { + if (status != ICAL_STATUS_COMPLETED) { + e_dialog_combo_box_set ( + priv->status_combo, + ICAL_STATUS_COMPLETED, + status_map); + } + gtk_spin_button_set_value ( + GTK_SPIN_BUTTON (priv->percent_complete), 100); + } + + comp_editor_page_set_updating (COMP_EDITOR_PAGE (tpage), FALSE); + + /* Notify upstream */ + dates.complete = &completed_tt; + comp_editor_page_notify_dates_changed (COMP_EDITOR_PAGE (tpage), &dates); +} + +static void +status_changed (GtkWidget *combo, + TaskPage *tpage) { - TaskPage *tpage; TaskPagePrivate *priv; - - tpage = TASK_PAGE (data); + icalproperty_status status; + CompEditor *editor; + time_t ctime = -1; + priv = tpage->priv; - - if (!priv->updating) - comp_editor_page_notify_changed (COMP_EDITOR_PAGE (tpage)); + + if (comp_editor_page_get_updating (COMP_EDITOR_PAGE (tpage))) + return; + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (tpage)); + + comp_editor_page_set_updating (COMP_EDITOR_PAGE (tpage), TRUE); + + status = e_dialog_combo_box_get (priv->status_combo, status_map); + if (status == ICAL_STATUS_NONE) { + gtk_spin_button_set_value ( + GTK_SPIN_BUTTON (priv->percent_complete), 0); + e_date_edit_set_time (E_DATE_EDIT (priv->completed_date), ctime); + complete_date_changed (tpage, 0, FALSE); + } else if (status == ICAL_STATUS_INPROCESS) { + gint percent_complete = gtk_spin_button_get_value_as_int ( + GTK_SPIN_BUTTON (priv->percent_complete)); + if (percent_complete <= 0 || percent_complete >= 100) + gtk_spin_button_set_value ( + GTK_SPIN_BUTTON (priv->percent_complete), 50); + + e_date_edit_set_time (E_DATE_EDIT (priv->completed_date), ctime); + complete_date_changed (tpage, 0, FALSE); + } else if (status == ICAL_STATUS_COMPLETED) { + gtk_spin_button_set_value ( + GTK_SPIN_BUTTON (priv->percent_complete), 100); + ctime = time (NULL); + e_date_edit_set_time (E_DATE_EDIT (priv->completed_date), ctime); + complete_date_changed (tpage, ctime, TRUE); + } + + comp_editor_page_set_updating (COMP_EDITOR_PAGE (tpage), FALSE); + + comp_editor_set_changed (editor, TRUE); +} + +static void +percent_complete_changed (GtkAdjustment *adj, + TaskPage *tpage) +{ + TaskPagePrivate *priv; + gint percent; + icalproperty_status status; + CompEditor *editor; + gboolean complete; + time_t ctime = -1; + + priv = tpage->priv; + + if (comp_editor_page_get_updating (COMP_EDITOR_PAGE (tpage))) + return; + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (tpage)); + + comp_editor_page_set_updating (COMP_EDITOR_PAGE (tpage), TRUE); + + percent = gtk_spin_button_get_value_as_int ( + GTK_SPIN_BUTTON (priv->percent_complete)); + if (percent == 100) { + complete = TRUE; + ctime = time (NULL); + status = ICAL_STATUS_COMPLETED; + } else { + complete = FALSE; + + if (percent == 0) + status = ICAL_STATUS_NONE; + else + status = ICAL_STATUS_INPROCESS; + } + + e_dialog_combo_box_set (priv->status_combo, status, status_map); + e_date_edit_set_time (E_DATE_EDIT (priv->completed_date), ctime); + complete_date_changed (tpage, ctime, complete); + + comp_editor_page_set_updating (COMP_EDITOR_PAGE (tpage), FALSE); + + comp_editor_set_changed (editor, TRUE); +} + +static gboolean +task_page_transform_classification_to_combo (GBinding *binding, + const GValue *source_value, + GValue *target_value, + gpointer user_data) +{ + gint action_value; + + g_return_val_if_fail (source_value != NULL, FALSE); + g_return_val_if_fail (target_value != NULL, FALSE); + + action_value = g_value_get_int (source_value); + g_value_set_int (target_value, action_value - 1); + + return TRUE; +} + +static gboolean +task_page_transform_classification_from_combo (GBinding *binding, + const GValue *source_value, + GValue *target_value, + gpointer user_data) +{ + gint combo_value; + + g_return_val_if_fail (source_value != NULL, FALSE); + g_return_val_if_fail (target_value != NULL, FALSE); + + combo_value = g_value_get_int (source_value); + g_value_set_int (target_value, combo_value + 1); + + return TRUE; } /* Hooks the widget signals */ static gboolean init_widgets (TaskPage *tpage) { + CompEditor *editor; TaskPagePrivate *priv; - char *location; + GtkAction *action; + GtkTextBuffer *text_buffer; icaltimezone *zone; + gboolean active; + GtkAdjustment *adjustment; priv = tpage->priv; + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (tpage)); + /* Make sure the EDateEdit widgets use our timezones to get the - current time. */ - e_date_edit_set_get_time_callback (E_DATE_EDIT (priv->start_date), - (EDateEditGetTimeCallback) comp_editor_get_current_time, - tpage, NULL); - e_date_edit_set_get_time_callback (E_DATE_EDIT (priv->due_date), - (EDateEditGetTimeCallback) comp_editor_get_current_time, - tpage, NULL); - + * current time. */ + e_date_edit_set_get_time_callback ( + E_DATE_EDIT (priv->start_date), + (EDateEditGetTimeCallback) comp_editor_get_current_time, + g_object_ref (editor), + (GDestroyNotify) g_object_unref); + e_date_edit_set_get_time_callback ( + E_DATE_EDIT (priv->due_date), + (EDateEditGetTimeCallback) comp_editor_get_current_time, + g_object_ref (editor), + (GDestroyNotify) g_object_unref); + + /* Generic informative messages */ + gtk_widget_hide (priv->info_hbox); + /* Summary */ - gtk_signal_connect (GTK_OBJECT (priv->summary), "changed", - GTK_SIGNAL_FUNC (summary_changed_cb), tpage); + g_signal_connect ( + priv->summary, "changed", + G_CALLBACK (summary_changed_cb), tpage); + + /* Description */ + text_buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (priv->description)); - /* Description - turn on word wrap. */ - gtk_text_set_word_wrap (GTK_TEXT (priv->description), TRUE); + gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (priv->description), GTK_WRAP_WORD); + + e_buffer_tagger_connect (GTK_TEXT_VIEW (priv->description)); /* Dates */ - gtk_signal_connect (GTK_OBJECT (priv->start_date), "changed", - GTK_SIGNAL_FUNC (date_changed_cb), tpage); - gtk_signal_connect (GTK_OBJECT (priv->due_date), "changed", - GTK_SIGNAL_FUNC (date_changed_cb), tpage); + g_signal_connect ( + priv->start_date, "changed", + G_CALLBACK (date_changed_cb), tpage); + g_signal_connect ( + priv->due_date, "changed", + G_CALLBACK (date_changed_cb), tpage); + + /* time zone changed */ + g_signal_connect ( + priv->timezone, "changed", + G_CALLBACK (timezone_changed_cb), tpage); - gtk_signal_connect (GTK_OBJECT (priv->due_timezone), "changed", - GTK_SIGNAL_FUNC (field_changed_cb), tpage); - gtk_signal_connect (GTK_OBJECT (priv->start_timezone), "changed", - GTK_SIGNAL_FUNC (field_changed_cb), tpage); + /* Categories button */ + g_signal_connect ( + priv->categories_btn, "clicked", + G_CALLBACK (categories_clicked_cb), tpage); - /* Classification */ - gtk_signal_connect (GTK_OBJECT (priv->classification_public), - "toggled", - GTK_SIGNAL_FUNC (field_changed_cb), tpage); - gtk_signal_connect (GTK_OBJECT (priv->classification_private), - "toggled", - GTK_SIGNAL_FUNC (field_changed_cb), tpage); - gtk_signal_connect (GTK_OBJECT (priv->classification_confidential), - "toggled", - GTK_SIGNAL_FUNC (field_changed_cb), tpage); + /* Source selector */ + g_signal_connect ( + priv->client_combo_box, "changed", + G_CALLBACK (source_changed_cb), tpage); /* Connect the default signal handler to use to make sure the "changed" - field gets set whenever a field is changed. */ - gtk_signal_connect (GTK_OBJECT (priv->description), "changed", - GTK_SIGNAL_FUNC (field_changed_cb), tpage); - gtk_signal_connect (GTK_OBJECT (priv->categories), "changed", - GTK_SIGNAL_FUNC (field_changed_cb), tpage); + * field gets set whenever a field is changed. */ + + /* Belongs to priv->description */ + g_signal_connect_swapped ( + text_buffer, "changed", + G_CALLBACK (comp_editor_page_changed), tpage); + g_signal_connect_swapped ( + priv->summary, "changed", + G_CALLBACK (comp_editor_page_changed), tpage); + g_signal_connect_swapped ( + priv->start_date, "changed", + G_CALLBACK (start_date_changed_cb), tpage); + g_signal_connect_swapped ( + priv->start_date, "changed", + G_CALLBACK (comp_editor_page_changed), tpage); + g_signal_connect_swapped ( + priv->due_date, "changed", + G_CALLBACK (due_date_changed_cb), tpage); + g_signal_connect_swapped ( + priv->due_date, "changed", + G_CALLBACK (comp_editor_page_changed), tpage); + g_signal_connect_swapped ( + priv->timezone, "changed", + G_CALLBACK (comp_editor_page_changed), tpage); + g_signal_connect_swapped ( + priv->categories, "changed", + G_CALLBACK (comp_editor_page_changed), tpage); + + g_signal_connect ( + priv->list_view, "event", + G_CALLBACK (list_view_event), tpage); + g_signal_connect ( + priv->list_view, "key_press_event", + G_CALLBACK (list_key_press), tpage); + + /* Add attendee button */ + g_signal_connect ( + priv->add, "clicked", + G_CALLBACK (add_clicked_cb), tpage); + + /* Edit attendee button */ + g_signal_connect ( + priv->edit, "clicked", + G_CALLBACK (edit_clicked_cb), tpage); + + /* Remove attendee button */ + g_signal_connect ( + priv->remove, "clicked", + G_CALLBACK (remove_clicked_cb), tpage); /* Contacts button */ - gtk_signal_connect (GTK_OBJECT (priv->contacts_btn), "clicked", - GTK_SIGNAL_FUNC (contacts_clicked_cb), tpage); - - /* Categories button */ - gtk_signal_connect (GTK_OBJECT (priv->categories_btn), "clicked", - GTK_SIGNAL_FUNC (categories_clicked_cb), tpage); + g_signal_connect ( + priv->invite, "clicked", + G_CALLBACK (invite_cb), tpage); + /* Meeting List View */ + g_signal_connect ( + priv->list_view, "attendee_added", + G_CALLBACK (attendee_added_cb), tpage); - /* Create the contacts entry, a corba control from the address book. */ - priv->corba_select_names = comp_editor_create_contacts_component (); - if (priv->corba_select_names == CORBA_OBJECT_NIL) - return FALSE; - - priv->contacts_entry = comp_editor_create_contacts_control (priv->corba_select_names); - if (priv->contacts_entry == NULL) - return FALSE; + /* Set the default timezone, so the timezone entry may be hidden. */ + zone = comp_editor_get_timezone (editor); + e_timezone_entry_set_default_timezone (E_TIMEZONE_ENTRY (priv->timezone), zone); - gtk_container_add (GTK_CONTAINER (priv->contacts_box), - priv->contacts_entry); + /* Make sure the EDateEdit widgets use our timezones to get the + * current time. */ + e_date_edit_set_get_time_callback ( + E_DATE_EDIT (priv->completed_date), + (EDateEditGetTimeCallback) comp_editor_get_current_time, + g_object_ref (editor), + (GDestroyNotify) g_object_unref); + + /* Connect signals. The Status, Percent Complete & Date Completed + * properties are closely related so whenever one changes we may need + * to update the other 2. */ + g_signal_connect ( + GTK_COMBO_BOX (priv->status_combo), "changed", + G_CALLBACK (status_changed), tpage); + + adjustment = gtk_spin_button_get_adjustment ( + GTK_SPIN_BUTTON (priv->percent_complete)); + g_signal_connect ( + adjustment, "value_changed", + G_CALLBACK (percent_complete_changed), tpage); + + /* Priority */ + g_signal_connect_swapped ( + GTK_COMBO_BOX (priv->priority_combo), "changed", + G_CALLBACK (comp_editor_page_changed), tpage); + + /* Completed Date */ + g_signal_connect ( + priv->completed_date, "changed", + G_CALLBACK (completed_date_changed_cb), tpage); + g_signal_connect_swapped ( + priv->completed_date, "changed", + G_CALLBACK (comp_editor_page_changed), tpage); + + /* URL */ + g_signal_connect_swapped ( + priv->web_page_entry, "changed", + G_CALLBACK (comp_editor_page_changed), tpage); + + action = comp_editor_get_action (editor, "view-time-zone"); + active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)); + task_page_set_show_timezone (tpage, active); + + e_meeting_list_view_column_set_visible ( + priv->list_view, E_MEETING_STORE_ATTENDEE_COL, TRUE); + + action = comp_editor_get_action (editor, "view-role"); + active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)); + e_meeting_list_view_column_set_visible ( + priv->list_view, E_MEETING_STORE_ROLE_COL, active); + + action = comp_editor_get_action (editor, "view-rsvp"); + active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)); + e_meeting_list_view_column_set_visible ( + priv->list_view, E_MEETING_STORE_RSVP_COL, active); + + action = comp_editor_get_action (editor, "view-status"); + active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)); + e_meeting_list_view_column_set_visible ( + priv->list_view, E_MEETING_STORE_STATUS_COL, active); + + action = comp_editor_get_action (editor, "view-type"); + active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)); + e_meeting_list_view_column_set_visible ( + priv->list_view, E_MEETING_STORE_TYPE_COL, active); + + action = comp_editor_get_action (editor, "view-categories"); + active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)); + task_page_set_show_categories (tpage, active); - /* Set the default timezone, so the timezone entry may be hidden. */ - location = calendar_config_get_timezone (); - zone = icaltimezone_get_builtin_timezone (location); - e_timezone_entry_set_default_timezone (E_TIMEZONE_ENTRY (priv->start_timezone), zone); - e_timezone_entry_set_default_timezone (E_TIMEZONE_ENTRY (priv->due_timezone), zone); + /* Classification */ + action = comp_editor_get_action (editor, "classify-public"); + g_object_bind_property_full ( + action, "current-value", + priv->classification_combo, "active", + G_BINDING_BIDIRECTIONAL | G_BINDING_SYNC_CREATE, + task_page_transform_classification_to_combo, + task_page_transform_classification_from_combo, + NULL, NULL); return TRUE; } - +static void +task_page_select_organizer (TaskPage *tpage, + const gchar *backend_address) +{ + TaskPagePrivate *priv = tpage->priv; + const gchar *default_address; + gint ii; + + /* Treat an empty backend address as NULL. */ + if (backend_address != NULL && *backend_address == '\0') + backend_address = NULL; + + default_address = priv->fallback_address; + + if (backend_address != NULL) { + for (ii = 0; priv->address_strings[ii] != NULL; ii++) { + if (g_strrstr (priv->address_strings[ii], backend_address) != NULL) { + default_address = priv->address_strings[ii]; + break; + } + } + } + + if (default_address != NULL) { + if (!priv->comp || !e_cal_component_has_organizer (priv->comp)) { + GtkEntry *entry = GTK_ENTRY (gtk_bin_get_child (GTK_BIN (priv->organizer))); + + g_signal_handlers_block_by_func (entry, organizer_changed_cb, tpage); + gtk_entry_set_text (entry, default_address); + g_signal_handlers_unblock_by_func (entry, organizer_changed_cb, tpage); + } + } else + g_warning ("No potential organizers!"); +} /** * task_page_construct: * @tpage: An task page. - * + * * Constructs an task page by loading its Glade data. - * + * * Return value: The same object as @tpage, or NULL if the widgets could not be * created. **/ TaskPage * -task_page_construct (TaskPage *tpage) +task_page_construct (TaskPage *tpage, + EMeetingStore *meeting_store, + ECalClient *client) { + EShell *shell; + CompEditor *editor; + ESourceRegistry *registry; TaskPagePrivate *priv; + GtkComboBox *combo_box; + GtkListStore *list_store; + GtkTreeModel *model; + GtkTreeIter iter; + gint ii; + + editor = comp_editor_page_get_editor (COMP_EDITOR_PAGE (tpage)); + shell = comp_editor_get_shell (editor); priv = tpage->priv; + priv->meeting_store = g_object_ref (meeting_store); + priv->client = client; - priv->xml = glade_xml_new (EVOLUTION_GLADEDIR "/task-page.glade", - NULL); - if (!priv->xml) { - g_message ("task_page_construct(): " - "Could not load the Glade XML file!"); - return NULL; - } + /* Make sure our custom widget classes are registered with + * GType before we load the GtkBuilder definition file. */ + g_type_ensure (E_TYPE_DATE_EDIT); + g_type_ensure (E_TYPE_TIMEZONE_ENTRY); + g_type_ensure (E_TYPE_SOURCE_COMBO_BOX); + g_type_ensure (E_TYPE_SPELL_ENTRY); + + priv->builder = gtk_builder_new (); + e_load_ui_builder_definition (priv->builder, "task-page.ui"); if (!get_widgets (tpage)) { - g_message ("task_page_construct(): " - "Could not find all widgets in the XML file!"); + g_message ( + "task_page_construct(): " + "Could not find all widgets in the XML file!"); return NULL; } + combo_box = GTK_COMBO_BOX (priv->organizer); + model = gtk_combo_box_get_model (combo_box); + list_store = GTK_LIST_STORE (model); + + registry = e_shell_get_registry (shell); + priv->address_strings = itip_get_user_identities (registry); + priv->fallback_address = itip_get_fallback_identity (registry); + + /* FIXME Could we just use a GtkComboBoxText? */ + for (ii = 0; priv->address_strings[ii] != NULL; ii++) { + gtk_list_store_append (list_store, &iter); + gtk_list_store_set ( + list_store, &iter, + 0, priv->address_strings[ii], -1); + } + + gtk_combo_box_set_active (combo_box, 0); + + g_signal_connect ( + gtk_bin_get_child (GTK_BIN (priv->organizer)), "changed", + G_CALLBACK (organizer_changed_cb), tpage); + if (!init_widgets (tpage)) { - g_message ("event_page_construct(): " - "Could not initialize the widgets!"); + g_message ( + "task_page_construct(): " + "Could not initialize the widgets!"); return NULL; } @@ -972,35 +2715,43 @@ task_page_construct (TaskPage *tpage) /** * task_page_new: - * + * * Creates a new task page. - * + * * Return value: A newly-created task page, or NULL if the page could * not be created. **/ TaskPage * -task_page_new (void) +task_page_new (EMeetingStore *model, + CompEditor *editor) { TaskPage *tpage; + ECalClient *client; - tpage = gtk_type_new (TYPE_TASK_PAGE); - if (!task_page_construct (tpage)) { - gtk_object_unref (GTK_OBJECT (tpage)); - return NULL; + tpage = g_object_new (TYPE_TASK_PAGE, "editor", editor, NULL); + client = comp_editor_get_client (editor); + if (!task_page_construct (tpage, model, client)) { + g_object_unref (tpage); + g_return_val_if_reached (NULL); } return tpage; } -GtkWidget *task_page_create_date_edit (void); - -GtkWidget * -task_page_create_date_edit (void) +ECalComponent * +task_page_get_cancel_comp (TaskPage *page) { - GtkWidget *dedit; + TaskPagePrivate *priv; + + g_return_val_if_fail (page != NULL, NULL); + g_return_val_if_fail (IS_TASK_PAGE (page), NULL); + + priv = page->priv; + + if (priv->deleted_attendees->len == 0) + return NULL; - dedit = comp_editor_new_date_edit (TRUE, TRUE, TRUE); - e_date_edit_set_allow_no_date_set (E_DATE_EDIT (dedit), TRUE); + set_attendees (priv->comp, priv->deleted_attendees); - return dedit; + return e_cal_component_clone (priv->comp); } |