diff options
Diffstat (limited to 'calendar/gui')
-rw-r--r-- | calendar/gui/dialogs/meeting-page.c | 15 | ||||
-rw-r--r-- | calendar/gui/dialogs/meeting-page.h | 9 | ||||
-rw-r--r-- | calendar/gui/e-itip-control.c | 21 | ||||
-rw-r--r-- | calendar/gui/itip-utils.c | 409 | ||||
-rw-r--r-- | calendar/gui/itip-utils.h | 2 |
5 files changed, 371 insertions, 85 deletions
diff --git a/calendar/gui/dialogs/meeting-page.c b/calendar/gui/dialogs/meeting-page.c index 3b20831093..22c8f6e05a 100644 --- a/calendar/gui/dialogs/meeting-page.c +++ b/calendar/gui/dialogs/meeting-page.c @@ -382,15 +382,21 @@ meeting_page_fill_widgets (CompEditorPage *page, CalComponent *comp) if (organizer.value != NULL) { const gchar *strip = itip_strip_mailto (organizer.value); - gchar *s = e_utf8_to_gtk_string (priv->existing_organizer, strip); + gchar *s, *string; gtk_widget_hide (priv->organizer_table); gtk_widget_show (priv->existing_organizer_table); gtk_widget_hide (priv->invite); + if (organizer.cn != NULL) + string = g_strdup_printf ("%s <%s>", organizer.cn, strip); + else + string = g_strdup (strip); + s = e_utf8_to_gtk_string (priv->existing_organizer, string); gtk_label_set_text (GTK_LABEL (priv->existing_organizer), s); g_free (s); - + g_free (string); + priv->existing = TRUE; } else { gtk_widget_hide (priv->other_organizer_lbl); @@ -655,11 +661,12 @@ other_clicked_cb (GtkWidget *widget, gpointer data) mpage = MEETING_PAGE (data); priv = mpage->priv; + gtk_widget_hide (priv->organizer_lbl); + gtk_widget_hide (priv->organizer); + gtk_widget_hide (priv->other_organizer_btn); gtk_widget_show (priv->other_organizer_lbl); gtk_widget_show (priv->other_organizer); - gtk_label_set_text (GTK_LABEL (priv->organizer_lbl), _("Sent By:")); - priv->other = TRUE; } diff --git a/calendar/gui/dialogs/meeting-page.h b/calendar/gui/dialogs/meeting-page.h index 206ee90bb8..b2503f7018 100644 --- a/calendar/gui/dialogs/meeting-page.h +++ b/calendar/gui/dialogs/meeting-page.h @@ -52,10 +52,11 @@ typedef struct { } MeetingPageClass; -GtkType meeting_page_get_type (void); -MeetingPage *meeting_page_construct (MeetingPage *mpage, EMeetingModel *emm); -MeetingPage *meeting_page_new (EMeetingModel *emm); -CalComponent *meeting_page_get_cancel_comp (MeetingPage *mpage); +GtkType meeting_page_get_type (void); +MeetingPage *meeting_page_construct (MeetingPage *mpage, + EMeetingModel *emm); +MeetingPage *meeting_page_new (EMeetingModel *emm); +CalComponent *meeting_page_get_cancel_comp (MeetingPage *mpage); diff --git a/calendar/gui/e-itip-control.c b/calendar/gui/e-itip-control.c index 20e17d7eac..1f26c5d765 100644 --- a/calendar/gui/e-itip-control.c +++ b/calendar/gui/e-itip-control.c @@ -857,15 +857,16 @@ static icalcomponent * get_next (icalcompiter *iter) { icalcomponent *ret = NULL; - icalcomponent_kind kind = ICAL_NO_COMPONENT; + icalcomponent_kind kind; - while (kind != ICAL_VEVENT_COMPONENT - && kind != ICAL_VTODO_COMPONENT - && kind != ICAL_VFREEBUSY_COMPONENT) { + do { icalcompiter_next (iter); ret = icalcompiter_deref (iter); kind = icalcomponent_isa (ret); - } + } while (ret != NULL + && kind != ICAL_VEVENT_COMPONENT + && kind != ICAL_VTODO_COMPONENT + && kind != ICAL_VFREEBUSY_COMPONENT); return ret; } @@ -990,6 +991,11 @@ e_itip_control_set_data (EItipControl *itip, const gchar *text) && kind != ICAL_VFREEBUSY_COMPONENT) priv->ical_comp = get_next (&priv->iter); + if (priv->ical_comp == NULL) { + write_error_html (itip, _("The attachment has no viewable calendar items")); + return; + } + priv->total = icalcomponent_count_components (priv->main_comp, ICAL_VEVENT_COMPONENT); priv->total += icalcomponent_count_components (priv->main_comp, ICAL_VTODO_COMPONENT); priv->total += icalcomponent_count_components (priv->main_comp, ICAL_VFREEBUSY_COMPONENT); @@ -1130,6 +1136,7 @@ update_attendee_status (EItipControl *itip) change_status (cal_component_get_icalcomponent (comp), itip_strip_mailto (a->value), partstat); + cal_component_rescan (comp); } else { dialog = gnome_warning_dialog (_("Attendee status could " "not be updated because " @@ -1334,14 +1341,17 @@ ok_clicked_cb (GtkHTML *html, const gchar *method, const gchar *url, const gchar break; case 'A': change_status (priv->ical_comp, priv->my_address, ICAL_PARTSTAT_ACCEPTED); + cal_component_rescan (priv->comp); update_item (itip); break; case 'T': change_status (priv->ical_comp, priv->my_address, ICAL_PARTSTAT_TENTATIVE); + cal_component_rescan (priv->comp); update_item (itip); break; case 'D': change_status (priv->ical_comp, priv->my_address, ICAL_PARTSTAT_DECLINED); + cal_component_rescan (priv->comp); update_item (itip); break; case 'F': @@ -1399,6 +1409,7 @@ ok_clicked_cb (GtkHTML *html, const gchar *method, const gchar *url, const gchar icalproperty_free (prop); } } + cal_component_rescan (comp); itip_send_comp (CAL_COMPONENT_METHOD_REPLY, comp); } else { GtkWidget *dialog; diff --git a/calendar/gui/itip-utils.c b/calendar/gui/itip-utils.c index dcef42307e..9180fbc400 100644 --- a/calendar/gui/itip-utils.c +++ b/calendar/gui/itip-utils.c @@ -78,10 +78,34 @@ error_dialog (gchar *str) gnome_dialog_run_and_close (GNOME_DIALOG (dlg)); } +static Bonobo_ConfigDatabase db = NULL; + +static ItipAddress * +get_address (long num) +{ + ItipAddress *a; + gchar *path; + + a = g_new0 (ItipAddress, 1); + + /* get the identity info */ + path = g_strdup_printf ("/Mail/Accounts/identity_name_%ld", num); + a->name = bonobo_config_get_string (db, path, NULL); + g_free (path); + + path = g_strdup_printf ("/Mail/Accounts/identity_address_%ld", num); + a->address = bonobo_config_get_string (db, path, NULL); + g_free (path); + + a->full = g_strdup_printf ("%s <%s>", a->name, a->address); + + return a; +} + GList * itip_addresses_get (void) { - static Bonobo_ConfigDatabase db = NULL; + CORBA_Environment ev; GList *addresses = NULL; glong len, def, i; @@ -106,29 +130,55 @@ itip_addresses_get (void) for (i = 0; i < len; i++) { ItipAddress *a; - gchar *path; - - a = g_new0 (ItipAddress, 1); - - /* get the identity info */ - path = g_strdup_printf ("/Mail/Accounts/identity_name_%ld", i); - a->name = bonobo_config_get_string (db, path, NULL); - g_free (path); - - path = g_strdup_printf ("/Mail/Accounts/identity_address_%ld", i); - a->address = bonobo_config_get_string (db, path, NULL); - g_free (path); + a = get_address (i); if (i == def) a->default_address = TRUE; - a->full = g_strdup_printf ("%s <%s>", a->name, a->address); addresses = g_list_append (addresses, a); } return addresses; } +ItipAddress * +itip_addresses_get_default (void) +{ + CORBA_Environment ev; + ItipAddress *a; + glong def; + + if (db == NULL) { + CORBA_exception_init (&ev); + + db = bonobo_get_object ("wombat:", + "Bonobo/ConfigDatabase", + &ev); + + if (BONOBO_EX (&ev) || db == CORBA_OBJECT_NIL) { + CORBA_exception_free (&ev); + return NULL; + } + + CORBA_exception_free (&ev); + } + + def = bonobo_config_get_long_with_default (db, "/Mail/Accounts/default_account", 0, NULL); + a = get_address (def); + a->default_address = TRUE; + + return a; +} + +void +itip_address_free (ItipAddress *address) +{ + g_free (address->name); + g_free (address->address); + g_free (address->full); + g_free (address); +} + void itip_addresses_free (GList *addresses) { @@ -136,11 +186,7 @@ itip_addresses_free (GList *addresses) for (l = addresses; l != NULL; l = l->next) { ItipAddress *a = l->data; - - g_free (a->name); - g_free (a->address); - g_free (a->full); - g_free (a); + itip_address_free (a); } g_list_free (addresses); } @@ -189,33 +235,6 @@ typedef struct { icalcomponent *icomp; } ItipUtilTZData; -static void -foreach_tzid_callback (icalparameter *param, gpointer data) -{ - ItipUtilTZData *tz_data = data; - const char *tzid; - icaltimezone *zone; - icalcomponent *vtimezone_comp; - - /* Get the TZID string from the parameter. */ - tzid = icalparameter_get_tzid (param); - if (!tzid || g_hash_table_lookup (tz_data->tzids, tzid)) - return; - - /* Check if it is a builtin timezone. If it isn't, return. */ - zone = icaltimezone_get_builtin_timezone_from_tzid (tzid); - if (!zone) - return; - - /* Convert it to a string and add it to the hash. */ - vtimezone_comp = icaltimezone_get_component (zone); - if (!vtimezone_comp) - return; - - icalcomponent_add_component (tz_data->icomp, icalcomponent_new_clone (vtimezone_comp)); - g_hash_table_insert (tz_data->tzids, (char *)tzid, (char *)tzid); -} - static GNOME_Evolution_Composer_RecipientList * comp_to_list (CalComponentItipMethod method, CalComponent *comp) { @@ -230,6 +249,11 @@ comp_to_list (CalComponentItipMethod method, CalComponent *comp) case CAL_COMPONENT_METHOD_CANCEL: cal_component_get_attendee_list (comp, &attendees); len = g_slist_length (attendees); + if (len <= 0) { + error_dialog (_("Atleast one attendee is necessary")); + cal_component_free_attendee_list (attendees); + return NULL; + } to_list = GNOME_Evolution_Composer_RecipientList__alloc (); to_list->_maximum = len; @@ -246,6 +270,7 @@ comp_to_list (CalComponentItipMethod method, CalComponent *comp) recipient->name = CORBA_string_dup (""); recipient->address = CORBA_string_dup (itip_strip_mailto (att->value)); } + cal_component_free_attendee_list (attendees); break; case CAL_COMPONENT_METHOD_REPLY: @@ -288,7 +313,6 @@ static CORBA_char * comp_subject (CalComponent *comp) { CalComponentText caltext; - cal_component_get_summary (comp, &caltext); if (caltext.value != NULL) return CORBA_string_dup (caltext.value); @@ -365,61 +389,295 @@ comp_description (CalComponent *comp) } } +static void +foreach_tzid_callback (icalparameter *param, gpointer data) +{ + ItipUtilTZData *tz_data = data; + const char *tzid; + icaltimezone *zone; + icalcomponent *vtimezone_comp; + + /* Get the TZID string from the parameter. */ + tzid = icalparameter_get_tzid (param); + if (!tzid || g_hash_table_lookup (tz_data->tzids, tzid)) + return; + + /* Check if it is a builtin timezone. If it isn't, return. */ + zone = icaltimezone_get_builtin_timezone_from_tzid (tzid); + if (!zone) + return; + + /* Convert it to a string and add it to the hash. */ + vtimezone_comp = icaltimezone_get_component (zone); + if (!vtimezone_comp) + return; + + icalcomponent_add_component (tz_data->icomp, icalcomponent_new_clone (vtimezone_comp)); + g_hash_table_insert (tz_data->tzids, (char *)tzid, (char *)tzid); +} + static char * comp_string (CalComponentItipMethod method, CalComponent *comp) { - CalComponent *clone; - icalcomponent *icomp, *iclone; + icalcomponent *top_level, *icomp; icalproperty *prop; icalvalue *value; gchar *ical_string; ItipUtilTZData tz_data; - icomp = cal_util_new_top_level (); + top_level = cal_util_new_top_level (); prop = icalproperty_new (ICAL_METHOD_PROPERTY); value = icalvalue_new_method (itip_methods_enum[method]); icalproperty_set_value (prop, value); - icalcomponent_add_property (icomp, prop); + icalcomponent_add_property (top_level, prop); + + icomp = cal_component_get_icalcomponent (comp); + + /* Add the timezones */ + tz_data.tzids = g_hash_table_new (g_str_hash, g_str_equal); + tz_data.icomp = top_level; + icalcomponent_foreach_tzid (icomp, foreach_tzid_callback, &tz_data); + g_hash_table_destroy (tz_data.tzids); + + icalcomponent_add_component (top_level, icomp); + ical_string = icalcomponent_as_ical_string (top_level); + icalcomponent_remove_component (top_level, icomp); + + icalcomponent_free (top_level); + + return ical_string; +} + +static gboolean +comp_limit_attendees (CalComponent *comp) +{ + icalcomponent *icomp; + GList *addresses; + icalproperty *prop; + gboolean found = FALSE, match = FALSE; + + icomp = cal_component_get_icalcomponent (comp); + addresses = itip_addresses_get (); + + for (prop = icalcomponent_get_first_property (icomp, ICAL_ATTENDEE_PROPERTY); + prop != NULL; + prop = icalcomponent_get_next_property (icomp, ICAL_ATTENDEE_PROPERTY)) + { + icalvalue *value; + const char *attendee, *text; + GList *l; + + /* If we've already found something, just erase the rest */ + if (found) { + icalcomponent_remove_property (icomp, prop); + icalproperty_free (prop); + continue; + } + + value = icalproperty_get_value (prop); + if (!value) + continue; + + attendee = icalvalue_get_string (value); + + text = itip_strip_mailto (attendee); + for (l = addresses; l != NULL; l = l->next) { + ItipAddress *a = l->data; + + if (strstr (text, a->address)) + found = match = TRUE; + } + if (!match) { + icalcomponent_remove_property (icomp, prop); + icalproperty_free (prop); + } + match = FALSE; + } + itip_addresses_free (addresses); - /* Strip off attributes barred from appearing */ + return found; +} + +static void +comp_sentby (CalComponent *comp) +{ + CalComponentOrganizer organizer; + GList *addresses, *l; + const char *strip; + gboolean is_user = FALSE; + + cal_component_get_organizer (comp, &organizer); + if (!organizer.value) { + ItipAddress *a = itip_addresses_get_default (); + + organizer.value = g_strdup_printf ("MAILTO:%s", a->address); + organizer.sentby = NULL; + organizer.cn = a->name; + organizer.language = NULL; + + cal_component_set_organizer (comp, &organizer); + g_free ((char *) organizer.value); + itip_address_free (a); + + return; + } + + strip = itip_strip_mailto (organizer.value); + + addresses = itip_addresses_get (); + for (l = addresses; l != NULL; l = l->next) { + ItipAddress *a = l->data; + + if (!strcmp (a->address, strip)) { + is_user = TRUE; + break; + } + } + if (!is_user) { + ItipAddress *a = itip_addresses_get_default (); + + organizer.value = g_strdup (organizer.value); + organizer.sentby = g_strdup_printf ("MAILTO:%s", a->address); + organizer.cn = g_strdup (organizer.cn); + organizer.language = g_strdup (organizer.language); + + cal_component_set_organizer (comp, &organizer); + + g_free ((char *)organizer.value); + g_free ((char *)organizer.sentby); + g_free ((char *)organizer.cn); + g_free ((char *)organizer.language); + itip_address_free (a); + } + + itip_addresses_free (addresses); +} +static CalComponent * +comp_minimal (CalComponent *comp, gboolean attendee) +{ + CalComponent *clone; + icalcomponent *icomp; + icalproperty *prop; + CalComponentOrganizer organizer; + const char *uid; + GSList *comments; + struct icaltimetype itt; + CalComponentRange *recur_id; + + clone = cal_component_new (); + cal_component_set_new_vtype (clone, cal_component_get_vtype (comp)); + + if (attendee) { + GSList *attendees; + + cal_component_get_attendee_list (comp, &attendees); + cal_component_set_attendee_list (clone, attendees); + + if (!comp_limit_attendees (clone)) { + error_dialog ("You are not an attendee!"); + goto error; + } + } + + itt = icaltime_from_timet_with_zone (time (NULL), FALSE, + icaltimezone_get_utc_timezone ()); + cal_component_set_dtstamp (clone, &itt); + + cal_component_get_organizer (comp, &organizer); + if (organizer.value == NULL) + goto error; + cal_component_set_organizer (clone, &organizer); + + cal_component_get_uid (comp, &uid); + cal_component_set_uid (clone, uid); + + cal_component_get_comment_list (comp, &comments); + if (g_slist_length (comments) <= 1) { + cal_component_set_comment_list (clone, comments); + } else { + GSList *l = comments; + + comments = g_slist_remove_link (comments, l); + cal_component_set_comment_list (clone, l); + cal_component_free_text_list (l); + } + cal_component_free_text_list (comments); + + cal_component_get_recurid (comp, &recur_id); + cal_component_set_recurid (clone, recur_id); + + icomp = cal_component_get_icalcomponent (comp); + for (prop = icalcomponent_get_first_property (icomp, ICAL_X_PROPERTY); + prop != NULL; + prop = icalcomponent_get_next_property (icomp, ICAL_X_PROPERTY)) + { + icalproperty *p; + + p = icalproperty_new_clone (prop); + icalcomponent_add_property (icomp, p); + } + + cal_component_rescan (clone); + + return clone; + + error: + gtk_object_unref (GTK_OBJECT (clone)); + return NULL; +} + +static CalComponent * +comp_compliant (CalComponentItipMethod method, CalComponent *comp) +{ + CalComponent *clone, *temp_clone; + clone = cal_component_clone (comp); + + /* We delete incoming alarms anyhow, and this helps with outlook */ + cal_component_remove_all_alarms (clone); + + /* Comply with itip spec */ switch (method) { case CAL_COMPONENT_METHOD_PUBLISH: + comp_sentby (clone); cal_component_set_attendee_list (clone, NULL); - break; - case CAL_COMPONENT_METHOD_REPLY: + break; + case CAL_COMPONENT_METHOD_REQUEST: + comp_sentby (clone); + break; case CAL_COMPONENT_METHOD_CANCEL: + comp_sentby (clone); + break; + case CAL_COMPONENT_METHOD_REPLY: + break; + case CAL_COMPONENT_METHOD_ADD: + break; case CAL_COMPONENT_METHOD_REFRESH: + /* Need to remove almost everything */ + temp_clone = comp_minimal (clone, TRUE); + gtk_object_unref (GTK_OBJECT (clone)); + clone = temp_clone; + break; + case CAL_COMPONENT_METHOD_COUNTER: + break; case CAL_COMPONENT_METHOD_DECLINECOUNTER: - cal_component_remove_all_alarms (clone); + /* Need to remove almost everything */ + temp_clone = comp_minimal (clone, FALSE); + clone = temp_clone; break; default: } - iclone = cal_component_get_icalcomponent (clone); - - /* Add the timezones */ - tz_data.tzids = g_hash_table_new (g_str_hash, g_str_equal); - tz_data.icomp = icomp; - icalcomponent_foreach_tzid (iclone, foreach_tzid_callback, &tz_data); - g_hash_table_destroy (tz_data.tzids); - - icalcomponent_add_component (icomp, iclone); - ical_string = icalcomponent_as_ical_string (icomp); - icalcomponent_remove_component (icomp, iclone); - - icalcomponent_free (icomp); - gtk_object_unref (GTK_OBJECT (clone)); - - return ical_string; + return clone; } void -itip_send_comp (CalComponentItipMethod method, CalComponent *comp) +itip_send_comp (CalComponentItipMethod method, CalComponent *send_comp) { BonoboObjectClient *bonobo_server; GNOME_Evolution_Composer composer_server; + CalComponent *comp = NULL; GNOME_Evolution_Composer_RecipientList *to_list = NULL; GNOME_Evolution_Composer_RecipientList *cc_list = NULL; GNOME_Evolution_Composer_RecipientList *bcc_list = NULL; @@ -437,6 +695,10 @@ itip_send_comp (CalComponentItipMethod method, CalComponent *comp) g_return_if_fail (bonobo_server != NULL); composer_server = BONOBO_OBJREF (bonobo_server); + comp = comp_compliant (method, send_comp); + if (comp == NULL) + goto cleanup; + to_list = comp_to_list (method, comp); if (to_list == NULL) goto cleanup; @@ -492,6 +754,9 @@ itip_send_comp (CalComponentItipMethod method, CalComponent *comp) cleanup: CORBA_exception_free (&ev); + if (comp != NULL) + gtk_object_unref (GTK_OBJECT (comp)); + if (to_list != NULL) CORBA_free (to_list); if (cc_list != NULL) diff --git a/calendar/gui/itip-utils.h b/calendar/gui/itip-utils.h index 06108a4481..bf34f13a03 100644 --- a/calendar/gui/itip-utils.h +++ b/calendar/gui/itip-utils.h @@ -27,6 +27,8 @@ typedef struct { } ItipAddress; GList *itip_addresses_get (void); +ItipAddress *itip_addresses_get_default (void); +void itip_address_free (ItipAddress *address); void itip_addresses_free (GList *addresses); const gchar *itip_strip_mailto (const gchar *address); |