summaryrefslogtreecommitdiffstats
path: root/src/gs-app-row.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/gs-app-row.c')
-rw-r--r--src/gs-app-row.c1240
1 files changed, 1240 insertions, 0 deletions
diff --git a/src/gs-app-row.c b/src/gs-app-row.c
new file mode 100644
index 0000000..0cc9895
--- /dev/null
+++ b/src/gs-app-row.c
@@ -0,0 +1,1240 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-
+ * vi:set noexpandtab tabstop=8 shiftwidth=8:
+ *
+ * Copyright (C) 2012-2016 Richard Hughes <richard@hughsie.com>
+ * Copyright (C) 2013 Matthias Clasen <mclasen@redhat.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-app-row.h"
+#include "gs-star-widget.h"
+#include "gs-progress-button.h"
+#include "gs-common.h"
+
+typedef struct
+{
+ GsApp *app;
+ GtkWidget *image;
+ GtkWidget *name_box;
+ GtkWidget *name_label;
+ GtkWidget *version_box;
+ GtkWidget *version_current_label;
+ GtkWidget *version_arrow_label;
+ GtkWidget *version_update_label;
+ GtkWidget *system_updates_label; /* Only for "System Updates" app */
+ GtkWidget *star;
+ GtkWidget *description_label;
+ GtkWidget *button_box;
+ GtkWidget *button_revealer;
+ GtkWidget *button;
+ GtkWidget *spinner;
+ GtkWidget *label;
+ GtkWidget *box_tag;
+ GtkWidget *label_warning;
+ GtkWidget *label_origin;
+ GtkWidget *label_installed;
+ GtkWidget *label_app_size;
+ gboolean colorful;
+ gboolean show_buttons;
+ gboolean show_rating;
+ gboolean show_description;
+ gboolean show_source;
+ gboolean show_update;
+ gboolean show_installed_size;
+ gboolean show_installed;
+ guint pending_refresh_id;
+ guint unreveal_in_idle_id;
+ gboolean is_narrow;
+} GsAppRowPrivate;
+
+G_DEFINE_TYPE_WITH_PRIVATE (GsAppRow, gs_app_row, GTK_TYPE_LIST_BOX_ROW)
+
+enum {
+ SIGNAL_BUTTON_CLICKED,
+ SIGNAL_UNREVEALED,
+ SIGNAL_LAST
+};
+
+static guint signals [SIGNAL_LAST] = { 0 };
+
+typedef enum {
+ PROP_APP = 1,
+ PROP_COLORFUL,
+ PROP_SHOW_DESCRIPTION,
+ PROP_SHOW_SOURCE,
+ PROP_SHOW_BUTTONS,
+ PROP_SHOW_RATING,
+ PROP_SHOW_UPDATE,
+ PROP_SHOW_INSTALLED_SIZE,
+ PROP_SHOW_INSTALLED,
+ PROP_IS_NARROW,
+} GsAppRowProperty;
+
+static GParamSpec *obj_props[PROP_IS_NARROW + 1] = { NULL, };
+
+/*
+ * gs_app_row_get_description:
+ *
+ * Return value: PangoMarkup or text
+ */
+static GString *
+gs_app_row_get_description (GsAppRow *app_row,
+ gboolean *out_is_markup)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+ const gchar *tmp = NULL;
+
+ *out_is_markup = FALSE;
+
+ /* convert the markdown update description into PangoMarkup */
+ if (priv->show_update) {
+ tmp = gs_app_get_update_details_markup (priv->app);
+ if (tmp != NULL && tmp[0] != '\0') {
+ *out_is_markup = TRUE;
+ return g_string_new (tmp);
+ }
+ }
+
+ /* if missing summary is set, return it without escaping in order to
+ * correctly show hyperlinks */
+ if (gs_app_get_state (priv->app) == GS_APP_STATE_UNAVAILABLE) {
+ tmp = gs_app_get_summary_missing (priv->app);
+ if (tmp != NULL && tmp[0] != '\0')
+ return g_string_new (tmp);
+ }
+
+ /* try all these things in order */
+ if (tmp == NULL || (tmp != NULL && tmp[0] == '\0'))
+ tmp = gs_app_get_summary (priv->app);
+ if (tmp == NULL || (tmp != NULL && tmp[0] == '\0'))
+ tmp = gs_app_get_description (priv->app);
+ if (tmp == NULL || (tmp != NULL && tmp[0] == '\0'))
+ tmp = gs_app_get_name (priv->app);
+ if (tmp == NULL)
+ return NULL;
+ return g_string_new (tmp);
+}
+
+static void
+gs_app_row_update_button_reveal (GsAppRow *app_row)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+ gboolean sensitive = gtk_widget_get_sensitive (priv->button);
+
+ gtk_widget_set_visible (priv->button_revealer, sensitive || !priv->is_narrow);
+}
+
+static void
+gs_app_row_refresh_button (GsAppRow *app_row, gboolean missing_search_result)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+ GtkStyleContext *context;
+
+ /* disabled */
+ if (!priv->show_buttons) {
+ gs_app_row_update_button_reveal (app_row);
+ gtk_widget_set_visible (priv->button, FALSE);
+ return;
+ }
+
+ /* label */
+ switch (gs_app_get_state (priv->app)) {
+ case GS_APP_STATE_UNAVAILABLE:
+ gtk_widget_set_visible (priv->button, TRUE);
+ if (missing_search_result) {
+ /* TRANSLATORS: this is a button next to the search results that
+ * allows the application to be easily installed */
+ gs_progress_button_set_label (GS_PROGRESS_BUTTON (priv->button), _("Visit Website"));
+ gs_progress_button_set_icon_name (GS_PROGRESS_BUTTON (priv->button), NULL);
+ } else {
+ /* TRANSLATORS: this is a button next to the search results that
+ * allows the application to be easily installed.
+ * The ellipsis indicates that further steps are required */
+ gs_progress_button_set_label (GS_PROGRESS_BUTTON (priv->button), _("Install…"));
+ gs_progress_button_set_icon_name (GS_PROGRESS_BUTTON (priv->button), NULL);
+ }
+ break;
+ case GS_APP_STATE_QUEUED_FOR_INSTALL:
+ gtk_widget_set_visible (priv->button, TRUE);
+ /* TRANSLATORS: this is a button next to the search results that
+ * allows to cancel a queued install of the application */
+ gs_progress_button_set_label (GS_PROGRESS_BUTTON (priv->button), _("Cancel"));
+ gs_progress_button_set_icon_name (GS_PROGRESS_BUTTON (priv->button), "edit-delete-symbolic");
+ break;
+ case GS_APP_STATE_AVAILABLE:
+ case GS_APP_STATE_AVAILABLE_LOCAL:
+ gtk_widget_set_visible (priv->button, TRUE);
+ /* TRANSLATORS: this is a button next to the search results that
+ * allows the application to be easily installed */
+ gs_progress_button_set_label (GS_PROGRESS_BUTTON (priv->button), _("Install"));
+ gs_progress_button_set_icon_name (GS_PROGRESS_BUTTON (priv->button), "list-add-symbolic");
+ break;
+ case GS_APP_STATE_UPDATABLE_LIVE:
+ gtk_widget_set_visible (priv->button, TRUE);
+ if (priv->show_update) {
+ /* TRANSLATORS: this is a button in the updates panel
+ * that allows the app to be easily updated live */
+ gs_progress_button_set_label (GS_PROGRESS_BUTTON (priv->button), _("Update"));
+ gs_progress_button_set_icon_name (GS_PROGRESS_BUTTON (priv->button), "software-update-available-symbolic");
+ } else {
+ /* TRANSLATORS: this is a button next to the search results that
+ * allows the application to be easily removed */
+ gs_progress_button_set_label (GS_PROGRESS_BUTTON (priv->button), _("Uninstall"));
+ gs_progress_button_set_icon_name (GS_PROGRESS_BUTTON (priv->button), "app-remove-symbolic");
+ }
+ break;
+ case GS_APP_STATE_UPDATABLE:
+ case GS_APP_STATE_INSTALLED:
+ if (!gs_app_has_quirk (priv->app, GS_APP_QUIRK_COMPULSORY))
+ gtk_widget_set_visible (priv->button, TRUE);
+ /* TRANSLATORS: this is a button next to the search results that
+ * allows the application to be easily removed */
+ gs_progress_button_set_label (GS_PROGRESS_BUTTON (priv->button), _("Uninstall"));
+ gs_progress_button_set_icon_name (GS_PROGRESS_BUTTON (priv->button), "app-remove-symbolic");
+ break;
+ case GS_APP_STATE_INSTALLING:
+ gtk_widget_set_visible (priv->button, TRUE);
+ /* TRANSLATORS: this is a button next to the search results that
+ * shows the status of an application being installed */
+ gs_progress_button_set_label (GS_PROGRESS_BUTTON (priv->button), _("Installing"));
+ gs_progress_button_set_icon_name (GS_PROGRESS_BUTTON (priv->button), NULL);
+ break;
+ case GS_APP_STATE_REMOVING:
+ gtk_widget_set_visible (priv->button, TRUE);
+ /* TRANSLATORS: this is a button next to the search results that
+ * shows the status of an application being erased */
+ gs_progress_button_set_label (GS_PROGRESS_BUTTON (priv->button), _("Uninstalling"));
+ gs_progress_button_set_icon_name (GS_PROGRESS_BUTTON (priv->button), NULL);
+ break;
+ default:
+ break;
+ }
+
+ /* visible */
+ switch (gs_app_get_state (priv->app)) {
+ case GS_APP_STATE_UNAVAILABLE:
+ case GS_APP_STATE_QUEUED_FOR_INSTALL:
+ case GS_APP_STATE_AVAILABLE:
+ case GS_APP_STATE_AVAILABLE_LOCAL:
+ case GS_APP_STATE_UPDATABLE_LIVE:
+ case GS_APP_STATE_INSTALLING:
+ case GS_APP_STATE_REMOVING:
+ gtk_widget_set_visible (priv->button, TRUE);
+ break;
+ case GS_APP_STATE_UPDATABLE:
+ case GS_APP_STATE_INSTALLED:
+ gtk_widget_set_visible (priv->button,
+ !gs_app_has_quirk (priv->app,
+ GS_APP_QUIRK_COMPULSORY));
+ break;
+ default:
+ gtk_widget_set_visible (priv->button, FALSE);
+ break;
+ }
+
+ /* colorful */
+ context = gtk_widget_get_style_context (priv->button);
+ if (!priv->colorful) {
+ gtk_style_context_remove_class (context, "destructive-action");
+ } else {
+ switch (gs_app_get_state (priv->app)) {
+ case GS_APP_STATE_UPDATABLE:
+ case GS_APP_STATE_INSTALLED:
+ gtk_style_context_add_class (context, "destructive-action");
+ break;
+ case GS_APP_STATE_UPDATABLE_LIVE:
+ if (priv->show_update)
+ gtk_style_context_remove_class (context, "destructive-action");
+ else
+ gtk_style_context_add_class (context, "destructive-action");
+ break;
+ default:
+ gtk_style_context_remove_class (context, "destructive-action");
+ break;
+ }
+ }
+
+ /* always insensitive when in selection mode */
+ switch (gs_app_get_state (priv->app)) {
+ case GS_APP_STATE_INSTALLING:
+ case GS_APP_STATE_REMOVING:
+ gtk_widget_set_sensitive (priv->button, FALSE);
+ break;
+ default:
+ gtk_widget_set_sensitive (priv->button, TRUE);
+ break;
+ }
+
+ gs_app_row_update_button_reveal (app_row);
+}
+
+static void
+gs_app_row_actually_refresh (GsAppRow *app_row)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+ GtkStyleContext *context;
+ GString *str = NULL;
+ const gchar *tmp;
+ gboolean missing_search_result;
+ gboolean is_markup = FALSE;
+ guint64 size_installed_bytes = 0;
+ GsSizeType size_installed_type = GS_SIZE_TYPE_UNKNOWN;
+ g_autoptr(GIcon) icon = NULL;
+
+ if (priv->app == NULL)
+ return;
+
+ /* is this a missing search result from the extras page? */
+ missing_search_result = (gs_app_get_state (priv->app) == GS_APP_STATE_UNAVAILABLE &&
+ gs_app_get_url_missing (priv->app) != NULL);
+
+ /* do a fill bar for the current progress */
+ switch (gs_app_get_state (priv->app)) {
+ case GS_APP_STATE_INSTALLING:
+ gs_progress_button_set_progress (GS_PROGRESS_BUTTON (priv->button),
+ gs_app_get_progress (priv->app));
+ gs_progress_button_set_show_progress (GS_PROGRESS_BUTTON (priv->button), TRUE);
+ break;
+ default:
+ gs_progress_button_set_show_progress (GS_PROGRESS_BUTTON (priv->button), FALSE);
+ break;
+ }
+
+ /* join the description lines */
+ str = gs_app_row_get_description (app_row, &is_markup);
+ if (str != NULL) {
+ as_gstring_replace (str, "\n", " ");
+ if (is_markup)
+ gtk_label_set_markup (GTK_LABEL (priv->description_label), str->str);
+ else
+ gtk_label_set_label (GTK_LABEL (priv->description_label), str->str);
+ g_string_free (str, TRUE);
+ } else {
+ gtk_label_set_text (GTK_LABEL (priv->description_label), NULL);
+ }
+
+ /* add warning */
+ if (gs_app_has_quirk (priv->app, GS_APP_QUIRK_REMOVABLE_HARDWARE)) {
+ gtk_label_set_text (GTK_LABEL (priv->label_warning),
+ /* TRANSLATORS: during the update the device
+ * will restart into a special update-only mode */
+ _("Device cannot be used during update."));
+ gtk_widget_show (priv->label_warning);
+ }
+
+ /* where did this app come from */
+ if (priv->show_source) {
+ tmp = gs_app_get_origin_hostname (priv->app);
+ if (tmp != NULL) {
+ g_autofree gchar *origin_tmp = NULL;
+ /* TRANSLATORS: this refers to where the app came from */
+ origin_tmp = g_strdup_printf (_("Source: %s"), tmp);
+ gtk_label_set_label (GTK_LABEL (priv->label_origin), origin_tmp);
+ }
+ gtk_widget_set_visible (priv->label_origin, tmp != NULL);
+ } else {
+ gtk_widget_set_visible (priv->label_origin, FALSE);
+ }
+
+ /* installed tag */
+ if (!priv->show_buttons) {
+ switch (gs_app_get_state (priv->app)) {
+ case GS_APP_STATE_UPDATABLE:
+ case GS_APP_STATE_UPDATABLE_LIVE:
+ case GS_APP_STATE_INSTALLED:
+ gtk_widget_set_visible (priv->label_installed, priv->show_installed);
+ break;
+ default:
+ gtk_widget_set_visible (priv->label_installed, FALSE);
+ break;
+ }
+ } else {
+ gtk_widget_set_visible (priv->label_installed, FALSE);
+ }
+
+ /* name */
+ gtk_label_set_label (GTK_LABEL (priv->name_label),
+ gs_app_get_name (priv->app));
+
+ if (priv->show_update) {
+ const gchar *version_current = NULL;
+ const gchar *version_update = NULL;
+
+ /* current version */
+ tmp = gs_app_get_version_ui (priv->app);
+ if (tmp != NULL && tmp[0] != '\0') {
+ version_current = tmp;
+ gtk_label_set_label (GTK_LABEL (priv->version_current_label),
+ version_current);
+ gtk_widget_show (priv->version_current_label);
+ } else {
+ gtk_widget_hide (priv->version_current_label);
+ }
+
+ /* update version */
+ tmp = gs_app_get_update_version_ui (priv->app);
+ if (tmp != NULL && tmp[0] != '\0' &&
+ g_strcmp0 (tmp, version_current) != 0) {
+ version_update = tmp;
+ gtk_label_set_label (GTK_LABEL (priv->version_update_label),
+ version_update);
+ gtk_widget_show (priv->version_update_label);
+ } else {
+ gtk_widget_hide (priv->version_update_label);
+ }
+
+ /* have both: show arrow */
+ if (version_current != NULL && version_update != NULL &&
+ g_strcmp0 (version_current, version_update) != 0) {
+ gtk_widget_show (priv->version_arrow_label);
+ } else {
+ gtk_widget_hide (priv->version_arrow_label);
+ }
+
+ /* ensure the arrow is the right way round for the text direction,
+ * as arrows are not bidi-mirrored automatically
+ * See section 2 of http://www.unicode.org/L2/L2017/17438-bidi-math-fdbk.html */
+ switch (gtk_widget_get_direction (priv->version_box)) {
+ case GTK_TEXT_DIR_RTL:
+ gtk_label_set_label (GTK_LABEL (priv->version_arrow_label), "←");
+ break;
+ case GTK_TEXT_DIR_NONE:
+ case GTK_TEXT_DIR_LTR:
+ default:
+ gtk_label_set_label (GTK_LABEL (priv->version_arrow_label), "→");
+ break;
+ }
+
+ /* show the box if we have either of the versions */
+ if (version_current != NULL || version_update != NULL)
+ gtk_widget_show (priv->version_box);
+ else
+ gtk_widget_hide (priv->version_box);
+
+ gtk_widget_hide (priv->star);
+ } else {
+ gtk_widget_hide (priv->version_box);
+ if (missing_search_result || gs_app_get_rating (priv->app) <= 0 || !priv->show_rating) {
+ gtk_widget_hide (priv->star);
+ } else {
+ gtk_widget_show (priv->star);
+ gtk_widget_set_sensitive (priv->star, FALSE);
+ gs_star_widget_set_rating (GS_STAR_WIDGET (priv->star),
+ gs_app_get_rating (priv->app));
+ }
+ }
+
+ if (priv->show_update && gs_app_get_special_kind (priv->app) == GS_APP_SPECIAL_KIND_OS_UPDATE) {
+ gtk_label_set_label (GTK_LABEL (priv->system_updates_label), gs_app_get_summary (priv->app));
+ gtk_widget_show (priv->system_updates_label);
+ } else {
+ gtk_widget_hide (priv->system_updates_label);
+ }
+
+ /* pixbuf */
+ icon = gs_app_get_icon_for_size (priv->app,
+ gtk_image_get_pixel_size (GTK_IMAGE (priv->image)),
+ gtk_widget_get_scale_factor (priv->image),
+ "system-component-application");
+ gtk_image_set_from_gicon (GTK_IMAGE (priv->image), icon);
+
+ context = gtk_widget_get_style_context (priv->image);
+ if (missing_search_result)
+ gtk_style_context_add_class (context, "dimmer-label");
+ else
+ gtk_style_context_remove_class (context, "dimmer-label");
+
+ /* pending label */
+ switch (gs_app_get_state (priv->app)) {
+ case GS_APP_STATE_QUEUED_FOR_INSTALL:
+ gtk_widget_set_visible (priv->label, TRUE);
+ gtk_label_set_label (GTK_LABEL (priv->label), _("Pending"));
+ break;
+ case GS_APP_STATE_PENDING_INSTALL:
+ gtk_widget_set_visible (priv->label, TRUE);
+ gtk_label_set_label (GTK_LABEL (priv->label), _("Pending install"));
+ break;
+ case GS_APP_STATE_PENDING_REMOVE:
+ gtk_widget_set_visible (priv->label, TRUE);
+ gtk_label_set_label (GTK_LABEL (priv->label), _("Pending remove"));
+ break;
+ default:
+ gtk_widget_set_visible (priv->label, FALSE);
+ break;
+ }
+
+ /* spinner */
+ switch (gs_app_get_state (priv->app)) {
+ case GS_APP_STATE_REMOVING:
+ gtk_spinner_start (GTK_SPINNER (priv->spinner));
+ gtk_widget_set_visible (priv->spinner, TRUE);
+ break;
+ default:
+ gtk_widget_set_visible (priv->spinner, FALSE);
+ break;
+ }
+
+ /* button */
+ gs_app_row_refresh_button (app_row, missing_search_result);
+
+ /* hide buttons in the update list, unless the app is live updatable */
+ switch (gs_app_get_state (priv->app)) {
+ case GS_APP_STATE_UPDATABLE_LIVE:
+ case GS_APP_STATE_INSTALLING:
+ gtk_widget_set_visible (priv->button_box, TRUE);
+ break;
+ default:
+ gtk_widget_set_visible (priv->button_box, !priv->show_update);
+ break;
+ }
+
+ /* show the right size */
+ if (priv->show_installed_size) {
+ size_installed_type = gs_app_get_size_installed (priv->app, &size_installed_bytes);
+ }
+ if (size_installed_type == GS_SIZE_TYPE_VALID && size_installed_bytes > 0) {
+ g_autofree gchar *sizestr = NULL;
+ sizestr = g_format_size (size_installed_bytes);
+ gtk_label_set_label (GTK_LABEL (priv->label_app_size), sizestr);
+ gtk_widget_show (priv->label_app_size);
+ } else {
+ gtk_widget_hide (priv->label_app_size);
+ }
+
+ /* add warning */
+ if (priv->show_update) {
+ g_autoptr(GString) warning = g_string_new (NULL);
+ const gchar *renamed_from;
+
+ if (gs_app_has_quirk (priv->app, GS_APP_QUIRK_NEW_PERMISSIONS))
+ g_string_append (warning, _("Requires additional permissions"));
+
+ renamed_from = gs_app_get_renamed_from (priv->app);
+ if (renamed_from && g_strcmp0 (renamed_from, gs_app_get_name (priv->app)) != 0) {
+ if (warning->len > 0)
+ g_string_append (warning, "\n");
+ /* Translators: A message to indicate that an app has been renamed. The placeholder is the old human-readable name. */
+ g_string_append_printf (warning, _("Renamed from %s"), renamed_from);
+ }
+
+ if (warning->len > 0) {
+ gtk_label_set_text (GTK_LABEL (priv->label_warning), warning->str);
+ gtk_widget_show (priv->label_warning);
+ }
+ }
+
+ gtk_widget_set_visible (priv->box_tag,
+ gtk_widget_get_visible (priv->label_origin) ||
+ gtk_widget_get_visible (priv->label_installed) ||
+ gtk_widget_get_visible (priv->label_warning));
+
+ gtk_label_set_max_width_chars (GTK_LABEL (priv->name_label),
+ gtk_widget_get_visible (priv->description_label) ? 20 : -1);
+}
+
+static void
+finish_unreveal (GsAppRow *app_row)
+{
+ gtk_widget_hide (GTK_WIDGET (app_row));
+
+ g_signal_emit (app_row, signals[SIGNAL_UNREVEALED], 0);
+}
+
+static void
+child_unrevealed (GObject *revealer, GParamSpec *pspec, gpointer user_data)
+{
+ GsAppRow *app_row = user_data;
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ /* return immediately if we are in destruction (this doesn't, however,
+ * catch the case where we are being removed from a container without
+ * having been destroyed first.)
+ */
+ if (priv->app == NULL || !gtk_widget_get_mapped (GTK_WIDGET (app_row)))
+ return;
+
+ finish_unreveal (app_row);
+}
+
+static gboolean
+child_unrevealed_unmapped_cb (gpointer user_data)
+{
+ GsAppRow *app_row = user_data;
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ priv->unreveal_in_idle_id = 0;
+
+ finish_unreveal (app_row);
+
+ return G_SOURCE_REMOVE;
+}
+
+/**
+ * gs_app_row_unreveal:
+ * @app_row: a #GsAppRow
+ *
+ * Hide the row with an animation. Once the animation is done
+ * the GsAppRow:unrevealed signal is emitted. This handles
+ * the case when the widget is not mapped as well, in which case
+ * the GsAppRow:unrevealed signal is emitted from an idle
+ * callback, to ensure async nature of the function call and
+ * the signal emission.
+ *
+ * Calling the function multiple times has no effect.
+ **/
+void
+gs_app_row_unreveal (GsAppRow *app_row)
+{
+ GtkWidget *child;
+ GtkWidget *revealer;
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ child = gtk_list_box_row_get_child (GTK_LIST_BOX_ROW (app_row));
+
+ /* This means the row is already hiding */
+ if (GTK_IS_REVEALER (child))
+ return;
+
+ gtk_widget_set_sensitive (child, FALSE);
+
+ /* Revealer does not animate when the widget is not mapped */
+ if (!gtk_widget_get_mapped (GTK_WIDGET (app_row))) {
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+ if (priv->unreveal_in_idle_id == 0)
+ priv->unreveal_in_idle_id = g_idle_add_full (G_PRIORITY_HIGH, child_unrevealed_unmapped_cb, app_row, NULL);
+ return;
+ }
+
+ revealer = gtk_revealer_new ();
+ gtk_revealer_set_reveal_child (GTK_REVEALER (revealer), TRUE);
+ gtk_widget_show (revealer);
+
+ g_object_ref (child);
+ gtk_list_box_row_set_child (GTK_LIST_BOX_ROW (app_row), revealer);
+ gtk_revealer_set_child (GTK_REVEALER (revealer), child);
+ g_object_unref (child);
+
+ g_signal_connect (revealer, "notify::child-revealed",
+ G_CALLBACK (child_unrevealed), app_row);
+ gtk_revealer_set_reveal_child (GTK_REVEALER (revealer), FALSE);
+}
+
+GsApp *
+gs_app_row_get_app (GsAppRow *app_row)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+ g_return_val_if_fail (GS_IS_APP_ROW (app_row), NULL);
+ return priv->app;
+}
+
+static gboolean
+gs_app_row_refresh_idle_cb (gpointer user_data)
+{
+ GsAppRow *app_row = GS_APP_ROW (user_data);
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+ priv->pending_refresh_id = 0;
+ gs_app_row_actually_refresh (app_row);
+ return G_SOURCE_REMOVE;
+}
+
+/* Schedule an idle call to gs_app_row_actually_refresh() unless one’s already pending. */
+static void
+gs_app_row_schedule_refresh (GsAppRow *app_row)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ if (priv->pending_refresh_id > 0)
+ return;
+ priv->pending_refresh_id = g_idle_add (gs_app_row_refresh_idle_cb, app_row);
+}
+
+static void
+gs_app_row_notify_props_changed_cb (GsApp *app,
+ GParamSpec *pspec,
+ GsAppRow *app_row)
+{
+ gs_app_row_schedule_refresh (app_row);
+}
+
+static void
+gs_app_row_set_app (GsAppRow *app_row, GsApp *app)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ priv->app = g_object_ref (app);
+
+ g_signal_connect_object (priv->app, "notify::state",
+ G_CALLBACK (gs_app_row_notify_props_changed_cb),
+ app_row, 0);
+ g_signal_connect_object (priv->app, "notify::rating",
+ G_CALLBACK (gs_app_row_notify_props_changed_cb),
+ app_row, 0);
+ g_signal_connect_object (priv->app, "notify::progress",
+ G_CALLBACK (gs_app_row_notify_props_changed_cb),
+ app_row, 0);
+ g_signal_connect_object (priv->app, "notify::allow-cancel",
+ G_CALLBACK (gs_app_row_notify_props_changed_cb),
+ app_row, 0);
+
+ gs_app_row_schedule_refresh (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_APP]);
+}
+
+static void
+gs_app_row_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
+{
+ GsAppRow *app_row = GS_APP_ROW (object);
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ switch ((GsAppRowProperty) prop_id) {
+ case PROP_APP:
+ g_value_set_object (value, priv->app);
+ break;
+ case PROP_COLORFUL:
+ g_value_set_boolean (value, priv->colorful);
+ break;
+ case PROP_SHOW_DESCRIPTION:
+ g_value_set_boolean (value, gs_app_row_get_show_description (app_row));
+ break;
+ case PROP_SHOW_SOURCE:
+ g_value_set_boolean (value, priv->show_source);
+ break;
+ case PROP_SHOW_BUTTONS:
+ g_value_set_boolean (value, priv->show_buttons);
+ break;
+ case PROP_SHOW_RATING:
+ g_value_set_boolean (value, priv->show_rating);
+ break;
+ case PROP_SHOW_UPDATE:
+ g_value_set_boolean (value, priv->show_update);
+ break;
+ case PROP_SHOW_INSTALLED_SIZE:
+ g_value_set_boolean (value, priv->show_installed_size);
+ break;
+ case PROP_SHOW_INSTALLED:
+ g_value_set_boolean (value, priv->show_installed);
+ break;
+ case PROP_IS_NARROW:
+ g_value_set_boolean (value, gs_app_row_get_is_narrow (app_row));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gs_app_row_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
+{
+ GsAppRow *app_row = GS_APP_ROW (object);
+
+ switch ((GsAppRowProperty) prop_id) {
+ case PROP_APP:
+ gs_app_row_set_app (app_row, g_value_get_object (value));
+ break;
+ case PROP_COLORFUL:
+ gs_app_row_set_colorful (app_row, g_value_get_boolean (value));
+ break;
+ case PROP_SHOW_DESCRIPTION:
+ gs_app_row_set_show_description (app_row, g_value_get_boolean (value));
+ break;
+ case PROP_SHOW_SOURCE:
+ gs_app_row_set_show_source (app_row, g_value_get_boolean (value));
+ break;
+ case PROP_SHOW_BUTTONS:
+ gs_app_row_set_show_buttons (app_row, g_value_get_boolean (value));
+ break;
+ case PROP_SHOW_RATING:
+ gs_app_row_set_show_rating (app_row, g_value_get_boolean (value));
+ break;
+ case PROP_SHOW_UPDATE:
+ gs_app_row_set_show_update (app_row, g_value_get_boolean (value));
+ break;
+ case PROP_SHOW_INSTALLED_SIZE:
+ gs_app_row_set_show_installed_size (app_row, g_value_get_boolean (value));
+ break;
+ case PROP_SHOW_INSTALLED:
+ gs_app_row_set_show_installed (app_row, g_value_get_boolean (value));
+ break;
+ case PROP_IS_NARROW:
+ gs_app_row_set_is_narrow (app_row, g_value_get_boolean (value));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gs_app_row_dispose (GObject *object)
+{
+ GsAppRow *app_row = GS_APP_ROW (object);
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ if (priv->app)
+ g_signal_handlers_disconnect_by_func (priv->app, gs_app_row_notify_props_changed_cb, app_row);
+
+ g_clear_object (&priv->app);
+ g_clear_handle_id (&priv->pending_refresh_id, g_source_remove);
+ g_clear_handle_id (&priv->unreveal_in_idle_id, g_source_remove);
+
+ G_OBJECT_CLASS (gs_app_row_parent_class)->dispose (object);
+}
+
+static void
+gs_app_row_class_init (GsAppRowClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+
+ object_class->get_property = gs_app_row_get_property;
+ object_class->set_property = gs_app_row_set_property;
+ object_class->dispose = gs_app_row_dispose;
+
+ /**
+ * GsAppRow:app:
+ *
+ * The #GsApp to show in this row.
+ *
+ * Since: 3.38
+ */
+ obj_props[PROP_APP] =
+ g_param_spec_object ("app", NULL, NULL,
+ GS_TYPE_APP,
+ G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
+
+ /**
+ * GsAppRow:colorful:
+ *
+ * Whether the buttons can be colorized in the row.
+ *
+ * Since: 42.1
+ */
+ obj_props[PROP_COLORFUL] =
+ g_param_spec_boolean ("colorful", NULL, NULL,
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * GsAppRow:show-description:
+ *
+ * Show the description of the app in the row.
+ *
+ * Since: 41
+ */
+ obj_props[PROP_SHOW_DESCRIPTION] =
+ g_param_spec_boolean ("show-description", NULL, NULL,
+ TRUE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * GsAppRow:show-source:
+ *
+ * Show the source of the app in the row.
+ *
+ * Since: 3.38
+ */
+ obj_props[PROP_SHOW_SOURCE] =
+ g_param_spec_boolean ("show-source", NULL, NULL,
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * GsAppRow:show-buttons:
+ *
+ * Show buttons (such as Install, Cancel or Update) in the app row.
+ *
+ * Since: 3.38
+ */
+ obj_props[PROP_SHOW_BUTTONS] =
+ g_param_spec_boolean ("show-buttons", NULL, NULL,
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * GsAppRow:show-rating:
+ *
+ * Show app rating in the app row.
+ *
+ * Since: 42.1
+ */
+ obj_props[PROP_SHOW_RATING] =
+ g_param_spec_boolean ("show-rating", NULL, NULL,
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * GsAppRow:show-update:
+ *
+ * Show update (version) information in the app row.
+ *
+ * Since: 42.1
+ */
+ obj_props[PROP_SHOW_UPDATE] =
+ g_param_spec_boolean ("show-update", NULL, NULL,
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * GsAppRow:show-installed:
+ *
+ * Show an "Installed" check in the app row, when the app is installed.
+ *
+ * Since: 42.1
+ */
+ obj_props[PROP_SHOW_INSTALLED] =
+ g_param_spec_boolean ("show-installed", NULL, NULL,
+ TRUE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * GsAppRow:show-installed-size:
+ *
+ * Show the installed size of the app in the row.
+ *
+ * Since: 3.38
+ */
+ obj_props[PROP_SHOW_INSTALLED_SIZE] =
+ g_param_spec_boolean ("show-installed-size", NULL, NULL,
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * GsAppRow:is-narrow:
+ *
+ * Whether the row is in narrow mode.
+ *
+ * In narrow mode, the row will take up less horizontal space, doing so
+ * by e.g. using icons rather than labels in buttons. This is needed to
+ * keep the UI useable on small form-factors like smartphones.
+ *
+ * Since: 41
+ */
+ obj_props[PROP_IS_NARROW] =
+ g_param_spec_boolean ("is-narrow", NULL, NULL,
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY);
+
+ g_object_class_install_properties (object_class, G_N_ELEMENTS (obj_props), obj_props);
+
+ signals [SIGNAL_BUTTON_CLICKED] =
+ g_signal_new ("button-clicked",
+ G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST,
+ G_STRUCT_OFFSET (GsAppRowClass, button_clicked),
+ NULL, NULL, g_cclosure_marshal_VOID__VOID,
+ G_TYPE_NONE, 0);
+
+ signals [SIGNAL_UNREVEALED] =
+ g_signal_new ("unrevealed",
+ G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST,
+ G_STRUCT_OFFSET (GsAppRowClass, unrevealed),
+ NULL, NULL, g_cclosure_marshal_VOID__VOID,
+ G_TYPE_NONE, 0);
+
+ gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/Software/gs-app-row.ui");
+
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, image);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, name_box);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, name_label);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, version_box);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, version_current_label);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, version_arrow_label);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, version_update_label);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, system_updates_label);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, star);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, description_label);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, button_box);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, button_revealer);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, button);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, spinner);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, label);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, box_tag);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, label_warning);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, label_origin);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, label_installed);
+ gtk_widget_class_bind_template_child_private (widget_class, GsAppRow, label_app_size);
+}
+
+static void
+button_clicked (GtkWidget *widget, GsAppRow *app_row)
+{
+ g_signal_emit (app_row, signals[SIGNAL_BUTTON_CLICKED], 0);
+}
+
+static void
+gs_app_row_init (GsAppRow *app_row)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ priv->show_description = TRUE;
+ priv->show_installed = TRUE;
+
+ gtk_widget_init_template (GTK_WIDGET (app_row));
+
+ g_signal_connect (priv->button, "clicked",
+ G_CALLBACK (button_clicked), app_row);
+
+ /* A fix for this is included in 4.6.4, apply workaround, if not running with new-enough gtk. */
+ if (gtk_get_major_version () < 4 ||
+ (gtk_get_major_version () == 4 && gtk_get_minor_version () < 6) ||
+ (gtk_get_major_version () == 4 && gtk_get_minor_version () == 6 && gtk_get_micro_version () < 4)) {
+ g_object_set (G_OBJECT (priv->name_label),
+ "wrap", FALSE,
+ "lines", 1,
+ NULL);
+ g_object_set (G_OBJECT (priv->description_label),
+ "wrap", FALSE,
+ "lines", 1,
+ NULL);
+ g_object_set (G_OBJECT (priv->label_warning),
+ "wrap", FALSE,
+ "lines", 1,
+ NULL);
+ g_object_set (G_OBJECT (priv->system_updates_label),
+ "wrap", FALSE,
+ "lines", 1,
+ NULL);
+ }
+}
+
+void
+gs_app_row_set_size_groups (GsAppRow *app_row,
+ GtkSizeGroup *name,
+ GtkSizeGroup *button_label,
+ GtkSizeGroup *button_image)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ if (name != NULL)
+ gtk_size_group_add_widget (name, priv->name_box);
+ gs_progress_button_set_size_groups (GS_PROGRESS_BUTTON (priv->button), button_label, button_image);
+}
+
+void
+gs_app_row_set_colorful (GsAppRow *app_row, gboolean colorful)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ if ((!priv->colorful) == (!colorful))
+ return;
+
+ priv->colorful = colorful;
+ gs_app_row_schedule_refresh (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_COLORFUL]);
+}
+
+void
+gs_app_row_set_show_buttons (GsAppRow *app_row, gboolean show_buttons)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ if ((!priv->show_buttons) == (!show_buttons))
+ return;
+
+ priv->show_buttons = show_buttons;
+ gs_app_row_schedule_refresh (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_SHOW_BUTTONS]);
+}
+
+void
+gs_app_row_set_show_rating (GsAppRow *app_row, gboolean show_rating)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ if ((!priv->show_rating) == (!show_rating))
+ return;
+
+ priv->show_rating = show_rating;
+ gs_app_row_schedule_refresh (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_SHOW_RATING]);
+}
+
+/**
+ * gs_app_row_get_show_description:
+ * @app_row: a #GsAppRow
+ *
+ * Get the value of #GsAppRow:show-description.
+ *
+ * Returns: %TRUE if the description is shown, %FALSE otherwise
+ *
+ * Since: 41
+ */
+gboolean
+gs_app_row_get_show_description (GsAppRow *app_row)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_val_if_fail (GS_IS_APP_ROW (app_row), FALSE);
+
+ return priv->show_description;
+}
+
+/**
+ * gs_app_row_set_show_description:
+ * @app_row: a #GsAppRow
+ * @show_description: %TRUE to show the description, %FALSE otherwise
+ *
+ * Set the value of #GsAppRow:show-description.
+ *
+ * Since: 41
+ */
+void
+gs_app_row_set_show_description (GsAppRow *app_row, gboolean show_description)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ show_description = !!show_description;
+
+ if (priv->show_description == show_description)
+ return;
+
+ priv->show_description = show_description;
+ gs_app_row_schedule_refresh (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_SHOW_DESCRIPTION]);
+}
+
+void
+gs_app_row_set_show_source (GsAppRow *app_row, gboolean show_source)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ if ((!priv->show_source) == (!show_source))
+ return;
+
+ priv->show_source = show_source;
+ gs_app_row_schedule_refresh (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_SHOW_SOURCE]);
+}
+
+void
+gs_app_row_set_show_installed_size (GsAppRow *app_row, gboolean show_size)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ if ((!priv->show_installed_size) == (!show_size))
+ return;
+
+ priv->show_installed_size = show_size;
+ gs_app_row_schedule_refresh (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_SHOW_INSTALLED_SIZE]);
+}
+
+/**
+ * gs_app_row_get_is_narrow:
+ * @app_row: a #GsAppRow
+ *
+ * Get the value of #GsAppRow:is-narrow.
+ *
+ * Retruns: %TRUE if the row is in narrow mode, %FALSE otherwise
+ *
+ * Since: 41
+ */
+gboolean
+gs_app_row_get_is_narrow (GsAppRow *app_row)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_val_if_fail (GS_IS_APP_ROW (app_row), FALSE);
+
+ return priv->is_narrow;
+}
+
+/**
+ * gs_app_row_set_is_narrow:
+ * @app_row: a #GsAppRow
+ * @is_narrow: %TRUE to set the row in narrow mode, %FALSE otherwise
+ *
+ * Set the value of #GsAppRow:is-narrow.
+ *
+ * Since: 41
+ */
+void
+gs_app_row_set_is_narrow (GsAppRow *app_row, gboolean is_narrow)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ is_narrow = !!is_narrow;
+
+ if (priv->is_narrow == is_narrow)
+ return;
+
+ priv->is_narrow = is_narrow;
+ gs_app_row_update_button_reveal (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_IS_NARROW]);
+}
+
+/**
+ * gs_app_row_set_show_update:
+ *
+ * Only really useful for the update panel to call
+ **/
+void
+gs_app_row_set_show_update (GsAppRow *app_row, gboolean show_update)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ if ((!priv->show_update) == (!show_update))
+ return;
+
+ priv->show_update = show_update;
+ gs_app_row_schedule_refresh (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_SHOW_UPDATE]);
+}
+
+/**
+ * gs_app_row_set_show_installed:
+ * @app_row: a #GsAppRow
+ * @show_installed: value to set
+ *
+ * Set whether to show "installed" label. Default is %TRUE. This has effect only
+ * when not showing buttons (gs_app_row_set_show_buttons()).
+ *
+ * Since: 42.1
+ **/
+void
+gs_app_row_set_show_installed (GsAppRow *app_row,
+ gboolean show_installed)
+{
+ GsAppRowPrivate *priv = gs_app_row_get_instance_private (app_row);
+
+ g_return_if_fail (GS_IS_APP_ROW (app_row));
+
+ if ((!show_installed) != (!priv->show_installed)) {
+ priv->show_installed = show_installed;
+ gs_app_row_schedule_refresh (app_row);
+ g_object_notify_by_pspec (G_OBJECT (app_row), obj_props[PROP_SHOW_INSTALLED]);
+ }
+}
+
+GtkWidget *
+gs_app_row_new (GsApp *app)
+{
+ g_return_val_if_fail (GS_IS_APP (app), NULL);
+
+ return g_object_new (GS_TYPE_APP_ROW,
+ "app", app,
+ NULL);
+}