diff options
Diffstat (limited to '')
-rw-r--r-- | src/gs-app-row.c | 1240 |
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); +} |