/* To-do widget for gncal * * Copyright (C) 1998 The Free Software Foundation * * Author: Federico Mena <quartic@gimp.org> */ #include <config.h> #include <string.h> #include <gnome.h> #include "gncal-todo.h" #include "main.h" #include "popup-menu.h" #include "eventedit.h" int todo_show_due_date = 0; int todo_due_date_overdue_highlight = 0; char *todo_overdue_font_text; gint todo_current_sort_column = 0; gint todo_current_sort_type = GTK_SORT_ASCENDING; gboolean todo_style_changed =0; gboolean todo_list_autoresize = 1; gboolean todo_list_redraw_in_progess = 0; static void gncal_todo_init (GncalTodo *todo); guint gncal_todo_get_type (void) { static guint todo_type = 0; if (!todo_type) { GtkTypeInfo todo_info = { "GncalTodo", sizeof (GncalTodo), sizeof (GncalTodoClass), (GtkClassInitFunc) NULL, (GtkObjectInitFunc) gncal_todo_init, (GtkArgSetFunc) NULL, (GtkArgGetFunc) NULL }; todo_type = gtk_type_unique (gtk_vbox_get_type (), &todo_info); } return todo_type; } static void ok_button (GtkWidget *widget, GnomeDialog *dialog) { iCalObject *ico; GncalTodo *todo; GtkEntry *entry; GnomeDateEdit *due_date; GtkText *comment; ico = gtk_object_get_user_data (GTK_OBJECT (dialog)); todo = GNCAL_TODO (gtk_object_get_data (GTK_OBJECT (dialog), "gncal_todo")); entry = GTK_ENTRY (gtk_object_get_data (GTK_OBJECT (dialog), "summary_entry")); due_date = GNOME_DATE_EDIT (gtk_object_get_data(GTK_OBJECT(dialog), "due_date")); comment = GTK_TEXT(gtk_object_get_data (GTK_OBJECT(dialog), "comment")); if (ico->summary) g_free (ico->summary); if (ico->comment) g_free (ico->comment); ico->dtend = gnome_date_edit_get_date (due_date); ico->summary = g_strdup (gtk_entry_get_text (entry)); ico->comment = gtk_editable_get_chars( GTK_EDITABLE(comment), 0, -1); ico->user_data = NULL; if (ico->new) { gnome_calendar_add_object (todo->calendar, ico); ico->new = FALSE; } else gnome_calendar_object_changed (todo->calendar, ico, CHANGE_ALL); /* ok, summary only... */ save_default_calendar (todo->calendar); gtk_widget_destroy (GTK_WIDGET (dialog)); } static void cancel_button (GtkWidget *widget, GnomeDialog *dialog) { iCalObject *ico; ico = gtk_object_get_user_data (GTK_OBJECT (dialog)); ico->user_data = NULL; if (ico->new) ical_object_destroy (ico); gtk_widget_destroy (GTK_WIDGET (dialog)); } static gint delete_event (GtkWidget *widget, GdkEvent *event, GnomeDialog *dialog) { cancel_button (NULL, dialog); return TRUE; } static void simple_todo_editor (GncalTodo *todo, iCalObject *ico) { GtkWidget *dialog; GtkWidget *hbox; GtkWidget *due_box; GtkWidget *due_label; GtkWidget *due_entry; GtkWidget *comment_box; GtkWidget *comment_label; GtkWidget *comment_text; GtkWidget *comment_internal_box; GtkWidget *comment_sep; GtkWidget *w; GtkWidget *entry; dialog = gnome_dialog_new (ico->new ? _("Create to-do item") : _("Edit to-do item"), GNOME_STOCK_BUTTON_OK, GNOME_STOCK_BUTTON_CANCEL, NULL); gnome_dialog_set_parent (GNOME_DIALOG (dialog), GTK_WINDOW (todo->calendar)); hbox = gtk_hbox_new (FALSE, 4); gtk_container_border_width (GTK_CONTAINER (hbox), 4); gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (dialog)->vbox), hbox, FALSE, FALSE, 0); gtk_widget_show (hbox); due_box = gtk_hbox_new (FALSE, 4); gtk_container_border_width (GTK_CONTAINER (due_box), 4); gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (dialog)->vbox), due_box, FALSE, FALSE, 0); gtk_widget_show (due_box); comment_box = gtk_hbox_new (FALSE, 4); gtk_container_border_width (GTK_CONTAINER (comment_box), 4); gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (dialog)->vbox), comment_box, FALSE, FALSE, 0); gtk_widget_show (comment_box); comment_internal_box = gtk_vbox_new(FALSE,2); gtk_container_border_width (GTK_CONTAINER (comment_internal_box), 4); gtk_box_pack_start (GTK_BOX (comment_box), comment_internal_box, TRUE, TRUE, 0); gtk_widget_show (comment_internal_box); w = gtk_label_new (_("Summary:")); gtk_box_pack_start (GTK_BOX (hbox), w, FALSE, FALSE, 0); gtk_widget_show (w); entry = gtk_entry_new (); gtk_entry_set_text (GTK_ENTRY (entry), ico->summary); gtk_box_pack_start (GTK_BOX (hbox), entry, TRUE, TRUE, 0); gtk_widget_show (entry); due_label = gtk_label_new (_("Due Date:")); gtk_box_pack_start (GTK_BOX (due_box), due_label, FALSE, FALSE, 0); gtk_widget_show (due_label); due_entry = gtk_entry_new (); due_entry = date_edit_new (ico->dtend, FALSE); gtk_box_pack_start (GTK_BOX (due_box), due_entry, TRUE, TRUE, 0); gtk_widget_show (due_entry); comment_sep = gtk_hseparator_new (); gtk_box_pack_start (GTK_BOX (comment_box), comment_sep, FALSE, FALSE, 0); gtk_widget_show(comment_sep); comment_label = gtk_label_new (_("Item Comments:")); gtk_label_set_justify(GTK_LABEL(comment_label), GTK_JUSTIFY_LEFT); gtk_box_pack_start (GTK_BOX (comment_internal_box), comment_label, TRUE, TRUE, 0); gtk_widget_show (comment_label); comment_text = gtk_text_new (NULL, NULL); gtk_text_set_editable (GTK_TEXT (comment_text), TRUE); gtk_text_set_word_wrap( GTK_TEXT(comment_text), TRUE); gtk_text_freeze(GTK_TEXT(comment_text)); if(ico->comment) { gtk_text_insert(GTK_TEXT(comment_text), NULL, NULL, NULL, ico->comment, strlen(ico->comment)); } gtk_text_thaw(GTK_TEXT(comment_text)); gtk_box_pack_start (GTK_BOX (comment_internal_box), comment_text, FALSE, TRUE, 0); gtk_widget_show (comment_text); ico->user_data = dialog; gtk_object_set_user_data (GTK_OBJECT (dialog), ico); gtk_object_set_data (GTK_OBJECT (dialog), "gncal_todo", todo); gtk_object_set_data (GTK_OBJECT (dialog), "summary_entry", entry); gtk_object_set_data (GTK_OBJECT (dialog), "due_date", due_entry); gtk_object_set_data (GTK_OBJECT (dialog), "comment", comment_text); gnome_dialog_button_connect (GNOME_DIALOG (dialog), 0, (GtkSignalFunc) ok_button, dialog); gnome_dialog_button_connect (GNOME_DIALOG (dialog), 1, (GtkSignalFunc) cancel_button, dialog); gtk_signal_connect (GTK_OBJECT (dialog), "delete_event", (GtkSignalFunc) delete_event, dialog); gnome_dialog_set_default (GNOME_DIALOG (dialog), 0); gnome_dialog_editable_enters (GNOME_DIALOG (dialog), GTK_EDITABLE(entry)); gtk_window_position (GTK_WINDOW (dialog), GTK_WIN_POS_MOUSE); gtk_widget_show (dialog); gtk_widget_grab_focus (entry); } static iCalObject * get_clist_selected_ico (GtkCList *clist) { gint sel; if (!clist->selection) return NULL; sel = GPOINTER_TO_INT(clist->selection->data); return gtk_clist_get_row_data (clist, sel); } static void add_todo (GncalTodo *todo) { iCalObject *ico; ico = ical_new ("", user_name, ""); ico->type = ICAL_TODO; ico->new = TRUE; simple_todo_editor (todo, ico); } static void edit_todo (GncalTodo *todo) { simple_todo_editor (todo, get_clist_selected_ico (todo->clist)); } static void delete_todo (GncalTodo *todo) { gnome_calendar_remove_object (todo->calendar, get_clist_selected_ico (todo->clist)); save_default_calendar (todo->calendar); } static void add_activated (GtkWidget *widget, GncalTodo *todo) { add_todo (todo); } static void edit_activated (GtkWidget *widget, GncalTodo *todo) { edit_todo (todo); } static void delete_activated (GtkWidget *widget, GncalTodo *todo) { delete_todo (todo); } static void clist_row_selected (GtkCList *clist, gint row, gint column, GdkEventButton *event, GncalTodo *todo) { static struct menu_item items[] = { { N_("Add to-do item..."), (GtkSignalFunc) add_activated, NULL, TRUE }, { N_("Edit this item..."), (GtkSignalFunc) edit_activated, NULL, TRUE }, { N_("Delete this item"), (GtkSignalFunc) delete_activated, NULL, TRUE } }; int i; gtk_widget_set_sensitive (todo->edit_button, (todo->clist->selection != NULL)); gtk_widget_set_sensitive (todo->delete_button, (todo->clist->selection != NULL)); if (!event) return; switch (event->button) { case 1: if (event->type == GDK_2BUTTON_PRESS) edit_todo (todo); break; case 3: for (i = 0; i < (sizeof (items) / sizeof (items[0])); i++) items[i].data = todo; popup_menu (items, sizeof (items) / sizeof (items[0]), event); break; default: break; } } /* * once we get a call back stating that a column * has been resized never ever automatically resize again */ void column_resized (GtkWidget *widget, GncalTodo *todo) { /* disabling autoresize of columns */ if (todo_list_autoresize && !todo_list_redraw_in_progess){ todo_list_autoresize = 0; } } /* * restore the previously set settings for sorting the * todo list */ static void init_column_sorting (GtkCList *clist) { /* due date isn't shown so we can't sort by it */ if (todo_current_sort_column == 1 && ! todo_show_due_date) todo_current_sort_column = 0; clist->sort_type = todo_current_sort_type; clist->sort_column = todo_current_sort_column; gtk_clist_set_sort_column (clist, todo_current_sort_column); gtk_clist_sort (clist); } static void todo_click_column (GtkCList *clist, gint column, gpointer data) { if (column == clist->sort_column) { if (clist->sort_type == GTK_SORT_ASCENDING) { clist->sort_type = GTK_SORT_DESCENDING; todo_current_sort_type = GTK_SORT_DESCENDING; } else { clist->sort_type = GTK_SORT_ASCENDING; todo_current_sort_type = GTK_SORT_ASCENDING; } } else { gtk_clist_set_sort_column (clist, column); todo_current_sort_column = column; } gtk_clist_sort (clist); /* * save the sorting preferences cause I hate to have the user * click twice */ gnome_config_set_int("/calendar/Todo/sort_column", todo_current_sort_column); gnome_config_set_int("/calendar/Todo/sort_type", todo_current_sort_type); gnome_config_sync(); } static void gncal_todo_init (GncalTodo *todo) { GtkWidget *w; GtkWidget *sw; GtkWidget *hbox; gchar *titles[2] = {"Summary","Due Date"}; gtk_box_set_spacing (GTK_BOX (todo), 4); /* Label */ w = gtk_label_new (_("To-do list")); gtk_misc_set_alignment (GTK_MISC (w), 0.0, 0.5); gtk_box_pack_start (GTK_BOX (todo), w, FALSE, FALSE, 0); gtk_widget_show (w); /* Clist */ sw = gtk_scrolled_window_new (NULL, NULL); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_box_pack_start (GTK_BOX (todo), sw, TRUE, TRUE, 0); gtk_widget_show (sw); w = gtk_clist_new_with_titles(2, titles); todo->clist = GTK_CLIST (w); gtk_clist_set_selection_mode (todo->clist, GTK_SELECTION_BROWSE); gtk_signal_connect (GTK_OBJECT (todo->clist), "select_row", (GtkSignalFunc) clist_row_selected, todo); gtk_signal_connect (GTK_OBJECT (todo->clist), "resize_column", (GtkSignalFunc) column_resized, todo); gtk_signal_connect (GTK_OBJECT (todo->clist), "click_column", (GtkSignalFunc) todo_click_column, NULL); gtk_container_add (GTK_CONTAINER (sw), w); gtk_widget_show (w); /* Box for buttons */ hbox = gtk_hbox_new (TRUE, 4); gtk_box_pack_start (GTK_BOX (todo), hbox, FALSE, FALSE, 0); gtk_widget_show (hbox); /* Add */ w = gtk_button_new_with_label (_("Add...")); gtk_signal_connect (GTK_OBJECT (w), "clicked", (GtkSignalFunc) add_activated, todo); gtk_box_pack_start (GTK_BOX (hbox), w, TRUE, TRUE, 0); gtk_widget_show (w); /* Edit */ w = gtk_button_new_with_label (_("Edit...")); todo->edit_button = w; gtk_widget_set_sensitive (w, FALSE); gtk_signal_connect (GTK_OBJECT (w), "clicked", (GtkSignalFunc) edit_activated, todo); gtk_box_pack_start (GTK_BOX (hbox), w, TRUE, TRUE, 0); gtk_widget_show (w); /* Delete */ w = gtk_button_new_with_label (_("Delete")); todo->delete_button = w; gtk_widget_set_sensitive (w, FALSE); gtk_signal_connect (GTK_OBJECT (w), "clicked", (GtkSignalFunc) delete_activated, todo); gtk_box_pack_start (GTK_BOX (hbox), w, TRUE, TRUE, 0); gtk_widget_show (w); } GtkWidget * gncal_todo_new (GnomeCalendar *calendar) { GncalTodo *todo; g_return_val_if_fail (calendar != NULL, NULL); todo = gtk_type_new (gncal_todo_get_type ()); todo->calendar = calendar; gncal_todo_update (todo, NULL, 0); return GTK_WIDGET (todo); } char * convert_time_t_to_char (time_t t) { char *buffer; struct tm *tm; buffer = g_malloc(15); tm = localtime (&t); strftime(buffer, 15, "%m/%d/%y", tm); return buffer; } GtkStyle * make_overdue_todo_style(GncalTodo *todo) { GtkStyle *overdue_style = NULL; GdkColor overdue_color; /*make the overdue color configurable */ overdue_color.red = color_props[COLOR_PROP_OVERDUE_TODO].r; overdue_color.green = color_props[COLOR_PROP_OVERDUE_TODO].g; overdue_color.blue = color_props[COLOR_PROP_OVERDUE_TODO].b; overdue_style = gtk_style_copy (GTK_WIDGET (todo->clist)->style); overdue_style->base[GTK_STATE_NORMAL] = overdue_color; return overdue_style; } static void insert_in_clist (GncalTodo *todo, iCalObject *ico) { int i; char *text[2]; static GtkStyle *overdue_style = NULL; /* setup the over due style if we haven't already, or it changed.*/ if (todo_style_changed || !overdue_style) { /* free the old style cause its not needed anymore */ if(!overdue_style) g_free(overdue_style); overdue_style = make_overdue_todo_style(todo); todo_style_changed = 0; } text[0] = ico->summary; /* * right now column 0 will be the summary * and column 1 will be the due date. * WISH: this should be able to be changed on the fly */ if(ico->dtend && todo_show_due_date) { text[1] = convert_time_t_to_char(ico->dtend); /* Append the data's pointer so later it can be properly freed */ todo->data_ptrs = g_slist_append (todo->data_ptrs, text[1]); } else text[1] = NULL; i = gtk_clist_append (todo->clist, text); gtk_clist_set_row_data (todo->clist, i, ico); /* * determine if the task is overdue.. * if so mark with the apropriate style */ if(todo_due_date_overdue_highlight) { if(ico->dtend < time(NULL)) gtk_clist_set_row_style(todo->clist, i, overdue_style); } /* keep the list in order */ gtk_clist_sort (todo->clist); } void gncal_todo_update (GncalTodo *todo, iCalObject *ico, int flags) { GList *list; GSList *current_list; g_return_if_fail (todo != NULL); g_return_if_fail (GNCAL_IS_TODO (todo)); /* * shut down the resize handler cause we are playing with the list. * In otherwords turn off the event handler */ todo_list_redraw_in_progess =1; /* freeze the list */ gtk_clist_freeze (todo->clist); init_column_sorting (todo->clist); /* * before here we have to free some of the memory that * stores the due date, or else we have a memory leak. * luckily all of the pointers are stored in todo->data_ptrs; */ /* check on the columns that we should display */ /* check for due date */ if(todo_show_due_date) gtk_clist_set_column_visibility (todo->clist, 1, 1); else gtk_clist_set_column_visibility (todo->clist, 1, 0); /* free the memory locations that were used in the previous display */ for (current_list = todo->data_ptrs; current_list != NULL; current_list = g_slist_next(current_list)){ g_free(current_list->data); } /* free the list and clear out the pointer */ g_slist_free(todo->data_ptrs); todo->data_ptrs = NULL; gtk_clist_clear (todo->clist); for (list = todo->calendar->cal->todo; list; list = list->next) insert_in_clist (todo, list->data); /* if we are autoresizing then do it now */ if(todo_list_autoresize && todo->clist->rows != 0) gtk_clist_columns_autosize (todo->clist); gtk_clist_thaw (todo->clist); gtk_widget_set_sensitive (todo->edit_button, (todo->clist->selection != NULL)); gtk_widget_set_sensitive (todo->delete_button, (todo->clist->selection != NULL)); todo_list_redraw_in_progess = 0; }