515 lines
19 KiB
C
515 lines
19 KiB
C
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
/*
|
|
* Copyright (C) 2012 Giovanni Campagna <scampa.giovanni@gmail.com>
|
|
* Copyright (C) 2015 Red Hat, Inc.
|
|
*
|
|
* This library 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) any later version.
|
|
*
|
|
* This library 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 this library; if not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <string.h>
|
|
#include <glib/gi18n-lib.h>
|
|
#include <glib.h>
|
|
#include <gio/gio.h>
|
|
#include <gio/gdesktopappinfo.h>
|
|
|
|
#include "list-box-helper.h"
|
|
#include "cc-notifications-panel.h"
|
|
#include "cc-edit-dialog.h"
|
|
|
|
/*
|
|
* Key Switch
|
|
*
|
|
* "enable", "notifications-switch" When set to off, all other switches in the dialog are insensitive
|
|
* "enable-sound-alerts", "sound-alerts-switch"
|
|
* "show-banners", "notification-banners-switch" Off and insensitive when corresponding panel switch is off
|
|
* "force-expanded", "notification-banners-content-switch" Off and insensitive when switch above is off
|
|
* "show-in-lock-screen", "lock-screen-notifications-switch" Off and insensitive when corresponding panel switch is off
|
|
* "details-in-lock-screen", "lock-screen-content-switch" Off and insensitive when switch above is off
|
|
*/
|
|
|
|
static void update_banner_switch (GtkWidget *dialog);
|
|
static void update_banner_content_switch (GtkWidget *dialog);
|
|
static void update_lock_screen_switch (GtkWidget *dialog);
|
|
static void update_lock_screen_content_switch (GtkWidget *dialog);
|
|
static void update_sound_switch (GtkWidget *dialog);
|
|
static void update_notification_switch (GtkWidget *dialog);
|
|
static void update_switches (GtkWidget *dialog);
|
|
|
|
static void
|
|
dialog_set_app_id (GtkWidget *dialog,
|
|
const char *app_id)
|
|
{
|
|
g_object_set_data_full (G_OBJECT (dialog), "app-id", g_strdup (app_id), g_free);
|
|
}
|
|
|
|
static const char *
|
|
dialog_get_app_id (GtkWidget *dialog)
|
|
{
|
|
return (const char*)g_object_get_data (G_OBJECT (dialog), "app-id");
|
|
}
|
|
|
|
static void
|
|
dialog_set_perm_store (GtkWidget *dialog,
|
|
GDBusProxy *proxy)
|
|
{
|
|
if (proxy)
|
|
g_object_set_data_full (G_OBJECT (dialog), "perm-store", g_object_ref (proxy), g_object_unref);
|
|
}
|
|
|
|
static GDBusProxy *
|
|
dialog_get_perm_store (GtkWidget *dialog)
|
|
{
|
|
return (GDBusProxy *)g_object_get_data (G_OBJECT (dialog), "perm-store");
|
|
}
|
|
|
|
static GtkWidget *
|
|
get_switch (GtkBuilder *builder,
|
|
const gchar *prefix)
|
|
{
|
|
g_autofree gchar *name = NULL;
|
|
|
|
name = g_strdup_printf ("%s-switch", prefix);
|
|
return GTK_WIDGET (gtk_builder_get_object (builder, name));
|
|
}
|
|
|
|
static void
|
|
set_key_from_switch (GtkWidget *dialog,
|
|
const gchar *key,
|
|
GtkSwitch *the_switch)
|
|
{
|
|
GSettings *settings;
|
|
|
|
settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "settings"));
|
|
|
|
g_settings_set_boolean (settings, key, gtk_switch_get_active (the_switch));
|
|
}
|
|
|
|
static void
|
|
on_perm_store_set_done (GObject *source_object,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
g_autoptr(GVariant) results = NULL;
|
|
g_autoptr(GError) error = NULL;
|
|
|
|
results = g_dbus_proxy_call_finish (G_DBUS_PROXY (source_object),
|
|
res,
|
|
&error);
|
|
if (results == NULL)
|
|
{
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
|
g_warning ("Failed to store permissions: %s", error->message);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
set_portal_permissions_for_app (GtkWidget *dialog, GtkSwitch *the_switch)
|
|
{
|
|
GDBusProxy *perm_store = dialog_get_perm_store (dialog);
|
|
const char *app_id = dialog_get_app_id (dialog);
|
|
gboolean allow = gtk_switch_get_active (the_switch);
|
|
g_autoptr(GVariant) perms = NULL;
|
|
g_autoptr(GVariant) new_perms = NULL;
|
|
g_autoptr(GVariant) data = NULL;
|
|
GVariantBuilder builder;
|
|
gboolean found;
|
|
int i;
|
|
const char *yes_strv[] = { "yes", NULL };
|
|
const char *no_strv[] = { "no", NULL };
|
|
g_autoptr(GVariant) reply = NULL;
|
|
g_autoptr(GError) error = NULL;
|
|
|
|
if (perm_store == NULL)
|
|
{
|
|
g_warning ("Could not find PermissionStore, not syncing notification permissions");
|
|
return;
|
|
}
|
|
|
|
new_perms = g_variant_new_strv (allow ? yes_strv : no_strv, 1);
|
|
g_variant_ref_sink (new_perms);
|
|
|
|
g_variant_builder_init (&builder, G_VARIANT_TYPE("a{sas}"));
|
|
found = FALSE;
|
|
|
|
reply = g_dbus_proxy_call_sync (perm_store,
|
|
"Lookup",
|
|
g_variant_new ("(ss)",
|
|
"notifications",
|
|
"notification"),
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
NULL);
|
|
if (reply)
|
|
{
|
|
g_variant_get (reply, "(@a{sas}v)", &perms, &data);
|
|
|
|
for (i = 0; i < g_variant_n_children (perms); i++)
|
|
{
|
|
const char *key;
|
|
g_autoptr(GVariant) value = NULL;
|
|
|
|
g_variant_get_child (perms, i, "{&s@as}", &key, &value);
|
|
if (g_strcmp0 (key, app_id) == 0)
|
|
{
|
|
found = TRUE;
|
|
g_variant_builder_add (&builder, "{s@as}", key, new_perms);
|
|
}
|
|
else
|
|
g_variant_builder_add (&builder, "{s@as}", key, value);
|
|
}
|
|
}
|
|
|
|
if (!found)
|
|
g_variant_builder_add (&builder, "{s@as}", app_id, new_perms);
|
|
|
|
|
|
g_dbus_proxy_call (perm_store,
|
|
"Set",
|
|
g_variant_new ("(sbsa{sas}v)",
|
|
"notifications",
|
|
TRUE,
|
|
"notification",
|
|
&builder,
|
|
data ? data : g_variant_new_byte (0)),
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
on_perm_store_set_done,
|
|
data);
|
|
}
|
|
|
|
static void
|
|
notifications_switch_state_set_cb (GtkSwitch *widget,
|
|
GParamSpec *pspec,
|
|
GtkWidget *dialog)
|
|
{
|
|
set_key_from_switch (dialog, "enable", widget);
|
|
set_portal_permissions_for_app (dialog, widget);
|
|
update_sound_switch (dialog);
|
|
update_banner_switch (dialog);
|
|
update_banner_content_switch (dialog);
|
|
update_lock_screen_switch (dialog);
|
|
update_lock_screen_content_switch (dialog);
|
|
}
|
|
|
|
static void
|
|
sound_alerts_switch_state_set_cb (GtkSwitch *widget,
|
|
GParamSpec *pspec,
|
|
GtkWidget *dialog)
|
|
{
|
|
set_key_from_switch (dialog, "enable-sound-alerts", widget);
|
|
}
|
|
|
|
static void
|
|
notification_banners_switch_state_set_cb (GtkSwitch *widget,
|
|
GParamSpec *pspec,
|
|
GtkWidget *dialog)
|
|
{
|
|
set_key_from_switch (dialog, "show-banners", widget);
|
|
update_banner_content_switch (dialog);
|
|
}
|
|
|
|
static void
|
|
notification_banners_content_switch_state_set_cb (GtkSwitch *widget,
|
|
GParamSpec *pspec,
|
|
GtkWidget *dialog)
|
|
{
|
|
set_key_from_switch (dialog, "force-expanded", widget);
|
|
}
|
|
|
|
static void
|
|
lock_screen_notifications_switch_state_set_cb (GtkSwitch *widget,
|
|
GParamSpec *pspec,
|
|
GtkWidget *dialog)
|
|
{
|
|
set_key_from_switch (dialog, "show-in-lock-screen", widget);
|
|
update_lock_screen_content_switch (dialog);
|
|
}
|
|
|
|
static void
|
|
lock_screen_content_switch_state_set_cb (GtkSwitch *widget,
|
|
GParamSpec *pspec,
|
|
GtkWidget *dialog)
|
|
{
|
|
set_key_from_switch (dialog, "details-in-lock-screen", widget);
|
|
}
|
|
|
|
static void
|
|
update_switches (GtkWidget *dialog)
|
|
{
|
|
update_notification_switch (dialog);
|
|
update_sound_switch (dialog);
|
|
update_banner_switch (dialog);
|
|
update_banner_content_switch (dialog);
|
|
update_lock_screen_switch (dialog);
|
|
update_lock_screen_content_switch (dialog);
|
|
}
|
|
|
|
static void
|
|
update_notification_switch (GtkWidget *dialog)
|
|
{
|
|
GtkBuilder *builder;
|
|
GSettings *settings;
|
|
GtkWidget *widget;
|
|
|
|
builder = GTK_BUILDER (g_object_get_data (G_OBJECT (dialog), "builder"));
|
|
settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "settings"));
|
|
|
|
widget = get_switch (builder, "notifications");
|
|
g_signal_handlers_block_by_func (G_OBJECT (widget), notifications_switch_state_set_cb, dialog);
|
|
gtk_switch_set_active (GTK_SWITCH (widget), g_settings_get_boolean (settings, "enable"));
|
|
g_signal_handlers_unblock_by_func (G_OBJECT (widget), notifications_switch_state_set_cb, dialog);
|
|
}
|
|
|
|
static void
|
|
update_sound_switch (GtkWidget *dialog)
|
|
{
|
|
GtkBuilder *builder;
|
|
GSettings *settings;
|
|
GtkWidget *widget;
|
|
|
|
builder = GTK_BUILDER (g_object_get_data (G_OBJECT (dialog), "builder"));
|
|
settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "settings"));
|
|
|
|
widget = get_switch (builder, "sound-alerts");
|
|
g_signal_handlers_block_by_func (G_OBJECT (widget), sound_alerts_switch_state_set_cb, dialog);
|
|
gtk_switch_set_active (GTK_SWITCH (widget), g_settings_get_boolean (settings, "enable-sound-alerts"));
|
|
g_signal_handlers_unblock_by_func (G_OBJECT (widget), sound_alerts_switch_state_set_cb, dialog);
|
|
gtk_widget_set_sensitive (widget, g_settings_get_boolean (settings, "enable"));
|
|
}
|
|
|
|
static void
|
|
update_banner_switch (GtkWidget *dialog)
|
|
{
|
|
GtkBuilder *builder;
|
|
GSettings *settings;
|
|
GSettings *master_settings;
|
|
GtkWidget *widget;
|
|
gboolean notifications_enabled;
|
|
gboolean show_banners;
|
|
gboolean active;
|
|
gboolean sensitive;
|
|
|
|
builder = GTK_BUILDER (g_object_get_data (G_OBJECT (dialog), "builder"));
|
|
settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "settings"));
|
|
master_settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "master-settings"));
|
|
|
|
show_banners = g_settings_get_boolean (master_settings, "show-banners");
|
|
notifications_enabled = g_settings_get_boolean (settings, "enable");
|
|
|
|
widget = get_switch (builder, "notification-banners");
|
|
active = g_settings_get_boolean (settings, "show-banners") &&
|
|
show_banners;
|
|
sensitive = notifications_enabled &&
|
|
show_banners;
|
|
g_signal_handlers_block_by_func (G_OBJECT (widget), notification_banners_switch_state_set_cb, dialog);
|
|
gtk_switch_set_active (GTK_SWITCH (widget), active);
|
|
g_signal_handlers_unblock_by_func (G_OBJECT (widget), notification_banners_switch_state_set_cb, dialog);
|
|
gtk_widget_set_sensitive (widget, sensitive);
|
|
}
|
|
|
|
static void
|
|
update_banner_content_switch (GtkWidget *dialog)
|
|
{
|
|
GtkBuilder *builder;
|
|
GSettings *settings;
|
|
GSettings *master_settings;
|
|
GtkWidget *widget;
|
|
gboolean notifications_enabled;
|
|
gboolean show_banners;
|
|
gboolean active;
|
|
gboolean sensitive;
|
|
|
|
builder = GTK_BUILDER (g_object_get_data (G_OBJECT (dialog), "builder"));
|
|
settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "settings"));
|
|
master_settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "master-settings"));
|
|
|
|
show_banners = g_settings_get_boolean (master_settings, "show-banners");
|
|
notifications_enabled = g_settings_get_boolean (settings, "enable");
|
|
|
|
widget = get_switch (builder, "notification-banners-content");
|
|
active = g_settings_get_boolean (settings, "force-expanded") &&
|
|
g_settings_get_boolean (settings, "show-banners") &&
|
|
show_banners;
|
|
sensitive = g_settings_get_boolean (settings, "show-banners") &&
|
|
notifications_enabled &&
|
|
show_banners;
|
|
g_signal_handlers_block_by_func (G_OBJECT (widget), notification_banners_content_switch_state_set_cb, dialog);
|
|
gtk_switch_set_active (GTK_SWITCH (widget), active);
|
|
g_signal_handlers_unblock_by_func (G_OBJECT (widget), notification_banners_content_switch_state_set_cb, dialog);
|
|
gtk_widget_set_sensitive (widget, sensitive);
|
|
}
|
|
|
|
static void
|
|
update_lock_screen_switch (GtkWidget *dialog)
|
|
{
|
|
GtkBuilder *builder;
|
|
GSettings *settings;
|
|
GSettings *master_settings;
|
|
GtkWidget *widget;
|
|
gboolean notifications_enabled;
|
|
gboolean show_in_lock_screen;
|
|
gboolean active;
|
|
gboolean sensitive;
|
|
|
|
builder = GTK_BUILDER (g_object_get_data (G_OBJECT (dialog), "builder"));
|
|
settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "settings"));
|
|
master_settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "master-settings"));
|
|
|
|
show_in_lock_screen = g_settings_get_boolean (master_settings, "show-in-lock-screen");
|
|
notifications_enabled = g_settings_get_boolean (settings, "enable");
|
|
|
|
widget = get_switch (builder, "lock-screen-notifications");
|
|
active = g_settings_get_boolean (settings, "show-in-lock-screen") &&
|
|
show_in_lock_screen;
|
|
sensitive = notifications_enabled &&
|
|
show_in_lock_screen;
|
|
|
|
g_signal_handlers_block_by_func (G_OBJECT (widget), lock_screen_notifications_switch_state_set_cb, dialog);
|
|
gtk_switch_set_active (GTK_SWITCH (widget), active);
|
|
g_signal_handlers_unblock_by_func (G_OBJECT (widget), lock_screen_notifications_switch_state_set_cb, dialog);
|
|
gtk_widget_set_sensitive (widget, sensitive);
|
|
}
|
|
|
|
static void
|
|
update_lock_screen_content_switch (GtkWidget *dialog)
|
|
{
|
|
GtkBuilder *builder;
|
|
GSettings *settings;
|
|
GSettings *master_settings;
|
|
GtkWidget *widget;
|
|
gboolean notifications_enabled;
|
|
gboolean show_in_lock_screen;
|
|
gboolean active;
|
|
gboolean sensitive;
|
|
|
|
builder = GTK_BUILDER (g_object_get_data (G_OBJECT (dialog), "builder"));
|
|
settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "settings"));
|
|
master_settings = G_SETTINGS (g_object_get_data (G_OBJECT (dialog), "master-settings"));
|
|
|
|
show_in_lock_screen = g_settings_get_boolean (master_settings, "show-in-lock-screen");
|
|
notifications_enabled = g_settings_get_boolean (settings, "enable");
|
|
|
|
widget = get_switch (builder, "lock-screen-content");
|
|
active = g_settings_get_boolean (settings, "details-in-lock-screen") &&
|
|
g_settings_get_boolean (settings, "show-in-lock-screen") &&
|
|
show_in_lock_screen;
|
|
sensitive = g_settings_get_boolean (settings, "show-in-lock-screen") &&
|
|
notifications_enabled &&
|
|
show_in_lock_screen;
|
|
g_signal_handlers_block_by_func (G_OBJECT (widget), lock_screen_content_switch_state_set_cb, dialog);
|
|
gtk_switch_set_active (GTK_SWITCH (widget), active);
|
|
g_signal_handlers_unblock_by_func (G_OBJECT (widget), lock_screen_content_switch_state_set_cb, dialog);
|
|
gtk_widget_set_sensitive (widget, sensitive);
|
|
}
|
|
|
|
void
|
|
cc_build_edit_dialog (CcNotificationsPanel *panel,
|
|
GAppInfo *app,
|
|
GSettings *settings,
|
|
GSettings *master_settings,
|
|
GDBusProxy *perm_store)
|
|
{
|
|
g_autoptr(GtkBuilder) builder = NULL;
|
|
GtkWindow *shell;
|
|
GtkWidget *dialog;
|
|
GtkWidget *listbox;
|
|
g_autoptr(GError) error = NULL;
|
|
gchar *objects[] = { "edit-dialog", NULL };
|
|
guint builder_result;
|
|
g_autofree gchar *app_id = NULL;
|
|
|
|
builder = gtk_builder_new ();
|
|
builder_result = gtk_builder_add_objects_from_resource (builder,
|
|
"/org/gnome/control-center/notifications/edit-dialog.ui",
|
|
objects,
|
|
&error);
|
|
|
|
if (builder_result == 0)
|
|
{
|
|
g_warning ("Could not load ui: %s", error->message);
|
|
return;
|
|
}
|
|
|
|
shell = GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (panel)));
|
|
|
|
dialog = GTK_WIDGET (gtk_builder_get_object (builder, "edit-dialog"));
|
|
|
|
app_id = g_strdup (g_app_info_get_id (app));
|
|
if (g_str_has_suffix (app_id, ".desktop"))
|
|
app_id[strlen (app_id) - strlen (".desktop")] = '\0';
|
|
dialog_set_app_id (dialog, app_id);
|
|
|
|
dialog_set_perm_store (dialog, perm_store);
|
|
|
|
g_object_set (dialog,
|
|
"title", g_app_info_get_name (app),
|
|
"transient-for", shell,
|
|
NULL);
|
|
|
|
listbox = GTK_WIDGET (gtk_builder_get_object (builder,
|
|
"main-listbox"));
|
|
|
|
gtk_list_box_set_header_func (GTK_LIST_BOX (listbox),
|
|
cc_list_box_update_header_func,
|
|
NULL, NULL);
|
|
|
|
/*
|
|
* Store builder, settings and master_settings to the dialog so we can
|
|
* access them from callbacks easily.
|
|
*/
|
|
g_object_set_data_full (G_OBJECT (dialog),
|
|
"builder",
|
|
g_object_ref (builder),
|
|
g_object_unref);
|
|
|
|
g_object_set_data_full (G_OBJECT (dialog),
|
|
"settings",
|
|
g_object_ref (settings),
|
|
g_object_unref);
|
|
|
|
g_object_set_data_full (G_OBJECT (dialog),
|
|
"master-settings",
|
|
g_object_ref (master_settings),
|
|
g_object_unref);
|
|
|
|
/* Connect signals */
|
|
gtk_builder_add_callback_symbols (builder,
|
|
"notifications_switch_state_set_cb",
|
|
G_CALLBACK (notifications_switch_state_set_cb),
|
|
"sound_alerts_switch_state_set_cb",
|
|
G_CALLBACK (sound_alerts_switch_state_set_cb),
|
|
"notification_banners_switch_state_set_cb",
|
|
G_CALLBACK (notification_banners_switch_state_set_cb),
|
|
"notification_banners_content_switch_state_set_cb",
|
|
G_CALLBACK (notification_banners_content_switch_state_set_cb),
|
|
"lock_screen_notifications_switch_state_set_cb",
|
|
G_CALLBACK (lock_screen_notifications_switch_state_set_cb),
|
|
"lock_screen_content_switch_state_set_cb",
|
|
G_CALLBACK (lock_screen_content_switch_state_set_cb),
|
|
NULL);
|
|
|
|
gtk_builder_connect_signals (builder, dialog);
|
|
|
|
/* Init states of switches */
|
|
update_switches (dialog);
|
|
|
|
/* Show the dialog */
|
|
gtk_widget_show_all (dialog);
|
|
}
|