/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
* Authors: Jeffrey Stedfast <fejj@ximian.com>
*
* Copyright 2002 Ximian, Inc. (www.ximian.com)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* 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 Street #330, Boston, MA 02111-1307, USA.
*
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <glade/glade.h>
#include <libgnomeui/gnome-pixmap.h>
#include "message-tag-followup.h"
static void message_tag_followup_class_init (MessageTagFollowUpClass *class);
static void message_tag_followup_init (MessageTagFollowUp *followup);
static void message_tag_followup_finalise (GtkObject *obj);
static const char *tag_get_name (MessageTagEditor *editor);
static const char *tag_get_value (MessageTagEditor *editor);
static void tag_set_value (MessageTagEditor *editor, const char *value);
static struct {
const char *i18n_name;
const char *name;
int value;
} available_flags[] = {
{ N_("Call"), "call", FOLLOWUP_FLAG_CALL },
{ N_("Do Not Forward"), "do-not-forward", FOLLOWUP_FLAG_DO_NOT_FORWARD },
{ N_("Follow-Up"), "follow-up", FOLLOWUP_FLAG_FOLLOWUP },
{ N_("For Your Information"), "fyi", FOLLOWUP_FLAG_FYI },
{ N_("Forward"), "forward", FOLLOWUP_FLAG_FORWARD },
{ N_("No Response Necessary"), "no-response", FOLLOWUP_FLAG_NO_RESPONSE_NECESSARY },
{ N_("Read"), "read", FOLLOWUP_FLAG_READ },
{ N_("Reply"), "reply", FOLLOWUP_FLAG_REPLY },
{ N_("Reply to All"), "reply-all", FOLLOWUP_FLAG_REPLY_ALL },
{ N_("Review"), "review", FOLLOWUP_FLAG_REVIEW },
{ N_("None"), NULL, FOLLOWUP_FLAG_NONE }
};
static MessageTagEditorClass *parent_class;
GtkType
message_tag_followup_get_type (void)
{
static GtkType type = 0;
if (!type) {
GtkTypeInfo type_info = {
"MessageTagFollowUp",
sizeof (MessageTagFollowUp),
sizeof (MessageTagFollowUpClass),
(GtkClassInitFunc) message_tag_followup_class_init,
(GtkObjectInitFunc) message_tag_followup_init,
(GtkArgSetFunc) NULL,
(GtkArgGetFunc) NULL
};
type = gtk_type_unique (message_tag_editor_get_type (), &type_info);
}
return type;
}
static void
message_tag_followup_class_init (MessageTagFollowUpClass *klass)
{
GtkObjectClass *object_class;
MessageTagEditorClass *editor_class;
object_class = (GtkObjectClass *) klass;
editor_class = (MessageTagEditorClass *) klass;
parent_class = gtk_type_class (message_tag_editor_get_type ());
object_class->finalize = message_tag_followup_finalise;
editor_class->get_name = tag_get_name;
editor_class->get_value = tag_get_value;
editor_class->set_value = tag_set_value;
}
static void
message_tag_followup_init (MessageTagFollowUp *editor)
{
editor->tag = g_new (struct _FollowUpTag, 1);
editor->tag->type = FOLLOWUP_FLAG_NONE;
editor->tag->target_date = time (NULL);
editor->tag->completed = 0;
editor->value = NULL;
editor->type = NULL;
editor->none = NULL;
editor->target_date = NULL;
editor->completed = NULL;
editor->clear = NULL;
}
static void
message_tag_followup_finalise (GtkObject *obj)
{
MessageTagFollowUp *editor = (MessageTagFollowUp *) obj;
g_free (editor->tag);
g_free (editor->value);
((GtkObjectClass *)(parent_class))->finalize (obj);
}
static const char *
tag_get_name (MessageTagEditor *editor)
{
return "follow-up";
}
static const char *
tag_get_value (MessageTagEditor *editor)
{
MessageTagFollowUp *followup = (MessageTagFollowUp *) editor;
g_free (followup->value);
followup->value = message_tag_followup_encode (followup->tag);
return followup->value;
}
static void
set_widget_values (MessageTagFollowUp *followup)
{
time_t completed;
gtk_option_menu_set_history (followup->type, followup->tag->type);
e_date_edit_set_time (followup->target_date, followup->tag->target_date);
completed = followup->tag->completed;
gtk_toggle_button_set_active (followup->completed, completed ? TRUE : FALSE);
if (completed)
followup->tag->completed = completed;
}
static void
tag_set_value (MessageTagEditor *editor, const char *value)
{
MessageTagFollowUp *followup = (MessageTagFollowUp *) editor;
g_free (followup->tag);
followup->tag = message_tag_followup_decode (value);
set_widget_values (followup);
}
struct _FollowUpTag *
message_tag_followup_decode (const char *value)
{
struct _FollowUpTag *tag;
const char *inptr;
int i;
tag = g_new (struct _FollowUpTag, 1);
for (i = 0; i < FOLLOWUP_FLAG_NONE; i++) {
if (!strncmp (value, available_flags[i].name, strlen (available_flags[i].name)))
break;
}
tag->type = i;
inptr = value + strlen (available_flags[i].name);
if (*inptr == ':') {
inptr++;
tag->target_date = strtoul (inptr, (char **) &inptr, 16);
if (*inptr == ':') {
inptr++;
tag->completed = strtoul (inptr, (char **) &inptr, 16);
} else
tag->completed = 0;
} else {
tag->target_date = time (NULL);
tag->completed = 0;
}
return tag;
}
char *
message_tag_followup_encode (struct _FollowUpTag *tag)
{
g_return_val_if_fail (tag != NULL, NULL);
if (tag->type == FOLLOWUP_FLAG_NONE)
return NULL;
return g_strdup_printf ("%s:%lx:%lx", available_flags[tag->type].name,
(unsigned long) tag->target_date,
(unsigned long) tag->completed);
}
const char *
message_tag_followup_i18n_name (int type)
{
g_return_val_if_fail (type >= 0 && type <= FOLLOWUP_FLAG_NONE, NULL);
if (type != FOLLOWUP_FLAG_NONE)
return _(available_flags[type].i18n_name);
else
return NULL;
}
static void
clear_clicked (GtkButton *button, gpointer user_data)
{
MessageTagFollowUp *followup = user_data;
gtk_widget_show (followup->none);
gtk_option_menu_set_history (followup->type, FOLLOWUP_FLAG_NONE);
gtk_signal_emit_by_name (GTK_OBJECT (followup->none), "activate", followup);
e_date_edit_set_time (followup->target_date, time (NULL));
gtk_toggle_button_set_active (followup->completed, FALSE);
}
static void
completed_toggled (GtkToggleButton *button, gpointer user_data)
{
MessageTagFollowUp *followup = user_data;
if (gtk_toggle_button_get_active (followup->completed))
followup->tag->completed = time (NULL);
else
followup->tag->completed = 0;
}
static void
type_changed (GtkWidget *item, gpointer user_data)
{
MessageTagFollowUp *followup = user_data;
followup->tag->type = GPOINTER_TO_INT (gtk_object_get_data (GTK_OBJECT (item), "value"));
if (item != followup->none)
gtk_widget_hide (followup->none);
}
static void
target_date_changed (EDateEdit *widget, gpointer user_data)
{
MessageTagFollowUp *followup = user_data;
followup->tag->target_date = e_date_edit_get_time (widget);
}
GtkWidget *target_date_new (const char *s1, const char *s2, int i1, int i2);
GtkWidget *
target_date_new (const char *s1, const char *s2, int i1, int i2)
{
GtkWidget *widget;
widget = e_date_edit_new ();
e_date_edit_set_show_date (E_DATE_EDIT (widget), TRUE);
e_date_edit_set_show_time (E_DATE_EDIT (widget), TRUE);
e_date_edit_set_week_start_day (E_DATE_EDIT (widget), 6);
/* FIXME: make this locale dependant?? */
e_date_edit_set_use_24_hour_format (E_DATE_EDIT (widget), FALSE);
e_date_edit_set_allow_no_date_set (E_DATE_EDIT (widget), FALSE);
e_date_edit_set_time_popup_range (E_DATE_EDIT (widget), 0, 24);
return widget;
}
static void
construct (MessageTagEditor *editor)
{
MessageTagFollowUp *followup = (MessageTagFollowUp *) editor;
GtkWidget *widget, *menu, *item;
GladeXML *gui;
int i;
gui = glade_xml_new (EVOLUTION_GLADEDIR "/message-tags.glade", "followup_editor");
widget = glade_xml_get_widget (gui, "toplevel");
/* reparent */
gtk_widget_reparent (widget, GNOME_DIALOG (editor)->vbox);
widget = glade_xml_get_widget (gui, "pixmap");
gnome_pixmap_load_file (GNOME_PIXMAP (widget), EVOLUTION_GLADEDIR "/flag-for-followup-48.png");
followup->message_list = GTK_CLIST (glade_xml_get_widget (gui, "message_list"));
followup->type = GTK_OPTION_MENU (glade_xml_get_widget (gui, "followup_type"));
gtk_option_menu_remove_menu (followup->type);
menu = gtk_menu_new ();
for (i = 0; i <= FOLLOWUP_FLAG_NONE; i++) {
item = gtk_menu_item_new_with_label (_(available_flags[i].i18n_name));
gtk_object_set_data (GTK_OBJECT (item), "value",
GINT_TO_POINTER (available_flags[i].value));
gtk_signal_connect (GTK_OBJECT (item), "activate",
type_changed, followup);
gtk_menu_append (GTK_MENU (menu), item);
gtk_widget_show (item);
}
followup->none = item;
gtk_option_menu_set_menu (followup->type, menu);
gtk_signal_emit_by_name (GTK_OBJECT (item), "activate", followup);
gtk_option_menu_set_history (followup->type, FOLLOWUP_FLAG_NONE);
followup->target_date = E_DATE_EDIT (glade_xml_get_widget (gui, "target_date"));
e_date_edit_set_time (followup->target_date, time (NULL));
gtk_signal_connect (GTK_OBJECT (followup->target_date), "changed",
target_date_changed, followup);
followup->completed = GTK_TOGGLE_BUTTON (glade_xml_get_widget (gui, "completed"));
gtk_signal_connect (GTK_OBJECT (followup->completed), "toggled",
completed_toggled, followup);
followup->clear = GTK_BUTTON (glade_xml_get_widget (gui, "clear"));
gtk_signal_connect (GTK_OBJECT (followup->clear), "clicked",
clear_clicked, followup);
gtk_object_unref (GTK_OBJECT (gui));
}
MessageTagEditor *
message_tag_followup_new (void)
{
MessageTagEditor *editor;
editor = (MessageTagEditor *) gtk_type_new (message_tag_followup_get_type ());
construct (editor);
return editor;
}
void
message_tag_followup_append_message (MessageTagFollowUp *editor,
const char *from,
const char *subject)
{
char *text[3];
g_return_if_fail (IS_MESSAGE_TAG_FOLLOWUP (editor));
text[0] = (char *) from;
text[1] = (char *) subject;
text[2] = NULL;
gtk_clist_append (editor->message_list, text);
}