/*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) version 3.
*
* 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with the program; if not, see
*
*
* Authors:
* Federico Mena Quintero
* Damon Chaplin
* Rodrigo Moya
* Nathan Owens
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
*/
#ifdef HAVE_CONFIG_H
#include
#endif
#include
#include
#include
#include
#include
#include
#include
#include "e-util/e-error.h"
#include "e-util/e-categories-config.h"
#include "e-util/e-util-private.h"
#include
#include
#include
#include "widgets/menus/gal-view-menus.h"
#include "dialogs/delete-error.h"
#include "calendar-config.h"
#include "cal-search-bar.h"
#include "comp-util.h"
#include "e-memo-table-config.h"
#include "misc.h"
#include "e-cal-component-memo-preview.h"
#include "e-memos.h"
#include "common/authentication.h"
/* Private part of the GnomeCalendar structure */
struct _EMemosPrivate {
/* The memo lists for display */
GHashTable *clients;
GList *clients_list;
ECal *default_client;
ECalView *query;
/* The EMemoTable showing the memos. */
GtkWidget *memos_view;
EMemoTableConfig *memos_view_config;
/* Calendar search bar for memos */
GtkWidget *search_bar;
/* The preview */
GtkWidget *preview;
gchar *current_uid;
char *sexp;
/* View instance and the view menus handler */
GalViewInstance *view_instance;
GalViewMenus *view_menus;
GList *notifications;
};
static void setup_widgets (EMemos *memos);
static void e_memos_destroy (GtkObject *object);
static void update_view (EMemos *memos);
static void categories_changed_cb (gpointer object, gpointer user_data);
/* Signal IDs */
enum {
SELECTION_CHANGED,
SOURCE_ADDED,
SOURCE_REMOVED,
LAST_SIGNAL
};
static guint e_memos_signals[LAST_SIGNAL] = { 0 };
G_DEFINE_TYPE (EMemos, e_memos, GTK_TYPE_TABLE)
/* 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)
{
EMemos *memos;
EMemosPrivate *priv;
memos = E_MEMOS (data);
priv = memos->priv;
if (priv->sexp)
g_free (priv->sexp);
priv->sexp = g_strdup (sexp);
update_view (memos);
}
/* 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)
{
EMemos *memos;
EMemosPrivate *priv;
ECalModel *model;
memos = E_MEMOS (data);
priv = memos->priv;
model = e_memo_table_get_model (E_MEMO_TABLE (priv->memos_view));
e_cal_model_set_default_category (model, category);
}
static void
set_timezone (EMemos *memos)
{
EMemosPrivate *priv;
icaltimezone *zone;
GList *l;
priv = memos->priv;
zone = calendar_config_get_icaltimezone ();
for (l = priv->clients_list; l != NULL; l = l->next) {
ECal *client = l->data;
if (e_cal_get_load_state (client) == E_CAL_LOAD_LOADED)
/* FIXME Error checking */
e_cal_set_default_timezone (client, zone, NULL);
}
if (priv->default_client && e_cal_get_load_state (priv->default_client) == E_CAL_LOAD_LOADED)
/* FIXME Error checking */
e_cal_set_default_timezone (priv->default_client, zone, NULL);
if (priv->preview)
e_cal_component_memo_preview_set_default_timezone (E_CAL_COMPONENT_MEMO_PREVIEW (priv->preview), zone);
}
static void
timezone_changed_cb (GConfClient *client, guint id, GConfEntry *entry, gpointer data)
{
EMemos *memos = data;
set_timezone (memos);
}
static void
update_view (EMemos *memos)
{
EMemosPrivate *priv;
ECalModel *model;
priv = memos->priv;
model = e_memo_table_get_model (E_MEMO_TABLE (priv->memos_view));
e_cal_model_set_search_query (model, priv->sexp);
e_cal_component_memo_preview_clear (E_CAL_COMPONENT_MEMO_PREVIEW (priv->preview));
}
static void
setup_config (EMemos *memos)
{
EMemosPrivate *priv;
guint not;
priv = memos->priv;
/* Timezone */
set_timezone (memos);
not = calendar_config_add_notification_timezone (timezone_changed_cb, memos);
priv->notifications = g_list_prepend (priv->notifications, GUINT_TO_POINTER (not));
}
struct AffectedComponents {
EMemoTable *memo_table;
GSList *components; /* contains pointers to ECalModelComponent */
};
/**
* get_selected_components_cb
* Helper function to fill list of selected components in EMemoTable.
* This function is called from e_table_selected_row_foreach.
**/
static void
get_selected_components_cb (int model_row, gpointer data)
{
struct AffectedComponents *ac = (struct AffectedComponents *) data;
if (!ac || !ac->memo_table)
return;
ac->components = g_slist_prepend (ac->components, e_cal_model_get_component_at (E_CAL_MODEL (e_memo_table_get_model (ac->memo_table)), model_row));
}
/**
* do_for_selected_components
* Calls function func for all selected components in memo_table.
*
* @param memo_table Table with selected components of our interest.
* @param func Function to be called on each selected component from cal_table.
* The first parameter of this function is a pointer to ECalModelComponent and
* the second parameter of this function is pointer to cal_table
* @param user_data User data, will be passed to func.
**/
static void
do_for_selected_components (EMemoTable *memo_table, GFunc func, gpointer user_data)
{
ETable *etable;
struct AffectedComponents ac;
g_return_if_fail (memo_table != NULL);
ac.memo_table = memo_table;
ac.components = NULL;
etable = e_table_scrolled_get_table (E_TABLE_SCROLLED (memo_table->etable));
e_table_selected_row_foreach (etable, get_selected_components_cb, &ac);
g_slist_foreach (ac.components, func, user_data);
g_slist_free (ac.components);
}
/**
* obtain_list_of_components
* As a callback function to convert each ECalModelComponent to string
* of format "source_uid\ncomponent_str" and add this newly allocated
* string to the list of components. Strings should be freed with g_free.
*
* @param data ECalModelComponent object.
* @param user_data Pointer to GSList list, where to put new strings.
**/
static void
obtain_list_of_components (gpointer data, gpointer user_data)
{
GSList **list;
ECalModelComponent *comp_data;
list = (GSList **) user_data;
comp_data = (ECalModelComponent *) data;
if (list && comp_data) {
char *comp_str;
icalcomponent *vcal;
vcal = e_cal_util_new_top_level ();
e_cal_util_add_timezones_from_component (vcal, comp_data->icalcomp);
icalcomponent_add_component (vcal, icalcomponent_new_clone (comp_data->icalcomp));
comp_str = icalcomponent_as_ical_string (vcal);
if (comp_str) {
ESource *source = e_cal_get_source (comp_data->client);
const char *source_uid = e_source_peek_uid (source);
*list = g_slist_prepend (*list, g_strdup_printf ("%s\n%s", source_uid, comp_str));
/* do not free this pointer, it owns libical */
/* g_free (comp_str); */
}
icalcomponent_free (vcal);
g_free (comp_str);
}
}
static void
table_drag_data_get (ETable *table,
int row,
int col,
GdkDragContext *context,
GtkSelectionData *selection_data,
guint info,
guint time,
EMemos *memos)
{
EMemosPrivate *priv;
priv = memos->priv;
if (info == TARGET_VCALENDAR) {
/* we will pass an icalcalendar component for both types */
GSList *components = NULL;
do_for_selected_components (E_MEMO_TABLE (priv->memos_view), obtain_list_of_components, &components);
if (components) {
cal_comp_selection_set_string_list (selection_data, components);
g_slist_foreach (components, (GFunc)g_free, NULL);
g_slist_free (components);
}
}
}
static void
setup_widgets (EMemos *memos)
{
EMemosPrivate *priv;
ETable *etable;
ECalModel *model;
priv = memos->priv;
priv->search_bar = cal_search_bar_new (CAL_SEARCH_MEMOS_DEFAULT);
g_signal_connect (priv->search_bar, "sexp_changed",
G_CALLBACK (search_bar_sexp_changed_cb), memos);
g_signal_connect (priv->search_bar, "category_changed",
G_CALLBACK (search_bar_category_changed_cb), memos);
categories_changed_cb (NULL, memos);
gtk_table_attach (GTK_TABLE (memos), priv->search_bar, 0, 1, 0, 1,
GTK_EXPAND | GTK_FILL | GTK_SHRINK, 0, 0, 0);
gtk_widget_show (priv->search_bar);
/* create the memo list */
priv->memos_view = e_memo_table_new ();
priv->memos_view_config = e_memo_table_config_new (E_MEMO_TABLE (priv->memos_view));
g_signal_connect (etable, "table_drag_data_get",
G_CALLBACK(table_drag_data_get), memos);
}
/* Class initialization function for the gnome calendar */
static void
e_memos_class_init (EMemosClass *klass)
{
GtkObjectClass *object_class;
object_class = (GtkObjectClass *) klass;
e_memos_signals[SELECTION_CHANGED] =
g_signal_new ("selection_changed",
G_TYPE_FROM_CLASS (object_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (EMemosClass, selection_changed),
NULL, NULL,
g_cclosure_marshal_VOID__INT,
G_TYPE_NONE, 1,
G_TYPE_INT);
e_memos_signals[SOURCE_ADDED] =
g_signal_new ("source_added",
G_TYPE_FROM_CLASS (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (EMemosClass, source_added),
NULL, NULL,
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE,
1,
G_TYPE_OBJECT);
e_memos_signals[SOURCE_REMOVED] =
g_signal_new ("source_removed",
G_TYPE_FROM_CLASS (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (EMemosClass, source_removed),
NULL, NULL,
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE,
1,
G_TYPE_OBJECT);
object_class->destroy = e_memos_destroy;
klass->selection_changed = NULL;
klass->source_added = NULL;
klass->source_removed = NULL;
}
static void
categories_changed_cb (gpointer object, gpointer user_data)
{
GList *cat_list;
GPtrArray *cat_array;
EMemosPrivate *priv;
EMemos *memos = user_data;
priv = memos->priv;
cat_array = g_ptr_array_new ();
cat_list = e_categories_get_list ();
while (cat_list != NULL) {
if (e_categories_is_searchable ((const char *) cat_list->data))
g_ptr_array_add (cat_array, cat_list->data);
cat_list = g_list_remove (cat_list, cat_list->data);
}
cal_search_bar_set_categories (CAL_SEARCH_BAR(priv->search_bar), cat_array);
g_ptr_array_free (cat_array, TRUE);
}
/* Object initialization function for the gnome calendar */
static void
e_memos_init (EMemos *memos)
{
EMemosPrivate *priv;
priv = g_new0 (EMemosPrivate, 1);
memos->priv = priv;
setup_config (memos);
setup_widgets (memos);
e_categories_register_change_listener (G_CALLBACK (categories_changed_cb), memos);
priv->clients = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_object_unref);
priv->query = NULL;
priv->view_instance = NULL;
priv->view_menus = NULL;
priv->current_uid = NULL;
priv->sexp = g_strdup ("#t");
priv->default_client = NULL;
update_view (memos);
}
GtkWidget *
e_memos_new (void)
{
EMemos *memos;
memos = g_object_new (e_memos_get_type (), NULL);
return GTK_WIDGET (memos);
}
static void
e_memos_destroy (GtkObject *object)
{
EMemos *memos;
EMemosPrivate *priv;
g_return_if_fail (object != NULL);
g_return_if_fail (E_IS_MEMOS (object));
memos = E_MEMOS (object);
priv = memos->priv;
if (priv) {
GList *l;
e_categories_unregister_change_listener (G_CALLBACK (categories_changed_cb), memos);
/* disconnect from signals on all the clients */
for (l = priv->clients_list; l != NULL; l = l->next) {
g_signal_handlers_disconnect_matched (l->data, G_SIGNAL_MATCH_DATA,
0, 0, NULL, NULL, memos);
}
g_hash_table_destroy (priv->clients);
g_list_free (priv->clients_list);
if (priv->current_uid) {
g_free (priv->current_uid);
priv->current_uid = NULL;
}
if (priv->sexp) {
g_free (priv->sexp);
priv->sexp = NULL;
}
if (priv->memos_view_config) {
g_object_unref (priv->memos_view_config);
priv->memos_view_config = NULL;
}
for (l = priv->notifications; l; l = l->next)
calendar_config_remove_notification (GPOINTER_TO_UINT (l->data));
priv->notifications = NULL;
g_free (priv);
memos->priv = NULL;
}
if (GTK_OBJECT_CLASS (e_memos_parent_class)->destroy)
(* GTK_OBJECT_CLASS (e_memos_parent_class)->destroy) (object);
}
static void
set_status_message (EMemos *memos, const char *message, ...)
{
EMemosPrivate *priv;
va_list args;
char sz[2048], *msg_string = NULL;
if (message) {
va_start (args, message);
vsnprintf (sz, sizeof sz, message, args);
va_end (args);
msg_string = sz;
}
priv = memos->priv;
e_memo_table_set_status_message (E_MEMO_TABLE (priv->memos_view), msg_string);
}
/* Callback from the calendar client when the calendar is opened */
static void
client_cal_opened_cb (ECal *ecal, ECalendarStatus status, EMemos *memos)
{
ECalModel *model;
ESource *source;
EMemosPrivate *priv;
priv = memos->priv;
source = e_cal_get_source (ecal);
switch (status) {
case E_CALENDAR_STATUS_OK :
g_signal_handlers_disconnect_matched (ecal, G_SIGNAL_MATCH_FUNC, 0, 0, NULL, client_cal_opened_cb, NULL);
set_status_message (memos, _("Loading memos"));
model = e_memo_table_get_model (E_MEMO_TABLE (priv->memos_view));
e_cal_model_add_client (model, ecal);
set_timezone (memos);
set_status_message (memos, NULL);
break;
case E_CALENDAR_STATUS_BUSY :
break;
case E_CALENDAR_STATUS_REPOSITORY_OFFLINE:
e_error_run (GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (memos))), "calendar:prompt-no-contents-offline-memos", NULL);
break;
default :
/* Make sure the source doesn't disappear on us */
g_object_ref (source);
priv->clients_list = g_list_remove (priv->clients_list, ecal);
g_signal_handlers_disconnect_matched (ecal, G_SIGNAL_MATCH_DATA,
0, 0, NULL, NULL, memos);
/* Do this last because it unrefs the client */
g_hash_table_remove (priv->clients, e_source_peek_uid (source));
g_signal_emit (memos, e_memos_signals[SOURCE_REMOVED], 0, source);
set_status_message (memos, NULL);
g_object_unref (source);
break;
}
}
static void
default_client_cal_opened_cb (ECal *ecal, ECalendarStatus status, EMemos *memos)
{
ECalModel *model;
ESource *source;
EMemosPrivate *priv;
priv = memos->priv;
source = e_cal_get_source (ecal);
switch (status) {
case E_CALENDAR_STATUS_OK :
g_signal_handlers_disconnect_matched (ecal, G_SIGNAL_MATCH_FUNC, 0, 0, NULL, default_client_cal_opened_cb, NULL);
model = e_memo_table_get_model (E_MEMO_TABLE (priv->memos_view));
set_timezone (memos);
e_cal_model_set_default_client (model, ecal);
set_status_message (memos, NULL);
break;
case E_CALENDAR_STATUS_BUSY:
break;
default :
/* Make sure the source doesn't disappear on us */
g_object_ref (source);
priv->clients_list = g_list_remove (priv->clients_list, ecal);
g_signal_handlers_disconnect_matched (ecal, G_SIGNAL_MATCH_DATA,
0, 0, NULL, NULL, memos);
/* Do this last because it unrefs the client */
g_hash_table_remove (priv->clients, e_source_peek_uid (source));
g_signal_emit (memos, e_memos_signals[SOURCE_REMOVED], 0, source);
set_status_message (memos, NULL);
g_object_unref (priv->default_client);
priv->default_client = NULL;
g_object_unref (source);
break;
}
}
typedef void (*open_func) (ECal *, ECalendarStatus, EMemos *);
static gboolean
open_ecal (EMemos *memos, ECal *cal, gboolean only_if_exists, open_func of)
{
set_status_message (memos, _("Opening memos at %s"), e_cal_get_uri (cal));
g_signal_connect (G_OBJECT (cal), "cal_opened", G_CALLBACK (of), memos);
e_cal_open_async (cal, only_if_exists);
return TRUE;
}
gboolean
e_memos_add_memo_source (EMemos *memos, ESource *source)
{
EMemosPrivate *priv;
ECal *client;
const char *uid;
g_return_val_if_fail (memos != NULL, FALSE);
g_return_val_if_fail (E_IS_MEMOS (memos), FALSE);
g_return_val_if_fail (E_IS_SOURCE (source), FALSE);
priv = memos->priv;
uid = e_source_peek_uid (source);
client = g_hash_table_lookup (priv->clients, uid);
if (client) {
/* We already have it */
return TRUE;
} else {
ESource *default_source;
if (priv->default_client) {
default_source = e_cal_get_source (priv->default_client);
/* We don't have it but the default client is it */
if (!strcmp (e_source_peek_uid (default_source), uid))
client = g_object_ref (priv->default_client);
}
/* Create a new one */
if (!client) {
client = auth_new_cal_from_source (source, E_CAL_SOURCE_TYPE_JOURNAL);
if (!client)
return FALSE;
}
}
g_signal_connect (G_OBJECT (client), "backend_error", G_CALLBACK (backend_error_cb), memos);
g_signal_connect (G_OBJECT (client), "backend_died", G_CALLBACK (backend_died_cb), memos);
/* add the client to internal structure */
g_hash_table_insert (priv->clients, g_strdup (uid) , client);
priv->clients_list = g_list_prepend (priv->clients_list, client);
g_signal_emit (memos, e_memos_signals[SOURCE_ADDED], 0, source);
open_ecal (memos, client, FALSE, client_cal_opened_cb);
return TRUE;
}
gboolean
e_memos_remove_memo_source (EMemos *memos, ESource *source)
{
EMemosPrivate *priv;
ECal *client;
ECalModel *model;
const char *uid;
g_return_val_if_fail (memos != NULL, FALSE);
g_return_val_if_fail (E_IS_MEMOS (memos), FALSE);
g_return_val_if_fail (E_IS_SOURCE (source), FALSE);
priv = memos->priv;
uid = e_source_peek_uid (source);
client = g_hash_table_lookup (priv->clients, uid);
if (!client)
return TRUE;
priv->clients_list = g_list_remove (priv->clients_list, client);
g_signal_handlers_disconnect_matched (client, G_SIGNAL_MATCH_DATA,
0, 0, NULL, NULL, memos);
model = e_memo_table_get_model (E_MEMO_TABLE (priv->memos_view));
e_cal_model_remove_client (model, client);
g_hash_table_remove (priv->clients, uid);
g_signal_emit (memos, e_memos_signals[SOURCE_REMOVED], 0, source);
return TRUE;
}
gboolean
e_memos_set_default_source (EMemos *memos, ESource *source)
{
EMemosPrivate *priv;
ECal *ecal;
g_return_val_if_fail (memos != NULL, FALSE);
g_return_val_if_fail (E_IS_MEMOS (memos), FALSE);
g_return_val_if_fail (E_IS_SOURCE (source), FALSE);
priv = memos->priv;
ecal = g_hash_table_lookup (priv->clients, e_source_peek_uid (source));
if (priv->default_client)
g_object_unref (priv->default_client);
if (ecal) {
priv->default_client = g_object_ref (ecal);
} else {
priv->default_client = auth_new_cal_from_source (source, E_CAL_SOURCE_TYPE_JOURNAL);
if (!priv->default_client)
return FALSE;
}
open_ecal (memos, priv->default_client, FALSE, default_client_cal_opened_cb);
return TRUE;
}
ECal *
e_memos_get_default_client (EMemos *memos)
{
EMemosPrivate *priv;
g_return_val_if_fail (memos != NULL, NULL);
g_return_val_if_fail (E_IS_MEMOS (memos), NULL);
priv = memos->priv;
return e_cal_model_get_default_client (e_memo_table_get_model (E_MEMO_TABLE (priv->memos_view)));
}