/* * e-cal-source-config.c * * 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 * */ #include "e-cal-source-config.h" #include #include #include "e-misc-utils.h" #define E_CAL_SOURCE_CONFIG_GET_PRIVATE(obj) \ (G_TYPE_INSTANCE_GET_PRIVATE \ ((obj), E_TYPE_CAL_SOURCE_CONFIG, ECalSourceConfigPrivate)) struct _ECalSourceConfigPrivate { ECalClientSourceType source_type; GtkWidget *color_button; GtkWidget *default_button; }; enum { PROP_0, PROP_SOURCE_TYPE }; G_DEFINE_TYPE ( ECalSourceConfig, e_cal_source_config, E_TYPE_SOURCE_CONFIG) static ESource * cal_source_config_ref_default (ESourceConfig *config) { ECalSourceConfigPrivate *priv; ESourceRegistry *registry; priv = E_CAL_SOURCE_CONFIG_GET_PRIVATE (config); registry = e_source_config_get_registry (config); if (priv->source_type == E_CAL_CLIENT_SOURCE_TYPE_EVENTS) return e_source_registry_ref_default_calendar (registry); else if (priv->source_type == E_CAL_CLIENT_SOURCE_TYPE_MEMOS) return e_source_registry_ref_default_memo_list (registry); else if (priv->source_type == E_CAL_CLIENT_SOURCE_TYPE_TASKS) return e_source_registry_ref_default_task_list (registry); g_return_val_if_reached (NULL); } static void cal_source_config_set_default (ESourceConfig *config, ESource *source) { ECalSourceConfigPrivate *priv; ESourceRegistry *registry; priv = E_CAL_SOURCE_CONFIG_GET_PRIVATE (config); registry = e_source_config_get_registry (config); if (priv->source_type == E_CAL_CLIENT_SOURCE_TYPE_EVENTS) e_source_registry_set_default_calendar (registry, source); else if (priv->source_type == E_CAL_CLIENT_SOURCE_TYPE_MEMOS) e_source_registry_set_default_memo_list (registry, source); else if (priv->source_type == E_CAL_CLIENT_SOURCE_TYPE_TASKS) e_source_registry_set_default_task_list (registry, source); } static void cal_source_config_set_source_type (ECalSourceConfig *config, ECalClientSourceType source_type) { config->priv->source_type = source_type; } static void cal_source_config_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) { switch (property_id) { case PROP_SOURCE_TYPE: cal_source_config_set_source_type ( E_CAL_SOURCE_CONFIG (object), g_value_get_enum (value)); return; } G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } static void cal_source_config_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) { switch (property_id) { case PROP_SOURCE_TYPE: g_value_set_enum ( value, e_cal_source_config_get_source_type ( E_CAL_SOURCE_CONFIG (object))); return; } G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } static void cal_source_config_dispose (GObject *object) { ECalSourceConfigPrivate *priv; priv = E_CAL_SOURCE_CONFIG_GET_PRIVATE (object); if (priv->color_button != NULL) { g_object_unref (priv->color_button); priv->color_button = NULL; } if (priv->default_button != NULL) { g_object_unref (priv->default_button); priv->default_button = NULL; } /* Chain up to parent's dispose() method. */ G_OBJECT_CLASS (e_cal_source_config_parent_class)->dispose (object); } static void cal_source_config_constructed (GObject *object) { ECalSourceConfigPrivate *priv; ESource *default_source; ESource *original_source; ESourceConfig *config; GObjectClass *class; GtkWidget *widget; const gchar *label; /* Chain up to parent's constructed() method. */ class = G_OBJECT_CLASS (e_cal_source_config_parent_class); class->constructed (object); config = E_SOURCE_CONFIG (object); priv = E_CAL_SOURCE_CONFIG_GET_PRIVATE (object); widget = gtk_color_button_new (); priv->color_button = g_object_ref_sink (widget); gtk_widget_show (widget); switch (priv->source_type) { case E_CAL_CLIENT_SOURCE_TYPE_EVENTS: label = _("Mark as default calendar"); break; case E_CAL_CLIENT_SOURCE_TYPE_TASKS: label = _("Mark as default task list"); break; case E_CAL_CLIENT_SOURCE_TYPE_MEMOS: label = _("Mark as default memo list"); break; default: /* No need to translate this string. */ label = "Invalid ECalSourceType value"; g_warn_if_reached (); } widget = gtk_check_button_new_with_label (label); priv->default_button = g_object_ref_sink (widget); gtk_widget_show (widget); default_source = cal_source_config_ref_default (config); original_source = e_source_config_get_original_source (config); if (original_source != NULL) { gboolean active; active = e_source_equal (original_source, default_source); g_object_set (priv->default_button, "active", active, NULL); } g_object_unref (default_source); e_source_config_insert_widget ( config, NULL, _("Color:"), priv->color_button); e_source_config_insert_widget ( config, NULL, NULL, priv->default_button); } static const gchar * cal_source_config_get_backend_extension_name (ESourceConfig *config) { ECalSourceConfig *cal_config; const gchar *extension_name; cal_config = E_CAL_SOURCE_CONFIG (config); switch (e_cal_source_config_get_source_type (cal_config)) { case E_CAL_CLIENT_SOURCE_TYPE_EVENTS: extension_name = E_SOURCE_EXTENSION_CALENDAR; break; case E_CAL_CLIENT_SOURCE_TYPE_TASKS: extension_name = E_SOURCE_EXTENSION_TASK_LIST; break; case E_CAL_CLIENT_SOURCE_TYPE_MEMOS: extension_name = E_SOURCE_EXTENSION_MEMO_LIST; break; default: g_return_val_if_reached (NULL); } return extension_name; } static GList * cal_source_config_list_eligible_collections (ESourceConfig *config) { GQueue trash = G_QUEUE_INIT; GList *list, *link; /* Chain up to parent's list_eligible_collections() method. */ list = E_SOURCE_CONFIG_CLASS (e_cal_source_config_parent_class)-> list_eligible_collections (config); for (link = list; link != NULL; link = g_list_next (link)) { ESource *source = E_SOURCE (link->data); ESourceCollection *extension; const gchar *extension_name; extension_name = E_SOURCE_EXTENSION_COLLECTION; extension = e_source_get_extension (source, extension_name); if (!e_source_collection_get_calendar_enabled (extension)) g_queue_push_tail (&trash, link); } /* Remove ineligible collections from the list. */ while ((link = g_queue_pop_head (&trash)) != NULL) { g_object_unref (link->data); list = g_list_delete_link (list, link); } return list; } static void cal_source_config_init_candidate (ESourceConfig *config, ESource *scratch_source) { ECalSourceConfigPrivate *priv; ESourceConfigClass *class; ESourceExtension *extension; const gchar *extension_name; /* Chain up to parent's init_candidate() method. */ class = E_SOURCE_CONFIG_CLASS (e_cal_source_config_parent_class); class->init_candidate (config, scratch_source); priv = E_CAL_SOURCE_CONFIG_GET_PRIVATE (config); extension_name = e_source_config_get_backend_extension_name (config); extension = e_source_get_extension (scratch_source, extension_name); g_object_bind_property_full ( extension, "color", priv->color_button, "color", G_BINDING_BIDIRECTIONAL | G_BINDING_SYNC_CREATE, e_binding_transform_string_to_color, e_binding_transform_color_to_string, NULL, (GDestroyNotify) NULL); } static void cal_source_config_commit_changes (ESourceConfig *config, ESource *scratch_source) { ECalSourceConfigPrivate *priv; GtkToggleButton *toggle_button; ESourceConfigClass *class; ESource *default_source; priv = E_CAL_SOURCE_CONFIG_GET_PRIVATE (config); toggle_button = GTK_TOGGLE_BUTTON (priv->default_button); /* Chain up to parent's commit_changes() method. */ class = E_SOURCE_CONFIG_CLASS (e_cal_source_config_parent_class); class->commit_changes (config, scratch_source); default_source = cal_source_config_ref_default (config); /* The default setting is a little tricky to get right. If * the toggle button is active, this ESource is now the default. * That much is simple. But if the toggle button is NOT active, * then we have to inspect the old default. If this ESource WAS * the default, reset the default to 'system'. If this ESource * WAS NOT the old default, leave it alone. */ if (gtk_toggle_button_get_active (toggle_button)) cal_source_config_set_default (config, scratch_source); else if (e_source_equal (scratch_source, default_source)) cal_source_config_set_default (config, NULL); g_object_unref (default_source); } static void e_cal_source_config_class_init (ECalSourceConfigClass *class) { GObjectClass *object_class; ESourceConfigClass *source_config_class; g_type_class_add_private (class, sizeof (ECalSourceConfigPrivate)); object_class = G_OBJECT_CLASS (class); object_class->set_property = cal_source_config_set_property; object_class->get_property = cal_source_config_get_property; object_class->dispose = cal_source_config_dispose; object_class->constructed = cal_source_config_constructed; source_config_class = E_SOURCE_CONFIG_CLASS (class); source_config_class->get_backend_extension_name = cal_source_config_get_backend_extension_name; source_config_class->list_eligible_collections = cal_source_config_list_eligible_collections; source_config_class->init_candidate = cal_source_config_init_candidate; source_config_class->commit_changes = cal_source_config_commit_changes; g_object_class_install_property ( object_class, PROP_SOURCE_TYPE, g_param_spec_enum ( "source-type", "Source Type", "The iCalendar object type", E_TYPE_CAL_CLIENT_SOURCE_TYPE, E_CAL_CLIENT_SOURCE_TYPE_EVENTS, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS)); } static void e_cal_source_config_init (ECalSourceConfig *config) { config->priv = E_CAL_SOURCE_CONFIG_GET_PRIVATE (config); } GtkWidget * e_cal_source_config_new (ESourceRegistry *registry, ESource *original_source, ECalClientSourceType source_type) { g_return_val_if_fail (E_IS_SOURCE_REGISTRY (registry), NULL); if (original_source != NULL) g_return_val_if_fail (E_IS_SOURCE (original_source), NULL); return g_object_new ( E_TYPE_CAL_SOURCE_CONFIG, "registry", registry, "original-source", original_source, "source-type", source_type, NULL); } ECalClientSourceType e_cal_source_config_get_source_type (ECalSourceConfig *config) { g_return_val_if_fail (E_IS_CAL_SOURCE_CONFIG (config), 0); return config->priv->source_type; } void e_cal_source_config_add_offline_toggle (ECalSourceConfig *config, ESource *scratch_source) { GtkWidget *widget; ESourceExtension *extension; const gchar *extension_name; const gchar *label; g_return_if_fail (E_IS_CAL_SOURCE_CONFIG (config)); g_return_if_fail (E_IS_SOURCE (scratch_source)); extension_name = E_SOURCE_EXTENSION_OFFLINE; extension = e_source_get_extension (scratch_source, extension_name); switch (e_cal_source_config_get_source_type (config)) { case E_CAL_CLIENT_SOURCE_TYPE_EVENTS: label = _("Copy calendar contents locally " "for offline operation"); break; case E_CAL_CLIENT_SOURCE_TYPE_TASKS: label = _("Copy task list contents locally " "for offline operation"); break; case E_CAL_CLIENT_SOURCE_TYPE_MEMOS: label = _("Copy memo list contents locally " "for offline operation"); break; default: g_return_if_reached (); } widget = gtk_check_button_new_with_label (label); e_source_config_insert_widget ( E_SOURCE_CONFIG (config), scratch_source, NULL, widget); gtk_widget_show (widget); g_object_bind_property ( extension, "stay-synchronized", widget, "active", G_BINDING_BIDIRECTIONAL | G_BINDING_SYNC_CREATE); }