/*
* 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 <http://www.gnu.org/licenses/>
*
*
* Authors:
* David Trowbridge <trowbrds@cs.colorado.edu>
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <string.h>
#include <gtk/gtk.h>
#include <glib/gi18n.h>
#include <gio/gio.h>
#include <calendar/gui/e-cal-config.h>
#include <shell/es-event.h>
#include <e-util/e-util.h>
#include <e-util/e-util-private.h>
#include <e-util/e-dialog-utils.h>
#include <shell/e-shell.h>
#include <shell/e-shell-view.h>
#include "url-editor-dialog.h"
#include "publish-format-fb.h"
#include "publish-format-ical.h"
#ifdef HAVE_LIBNOTIFY
#include <libnotify/notify.h>
#endif
static GtkListStore *store = NULL;
static GHashTable *uri_timeouts = NULL;
static GSList *publish_uris = NULL;
static GSList *queued_publishes = NULL;
static gint online = 0;
static GSList *error_queue = NULL;
static GStaticMutex error_queue_lock = G_STATIC_MUTEX_INIT;
static guint error_queue_show_idle_id = 0;
static void error_queue_add (gchar *descriptions, GError *error);
gint e_plugin_lib_enable (EPlugin *ep, gint enable);
GtkWidget *publish_calendar_locations (EPlugin *epl, EConfigHookItemFactoryData *data);
static void update_timestamp (EPublishUri *uri);
static void publish (EPublishUri *uri, gboolean can_report_success);
static GtkStatusIcon *status_icon = NULL;
static guint status_icon_timeout_id = 0;
#ifdef HAVE_LIBNOTIFY
static NotifyNotification *notify = NULL;
static gboolean
show_notify_cb (gpointer data)
{
return notify && !notify_notification_show (notify, NULL);
}
#endif
static gboolean
remove_notification (gpointer data)
{
if (status_icon_timeout_id)
g_source_remove (status_icon_timeout_id);
status_icon_timeout_id = 0;
#ifdef HAVE_LIBNOTIFY
if (notify)
notify_notification_close (notify, NULL);
notify = NULL;
#endif
gtk_status_icon_set_visible (status_icon, FALSE);
g_object_unref (status_icon);
status_icon = NULL;
return FALSE;
}
static void
update_publish_notification (GtkMessageType msg_type,
const gchar *msg_text)
{
static GString *actual_msg = NULL;
#ifdef HAVE_LIBNOTIFY
static gboolean can_notify = TRUE;
#endif
gboolean new_icon = !status_icon;
const gchar *stock_name;
g_return_if_fail (msg_text != NULL);
if (new_icon) {
status_icon = gtk_status_icon_new ();
if (actual_msg) {
g_string_free (actual_msg, TRUE);
actual_msg = NULL;
}
} else if (status_icon_timeout_id) {
g_source_remove (status_icon_timeout_id);
}
switch (msg_type) {
case GTK_MESSAGE_WARNING:
stock_name = GTK_STOCK_DIALOG_WARNING;
break;
case GTK_MESSAGE_ERROR:
stock_name = GTK_STOCK_DIALOG_ERROR;
break;
default:
stock_name = GTK_STOCK_DIALOG_INFO;
break;
}
if (!actual_msg) {
actual_msg = g_string_new (msg_text);
} else {
g_string_append (actual_msg, "\n");
g_string_append (actual_msg, msg_text);
}
gtk_status_icon_set_from_stock (status_icon, stock_name);
gtk_status_icon_set_tooltip_text (status_icon, actual_msg->str);
#ifdef HAVE_LIBNOTIFY
if (can_notify) {
if (notify) {
notify_notification_update (notify, _("Calendar Publishing"), actual_msg->str, stock_name);
} else {
if (!notify_init ("evolution-publish-calendar")) {
can_notify = FALSE;
return;
}
notify = notify_notification_new (_("Calendar Publishing"), actual_msg->str, stock_name);
notify_notification_set_urgency (notify, NOTIFY_URGENCY_NORMAL);
notify_notification_set_timeout (notify, NOTIFY_EXPIRES_DEFAULT);
g_timeout_add (500, show_notify_cb, NULL);
g_signal_connect (
notify, "closed",
G_CALLBACK (remove_notification), NULL);
}
}
#endif
status_icon_timeout_id = g_timeout_add_seconds (15, remove_notification, NULL);
if (new_icon) {
g_signal_connect (
status_icon, "activate",
G_CALLBACK (remove_notification), NULL);
}
}
static void
publish_no_succ_info (EPublishUri *uri)
{
publish (uri, FALSE);
}
static void
publish_uri_async (EPublishUri *uri)
{
GThread *thread = NULL;
GError *error = NULL;
thread = g_thread_create ((GThreadFunc) publish_no_succ_info, uri, FALSE, &error);
if (!thread) {
g_warning (G_STRLOC ": %s", error->message);
g_error_free (error);
}
}
static void
publish_online (EPublishUri *uri,
GFile *file,
GError **perror,
gboolean can_report_success)
{
GOutputStream *stream;
GError *error = NULL;
stream = G_OUTPUT_STREAM (g_file_replace (file, NULL, FALSE, G_FILE_CREATE_NONE, NULL, &error));
if (!stream || error) {
if (stream)
g_object_unref (stream);
if (perror) {
*perror = error;
} else if (error) {
error_queue_add (g_strdup_printf (_("Could not open %s:"), uri->location), error);
} else {
error_queue_add (g_strdup_printf (_("Could not open %s: Unknown error"), uri->location), NULL);
}
return;
}
switch (uri->publish_format) {
case URI_PUBLISH_AS_ICAL:
publish_calendar_as_ical (stream, uri, &error);
break;
case URI_PUBLISH_AS_FB:
publish_calendar_as_fb (stream, uri, &error);
break;
/*
case URI_PUBLISH_AS_HTML:
publish_calendar_as_html (handle, uri);
break;
*/
}
if (error)
error_queue_add (g_strdup_printf (_("There was an error while publishing to %s:"), uri->location), error);
else if (can_report_success)
error_queue_add (g_strdup_printf (_("Publishing to %s finished successfully"), uri->location), NULL);
update_timestamp (uri);
g_output_stream_close (stream, NULL, NULL);
g_object_unref (stream);
}
static void
unmount_done_cb (GObject *source_object,
GAsyncResult *result,
gpointer user_data)
{
GError *error = NULL;
g_mount_unmount_with_operation_finish (G_MOUNT (source_object), result, &error);
if (error) {
g_warning ("Unmount failed: %s", error->message);
g_error_free (error);
}
g_object_unref (source_object);
}
struct mnt_struct {
EPublishUri *uri;
GFile *file;
GMountOperation *mount_op;
gboolean can_report_success;
};
static void
mount_ready_cb (GObject *source_object,
GAsyncResult *result,
gpointer user_data)
{
struct mnt_struct *ms = (struct mnt_struct *) user_data;
GError *error = NULL;
GMount *mount;
g_file_mount_enclosing_volume_finish (G_FILE (source_object), result, &error);
if (error) {
error_queue_add (g_strdup_printf (_("Mount of %s failed:"), ms->uri->location), error);
if (ms)
g_object_unref (ms->mount_op);
g_free (ms);
g_object_unref (source_object);
return;
}
g_return_if_fail (ms != NULL);
publish_online (ms->uri, ms->file, NULL, ms->can_report_success);
g_object_unref (ms->mount_op);
g_free (ms);
mount = g_file_find_enclosing_mount (G_FILE (source_object), NULL, NULL);
if (mount)
g_mount_unmount_with_operation (mount, G_MOUNT_UNMOUNT_NONE, NULL, NULL, unmount_done_cb, NULL);
g_object_unref (source_object);
}
static void
ask_password (GMountOperation *op,
const gchar *message,
const gchar *default_user,
const gchar *default_domain,
GAskPasswordFlags flags,
gpointer user_data)
{
struct mnt_struct *ms = (struct mnt_struct *) user_data;
const gchar *username;
gchar *password;
gboolean req_pass = FALSE;
SoupURI *soup_uri;
g_return_if_fail (ms != NULL);
/* we can ask only for a password */
if ((flags & G_ASK_PASSWORD_NEED_PASSWORD) == 0)
return;
soup_uri = soup_uri_new (ms->uri->location);
g_return_if_fail (soup_uri != NULL);
username = soup_uri_get_user (soup_uri);
password = e_passwords_get_password (NULL, ms->uri->location);
req_pass = ((username && *username) && !(ms->uri->service_type == TYPE_ANON_FTP &&
!strcmp (username, "anonymous"))) ? TRUE:FALSE;
if (!password && req_pass) {
gboolean remember = FALSE;
password = e_passwords_ask_password (_("Enter password"), NULL, ms->uri->location, message,
E_PASSWORDS_REMEMBER_FOREVER | E_PASSWORDS_SECRET | E_PASSWORDS_ONLINE,
&remember,
NULL);
if (!password) {
/* user canceled password dialog */
g_mount_operation_reply (op, G_MOUNT_OPERATION_ABORTED);
soup_uri_free (soup_uri);
return;
}
}
if (!req_pass)
g_mount_operation_set_anonymous (op, TRUE);
else {
g_mount_operation_set_anonymous (op, FALSE);
g_mount_operation_set_username (op, username);
g_mount_operation_set_password (op, password);
}
g_mount_operation_reply (op, G_MOUNT_OPERATION_HANDLED);
soup_uri_free (soup_uri);
}
static void
ask_question (GMountOperation *op,
const gchar *message,
const gchar *choices[])
{
/* this has been stolen from file-chooser */
GtkWidget *dialog;
gint cnt, len;
gchar *primary;
const gchar *secondary = NULL;
gint res;
primary = strstr (message, "\n");
if (primary) {
secondary = primary + 1;
primary = g_strndup (message, strlen (message) - strlen (primary));
}
gdk_threads_enter ();
dialog = gtk_message_dialog_new (NULL,
0, GTK_MESSAGE_QUESTION,
GTK_BUTTONS_NONE, "%s", primary);
g_free (primary);
if (secondary) {
gtk_message_dialog_format_secondary_text (GTK_MESSAGE_DIALOG (dialog),
"%s", secondary);
}
if (choices) {
/* First count the items in the list then
* add the buttons in reverse order */
for (len = 0; choices[len] != NULL; len++) {
;
}
for (cnt = len - 1; cnt >= 0; cnt--) {
gtk_dialog_add_button (GTK_DIALOG (dialog), choices[cnt], cnt);
}
}
res = gtk_dialog_run (GTK_DIALOG (dialog));
if (res >= 0) {
g_mount_operation_set_choice (op, res);
g_mount_operation_reply (op, G_MOUNT_OPERATION_HANDLED);
} else {
g_mount_operation_reply (op, G_MOUNT_OPERATION_ABORTED);
}
gtk_widget_destroy (GTK_WIDGET (dialog));
gdk_threads_leave ();
}
static void
mount_first (EPublishUri *uri,
GFile *file,
gboolean can_report_success)
{
struct mnt_struct *ms = g_malloc (sizeof (struct mnt_struct));
ms->uri = uri;
ms->file = g_object_ref (file);
ms->mount_op = g_mount_operation_new ();
ms->can_report_success = can_report_success;
g_signal_connect (
ms->mount_op, "ask-password",
G_CALLBACK (ask_password), ms);
g_signal_connect (
ms->mount_op, "ask-question",
G_CALLBACK (ask_question), ms);
g_file_mount_enclosing_volume (file, G_MOUNT_MOUNT_NONE, ms->mount_op, NULL, mount_ready_cb, ms);
}
static void
publish (EPublishUri *uri,
gboolean can_report_success)
{
if (online) {
GError *error = NULL;
GFile *file;
if (g_slist_find (queued_publishes, uri))
queued_publishes = g_slist_remove (queued_publishes, uri);
if (!uri->enabled)
return;
file = g_file_new_for_uri (uri->location);
g_return_if_fail (file != NULL);
publish_online (uri, file, &error, can_report_success);
if (error && error->domain == G_IO_ERROR && error->code == G_IO_ERROR_NOT_MOUNTED) {
g_error_free (error);
error = NULL;
mount_first (uri, file, can_report_success);
}
if (error)
error_queue_add (g_strdup_printf (_("Could not open %s:"), uri->location), error);
g_object_unref (file);
} else {
if (g_slist_find (queued_publishes, uri) == NULL)
queued_publishes = g_slist_prepend (queued_publishes, uri);
}
}
typedef struct {
GSettings *settings;
GtkWidget *treeview;
GtkWidget *url_add;
GtkWidget *url_edit;
GtkWidget *url_remove;
GtkWidget *url_enable;
} PublishUIData;
static void
add_timeout (EPublishUri *uri)
{
guint id;
/* Set the timeout for now+frequency */
switch (uri->publish_frequency) {
case URI_PUBLISH_DAILY:
id = g_timeout_add_seconds (24 * 60 * 60, (GSourceFunc) publish, uri);
g_hash_table_insert (uri_timeouts, uri, GUINT_TO_POINTER (id));
break;
case URI_PUBLISH_WEEKLY:
id = g_timeout_add_seconds (7 * 24 * 60 * 60, (GSourceFunc) publish, uri);
g_hash_table_insert (uri_timeouts, uri, GUINT_TO_POINTER (id));
break;
}
}
static void
update_timestamp (EPublishUri *uri)
{
GSettings *settings;
gchar **set_uris;
GPtrArray *uris_array;
gboolean found = FALSE;
gchar *xml;
gint ii;
guint id;
/* Remove timeout if we have one */
id = GPOINTER_TO_UINT (g_hash_table_lookup (uri_timeouts, uri));
if (id) {
g_source_remove (id);
add_timeout (uri);
}
/* Update timestamp in settings */
xml = e_publish_uri_to_xml (uri);
if (uri->last_pub_time)
g_free (uri->last_pub_time);
uri->last_pub_time = g_strdup_printf ("%d", (gint) time (NULL));
uris_array = g_ptr_array_new_full (3, g_free);
settings = g_settings_new (PC_SETTINGS_ID);
set_uris = g_settings_get_strv (settings, PC_SETTINGS_URIS);
for (ii = 0; set_uris && set_uris[ii]; ii++) {
const gchar *d = set_uris[ii];
if (!found && g_str_equal (d, xml)) {
found = TRUE;
g_ptr_array_add (uris_array, e_publish_uri_to_xml (uri));
} else {
g_ptr_array_add (uris_array, g_strdup (d));
}
}
g_strfreev (set_uris);
g_free (xml);
/* this should not happen, right? */
if (!found)
g_ptr_array_add (uris_array, e_publish_uri_to_xml (uri));
g_ptr_array_add (uris_array, NULL);
g_settings_set_strv (settings, PC_SETTINGS_URIS, (const gchar * const *) uris_array->pdata);
g_object_unref (settings);
g_ptr_array_free (uris_array, TRUE);
}
static void
add_offset_timeout (EPublishUri *uri)
{
guint id;
time_t offset = atoi (uri->last_pub_time);
time_t current = time (NULL);
gint elapsed = current - offset;
switch (uri->publish_frequency) {
case URI_PUBLISH_DAILY:
if (elapsed > 24 * 60 * 60) {
publish (uri, FALSE);
add_timeout (uri);
} else {
id = g_timeout_add_seconds (24 * 60 * 60 - elapsed, (GSourceFunc) publish, uri);
g_hash_table_insert (uri_timeouts, uri, GUINT_TO_POINTER (id));
break;
}
break;
case URI_PUBLISH_WEEKLY:
if (elapsed > 7 * 24 * 60 * 60) {
publish (uri, FALSE);
add_timeout (uri);
} else {
id = g_timeout_add_seconds (7 * 24 * 60 * 60 - elapsed, (GSourceFunc) publish, uri);
g_hash_table_insert (uri_timeouts, uri, GUINT_TO_POINTER (id));
break;
}
break;
}
}
static void
url_list_changed (PublishUIData *ui)
{
GtkTreeModel *model = NULL;
GPtrArray *uris;
GtkTreeIter iter;
gboolean valid;
GSettings *settings;
uris = g_ptr_array_new_full (3, g_free);
model = gtk_tree_view_get_model (GTK_TREE_VIEW (ui->treeview));
valid = gtk_tree_model_get_iter_first (model, &iter);
while (valid) {
EPublishUri *url;
gchar *xml;
gtk_tree_model_get (model, &iter, URL_LIST_URL_COLUMN, &url, -1);
if ((xml = e_publish_uri_to_xml (url)) != NULL)
g_ptr_array_add (uris, xml);
valid = gtk_tree_model_iter_next (model, &iter);
}
g_ptr_array_add (uris, NULL);
settings = g_settings_new (PC_SETTINGS_ID);
g_settings_set_strv (settings, PC_SETTINGS_URIS, (const gchar * const *) uris->pdata);
g_object_unref (settings);
g_ptr_array_free (uris, TRUE);
}
static void
update_url_enable_button (EPublishUri *url,
GtkWidget *url_enable)
{
g_return_if_fail (url_enable != NULL);
g_return_if_fail (GTK_IS_BUTTON (url_enable));
gtk_button_set_label (GTK_BUTTON (url_enable), url && url->enabled ? _("_Disable") : _("E_nable"));
}
static void
url_list_enable_toggled (GtkCellRendererToggle *renderer,
const gchar *path_string,
PublishUIData *ui)
{
EPublishUri *url = NULL;
GtkTreeModel *model;
GtkTreePath *path;
GtkTreeIter iter;
path = gtk_tree_path_new_from_string (path_string);
model = gtk_tree_view_get_model (GTK_TREE_VIEW (ui->treeview));
if (gtk_tree_model_get_iter (model, &iter, path)) {
gtk_tree_model_get (model, &iter, URL_LIST_URL_COLUMN, &url, -1);
url->enabled = !url->enabled;
update_url_enable_button (url, ui->url_enable);
gtk_list_store_set (GTK_LIST_STORE (model), &iter, URL_LIST_ENABLED_COLUMN, url->enabled, -1);
url_list_changed (ui);
}
gtk_tree_path_free (path);
}
static void
selection_changed (GtkTreeSelection *selection,
PublishUIData *ui)
{
GtkTreeModel *model;
GtkTreeIter iter;
EPublishUri *url = NULL;
if (gtk_tree_selection_get_selected (selection, &model, &iter)) {
gtk_tree_model_get (model, &iter, URL_LIST_URL_COLUMN, &url, -1);
gtk_widget_set_sensitive (ui->url_edit, TRUE);
gtk_widget_set_sensitive (ui->url_remove, TRUE);
gtk_widget_set_sensitive (ui->url_enable, TRUE);
} else {
gtk_widget_set_sensitive (ui->url_edit, FALSE);
gtk_widget_set_sensitive (ui->url_remove, FALSE);
gtk_widget_set_sensitive (ui->url_enable, FALSE);
}
update_url_enable_button (url, ui->url_enable);
}
static void
url_add_clicked (GtkButton *button,
PublishUIData *ui)
{
GtkTreeModel *model;
GtkTreeIter iter;
GtkWidget *url_editor;
EPublishUri *uri;
model = gtk_tree_view_get_model (GTK_TREE_VIEW (ui->treeview));
url_editor = url_editor_dialog_new (model, NULL);
if (url_editor_dialog_run ((UrlEditorDialog *) url_editor)) {
uri = URL_EDITOR_DIALOG (url_editor)->uri;
if (uri->location) {
gtk_list_store_append (GTK_LIST_STORE (model), &iter);
gtk_list_store_set (
GTK_LIST_STORE (model), &iter,
URL_LIST_ENABLED_COLUMN, uri->enabled,
URL_LIST_LOCATION_COLUMN, uri->location,
URL_LIST_URL_COLUMN, uri, -1);
url_list_changed (ui);
publish_uris = g_slist_prepend (publish_uris, uri);
add_timeout (uri);
publish_uri_async (uri);
} else {
g_free (uri);
}
}
gtk_widget_destroy (url_editor);
}
static void
url_edit_clicked (GtkButton *button,
PublishUIData *ui)
{
GtkTreeSelection *selection;
GtkTreeModel *model;
GtkTreeIter iter;
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (ui->treeview));
if (gtk_tree_selection_get_selected (selection, &model, &iter)) {
EPublishUri *uri;
GtkWidget *url_editor;
guint id;
gtk_tree_model_get (GTK_TREE_MODEL (model), &iter, 2, &uri, -1);
url_editor = url_editor_dialog_new (model, uri);
if (url_editor_dialog_run ((UrlEditorDialog *) url_editor)) {
gtk_list_store_set (GTK_LIST_STORE (model), &iter,
URL_LIST_ENABLED_COLUMN, uri->enabled,
URL_LIST_LOCATION_COLUMN, uri->location,
URL_LIST_URL_COLUMN, uri, -1);
id = GPOINTER_TO_UINT (g_hash_table_lookup (uri_timeouts, uri));
if (id)
g_source_remove (id);
add_timeout (uri);
url_list_changed (ui);
publish_uri_async (uri);
}
gtk_widget_destroy (url_editor);
}
}
static void
url_list_double_click (GtkTreeView *treeview,
GtkTreePath *path,
GtkTreeViewColumn *column,
PublishUIData *ui)
{
url_edit_clicked (NULL, ui);
}
static void
url_remove_clicked (GtkButton *button,
PublishUIData *ui)
{
EPublishUri *url = NULL;
GtkTreeSelection *selection;
GtkTreeModel *model;
GtkTreeIter iter;
GtkWidget *confirm;
gint response;
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (ui->treeview));
if (!gtk_tree_selection_get_selected (selection, &model, &iter))
return;
gtk_tree_model_get (model, &iter, URL_LIST_URL_COLUMN, &url, -1);
confirm = gtk_message_dialog_new (NULL, GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
GTK_MESSAGE_QUESTION, GTK_BUTTONS_NONE,
_("Are you sure you want to remove this location?"));
gtk_dialog_add_button (GTK_DIALOG (confirm), GTK_STOCK_CANCEL, GTK_RESPONSE_NO);
gtk_dialog_add_button (GTK_DIALOG (confirm), GTK_STOCK_REMOVE, GTK_RESPONSE_YES);
gtk_dialog_set_default_response (GTK_DIALOG (confirm), GTK_RESPONSE_CANCEL);
response = gtk_dialog_run (GTK_DIALOG (confirm));
gtk_widget_destroy (confirm);
if (response == GTK_RESPONSE_YES) {
gint len;
guint id;
gtk_list_store_remove (GTK_LIST_STORE (model), &iter);
len = gtk_tree_model_iter_n_children (model, NULL);
if (len > 0) {
gtk_tree_selection_select_iter (selection, &iter);
} else {
gtk_widget_set_sensitive (ui->url_edit, FALSE);
gtk_widget_set_sensitive (ui->url_remove, FALSE);
gtk_widget_set_sensitive (ui->url_enable, FALSE);
update_url_enable_button (NULL, ui->url_enable);
}
publish_uris = g_slist_remove (publish_uris, url);
id = GPOINTER_TO_UINT (g_hash_table_lookup (uri_timeouts, url));
if (id)
g_source_remove (id);
g_free (url);
url_list_changed (ui);
}
}
static void
url_enable_clicked (GtkButton *button,
PublishUIData *ui)
{
EPublishUri *url = NULL;
GtkTreeSelection *selection;
GtkTreeModel *model;
GtkTreeIter iter;
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (ui->treeview));
if (gtk_tree_selection_get_selected (selection, &model, &iter)) {
gtk_tree_model_get (model, &iter, URL_LIST_URL_COLUMN, &url, -1);
url->enabled = !url->enabled;
update_url_enable_button (url, ui->url_enable);
gtk_list_store_set (GTK_LIST_STORE (model), &iter, URL_LIST_ENABLED_COLUMN, url->enabled, -1);
gtk_tree_selection_select_iter (selection, &iter);
url_list_changed (ui);
}
}
static void
online_state_changed (EShell *shell)
{
online = e_shell_get_online (shell);
if (online)
while (queued_publishes)
publish (queued_publishes->data, FALSE);
}
GtkWidget *
publish_calendar_locations (EPlugin *epl,
EConfigHookItemFactoryData *data)
{
GtkBuilder *builder;
GtkCellRenderer *renderer;
GtkTreeSelection *selection;
GtkWidget *toplevel;
PublishUIData *ui = g_new0 (PublishUIData, 1);
GSList *l;
GtkTreeIter iter;
builder = gtk_builder_new ();
e_load_ui_builder_definition (builder, "publish-calendar.ui");
ui->treeview = e_builder_get_widget (builder, "url list");
if (store == NULL)
store = gtk_list_store_new (URL_LIST_N_COLUMNS, G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_POINTER);
else
gtk_list_store_clear (store);
gtk_tree_view_set_model (GTK_TREE_VIEW (ui->treeview), GTK_TREE_MODEL (store));
renderer = gtk_cell_renderer_toggle_new ();
g_object_set (renderer, "activatable", TRUE, NULL);
gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (ui->treeview), -1, _("Enabled"),
renderer, "active", URL_LIST_ENABLED_COLUMN, NULL);
g_signal_connect (
renderer, "toggled",
G_CALLBACK (url_list_enable_toggled), ui);
renderer = gtk_cell_renderer_text_new ();
gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (ui->treeview), -1, _("Location"),
renderer, "text", URL_LIST_LOCATION_COLUMN, NULL);
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (ui->treeview));
gtk_tree_selection_set_mode (selection, GTK_SELECTION_SINGLE);
g_signal_connect (
selection, "changed",
G_CALLBACK (selection_changed), ui);
gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (ui->treeview), TRUE);
g_signal_connect (
ui->treeview, "row-activated",
G_CALLBACK (url_list_double_click), ui);
ui->url_add = e_builder_get_widget (builder, "url add");
ui->url_edit = e_builder_get_widget (builder, "url edit");
ui->url_remove = e_builder_get_widget (builder, "url remove");
ui->url_enable = e_builder_get_widget (builder, "url enable");
update_url_enable_button (NULL, ui->url_enable);
g_signal_connect (
ui->url_add, "clicked",
G_CALLBACK (url_add_clicked), ui);
g_signal_connect (
ui->url_edit, "clicked",
G_CALLBACK (url_edit_clicked), ui);
g_signal_connect (
ui->url_remove, "clicked",
G_CALLBACK (url_remove_clicked), ui);
g_signal_connect (
ui->url_enable, "clicked",
G_CALLBACK (url_enable_clicked), ui);
gtk_widget_set_sensitive (GTK_WIDGET (ui->url_edit), FALSE);
gtk_widget_set_sensitive (GTK_WIDGET (ui->url_remove), FALSE);
gtk_widget_set_sensitive (GTK_WIDGET (ui->url_enable), FALSE);
gtk_button_set_image (GTK_BUTTON (ui->url_enable), gtk_image_new_from_stock (GTK_STOCK_APPLY, GTK_ICON_SIZE_BUTTON));
gtk_button_set_use_underline (GTK_BUTTON (ui->url_enable), TRUE);
l = publish_uris;
while (l) {
EPublishUri *url = (EPublishUri *) l->data;
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter,
URL_LIST_ENABLED_COLUMN, url->enabled,
URL_LIST_LOCATION_COLUMN, url->location,
URL_LIST_URL_COLUMN, url, -1);
l = g_slist_next (l);
}
if (gtk_tree_model_get_iter_first (GTK_TREE_MODEL (store), &iter))
gtk_tree_selection_select_iter (selection, &iter);
toplevel = e_builder_get_widget (builder, "toplevel");
gtk_widget_show_all (toplevel);
gtk_box_pack_start (GTK_BOX (data->parent), toplevel, FALSE, TRUE, 0);
g_object_unref (builder);
return toplevel;
}
static gpointer
publish_urls (gpointer data)
{
GSList *l;
for (l = publish_uris; l; l = g_slist_next (l)) {
EPublishUri *uri = l->data;
publish (uri, TRUE);
}
return GINT_TO_POINTER (0);
}
static gpointer
publish_uris_set_timeout (gchar **uris)
{
gint ii;
uri_timeouts = g_hash_table_new (g_direct_hash, g_direct_equal);
for (ii = 0; uris && uris[ii]; ii++) {
const gchar *xml = uris[ii];
EPublishUri *uri = e_publish_uri_from_xml (xml);
if (!uri->location) {
g_free (uri);
continue;
}
publish_uris = g_slist_prepend (publish_uris, uri);
/* Add a timeout based on the last publish time */
add_offset_timeout (uri);
}
g_strfreev (uris);
return NULL;
}
gint
e_plugin_lib_enable (EPlugin *ep,
gint enable)
{
EShell *shell = e_shell_get_default ();
if (shell) {
g_signal_handlers_disconnect_by_func (shell, G_CALLBACK (online_state_changed), NULL);
if (enable) {
online = e_shell_get_online (shell);
g_signal_connect (
shell, "notify::online",
G_CALLBACK (online_state_changed), NULL);
}
}
if (enable) {
GSettings *settings;
gchar **uris;
GThread *thread = NULL;
GError *error = NULL;
settings = g_settings_new (PC_SETTINGS_ID);
uris = g_settings_get_strv (settings, PC_SETTINGS_URIS);
g_object_unref (settings);
thread = g_thread_create ((GThreadFunc) publish_uris_set_timeout, uris, FALSE, &error);
if (!thread) {
g_warning ("Could create thread to set timeout for publishing uris : %s", error->message);
g_error_free (error);
}
}
return 0;
}
struct eq_data {
gchar *description;
GError *error;
};
static gboolean
error_queue_show_idle (gpointer user_data)
{
GString *info = NULL;
GSList *l;
gboolean has_error = FALSE, has_info = FALSE;
g_static_mutex_lock (&error_queue_lock);
for (l = error_queue; l; l = l->next) {
struct eq_data *data = l->data;
if (data) {
if (data->description) {
if (!info) {
info = g_string_new (data->description);
} else {
g_string_append (info, "\n\n");
g_string_append (info, data->description);
}
g_free (data->description);
}
if (data->error) {
has_error = TRUE;
if (!info) {
info = g_string_new (data->error->message);
} else if (data->description) {
g_string_append (info, " ");
g_string_append (info, data->error->message);
} else {
g_string_append (info, "\n\n");
g_string_append (info, data->error->message);
}
g_error_free (data->error);
} else if (data->description) {
has_info = TRUE;
}
g_free (data);
}
}
g_slist_free (error_queue);
error_queue = NULL;
error_queue_show_idle_id = 0;
g_static_mutex_unlock (&error_queue_lock);
if (info) {
update_publish_notification (has_error && has_info ? GTK_MESSAGE_WARNING : has_error ? GTK_MESSAGE_ERROR : GTK_MESSAGE_INFO, info->str);
g_string_free (info, TRUE);
}
return FALSE;
}
void
error_queue_add (gchar *description,
GError *error)
{
struct eq_data *data;
if (!error && !description)
return;
data = g_new0 (struct eq_data, 1);
data->description = description;
data->error = error;
g_static_mutex_lock (&error_queue_lock);
error_queue = g_slist_append (error_queue, data);
if (error_queue_show_idle_id == 0)
error_queue_show_idle_id = g_idle_add (error_queue_show_idle, NULL);
g_static_mutex_unlock (&error_queue_lock);
}
static void
action_calendar_publish_cb (GtkAction *action,
EShellView *shell_view)
{
GThread *thread = NULL;
GError *error = NULL;
thread = g_thread_create ((GThreadFunc) publish_urls, NULL, FALSE, &error);
if (!thread) {
/* To Translators: This is shown to a user when creation of a new thread,
* where the publishing should be done, fails. Basically, this shouldn't
* ever happen, and if so, then something is really wrong. */
error_queue_add (g_strdup (_("Could not create publish thread.")), error);
}
}
static GtkActionEntry entries[] = {
{ "calendar-publish",
NULL,
N_("_Publish Calendar Information"),
NULL,
NULL, /* XXX Add a tooltip! */
G_CALLBACK (action_calendar_publish_cb) }
};
gboolean e_plugin_ui_init (GtkUIManager *ui_manager, EShellView *shell_view);
gboolean
e_plugin_ui_init (GtkUIManager *ui_manager,
EShellView *shell_view)
{
EShellWindow *shell_window;
GtkActionGroup *action_group;
shell_window = e_shell_view_get_shell_window (shell_view);
action_group = e_shell_window_get_action_group (shell_window, "calendar");
gtk_action_group_add_actions (
action_group, entries,
G_N_ELEMENTS (entries), shell_view);
return TRUE;
}