/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* Evolution calendar - Main page of the task editor dialog * * Copyright (C) 2001 Ximian, Inc. * * Authors: Federico Mena-Quintero * Miguel de Icaza * Seth Alves * JP Rosevear * * This program is free software; you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA. */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../calendar-component.h" #include "../e-meeting-attendee.h" #include "../e-meeting-store.h" #include "../e-meeting-list-view.h" #include "../itip-utils.h" #include "comp-editor-util.h" #include "e-delegate-dialog.h" #include "meeting-page.h" #include "../e-cal-popup.h" /* Private part of the MeetingPage structure */ struct _MeetingPagePrivate { /* Lists of attendees */ GPtrArray *deleted_attendees; /* To use in case of cancellation */ ECalComponent *comp; /* List of identities */ EAccountList *accounts; EMeetingAttendee *ia; char *default_address; char *user_add; /* Glade XML data */ GladeXML *xml; /* Widgets from the Glade file */ GtkWidget *main; GtkWidget *list_box; GtkWidget *organizer_table; GtkWidget *organizer; GtkWidget *existing_organizer_table; GtkWidget *existing_organizer; GtkWidget *existing_organizer_btn; GtkWidget *add; GtkWidget *remove; GtkWidget *invite; GtkWidget *att_label; GtkWidget *org_label; /* ListView stuff */ EMeetingStore *model; EMeetingListView *list_view; gint row; /* For handling who the organizer is */ gboolean user_org; gboolean existing; gboolean updating; }; static void meeting_page_finalize (GObject *object); static GtkWidget *meeting_page_get_widget (CompEditorPage *page); static void meeting_page_focus_main_widget (CompEditorPage *page); static gboolean meeting_page_fill_widgets (CompEditorPage *page, ECalComponent *comp); static gboolean meeting_page_fill_component (CompEditorPage *page, ECalComponent *comp); static void attendee_added_cb (EMeetingListView *emlv, EMeetingAttendee *attendee, gpointer user_data); G_DEFINE_TYPE (MeetingPage, meeting_page, TYPE_COMP_EDITOR_PAGE) /* Class initialization function for the task page */ static void meeting_page_class_init (MeetingPageClass *class) { CompEditorPageClass *editor_page_class; GObjectClass *object_class; editor_page_class = (CompEditorPageClass *) class; object_class = (GObjectClass *) class; editor_page_class->get_widget = meeting_page_get_widget; editor_page_class->focus_main_widget = meeting_page_focus_main_widget; editor_page_class->fill_widgets = meeting_page_fill_widgets; editor_page_class->fill_component = meeting_page_fill_component; editor_page_class->set_summary = NULL; editor_page_class->set_dates = NULL; object_class->finalize = meeting_page_finalize; } /* Object initialization function for the task page */ static void meeting_page_init (MeetingPage *mpage) { MeetingPagePrivate *priv; priv = g_new0 (MeetingPagePrivate, 1); mpage->priv = priv; priv->deleted_attendees = g_ptr_array_new (); priv->comp = NULL; priv->accounts = NULL; priv->ia = NULL; priv->default_address = NULL; priv->xml = NULL; priv->main = NULL; priv->invite = NULL; priv->model = NULL; priv->list_view = NULL; priv->updating = FALSE; } static EAccount * get_current_account (MeetingPage *mpage) { MeetingPagePrivate *priv; EIterator *it; const char *str; priv = mpage->priv; str = gtk_entry_get_text (GTK_ENTRY (GTK_COMBO (priv->organizer)->entry)); if (!str) return NULL; for (it = e_list_get_iterator((EList *)priv->accounts); e_iterator_is_valid(it); e_iterator_next(it)) { EAccount *a = (EAccount *)e_iterator_get(it); char *full = g_strdup_printf("%s <%s>", a->id->name, a->id->address); if (!strcmp (full, str)) { g_free (full); g_object_unref (it); return a; } g_free (full); } g_object_unref (it); return NULL; } static void set_attendees (ECalComponent *comp, const GPtrArray *attendees) { GSList *comp_attendees = NULL, *l; int i; 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); 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 cleanup_attendees (GPtrArray *attendees) { int i; for (i = 0; i < attendees->len; i++) g_object_unref (g_ptr_array_index (attendees, i)); } /* Destroy handler for the task page */ static void meeting_page_finalize (GObject *object) { MeetingPage *mpage; MeetingPagePrivate *priv; g_return_if_fail (object != NULL); g_return_if_fail (IS_MEETING_PAGE (object)); mpage = MEETING_PAGE (object); priv = mpage->priv; if (priv->comp != NULL) g_object_unref (priv->comp); cleanup_attendees (priv->deleted_attendees); g_ptr_array_free (priv->deleted_attendees, TRUE); if (priv->ia != NULL) g_object_unref (priv->ia); g_object_unref (priv->model); if (priv->main) gtk_widget_unref (priv->main); if (priv->xml) { g_object_unref (priv->xml); priv->xml = NULL; } if (priv->default_address) { g_free (priv->default_address); priv->default_address = NULL; } if (priv->user_add) { g_free (priv->user_add); priv->user_add = NULL; } g_free (priv); mpage->priv = NULL; if (G_OBJECT_CLASS (meeting_page_parent_class)->finalize) (* G_OBJECT_CLASS (meeting_page_parent_class)->finalize) (object); } /* get_widget handler for the task page */ static GtkWidget * meeting_page_get_widget (CompEditorPage *page) { MeetingPage *mpage; MeetingPagePrivate *priv; mpage = MEETING_PAGE (page); priv = mpage->priv; return priv->main; } /* focus_main_widget handler for the task page */ static void meeting_page_focus_main_widget (CompEditorPage *page) { MeetingPage *mpage; MeetingPagePrivate *priv; mpage = MEETING_PAGE (page); priv = mpage->priv; gtk_widget_grab_focus (priv->organizer); } /* Fills the widgets with default values */ static void clear_widgets (MeetingPage *mpage) { MeetingPagePrivate *priv; gchar *str; priv = mpage->priv; if (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE) { str = g_strdup_printf ("%s", _("Dele_gatees")); gtk_label_set_markup_with_mnemonic ((GtkLabel *)priv->att_label, str); g_free (str); } if (e_cal_get_static_capability (COMP_EDITOR_PAGE (mpage)->client, CAL_STATIC_CAPABILITY_NO_ORGANIZER)) { str = g_strdup_printf ("%s", _("From:")); gtk_label_set_markup (GTK_LABEL (priv->org_label), str); g_free (str); gtk_widget_hide (priv->existing_organizer_btn); } gtk_entry_set_text (GTK_ENTRY (GTK_COMBO (priv->organizer)->entry), priv->default_address); gtk_label_set_text (GTK_LABEL (priv->existing_organizer), _("None")); gtk_widget_show (priv->organizer_table); gtk_widget_hide (priv->existing_organizer_table); priv->existing = FALSE; priv->user_org = TRUE; } static void sensitize_widgets (MeetingPage *mpage) { gboolean read_only = FALSE; MeetingPagePrivate *priv = mpage->priv; GError *error = NULL; guint32 flags; gboolean delegate; flags = COMP_EDITOR_PAGE (mpage)->flags; delegate = flags & COMP_EDITOR_PAGE_DELEGATE; if (!e_cal_is_read_only (COMP_EDITOR_PAGE (mpage)->client, &read_only, &error)) { if (error->code != E_CALENDAR_STATUS_BUSY) read_only = TRUE; g_error_free (error); } gtk_widget_set_sensitive (priv->organizer, !read_only); gtk_widget_set_sensitive (priv->existing_organizer_btn, (!read_only && priv->user_org)); gtk_widget_set_sensitive (priv->add, (!read_only && priv->user_org) || delegate); gtk_widget_set_sensitive (priv->remove, (!read_only && priv->user_org) || delegate); gtk_widget_set_sensitive (priv->invite, (!read_only && priv->user_org) || delegate); gtk_widget_set_sensitive (GTK_WIDGET (priv->list_view), !read_only); } /* fill_widgets handler for the meeting page */ static gboolean meeting_page_fill_widgets (CompEditorPage *page, ECalComponent *comp) { MeetingPage *mpage; MeetingPagePrivate *priv; ECalComponentOrganizer organizer; mpage = MEETING_PAGE (page); priv = mpage->priv; priv->updating = TRUE; /* Clean out old data */ if (priv->comp != NULL) g_object_unref (priv->comp); priv->comp = NULL; cleanup_attendees (priv->deleted_attendees); g_ptr_array_set_size (priv->deleted_attendees, 0); /* Clean the screen */ clear_widgets (mpage); /* Component for cancellation */ priv->comp = e_cal_component_clone (comp); priv->user_add = itip_get_comp_attendee (comp, COMP_EDITOR_PAGE (mpage)->client); /* 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; gtk_widget_hide (priv->organizer_table); gtk_widget_show (priv->existing_organizer_table); if (itip_organizer_is_user (comp, page->client)) { if (e_cal_get_static_capability ( page->client, CAL_STATIC_CAPABILITY_ORGANIZER_NOT_EMAIL_ADDRESS)) gtk_widget_hide (priv->existing_organizer_btn); priv->user_org = TRUE; } else { if (e_cal_get_static_capability ( page->client, CAL_STATIC_CAPABILITY_ORGANIZER_NOT_EMAIL_ADDRESS)) gtk_widget_hide (priv->existing_organizer_btn); gtk_widget_set_sensitive (priv->invite, FALSE); gtk_widget_set_sensitive (priv->add, FALSE); gtk_widget_set_sensitive (priv->remove, FALSE); priv->user_org = FALSE; } if (e_cal_get_static_capability (COMP_EDITOR_PAGE (mpage)->client, CAL_STATIC_CAPABILITY_NO_ORGANIZER) && (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_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); gtk_label_set_text (GTK_LABEL (priv->existing_organizer), string); g_free (string); priv->existing = TRUE; } } else { EAccount *a; a = get_current_account (mpage); if (a != NULL) { priv->ia = e_meeting_store_add_attendee_with_defaults (priv->model); g_object_ref (priv->ia); e_meeting_attendee_set_address (priv->ia, g_strdup_printf ("MAILTO:%s", a->id->address)); e_meeting_attendee_set_cn (priv->ia, g_strdup (a->id->name)); if (page->client && e_cal_get_organizer_must_accept (page->client)) e_meeting_attendee_set_status (priv->ia, ICAL_PARTSTAT_NEEDSACTION); else e_meeting_attendee_set_status (priv->ia, ICAL_PARTSTAT_ACCEPTED); } } priv->updating = FALSE; priv->user_org = page->flags & COMP_EDITOR_PAGE_USER_ORG; sensitize_widgets (mpage); return TRUE; } /* fill_component handler for the meeting page */ static gboolean meeting_page_fill_component (CompEditorPage *page, ECalComponent *comp) { MeetingPage *mpage; MeetingPagePrivate *priv; ECalComponentOrganizer organizer = {NULL, NULL, NULL, NULL}; mpage = MEETING_PAGE (page); priv = mpage->priv; if (!priv->existing) { EAccount *a; gchar *addr = NULL; /* Find the identity for the organizer or sentby field */ a = get_current_account (mpage); /* Sanity Check */ if (a == NULL) { e_notice (page, GTK_MESSAGE_ERROR, _("The organizer selected no longer has an account.")); return FALSE; } if (a->id->address == NULL || strlen (a->id->address) == 0) { e_notice (page, GTK_MESSAGE_ERROR, _("An organizer is required.")); return FALSE; } addr = g_strdup_printf ("MAILTO:%s", a->id->address); organizer.value = addr; organizer.cn = a->id->name; e_cal_component_set_organizer (comp, &organizer); g_free (addr); } if (e_meeting_store_count_actual_attendees (priv->model) < 1) { e_notice (page, GTK_MESSAGE_ERROR, _("At least one attendee is required.")); return FALSE; } if (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE ) { GSList *attendee_list, *l; int i; const GPtrArray *attendees = e_meeting_store_get_attendees (priv->model); 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_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->model)); return TRUE; } /* Gets the widgets from the XML file and returns if they are all available. */ static gboolean get_widgets (MeetingPage *mpage) { CompEditorPage *page = COMP_EDITOR_PAGE (mpage); MeetingPagePrivate *priv; GSList *accel_groups; GtkWidget *toplevel; priv = mpage->priv; #define GW(name) glade_xml_get_widget (priv->xml, name) priv->main = GW ("meeting-page"); if (!priv->main) return FALSE; priv->list_box = GW ("list-box"); /* Get the GtkAccelGroup from the toplevel window, so we can install it when the notebook page is mapped. */ toplevel = gtk_widget_get_toplevel (priv->main); accel_groups = gtk_accel_groups_from_object (G_OBJECT (toplevel)); if (accel_groups) { page->accel_group = accel_groups->data; gtk_accel_group_ref (page->accel_group); } gtk_widget_ref (priv->main); gtk_container_remove (GTK_CONTAINER (priv->main->parent), priv->main); /* For making the user the organizer */ priv->organizer_table = GW ("organizer-table"); priv->organizer = GW ("organizer"); gtk_combo_set_value_in_list (GTK_COMBO (priv->organizer), FALSE, FALSE); gtk_entry_set_editable (GTK_ENTRY (GTK_COMBO (priv->organizer)->entry), FALSE); /* For showing existing organizers */ priv->existing_organizer_table = GW ("existing-organizer-table"); priv->existing_organizer = GW ("existing-organizer"); priv->existing_organizer_btn = GW ("existing-organizer-button"); /* Buttons */ priv->add = GW ("add-attendee"); priv->remove = GW ("remove-attendee"); priv->invite = GW ("invite"); /* Attendees Label */ priv->att_label = GW ("attendees-label"); priv->org_label = GW ("org-label"); #undef GW return (priv->list_box && priv->att_label && priv->invite && priv->add && priv->remove && priv->organizer_table && priv->organizer && priv->existing_organizer_table && priv->existing_organizer && priv->existing_organizer_btn); } static void org_changed_cb (GtkWidget *widget, gpointer data) { MeetingPage *mpage; MeetingPagePrivate *priv; mpage = MEETING_PAGE (data); priv = mpage->priv; if (priv->updating) return; if (!priv->existing && priv->ia != NULL) { EAccount *a; a = get_current_account (mpage); if (a != NULL) { e_meeting_attendee_set_address (priv->ia, g_strdup_printf ("MAILTO:%s", a->id->address)); e_meeting_attendee_set_cn (priv->ia, g_strdup (a->id->name)); if (!e_meeting_store_find_attendee (priv->model, e_meeting_attendee_get_address (priv->ia), NULL)) e_meeting_store_add_attendee (priv->model, priv->ia); } else { e_meeting_store_remove_attendee (priv->model, priv->ia); } } comp_editor_page_notify_changed (COMP_EDITOR_PAGE (mpage)); } /* Function called to change the organizer */ static void change_clicked_cb (GtkWidget *widget, gpointer data) { MeetingPage *mpage; MeetingPagePrivate *priv; mpage = MEETING_PAGE (data); priv = mpage->priv; gtk_widget_show (priv->organizer_table); gtk_widget_hide (priv->existing_organizer_table); gtk_widget_set_sensitive (priv->invite, TRUE); gtk_widget_set_sensitive (priv->add, TRUE); gtk_widget_set_sensitive (priv->remove, TRUE); comp_editor_page_notify_changed (COMP_EDITOR_PAGE (mpage)); priv->existing = FALSE; priv->user_org = TRUE; } static void add_clicked_cb (GtkButton *btn, MeetingPage *mpage) { EMeetingAttendee *attendee; attendee = e_meeting_store_add_attendee_with_defaults (mpage->priv->model); if (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE) { e_meeting_attendee_set_delfrom (attendee, g_strdup_printf ("MAILTO:%s", mpage->priv->user_add)); } e_meeting_list_view_edit (mpage->priv->list_view, attendee); } static gboolean existing_attendee (EMeetingAttendee *ia, ECalComponent *comp) { GSList *attendees, *l; const gchar *ia_address; ia_address = itip_strip_mailto (e_meeting_attendee_get_address (ia)); if (!ia_address) return FALSE; e_cal_component_get_attendee_list (comp, &attendees); for (l = attendees; l; l = l->next) { ECalComponentAttendee *attendee = l->data; const char *address; address = itip_strip_mailto (attendee->value); if (address && !g_ascii_strcasecmp (ia_address, address)) { e_cal_component_free_attendee_list (attendees); return TRUE; } } e_cal_component_free_attendee_list (attendees); return FALSE; } static void remove_attendee (MeetingPage *mpage, EMeetingAttendee *ia) { MeetingPagePrivate *priv; int pos = 0; gboolean delegate = (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE); priv = mpage->priv; /* 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->model, e_meeting_attendee_get_delfrom (ia), &pos); if (ib != NULL) { e_meeting_attendee_set_delto (ib, NULL); if (!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)) { 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->model, e_meeting_attendee_get_delto (ia), NULL); e_meeting_store_remove_attendee (priv->model, ia); ia = ib; } sensitize_widgets (mpage); } static void remove_clicked_cb (GtkButton *btn, MeetingPage *mpage) { MeetingPagePrivate *priv; EMeetingAttendee *ia; GtkTreeSelection *selection; GList *paths = NULL, *tmp; GtkTreeIter iter; GtkTreePath *path = NULL; gboolean valid_iter; char *address; priv = mpage->priv; selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (priv->list_view)); if (!(paths = gtk_tree_selection_get_selected_rows (selection, &(priv->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->model), &iter, path); gtk_tree_model_get (GTK_TREE_MODEL (priv->model), &iter, E_MEETING_STORE_ADDRESS_COL, &address, -1); ia = e_meeting_store_find_attendee (priv->model, 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 (mpage, ia); } /* Select closest item after removal */ valid_iter = gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->model), &iter, path); if (!valid_iter) { gtk_tree_path_prev (path); valid_iter = gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->model), &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); } /* Function called to invite more people */ static void invite_cb (GtkWidget *widget, gpointer data) { MeetingPage *mpage; MeetingPagePrivate *priv; mpage = MEETING_PAGE (data); priv = mpage->priv; e_meeting_list_view_invite_others_dialog (priv->list_view); } /* Hooks the widget signals */ static void init_widgets (MeetingPage *mpage) { MeetingPagePrivate *priv; priv = mpage->priv; /* Organizer */ g_signal_connect (GTK_COMBO (priv->organizer)->entry, "changed", G_CALLBACK (org_changed_cb), mpage); g_signal_connect (priv->existing_organizer_btn, "clicked", G_CALLBACK (change_clicked_cb), mpage); /* Add attendee button */ g_signal_connect (priv->add, "clicked", G_CALLBACK (add_clicked_cb), mpage); /* Remove attendee button */ g_signal_connect (priv->remove, "clicked", G_CALLBACK (remove_clicked_cb), mpage); /* Contacts button */ g_signal_connect(priv->invite, "clicked", G_CALLBACK (invite_cb), mpage); /* Meeting List View */ g_signal_connect (priv->list_view, "attendee_added", G_CALLBACK (attendee_added_cb), mpage); } static void attendee_added_cb (EMeetingListView *emlv, EMeetingAttendee *ia, gpointer user_data) { MeetingPage *mpage = MEETING_PAGE (user_data); MeetingPagePrivate *priv; gboolean delegate = (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE); priv = mpage->priv; if (delegate) { if (existing_attendee (ia, priv->comp)) e_meeting_store_remove_attendee (priv->model, ia); else { if (!e_cal_get_static_capability (COMP_EDITOR_PAGE(mpage)->client, CAL_STATIC_CAPABILITY_DELEGATE_TO_MANY)) { const char *delegator_id = e_meeting_attendee_get_delfrom (ia); EMeetingAttendee *delegator; delegator = e_meeting_store_find_attendee (priv->model, delegator_id, NULL); e_meeting_attendee_set_delto (delegator, g_strdup (e_meeting_attendee_get_address (ia))); gtk_widget_set_sensitive (priv->invite, FALSE); gtk_widget_set_sensitive (priv->add, FALSE); } } } } static void client_changed_cb (CompEditorPage *page, ECal *client, gpointer user_data) { MeetingPage *mpage = MEETING_PAGE (page); sensitize_widgets (mpage); } static void popup_add_cb (EPopup *ep, EPopupItem *pitem, void *data) { MeetingPage *mpage = data; add_clicked_cb (NULL, mpage); } static void popup_delete_cb (EPopup *ep, EPopupItem *pitem, void *data) { MeetingPage *mpage = data; remove_clicked_cb (NULL, mpage); } enum { ATTENDEE_CAN_DELEGATE = 1<<1, ATTENDEE_CAN_DELETE = 1<<2, ATTENDEE_CAN_ADD = 1<<3, ATTENDEE_LAST = 1<<4, }; static EPopupItem context_menu_items[] = { { E_POPUP_ITEM, "10.delete", N_("_Remove"), popup_delete_cb, NULL, GTK_STOCK_REMOVE, ATTENDEE_CAN_DELETE }, { E_POPUP_ITEM, "15.add", N_("_Add "), popup_add_cb, NULL, GTK_STOCK_ADD }, }; static void context_popup_free(EPopup *ep, GSList *items, void *data) { g_slist_free(items); } static gint button_press_event (GtkWidget *widget, GdkEventButton *event, MeetingPage *mpage) { MeetingPagePrivate *priv; GtkMenu *menu; EMeetingAttendee *ia; GtkTreePath *path; GtkTreeIter iter; char *address; guint32 disable_mask = ~0; GSList *menus = NULL; ECalPopup *ep; int i; priv = mpage->priv; /* only process right-clicks */ if (event->button != 3 || event->type != GDK_BUTTON_PRESS) return FALSE; /* only if we right-click on an attendee */ if (gtk_tree_view_get_path_at_pos (GTK_TREE_VIEW (priv->list_view), event->x, event->y, &path, NULL, NULL, NULL)) { GtkTreeSelection *selection; if (gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->model), &iter, path)) { gtk_tree_model_get (GTK_TREE_MODEL (priv->model), &iter, E_MEETING_STORE_ADDRESS_COL, &address, -1); ia = e_meeting_store_find_attendee (priv->model, address, &priv->row); g_free (address); if (ia) { selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (priv->list_view)); gtk_tree_selection_unselect_all (selection); gtk_tree_selection_select_path (selection, path); if (e_meeting_attendee_get_edit_level (ia) == E_MEETING_ATTENDEE_EDIT_FULL) disable_mask &= ~ATTENDEE_CAN_DELETE; } } } if (GTK_WIDGET_IS_SENSITIVE(priv->add)) disable_mask &= ~ATTENDEE_CAN_ADD; else if (priv->user_org) disable_mask &= ~ATTENDEE_CAN_ADD; ep = e_cal_popup_new("org.gnome.evolution.calendar.meeting.popup"); for (i=0;ibutton, event->time); return TRUE; } static gboolean list_view_event (EMeetingListView *list_view, GdkEvent *event, MeetingPage *mpage) { MeetingPagePrivate *priv= mpage->priv; if (event->type == GDK_2BUTTON_PRESS && mpage->priv->user_org) { EMeetingAttendee *attendee; attendee = e_meeting_store_add_attendee_with_defaults (priv->model); if (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE) { e_meeting_attendee_set_delfrom (attendee, g_strdup_printf ("MAILTO:%s", mpage->priv->user_add)); } e_meeting_list_view_edit (mpage->priv->list_view, attendee); return TRUE; } return FALSE; } static gboolean list_key_press (EMeetingListView *list_view, GdkEventKey *event, MeetingPage *mpage) { if (event->keyval == GDK_Delete) { remove_clicked_cb (NULL, mpage); return TRUE; } else if (event->keyval == GDK_Insert) { add_clicked_cb (NULL, mpage); return TRUE; } return FALSE; } /** * meeting_page_construct: * @mpage: An task details page. * * Constructs an task page by loading its Glade data. * * Return value: The same object as @mpage, or NULL if the widgets could not * be created. **/ MeetingPage * meeting_page_construct (MeetingPage *mpage, EMeetingStore *ems, ECal *client) { MeetingPagePrivate *priv; char *backend_address = NULL; EIterator *it; EAccount *def_account; GList *address_strings = NULL, *l; GtkWidget *sw; EAccount *a; GtkTreeSelection *selection; char *gladefile; priv = mpage->priv; gladefile = g_build_filename (EVOLUTION_GLADEDIR, "meeting-page.glade", NULL); priv->xml = glade_xml_new (gladefile, NULL, NULL); g_free (gladefile); if (!priv->xml) { g_message (G_STRLOC ": Could not load the Glade XML file!"); return NULL; } if (!get_widgets (mpage)) { g_message (G_STRLOC ": Could not find all widgets in the XML file!"); return NULL; } /* Address information */ if (!e_cal_get_cal_address (client, &backend_address, NULL)) return NULL; priv->accounts = itip_addresses_get (); def_account = itip_addresses_get_default(); for (it = e_list_get_iterator((EList *)priv->accounts); e_iterator_is_valid(it); e_iterator_next(it)) { a = (EAccount *)e_iterator_get(it); char *full; full = g_strdup_printf("%s <%s>", a->id->name, a->id->address); address_strings = g_list_append(address_strings, full); /* Note that the address specified by the backend gets * precedence over the default mail address. */ if (backend_address && !strcmp (backend_address, a->id->address)) { if (priv->default_address) g_free (priv->default_address); priv->default_address = g_strdup (full); } else if (a == def_account && !priv->default_address) { priv->default_address = g_strdup (full); } } if (backend_address) g_free (backend_address); g_object_unref(it); if (address_strings) gtk_combo_set_popdown_strings (GTK_COMBO (priv->organizer), address_strings); else g_warning ("No potential organizers!"); for (l = address_strings; l != NULL; l = l->next) g_free (l->data); g_list_free (address_strings); /* The etable displaying attendees and their status */ g_object_ref (ems); priv->model = ems; priv->list_view = e_meeting_list_view_new (priv->model); selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (priv->list_view)); gtk_tree_selection_set_mode (selection, GTK_SELECTION_MULTIPLE); g_signal_connect (G_OBJECT (priv->list_view), "button_press_event", G_CALLBACK (button_press_event), mpage); g_signal_connect (G_OBJECT (priv->list_view), "event", G_CALLBACK (list_view_event), mpage); 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); /* Set the mnemonic widget for the Attendees label */ gtk_label_set_mnemonic_widget (GTK_LABEL (priv->att_label), GTK_WIDGET (priv->list_view)); /* Init the widget signals */ init_widgets (mpage); g_signal_connect_after (G_OBJECT (mpage), "client_changed", G_CALLBACK (client_changed_cb), NULL); g_signal_connect (priv->list_view, "key_press_event", G_CALLBACK (list_key_press), mpage); return mpage; } /** * meeting_page_new: * * Creates a new task details page. * * Return value: A newly-created task details page, or NULL if the page could * not be created. **/ MeetingPage * meeting_page_new (EMeetingStore *ems, ECal *client) { MeetingPage *mpage; mpage = g_object_new (TYPE_MEETING_PAGE, NULL); if (!meeting_page_construct (mpage, ems, client)) { g_object_unref (mpage); return NULL; } return mpage; } /** * meeting_page_get_cancel_comp: * @mpage: * * * * Return value: **/ ECalComponent * meeting_page_get_cancel_comp (MeetingPage *mpage) { MeetingPagePrivate *priv; g_return_val_if_fail (mpage != NULL, NULL); g_return_val_if_fail (IS_MEETING_PAGE (mpage), NULL); priv = mpage->priv; if (priv->deleted_attendees->len == 0) return NULL; set_attendees (priv->comp, priv->deleted_attendees); return e_cal_component_clone (priv->comp); }