/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ /* e-tasks.c * * Copyright (C) 2001-2003 Ximian, Inc. * * 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. * * Authors: Federico Mena Quintero <federico@ximian.com> * Damon Chaplin <damon@ximian.com> * Rodrigo Moya <rodrigo@ximian.com> */ #include <config.h> #include <gnome.h> #include <libgnomevfs/gnome-vfs-ops.h> #include <gal/util/e-util.h> #include <gal/e-table/e-table-scrolled.h> #include <gal/menus/gal-view-instance.h> #include <gal/menus/gal-view-factory-etable.h> #include <gal/menus/gal-view-etable.h> #include <gtkhtml/gtkhtml.h> #include <gtkhtml/gtkhtml-stream.h> #include "e-util/e-categories-config.h" #include "e-util/e-time-utils.h" #include "e-util/e-url.h" #include "cal-util/timeutil.h" #include "widgets/menus/gal-view-menus.h" #include "dialogs/delete-error.h" #include "dialogs/task-editor.h" #include "cal-search-bar.h" #include "calendar-config.h" #include "calendar-component.h" #include "comp-util.h" #include "misc.h" #include "e-tasks.h" /* A list of all of the ETasks widgets in use. We use this to update the user preference settings. This will change when we switch to GConf. */ static GList *all_tasks = NULL; /* Private part of the GnomeCalendar structure */ struct _ETasksPrivate { /* The calendar client object we monitor */ CalClient *client; CalQuery *query; /* The ECalendarTable showing the tasks. */ GtkWidget *tasks_view; /* Calendar search bar for tasks */ GtkWidget *search_bar; /* The HTML widget to display the task's details */ GtkWidget *html; gchar *current_uid; /* View instance and the view menus handler */ GalViewInstance *view_instance; GalViewMenus *view_menus; }; static void e_tasks_class_init (ETasksClass *class); static void e_tasks_init (ETasks *tasks); static void setup_widgets (ETasks *tasks); static void e_tasks_destroy (GtkObject *object); static void cal_opened_cb (CalClient *client, CalClientOpenStatus status, gpointer data); static void backend_error_cb (CalClient *client, const char *message, gpointer data); /* Signal IDs */ enum { SELECTION_CHANGED, LAST_SIGNAL }; static GtkTableClass *parent_class; static guint e_tasks_signals[LAST_SIGNAL] = { 0 }; E_MAKE_TYPE (e_tasks, "ETasks", ETasks, e_tasks_class_init, e_tasks_init, GTK_TYPE_TABLE) /* Class initialization function for the gnome calendar */ static void e_tasks_class_init (ETasksClass *class) { GtkObjectClass *object_class; object_class = (GtkObjectClass *) class; parent_class = gtk_type_class (GTK_TYPE_TABLE); e_tasks_signals[SELECTION_CHANGED] = gtk_signal_new ("selection_changed", GTK_RUN_LAST, G_TYPE_FROM_CLASS (object_class), GTK_SIGNAL_OFFSET (ETasksClass, selection_changed), gtk_marshal_NONE__INT, GTK_TYPE_NONE, 1, GTK_TYPE_INT); object_class->destroy = e_tasks_destroy; class->selection_changed = NULL; } /* Object initialization function for the gnome calendar */ static void e_tasks_init (ETasks *tasks) { ETasksPrivate *priv; priv = g_new0 (ETasksPrivate, 1); tasks->priv = priv; priv->client = NULL; priv->query = NULL; priv->view_instance = NULL; priv->view_menus = NULL; priv->current_uid = NULL; } /* Converts a time_t to a string, relative to the specified timezone */ static char * timet_to_str_with_zone (time_t t, icaltimezone *zone) { struct icaltimetype itt; struct tm tm; char buf[256]; if (t == -1) return g_strdup (_("invalid time")); itt = icaltime_from_timet_with_zone (t, FALSE, zone); tm = icaltimetype_to_tm (&itt); e_time_format_date_and_time (&tm, calendar_config_get_24_hour_format (), FALSE, FALSE, buf, sizeof (buf)); return g_strdup (buf); } static void write_html (GtkHTMLStream *stream, CalComponent *comp) { CalComponentText text; CalComponentDateTime dt; gchar *buf, *str; icaltimezone *current_zone; GSList *l; icalproperty_status status; int *priority_value; g_return_if_fail (IS_CAL_COMPONENT (comp)); str = calendar_config_get_timezone (); if (str && str[0]) { current_zone = icaltimezone_get_builtin_timezone (str); } else current_zone = icaltimezone_get_utc_timezone (); /* write document header */ cal_component_get_summary (comp, &text); gtk_html_stream_printf (stream, "<HTML><BODY><H1>%s</H1>", text.value); /* write icons for the categories */ cal_component_get_categories_list (comp, &l); if (l) { GSList *node; for (node = l; node != NULL; node = node->next) { const char *icon_file; icon_file = e_categories_config_get_icon_file_for ((const char *) node->data); if (icon_file) { gtk_html_stream_printf (stream, "<IMG ALT=\"%s\" SRC=\"file://%s\">", (const char *) node->data, icon_file); } } cal_component_free_categories_list (l); } /* write summary */ gtk_html_stream_printf (stream, "<BR><BR><BR><TABLE BORDER=\"0\" WIDTH=\"80%%\">" "<TR><TD VALIGN=\"TOP\" ALIGN=\"RIGHT\" WIDTH=\"15%%\"><B>%s</B></TD><TD>%s</TD></TR>", _("Summary:"), text.value); /* write start date */ cal_component_get_dtstart (comp, &dt); if (dt.value != NULL) { buf = timet_to_str_with_zone (icaltime_as_timet (*dt.value), current_zone); str = g_locale_to_utf8 (buf, -1, NULL, NULL, NULL); g_free (buf); } else str = g_strdup (""); gtk_html_stream_printf (stream, "<TR><TD VALIGN=\"TOP\" ALIGN=\"RIGHT\"><B>%s</B></TD><TD>%s</TD></TR>", _("Start Date:"), str); cal_component_free_datetime (&dt); g_free (str); /* write Due Date */ cal_component_get_due (comp, &dt); if (dt.value != NULL) { buf = timet_to_str_with_zone (icaltime_as_timet (*dt.value), current_zone); str = g_locale_to_utf8 (buf, -1, NULL, NULL, NULL); g_free (buf); } else str = g_strdup (""); gtk_html_stream_printf (stream, "<TR><TD VALIGN=\"TOP\" ALIGN=\"RIGHT\"><B>%s</B></TD><TD>%s</TD></TR>", _("Due Date:"), str); cal_component_free_datetime (&dt); g_free (str); /* write status */ gtk_html_stream_printf (stream, "<TR><TD VALIGN=\"TOP\" ALIGN=\"RIGHT\"><B>%s</B></TD>", _("Status:")); cal_component_get_status (comp, &status); switch (status) { case ICAL_STATUS_INPROCESS : str = g_strdup (_("In Progress")); break; case ICAL_STATUS_COMPLETED : str = g_strdup (_("Completed")); break; case ICAL_STATUS_CANCELLED : str = g_strdup (_("Cancelled")); break; case ICAL_STATUS_NONE : default : str = g_strdup (_("Not Started")); break; } gtk_html_stream_printf (stream, "<TD>%s</TD></TR>", str); g_free (str); /* write priority */ gtk_html_stream_printf (stream, "<TR><TD VALIGN=\"TOP\" ALIGN=\"RIGHT\"><B>%s</B></TD>", _("Priority:")); cal_component_get_priority (comp, &priority_value); if (priority_value) { if (*priority_value == 0) str = g_strdup (""); else if (*priority_value <= 4) str = g_strdup (_("High")); else if (*priority_value == 5) str = g_strdup (_("Normal")); else str = g_strdup (_("Low")); gtk_html_stream_printf (stream, "<TD>%s</TD></TR>", str); g_free (str); cal_component_free_priority (priority_value); } else gtk_html_stream_printf (stream, "<TD></TD></TR>"); /* write description and URL */ gtk_html_stream_printf (stream, "<TR><TD COLSPAN=\"2\"><HR></TD></TR>"); gtk_html_stream_printf (stream, "<TR><TD VALIGN=\"TOP\" ALIGN=\"RIGHT\"><B>%s</B></TD>", _("Description:")); cal_component_get_description_list (comp, &l); if (l) { GSList *node; gtk_html_stream_printf (stream, "<TD>"); for (node = l; node != NULL; node = node->next) { gint i; GString *str = g_string_new ("");; text = * (CalComponentText *) node->data; for (i = 0; i < strlen (text.value ? text.value : 0); i++) { if (text.value[i] == '\n') str = g_string_append (str, "<BR>"); else if (text.value[i] == '<') str = g_string_append (str, "<"); else if (text.value[i] == '>') str = g_string_append (str, ">"); else str = g_string_append_c (str, text.value[i]); } gtk_html_stream_printf (stream, str->str); g_string_free (str, TRUE); } gtk_html_stream_printf (stream, "</TD></TR>"); cal_component_free_text_list (l); } else gtk_html_stream_printf (stream, "<TD></TD></TR>"); /* URL */ gtk_html_stream_printf (stream, "<TR><TD VALIGN=\"TOP\" ALIGN=\"RIGHT\"><B>%s</B></TD>", _("Web Page:")); cal_component_get_url (comp, (const char **) &str); if (str) gtk_html_stream_printf (stream, "<TD><A HREF=\"%s\">%s</A></TD></TR>", str, str); else gtk_html_stream_printf (stream, "<TD></TD></TR>"); gtk_html_stream_printf (stream, "</TABLE>"); /* close document */ gtk_html_stream_printf (stream, "</BODY></HTML>"); } static void on_link_clicked (GtkHTML *html, const char *url, gpointer data) { GError *err = NULL; gnome_url_show (url, &err); if (err) { g_warning ("gnome_url_show: %s", err->message); g_error_free (err); } } /* Callback used when the cursor changes in the table */ static void table_cursor_change_cb (ETable *etable, int row, gpointer data) { ETasks *tasks; ETasksPrivate *priv; int n_selected; tasks = E_TASKS (data); priv = tasks->priv; n_selected = e_table_selected_count (etable); /* update the HTML widget */ if (n_selected == 1) { GtkHTMLStream *stream; ECalModel *model; ECalModelComponent *comp_data; CalComponent *comp; const char *uid; model = e_calendar_table_get_model (E_CALENDAR_TABLE (priv->tasks_view)); stream = gtk_html_begin (GTK_HTML (priv->html)); comp_data = e_cal_model_get_component_at (model, e_table_get_cursor_row (etable)); comp = cal_component_new (); cal_component_set_icalcomponent (comp, icalcomponent_new_clone (comp_data->icalcomp)); write_html (stream, comp); gtk_html_stream_close (stream, GTK_HTML_STREAM_OK); cal_component_get_uid (comp, &uid); if (priv->current_uid) g_free (priv->current_uid); priv->current_uid = g_strdup (uid); g_object_unref (comp); } else gtk_html_load_empty (GTK_HTML (priv->html)); } /* Callback used when the selection changes in the table. */ static void table_selection_change_cb (ETable *etable, gpointer data) { ETasks *tasks; int n_selected; tasks = E_TASKS (data); n_selected = e_table_selected_count (etable); gtk_signal_emit (GTK_OBJECT (tasks), e_tasks_signals[SELECTION_CHANGED], n_selected); } /* Callback used when the sexp in the search bar changes */ static void search_bar_sexp_changed_cb (CalSearchBar *cal_search, const char *sexp, gpointer data) { ETasks *tasks; ETasksPrivate *priv; ECalModel *model; tasks = E_TASKS (data); priv = tasks->priv; model = e_calendar_table_get_model (E_CALENDAR_TABLE (priv->tasks_view)); e_cal_model_set_query (model, sexp); } /* Callback used when the selected category in the search bar changes */ static void search_bar_category_changed_cb (CalSearchBar *cal_search, const char *category, gpointer data) { ETasks *tasks; ETasksPrivate *priv; ECalModel *model; tasks = E_TASKS (data); priv = tasks->priv; model = e_calendar_table_get_model (E_CALENDAR_TABLE (priv->tasks_view)); e_cal_model_set_default_category (model, category); } /* Callback used when the user selects a URL in the HTML widget */ static void url_requested_cb (GtkHTML *html, const char *url, GtkHTMLStream *stream, gpointer data) { if (!strncmp ("file:///", url, strlen ("file:///"))) { GnomeVFSHandle *handle; GnomeVFSResult result; char buffer[4096]; if (gnome_vfs_open (&handle, url, GNOME_VFS_OPEN_READ) == GNOME_VFS_OK) { do { GnomeVFSFileSize bread; result = gnome_vfs_read (handle, buffer, sizeof (buffer), &bread); if (result == GNOME_VFS_OK) gtk_html_stream_write (stream, buffer, bread); } while (result == GNOME_VFS_OK); gnome_vfs_close (handle); } } } static gboolean vpaned_resized_cb (GtkWidget *widget, GdkEventButton *event, ETasks *tasks) { calendar_config_set_task_vpane_pos (gtk_paned_get_position (GTK_PANED (widget))); return FALSE; } #define E_TASKS_TABLE_DEFAULT_STATE \ "<?xml version=\"1.0\"?>" \ "<ETableState>" \ "<column source=\"13\"/>" \ "<column source=\"14\"/>" \ "<column source=\"9\"/>" \ "<column source=\"5\"/>" \ "<grouping/>" \ "</ETableState>" static void setup_widgets (ETasks *tasks) { ETasksPrivate *priv; ETable *etable; GtkWidget *paned, *scroll; priv = tasks->priv; priv->search_bar = cal_search_bar_new (); g_signal_connect (priv->search_bar, "sexp_changed", G_CALLBACK (search_bar_sexp_changed_cb), tasks); g_signal_connect (priv->search_bar, "category_changed", G_CALLBACK (search_bar_category_changed_cb), tasks); gtk_table_attach (GTK_TABLE (tasks), priv->search_bar, 0, 1, 0, 1, GTK_EXPAND | GTK_FILL | GTK_SHRINK, 0, 0, 0); gtk_widget_show (priv->search_bar); /* add the paned widget for the task list and task detail areas */ paned = gtk_vpaned_new (); gtk_paned_set_position (GTK_PANED (paned), calendar_config_get_task_vpane_pos ()); g_signal_connect (G_OBJECT (paned), "button_release_event", G_CALLBACK (vpaned_resized_cb), tasks); gtk_table_attach (GTK_TABLE (tasks), paned, 0, 1, 1, 2, GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0); gtk_widget_show (paned); /* create the task list */ priv->tasks_view = e_calendar_table_new (); etable = e_table_scrolled_get_table ( E_TABLE_SCROLLED (E_CALENDAR_TABLE (priv->tasks_view)->etable)); e_table_set_state (etable, E_TASKS_TABLE_DEFAULT_STATE); gtk_paned_add1 (GTK_PANED (paned), priv->tasks_view); gtk_widget_show (priv->tasks_view); calendar_config_configure_e_calendar_table (E_CALENDAR_TABLE (priv->tasks_view)); g_signal_connect (etable, "cursor_change", G_CALLBACK (table_cursor_change_cb), tasks); g_signal_connect (etable, "selection_change", G_CALLBACK (table_selection_change_cb), tasks); /* create the task detail */ priv->html = gtk_html_new (); gtk_html_set_default_content_type (GTK_HTML (priv->html), "charset=utf-8"); gtk_html_load_empty (GTK_HTML (priv->html)); g_signal_connect (G_OBJECT (priv->html), "url_requested", G_CALLBACK (url_requested_cb), NULL); g_signal_connect (G_OBJECT (priv->html), "link_clicked", G_CALLBACK (on_link_clicked), tasks); gtk_widget_pop_colormap (); scroll = gtk_scrolled_window_new (NULL, NULL); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scroll), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scroll), GTK_SHADOW_IN); gtk_container_add (GTK_CONTAINER (scroll), priv->html); gtk_paned_add2 (GTK_PANED (paned), scroll); gtk_widget_show_all (scroll); } /* Callback used when the set of categories changes in the calendar client */ static void client_categories_changed_cb (CalClient *client, GPtrArray *categories, gpointer data) { ETasks *tasks; ETasksPrivate *priv; tasks = E_TASKS (data); priv = tasks->priv; cal_search_bar_set_categories (CAL_SEARCH_BAR (priv->search_bar), categories); } static void client_obj_updated_cb (CalClient *client, const char *uid, gpointer data) { ETasks *tasks; ETasksPrivate *priv; tasks = E_TASKS (data); priv = tasks->priv; if (priv->current_uid) { if (!strcmp (uid, priv->current_uid)) { ETable *etable; etable = e_table_scrolled_get_table ( E_TABLE_SCROLLED (E_CALENDAR_TABLE (priv->tasks_view)->etable)); table_cursor_change_cb (etable, 0, tasks); } } } GtkWidget * e_tasks_construct (ETasks *tasks) { ETasksPrivate *priv; ECalModel *model; g_return_val_if_fail (tasks != NULL, NULL); g_return_val_if_fail (E_IS_TASKS (tasks), NULL); priv = tasks->priv; setup_widgets (tasks); priv->client = cal_client_new (); if (!priv->client) return NULL; g_signal_connect (priv->client, "cal_opened", G_CALLBACK (cal_opened_cb), tasks); g_signal_connect (priv->client, "backend_error", G_CALLBACK (backend_error_cb), tasks); g_signal_connect (priv->client, "categories_changed", G_CALLBACK (client_categories_changed_cb), tasks); g_signal_connect (priv->client, "obj_updated", G_CALLBACK (client_obj_updated_cb), tasks); model = e_calendar_table_get_model (E_CALENDAR_TABLE (priv->tasks_view)); g_assert (model != NULL); e_cal_model_add_client (model, priv->client); return GTK_WIDGET (tasks); } GtkWidget * e_tasks_new (void) { ETasks *tasks; tasks = g_object_new (e_tasks_get_type (), NULL); if (!e_tasks_construct (tasks)) { g_message ("e_tasks_new(): Could not construct the tasks GUI"); g_object_unref (tasks); return NULL; } all_tasks = g_list_prepend (all_tasks, tasks); return GTK_WIDGET (tasks); } void e_tasks_set_ui_component (ETasks *tasks, BonoboUIComponent *ui_component) { g_return_if_fail (E_IS_TASKS (tasks)); g_return_if_fail (ui_component == NULL || BONOBO_IS_UI_COMPONENT (ui_component)); e_search_bar_set_ui_component (E_SEARCH_BAR (tasks->priv->search_bar), ui_component); } static void e_tasks_destroy (GtkObject *object) { ETasks *tasks; ETasksPrivate *priv; g_return_if_fail (object != NULL); g_return_if_fail (E_IS_TASKS (object)); tasks = E_TASKS (object); priv = tasks->priv; if (priv) { if (priv->client) { g_object_unref (priv->client); priv->client = NULL; } if (priv->current_uid) { g_free (priv->current_uid); priv->current_uid = NULL; } g_free (priv); tasks->priv = NULL; all_tasks = g_list_remove (all_tasks, tasks); } if (GTK_OBJECT_CLASS (parent_class)->destroy) (* GTK_OBJECT_CLASS (parent_class)->destroy) (object); } static void set_status_message (ETasks *tasks, const char *message) { ETasksPrivate *priv; priv = tasks->priv; e_calendar_table_set_status_message (E_CALENDAR_TABLE (priv->tasks_view), message); } gboolean e_tasks_open (ETasks *tasks, char *file) { ETasksPrivate *priv; char *message; EUri *uri; char *real_uri; char *urinopwd; g_return_val_if_fail (tasks != NULL, FALSE); g_return_val_if_fail (E_IS_TASKS (tasks), FALSE); g_return_val_if_fail (file != NULL, FALSE); priv = tasks->priv; uri = e_uri_new (file); if (!uri || !g_strncasecmp (uri->protocol, "file", 4)) real_uri = g_concat_dir_and_file (file, "tasks.ics"); else real_uri = g_strdup (file); urinopwd = get_uri_without_password (real_uri); message = g_strdup_printf (_("Opening tasks at %s"), urinopwd); set_status_message (tasks, message); g_free (message); g_free (urinopwd); if (!cal_client_open_calendar (priv->client, real_uri, FALSE)) { g_message ("e_tasks_open(): Could not issue the request"); g_free (real_uri); e_uri_free (uri); return FALSE; } g_free (real_uri); e_uri_free (uri); return TRUE; } /* Displays an error to indicate that loading a calendar failed */ static void load_error (ETasks *tasks, const char *uri) { char *msg; char *urinopwd; urinopwd = get_uri_without_password (uri); msg = g_strdup_printf (_("Could not load the tasks in `%s'"), urinopwd); gnome_error_dialog_parented (msg, GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (tasks)))); g_free (msg); g_free (urinopwd); } /* Displays an error to indicate that the specified URI method is not supported */ static void method_error (ETasks *tasks, const char *uri) { char *msg; char *urinopwd; urinopwd = get_uri_without_password (uri); msg = g_strdup_printf (_("The method required to load `%s' is not supported"), urinopwd); gnome_error_dialog_parented (msg, GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (tasks)))); g_free (msg); g_free (urinopwd); } /* Displays an error to indicate permission problems */ static void permission_error (ETasks *tasks, const char *uri) { char *msg; char *urinopwd; urinopwd = get_uri_without_password (uri); msg = g_strdup_printf (_("You don't have permission to open the folder in `%s'"), urinopwd); gnome_error_dialog_parented (msg, GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (tasks)))); g_free (msg); g_free (urinopwd); } /* Callback from the calendar client when a calendar is opened */ static void cal_opened_cb (CalClient *client, CalClientOpenStatus status, gpointer data) { ETasks *tasks; ETasksPrivate *priv; char *location; icaltimezone *zone; tasks = E_TASKS (data); priv = tasks->priv; set_status_message (tasks, NULL); switch (status) { case CAL_CLIENT_OPEN_SUCCESS: /* Everything is OK */ /* Set the client's default timezone, if we have one. */ location = calendar_config_get_timezone (); zone = icaltimezone_get_builtin_timezone (location); if (zone) cal_client_set_default_timezone (client, zone); return; case CAL_CLIENT_OPEN_ERROR: load_error (tasks, cal_client_get_uri (client)); break; case CAL_CLIENT_OPEN_NOT_FOUND: /* bullshit; we did not specify only_if_exists */ g_assert_not_reached (); return; case CAL_CLIENT_OPEN_METHOD_NOT_SUPPORTED: method_error (tasks, cal_client_get_uri (client)); break; case CAL_CLIENT_OPEN_PERMISSION_DENIED: permission_error (tasks, cal_client_get_uri (client)); break; default: g_assert_not_reached (); } } /* Callback from the calendar client when an error occurs in the backend */ static void backend_error_cb (CalClient *client, const char *message, gpointer data) { ETasks *tasks; ETasksPrivate *priv; char *errmsg; char *urinopwd; tasks = E_TASKS (data); priv = tasks->priv; urinopwd = get_uri_without_password (cal_client_get_uri (client)); errmsg = g_strdup_printf (_("Error on %s:\n %s"), urinopwd, message); gnome_error_dialog_parented (errmsg, GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (tasks)))); g_free (errmsg); g_free (urinopwd); } /** * e_tasks_get_cal_client: * @tasks: An #ETasks. * * Queries the calendar client interface object that a tasks view is using. * * Return value: A calendar client interface object. **/ CalClient * e_tasks_get_cal_client (ETasks *tasks) { ETasksPrivate *priv; g_return_val_if_fail (E_IS_TASKS (tasks), NULL); priv = tasks->priv; return priv->client; } void e_tasks_new_task (ETasks *tasks) { ETasksPrivate *priv; TaskEditor *tedit; CalComponent *comp; const char *category; g_return_if_fail (E_IS_TASKS (tasks)); priv = tasks->priv; tedit = task_editor_new (priv->client); comp = cal_comp_task_new_with_defaults (priv->client); category = cal_search_bar_get_category (CAL_SEARCH_BAR (priv->search_bar)); cal_component_set_categories (comp, category); comp_editor_edit_comp (COMP_EDITOR (tedit), comp); g_object_unref (comp); comp_editor_focus (COMP_EDITOR (tedit)); } /** * e_tasks_complete_selected: * @tasks: A tasks control widget * * Marks the selected tasks complete **/ void e_tasks_complete_selected (ETasks *tasks) { ETasksPrivate *priv; ECalendarTable *cal_table; g_return_if_fail (tasks != NULL); g_return_if_fail (E_IS_TASKS (tasks)); priv = tasks->priv; cal_table = E_CALENDAR_TABLE (priv->tasks_view); set_status_message (tasks, _("Completing tasks...")); e_calendar_table_complete_selected (cal_table); set_status_message (tasks, NULL); } /** * e_tasks_delete_selected: * @tasks: A tasks control widget. * * Deletes the selected tasks in the task list. **/ void e_tasks_delete_selected (ETasks *tasks) { ETasksPrivate *priv; ECalendarTable *cal_table; g_return_if_fail (tasks != NULL); g_return_if_fail (E_IS_TASKS (tasks)); priv = tasks->priv; cal_table = E_CALENDAR_TABLE (priv->tasks_view); set_status_message (tasks, _("Deleting selected objects...")); e_calendar_table_delete_selected (cal_table); set_status_message (tasks, NULL); } static char * create_sexp (void) { char *sexp; sexp = g_strdup ("(and (= (get-vtype) \"VTODO\") (is-completed?))"); #if 0 g_print ("Calendar model sexp:\n%s\n", sexp); #endif return sexp; } /* Callback used when a component is updated in the live query */ static void query_obj_updated_cb (CalQuery *query, const char *uid, gboolean query_in_progress, int n_scanned, int total, gpointer data) { ETasks *tasks; ETasksPrivate *priv; tasks = E_TASKS (data); priv = tasks->priv; delete_error_dialog (cal_client_remove_object (priv->client, uid), CAL_COMPONENT_TODO); } /* Callback used when an evaluation error occurs when running a query */ static void query_eval_error_cb (CalQuery *query, const char *error_str, gpointer data) { ETasks *tasks; ETasksPrivate *priv; tasks = E_TASKS (data); priv = tasks->priv; g_warning ("eval error: %s\n", error_str); set_status_message (tasks, NULL); g_signal_handlers_disconnect_matched (priv->query, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, tasks); g_object_unref (priv->query); priv->query = NULL; } static void query_query_done_cb (CalQuery *query, CalQueryDoneStatus status, const char *error_str, gpointer data) { ETasks *tasks; ETasksPrivate *priv; tasks = E_TASKS (data); priv = tasks->priv; if (status != CAL_QUERY_DONE_SUCCESS) g_warning ("query done: %s\n", error_str); set_status_message (tasks, NULL); g_signal_handlers_disconnect_matched (priv->query, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, tasks); g_object_unref (priv->query); priv->query = NULL; } /** * e_tasks_expunge: * @tasks: A tasks control widget * * Removes all tasks marked as completed **/ void e_tasks_delete_completed (ETasks *tasks) { ETasksPrivate *priv; char *sexp; g_return_if_fail (tasks != NULL); g_return_if_fail (E_IS_TASKS (tasks)); priv = tasks->priv; /* If we have a query, we are already expunging */ if (priv->query) return; sexp = create_sexp (); set_status_message (tasks, _("Expunging")); priv->query = cal_client_get_query (priv->client, sexp); g_free (sexp); if (!priv->query) { set_status_message (tasks, NULL); g_message ("update_query(): Could not create the query"); return; } g_signal_connect (priv->query, "obj_updated", G_CALLBACK (query_obj_updated_cb), tasks); g_signal_connect (priv->query, "query_done", G_CALLBACK (query_query_done_cb), tasks); g_signal_connect (priv->query, "eval_error", G_CALLBACK (query_eval_error_cb), tasks); } /* Callback used from the view collection when we need to display a new view */ static void display_view_cb (GalViewInstance *instance, GalView *view, gpointer data) { ETasks *tasks; tasks = E_TASKS (data); if (GAL_IS_VIEW_ETABLE (view)) { gal_view_etable_attach_table (GAL_VIEW_ETABLE (view), e_table_scrolled_get_table (E_TABLE_SCROLLED (E_CALENDAR_TABLE (tasks->priv->tasks_view)->etable))); } } /** * e_tasks_setup_view_menus: * @tasks: A tasks widget. * @uic: UI controller to use for the menus. * * Sets up the #GalView menus for a tasks control. This function should be * called from the Bonobo control activation callback for this tasks control. * Also, the menus should be discarded using e_tasks_discard_view_menus(). **/ void e_tasks_setup_view_menus (ETasks *tasks, BonoboUIComponent *uic) { ETasksPrivate *priv; GalViewFactory *factory; ETableSpecification *spec; char *dir; static GalViewCollection *collection = NULL; g_return_if_fail (tasks != NULL); g_return_if_fail (E_IS_TASKS (tasks)); g_return_if_fail (uic != NULL); g_return_if_fail (BONOBO_IS_UI_COMPONENT (uic)); priv = tasks->priv; g_return_if_fail (priv->view_instance == NULL); g_assert (priv->view_instance == NULL); g_assert (priv->view_menus == NULL); /* Create the view instance */ if (collection == NULL) { collection = gal_view_collection_new (); gal_view_collection_set_title (collection, _("Tasks")); dir = gnome_util_prepend_user_home ("/evolution/views/tasks/"); gal_view_collection_set_storage_directories (collection, EVOLUTION_GALVIEWSDIR "/tasks/", dir); g_free (dir); /* Create the views */ spec = e_table_specification_new (); e_table_specification_load_from_file (spec, EVOLUTION_ETSPECDIR "/e-calendar-table.etspec"); factory = gal_view_factory_etable_new (spec); g_object_unref (spec); gal_view_collection_add_factory (collection, factory); g_object_unref (factory); /* Load the collection and create the menus */ gal_view_collection_load (collection); } priv->view_instance = gal_view_instance_new (collection, cal_client_get_uri (priv->client)); priv->view_menus = gal_view_menus_new (priv->view_instance); gal_view_menus_apply (priv->view_menus, uic, NULL); g_signal_connect (priv->view_instance, "display_view", G_CALLBACK (display_view_cb), tasks); display_view_cb (priv->view_instance, gal_view_instance_get_current_view (priv->view_instance), tasks); } /** * e_tasks_discard_view_menus: * @tasks: A tasks widget. * * Discards the #GalView menus used by a tasks control. This function should be * called from the Bonobo control deactivation callback for this tasks control. * The menus should have been set up with e_tasks_setup_view_menus(). **/ void e_tasks_discard_view_menus (ETasks *tasks) { ETasksPrivate *priv; g_return_if_fail (tasks != NULL); g_return_if_fail (E_IS_TASKS (tasks)); priv = tasks->priv; g_return_if_fail (priv->view_instance != NULL); g_assert (priv->view_instance != NULL); g_assert (priv->view_menus != NULL); g_object_unref (priv->view_instance); priv->view_instance = NULL; g_object_unref (priv->view_menus); priv->view_menus = NULL; } /** * e_tasks_get_calendar_table: * @tasks: A tasks widget. * * Queries the #ECalendarTable contained in a tasks widget. * * Return value: The #ECalendarTable that the tasks widget uses to display its * information. **/ ECalendarTable * e_tasks_get_calendar_table (ETasks *tasks) { ETasksPrivate *priv; g_return_val_if_fail (tasks != NULL, NULL); g_return_val_if_fail (E_IS_TASKS (tasks), NULL); priv = tasks->priv; return E_CALENDAR_TABLE (priv->tasks_view); } /* This updates all the preference settings for all the ETasks widgets in use. */ void e_tasks_update_all_config_settings (void) { ETasks *tasks; ETasksPrivate *priv; GList *elem; char *location; icaltimezone *zone; location = calendar_config_get_timezone (); zone = icaltimezone_get_builtin_timezone (location); for (elem = all_tasks; elem; elem = elem->next) { tasks = E_TASKS (elem->data); priv = tasks->priv; calendar_config_configure_e_calendar_table (E_CALENDAR_TABLE (priv->tasks_view)); if (zone) cal_client_set_default_timezone (priv->client, zone); } }