spice-gtk/src/usb-device-widget.c
Marc-André Lureau bc3d12efb2 gtk: make SpiceUsbDeviceWidget private
Do not leak internals of SpiceUsbDeviceWidget in public headers. This makes also
the class final, which let us extend more easily without fear of breaking ABI.

Signed-off-by: Marc-André Lureau <marcandre.lureau@gmail.com>
Acked-by: Fabiano Fidêncio <fidencio@redhat.com>
2016-03-14 12:02:49 +01:00

586 lines
20 KiB
C

/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
/*
Copyright (C) 2012 Red Hat, Inc.
Red Hat Authors:
Hans de Goede <hdegoede@redhat.com>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library 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
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
#include "config.h"
#include <glib/gi18n.h>
#include "glib-compat.h"
#include "spice-client.h"
#include "spice-marshal.h"
#include "usb-device-widget.h"
/**
* SECTION:usb-device-widget
* @short_description: USB device selection widget
* @title: Spice USB device selection widget
* @section_id:
* @see_also:
* @stability: Stable
* @include: spice-client-gtk.h
*
* #SpiceUsbDeviceWidget is a gtk widget which apps can use to easily
* add an UI to select USB devices to redirect (or unredirect).
*/
struct _SpiceUsbDeviceWidget
{
GtkVBox parent;
SpiceUsbDeviceWidgetPrivate *priv;
};
struct _SpiceUsbDeviceWidgetClass
{
GtkVBoxClass parent_class;
/* signals */
void (*connect_failed) (SpiceUsbDeviceWidget *widget,
SpiceUsbDevice *device, GError *error);
};
/* ------------------------------------------------------------------ */
/* Prototypes for callbacks */
static void device_added_cb(SpiceUsbDeviceManager *manager,
SpiceUsbDevice *device, gpointer user_data);
static void device_removed_cb(SpiceUsbDeviceManager *manager,
SpiceUsbDevice *device, gpointer user_data);
static void device_error_cb(SpiceUsbDeviceManager *manager,
SpiceUsbDevice *device, GError *err, gpointer user_data);
static gboolean spice_usb_device_widget_update_status(gpointer user_data);
/* ------------------------------------------------------------------ */
/* gobject glue */
#define SPICE_USB_DEVICE_WIDGET_GET_PRIVATE(obj) \
(G_TYPE_INSTANCE_GET_PRIVATE((obj), SPICE_TYPE_USB_DEVICE_WIDGET, \
SpiceUsbDeviceWidgetPrivate))
enum {
PROP_0,
PROP_SESSION,
PROP_DEVICE_FORMAT_STRING,
};
enum {
CONNECT_FAILED,
LAST_SIGNAL,
};
struct _SpiceUsbDeviceWidgetPrivate {
SpiceSession *session;
gchar *device_format_string;
SpiceUsbDeviceManager *manager;
GtkWidget *info_bar;
GtkWidget *label;
gchar *err_msg;
gsize device_count;
};
static guint signals[LAST_SIGNAL] = { 0, };
#if GTK_CHECK_VERSION(3,0,0)
G_DEFINE_TYPE(SpiceUsbDeviceWidget, spice_usb_device_widget, GTK_TYPE_BOX);
#else
G_DEFINE_TYPE(SpiceUsbDeviceWidget, spice_usb_device_widget, GTK_TYPE_VBOX);
#endif
static void spice_usb_device_widget_get_property(GObject *gobject,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
SpiceUsbDeviceWidget *self = SPICE_USB_DEVICE_WIDGET(gobject);
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
switch (prop_id) {
case PROP_SESSION:
g_value_set_object(value, priv->session);
break;
case PROP_DEVICE_FORMAT_STRING:
g_value_set_string(value, priv->device_format_string);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID(gobject, prop_id, pspec);
break;
}
}
static void spice_usb_device_widget_set_property(GObject *gobject,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
SpiceUsbDeviceWidget *self = SPICE_USB_DEVICE_WIDGET(gobject);
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
switch (prop_id) {
case PROP_SESSION:
priv->session = g_value_dup_object(value);
break;
case PROP_DEVICE_FORMAT_STRING:
priv->device_format_string = g_value_dup_string(value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID(gobject, prop_id, pspec);
break;
}
}
static void spice_usb_device_widget_hide_info_bar(SpiceUsbDeviceWidget *self)
{
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
if (priv->info_bar) {
gtk_widget_destroy(priv->info_bar);
priv->info_bar = NULL;
}
}
static void
spice_usb_device_widget_show_info_bar(SpiceUsbDeviceWidget *self,
const gchar *message,
GtkMessageType message_type,
const gchar *stock_icon_id)
{
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
GtkWidget *info_bar, *content_area, *hbox, *widget;
spice_usb_device_widget_hide_info_bar(self);
info_bar = gtk_info_bar_new();
gtk_info_bar_set_message_type(GTK_INFO_BAR(info_bar), message_type);
content_area = gtk_info_bar_get_content_area(GTK_INFO_BAR(info_bar));
#if GTK_CHECK_VERSION(3,0,0)
hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 12);
#else
hbox = gtk_hbox_new(FALSE, 12);
#endif
gtk_container_add(GTK_CONTAINER(content_area), hbox);
widget = gtk_image_new_from_stock(stock_icon_id,
GTK_ICON_SIZE_SMALL_TOOLBAR);
gtk_box_pack_start(GTK_BOX(hbox), widget, FALSE, FALSE, 0);
widget = gtk_label_new(message);
gtk_box_pack_start(GTK_BOX(hbox), widget, TRUE, TRUE, 0);
priv->info_bar = gtk_alignment_new(0.0, 0.0, 1.0, 0.0);
gtk_alignment_set_padding(GTK_ALIGNMENT(priv->info_bar), 0, 0, 12, 0);
gtk_container_add(GTK_CONTAINER(priv->info_bar), info_bar);
gtk_box_pack_start(GTK_BOX(self), priv->info_bar, FALSE, FALSE, 0);
gtk_widget_show_all(priv->info_bar);
}
static GObject *spice_usb_device_widget_constructor(
GType gtype, guint n_properties, GObjectConstructParam *properties)
{
GObject *obj;
SpiceUsbDeviceWidget *self;
SpiceUsbDeviceWidgetPrivate *priv;
GPtrArray *devices = NULL;
GError *err = NULL;
gchar *str;
int i;
{
/* Always chain up to the parent constructor */
GObjectClass *parent_class;
parent_class = G_OBJECT_CLASS(spice_usb_device_widget_parent_class);
obj = parent_class->constructor(gtype, n_properties, properties);
}
self = SPICE_USB_DEVICE_WIDGET(obj);
priv = self->priv;
if (!priv->session)
g_error("SpiceUsbDeviceWidget constructed without a session");
priv->label = gtk_label_new(NULL);
str = g_strdup_printf("<b>%s</b>", _("Select USB devices to redirect"));
gtk_label_set_markup(GTK_LABEL (priv->label), str);
g_free(str);
gtk_misc_set_alignment(GTK_MISC(priv->label), 0.0, 0.5);
gtk_box_pack_start(GTK_BOX(self), priv->label, FALSE, FALSE, 0);
priv->manager = spice_usb_device_manager_get(priv->session, &err);
if (err) {
spice_usb_device_widget_show_info_bar(self, err->message,
GTK_MESSAGE_WARNING,
GTK_STOCK_DIALOG_WARNING);
g_clear_error(&err);
return obj;
}
g_signal_connect(priv->manager, "device-added",
G_CALLBACK(device_added_cb), self);
g_signal_connect(priv->manager, "device-removed",
G_CALLBACK(device_removed_cb), self);
g_signal_connect(priv->manager, "device-error",
G_CALLBACK(device_error_cb), self);
devices = spice_usb_device_manager_get_devices(priv->manager);
if (!devices)
goto end;
for (i = 0; i < devices->len; i++)
device_added_cb(NULL, g_ptr_array_index(devices, i), self);
g_ptr_array_unref(devices);
end:
spice_usb_device_widget_update_status(self);
return obj;
}
static void spice_usb_device_widget_finalize(GObject *object)
{
SpiceUsbDeviceWidget *self = SPICE_USB_DEVICE_WIDGET(object);
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
if (priv->manager) {
g_signal_handlers_disconnect_by_func(priv->manager,
device_added_cb, self);
g_signal_handlers_disconnect_by_func(priv->manager,
device_removed_cb, self);
g_signal_handlers_disconnect_by_func(priv->manager,
device_error_cb, self);
}
g_object_unref(priv->session);
g_free(priv->device_format_string);
if (G_OBJECT_CLASS(spice_usb_device_widget_parent_class)->finalize)
G_OBJECT_CLASS(spice_usb_device_widget_parent_class)->finalize(object);
}
static void spice_usb_device_widget_class_init(
SpiceUsbDeviceWidgetClass *klass)
{
GObjectClass *gobject_class = (GObjectClass *)klass;
GParamSpec *pspec;
g_type_class_add_private (klass, sizeof (SpiceUsbDeviceWidgetPrivate));
gobject_class->constructor = spice_usb_device_widget_constructor;
gobject_class->finalize = spice_usb_device_widget_finalize;
gobject_class->get_property = spice_usb_device_widget_get_property;
gobject_class->set_property = spice_usb_device_widget_set_property;
/**
* SpiceUsbDeviceWidget:session:
*
* #SpiceSession this #SpiceUsbDeviceWidget is associated with
*
**/
pspec = g_param_spec_object("session",
"Session",
"SpiceSession",
SPICE_TYPE_SESSION,
G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
G_PARAM_STATIC_STRINGS);
g_object_class_install_property(gobject_class, PROP_SESSION, pspec);
/**
* SpiceUsbDeviceWidget:device-format-string:
*
* Format string to pass to spice_usb_device_get_description() for getting
* the device USB descriptions.
*/
pspec = g_param_spec_string("device-format-string",
"Device format string",
"Format string for device description",
NULL,
G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
G_PARAM_STATIC_STRINGS);
g_object_class_install_property(gobject_class, PROP_DEVICE_FORMAT_STRING,
pspec);
/**
* SpiceUsbDeviceWidget::connect-failed:
* @widget: The #SpiceUsbDeviceWidget that emitted the signal
* @device: #SpiceUsbDevice boxed object corresponding to the added device
* @error: #GError describing the reason why the connect failed
*
* The #SpiceUsbDeviceWidget::connect-failed signal is emitted whenever
* the user has requested for a device to be redirected and this has
* failed.
**/
signals[CONNECT_FAILED] =
g_signal_new("connect-failed",
G_OBJECT_CLASS_TYPE(gobject_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET(SpiceUsbDeviceWidgetClass, connect_failed),
NULL, NULL,
g_cclosure_user_marshal_VOID__BOXED_BOXED,
G_TYPE_NONE,
2,
SPICE_TYPE_USB_DEVICE,
G_TYPE_ERROR);
}
static void spice_usb_device_widget_init(SpiceUsbDeviceWidget *self)
{
self->priv = SPICE_USB_DEVICE_WIDGET_GET_PRIVATE(self);
}
/* ------------------------------------------------------------------ */
/* public api */
/**
* spice_usb_device_widget_new:
* @session: #SpiceSession for which to widget will control USB redirection
* @device_format_string: (allow-none): String passed to
* spice_usb_device_get_description()
*
* Creates a new widget to control USB redirection.
*
* Returns: a new #SpiceUsbDeviceWidget instance
*/
GtkWidget *spice_usb_device_widget_new(SpiceSession *session,
const gchar *device_format_string)
{
return g_object_new(SPICE_TYPE_USB_DEVICE_WIDGET,
"orientation", GTK_ORIENTATION_VERTICAL,
"session", session,
"device-format-string", device_format_string,
"spacing", 6,
NULL);
}
/* ------------------------------------------------------------------ */
/* callbacks */
static SpiceUsbDevice *get_usb_device(GtkWidget *widget)
{
if (!GTK_IS_ALIGNMENT(widget))
return NULL;
widget = gtk_bin_get_child(GTK_BIN(widget));
return g_object_get_data(G_OBJECT(widget), "usb-device");
}
static void check_can_redirect(GtkWidget *widget, gpointer user_data)
{
SpiceUsbDeviceWidget *self = SPICE_USB_DEVICE_WIDGET(user_data);
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
SpiceUsbDevice *device;
gboolean can_redirect;
GError *err = NULL;
device = get_usb_device(widget);
if (!device)
return; /* Non device widget, ie the info_bar */
priv->device_count++;
can_redirect = spice_usb_device_manager_can_redirect_device(priv->manager,
device, &err);
gtk_widget_set_sensitive(widget, can_redirect);
/* If we cannot redirect this device, append the error message to
err_msg, but only if it is *not* already there! */
if (!can_redirect) {
if (priv->err_msg) {
if (!strstr(priv->err_msg, err->message)) {
gchar *old_err_msg = priv->err_msg;
priv->err_msg = g_strdup_printf("%s\n%s", priv->err_msg,
err->message);
g_free(old_err_msg);
}
} else {
priv->err_msg = g_strdup(err->message);
}
}
g_clear_error(&err);
}
static gboolean spice_usb_device_widget_update_status(gpointer user_data)
{
SpiceUsbDeviceWidget *self = SPICE_USB_DEVICE_WIDGET(user_data);
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
gchar *str, *markup_str;
const gchar *free_channels_str;
int free_channels;
g_object_get(priv->manager, "free-channels", &free_channels, NULL);
free_channels_str = ngettext(_("Select USB devices to redirect (%d free channel)"),
_("Select USB devices to redirect (%d free channels)"),
free_channels);
str = g_strdup_printf(free_channels_str, free_channels);
markup_str = g_strdup_printf("<b>%s</b>", str);
gtk_label_set_markup(GTK_LABEL (priv->label), markup_str);
g_free(markup_str);
g_free(str);
priv->device_count = 0;
gtk_container_foreach(GTK_CONTAINER(self), check_can_redirect, self);
if (priv->err_msg) {
spice_usb_device_widget_show_info_bar(self, priv->err_msg,
GTK_MESSAGE_INFO,
GTK_STOCK_DIALOG_WARNING);
g_free(priv->err_msg);
priv->err_msg = NULL;
} else {
spice_usb_device_widget_hide_info_bar(self);
}
if (priv->device_count == 0)
spice_usb_device_widget_show_info_bar(self, _("No USB devices detected"),
GTK_MESSAGE_INFO,
GTK_STOCK_DIALOG_INFO);
return FALSE;
}
typedef struct _connect_cb_data {
GtkWidget *check;
SpiceUsbDeviceWidget *self;
} connect_cb_data;
static void connect_cb(GObject *gobject, GAsyncResult *res, gpointer user_data)
{
SpiceUsbDeviceManager *manager = SPICE_USB_DEVICE_MANAGER(gobject);
connect_cb_data *data = user_data;
SpiceUsbDeviceWidget *self = data->self;
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
SpiceUsbDevice *device;
GError *err = NULL;
gchar *desc;
spice_usb_device_manager_connect_device_finish(manager, res, &err);
if (err) {
device = g_object_get_data(G_OBJECT(data->check), "usb-device");
desc = spice_usb_device_get_description(device,
priv->device_format_string);
g_prefix_error(&err, "Could not redirect %s: ", desc);
g_free(desc);
SPICE_DEBUG("%s", err->message);
g_signal_emit(self, signals[CONNECT_FAILED], 0, device, err);
g_error_free(err);
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(data->check), FALSE);
spice_usb_device_widget_update_status(self);
}
g_object_unref(data->check);
g_object_unref(data->self);
g_free(data);
}
static void checkbox_clicked_cb(GtkWidget *check, gpointer user_data)
{
SpiceUsbDeviceWidget *self = SPICE_USB_DEVICE_WIDGET(user_data);
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
SpiceUsbDevice *device;
device = g_object_get_data(G_OBJECT(check), "usb-device");
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(check))) {
connect_cb_data *data = g_new(connect_cb_data, 1);
data->check = g_object_ref(check);
data->self = g_object_ref(self);
spice_usb_device_manager_connect_device_async(priv->manager,
device,
NULL,
connect_cb,
data);
} else {
spice_usb_device_manager_disconnect_device(priv->manager,
device);
}
spice_usb_device_widget_update_status(self);
}
static void checkbox_usb_device_destroy_notify(gpointer data)
{
g_boxed_free(spice_usb_device_get_type(), data);
}
static void device_added_cb(SpiceUsbDeviceManager *manager,
SpiceUsbDevice *device, gpointer user_data)
{
SpiceUsbDeviceWidget *self = SPICE_USB_DEVICE_WIDGET(user_data);
SpiceUsbDeviceWidgetPrivate *priv = self->priv;
GtkWidget *align, *check;
gchar *desc;
desc = spice_usb_device_get_description(device,
priv->device_format_string);
check = gtk_check_button_new_with_label(desc);
g_free(desc);
if (spice_usb_device_manager_is_device_connected(priv->manager,
device))
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check), TRUE);
g_object_set_data_full(
G_OBJECT(check), "usb-device",
g_boxed_copy(spice_usb_device_get_type(), device),
checkbox_usb_device_destroy_notify);
g_signal_connect(G_OBJECT(check), "clicked",
G_CALLBACK(checkbox_clicked_cb), self);
align = gtk_alignment_new(0, 0, 0, 0);
gtk_alignment_set_padding(GTK_ALIGNMENT(align), 0, 0, 12, 0);
gtk_container_add(GTK_CONTAINER(align), check);
gtk_box_pack_end(GTK_BOX(self), align, FALSE, FALSE, 0);
spice_usb_device_widget_update_status(self);
gtk_widget_show_all(align);
}
static void destroy_widget_by_usb_device(GtkWidget *widget, gpointer user_data)
{
if (get_usb_device(widget) == user_data)
gtk_widget_destroy(widget);
}
static void device_removed_cb(SpiceUsbDeviceManager *manager,
SpiceUsbDevice *device, gpointer user_data)
{
SpiceUsbDeviceWidget *self = SPICE_USB_DEVICE_WIDGET(user_data);
gtk_container_foreach(GTK_CONTAINER(self),
destroy_widget_by_usb_device, device);
spice_usb_device_widget_update_status(self);
}
static void set_inactive_by_usb_device(GtkWidget *widget, gpointer user_data)
{
if (get_usb_device(widget) == user_data) {
GtkWidget *check = gtk_bin_get_child(GTK_BIN(widget));
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check), FALSE);
}
}
static void device_error_cb(SpiceUsbDeviceManager *manager,
SpiceUsbDevice *device, GError *err, gpointer user_data)
{
SpiceUsbDeviceWidget *self = SPICE_USB_DEVICE_WIDGET(user_data);
gtk_container_foreach(GTK_CONTAINER(self),
set_inactive_by_usb_device, device);
spice_usb_device_widget_update_status(self);
}