Files
NetworkManager/clients/tui/nmt-widget-list.c
Thomas Haller 8bace23beb all: cleanup includes and let "nm-default.h" include "config.h"
- All internal source files (except "examples", which are not internal)
  should include "config.h" first. As also all internal source
  files should include "nm-default.h", let "config.h" be included
  by "nm-default.h" and include "nm-default.h" as first in every
  source file.
  We already wanted to include "nm-default.h" before other headers
  because it might contains some fixes (like "nm-glib.h" compatibility)
  that is required first.

- After including "nm-default.h", we optinally allow for including the
  corresponding header file for the source file at hand. The idea
  is to ensure that each header file is self contained.

- Don't include "config.h" or "nm-default.h" in any header file
  (except "nm-sd-adapt.h"). Public headers anyway must not include
  these headers, and internal headers are never included after
  "nm-default.h", as of the first previous point.

- Include all internal headers with quotes instead of angle brackets.
  In practice it doesn't matter, because in our public headers we must
  include other headers with angle brackets. As we use our public
  headers also to compile our interal source files, effectively the
  result must be the same. Still do it for consistency.

- Except for <config.h> itself. Include it with angle brackets as suggested by
  https://www.gnu.org/software/autoconf/manual/autoconf.html#Configuration-Headers
2016-02-19 17:53:25 +01:00

495 lines
14 KiB
C

