/* gtd-provider-selector.c * * Copyright (C) 2015-2020 Georges Basile Stavracas Neto * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * 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, see . */ #define G_LOG_DOMAIN "GtdProviderSelector" #include "gtd-application.h" #include "gtd-manager.h" #include "gtd-provider.h" #include "gtd-provider-row.h" #include "gtd-provider-selector.h" #include struct _GtdProviderSelector { GtkBox parent; GtkListBox *listbox; GtkWidget *local_check; /* stub rows */ GtkWidget *exchange_stub_row; GtkWidget *google_stub_row; GtkWidget *owncloud_stub_row; GtkWidget *local_row; gint show_local_provider : 1; gint show_stub_rows : 1; }; G_DEFINE_TYPE (GtdProviderSelector, gtd_provider_selector, GTK_TYPE_BOX) enum { PROP_0, PROP_SHOW_LOCAL, PROP_SHOW_STUB_ROWS, LAST_PROP }; enum { PROVIDER_SELECTED, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0, }; static void spawn (const gchar *action, const gchar *arg) { const gchar *command[] = {"gnome-control-center", "online-accounts", action, arg, NULL}; g_spawn_async (NULL, (gchar **) command, NULL, G_SPAWN_SEARCH_PATH | G_SPAWN_STDOUT_TO_DEV_NULL, NULL, NULL, NULL, NULL); } /** * display_header_func: * * Shows a separator before each row. */ static void display_header_func (GtkListBoxRow *row, GtkListBoxRow *before, gpointer user_data) { if (before != NULL) { GtkWidget *header; header = gtk_separator_new (GTK_ORIENTATION_HORIZONTAL); gtk_list_box_row_set_header (row, header); gtk_widget_show (header); } } static void unselect_rows (GtdProviderSelector *self) { GtkWidget *child; for (child = gtk_widget_get_first_child (GTK_WIDGET (self->listbox)); child; child = gtk_widget_get_next_sibling (child)) { if (GTD_IS_PROVIDER_ROW (child)) gtd_provider_row_set_selected (GTD_PROVIDER_ROW (child), FALSE); } } static void default_provider_changed_cb (GtdProviderSelector *selector) { GtdProvider *current; GtdManager *manager; GtkWidget *child; manager = gtd_manager_get_default (); current = gtd_manager_get_default_provider (manager); for (child = gtk_widget_get_first_child (GTK_WIDGET (selector->listbox)); child; child = gtk_widget_get_next_sibling (child)) { GtdProviderRow *provider_row; GtdProvider *provider; if (!GTD_IS_PROVIDER_ROW (child)) continue; provider_row = GTD_PROVIDER_ROW (child); provider = gtd_provider_row_get_provider (provider_row); gtd_provider_row_set_selected (provider_row, provider == current); } g_signal_emit (selector, signals[PROVIDER_SELECTED], 0, current); } static void gtd_provider_selector__listbox_row_activated (GtdProviderSelector *selector, GtkWidget *row) { g_return_if_fail (GTD_IS_PROVIDER_SELECTOR (selector)); /* The row is either one of the stub rows, or a GtdGoaRow */ if (row == selector->google_stub_row) { spawn ("add", "google"); } else if (row == selector->owncloud_stub_row) { spawn ("add", "owncloud"); } else if (row == selector->exchange_stub_row) { spawn ("add", "exchange"); } else { GtdProvider *provider; provider = gtd_provider_row_get_provider (GTD_PROVIDER_ROW (row)); unselect_rows (selector); /* * If the account has it's calendars disabled, we cannot let it * be a default provider location. Instead, open the Control Center * to give the user the ability to change it. */ if (gtd_provider_get_enabled (provider)) { gtd_provider_row_set_selected (GTD_PROVIDER_ROW (row), TRUE); g_signal_emit (selector, signals[PROVIDER_SELECTED], 0, provider); } else { spawn ((gchar*) gtd_provider_get_id (provider), NULL); } } } static void on_local_check_toggled_cb (GtdProviderSelector *selector, GtkCheckButton *check) { g_return_if_fail (GTD_IS_PROVIDER_SELECTOR (selector)); /* * Unset the currently selected provider location row when the check button is * activated. No need to do this when deactivated, since we already did. */ if (gtk_check_button_get_active (check)) { GtdProvider *local_provider; local_provider = gtd_provider_row_get_provider (GTD_PROVIDER_ROW (selector->local_row)); unselect_rows (selector); /* * Sets the provider location to "local", and don't unset it if the * check gets deactivated. */ g_signal_emit (selector, signals[PROVIDER_SELECTED], 0, local_provider); } else { g_signal_emit (selector, signals[PROVIDER_SELECTED], 0, NULL); } } static void remove_provider (GtdProviderSelector *selector, GtdProvider *provider) { GtkWidget *child; gint exchange; gint google; gint owncloud; g_return_if_fail (GTD_IS_PROVIDER_SELECTOR (selector)); g_return_if_fail (GTD_IS_PROVIDER (provider)); exchange = google = owncloud = 0; for (child = gtk_widget_get_first_child (GTK_WIDGET (selector->listbox)); child; child = gtk_widget_get_next_sibling (child)) { GtdProviderRow *provider_row; GtdProvider *row_provider; const gchar *provider_id; if (!GTD_IS_PROVIDER_ROW (child)) continue; provider_row = GTD_PROVIDER_ROW (child); row_provider = gtd_provider_row_get_provider (provider_row); provider_id = gtd_provider_get_id (row_provider); if (row_provider == provider) { gtk_list_box_remove (selector->listbox, child); } else { if (g_strcmp0 (provider_id, "exchange") == 0) exchange++; else if (g_strcmp0 (provider_id, "google") == 0) google++; else if (g_strcmp0 (provider_id, "owncloud") == 0) owncloud++; } } gtk_widget_set_visible (selector->exchange_stub_row, exchange == 0); gtk_widget_set_visible (selector->google_stub_row, google == 0); gtk_widget_set_visible (selector->owncloud_stub_row, owncloud == 0); } static void add_provider (GtdProviderSelector *selector, GtdProvider *provider) { GtkWidget *row; const gchar *provider_id; g_return_if_fail (GTD_IS_PROVIDER_SELECTOR (selector)); g_return_if_fail (GTD_IS_PROVIDER (provider)); row = gtd_provider_row_new (provider); provider_id = gtd_provider_get_id (provider); g_debug ("Adding provider %s", provider_id); gtk_list_box_insert (selector->listbox, row, -1); /* track the local provider row */ if (g_str_has_prefix (provider_id, "local") == 0) { gtk_widget_set_visible (row, selector->show_local_provider); selector->local_row = row; } /* Auto selects the default provider row when needed */ if (!gtd_provider_selector_get_selected_provider (selector)) gtd_provider_selector_set_selected_provider (selector, provider); /* hide the related stub row */ if (g_str_has_prefix (provider_id, "exchange") == 0) gtk_widget_hide (selector->exchange_stub_row); else if (g_str_has_prefix (provider_id, "google") == 0) gtk_widget_hide (selector->google_stub_row); else if (g_str_has_prefix (provider_id, "owncloud") == 0) gtk_widget_hide (selector->owncloud_stub_row); } static void gtd_provider_selector__fill_accounts (GtdProviderSelector *selector) { GtdManager *manager; GList *providers; GList *l; g_return_if_fail (GTD_IS_PROVIDER_SELECTOR (selector)); manager = gtd_manager_get_default (); /* load accounts */ providers = gtd_manager_get_providers (manager); for (l = providers; l != NULL; l = l->next) add_provider (selector, l->data); g_list_free (providers); } static gint sort_func (GtkListBoxRow *row1, GtkListBoxRow *row2, gpointer user_data) { GtdProvider *provider1; GtdProvider *provider2; if (!GTD_IS_PROVIDER_ROW (row1)) return 1; else if (!GTD_IS_PROVIDER_ROW (row2)) return -1; provider1 = gtd_provider_row_get_provider (GTD_PROVIDER_ROW (row1)); provider2 = gtd_provider_row_get_provider (GTD_PROVIDER_ROW (row2)); return provider2 != provider1;//gtd_provider_compare (provider1, provider2); } static void gtd_provider_selector_constructed (GObject *object) { GtdProviderSelector *self = GTD_PROVIDER_SELECTOR (object); G_OBJECT_CLASS (gtd_provider_selector_parent_class)->constructed (object); gtk_list_box_set_header_func (GTK_LIST_BOX (self->listbox), display_header_func, NULL, NULL); gtk_list_box_set_sort_func (GTK_LIST_BOX (self->listbox), (GtkListBoxSortFunc) sort_func, NULL, NULL); } static void gtd_provider_selector_finalize (GObject *object) { G_OBJECT_CLASS (gtd_provider_selector_parent_class)->finalize (object); } static void gtd_provider_selector_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GtdProviderSelector *self = GTD_PROVIDER_SELECTOR (object); switch (prop_id) { case PROP_SHOW_LOCAL: g_value_set_boolean (value, self->show_local_provider); break; case PROP_SHOW_STUB_ROWS: g_value_set_boolean (value, self->show_stub_rows); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void gtd_provider_selector_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GtdProviderSelector *self = GTD_PROVIDER_SELECTOR (object); switch (prop_id) { case PROP_SHOW_LOCAL: gtd_provider_selector_show_local (self, g_value_get_boolean (value)); break; case PROP_SHOW_STUB_ROWS: gtd_provider_selector_set_show_stub_rows (self, g_value_get_boolean (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void gtd_provider_selector_class_init (GtdProviderSelectorClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); object_class->finalize = gtd_provider_selector_finalize; object_class->constructed = gtd_provider_selector_constructed; object_class->get_property = gtd_provider_selector_get_property; object_class->set_property = gtd_provider_selector_set_property; /** * GtdProviderSelector::location-selected: * * Emitted when a provider location is selected. */ signals[PROVIDER_SELECTED] = g_signal_new ("provider-selected", GTD_TYPE_PROVIDER_SELECTOR, G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 1, GTD_TYPE_PROVIDER); /** * GtdProviderSelector::show-local-provider: * * Whether it should show a row for the local provider. */ g_object_class_install_property ( object_class, PROP_SHOW_LOCAL, g_param_spec_boolean ("show-local", "Show local provider row", "Whether should show a local provider row instead of a checkbox", FALSE, G_PARAM_READWRITE)); /** * GtdProviderSelector::show-stub-rows: * * Whether it should show stub rows for non-added accounts. */ g_object_class_install_property ( object_class, PROP_SHOW_STUB_ROWS, g_param_spec_boolean ("show-stub-rows", "Show stub rows", "Whether should show stub rows for non-added accounts", TRUE, G_PARAM_READWRITE)); gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/todo/ui/gtd-provider-selector.ui"); gtk_widget_class_bind_template_child (widget_class, GtdProviderSelector, exchange_stub_row); gtk_widget_class_bind_template_child (widget_class, GtdProviderSelector, google_stub_row); gtk_widget_class_bind_template_child (widget_class, GtdProviderSelector, listbox); gtk_widget_class_bind_template_child (widget_class, GtdProviderSelector, local_check); gtk_widget_class_bind_template_child (widget_class, GtdProviderSelector, owncloud_stub_row); gtk_widget_class_bind_template_callback (widget_class, on_local_check_toggled_cb); gtk_widget_class_bind_template_callback (widget_class, gtd_provider_selector__listbox_row_activated); } static void gtd_provider_selector_init (GtdProviderSelector *self) { GtdManager *manager; self->show_stub_rows = TRUE; /* Setup the manager */ manager = gtd_manager_get_default (); gtk_widget_init_template (GTK_WIDGET (self)); gtd_provider_selector__fill_accounts (self); g_signal_connect_swapped (manager, "notify::default-provider", G_CALLBACK (default_provider_changed_cb), self); g_signal_connect_swapped (manager, "provider-added", G_CALLBACK (add_provider), self); g_signal_connect_swapped (manager, "provider-removed", G_CALLBACK (remove_provider), self); } /** * gtd_provider_selector_new: * * Creates a new #GtdProviderSelector. * * Returns: (transfer full): a new #GtdProviderSelector */ GtkWidget* gtd_provider_selector_new (void) { return g_object_new (GTD_TYPE_PROVIDER_SELECTOR, NULL); } /** * gtd_provider_selector_show_local: * * Shows a row for local provider item. */ void gtd_provider_selector_show_local (GtdProviderSelector *selector, gboolean show) { g_return_if_fail (GTD_IS_PROVIDER_SELECTOR (selector)); if (selector->show_local_provider != show) { selector->show_local_provider = show; gtk_widget_set_visible (selector->local_check, !show); if (selector->local_row) gtk_widget_set_visible (selector->local_row, show); g_object_notify (G_OBJECT (selector), "show-local"); } } /** * gtd_provider_selector_get_selected_provider: * @selector: a #GtdProviderSelector * * Retrieves the currently selected #GtdProvider, or %NULL if * none is selected. * * Returns: (transfer none): the selected #GtdProvider */ GtdProvider* gtd_provider_selector_get_selected_provider (GtdProviderSelector *selector) { GtdProvider *provider; GtkWidget *child; g_return_val_if_fail (GTD_IS_PROVIDER_SELECTOR (selector), NULL); provider = NULL; for (child = gtk_widget_get_first_child (GTK_WIDGET (selector->listbox)); child; child = gtk_widget_get_next_sibling (child)) { GtdProviderRow *provider_row; if (!GTD_IS_PROVIDER_ROW (child)) continue; provider_row = GTD_PROVIDER_ROW (child); if (gtd_provider_row_get_selected (provider_row)) { provider = gtd_provider_row_get_provider (provider_row); break; } } return provider; } /** * gtd_provider_selector_set_selected_provider: * @selector: a #GtdProviderSelector * @provider: a #GtdProvider * * Selects @provider in the given #GtdProviderSelector. */ void gtd_provider_selector_set_selected_provider (GtdProviderSelector *selector, GtdProvider *provider) { GtkWidget *child; g_return_if_fail (GTD_IS_PROVIDER_SELECTOR (selector)); for (child = gtk_widget_get_first_child (GTK_WIDGET (selector->listbox)); child; child = gtk_widget_get_next_sibling (child)) { GtdProviderRow *provider_row; if (!GTD_IS_PROVIDER_ROW (child)) continue; provider_row = GTD_PROVIDER_ROW (child); gtd_provider_row_set_selected (provider_row, gtd_provider_row_get_provider (provider_row) == provider); g_signal_emit (selector, signals[PROVIDER_SELECTED], 0, provider); } } /** * gtd_provider_selector_get_show_stub_rows: * @selector: a #GtdProviderSelector * * Retrieves the ::show-stub-rows property. * * Returns: %TRUE if it shows stub rows, %FALSE if it hides them. */ gboolean gtd_provider_selector_get_show_stub_rows (GtdProviderSelector *selector) { g_return_val_if_fail (GTD_IS_PROVIDER_SELECTOR (selector), FALSE); return selector->show_stub_rows; } /** * gtd_provider_selector_set_show_stub_rows: * @selector: a #GtdProviderSelector * @show_stub_rows: %TRUE to show stub rows, %FALSE to hide them. * * Sets the #GtdProviderSelector::show-stub-rows property. * * Returns: */ void gtd_provider_selector_set_show_stub_rows (GtdProviderSelector *selector, gboolean show_stub_rows) { g_return_if_fail (GTD_IS_PROVIDER_SELECTOR (selector)); if (selector->show_stub_rows != show_stub_rows) { selector->show_stub_rows = show_stub_rows; /* * If we're showing the stub rows, it must check which ones should be shown. * We don't want to show stub rows for */ if (show_stub_rows) { GtkWidget *child; gint google_counter; gint exchange_counter; gint owncloud_counter; google_counter = 0; exchange_counter = 0; owncloud_counter = 0; for (child = gtk_widget_get_first_child (GTK_WIDGET (selector->listbox)); child; child = gtk_widget_get_next_sibling (child)) { if (GTD_IS_PROVIDER_ROW (child)) { GtdProvider *provider = gtd_provider_row_get_provider (GTD_PROVIDER_ROW (child)); const gchar *type; type = gtd_provider_get_id (provider); if (g_strcmp0 (type, "google") == 0) google_counter++; else if (g_strcmp0 (type, "exchange") == 0) exchange_counter++; else if (g_strcmp0 (type, "owncloud") == 0) owncloud_counter++; } } gtk_widget_set_visible (selector->google_stub_row, google_counter == 0); gtk_widget_set_visible (selector->exchange_stub_row, exchange_counter == 0); gtk_widget_set_visible (selector->owncloud_stub_row, owncloud_counter == 0); } else { gtk_widget_hide (selector->exchange_stub_row); gtk_widget_hide (selector->google_stub_row); gtk_widget_hide (selector->owncloud_stub_row); } g_object_notify (G_OBJECT (selector), "show-stub-rows"); } }