summaryrefslogtreecommitdiffstats
path: root/src/gs-update-dialog.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/gs-update-dialog.c842
1 files changed, 842 insertions, 0 deletions
diff --git a/src/gs-update-dialog.c b/src/gs-update-dialog.c
new file mode 100644
index 0000000..cd3c0b6
--- /dev/null
+++ b/src/gs-update-dialog.c
@@ -0,0 +1,842 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-
+ * vi:set noexpandtab tabstop=8 shiftwidth=8:
+ *
+ * Copyright (C) 2013-2016 Richard Hughes <richard@hughsie.com>
+ * Copyright (C) 2014-2018 Kalev Lember <klember@redhat.com>
+ *
+ * SPDX-License-Identifier: GPL-2.0+
+ */
+
+#include "config.h"
+
+#include <glib/gi18n.h>
+
+#include "gs-update-dialog.h"
+#include "gs-app-row.h"
+#include "gs-update-list.h"
+#include "gs-common.h"
+
+typedef struct {
+ gchar *title;
+ gchar *stack_page;
+ GtkWidget *focus;
+} BackEntry;
+
+typedef enum {
+ GS_UPDATE_DIALOG_SECTION_ADDITIONS,
+ GS_UPDATE_DIALOG_SECTION_REMOVALS,
+ GS_UPDATE_DIALOG_SECTION_UPDATES,
+ GS_UPDATE_DIALOG_SECTION_DOWNGRADES,
+ GS_UPDATE_DIALOG_SECTION_LAST,
+} GsUpdateDialogSection;
+
+struct _GsUpdateDialog
+{
+ GtkDialog parent_instance;
+
+ GQueue *back_entry_stack;
+ GCancellable *cancellable;
+ GsPluginLoader *plugin_loader;
+ GtkWidget *box_header;
+ GtkWidget *button_back;
+ GtkWidget *image_icon;
+ GtkWidget *label_details;
+ GtkWidget *label_name;
+ GtkWidget *label_summary;
+ GtkWidget *list_boxes[GS_UPDATE_DIALOG_SECTION_LAST];
+ GtkWidget *list_box_installed_updates;
+ GtkWidget *os_update_description;
+ GtkWidget *os_update_box;
+ GtkWidget *scrolledwindow;
+ GtkWidget *scrolledwindow_details;
+ GtkWidget *spinner;
+ GtkWidget *stack;
+ GtkWidget *permissions_section_box;
+ GtkWidget *permissions_section_content;
+};
+
+G_DEFINE_TYPE (GsUpdateDialog, gs_update_dialog, GTK_TYPE_DIALOG)
+
+static void
+save_back_entry (GsUpdateDialog *dialog)
+{
+ BackEntry *entry;
+
+ entry = g_slice_new0 (BackEntry);
+ entry->stack_page = g_strdup (gtk_stack_get_visible_child_name (GTK_STACK (dialog->stack)));
+ entry->title = g_strdup (gtk_window_get_title (GTK_WINDOW (dialog)));
+
+ entry->focus = gtk_window_get_focus (GTK_WINDOW (dialog));
+ if (entry->focus != NULL)
+ g_object_add_weak_pointer (G_OBJECT (entry->focus),
+ (gpointer *) &entry->focus);
+
+ g_queue_push_head (dialog->back_entry_stack, entry);
+}
+
+static void
+back_entry_free (BackEntry *entry)
+{
+ if (entry->focus != NULL)
+ g_object_remove_weak_pointer (G_OBJECT (entry->focus),
+ (gpointer *) &entry->focus);
+ g_free (entry->stack_page);
+ g_free (entry->title);
+ g_slice_free (BackEntry, entry);
+}
+
+static struct {
+ GsAppPermissions permission;
+ const char *title;
+ const char *subtitle;
+} permission_display_data[] = {
+ { GS_APP_PERMISSIONS_NETWORK, N_("Network"), N_("Can communicate over the network") },
+ { GS_APP_PERMISSIONS_SYSTEM_BUS, N_("System Services"), N_("Can access D-Bus services on the system bus") },
+ { GS_APP_PERMISSIONS_SESSION_BUS, N_("Session Services"), N_("Can access D-Bus services on the session bus") },
+ { GS_APP_PERMISSIONS_DEVICES, N_("Devices"), N_("Can access system device files") },
+ { GS_APP_PERMISSIONS_HOME_FULL, N_("Home folder"), N_("Can view, edit and create files") },
+ { GS_APP_PERMISSIONS_HOME_READ, N_("Home folder"), N_("Can view files") },
+ { GS_APP_PERMISSIONS_FILESYSTEM_FULL, N_("File system"), N_("Can view, edit and create files") },
+ { GS_APP_PERMISSIONS_FILESYSTEM_READ, N_("File system"), N_("Can view files") },
+ { GS_APP_PERMISSIONS_DOWNLOADS_FULL, N_("Downloads folder"), N_("Can view, edit and create files") },
+ { GS_APP_PERMISSIONS_DOWNLOADS_READ, N_("Downloads folder"), N_("Can view files") },
+ { GS_APP_PERMISSIONS_SETTINGS, N_("Settings"), N_("Can view and change any settings") },
+ { GS_APP_PERMISSIONS_X11, N_("Legacy display system"), N_("Uses an old, insecure display system") },
+ { GS_APP_PERMISSIONS_ESCAPE_SANDBOX, N_("Sandbox escape"), N_("Can escape the sandbox and circumvent any other restrictions") },
+};
+
+static void
+populate_permissions_section (GsUpdateDialog *dialog, GsAppPermissions permissions)
+{
+ GList *children;
+
+ children = gtk_container_get_children (GTK_CONTAINER (dialog->permissions_section_content));
+ for (GList *l = children; l != NULL; l = l->next)
+ gtk_widget_destroy (GTK_WIDGET (l->data));
+ g_list_free (children);
+
+ for (gsize i = 0; i < G_N_ELEMENTS (permission_display_data); i++) {
+ GtkWidget *row, *image, *box, *label;
+
+ if ((permissions & permission_display_data[i].permission) == 0)
+ continue;
+
+ row = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 12);
+ gtk_widget_show (row);
+ if ((permission_display_data[i].permission & ~MEDIUM_PERMISSIONS) != 0) {
+ gtk_style_context_add_class (gtk_widget_get_style_context (row), "permission-row-warning");
+ }
+
+ image = gtk_image_new_from_icon_name ("dialog-warning-symbolic", GTK_ICON_SIZE_MENU);
+ if ((permission_display_data[i].permission & ~MEDIUM_PERMISSIONS) == 0)
+ gtk_widget_set_opacity (image, 0);
+
+ gtk_widget_show (image);
+ gtk_container_add (GTK_CONTAINER (row), image);
+
+ box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
+ gtk_widget_show (box);
+ gtk_container_add (GTK_CONTAINER (row), box);
+
+ label = gtk_label_new (_(permission_display_data[i].title));
+ gtk_label_set_xalign (GTK_LABEL (label), 0);
+ gtk_widget_show (label);
+ gtk_container_add (GTK_CONTAINER (box), label);
+
+ label = gtk_label_new (_(permission_display_data[i].subtitle));
+ gtk_label_set_xalign (GTK_LABEL (label), 0);
+ gtk_style_context_add_class (gtk_widget_get_style_context (label), "dim-label");
+ gtk_widget_show (label);
+ gtk_container_add (GTK_CONTAINER (box), label);
+
+ gtk_container_add (GTK_CONTAINER (dialog->permissions_section_content), row);
+ }
+}
+
+static void
+set_updates_description_ui (GsUpdateDialog *dialog, GsApp *app)
+{
+ AsAppKind kind;
+ const GdkPixbuf *pixbuf;
+ const gchar *update_details;
+
+ /* set window title */
+ kind = gs_app_get_kind (app);
+ if (kind == AS_APP_KIND_OS_UPDATE) {
+ gtk_window_set_title (GTK_WINDOW (dialog), gs_app_get_name (app));
+ } else if (gs_app_get_source_default (app) != NULL &&
+ gs_app_get_update_version (app) != NULL) {
+ g_autofree gchar *tmp = NULL;
+ tmp = g_strdup_printf ("%s %s",
+ gs_app_get_source_default (app),
+ gs_app_get_update_version (app));
+ gtk_window_set_title (GTK_WINDOW (dialog), tmp);
+ } else if (gs_app_get_source_default (app) != NULL) {
+ gtk_window_set_title (GTK_WINDOW (dialog),
+ gs_app_get_source_default (app));
+ } else {
+ gtk_window_set_title (GTK_WINDOW (dialog),
+ gs_app_get_update_version (app));
+ }
+
+ /* set update header */
+ gtk_widget_set_visible (dialog->box_header, kind == AS_APP_KIND_DESKTOP);
+ update_details = gs_app_get_update_details (app);
+ if (update_details == NULL) {
+ /* TRANSLATORS: this is where the packager did not write
+ * a description for the update */
+ update_details = _("No update description available.");
+ }
+ gtk_label_set_label (GTK_LABEL (dialog->label_details), update_details);
+ gtk_label_set_label (GTK_LABEL (dialog->label_name), gs_app_get_name (app));
+ gtk_label_set_label (GTK_LABEL (dialog->label_summary), gs_app_get_summary (app));
+
+ pixbuf = gs_app_get_pixbuf (app);
+ if (pixbuf != NULL)
+ gs_image_set_from_pixbuf (GTK_IMAGE (dialog->image_icon), pixbuf);
+
+ /* show the back button if needed */
+ gtk_widget_set_visible (dialog->button_back, !g_queue_is_empty (dialog->back_entry_stack));
+
+ if (gs_app_has_quirk (app, GS_APP_QUIRK_NEW_PERMISSIONS)) {
+ gtk_widget_show (dialog->permissions_section_box);
+ populate_permissions_section (dialog, gs_app_get_update_permissions (app));
+ } else {
+ gtk_widget_hide (dialog->permissions_section_box);
+ }
+}
+
+static void
+row_activated_cb (GtkListBox *list_box,
+ GtkListBoxRow *row,
+ GsUpdateDialog *dialog)
+{
+ GsApp *app;
+
+ app = GS_APP (g_object_get_data (G_OBJECT (gtk_bin_get_child (GTK_BIN (row))), "app"));
+
+ /* save the current stack state for the back button */
+ save_back_entry (dialog);
+
+ /* setup package view */
+ gs_update_dialog_show_update_details (dialog, app);
+}
+
+static void
+installed_updates_row_activated_cb (GtkListBox *list_box,
+ GtkListBoxRow *row,
+ GsUpdateDialog *dialog)
+{
+ GsApp *app;
+
+ app = gs_app_row_get_app (GS_APP_ROW (row));
+
+ /* save the current stack state for the back button */
+ save_back_entry (dialog);
+
+ gs_update_dialog_show_update_details (dialog, app);
+}
+
+static void
+get_installed_updates_cb (GsPluginLoader *plugin_loader,
+ GAsyncResult *res,
+ GsUpdateDialog *dialog)
+{
+ guint i;
+ guint64 install_date;
+ g_autoptr(GsAppList) list = NULL;
+ g_autoptr(GError) error = NULL;
+
+ /* get the results */
+ list = gs_plugin_loader_job_process_finish (plugin_loader, res, &error);
+
+ /* if we're in teardown, short-circuit and return immediately without
+ * dereferencing priv variables */
+ if (g_error_matches (error, GS_PLUGIN_ERROR, GS_PLUGIN_ERROR_CANCELLED) ||
+ dialog->spinner == NULL) {
+ g_debug ("get installed updates cancelled");
+ return;
+ }
+
+ gs_stop_spinner (GTK_SPINNER (dialog->spinner));
+
+ /* error */
+ if (list == NULL) {
+ g_warning ("failed to get installed updates: %s", error->message);
+ gtk_stack_set_visible_child_name (GTK_STACK (dialog->stack), "empty");
+ return;
+ }
+
+ /* no results */
+ if (gs_app_list_length (list) == 0) {
+ g_debug ("no installed updates to show");
+ gtk_stack_set_visible_child_name (GTK_STACK (dialog->stack), "empty");
+ return;
+ }
+
+ /* set the header title using any one of the applications */
+ install_date = gs_app_get_install_date (gs_app_list_index (list, 0));
+ if (install_date > 0) {
+ GtkWidget *header;
+ g_autoptr(GDateTime) date = NULL;
+ g_autofree gchar *date_str = NULL;
+ g_autofree gchar *subtitle = NULL;
+
+ date = g_date_time_new_from_unix_utc ((gint64) install_date);
+ date_str = g_date_time_format (date, "%x");
+
+ /* TRANSLATORS: this is the subtitle of the installed updates dialog window.
+ %s will be replaced by the date when the updates were installed.
+ The date format is defined by the locale's preferred date representation
+ ("%x" in strftime.) */
+ subtitle = g_strdup_printf (_("Installed on %s"), date_str);
+ header = gtk_dialog_get_header_bar (GTK_DIALOG (dialog));
+ gtk_header_bar_set_subtitle (GTK_HEADER_BAR (header), subtitle);
+ }
+
+ gtk_stack_set_visible_child_name (GTK_STACK (dialog->stack), "installed-updates-list");
+
+ gs_container_remove_all (GTK_CONTAINER (dialog->list_box_installed_updates));
+ for (i = 0; i < gs_app_list_length (list); i++) {
+ gs_update_list_add_app (GS_UPDATE_LIST (dialog->list_box_installed_updates),
+ gs_app_list_index (list, i));
+ }
+}
+
+void
+gs_update_dialog_show_installed_updates (GsUpdateDialog *dialog)
+{
+ g_autoptr(GsPluginJob) plugin_job = NULL;
+
+ /* TRANSLATORS: this is the title of the installed updates dialog window */
+ gtk_window_set_title (GTK_WINDOW (dialog), _("Installed Updates"));
+
+ gtk_widget_set_visible (dialog->button_back, !g_queue_is_empty (dialog->back_entry_stack));
+ gs_start_spinner (GTK_SPINNER (dialog->spinner));
+ gtk_stack_set_visible_child_name (GTK_STACK (dialog->stack), "spinner");
+
+ plugin_job = gs_plugin_job_newv (GS_PLUGIN_ACTION_GET_UPDATES_HISTORICAL,
+ "refine-flags", GS_PLUGIN_REFINE_FLAGS_REQUIRE_UPDATE_DETAILS |
+ GS_PLUGIN_REFINE_FLAGS_REQUIRE_ICON |
+ GS_PLUGIN_REFINE_FLAGS_REQUIRE_VERSION,
+ NULL);
+ gs_plugin_loader_job_process_async (dialog->plugin_loader, plugin_job,
+ dialog->cancellable,
+ (GAsyncReadyCallback) get_installed_updates_cb,
+ dialog);
+}
+
+static void
+unset_focus (GtkWidget *widget)
+{
+ GtkWidget *focus;
+
+ focus = gtk_window_get_focus (GTK_WINDOW (widget));
+ if (GTK_IS_LABEL (focus))
+ gtk_label_select_region (GTK_LABEL (focus), 0, 0);
+ gtk_window_set_focus (GTK_WINDOW (widget), NULL);
+}
+
+static gchar *
+format_version_update (GsApp *app)
+{
+ const gchar *tmp;
+ const gchar *version_current = NULL;
+ const gchar *version_update = NULL;
+
+ /* current version */
+ tmp = gs_app_get_version (app);
+ if (tmp != NULL && tmp[0] != '\0')
+ version_current = tmp;
+
+ /* update version */
+ tmp = gs_app_get_update_version (app);
+ if (tmp != NULL && tmp[0] != '\0')
+ version_update = tmp;
+
+ /* have both */
+ if (version_current != NULL && version_update != NULL &&
+ g_strcmp0 (version_current, version_update) != 0) {
+ return g_strdup_printf ("%s → %s",
+ version_current,
+ version_update);
+ }
+
+ /* just update */
+ if (version_update)
+ return g_strdup (version_update);
+
+ /* we have nothing, nada, zilch */
+ return NULL;
+}
+
+static GtkWidget *
+create_app_row (GsApp *app)
+{
+ GtkWidget *row, *label;
+
+ row = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 12);
+ g_object_set_data_full (G_OBJECT (row),
+ "app",
+ g_object_ref (app),
+ g_object_unref);
+ label = gtk_label_new (gs_app_get_source_default (app));
+ g_object_set (label,
+ "margin-start", 20,
+ "margin-end", 0,
+ "margin-top", 6,
+ "margin-bottom", 6,
+ "xalign", 0.0,
+ "ellipsize", PANGO_ELLIPSIZE_END,
+ NULL);
+ gtk_widget_set_halign (label, GTK_ALIGN_START);
+ gtk_widget_set_hexpand (label, TRUE);
+ gtk_widget_set_valign (label, GTK_ALIGN_CENTER);
+ gtk_container_add (GTK_CONTAINER (row), label);
+ if (gs_app_get_state (app) == AS_APP_STATE_UPDATABLE ||
+ gs_app_get_state (app) == AS_APP_STATE_UPDATABLE_LIVE) {
+ g_autofree gchar *verstr = format_version_update (app);
+ label = gtk_label_new (verstr);
+ } else {
+ label = gtk_label_new (gs_app_get_version (app));
+ }
+ g_object_set (label,
+ "margin-start", 0,
+ "margin-end", 20,
+ "margin-top", 6,
+ "margin-bottom", 6,
+ "xalign", 1.0,
+ "ellipsize", PANGO_ELLIPSIZE_END,
+ NULL);
+ gtk_widget_set_halign (label, GTK_ALIGN_END);
+ gtk_widget_set_valign (label, GTK_ALIGN_CENTER);
+ gtk_container_add (GTK_CONTAINER (row), label);
+ gtk_widget_show_all (row);
+
+ return row;
+}
+
+static gboolean
+is_downgrade (const gchar *evr1,
+ const gchar *evr2)
+{
+ gint rc;
+ g_autofree gchar *epoch1 = NULL;
+ g_autofree gchar *epoch2 = NULL;
+ g_autofree gchar *version1 = NULL;
+ g_autofree gchar *version2 = NULL;
+ g_autofree gchar *release1 = NULL;
+ g_autofree gchar *release2 = NULL;
+
+ if (evr1 == NULL || evr2 == NULL)
+ return FALSE;
+
+ /* split into epoch-version-release */
+ if (!gs_utils_parse_evr (evr1, &epoch1, &version1, &release1))
+ return FALSE;
+ if (!gs_utils_parse_evr (evr2, &epoch2, &version2, &release2))
+ return FALSE;
+
+ /* ignore epoch here as it's a way to make downgrades happen and not
+ * part of the semantic version */
+
+ /* check version */
+#if AS_CHECK_VERSION(0,7,15)
+ rc = as_utils_vercmp_full (version1, version2,
+ AS_VERSION_COMPARE_FLAG_NONE);
+#else
+ rc = as_utils_vercmp (version1, version2);
+#endif
+ if (rc != 0)
+ return rc > 0;
+
+ /* check release */
+#if AS_CHECK_VERSION(0,7,15)
+ rc = as_utils_vercmp_full (version1, version2,
+ AS_VERSION_COMPARE_FLAG_NONE);
+#else
+ rc = as_utils_vercmp (release1, release2);
+#endif
+ if (rc != 0)
+ return rc > 0;
+
+ return FALSE;
+}
+
+static GsUpdateDialogSection
+get_app_section (GsApp *app)
+{
+ GsUpdateDialogSection section;
+
+ /* Sections:
+ * 1. additions
+ * 2. removals
+ * 3. updates
+ * 4. downgrades */
+ switch (gs_app_get_state (app)) {
+ case AS_APP_STATE_AVAILABLE:
+ section = GS_UPDATE_DIALOG_SECTION_ADDITIONS;
+ break;
+ case AS_APP_STATE_UNAVAILABLE:
+ section = GS_UPDATE_DIALOG_SECTION_REMOVALS;
+ break;
+ case AS_APP_STATE_UPDATABLE:
+ case AS_APP_STATE_UPDATABLE_LIVE:
+ if (is_downgrade (gs_app_get_version (app),
+ gs_app_get_update_version (app)))
+ section = GS_UPDATE_DIALOG_SECTION_DOWNGRADES;
+ else
+ section = GS_UPDATE_DIALOG_SECTION_UPDATES;
+ break;
+ default:
+ g_warning ("get_app_section: unhandled state %s for %s",
+ as_app_state_to_string (gs_app_get_state (app)),
+ gs_app_get_unique_id (app));
+ section = GS_UPDATE_DIALOG_SECTION_UPDATES;
+ break;
+ }
+
+ return section;
+}
+
+static gint
+os_updates_sort_func (GtkListBoxRow *a,
+ GtkListBoxRow *b,
+ gpointer user_data)
+{
+ GObject *o1 = G_OBJECT (gtk_bin_get_child (GTK_BIN (a)));
+ GObject *o2 = G_OBJECT (gtk_bin_get_child (GTK_BIN (b)));
+ GsApp *a1 = g_object_get_data (o1, "app");
+ GsApp *a2 = g_object_get_data (o2, "app");
+ const gchar *key1 = gs_app_get_source_default (a1);
+ const gchar *key2 = gs_app_get_source_default (a2);
+
+ return g_strcmp0 (key1, key2);
+}
+
+static GtkWidget *
+get_section_header (GsUpdateDialog *dialog, GsUpdateDialogSection section)
+{
+ GtkStyleContext *context;
+ GtkWidget *header;
+ GtkWidget *label;
+
+ /* get labels and buttons for everything */
+ if (section == GS_UPDATE_DIALOG_SECTION_ADDITIONS) {
+ /* TRANSLATORS: This is the header for package additions during
+ * a system update */
+ label = gtk_label_new (_("Additions"));
+ } else if (section == GS_UPDATE_DIALOG_SECTION_REMOVALS) {
+ /* TRANSLATORS: This is the header for package removals during
+ * a system update */
+ label = gtk_label_new (_("Removals"));
+ } else if (section == GS_UPDATE_DIALOG_SECTION_UPDATES) {
+ /* TRANSLATORS: This is the header for package updates during
+ * a system update */
+ label = gtk_label_new (_("Updates"));
+ } else if (section == GS_UPDATE_DIALOG_SECTION_DOWNGRADES) {
+ /* TRANSLATORS: This is the header for package downgrades during
+ * a system update */
+ label = gtk_label_new (_("Downgrades"));
+ } else {
+ g_assert_not_reached ();
+ }
+
+ /* create header */
+ header = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 3);
+ context = gtk_widget_get_style_context (header);
+ gtk_style_context_add_class (context, "app-listbox-header");
+
+ /* put label into the header */
+ gtk_widget_set_hexpand (label, TRUE);
+ gtk_container_add (GTK_CONTAINER (header), label);
+ gtk_widget_set_visible (label, TRUE);
+ gtk_widget_set_margin_start (label, 6);
+ gtk_label_set_xalign (GTK_LABEL (label), 0.0);
+ context = gtk_widget_get_style_context (label);
+ gtk_style_context_add_class (context, "app-listbox-header-title");
+
+ /* success */
+ return header;
+}
+
+static void
+list_header_func (GtkListBoxRow *row,
+ GtkListBoxRow *before,
+ gpointer user_data)
+{
+ GsUpdateDialog *dialog = (GsUpdateDialog *) user_data;
+ GObject *o = G_OBJECT (gtk_bin_get_child (GTK_BIN (row)));
+ GsApp *app = g_object_get_data (o, "app");
+ GtkWidget *header = NULL;
+
+ if (before == NULL)
+ header = get_section_header (dialog, get_app_section (app));
+ else
+ header = gtk_separator_new (GTK_ORIENTATION_HORIZONTAL);
+ gtk_list_box_row_set_header (row, header);
+}
+
+static void
+create_section (GsUpdateDialog *dialog, GsUpdateDialogSection section)
+{
+ GtkStyleContext *context;
+
+ dialog->list_boxes[section] = gtk_list_box_new ();
+ gtk_list_box_set_selection_mode (GTK_LIST_BOX (dialog->list_boxes[section]),
+ GTK_SELECTION_NONE);
+ gtk_list_box_set_sort_func (GTK_LIST_BOX (dialog->list_boxes[section]),
+ os_updates_sort_func,
+ dialog, NULL);
+ gtk_list_box_set_header_func (GTK_LIST_BOX (dialog->list_boxes[section]),
+ list_header_func,
+ dialog, NULL);
+ g_signal_connect (GTK_LIST_BOX (dialog->list_boxes[section]), "row-activated",
+ G_CALLBACK (row_activated_cb), dialog);
+ gtk_widget_set_visible (dialog->list_boxes[section], TRUE);
+ gtk_widget_set_vexpand (dialog->list_boxes[section], TRUE);
+ gtk_container_add (GTK_CONTAINER (dialog->os_update_box), dialog->list_boxes[section]);
+ gtk_widget_set_margin_top (dialog->list_boxes[section], 24);
+
+ /* reorder the children */
+ for (guint i = 0; i < GS_UPDATE_DIALOG_SECTION_LAST; i++) {
+ if (dialog->list_boxes[i] == NULL)
+ continue;
+ gtk_box_reorder_child (GTK_BOX (dialog->os_update_box),
+ dialog->list_boxes[i], i);
+ }
+
+ /* make rounded edges */
+ context = gtk_widget_get_style_context (dialog->list_boxes[section]);
+ gtk_style_context_add_class (context, "app-updates-section");
+}
+
+void
+gs_update_dialog_show_update_details (GsUpdateDialog *dialog, GsApp *app)
+{
+ AsAppKind kind;
+ g_autofree gchar *str = NULL;
+
+ /* debug */
+ str = gs_app_to_string (app);
+ g_debug ("%s", str);
+
+ /* set update header */
+ set_updates_description_ui (dialog, app);
+
+ /* workaround a gtk+ issue where the dialog comes up with a label selected,
+ * https://bugzilla.gnome.org/show_bug.cgi?id=734033 */
+ unset_focus (GTK_WIDGET (dialog));
+
+ /* set update description */
+ kind = gs_app_get_kind (app);
+ if (kind == AS_APP_KIND_OS_UPDATE) {
+ GsAppList *related;
+ GsApp *app_related;
+ GsUpdateDialogSection section;
+ GtkWidget *row;
+
+ gtk_label_set_text (GTK_LABEL (dialog->os_update_description),
+ gs_app_get_description (app));
+
+ /* clear existing data */
+ for (guint i = 0; i < GS_UPDATE_DIALOG_SECTION_LAST; i++) {
+ if (dialog->list_boxes[i] == NULL)
+ continue;
+ gs_container_remove_all (GTK_CONTAINER (dialog->list_boxes[i]));
+ }
+
+ /* add new apps */
+ related = gs_app_get_related (app);
+ for (guint i = 0; i < gs_app_list_length (related); i++) {
+ app_related = gs_app_list_index (related, i);
+
+ section = get_app_section (app_related);
+ if (dialog->list_boxes[section] == NULL)
+ create_section (dialog, section);
+
+ row = create_app_row (app_related);
+ gtk_list_box_insert (GTK_LIST_BOX (dialog->list_boxes[section]), row, -1);
+ }
+ gtk_stack_set_transition_type (GTK_STACK (dialog->stack), GTK_STACK_TRANSITION_TYPE_SLIDE_LEFT);
+ gtk_stack_set_visible_child_name (GTK_STACK (dialog->stack), "os-update-list");
+ gtk_stack_set_transition_type (GTK_STACK (dialog->stack), GTK_STACK_TRANSITION_TYPE_NONE);
+ } else {
+ gtk_stack_set_transition_type (GTK_STACK (dialog->stack), GTK_STACK_TRANSITION_TYPE_SLIDE_LEFT);
+ gtk_stack_set_visible_child_name (GTK_STACK (dialog->stack), "package-details");
+ gtk_stack_set_transition_type (GTK_STACK (dialog->stack), GTK_STACK_TRANSITION_TYPE_NONE);
+ }
+}
+
+static void
+button_back_cb (GtkWidget *widget, GsUpdateDialog *dialog)
+{
+ BackEntry *entry;
+
+ /* return to the previous view */
+ entry = g_queue_pop_head (dialog->back_entry_stack);
+
+ gtk_stack_set_transition_type (GTK_STACK (dialog->stack), GTK_STACK_TRANSITION_TYPE_SLIDE_RIGHT);
+ gtk_stack_set_visible_child_name (GTK_STACK (dialog->stack), entry->stack_page);
+ gtk_stack_set_transition_type (GTK_STACK (dialog->stack), GTK_STACK_TRANSITION_TYPE_NONE);
+
+ gtk_window_set_title (GTK_WINDOW (dialog), entry->title);
+ if (entry->focus)
+ gtk_widget_grab_focus (entry->focus);
+ back_entry_free (entry);
+
+ gtk_widget_set_visible (dialog->button_back, !g_queue_is_empty (dialog->back_entry_stack));
+}
+
+static void
+scrollbar_mapped_cb (GtkWidget *sb, GtkScrolledWindow *swin)
+{
+ GtkWidget *frame;
+
+ frame = gtk_bin_get_child (GTK_BIN (gtk_bin_get_child (GTK_BIN (swin))));
+
+ if (gtk_widget_get_mapped (GTK_WIDGET (sb))) {
+ gtk_scrolled_window_set_shadow_type (swin, GTK_SHADOW_IN);
+ if (GTK_IS_FRAME (frame))
+ gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_NONE);
+ } else {
+ if (GTK_IS_FRAME (frame))
+ gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
+ gtk_scrolled_window_set_shadow_type (swin, GTK_SHADOW_NONE);
+ }
+}
+
+static gboolean
+key_press_event (GtkWidget *widget, GdkEventKey *event, gpointer user_data)
+{
+ GsUpdateDialog *dialog = (GsUpdateDialog *) widget;
+ GdkKeymap *keymap;
+ GdkModifierType state;
+ gboolean is_rtl;
+
+ if (!gtk_widget_is_visible (dialog->button_back) || !gtk_widget_is_sensitive (dialog->button_back))
+ return GDK_EVENT_PROPAGATE;
+
+ state = event->state;
+ keymap = gdk_keymap_get_for_display (gtk_widget_get_display (widget));
+ gdk_keymap_add_virtual_modifiers (keymap, &state);
+ state = state & gtk_accelerator_get_default_mod_mask ();
+ is_rtl = gtk_widget_get_direction (dialog->button_back) == GTK_TEXT_DIR_RTL;
+
+ if ((!is_rtl && state == GDK_MOD1_MASK && event->keyval == GDK_KEY_Left) ||
+ (is_rtl && state == GDK_MOD1_MASK && event->keyval == GDK_KEY_Right) ||
+ event->keyval == GDK_KEY_Back) {
+ gtk_widget_activate (dialog->button_back);
+ return GDK_EVENT_STOP;
+ }
+
+ return GDK_EVENT_PROPAGATE;
+}
+
+static gboolean
+button_press_event (GsUpdateDialog *dialog, GdkEventButton *event)
+{
+ /* Mouse hardware back button is 8 */
+ if (event->button != 8)
+ return GDK_EVENT_PROPAGATE;
+
+ if (!gtk_widget_is_visible (dialog->button_back) || !gtk_widget_is_sensitive (dialog->button_back))
+ return GDK_EVENT_PROPAGATE;
+
+ gtk_widget_activate (dialog->button_back);
+ return GDK_EVENT_STOP;
+}
+
+static void
+set_plugin_loader (GsUpdateDialog *dialog, GsPluginLoader *plugin_loader)
+{
+ dialog->plugin_loader = g_object_ref (plugin_loader);
+}
+
+static void
+gs_update_dialog_dispose (GObject *object)
+{
+ GsUpdateDialog *dialog = GS_UPDATE_DIALOG (object);
+
+ if (dialog->back_entry_stack != NULL) {
+ g_queue_free_full (dialog->back_entry_stack, (GDestroyNotify) back_entry_free);
+ dialog->back_entry_stack = NULL;
+ }
+
+ g_cancellable_cancel (dialog->cancellable);
+ g_clear_object (&dialog->cancellable);
+
+ g_clear_object (&dialog->plugin_loader);
+
+ G_OBJECT_CLASS (gs_update_dialog_parent_class)->dispose (object);
+}
+
+static void
+gs_update_dialog_init (GsUpdateDialog *dialog)
+{
+ GtkWidget *scrollbar;
+
+ gtk_widget_init_template (GTK_WIDGET (dialog));
+
+ dialog->back_entry_stack = g_queue_new ();
+ dialog->cancellable = g_cancellable_new ();
+
+ g_signal_connect (GTK_LIST_BOX (dialog->list_box_installed_updates), "row-activated",
+ G_CALLBACK (installed_updates_row_activated_cb), dialog);
+
+ g_signal_connect (dialog->button_back, "clicked",
+ G_CALLBACK (button_back_cb),
+ dialog);
+
+ g_signal_connect_after (dialog, "show", G_CALLBACK (unset_focus), NULL);
+
+ scrollbar = gtk_scrolled_window_get_vscrollbar (GTK_SCROLLED_WINDOW (dialog->scrolledwindow_details));
+ g_signal_connect (scrollbar, "map", G_CALLBACK (scrollbar_mapped_cb), dialog->scrolledwindow_details);
+ g_signal_connect (scrollbar, "unmap", G_CALLBACK (scrollbar_mapped_cb), dialog->scrolledwindow_details);
+
+ /* global keynav and mouse back button */
+ g_signal_connect (dialog, "key-press-event",
+ G_CALLBACK (key_press_event), NULL);
+ g_signal_connect (dialog, "button-press-event",
+ G_CALLBACK (button_press_event), NULL);
+}
+
+static void
+gs_update_dialog_class_init (GsUpdateDialogClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+
+ object_class->dispose = gs_update_dialog_dispose;
+
+ gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/Software/gs-update-dialog.ui");
+
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, box_header);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, button_back);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, image_icon);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, label_details);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, label_name);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, label_summary);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, list_box_installed_updates);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, os_update_description);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, os_update_box);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, scrolledwindow);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, scrolledwindow_details);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, spinner);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, stack);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, permissions_section_box);
+ gtk_widget_class_bind_template_child (widget_class, GsUpdateDialog, permissions_section_content);
+}
+
+GtkWidget *
+gs_update_dialog_new (GsPluginLoader *plugin_loader)
+{
+ GsUpdateDialog *dialog;
+
+ dialog = g_object_new (GS_TYPE_UPDATE_DIALOG,
+ "use-header-bar", TRUE,
+ NULL);
+ set_plugin_loader (dialog, plugin_loader);
+
+ return GTK_WIDGET (dialog);
+}