1395 lines
42 KiB
C
1395 lines
42 KiB
C
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
|
|
*
|
|
* Copyright (C) 2012 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 <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Author: Matthias Clasen <mclasen@redhat.com>
|
|
*/
|
|
|
|
#include "shell/cc-object-storage.h"
|
|
#include "list-box-helper.h"
|
|
#include "cc-privacy-panel.h"
|
|
#include "cc-privacy-resources.h"
|
|
#include "cc-util.h"
|
|
|
|
#include <gio/gdesktopappinfo.h>
|
|
#include <glib/gi18n.h>
|
|
|
|
CC_PANEL_REGISTER (CcPrivacyPanel, cc_privacy_panel)
|
|
|
|
#define WID(s) GTK_WIDGET (gtk_builder_get_object (self->priv->builder, s))
|
|
|
|
#define REMEMBER_RECENT_FILES "remember-recent-files"
|
|
#define RECENT_FILES_MAX_AGE "recent-files-max-age"
|
|
#define REMOVE_OLD_TRASH_FILES "remove-old-trash-files"
|
|
#define REMOVE_OLD_TEMP_FILES "remove-old-temp-files"
|
|
#define OLD_FILES_AGE "old-files-age"
|
|
#define SEND_SOFTWARE_USAGE_STATS "send-software-usage-stats"
|
|
#define REPORT_TECHNICAL_PROBLEMS "report-technical-problems"
|
|
#define LOCATION_ENABLED "enabled"
|
|
|
|
#define APP_PERMISSIONS_TABLE "gnome"
|
|
#define APP_PERMISSIONS_ID "geolocation"
|
|
|
|
struct _CcPrivacyPanelPrivate
|
|
{
|
|
GtkBuilder *builder;
|
|
GtkWidget *recent_dialog;
|
|
GtkWidget *screen_lock_dialog;
|
|
GtkWidget *location_dialog;
|
|
GtkWidget *location_label;
|
|
GtkWidget *trash_dialog;
|
|
GtkWidget *software_dialog;
|
|
GtkWidget *list_box;
|
|
GtkWidget *location_apps_list_box;
|
|
GtkWidget *location_apps_label;
|
|
GtkWidget *location_apps_frame;
|
|
|
|
GSettings *lockdown_settings;
|
|
GSettings *lock_settings;
|
|
GSettings *privacy_settings;
|
|
GSettings *notification_settings;
|
|
GSettings *location_settings;
|
|
|
|
GtkWidget *abrt_dialog;
|
|
GtkWidget *abrt_row;
|
|
guint abrt_watch_id;
|
|
|
|
GCancellable *cancellable;
|
|
|
|
GDBusProxy *gclue_manager;
|
|
GDBusProxy *perm_store;
|
|
GVariant *location_apps_perms;
|
|
GVariant *location_apps_data;
|
|
GHashTable *location_app_switches;
|
|
|
|
GtkSizeGroup *location_icon_size_group;
|
|
};
|
|
|
|
static char *
|
|
get_os_name (void)
|
|
{
|
|
char *buffer;
|
|
char *name;
|
|
|
|
name = NULL;
|
|
|
|
if (g_file_get_contents ("/etc/os-release", &buffer, NULL, NULL))
|
|
{
|
|
char *start, *end;
|
|
|
|
start = end = NULL;
|
|
if ((start = strstr (buffer, "NAME=")) != NULL)
|
|
{
|
|
start += strlen ("NAME=");
|
|
end = strchr (start, '\n');
|
|
}
|
|
|
|
if (start != NULL && end != NULL)
|
|
{
|
|
name = g_strndup (start, end - start);
|
|
}
|
|
|
|
g_free (buffer);
|
|
}
|
|
|
|
if (name && *name != '\0')
|
|
{
|
|
char *tmp;
|
|
tmp = g_shell_unquote (name, NULL);
|
|
g_free (name);
|
|
name = tmp;
|
|
}
|
|
|
|
if (name == NULL)
|
|
name = g_strdup ("GNOME");
|
|
|
|
return name;
|
|
}
|
|
|
|
static char *
|
|
get_privacy_policy_url (void)
|
|
{
|
|
char *buffer;
|
|
char *url;
|
|
|
|
url = NULL;
|
|
|
|
if (g_file_get_contents ("/etc/os-release", &buffer, NULL, NULL))
|
|
{
|
|
char *start, *end;
|
|
|
|
start = end = NULL;
|
|
if ((start = strstr (buffer, "PRIVACY_POLICY_URL=")) != NULL)
|
|
{
|
|
start += strlen ("PRIVACY_POLICY_URL=");
|
|
end = strchr (start, '\n');
|
|
}
|
|
|
|
if (start != NULL && end != NULL)
|
|
{
|
|
url = g_strndup (start, end - start);
|
|
}
|
|
|
|
g_free (buffer);
|
|
}
|
|
|
|
if (url && *url != '\0')
|
|
{
|
|
char *tmp;
|
|
tmp = g_shell_unquote (url, NULL);
|
|
g_free (url);
|
|
url = tmp;
|
|
}
|
|
|
|
if (url == NULL)
|
|
url = g_strdup ("http://www.gnome.org/privacy-policy");
|
|
|
|
return url;
|
|
}
|
|
|
|
static void
|
|
update_lock_screen_sensitivity (CcPrivacyPanel *self)
|
|
{
|
|
GtkWidget *widget;
|
|
gboolean locked;
|
|
|
|
locked = g_settings_get_boolean (self->priv->lockdown_settings, "disable-lock-screen");
|
|
|
|
widget = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "screen_lock_dialog_grid"));
|
|
gtk_widget_set_sensitive (widget, !locked);
|
|
}
|
|
|
|
static void
|
|
on_lockdown_settings_changed (GSettings *settings,
|
|
const char *key,
|
|
CcPrivacyPanel *panel)
|
|
{
|
|
if (g_str_equal (key, "disable-lock-screen") == FALSE)
|
|
return;
|
|
|
|
update_lock_screen_sensitivity (panel);
|
|
}
|
|
|
|
static gboolean
|
|
on_off_label_mapping_get (GValue *value,
|
|
GVariant *variant,
|
|
gpointer user_data)
|
|
{
|
|
g_value_set_string (value, g_variant_get_boolean (variant) ? _("On") : _("Off"));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GtkWidget *
|
|
get_on_off_label (GSettings *settings,
|
|
const gchar *key)
|
|
{
|
|
GtkWidget *w;
|
|
|
|
w = gtk_label_new ("");
|
|
g_settings_bind_with_mapping (settings, key,
|
|
w, "label",
|
|
G_SETTINGS_BIND_GET,
|
|
on_off_label_mapping_get,
|
|
NULL,
|
|
NULL,
|
|
NULL);
|
|
return w;
|
|
}
|
|
|
|
static gboolean
|
|
abrt_label_mapping_get (GValue *value,
|
|
GVariant *variant,
|
|
gpointer user_data)
|
|
{
|
|
g_value_set_string (value, g_variant_get_boolean (variant) ? _("Automatic") : _("Manual"));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GtkWidget *
|
|
get_abrt_label (GSettings *settings,
|
|
const gchar *key)
|
|
{
|
|
GtkWidget *w;
|
|
|
|
w = gtk_label_new ("");
|
|
g_settings_bind_with_mapping (settings, key,
|
|
w, "label",
|
|
G_SETTINGS_BIND_GET,
|
|
abrt_label_mapping_get,
|
|
NULL,
|
|
NULL,
|
|
NULL);
|
|
return w;
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
GtkWidget *label;
|
|
const gchar *key1;
|
|
const gchar *key2;
|
|
} Label2Data;
|
|
|
|
static void
|
|
set_on_off_label2 (GSettings *settings,
|
|
const gchar *key,
|
|
gpointer user_data)
|
|
{
|
|
Label2Data *data = user_data;
|
|
gboolean v1, v2;
|
|
|
|
v1 = g_settings_get_boolean (settings, data->key1);
|
|
v2 = g_settings_get_boolean (settings, data->key2);
|
|
|
|
gtk_label_set_label (GTK_LABEL (data->label), (v1 || v2) ? _("On") : _("Off"));
|
|
}
|
|
|
|
static GtkWidget *
|
|
get_on_off_label2 (GSettings *settings,
|
|
const gchar *key1,
|
|
const gchar *key2)
|
|
{
|
|
Label2Data *data;
|
|
|
|
data = g_new (Label2Data, 1);
|
|
data->label = gtk_label_new ("");
|
|
data->key1 = g_strdup (key1);
|
|
data->key2 = g_strdup (key2);
|
|
|
|
g_signal_connect (settings, "changed",
|
|
G_CALLBACK (set_on_off_label2), data);
|
|
|
|
set_on_off_label2 (settings, key1, data);
|
|
|
|
return data->label;
|
|
}
|
|
|
|
static GtkWidget *
|
|
add_row (CcPrivacyPanel *self,
|
|
const gchar *label,
|
|
const gchar *dialog_id,
|
|
GtkWidget *status)
|
|
{
|
|
GtkWidget *box, *row, *w;
|
|
|
|
row = gtk_list_box_row_new ();
|
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 50);
|
|
gtk_container_add (GTK_CONTAINER (row), box);
|
|
g_object_set_data (G_OBJECT (row), "dialog-id", (gpointer)dialog_id);
|
|
gtk_widget_set_hexpand (box, TRUE);
|
|
gtk_container_add (GTK_CONTAINER (self->priv->list_box), row);
|
|
|
|
w = gtk_label_new (label);
|
|
gtk_widget_set_halign (w, GTK_ALIGN_START);
|
|
gtk_widget_set_margin_start (w, 20);
|
|
gtk_widget_set_margin_end (w, 20);
|
|
gtk_widget_set_margin_top (w, 18);
|
|
gtk_widget_set_margin_bottom (w, 18);
|
|
gtk_widget_set_halign (w, GTK_ALIGN_START);
|
|
gtk_widget_set_valign (w, GTK_ALIGN_CENTER);
|
|
gtk_widget_set_hexpand (w, TRUE);
|
|
gtk_box_pack_start (GTK_BOX (box), w, TRUE, TRUE, 0);
|
|
gtk_widget_set_margin_start (status, 20);
|
|
gtk_widget_set_margin_end (status, 20);
|
|
gtk_widget_set_halign (status, GTK_ALIGN_END);
|
|
gtk_widget_set_valign (status, GTK_ALIGN_CENTER);
|
|
gtk_box_pack_end (GTK_BOX (box), status, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show_all (row);
|
|
|
|
return row;
|
|
}
|
|
|
|
static void
|
|
lock_combo_changed_cb (GtkWidget *widget,
|
|
CcPrivacyPanel *self)
|
|
{
|
|
GtkTreeIter iter;
|
|
GtkTreeModel *model;
|
|
guint delay;
|
|
gboolean ret;
|
|
|
|
/* no selection */
|
|
ret = gtk_combo_box_get_active_iter (GTK_COMBO_BOX(widget), &iter);
|
|
if (!ret)
|
|
return;
|
|
|
|
/* get entry */
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX(widget));
|
|
gtk_tree_model_get (model, &iter,
|
|
1, &delay,
|
|
-1);
|
|
g_settings_set (self->priv->lock_settings, "lock-delay", "u", delay);
|
|
}
|
|
|
|
static void
|
|
set_lock_value_for_combo (GtkComboBox *combo_box,
|
|
CcPrivacyPanel *self)
|
|
{
|
|
GtkTreeIter iter;
|
|
GtkTreeModel *model;
|
|
guint value;
|
|
gint value_tmp, value_prev;
|
|
gboolean ret;
|
|
guint i;
|
|
|
|
/* get entry */
|
|
model = gtk_combo_box_get_model (combo_box);
|
|
ret = gtk_tree_model_get_iter_first (model, &iter);
|
|
if (!ret)
|
|
return;
|
|
|
|
value_prev = 0;
|
|
i = 0;
|
|
|
|
/* try to make the UI match the lock setting */
|
|
g_settings_get (self->priv->lock_settings, "lock-delay", "u", &value);
|
|
do
|
|
{
|
|
gtk_tree_model_get (model, &iter,
|
|
1, &value_tmp,
|
|
-1);
|
|
if (value == value_tmp ||
|
|
(value_tmp > value_prev && value < value_tmp))
|
|
{
|
|
gtk_combo_box_set_active_iter (combo_box, &iter);
|
|
return;
|
|
}
|
|
value_prev = value_tmp;
|
|
i++;
|
|
} while (gtk_tree_model_iter_next (model, &iter));
|
|
|
|
/* If we didn't find the setting in the list */
|
|
gtk_combo_box_set_active (combo_box, i - 1);
|
|
}
|
|
|
|
static void
|
|
add_screen_lock (CcPrivacyPanel *self)
|
|
{
|
|
GtkWidget *w;
|
|
GtkWidget *dialog;
|
|
GtkWidget *label;
|
|
|
|
w = get_on_off_label (self->priv->lock_settings, "lock-enabled");
|
|
add_row (self, _("Screen Lock"), "screen_lock_dialog", w);
|
|
|
|
dialog = self->priv->screen_lock_dialog;
|
|
g_signal_connect (dialog, "delete-event",
|
|
G_CALLBACK (gtk_widget_hide_on_delete), NULL);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "automatic_screen_lock"));
|
|
g_settings_bind (self->priv->lock_settings, "lock-enabled",
|
|
w, "active",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "lock_after_combo"));
|
|
g_settings_bind (self->priv->lock_settings, "lock-enabled",
|
|
w, "sensitive",
|
|
G_SETTINGS_BIND_GET);
|
|
|
|
label = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "lock_after_label"));
|
|
|
|
g_object_bind_property (w, "sensitive", label, "sensitive", G_BINDING_DEFAULT);
|
|
|
|
set_lock_value_for_combo (GTK_COMBO_BOX (w), self);
|
|
g_signal_connect (w, "changed",
|
|
G_CALLBACK (lock_combo_changed_cb), self);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "show_notifications"));
|
|
g_settings_bind (self->priv->notification_settings, "show-in-lock-screen",
|
|
w, "active",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
}
|
|
|
|
static void
|
|
update_location_label (CcPrivacyPanel *self)
|
|
{
|
|
CcPrivacyPanelPrivate *priv = self->priv;
|
|
gboolean in_use = FALSE, on;
|
|
const gchar *label;
|
|
|
|
if (priv->gclue_manager != NULL)
|
|
{
|
|
GVariant *variant;
|
|
|
|
variant = g_dbus_proxy_get_cached_property (priv->gclue_manager, "InUse");
|
|
if (variant != NULL)
|
|
{
|
|
in_use = g_variant_get_boolean (variant);
|
|
g_variant_unref (variant);
|
|
}
|
|
}
|
|
|
|
if (in_use)
|
|
{
|
|
gtk_label_set_label (GTK_LABEL (priv->location_label), _("In use"));
|
|
return;
|
|
}
|
|
|
|
on = g_settings_get_boolean (priv->location_settings, LOCATION_ENABLED);
|
|
label = on ? C_("Location services status", "On") :
|
|
C_("Location services status", "Off");
|
|
gtk_label_set_label (GTK_LABEL (priv->location_label), label);
|
|
}
|
|
|
|
static void
|
|
on_gclue_manager_props_changed (GDBusProxy *manager,
|
|
GVariant *changed_properties,
|
|
GStrv invalidated_properties,
|
|
gpointer user_data)
|
|
{
|
|
update_location_label (user_data);
|
|
}
|
|
|
|
static void
|
|
on_gclue_manager_ready (GObject *source_object,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
CcPrivacyPanel *self;
|
|
GDBusProxy *proxy;
|
|
GError *error = NULL;
|
|
|
|
proxy = cc_object_storage_create_dbus_proxy_finish (res, &error);
|
|
if (proxy == NULL)
|
|
{
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
|
g_warning ("Failed to connect to Geoclue: %s", error->message);
|
|
g_error_free (error);
|
|
|
|
return;
|
|
}
|
|
self = user_data;
|
|
self->priv->gclue_manager = proxy;
|
|
|
|
g_signal_connect_object (self->priv->gclue_manager,
|
|
"g-properties-changed",
|
|
G_CALLBACK (on_gclue_manager_props_changed),
|
|
self,
|
|
0);
|
|
|
|
update_location_label (self);
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
CcPrivacyPanel *self;
|
|
GtkWidget *widget;
|
|
gchar *app_id;
|
|
gboolean changing_state;
|
|
gboolean pending_state;
|
|
} LocationAppStateData;
|
|
|
|
static void
|
|
location_app_state_data_free (LocationAppStateData *data)
|
|
{
|
|
g_free (data->app_id);
|
|
g_slice_free (LocationAppStateData, data);
|
|
}
|
|
|
|
static void
|
|
on_perm_store_set_done (GObject *source_object,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
LocationAppStateData *data;
|
|
GVariant *results;
|
|
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);
|
|
g_error_free (error);
|
|
|
|
return;
|
|
}
|
|
g_variant_unref (results);
|
|
|
|
data = (LocationAppStateData *) user_data;
|
|
data->changing_state = FALSE;
|
|
gtk_switch_set_state (GTK_SWITCH (data->widget), data->pending_state);
|
|
}
|
|
|
|
static gboolean
|
|
on_location_app_state_set (GtkSwitch *widget,
|
|
gboolean state,
|
|
gpointer user_data)
|
|
{
|
|
LocationAppStateData *data = (LocationAppStateData *) user_data;
|
|
CcPrivacyPanel *self = data->self;
|
|
GVariant *params;
|
|
GVariantIter iter;
|
|
gchar *key;
|
|
gchar **value;
|
|
GVariantBuilder builder;
|
|
|
|
if (data->changing_state)
|
|
return TRUE;
|
|
|
|
data->changing_state = TRUE;
|
|
data->pending_state = state;
|
|
|
|
g_variant_iter_init (&iter, self->priv->location_apps_perms);
|
|
g_variant_builder_init (&builder, G_VARIANT_TYPE_ARRAY);
|
|
while (g_variant_iter_loop (&iter, "{s^as}", &key, &value))
|
|
{
|
|
gchar *tmp = NULL;
|
|
|
|
if (g_strv_length (value) < 2)
|
|
/* It's OK to drop the entry if it's not in expected format */
|
|
continue;
|
|
|
|
if (g_strcmp0 (data->app_id, key) == 0)
|
|
{
|
|
tmp = value[0];
|
|
value[0] = state ? "EXACT" : "NONE";
|
|
}
|
|
|
|
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->priv->location_apps_data);
|
|
|
|
g_dbus_proxy_call (self->priv->perm_store,
|
|
"Set",
|
|
params,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
self->priv->cancellable,
|
|
on_perm_store_set_done,
|
|
data);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
add_location_app (CcPrivacyPanel *self,
|
|
const gchar *app_id,
|
|
gboolean enabled,
|
|
gint64 last_used)
|
|
{
|
|
CcPrivacyPanelPrivate *priv = self->priv;
|
|
GDesktopAppInfo *app_info;
|
|
char *desktop_id;
|
|
GtkWidget *box, *row, *w;
|
|
GIcon *icon;
|
|
GDateTime *t;
|
|
char *last_used_str;
|
|
LocationAppStateData *data;
|
|
|
|
w = g_hash_table_lookup (priv->location_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);
|
|
g_free (desktop_id);
|
|
if (app_info == NULL)
|
|
return;
|
|
|
|
row = gtk_list_box_row_new ();
|
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_margin_start (box, 12);
|
|
gtk_widget_set_margin_end (box, 6);
|
|
gtk_widget_set_margin_top (box, 12);
|
|
gtk_widget_set_margin_bottom (box, 12);
|
|
gtk_container_add (GTK_CONTAINER (row), box);
|
|
gtk_widget_set_hexpand (box, TRUE);
|
|
gtk_container_add (GTK_CONTAINER (priv->location_apps_list_box), row);
|
|
|
|
icon = g_app_info_get_icon (G_APP_INFO (app_info));
|
|
w = gtk_image_new_from_gicon (icon, GTK_ICON_SIZE_LARGE_TOOLBAR);
|
|
gtk_widget_set_halign (w, GTK_ALIGN_CENTER);
|
|
gtk_widget_set_valign (w, GTK_ALIGN_CENTER);
|
|
gtk_size_group_add_widget (priv->location_icon_size_group, w);
|
|
gtk_box_pack_start (GTK_BOX (box), w, FALSE, FALSE, 0);
|
|
|
|
w = gtk_label_new (g_app_info_get_name (G_APP_INFO (app_info)));
|
|
gtk_widget_set_margin_start (w, 12);
|
|
gtk_widget_set_margin_end (w, 12);
|
|
gtk_widget_set_halign (w, GTK_ALIGN_START);
|
|
gtk_widget_set_valign (w, GTK_ALIGN_CENTER);
|
|
gtk_label_set_xalign (GTK_LABEL (w), 0);
|
|
gtk_box_pack_start (GTK_BOX (box), w, FALSE, FALSE, 0);
|
|
|
|
t = g_date_time_new_from_unix_utc (last_used);
|
|
last_used_str = cc_util_get_smart_date (t);
|
|
w = gtk_label_new (last_used_str);
|
|
g_free (last_used_str);
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (w), "dim-label");
|
|
gtk_widget_set_margin_start (w, 12);
|
|
gtk_widget_set_margin_end (w, 12);
|
|
gtk_widget_set_halign (w, GTK_ALIGN_END);
|
|
gtk_widget_set_valign (w, GTK_ALIGN_CENTER);
|
|
gtk_box_pack_start (GTK_BOX (box), w, TRUE, TRUE, 0);
|
|
|
|
w = gtk_switch_new ();
|
|
gtk_switch_set_active (GTK_SWITCH (w), enabled);
|
|
gtk_widget_set_halign (w, GTK_ALIGN_END);
|
|
gtk_widget_set_valign (w, GTK_ALIGN_CENTER);
|
|
gtk_box_pack_start (GTK_BOX (box), w, FALSE, FALSE, 0);
|
|
g_settings_bind (priv->location_settings, LOCATION_ENABLED,
|
|
w, "sensitive",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
g_hash_table_insert (priv->location_app_switches,
|
|
g_strdup (app_id),
|
|
g_object_ref (w));
|
|
|
|
data = g_slice_new (LocationAppStateData);
|
|
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_location_app_state_set),
|
|
data,
|
|
(GClosureNotify) location_app_state_data_free,
|
|
0);
|
|
|
|
gtk_widget_show_all (row);
|
|
}
|
|
|
|
/* Steals permissions and permissions_data references */
|
|
static void
|
|
update_perm_store (CcPrivacyPanel *self,
|
|
GVariant *permissions,
|
|
GVariant *permissions_data)
|
|
{
|
|
CcPrivacyPanelPrivate *priv;
|
|
GVariantIter iter;
|
|
gchar *key;
|
|
gchar **value;
|
|
GList *children;
|
|
|
|
priv = self->priv;
|
|
|
|
g_clear_pointer (&priv->location_apps_perms, g_variant_unref);
|
|
priv->location_apps_perms = permissions;
|
|
g_clear_pointer (&priv->location_apps_data, g_variant_unref);
|
|
priv->location_apps_data = permissions_data;
|
|
|
|
g_variant_iter_init (&iter, permissions);
|
|
while (g_variant_iter_loop (&iter, "{s^as}", &key, &value))
|
|
{
|
|
gboolean enabled;
|
|
gint64 last_used;
|
|
|
|
if (g_strv_length (value) < 2)
|
|
{
|
|
g_debug ("Permissions for %s in incorrect format, ignoring..", key);
|
|
continue;
|
|
}
|
|
|
|
enabled = (g_strcmp0 (value[0], "NONE") != 0);
|
|
last_used = g_ascii_strtoll (value[1], NULL, 10);
|
|
|
|
add_location_app (self, key, enabled, last_used);
|
|
}
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (priv->location_apps_list_box));
|
|
if (g_list_length (children) > 0)
|
|
{
|
|
gtk_widget_set_visible (priv->location_apps_label, TRUE);
|
|
gtk_widget_set_visible (priv->location_apps_frame, TRUE);
|
|
}
|
|
g_list_free (children);
|
|
}
|
|
|
|
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)
|
|
{
|
|
GVariant *ret, *permissions, *permissions_data;
|
|
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);
|
|
g_error_free (error);
|
|
|
|
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),
|
|
user_data,
|
|
0);
|
|
}
|
|
|
|
static void
|
|
on_perm_store_ready (GObject *source_object,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
CcPrivacyPanel *self;
|
|
GDBusProxy *proxy;
|
|
GVariant *params;
|
|
GError *error = NULL;
|
|
|
|
proxy = cc_object_storage_create_dbus_proxy_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);
|
|
g_error_free (error);
|
|
|
|
return;
|
|
}
|
|
self = user_data;
|
|
self->priv->perm_store = proxy;
|
|
|
|
params = g_variant_new ("(ss)",
|
|
APP_PERMISSIONS_TABLE,
|
|
APP_PERMISSIONS_ID);
|
|
g_dbus_proxy_call (self->priv->perm_store,
|
|
"Lookup",
|
|
params,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
self->priv->cancellable,
|
|
on_perm_store_lookup_done,
|
|
self);
|
|
}
|
|
|
|
static void
|
|
add_location (CcPrivacyPanel *self)
|
|
{
|
|
CcPrivacyPanelPrivate *priv = self->priv;
|
|
GtkWidget *w;
|
|
GtkWidget *dialog;
|
|
|
|
priv->location_label = gtk_label_new ("");
|
|
update_location_label (self);
|
|
|
|
add_row (self,
|
|
_("Location Services"),
|
|
"location_dialog",
|
|
priv->location_label);
|
|
|
|
dialog = priv->location_dialog;
|
|
g_signal_connect (dialog, "delete-event",
|
|
G_CALLBACK (gtk_widget_hide_on_delete), NULL);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (priv->builder, "location_services_switch"));
|
|
g_settings_bind (priv->location_settings, LOCATION_ENABLED,
|
|
w, "active",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
|
|
g_signal_connect_object (priv->location_settings, "changed::" LOCATION_ENABLED,
|
|
G_CALLBACK (update_location_label), self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
priv->location_app_switches = g_hash_table_new_full (g_str_hash,
|
|
g_str_equal,
|
|
g_free,
|
|
g_object_unref);
|
|
|
|
cc_object_storage_create_dbus_proxy (G_BUS_TYPE_SYSTEM,
|
|
G_DBUS_PROXY_FLAGS_NONE,
|
|
"org.freedesktop.GeoClue2",
|
|
"/org/freedesktop/GeoClue2/Manager",
|
|
"org.freedesktop.GeoClue2.Manager",
|
|
priv->cancellable,
|
|
on_gclue_manager_ready,
|
|
self);
|
|
|
|
cc_object_storage_create_dbus_proxy (G_BUS_TYPE_SESSION,
|
|
G_DBUS_PROXY_FLAGS_NONE,
|
|
"org.freedesktop.impl.portal.PermissionStore",
|
|
"/org/freedesktop/impl/portal/PermissionStore",
|
|
"org.freedesktop.impl.portal.PermissionStore",
|
|
priv->cancellable,
|
|
on_perm_store_ready,
|
|
self);
|
|
}
|
|
|
|
static void
|
|
retain_history_combo_changed_cb (GtkWidget *widget,
|
|
CcPrivacyPanel *self)
|
|
{
|
|
GtkTreeIter iter;
|
|
GtkTreeModel *model;
|
|
gint value;
|
|
gboolean ret;
|
|
|
|
/* no selection */
|
|
ret = gtk_combo_box_get_active_iter (GTK_COMBO_BOX(widget), &iter);
|
|
if (!ret)
|
|
return;
|
|
|
|
/* get entry */
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX(widget));
|
|
gtk_tree_model_get (model, &iter,
|
|
1, &value,
|
|
-1);
|
|
g_settings_set (self->priv->privacy_settings, RECENT_FILES_MAX_AGE, "i", value);
|
|
}
|
|
|
|
static void
|
|
set_retain_history_value_for_combo (GtkComboBox *combo_box,
|
|
CcPrivacyPanel *self)
|
|
{
|
|
GtkTreeIter iter;
|
|
GtkTreeModel *model;
|
|
gint value;
|
|
gint value_tmp, value_prev;
|
|
gboolean ret;
|
|
guint i;
|
|
|
|
/* get entry */
|
|
model = gtk_combo_box_get_model (combo_box);
|
|
ret = gtk_tree_model_get_iter_first (model, &iter);
|
|
if (!ret)
|
|
return;
|
|
|
|
value_prev = 0;
|
|
i = 0;
|
|
|
|
/* try to make the UI match the setting */
|
|
g_settings_get (self->priv->privacy_settings, RECENT_FILES_MAX_AGE, "i", &value);
|
|
do
|
|
{
|
|
gtk_tree_model_get (model, &iter,
|
|
1, &value_tmp,
|
|
-1);
|
|
if (value == value_tmp ||
|
|
(value > 0 && value_tmp > value_prev && value < value_tmp))
|
|
{
|
|
gtk_combo_box_set_active_iter (combo_box, &iter);
|
|
return;
|
|
}
|
|
value_prev = value_tmp;
|
|
i++;
|
|
} while (gtk_tree_model_iter_next (model, &iter));
|
|
|
|
/* If we didn't find the setting in the list */
|
|
gtk_combo_box_set_active (combo_box, i - 1);
|
|
}
|
|
|
|
static void
|
|
clear_recent (CcPrivacyPanel *self)
|
|
{
|
|
GtkRecentManager *m;
|
|
|
|
m = gtk_recent_manager_get_default ();
|
|
gtk_recent_manager_purge_items (m, NULL);
|
|
}
|
|
|
|
static void
|
|
add_usage_history (CcPrivacyPanel *self)
|
|
{
|
|
GtkWidget *w;
|
|
GtkWidget *dialog;
|
|
GtkWidget *label;
|
|
|
|
w = get_on_off_label (self->priv->privacy_settings, REMEMBER_RECENT_FILES);
|
|
add_row (self, _("Usage & History"), "recent_dialog", w);
|
|
|
|
dialog = self->priv->recent_dialog;
|
|
g_signal_connect (dialog, "delete-event",
|
|
G_CALLBACK (gtk_widget_hide_on_delete), NULL);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "recently_used_switch"));
|
|
g_settings_bind (self->priv->privacy_settings, REMEMBER_RECENT_FILES,
|
|
w, "active",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "retain_history_combo"));
|
|
set_retain_history_value_for_combo (GTK_COMBO_BOX (w), self);
|
|
g_signal_connect (w, "changed",
|
|
G_CALLBACK (retain_history_combo_changed_cb), self);
|
|
|
|
g_settings_bind (self->priv->privacy_settings, REMEMBER_RECENT_FILES,
|
|
w, "sensitive",
|
|
G_SETTINGS_BIND_GET);
|
|
|
|
label = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "retain_history_label"));
|
|
|
|
g_object_bind_property (w, "sensitive", label, "sensitive", G_BINDING_DEFAULT);
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "clear_recent_button"));
|
|
g_signal_connect_swapped (w, "clicked",
|
|
G_CALLBACK (clear_recent), self);
|
|
|
|
}
|
|
|
|
static void
|
|
purge_after_combo_changed_cb (GtkWidget *widget,
|
|
CcPrivacyPanel *self)
|
|
{
|
|
GtkTreeIter iter;
|
|
GtkTreeModel *model;
|
|
guint value;
|
|
gboolean ret;
|
|
|
|
/* no selection */
|
|
ret = gtk_combo_box_get_active_iter (GTK_COMBO_BOX(widget), &iter);
|
|
if (!ret)
|
|
return;
|
|
|
|
/* get entry */
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX(widget));
|
|
gtk_tree_model_get (model, &iter,
|
|
1, &value,
|
|
-1);
|
|
g_settings_set (self->priv->privacy_settings, OLD_FILES_AGE, "u", value);
|
|
}
|
|
|
|
static void
|
|
set_purge_after_value_for_combo (GtkComboBox *combo_box,
|
|
CcPrivacyPanel *self)
|
|
{
|
|
GtkTreeIter iter;
|
|
GtkTreeModel *model;
|
|
guint value;
|
|
gint value_tmp, value_prev;
|
|
gboolean ret;
|
|
guint i;
|
|
|
|
/* get entry */
|
|
model = gtk_combo_box_get_model (combo_box);
|
|
ret = gtk_tree_model_get_iter_first (model, &iter);
|
|
if (!ret)
|
|
return;
|
|
|
|
value_prev = 0;
|
|
i = 0;
|
|
|
|
/* try to make the UI match the purge setting */
|
|
g_settings_get (self->priv->privacy_settings, OLD_FILES_AGE, "u", &value);
|
|
do
|
|
{
|
|
gtk_tree_model_get (model, &iter,
|
|
1, &value_tmp,
|
|
-1);
|
|
if (value == value_tmp ||
|
|
(value_tmp > value_prev && value < value_tmp))
|
|
{
|
|
gtk_combo_box_set_active_iter (combo_box, &iter);
|
|
return;
|
|
}
|
|
value_prev = value_tmp;
|
|
i++;
|
|
} while (gtk_tree_model_iter_next (model, &iter));
|
|
|
|
/* If we didn't find the setting in the list */
|
|
gtk_combo_box_set_active (combo_box, i - 1);
|
|
}
|
|
|
|
static gboolean
|
|
run_warning (GtkWindow *parent, char *prompt, char *text, char *button_title)
|
|
{
|
|
GtkWidget *dialog;
|
|
GtkWidget *button;
|
|
int result;
|
|
dialog = gtk_message_dialog_new (parent,
|
|
0,
|
|
GTK_MESSAGE_WARNING,
|
|
GTK_BUTTONS_NONE,
|
|
NULL);
|
|
g_object_set (dialog,
|
|
"text", prompt,
|
|
"secondary-text", text,
|
|
NULL);
|
|
gtk_dialog_add_button (GTK_DIALOG (dialog), _("_Cancel"), GTK_RESPONSE_CANCEL);
|
|
gtk_dialog_add_button (GTK_DIALOG (dialog), button_title, GTK_RESPONSE_OK);
|
|
|
|
gtk_dialog_set_default_response (GTK_DIALOG (dialog), FALSE);
|
|
|
|
button = gtk_dialog_get_widget_for_response (GTK_DIALOG (dialog), GTK_RESPONSE_OK);
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (button), "destructive-action");
|
|
|
|
result = gtk_dialog_run (GTK_DIALOG (dialog));
|
|
gtk_widget_destroy (dialog);
|
|
|
|
return result == GTK_RESPONSE_OK;
|
|
}
|
|
|
|
static void
|
|
empty_trash (CcPrivacyPanel *self)
|
|
{
|
|
GDBusConnection *bus;
|
|
gboolean result;
|
|
GtkWidget *dialog;
|
|
|
|
dialog = WID ("trash_dialog");
|
|
result = run_warning (GTK_WINDOW (dialog), _("Empty all items from Trash?"),
|
|
_("All items in the Trash will be permanently deleted."),
|
|
_("_Empty Trash"));
|
|
|
|
if (!result)
|
|
return;
|
|
|
|
bus = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, NULL);
|
|
g_dbus_connection_call (bus,
|
|
"org.gnome.SettingsDaemon.Housekeeping",
|
|
"/org/gnome/SettingsDaemon/Housekeeping",
|
|
"org.gnome.SettingsDaemon.Housekeeping",
|
|
"EmptyTrash",
|
|
NULL, NULL, 0, -1, NULL, NULL, NULL);
|
|
g_object_unref (bus);
|
|
}
|
|
|
|
static void
|
|
purge_temp (CcPrivacyPanel *self)
|
|
{
|
|
GDBusConnection *bus;
|
|
gboolean result;
|
|
GtkWidget *dialog;
|
|
|
|
dialog = WID ("trash_dialog");
|
|
result = run_warning (GTK_WINDOW (dialog), _("Delete all the temporary files?"),
|
|
_("All the temporary files will be permanently deleted."),
|
|
_("_Purge Temporary Files"));
|
|
|
|
if (!result)
|
|
return;
|
|
|
|
bus = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, NULL);
|
|
g_dbus_connection_call (bus,
|
|
"org.gnome.SettingsDaemon.Housekeeping",
|
|
"/org/gnome/SettingsDaemon/Housekeeping",
|
|
"org.gnome.SettingsDaemon.Housekeeping",
|
|
"RemoveTempFiles",
|
|
NULL, NULL, 0, -1, NULL, NULL, NULL);
|
|
g_object_unref (bus);
|
|
}
|
|
|
|
static void
|
|
add_trash_temp (CcPrivacyPanel *self)
|
|
{
|
|
GtkWidget *w;
|
|
GtkWidget *dialog;
|
|
|
|
w = get_on_off_label2 (self->priv->privacy_settings, REMOVE_OLD_TRASH_FILES, REMOVE_OLD_TEMP_FILES);
|
|
add_row (self, _("Purge Trash & Temporary Files"), "trash_dialog", w);
|
|
|
|
dialog = self->priv->trash_dialog;
|
|
g_signal_connect (dialog, "delete-event",
|
|
G_CALLBACK (gtk_widget_hide_on_delete), NULL);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "purge_trash_switch"));
|
|
g_settings_bind (self->priv->privacy_settings, REMOVE_OLD_TRASH_FILES,
|
|
w, "active",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "purge_temp_switch"));
|
|
g_settings_bind (self->priv->privacy_settings, REMOVE_OLD_TEMP_FILES,
|
|
w, "active",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "purge_after_combo"));
|
|
|
|
set_purge_after_value_for_combo (GTK_COMBO_BOX (w), self);
|
|
g_signal_connect (w, "changed",
|
|
G_CALLBACK (purge_after_combo_changed_cb), self);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "empty_trash_button"));
|
|
g_signal_connect_swapped (w, "clicked", G_CALLBACK (empty_trash), self);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "purge_temp_button"));
|
|
g_signal_connect_swapped (w, "clicked", G_CALLBACK (purge_temp), self);
|
|
}
|
|
|
|
static void
|
|
add_software (CcPrivacyPanel *self)
|
|
{
|
|
GtkWidget *w;
|
|
GtkWidget *dialog;
|
|
|
|
/* disable until all the points on the bug are fixed:
|
|
* https://bugzilla.gnome.org/show_bug.cgi?id=726234 */
|
|
return;
|
|
|
|
w = get_on_off_label (self->priv->privacy_settings, SEND_SOFTWARE_USAGE_STATS);
|
|
add_row (self, _("Software Usage"), "software_dialog", w);
|
|
|
|
dialog = self->priv->software_dialog;
|
|
g_signal_connect (dialog, "delete-event",
|
|
G_CALLBACK (gtk_widget_hide_on_delete), NULL);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "software_usage_switch"));
|
|
g_settings_bind (self->priv->privacy_settings, SEND_SOFTWARE_USAGE_STATS,
|
|
w, "active",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
}
|
|
|
|
static void
|
|
abrt_appeared_cb (GDBusConnection *connection,
|
|
const gchar *name,
|
|
const gchar *name_owner,
|
|
gpointer user_data)
|
|
{
|
|
CcPrivacyPanel *self = user_data;
|
|
g_debug ("ABRT appeared");
|
|
gtk_widget_show (self->priv->abrt_row);
|
|
}
|
|
|
|
static void
|
|
abrt_vanished_cb (GDBusConnection *connection,
|
|
const gchar *name,
|
|
gpointer user_data)
|
|
{
|
|
CcPrivacyPanel *self = user_data;
|
|
g_debug ("ABRT vanished");
|
|
gtk_widget_hide (self->priv->abrt_row);
|
|
}
|
|
|
|
static void
|
|
add_abrt (CcPrivacyPanel *self)
|
|
{
|
|
GtkWidget *w;
|
|
GtkWidget *dialog;
|
|
char *os_name, *url, *msg;
|
|
|
|
w = get_abrt_label (self->priv->privacy_settings, REPORT_TECHNICAL_PROBLEMS);
|
|
self->priv->abrt_row = add_row (self, _("Problem Reporting"), "abrt_dialog", w);
|
|
gtk_widget_hide (self->priv->abrt_row);
|
|
|
|
dialog = self->priv->abrt_dialog;
|
|
g_signal_connect (dialog, "delete-event",
|
|
G_CALLBACK (gtk_widget_hide_on_delete), NULL);
|
|
|
|
w = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "abrt_switch"));
|
|
g_settings_bind (self->priv->privacy_settings, REPORT_TECHNICAL_PROBLEMS,
|
|
w, "active",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
|
|
os_name = get_os_name ();
|
|
/* translators: '%s' is the distributor's name, such as 'Fedora' */
|
|
msg = g_strdup_printf (_("Sending reports of technical problems helps us improve %s. Reports are sent anonymously and are scrubbed of personal data."),
|
|
os_name);
|
|
g_free (os_name);
|
|
gtk_label_set_text (GTK_LABEL (gtk_builder_get_object (self->priv->builder, "abrt_explanation_label")), msg);
|
|
g_free (msg);
|
|
|
|
url = get_privacy_policy_url ();
|
|
if (!url)
|
|
{
|
|
g_debug ("Not watching for ABRT appearing, /etc/os-release lacks a privacy policy URL");
|
|
return;
|
|
}
|
|
msg = g_strdup_printf ("<a href=\"%s\">%s</a>", url, _("Privacy Policy"));
|
|
g_free (url);
|
|
gtk_label_set_markup (GTK_LABEL (gtk_builder_get_object (self->priv->builder, "abrt_policy_linklabel")), msg);
|
|
g_free (msg);
|
|
|
|
self->priv->abrt_watch_id = g_bus_watch_name (G_BUS_TYPE_SYSTEM,
|
|
"org.freedesktop.problems.daemon",
|
|
G_BUS_NAME_WATCHER_FLAGS_NONE,
|
|
abrt_appeared_cb,
|
|
abrt_vanished_cb,
|
|
self,
|
|
NULL);
|
|
}
|
|
|
|
static void
|
|
cc_privacy_panel_finalize (GObject *object)
|
|
{
|
|
CcPrivacyPanelPrivate *priv = CC_PRIVACY_PANEL (object)->priv;
|
|
|
|
if (priv->abrt_watch_id > 0)
|
|
{
|
|
g_bus_unwatch_name (priv->abrt_watch_id);
|
|
priv->abrt_watch_id = 0;
|
|
}
|
|
|
|
g_cancellable_cancel (priv->cancellable);
|
|
g_clear_pointer (&priv->recent_dialog, gtk_widget_destroy);
|
|
g_clear_pointer (&priv->screen_lock_dialog, gtk_widget_destroy);
|
|
g_clear_pointer (&priv->location_dialog, gtk_widget_destroy);
|
|
g_clear_pointer (&priv->trash_dialog, gtk_widget_destroy);
|
|
g_clear_pointer (&priv->software_dialog, gtk_widget_destroy);
|
|
g_clear_pointer (&priv->abrt_dialog, gtk_widget_destroy);
|
|
g_clear_object (&priv->builder);
|
|
g_clear_object (&priv->lockdown_settings);
|
|
g_clear_object (&priv->lock_settings);
|
|
g_clear_object (&priv->privacy_settings);
|
|
g_clear_object (&priv->notification_settings);
|
|
g_clear_object (&priv->location_settings);
|
|
g_clear_object (&priv->gclue_manager);
|
|
g_clear_object (&priv->cancellable);
|
|
g_clear_object (&priv->perm_store);
|
|
g_clear_object (&priv->location_icon_size_group);
|
|
g_clear_pointer (&priv->location_apps_perms, g_variant_unref);
|
|
g_clear_pointer (&priv->location_apps_data, g_variant_unref);
|
|
g_clear_pointer (&priv->location_app_switches, g_hash_table_unref);
|
|
|
|
G_OBJECT_CLASS (cc_privacy_panel_parent_class)->finalize (object);
|
|
}
|
|
|
|
static const char *
|
|
cc_privacy_panel_get_help_uri (CcPanel *panel)
|
|
{
|
|
return "help:gnome-help/privacy";
|
|
}
|
|
|
|
static void
|
|
activate_row (CcPrivacyPanel *self,
|
|
GtkListBoxRow *row)
|
|
{
|
|
GObject *w;
|
|
const gchar *dialog_id;
|
|
GtkWidget *toplevel;
|
|
|
|
dialog_id = g_object_get_data (G_OBJECT (row), "dialog-id");
|
|
w = gtk_builder_get_object (self->priv->builder, dialog_id);
|
|
if (w == NULL)
|
|
{
|
|
g_warning ("No such dialog: %s", dialog_id);
|
|
return;
|
|
}
|
|
|
|
toplevel = gtk_widget_get_toplevel (GTK_WIDGET (self));
|
|
gtk_window_set_transient_for (GTK_WINDOW (w), GTK_WINDOW (toplevel));
|
|
gtk_window_set_modal (GTK_WINDOW (w), TRUE);
|
|
gtk_window_present (GTK_WINDOW (w));
|
|
}
|
|
|
|
static void
|
|
cc_privacy_panel_init (CcPrivacyPanel *self)
|
|
{
|
|
GError *error;
|
|
GtkWidget *widget;
|
|
GtkWidget *frame;
|
|
guint res;
|
|
|
|
self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, CC_TYPE_PRIVACY_PANEL, CcPrivacyPanelPrivate);
|
|
g_resources_register (cc_privacy_get_resource ());
|
|
|
|
self->priv->cancellable = g_cancellable_new ();
|
|
self->priv->builder = gtk_builder_new ();
|
|
|
|
error = NULL;
|
|
res = gtk_builder_add_from_resource (self->priv->builder,
|
|
"/org/gnome/control-center/privacy/privacy.ui",
|
|
&error);
|
|
|
|
if (res == 0)
|
|
{
|
|
g_warning ("Could not load interface file: %s",
|
|
(error != NULL) ? error->message : "unknown error");
|
|
g_clear_error (&error);
|
|
return;
|
|
}
|
|
|
|
self->priv->recent_dialog = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "recent_dialog"));
|
|
self->priv->screen_lock_dialog = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "screen_lock_dialog"));
|
|
self->priv->location_dialog = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "location_dialog"));
|
|
self->priv->trash_dialog = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "trash_dialog"));
|
|
self->priv->software_dialog = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "software_dialog"));
|
|
self->priv->abrt_dialog = GTK_WIDGET (gtk_builder_get_object (self->priv->builder, "abrt_dialog"));
|
|
|
|
frame = WID ("frame");
|
|
widget = gtk_list_box_new ();
|
|
gtk_list_box_set_selection_mode (GTK_LIST_BOX (widget), GTK_SELECTION_NONE);
|
|
gtk_container_add (GTK_CONTAINER (frame), widget);
|
|
self->priv->list_box = widget;
|
|
gtk_widget_show (widget);
|
|
self->priv->location_apps_list_box = WID ("location_apps_list_box");
|
|
gtk_list_box_set_header_func (GTK_LIST_BOX (self->priv->location_apps_list_box),
|
|
cc_list_box_update_header_func,
|
|
NULL, NULL);
|
|
self->priv->location_apps_frame = WID ("location_apps_frame");
|
|
self->priv->location_apps_label = WID ("location_apps_label");
|
|
self->priv->location_icon_size_group = gtk_size_group_new (GTK_SIZE_GROUP_BOTH);
|
|
|
|
g_signal_connect_swapped (widget, "row-activated",
|
|
G_CALLBACK (activate_row), self);
|
|
|
|
gtk_list_box_set_header_func (GTK_LIST_BOX (widget),
|
|
cc_list_box_update_header_func,
|
|
NULL, NULL);
|
|
|
|
self->priv->lockdown_settings = g_settings_new ("org.gnome.desktop.lockdown");
|
|
self->priv->lock_settings = g_settings_new ("org.gnome.desktop.screensaver");
|
|
self->priv->privacy_settings = g_settings_new ("org.gnome.desktop.privacy");
|
|
self->priv->notification_settings = g_settings_new ("org.gnome.desktop.notifications");
|
|
self->priv->location_settings = g_settings_new ("org.gnome.system.location");
|
|
|
|
add_screen_lock (self);
|
|
add_location (self);
|
|
add_usage_history (self);
|
|
add_trash_temp (self);
|
|
add_software (self);
|
|
add_abrt (self);
|
|
|
|
g_signal_connect (self->priv->lockdown_settings, "changed",
|
|
G_CALLBACK (on_lockdown_settings_changed), self);
|
|
update_lock_screen_sensitivity (self);
|
|
|
|
widget = WID ("privacy_vbox");
|
|
gtk_container_add (GTK_CONTAINER (self), widget);
|
|
}
|
|
|
|
static void
|
|
cc_privacy_panel_class_init (CcPrivacyPanelClass *klass)
|
|
{
|
|
GObjectClass *oclass = G_OBJECT_CLASS (klass);
|
|
CcPanelClass *panel_class = CC_PANEL_CLASS (klass);
|
|
|
|
panel_class->get_help_uri = cc_privacy_panel_get_help_uri;
|
|
|
|
oclass->finalize = cc_privacy_panel_finalize;
|
|
|
|
g_type_class_add_private (klass, sizeof (CcPrivacyPanelPrivate));
|
|
}
|