/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
*
* Copyright (C) 2018 Red Hat, Inc
*
* 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 2 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 .
*
* Author: Matthias Clasen
*/
#include "cc-camera-panel.h"
#include "cc-camera-resources.h"
#include "cc-util.h"
#include
#include
#include
#define APP_PERMISSIONS_TABLE "devices"
#define APP_PERMISSIONS_ID "camera"
struct _CcCameraPanel
{
CcPanel parent_instance;
GtkStack *stack;
GtkListBox *camera_apps_list_box;
GtkSwitch *main_switch;
GSettings *privacy_settings;
GDBusProxy *perm_store;
GVariant *camera_apps_perms;
GVariant *camera_apps_data;
GHashTable *camera_app_switches;
GtkSizeGroup *camera_icon_size_group;
};
CC_PANEL_REGISTER (CcCameraPanel, cc_camera_panel)
typedef struct
{
CcCameraPanel *self;
GtkWidget *widget;
gchar *app_id;
gboolean changing_state;
gboolean pending_state;
} CameraAppStateData;
static void
camera_app_state_data_free (CameraAppStateData *data)
{
g_free (data->app_id);
g_slice_free (CameraAppStateData, data);
}
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;
CameraAppStateData *data;
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;
}
data = (CameraAppStateData *) user_data;
data->changing_state = FALSE;
gtk_switch_set_state (GTK_SWITCH (data->widget), data->pending_state);
}
static gboolean
on_camera_app_state_set (GtkSwitch *widget,
gboolean state,
gpointer user_data)
{
CameraAppStateData *data = (CameraAppStateData *) user_data;
GVariantBuilder builder;
CcCameraPanel *self;
GVariantIter iter;
GVariant *params;
const gchar *key;
gchar **value;
self = data->self;
if (data->changing_state)
return TRUE;
data->changing_state = TRUE;
data->pending_state = state;
g_variant_iter_init (&iter, self->camera_apps_perms);
g_variant_builder_init (&builder, G_VARIANT_TYPE_ARRAY);
while (g_variant_iter_loop (&iter, "{&s^a&s}", &key, &value))
{
gchar *tmp = NULL;
/* It's OK to drop the entry if it's not in expected format */
if (g_strv_length (value) != 1)
continue;
if (g_strcmp0 (data->app_id, key) == 0)
{
tmp = value[0];
value[0] = state ? "yes" : "no";
}
g_variant_builder_add (&builder, "{s^as}", key, value);
if (tmp != NULL)
value[0] = tmp;
}
params = g_variant_new ("(sbsa{sas}v)",
APP_PERMISSIONS_TABLE,
TRUE,
APP_PERMISSIONS_ID,
&builder,
self->camera_apps_data);
g_dbus_proxy_call (self->perm_store,
"Set",
params,
G_DBUS_CALL_FLAGS_NONE,
-1,
cc_panel_get_cancellable (CC_PANEL (self)),
on_perm_store_set_done,
data);
return TRUE;
}
static void
add_camera_app (CcCameraPanel *self,
const gchar *app_id,
gboolean enabled)
{
g_autofree gchar *desktop_id = NULL;
CameraAppStateData *data;
GDesktopAppInfo *app_info;
GtkWidget *row, *w;
GIcon *icon;
w = g_hash_table_lookup (self->camera_app_switches, app_id);
if (w != NULL)
{
gtk_switch_set_active (GTK_SWITCH (w), enabled);
return;
}
desktop_id = g_strdup_printf ("%s.desktop", app_id);
app_info = g_desktop_app_info_new (desktop_id);
if (!app_info)
return;
row = adw_action_row_new ();
gtk_list_box_append (self->camera_apps_list_box, row);
icon = g_app_info_get_icon (G_APP_INFO (app_info));
w = gtk_image_new_from_gicon (icon);
gtk_widget_set_valign (w, GTK_ALIGN_CENTER);
gtk_size_group_add_widget (self->camera_icon_size_group, w);
adw_action_row_add_prefix (ADW_ACTION_ROW (row), w);
adw_preferences_row_set_title (ADW_PREFERENCES_ROW (row),
g_app_info_get_name (G_APP_INFO (app_info)));
w = gtk_switch_new ();
gtk_switch_set_active (GTK_SWITCH (w), enabled);
gtk_widget_set_valign (w, GTK_ALIGN_CENTER);
adw_action_row_add_suffix (ADW_ACTION_ROW (row), w);
g_settings_bind (self->privacy_settings,
"disable-camera",
w,
"sensitive",
G_SETTINGS_BIND_INVERT_BOOLEAN);
g_hash_table_insert (self->camera_app_switches,
g_strdup (app_id),
g_object_ref (w));
data = g_slice_new (CameraAppStateData);
data->self = self;
data->app_id = g_strdup (app_id);
data->widget = w;
data->changing_state = FALSE;
g_signal_connect_data (G_OBJECT (w),
"state-set",
G_CALLBACK (on_camera_app_state_set),
data,
(GClosureNotify) camera_app_state_data_free,
0);
}
/* Steals permissions and permissions_data references */
static gboolean
to_child_name (GBinding *binding,
const GValue *from,
GValue *to,
gpointer user_data)
{
if (g_value_get_boolean (from))
g_value_set_string (to, "content");
else
g_value_set_string (to, "empty");
return TRUE;
}
static void
update_perm_store (CcCameraPanel *self,
GVariant *permissions,
GVariant *permissions_data)
{
GVariantIter iter;
const gchar *key;
gchar **value;
g_clear_pointer (&self->camera_apps_perms, g_variant_unref);
self->camera_apps_perms = permissions;
g_clear_pointer (&self->camera_apps_data, g_variant_unref);
self->camera_apps_data = permissions_data;
g_variant_iter_init (&iter, permissions);
while (g_variant_iter_loop (&iter, "{&s^a&s}", &key, &value))
{
gboolean enabled;
if (g_strv_length (value) != 1)
{
g_debug ("Permissions for %s in incorrect format, ignoring..", key);
continue;
}
enabled = (g_strcmp0 (value[0], "no") != 0);
add_camera_app (self, key, enabled);
}
}
static void
on_perm_store_signal (GDBusProxy *proxy,
gchar *sender_name,
gchar *signal_name,
GVariant *parameters,
gpointer user_data)
{
GVariant *permissions, *permissions_data;
if (g_strcmp0 (signal_name, "Changed") != 0)
return;
permissions = g_variant_get_child_value (parameters, 4);
permissions_data = g_variant_get_child_value (parameters, 3);
update_perm_store (user_data, permissions, permissions_data);
}
static void
on_perm_store_lookup_done (GObject *source_object,
GAsyncResult *res,
gpointer user_data)
{
CcCameraPanel *self = user_data;
GVariant *ret, *permissions, *permissions_data;
g_autoptr(GError) error = NULL;
ret = g_dbus_proxy_call_finish (G_DBUS_PROXY (source_object),
res,
&error);
if (ret == NULL)
{
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
g_warning ("Failed fetch permissions from flatpak permission store: %s", error->message);
return;
}
permissions = g_variant_get_child_value (ret, 0);
permissions_data = g_variant_get_child_value (ret, 1);
update_perm_store (user_data, permissions, permissions_data);
g_signal_connect_object (source_object,
"g-signal",
G_CALLBACK (on_perm_store_signal),
self,
0);
}
static void
on_perm_store_ready (GObject *source_object,
GAsyncResult *res,
gpointer user_data)
{
CcCameraPanel *self;
g_autoptr(GVariant) params = NULL;
g_autoptr(GError) error = NULL;
GDBusProxy *proxy;
proxy = g_dbus_proxy_new_for_bus_finish (res, &error);
if (proxy == NULL)
{
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
g_warning ("Failed to connect to flatpak permission store: %s", error->message);
return;
}
self = user_data;
self->perm_store = proxy;
params = g_variant_new ("(ss)",
APP_PERMISSIONS_TABLE,
APP_PERMISSIONS_ID);
g_dbus_proxy_call (self->perm_store,
"Lookup",
params,
G_DBUS_CALL_FLAGS_NONE,
-1,
cc_panel_get_cancellable (CC_PANEL (self)),
on_perm_store_lookup_done,
self);
}
static void
cc_camera_panel_finalize (GObject *object)
{
CcCameraPanel *self = CC_CAMERA_PANEL (object);
g_clear_object (&self->privacy_settings);
g_clear_object (&self->perm_store);
g_clear_object (&self->camera_icon_size_group);
g_clear_pointer (&self->camera_apps_perms, g_variant_unref);
g_clear_pointer (&self->camera_apps_data, g_variant_unref);
g_clear_pointer (&self->camera_app_switches, g_hash_table_unref);
G_OBJECT_CLASS (cc_camera_panel_parent_class)->finalize (object);
}
static const char *
cc_camera_panel_get_help_uri (CcPanel *panel)
{
return "help:gnome-help/camera";
}
static void
cc_camera_panel_class_init (CcCameraPanelClass *klass)
{
CcPanelClass *panel_class = CC_PANEL_CLASS (klass);
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
GObjectClass *object_class = G_OBJECT_CLASS (klass);
panel_class->get_help_uri = cc_camera_panel_get_help_uri;
object_class->finalize = cc_camera_panel_finalize;
gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/control-center/camera/cc-camera-panel.ui");
gtk_widget_class_bind_template_child (widget_class, CcCameraPanel, stack);
gtk_widget_class_bind_template_child (widget_class, CcCameraPanel, camera_apps_list_box);
gtk_widget_class_bind_template_child (widget_class, CcCameraPanel, main_switch);
}
static void
cc_camera_panel_init (CcCameraPanel *self)
{
g_resources_register (cc_camera_get_resource ());
gtk_widget_init_template (GTK_WIDGET (self));
self->camera_icon_size_group = gtk_size_group_new (GTK_SIZE_GROUP_BOTH);
self->privacy_settings = g_settings_new ("org.gnome.desktop.privacy");
g_settings_bind (self->privacy_settings, "disable-camera",
self->main_switch, "active",
G_SETTINGS_BIND_INVERT_BOOLEAN);
g_object_bind_property_full (self->main_switch, "active",
self->stack, "visible-child-name",
G_BINDING_SYNC_CREATE,
to_child_name,
NULL,
NULL, NULL);
self->camera_app_switches = g_hash_table_new_full (g_str_hash,
g_str_equal,
g_free,
g_object_unref);
g_dbus_proxy_new_for_bus (G_BUS_TYPE_SESSION,
G_DBUS_PROXY_FLAGS_NONE,
NULL,
"org.freedesktop.impl.portal.PermissionStore",
"/org/freedesktop/impl/portal/PermissionStore",
"org.freedesktop.impl.portal.PermissionStore",
cc_panel_get_cancellable (CC_PANEL (self)),
on_perm_store_ready,
self);
}