qubes-installer-qubes-os/anaconda/widgets/src/DiskOverview.c
Marek Marczykowski-Górecki 6bc5671491
anaconda: update to 25.20.9-1
Apply:
  git diff --full-index --binary anaconda-23.19.10-1..anaconda-25.20.9-1

And resolve conflicts.

QubesOS/qubes-issues#2574
2017-02-14 02:36:20 +01:00

558 lines
21 KiB
C

/*
* Copyright (C) 2011-2014 Red Hat, Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "config.h"
#include <atk/atk.h>
#include <gdk/gdk.h>
#include <gio/gio.h>
#include <string.h>
#include "DiskOverview.h"
#include "intl.h"
#include "widgets-common.h"
/**
* SECTION: DiskOverview
* @title: AnacondaDiskOverview
* @short_description: A widget that displays basic information about a disk
*
* A #AnacondaDiskOverview is a potentially selectable widget that displays a
* disk device's size, kind, and a prominant icon based on the kind of device.
* This widget can come in different sizes, depending on where it needs to be
* used.
*
* As a #AnacondaDiskOverview is a subclass of a #GtkEventBox, any signals
* may be caught. The #GtkWidget::button-press-event signal is already
* handled internally to change the background color, but may also be handled
* by user code in order to take some action based on the disk clicked upon.
*
* # CSS nodes
*
* |[<!-- language="plain" -->
* AnacondaDiskOverview
* ├── #anaconda-disk-capacity
* ├── #anaconda-disk-kind
* ├── #anaconda-disk-description
* ├── #anaconda-disk-name[.anaconda-disk-details]
* ├── #anaconda-disk-separator[.anaconda-disk-details]
* ╰── #anaconda-disk-free[.anaconda-disk-details]
* ]|
*
* The internal widgets are accessible by name for the purposes of CSS
* selectors.
*
* - anaconda-disk-capacity
*
* The total size of the disk, plus units
*
* - anaconda-disk-kind
*
* The icon displayed for the type of disk this widget is describing
*
* - anaconda-disk-description
*
* The string describing the disk
*
* - anaconda-disk-name
*
* The name of the device
*
* - anaconda-disk-free
*
* The amount of free, unpartitioned space on the disk, plus units
*
* - anaconda-disk-separator
*
* A string separating the name and free labels.
*
* Additionally, anaconda-disk-name, anaconda-disk-free, and
* anaconda-disk-separator are all accessible via the .anaconda-disk-details
* class.
*/
enum {
PROP_DESCRIPTION = 1,
PROP_KIND,
PROP_FREE,
PROP_CAPACITY,
PROP_NAME,
PROP_POPUP_INFO
};
/* Defaults for each property. */
#define DEFAULT_DESCRIPTION N_("New Device")
#define DEFAULT_KIND "drive-harddisk"
#define DEFAULT_CAPACITY N_("0 MB")
#define DEFAULT_FREE N_("0 MB")
#define DEFAULT_NAME ""
#define DEFAULT_POPUP_INFO ""
#define ICON_SIZE 48
struct _AnacondaDiskOverviewPrivate {
GtkWidget *grid;
GtkWidget *kind_icon;
GtkWidget *description_label;
GtkWidget *capacity_label, *free_label;
GtkWidget *name_label;
GtkWidget *tooltip;
GdkCursor *cursor;
gchar *kind;
gboolean chosen;
};
G_DEFINE_TYPE(AnacondaDiskOverview, anaconda_disk_overview, GTK_TYPE_EVENT_BOX)
gboolean anaconda_disk_overview_clicked(AnacondaDiskOverview *widget, GdkEvent *event);
static void anaconda_disk_overview_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec);
static void anaconda_disk_overview_set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec);
static void anaconda_disk_overview_toggle_background(AnacondaDiskOverview *widget);
static void anaconda_disk_overview_realize(GtkWidget *widget, gpointer user_data);
static void anaconda_disk_overview_finalize(GObject *object);
static gboolean anaconda_disk_overview_focus_changed(GtkWidget *widget, GdkEventFocus *event, gpointer user_data);
static void anaconda_disk_overview_class_init(AnacondaDiskOverviewClass *klass) {
GObjectClass *object_class = G_OBJECT_CLASS(klass);
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS(klass);
object_class->set_property = anaconda_disk_overview_set_property;
object_class->get_property = anaconda_disk_overview_get_property;
object_class->finalize = anaconda_disk_overview_finalize;
/**
* AnacondaDiskOverview:kind:
*
* The #AnacondaDiskOverview:kind string specifies what type of disk device this is, used to
* figure out what icon to be displayed. This should be something like
* "drive-harddisk", "drive-removable-media", etc.
*
* Since: 1.0
*/
g_object_class_install_property(object_class,
PROP_KIND,
g_param_spec_string("kind",
P_("kind"),
P_("Drive kind icon"),
DEFAULT_KIND,
G_PARAM_READWRITE));
/**
* AnacondaDiskOverview:description:
*
* The #AnacondaDiskOverview:description string is a very basic description of the device
* and is displayed in bold letters under the icon.
*
* Since: 1.0
*/
g_object_class_install_property(object_class,
PROP_DESCRIPTION,
g_param_spec_string("description",
P_("Description"),
P_("The drive description"),
DEFAULT_DESCRIPTION,
G_PARAM_READWRITE));
/**
* AnacondaDiskOverview:capacity:
*
* The #AnacondaDiskOverview:capacity string is the total size of the disk, plus units.
*
* Since: 1.0
*/
g_object_class_install_property(object_class,
PROP_CAPACITY,
g_param_spec_string("capacity",
P_("Capacity"),
P_("The drive size (including units)"),
DEFAULT_CAPACITY,
G_PARAM_READWRITE));
/**
* AnacondaDiskOverview:free:
*
* The #AnacondaDiskOverview:free string is the amount of free, unpartitioned space on the disk,
* plus units.
*
* Since: 1.0
*/
g_object_class_install_property(object_class,
PROP_FREE,
g_param_spec_string("free",
P_("Free space"),
P_("The drive's unpartitioned free space (including units)"),
DEFAULT_FREE,
G_PARAM_READWRITE));
/**
* AnacondaDiskOverview:name:
*
* The #AnacondaDiskOverview:name string provides this device's node name (like 'sda'). Note
* that these names aren't guaranteed to be consistent across reboots but
* their use is so ingrained that we need to continue displaying them.
*
* Since: 1.0
*/
g_object_class_install_property(object_class,
PROP_NAME,
g_param_spec_string("name",
P_("Device node name"),
P_("Device node name"),
DEFAULT_NAME,
G_PARAM_READWRITE));
/**
* AnacondaDiskOverview:popup-info:
*
* The #AnacondaDiskOverview:popup-info string is text that should appear in a tooltip when the
* #AnacondaDiskOverview is hovered over. For normal disk devices, this
* could be available space information. For more complex devics, this
* could be WWID, LUN, and so forth.
*
* Since: 1.0
*/
g_object_class_install_property(object_class,
PROP_POPUP_INFO,
g_param_spec_string("popup-info",
P_("Detailed Disk Information"),
P_("Tooltip information for this drive"),
DEFAULT_POPUP_INFO,
G_PARAM_READWRITE));
g_type_class_add_private(object_class, sizeof(AnacondaDiskOverviewPrivate));
gtk_widget_class_set_css_name(widget_class, "AnacondaDiskOverview");
}
/**
* anaconda_disk_overview_new:
*
* Creates a new #AnacondaDiskOverview, which is a potentially selectable
* widget that displays basic information about a single storage device, be
* that a regular disk or a more complicated network device.
*
* Returns: A new #AnacondaDiskOverview.
*/
GtkWidget *anaconda_disk_overview_new() {
return g_object_new(ANACONDA_TYPE_DISK_OVERVIEW, NULL);
}
static void set_icon(AnacondaDiskOverview *widget, const char *icon_name) {
GError *err = NULL;
GIcon *base_icon, *emblem_icon, *icon;
GEmblem *emblem = NULL;
if (!icon_name)
return;
if (widget->priv->kind_icon)
gtk_widget_destroy(widget->priv->kind_icon);
if (widget->priv->chosen) {
base_icon = g_icon_new_for_string(icon_name, &err);
if (!base_icon) {
fprintf(stderr, "could not create icon: %s\n", err->message);
g_error_free(err);
return;
}
emblem_icon = g_icon_new_for_string("resource://" ANACONDA_RESOURCE_PATH "anaconda-selected-icon.svg", &err);
if (!emblem_icon) {
fprintf(stderr, "could not create emblem: %s\n", err->message);
g_error_free(err);
}
else {
emblem = g_emblem_new(emblem_icon);
}
icon = g_emblemed_icon_new(base_icon, emblem);
g_object_unref(base_icon);
}
else {
icon = g_icon_new_for_string(icon_name, &err);
if (!icon) {
fprintf(stderr, "could not create icon: %s\n", err->message);
g_error_free(err);
return;
}
}
widget->priv->kind_icon = gtk_image_new_from_gicon(icon, GTK_ICON_SIZE_DIALOG);
gtk_image_set_pixel_size(GTK_IMAGE(widget->priv->kind_icon), ICON_SIZE);
}
/* Initialize the widgets in a newly allocated DiskOverview. */
static void anaconda_disk_overview_init(AnacondaDiskOverview *widget) {
AtkObject *atk;
AtkRole role;
GtkWidget *separator;
GtkStyleContext *context;
widget->priv = G_TYPE_INSTANCE_GET_PRIVATE(widget,
ANACONDA_TYPE_DISK_OVERVIEW,
AnacondaDiskOverviewPrivate);
gtk_widget_set_valign(GTK_WIDGET(widget), GTK_ALIGN_CENTER);
/* Allow tabbing from one DiskOverview to the next. */
gtk_widget_set_can_focus(GTK_WIDGET(widget), TRUE);
gtk_widget_add_events(GTK_WIDGET(widget), GDK_FOCUS_CHANGE_MASK|GDK_KEY_RELEASE_MASK);
g_signal_connect(widget, "focus-in-event", G_CALLBACK(anaconda_disk_overview_focus_changed), NULL);
g_signal_connect(widget, "focus-out-event", G_CALLBACK(anaconda_disk_overview_focus_changed), NULL);
/* Set "hand" cursor shape when over the selector */
widget->priv->cursor = gdk_cursor_new_for_display(gdk_display_get_default(), GDK_HAND2);
g_signal_connect(widget, "realize", G_CALLBACK(anaconda_disk_overview_realize), NULL);
/* Set some properties. */
widget->priv->chosen = FALSE;
/* Create the grid. */
widget->priv->grid = gtk_grid_new();
gtk_grid_set_row_spacing(GTK_GRID(widget->priv->grid), 2);
gtk_grid_set_column_spacing(GTK_GRID(widget->priv->grid), 6);
gtk_container_set_border_width(GTK_CONTAINER(widget->priv->grid), 6);
/* Create the capacity label. */
widget->priv->capacity_label = gtk_label_new(_(DEFAULT_CAPACITY));
gtk_widget_set_name(widget->priv->capacity_label, "anaconda-disk-capacity");
/* Create the spoke's icon. */
set_icon(widget, DEFAULT_KIND);
/* Create the description label. */
widget->priv->description_label = gtk_label_new(_(DEFAULT_DESCRIPTION));
gtk_label_set_justify(GTK_LABEL(widget->priv->description_label), GTK_JUSTIFY_CENTER);
gtk_widget_set_name(widget->priv->description_label, "anaconda-disk-description");
/* Create the name label. */
widget->priv->name_label = gtk_label_new(NULL);
gtk_widget_set_halign(widget->priv->name_label, GTK_ALIGN_END);
gtk_widget_set_name(widget->priv->name_label, "anaconda-disk-name");
/* Create the free space label. */
widget->priv->free_label = gtk_label_new(_(DEFAULT_FREE));
gtk_widget_set_halign(widget->priv->free_label, GTK_ALIGN_START);
gtk_widget_set_name(widget->priv->free_label, "anaconda-disk-free");
/* Create the separator label. */
separator = gtk_label_new("/");
gtk_widget_set_name(separator, "anaconda-disk-separator");
/* Apply the details style class to the widgets in the bottom row */
context = gtk_widget_get_style_context(widget->priv->name_label);
gtk_style_context_add_class(context, "anaconda-disk-details");
context = gtk_widget_get_style_context(widget->priv->free_label);
gtk_style_context_add_class(context, "anaconda-disk-details");
context = gtk_widget_get_style_context(separator);
gtk_style_context_add_class(context, "anaconda-disk-details");
/* Add everything to the grid, add the grid to the widget. */
gtk_grid_attach(GTK_GRID(widget->priv->grid), widget->priv->capacity_label, 0, 0, 3, 1);
gtk_grid_attach(GTK_GRID(widget->priv->grid), widget->priv->kind_icon, 0, 1, 3, 1);
gtk_grid_attach(GTK_GRID(widget->priv->grid), widget->priv->description_label, 0, 2, 3, 1);
gtk_grid_attach(GTK_GRID(widget->priv->grid), widget->priv->name_label, 0, 3, 1, 1);
gtk_grid_attach(GTK_GRID(widget->priv->grid), separator, 1, 3, 1, 1);
gtk_grid_attach(GTK_GRID(widget->priv->grid), widget->priv->free_label, 2, 3, 1, 1);
gtk_container_add(GTK_CONTAINER(widget), widget->priv->grid);
/* We need to handle button-press-event in order to change the background color. */
g_signal_connect(widget, "button-press-event", G_CALLBACK(anaconda_disk_overview_clicked), NULL);
/* And this one is to handle when you select a DiskOverview via keyboard. */
g_signal_connect(widget, "key-release-event", G_CALLBACK(anaconda_disk_overview_clicked), NULL);
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
/* No existing role is appropriate for this, so ignore the warning raised
by registering a new role. */
role = atk_role_register("disk overview");
G_GNUC_END_IGNORE_DEPRECATIONS
atk = gtk_widget_get_accessible(GTK_WIDGET(widget));
atk_object_set_role(atk, role);
/* Apply the "fallback" style so that the widgets are colored correctly when
* selected, insensitive, etc */
context = gtk_widget_get_style_context(GTK_WIDGET(widget));
gtk_style_context_add_class(context, "gtkstyle-fallback");
/* Add the style data to widgets with stylesheets */
anaconda_widget_apply_stylesheet(widget->priv->capacity_label, "DiskOverview-capacity");
anaconda_widget_apply_stylesheet(widget->priv->description_label, "DiskOverview-description");
anaconda_widget_apply_stylesheet(widget->priv->name_label, "DiskOverview-name");
anaconda_widget_apply_stylesheet(widget->priv->free_label, "DiskOverview-free");
}
gboolean anaconda_disk_overview_clicked(AnacondaDiskOverview *widget, GdkEvent *event) {
/* This handler runs for mouse presses and key releases. For key releases, it only
* runs for activate-type keys (enter, space, etc.).
*/
gtk_widget_grab_focus(GTK_WIDGET(widget));
if (event->type != GDK_BUTTON_PRESS && event->type != GDK_KEY_RELEASE)
return FALSE;
else if (event->type == GDK_KEY_RELEASE &&
(event->key.keyval != GDK_KEY_space && event->key.keyval != GDK_KEY_Return &&
event->key.keyval != GDK_KEY_ISO_Enter && event->key.keyval != GDK_KEY_KP_Enter &&
event->key.keyval != GDK_KEY_KP_Space))
return FALSE;
widget->priv->chosen = !widget->priv->chosen;
anaconda_disk_overview_toggle_background(widget);
return FALSE;
}
static void anaconda_disk_overview_toggle_background(AnacondaDiskOverview *widget) {
set_icon(widget, widget->priv->kind);
gtk_grid_attach(GTK_GRID(widget->priv->grid), widget->priv->kind_icon, 0, 1, 3, 1);
gtk_widget_show(widget->priv->kind_icon);
}
static void anaconda_disk_overview_finalize(GObject *object) {
AnacondaDiskOverview *widget = ANACONDA_DISK_OVERVIEW(object);
g_object_unref(widget->priv->cursor);
G_OBJECT_CLASS(anaconda_disk_overview_parent_class)->finalize(object);
}
static void anaconda_disk_overview_realize(GtkWidget *widget, gpointer user_data) {
AnacondaDiskOverview *overview = ANACONDA_DISK_OVERVIEW(widget);
gdk_window_set_cursor(gtk_widget_get_window(widget), overview->priv->cursor);
}
static void anaconda_disk_overview_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) {
AnacondaDiskOverview *widget = ANACONDA_DISK_OVERVIEW(object);
AnacondaDiskOverviewPrivate *priv = widget->priv;
switch(prop_id) {
case PROP_DESCRIPTION:
g_value_set_string (value, gtk_label_get_text(GTK_LABEL(priv->description_label)));
break;
case PROP_KIND:
g_value_set_object (value, (GObject *)priv->kind_icon);
break;
case PROP_FREE:
g_value_set_string (value, gtk_label_get_text(GTK_LABEL(priv->free_label)));
break;
case PROP_CAPACITY:
g_value_set_string (value, gtk_label_get_text(GTK_LABEL(priv->capacity_label)));
break;
case PROP_NAME:
g_value_set_string (value, gtk_label_get_text(GTK_LABEL(priv->name_label)));
break;
case PROP_POPUP_INFO:
g_value_set_string (value, gtk_widget_get_tooltip_text(GTK_WIDGET(widget)));
break;
}
}
static void anaconda_disk_overview_set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) {
AnacondaDiskOverview *widget = ANACONDA_DISK_OVERVIEW(object);
AnacondaDiskOverviewPrivate *priv = widget->priv;
switch(prop_id) {
case PROP_DESCRIPTION: {
gtk_label_set_text(GTK_LABEL(priv->description_label), g_value_get_string(value));
break;
}
case PROP_KIND:
if (widget->priv->kind)
g_free(widget->priv->kind);
widget->priv->kind = g_strdup(g_value_get_string(value));
set_icon(widget, widget->priv->kind);
gtk_grid_attach(GTK_GRID(widget->priv->grid), widget->priv->kind_icon, 0, 1, 3, 1);
break;
case PROP_FREE: {
gtk_label_set_text(GTK_LABEL(priv->free_label), g_value_get_string(value));
break;
}
case PROP_CAPACITY: {
gtk_label_set_text(GTK_LABEL(priv->capacity_label), g_value_get_string(value));
break;
}
case PROP_NAME: {
gtk_label_set_text(GTK_LABEL(priv->name_label), g_value_get_string(value));
break;
}
case PROP_POPUP_INFO: {
if (!strcmp(g_value_get_string(value), ""))
gtk_widget_set_has_tooltip(GTK_WIDGET(widget), FALSE);
else {
gtk_widget_set_tooltip_text(GTK_WIDGET(widget), g_value_get_string(value));
break;
}
}
}
}
/**
* anaconda_disk_overview_get_chosen:
* @widget: a #AnacondaDiskOverview
*
* Returns whether or not this disk has been chosen by the user.
*
* Returns: Whether @widget has been chosen.
*
* Since: 1.0
*/
gboolean anaconda_disk_overview_get_chosen(AnacondaDiskOverview *widget) {
return widget->priv->chosen;
}
/**
* anaconda_disk_overview_set_chosen:
* @widget: a #AnacondaDiskOverview
* @is_chosen: %TRUE if this disk is chosen.
*
* Specifies whether the disk shown by this overview has been chosen by
* the user for inclusion in installation. If so, a special background will
* be set as a visual indicator.
*
* Since: 1.0
*/
void anaconda_disk_overview_set_chosen(AnacondaDiskOverview *widget, gboolean is_chosen) {
widget->priv->chosen = is_chosen;
anaconda_disk_overview_toggle_background(widget);
}
static gboolean anaconda_disk_overview_focus_changed(GtkWidget *widget, GdkEventFocus *event, gpointer user_data) {
GtkStateFlags new_state;
new_state = gtk_widget_get_state_flags(widget) & ~GTK_STATE_FLAG_SELECTED;
if (event->in)
new_state |= GTK_STATE_FLAG_SELECTED;
gtk_widget_set_state_flags(widget, new_state, TRUE);
return FALSE;
}