/* Evolution calendar - Alarm options dialog
*
* Copyright (C) 2001 Ximian, Inc.
*
* Authors: Federico Mena-Quintero <federico@ximian.com>
*
* 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 <config.h>
#endif
#include <string.h>
#include <glib.h>
#include <libgnome/gnome-i18n.h>
#include <gtk/gtkmain.h>
#include <gtk/gtkcheckbutton.h>
#include <gtk/gtksignal.h>
#include <gtk/gtkwindow.h>
#include <glade/glade.h>
#include "e-util/e-dialog-widgets.h"
#include "alarm-options.h"
typedef struct {
/* Whether the dialog was accepted or canceled */
gboolean canceled;
/* Glade XML data */
GladeXML *xml;
/* Toplevel */
GtkWidget *toplevel;
/* Buttons */
GtkWidget *button_ok;
GtkWidget *button_cancel;
/* Alarm repeat widgets */
GtkWidget *repeat_toggle;
GtkWidget *repeat_group;
GtkWidget *repeat_quantity;
GtkWidget *repeat_value;
GtkWidget *repeat_unit;
/* Display alarm widgets */
GtkWidget *dalarm_group;
GtkWidget *dalarm_description;
/* Audio alarm widgets */
GtkWidget *aalarm_group;
GtkWidget *aalarm_attach;
/* FIXME: Mail alarm widgets */
GtkWidget *malarm_group;
/* Procedure alarm widgets */
GtkWidget *palarm_group;
GtkWidget *palarm_program;
GtkWidget *palarm_args;
} Dialog;
/* Gets the widgets from the XML file and returns if they are all available. */
static gboolean
get_widgets (Dialog *dialog)
{
#define GW(name) glade_xml_get_widget (dialog->xml, name)
dialog->toplevel = GW ("alarm-options-toplevel");
dialog->button_ok = GW ("button-ok");
dialog->button_cancel = GW ("button-cancel");
dialog->repeat_toggle = GW ("repeat-toggle");
dialog->repeat_group = GW ("repeat-group");
dialog->repeat_quantity = GW ("repeat-quantity");
dialog->repeat_value = GW ("repeat-value");
dialog->repeat_unit = GW ("repeat-unit");
dialog->dalarm_group = GW ("dalarm-group");
dialog->dalarm_description = GW ("dalarm-description");
dialog->aalarm_group = GW ("aalarm-group");
dialog->aalarm_attach = GW ("aalarm-attach");
dialog->malarm_group = GW ("malarm-group");
dialog->palarm_group = GW ("palarm-group");
dialog->palarm_program = GW ("palarm-program");
dialog->palarm_args = GW ("palarm-args");
return (dialog->toplevel
&& dialog->button_ok
&& dialog->button_cancel
&& dialog->repeat_toggle
&& dialog->repeat_group
&& dialog->repeat_quantity
&& dialog->repeat_value
&& dialog->repeat_unit
&& dialog->dalarm_group
&& dialog->dalarm_description
&& dialog->aalarm_group
&& dialog->aalarm_attach
&& dialog->malarm_group
&& dialog->palarm_group
&& dialog->palarm_program
&& dialog->palarm_args);
}
/* Closes the dialog by terminating its main loop */
static void
close_dialog (Dialog *dialog, gboolean canceled)
{
dialog->canceled = canceled;
gtk_main_quit ();
}
/* Callback used when the toplevel window is deleted */
static guint
toplevel_delete_event_cb (GtkWidget *widget, GdkEventAny *event, gpointer data)
{
Dialog *dialog;
dialog = data;
close_dialog (dialog, TRUE);
return TRUE;
}
/* Callback used when the OK button is clicked */
static void
button_ok_clicked_cb (GtkWidget *button, gpointer data)
{
Dialog *dialog;
dialog = data;
close_dialog (dialog, FALSE);
}
/* Callback used when the Cancel button is clicked */
static void
button_cancel_clicked_cb (GtkWidget *button, gpointer data)
{
Dialog *dialog;
dialog = data;
close_dialog (dialog, TRUE);
}
/* Callback used when the repeat toggle button is toggled. We sensitize the
* repeat group options as appropriate.
*/
static void
repeat_toggle_toggled_cb (GtkToggleButton *toggle, gpointer data)
{
Dialog *dialog;
gboolean active;
dialog = data;
active = gtk_toggle_button_get_active (toggle);
gtk_widget_set_sensitive (dialog->repeat_group, active);
}
/* Hooks the widget signals */
static void
init_widgets (Dialog *dialog)
{
/* Toplevel, buttons */
dialog->canceled = TRUE;
g_signal_connect((dialog->toplevel), "delete_event",
G_CALLBACK (toplevel_delete_event_cb), dialog);
g_signal_connect((dialog->button_ok), "clicked",
G_CALLBACK (button_ok_clicked_cb), dialog);
g_signal_connect((dialog->button_cancel), "clicked",
G_CALLBACK (button_cancel_clicked_cb), dialog);
/* Alarm repeat */
g_signal_connect((dialog->repeat_toggle), "toggled",
G_CALLBACK (repeat_toggle_toggled_cb), dialog);
}
/* Fills the audio alarm widgets with the values from the alarm component */
static void
alarm_to_aalarm_widgets (Dialog *dialog, CalComponentAlarm *alarm)
{
icalattach *attach;
const char *url;
cal_component_alarm_get_attach (alarm, &attach);
if (!attach) {
e_dialog_editable_set (dialog->aalarm_attach, NULL);
return;
}
/* FIXME: this does not support inline data */
url = NULL;
if (icalattach_get_is_url (attach))
url = icalattach_get_url (attach);
else
g_message ("alarm_to_aalarm_widgets(): FIXME: we don't support inline data yet");
e_dialog_editable_set (dialog->aalarm_attach, url);
icalattach_unref (attach);
}
/* Fills the display alarm widgets with the values from the alarm component */
static void
alarm_to_dalarm_widgets (Dialog *dialog, CalComponentAlarm *alarm)
{
CalComponentText description;
cal_component_alarm_get_description (alarm, &description);
e_dialog_editable_set (dialog->dalarm_description, description.value);
}
/* Fills the mail alarm widgets with the values from the alarm component */
static void
alarm_to_malarm_widgets (Dialog *dialog, CalComponentAlarm *alarm)
{
/* FIXME: nothing for now; we don't support mail alarms */
}
/* Fills the procedure alarm widgets with the values from the alarm component */
static void
alarm_to_palarm_widgets (Dialog *dialog, CalComponentAlarm *alarm)
{
icalattach *attach;
CalComponentText description;
cal_component_alarm_get_attach (alarm, &attach);
cal_component_alarm_get_description (alarm, &description);
if (attach) {
const char *url;
if (icalattach_get_is_url (attach)) {
url = icalattach_get_url (attach);
e_dialog_editable_set (dialog->palarm_program, url);
} else
g_message ("alarm_to_palarm_widgets(): Don't know what to do with non-URL "
"attachments");
icalattach_unref (attach);
}
e_dialog_editable_set (dialog->palarm_args, description.value);
}
enum duration_units {
DUR_MINUTES,
DUR_HOURS,
DUR_DAYS
};
static const int duration_units_map[] = {
DUR_MINUTES,
DUR_HOURS,
DUR_DAYS,
-1
};
/* Sigh. Takes an overcomplicated duration value and reduces it to its lowest
* common denominator.
*/
static void
normalize_duration (struct icaldurationtype dur, int *value, enum duration_units *units)
{
if (dur.seconds != 0 || dur.minutes != 0) {
*value = ((((dur.weeks * 7 + dur.days) * 24 + dur.hours) * 60) + dur.minutes
+ dur.seconds / 60 + ((dur.seconds % 60) >= 30 ? 1 : 0));
*units = DUR_MINUTES;
} else if (dur.hours) {
*value = ((dur.weeks * 7) + dur.days) * 24 + dur.hours;
*units = DUR_HOURS;
} else if (dur.days != 0 || dur.weeks != 0) {
*value = dur.weeks * 7 + dur.days;
*units = DUR_DAYS;
} else {
*value = 0;
*units = DUR_MINUTES;
}
}
/* Fills the repeat widgets with the values from the alarm component */
static void
alarm_to_repeat_widgets (Dialog *dialog, CalComponentAlarm *alarm)
{
CalAlarmRepeat repeat;
int value;
enum duration_units units;
cal_component_alarm_get_repeat (alarm, &repeat);
/* Sensitivity */
if (repeat.repetitions == 0) {
gtk_widget_set_sensitive (dialog->repeat_group, FALSE);
e_dialog_toggle_set (dialog->repeat_toggle, FALSE);
return;
}
gtk_widget_set_sensitive (dialog->repeat_group, TRUE);
e_dialog_toggle_set (dialog->repeat_toggle, TRUE);
/* Repetitions */
e_dialog_spin_set (dialog->repeat_quantity, repeat.repetitions);
/* Duration */
normalize_duration (repeat.duration, &value, &units);
e_dialog_spin_set (dialog->repeat_value, value);
e_dialog_option_menu_set (dialog->repeat_unit, units, duration_units_map);
}
/* Fills the widgets with the values from the alarm component */
static void
alarm_to_dialog (Dialog *dialog, CalComponentAlarm *alarm)
{
CalAlarmAction action;
alarm_to_repeat_widgets (dialog, alarm);
cal_component_alarm_get_action (alarm, &action);
switch (action) {
case CAL_ALARM_NONE:
g_assert_not_reached ();
return;
case CAL_ALARM_AUDIO:
gtk_window_set_title (GTK_WINDOW (dialog->toplevel), _("Audio Alarm Options"));
gtk_widget_show (dialog->aalarm_group);
gtk_widget_hide (dialog->dalarm_group);
gtk_widget_hide (dialog->malarm_group);
gtk_widget_hide (dialog->palarm_group);
alarm_to_aalarm_widgets (dialog, alarm);
break;
case CAL_ALARM_DISPLAY:
gtk_window_set_title (GTK_WINDOW (dialog->toplevel), _("Message Alarm Options"));
gtk_widget_hide (dialog->aalarm_group);
gtk_widget_show (dialog->dalarm_group);
gtk_widget_hide (dialog->malarm_group);
gtk_widget_hide (dialog->palarm_group);
alarm_to_dalarm_widgets (dialog, alarm);
break;
case CAL_ALARM_EMAIL:
gtk_window_set_title (GTK_WINDOW (dialog->toplevel), _("Mail Alarm Options"));
gtk_widget_hide (dialog->aalarm_group);
gtk_widget_hide (dialog->dalarm_group);
gtk_widget_show (dialog->malarm_group);
gtk_widget_hide (dialog->palarm_group);
alarm_to_malarm_widgets (dialog, alarm);
break;
case CAL_ALARM_PROCEDURE:
gtk_window_set_title (GTK_WINDOW (dialog->toplevel), _("Program Alarm Options"));
gtk_widget_hide (dialog->aalarm_group);
gtk_widget_hide (dialog->dalarm_group);
gtk_widget_hide (dialog->malarm_group);
gtk_widget_show (dialog->palarm_group);
alarm_to_palarm_widgets (dialog, alarm);
break;
case CAL_ALARM_UNKNOWN:
gtk_window_set_title (GTK_WINDOW (dialog->toplevel), _("Unknown Alarm Options"));
break;
default:
g_assert_not_reached ();
return;
}
}
/* Fills the alarm data with the values from the repeat/duration widgets */
static void
repeat_widgets_to_alarm (Dialog *dialog, CalComponentAlarm *alarm)
{
CalAlarmRepeat repeat;
if (!e_dialog_toggle_get (dialog->repeat_toggle)) {
repeat.repetitions = 0;
cal_component_alarm_set_repeat (alarm, repeat);
return;
}
repeat.repetitions = e_dialog_spin_get_int (dialog->repeat_quantity);
memset (&repeat.duration, 0, sizeof (repeat.duration));
switch (e_dialog_option_menu_get (dialog->repeat_unit, duration_units_map)) {
case DUR_MINUTES:
repeat.duration.minutes = e_dialog_spin_get_int (dialog->repeat_value);
break;
case DUR_HOURS:
repeat.duration.hours = e_dialog_spin_get_int (dialog->repeat_value);
break;
case DUR_DAYS:
repeat.duration.days = e_dialog_spin_get_int (dialog->repeat_value);
break;
default:
g_assert_not_reached ();
}
cal_component_alarm_set_repeat (alarm, repeat);
}
/* Fills the audio alarm data with the values from the widgets */
static void
aalarm_widgets_to_alarm (Dialog *dialog, CalComponentAlarm *alarm)
{
char *url;
icalattach *attach;
url = e_dialog_editable_get (dialog->aalarm_attach);
attach = icalattach_new_from_url (url ? url : "");
g_free (url);
cal_component_alarm_set_attach (alarm, attach);
icalattach_unref (attach);
}
/* Fills the display alarm data with the values from the widgets */
static void
dalarm_widgets_to_alarm (Dialog *dialog, CalComponentAlarm *alarm)
{
char *str;
CalComponentText description;
icalcomponent *icalcomp;
icalproperty *icalprop;
str = e_dialog_editable_get (dialog->dalarm_description);
description.value = str;
description.altrep = NULL;
cal_component_alarm_set_description (alarm, &description);
g_free (str);
/* remove the X-EVOLUTION-NEEDS-DESCRIPTION property, so that
* we don't re-set the alarm's description */
icalcomp = cal_component_alarm_get_icalcomponent (alarm);
icalprop = icalcomponent_get_first_property (icalcomp, ICAL_X_PROPERTY);
while (icalprop) {
const char *x_name;
x_name = icalproperty_get_x_name (icalprop);
if (!strcmp (x_name, "X-EVOLUTION-NEEDS-DESCRIPTION")) {
icalcomponent_remove_property (icalcomp, icalprop);
break;
}
icalprop = icalcomponent_get_next_property (icalcomp, ICAL_X_PROPERTY);
}
}
/* Fills the mail alarm data with the values from the widgets */
static void
malarm_widgets_to_alarm (Dialog *dialog, CalComponentAlarm *alarm)
{
/* FIXME: nothing for now; we don't support mail alarms */
}
/* Fills the procedure alarm data with the values from the widgets */
static void
palarm_widgets_to_alarm (Dialog *dialog, CalComponentAlarm *alarm)
{
char *program;
icalattach *attach;
char *str;
CalComponentText description;
icalcomponent *icalcomp;
icalproperty *icalprop;
program = e_dialog_editable_get (dialog->palarm_program);
attach = icalattach_new_from_url (program ? program : "");
g_free (program);
cal_component_alarm_set_attach (alarm, attach);
icalattach_unref (attach);
str = e_dialog_editable_get (dialog->palarm_args);
description.value = str;
description.altrep = NULL;
cal_component_alarm_set_description (alarm, &description);
g_free (str);
/* remove the X-EVOLUTION-NEEDS-DESCRIPTION property, so that
* we don't re-set the alarm's description */
icalcomp = cal_component_alarm_get_icalcomponent (alarm);
icalprop = icalcomponent_get_first_property (icalcomp, ICAL_X_PROPERTY);
while (icalprop) {
const char *x_name;
x_name = icalproperty_get_x_name (icalprop);
if (!strcmp (x_name, "X-EVOLUTION-NEEDS-DESCRIPTION")) {
icalcomponent_remove_property (icalcomp, icalprop);
break;
}
icalprop = icalcomponent_get_next_property (icalcomp, ICAL_X_PROPERTY);
}
}
/* Fills the alarm data with the values from the widgets */
static void
dialog_to_alarm (Dialog *dialog, CalComponentAlarm *alarm)
{
CalAlarmAction action;
repeat_widgets_to_alarm (dialog, alarm);
cal_component_alarm_get_action (alarm, &action);
switch (action) {
case CAL_ALARM_NONE:
g_assert_not_reached ();
break;
case CAL_ALARM_AUDIO:
aalarm_widgets_to_alarm (dialog, alarm);
break;
case CAL_ALARM_DISPLAY:
dalarm_widgets_to_alarm (dialog, alarm);
break;
case CAL_ALARM_EMAIL:
malarm_widgets_to_alarm (dialog, alarm);
break;
case CAL_ALARM_PROCEDURE:
palarm_widgets_to_alarm (dialog, alarm);
break;
case CAL_ALARM_UNKNOWN:
break;
default:
g_assert_not_reached ();
}
}
/**
* alarm_options_dialog_run:
* @alarm: Alarm that is to be edited.
*
* Runs an alarm options dialog modally.
*
* Return value: TRUE if the dialog could be created, FALSE otherwise.
**/
gboolean
alarm_options_dialog_run (CalComponentAlarm *alarm)
{
Dialog dialog;
g_return_val_if_fail (alarm != NULL, FALSE);
dialog.xml = glade_xml_new (EVOLUTION_GLADEDIR "/alarm-options.glade", NULL, NULL);
if (!dialog.xml) {
g_message ("alarm_options_dialog_new(): Could not load the Glade XML file!");
return FALSE;
}
if (!get_widgets (&dialog)) {
g_object_unref((dialog.xml));
return FALSE;
}
init_widgets (&dialog);
alarm_to_dialog (&dialog, alarm);
gtk_widget_show (dialog.toplevel);
gtk_main ();
if (!dialog.canceled)
dialog_to_alarm (&dialog, alarm);
gtk_widget_destroy (dialog.toplevel);
g_object_unref((dialog.xml));
return TRUE;
}