/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */
/*
* 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/>.
*
* Copyright 2013 Red Hat, Inc.
*/
/**
* SECTION:nmt-widget-list
* @short_description: A list of widgets, with Add and Remove buttons
*
* #NmtWidgetList presents a homogeneous list of widgets, with "Remove"
* buttons next to each one, and an "Add" button at the button to add
* new ones.
*
* It is the base class for #NmtAddressList, and is used internally by
* #NmtRouteTable.
*
* FIXME: The way this works is sort of weird.
*/
#include "nm-default.h"
#include <arpa/inet.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <string.h>
#include "nmt-widget-list.h"
#include "nmt-newt.h"
G_DEFINE_TYPE (NmtWidgetList, nmt_widget_list, NMT_TYPE_NEWT_GRID)
#define NMT_WIDGET_LIST_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), NMT_TYPE_WIDGET_LIST, NmtWidgetListPrivate))
typedef struct {
int length;
NmtWidgetListCallback create_callback;
gpointer user_data;
GDestroyNotify destroy_notify;
NmtNewtWidget *empty_widget;
GPtrArray *widgets;
GPtrArray *remove_buttons;
NmtNewtWidget *add_button;
GBinding *add_sensitivity;
} NmtWidgetListPrivate;
enum {
PROP_0,
PROP_CREATE_CALLBACK,
PROP_USER_DATA,
PROP_DESTROY_NOTIFY,
PROP_EMPTY_WIDGET,
PROP_LENGTH,
LAST_PROP
};
enum {
ADD_CLICKED,
REMOVE_CLICKED,
LAST_SIGNAL
};
static guint signals[LAST_SIGNAL] = { 0 };
static void add_clicked (NmtNewtButton *button, gpointer user_data);
static void remove_clicked (NmtNewtButton *button, gpointer user_data);
/**
* NmtWidgetListCallback:
* @list: the #NmtWidgetList
* @n: the number of the widget being added
* @user_data: the callback's user data
*
* Called by #NmtWidgetList to ask for a new widget to be created.
*
* Note that the widget is not created to go with any particular
* value, but rather is created to be at a certain spot in the list.
* When an element is deleted from the list, it is actually always
* the last widget in the list that is removed, but it is assumed
* that the widget list is bound to some array-valued property, and
* so when an element is deleted from that array, the widgets will
* all update themselves automatically when the array changes.
*
* Returns: a new widget for the list
*/
/**
* nmt_widget_list_new:
* @create_callback: callback to create new widgets
* @user_data: user data for @create_callback
* @destroy_notify: #GDestroyNotify for @user_data
* @empty_widget: (allow-none): a widget to display when there are
* no "real" widgets in the list.
*
* Creates a new #NmtWidgetList.
*
* Returns: a new #NmtWidgetList.
*/
NmtNewtWidget *
nmt_widget_list_new (NmtWidgetListCallback create_callback,
gpointer user_data,
GDestroyNotify destroy_notify,
NmtNewtWidget *empty_widget)
{
return g_object_new (NMT_TYPE_WIDGET_LIST,
"create-callback", create_callback,
"user-data", user_data,
"destroy-notify", destroy_notify,
"empty-widget", empty_widget,
NULL);
}
static void
nmt_widget_list_init (NmtWidgetList *list)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (list);
priv->widgets = g_ptr_array_new ();
priv->remove_buttons = g_ptr_array_new ();
priv->add_button = nmt_newt_button_new (_("Add..."));
g_signal_connect (priv->add_button, "clicked",
G_CALLBACK (add_clicked), list);
nmt_newt_grid_add (NMT_NEWT_GRID (list), priv->add_button, 0, 0);
}
static void
nmt_widget_list_constructed (GObject *object)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (object);
if (priv->length == 0 && priv->empty_widget) {
nmt_newt_widget_set_visible (priv->empty_widget, TRUE);
nmt_newt_grid_move (NMT_NEWT_GRID (object), priv->add_button, 0, 1);
}
G_OBJECT_CLASS (nmt_widget_list_parent_class)->constructed (object);
}
static void
nmt_widget_list_finalize (GObject *object)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (object);
g_ptr_array_unref (priv->widgets);
g_ptr_array_unref (priv->remove_buttons);
if (priv->user_data && priv->destroy_notify)
priv->destroy_notify (priv->user_data);
g_clear_object (&priv->empty_widget);
G_OBJECT_CLASS (nmt_widget_list_parent_class)->finalize (object);
}
static void
ensure_widgets (NmtWidgetList *list)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (list);
NmtNewtWidget *widget, *button, *focus;
gboolean was_empty;
NmtNewtForm *form;
int i;
was_empty = priv->widgets->len == 0;
if (priv->length < priv->widgets->len) {
/* remove excess widgets */
for (i = priv->length; i < priv->widgets->len; i++) {
nmt_newt_container_remove (NMT_NEWT_CONTAINER (list), priv->widgets->pdata[i]);
nmt_newt_container_remove (NMT_NEWT_CONTAINER (list), priv->remove_buttons->pdata[i]);
}
g_ptr_array_set_size (priv->widgets, priv->length);
g_ptr_array_set_size (priv->remove_buttons, priv->length);
} else if (priv->length > priv->widgets->len) {
/* add new widgets */
for (i = priv->widgets->len; i < priv->length; i++) {
widget = NMT_WIDGET_LIST_GET_CLASS (list)->create_widget (list, i);
nmt_newt_grid_add (NMT_NEWT_GRID (list), widget, 0, i);
g_ptr_array_add (priv->widgets, widget);
button = nmt_newt_button_new (_("Remove"));
g_signal_connect (button, "clicked",
G_CALLBACK (remove_clicked), list);
nmt_newt_grid_add (NMT_NEWT_GRID (list), button, 1, i);
nmt_newt_widget_set_padding (button, 1, 0, 0, 0);
g_ptr_array_add (priv->remove_buttons, button);
}
} else
return;
if (priv->widgets->len == 0 && priv->empty_widget) {
nmt_newt_widget_set_visible (priv->empty_widget, TRUE);
nmt_newt_grid_move (NMT_NEWT_GRID (list), priv->add_button, 0, 1);
} else {
if (was_empty && priv->empty_widget)
nmt_newt_widget_set_visible (priv->empty_widget, FALSE);
nmt_newt_grid_move (NMT_NEWT_GRID (list), priv->add_button, 0, priv->length);
}
form = nmt_newt_widget_get_form (NMT_NEWT_WIDGET (list));
if (form) {
if (priv->widgets->len) {
if (was_empty)
focus = priv->widgets->pdata[0];
else
focus = priv->widgets->pdata[priv->widgets->len - 1];
} else
focus = priv->add_button;
nmt_newt_form_set_focus (form, focus);
}
g_clear_object (&priv->add_sensitivity);
if (priv->widgets->len) {
widget = priv->widgets->pdata[priv->widgets->len - 1];
priv->add_sensitivity = g_object_bind_property (widget, "valid",
priv->add_button, "sensitive",
G_BINDING_SYNC_CREATE);
g_object_add_weak_pointer (G_OBJECT (priv->add_sensitivity),
(gpointer *)&priv->add_sensitivity);
} else
nmt_newt_component_set_sensitive (NMT_NEWT_COMPONENT (priv->add_button), TRUE);
}
static void
add_clicked (NmtNewtButton *button, gpointer list)
{
g_signal_emit (G_OBJECT (list), signals[ADD_CLICKED], 0, NULL);
}
static void
remove_clicked (NmtNewtButton *button, gpointer list)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (list);
int i;
for (i = 0; i < priv->remove_buttons->len; i++) {
if (priv->remove_buttons->pdata[i] == (gpointer)button)
break;
}
g_return_if_fail (i < priv->remove_buttons->len);
g_signal_emit (G_OBJECT (list), signals[REMOVE_CLICKED], 0, i, NULL);
}
static NmtNewtWidget *
nmt_widget_list_real_create_widget (NmtWidgetList *list,
int n)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (list);
g_return_val_if_fail (priv->create_callback != NULL, NULL);
return priv->create_callback (list, n, priv->user_data);
}
/**
* nmt_widget_list_get_length:
* @list: the #NmtNewtWidgetList
*
* Gets the number of widgets in the list.
*
* Returns: the number of widgets in the list.
*/
int
nmt_widget_list_get_length (NmtWidgetList *list)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (list);
return priv->length;
}
/**
* nmt_widget_list_set_length:
* @list: the #NmtNewtWidgetList
* @length: the new length
*
* Changes the number of widgets in the list. Widgets will be added or
* deleted as necessary.
*/
void
nmt_widget_list_set_length (NmtWidgetList *list,
int length)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (list);
if (priv->length != length) {
priv->length = length;
g_object_notify (G_OBJECT (list), "length");
}
ensure_widgets (list);
}
static void
nmt_widget_list_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (object);
switch (prop_id) {
case PROP_CREATE_CALLBACK:
priv->create_callback = g_value_get_pointer (value);
break;
case PROP_USER_DATA:
priv->user_data = g_value_get_pointer (value);
break;
case PROP_DESTROY_NOTIFY:
priv->destroy_notify = g_value_get_pointer (value);
break;
case PROP_LENGTH:
priv->length = g_value_get_int (value);
ensure_widgets (NMT_WIDGET_LIST (object));
break;
case PROP_EMPTY_WIDGET:
priv->empty_widget = g_value_get_object (value);
if (priv->empty_widget) {
g_object_ref_sink (priv->empty_widget);
nmt_newt_grid_add (NMT_NEWT_GRID (object), priv->empty_widget, 0, 0);
}
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
nmt_widget_list_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
NmtWidgetListPrivate *priv = NMT_WIDGET_LIST_GET_PRIVATE (object);
switch (prop_id) {
case PROP_CREATE_CALLBACK:
g_value_set_pointer (value, priv->create_callback);
break;
case PROP_USER_DATA:
g_value_set_pointer (value, priv->user_data);
break;
case PROP_DESTROY_NOTIFY:
g_value_set_pointer (value, priv->destroy_notify);
break;
case PROP_LENGTH:
g_value_set_int (value, priv->length);
break;
case PROP_EMPTY_WIDGET:
g_value_set_object (value, priv->empty_widget);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
nmt_widget_list_class_init (NmtWidgetListClass *list_class)
{
GObjectClass *object_class = G_OBJECT_CLASS (list_class);
g_type_class_add_private (list_class, sizeof (NmtWidgetListPrivate));
/* virtual methods */
object_class->constructed = nmt_widget_list_constructed;
object_class->set_property = nmt_widget_list_set_property;
object_class->get_property = nmt_widget_list_get_property;
object_class->finalize = nmt_widget_list_finalize;
list_class->create_widget = nmt_widget_list_real_create_widget;
/* signals */
/**
* NmtNewtWidget::add-clicked:
* @list: the #NmtNewtWidgetList
*
* Emitted when the user clicks the "Add" button. The caller can
* decide whether or not to add a new widget, and call
* nmt_widget_list_set_length() with the new length if so.
*
* FIXME: the "Add" button should be insensitive if it's
* not going to work.
*/
signals[ADD_CLICKED] =
g_signal_new ("add-clicked",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (NmtWidgetListClass, add_clicked),
NULL, NULL, NULL,
G_TYPE_NONE, 0);
/**
* NmtNewtWidget::remove-clicked:
* @list: the #NmtNewtWidgetList
* @n: the widget being removed
*
* Emitted when the user clicks one of the "Remove" buttons. The
* caller can decide whether or not to remove the widget, and
* call nmt_widget_list_set_length() with the new length if so.
*
* FIXME: the "Remove" button should be insensitive if it's not
* going to work.
*/
signals[REMOVE_CLICKED] =
g_signal_new ("remove-clicked",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (NmtWidgetListClass, remove_clicked),
NULL, NULL, NULL,
G_TYPE_NONE, 1, G_TYPE_INT);
/* properties */
/**
* NmtWidgetList:create-callback:
*
* Callback called to create a new widget.
*/
g_object_class_install_property
(object_class, PROP_CREATE_CALLBACK,
g_param_spec_pointer ("create-callback", "", "",
G_PARAM_READWRITE |
G_PARAM_STATIC_STRINGS));
/**
* NmtWidgetList:user-data:
*
* User data for #NmtWidgetList:create-callback
*/
g_object_class_install_property
(object_class, PROP_USER_DATA,
g_param_spec_pointer ("user-data", "", "",
G_PARAM_READWRITE |
G_PARAM_STATIC_STRINGS));
/**
* NmtWidgetList:destroy-notify:
*
* #GDestroyNotify for #NmtWidgetList:user-data
*/
g_object_class_install_property
(object_class, PROP_DESTROY_NOTIFY,
g_param_spec_pointer ("destroy-notify", "", "",
G_PARAM_READWRITE |
G_PARAM_STATIC_STRINGS));
/**
* NmtWidgetList:length:
*
* The length of the widget list; changing this value will add or
* remove widgets from the list.
*/
g_object_class_install_property
(object_class, PROP_LENGTH,
g_param_spec_int ("length", "", "",
0, G_MAXINT, 0,
G_PARAM_READWRITE |
G_PARAM_STATIC_STRINGS));
/**
* NmtWidgetList:empty-widget:
*
* If non-%NULL, this widget will be displayed when there are
* no "real" widgets in the list.
*/
g_object_class_install_property
(object_class, PROP_EMPTY_WIDGET,
g_param_spec_object ("empty-widget", "", "",
NMT_TYPE_NEWT_WIDGET,
G_PARAM_READWRITE |
G_PARAM_CONSTRUCT_ONLY |
G_PARAM_STATIC_STRINGS));
}