Blob Blame History Raw
// SPDX-License-Identifier: GPL-2.0+
/*
 * Copyright (C) 2013 Red Hat, Inc.
 */

/**
 * SECTION:nmt-newt-listbox
 * @short_description: Single-choice listboxes
 *
 * #NmtNewtListbox implements a single-choice listbox.
 *
 * A listbox has some number of rows, each associated with an
 * arbitrary pointer value. The pointer values do not need to be
 * unique, but some APIs will not be usable if they aren't. You
 * can also cause rows with %NULL keys to be treated specially.
 *
 * The listbox will emit #NmtNewtWidget::activate when the user
 * presses Return on a selection.
 */

#include "nm-default.h"

#include "nmt-newt-listbox.h"
#include "nmt-newt-form.h"
#include "nmt-newt-utils.h"

G_DEFINE_TYPE (NmtNewtListbox, nmt_newt_listbox, NMT_TYPE_NEWT_COMPONENT)

#define NMT_NEWT_LISTBOX_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), NMT_TYPE_NEWT_LISTBOX, NmtNewtListboxPrivate))

typedef struct {
	int height, alloc_height, width;
	gboolean fixed_height;
	NmtNewtListboxFlags flags;

	GPtrArray *entries;
	GPtrArray *keys;

	int active;
	gpointer active_key;
	gboolean skip_null_keys;

} NmtNewtListboxPrivate;

enum {
	PROP_0,
	PROP_HEIGHT,
	PROP_FLAGS,
	PROP_ACTIVE,
	PROP_ACTIVE_KEY,
	PROP_SKIP_NULL_KEYS,

	LAST_PROP
};

/**
 * NmtNewtListboxFlags:
 * @NMT_NEWT_LISTBOX_SCROLL: the listbox should have a scroll bar.
 * @NMT_NEWT_LISTBOX_BORDER: the listbox should have a border around it.
 *
 * Flags describing an #NmtNewtListbox
 */

/**
 * nmt_newt_listbox_new:
 * @height: the height of the listbox, or -1 for no fixed height
 * @flags: the listbox flags
 *
 * Creates a new #NmtNewtListbox
 *
 * Returns: a new #NmtNewtListbox
 */
NmtNewtWidget *
nmt_newt_listbox_new (int                 height,
                      NmtNewtListboxFlags flags)
{
	return g_object_new (NMT_TYPE_NEWT_LISTBOX,
	                     "height", height,
	                     "flags", flags,
	                     NULL);
}

/**
 * nmt_newt_listbox_append:
 * @listbox: an #NmtNewtListbox
 * @entry: the text for the new row
 * @key: (allow-none): the key associated with @entry
 *
 * Adds a row to @listbox.
 */
void
nmt_newt_listbox_append (NmtNewtListbox *listbox,
                         const char     *entry,
                         gpointer        key)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);

	g_ptr_array_add (priv->entries, nmt_newt_locale_from_utf8 (entry));
	g_ptr_array_add (priv->keys, key);
	nmt_newt_widget_needs_rebuild (NMT_NEWT_WIDGET (listbox));
}

/**
 * nmt_newt_listbox_clear:
 * @listbox: an #NmtNewtListbox
 *
 * Clears the contents of @listbox.
 */
void
nmt_newt_listbox_clear (NmtNewtListbox *listbox)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);

	g_ptr_array_set_size (priv->entries, 0);
	g_ptr_array_set_size (priv->keys, 0);

	priv->active = -1;
	priv->active_key = NULL;

	nmt_newt_widget_needs_rebuild (NMT_NEWT_WIDGET (listbox));
}

/**
 * nmt_newt_listbox_set_active:
 * @listbox: an #NmtNewtListbox
 * @active: the row to make active
 *
 * Sets @active to be the currently-selected row in @listbox,
 * scrolling it into view if needed.
 */
void
nmt_newt_listbox_set_active (NmtNewtListbox *listbox,
                             int             active)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);

	if (active == priv->active)
		return;

	g_return_if_fail (active >= 0 && active < priv->entries->len);
	g_return_if_fail (!priv->skip_null_keys || priv->keys->pdata[active]);

	priv->active = active;
	priv->active_key = priv->keys->pdata[active];

	g_object_notify (G_OBJECT (listbox), "active");
	g_object_notify (G_OBJECT (listbox), "active-key");
}

/**
 * nmt_newt_listbox_set_active_key:
 * @listbox: an #NmtNewtListbox
 * @active_key: the key for the row to make active
 *
 * Selects the (first) row in @listbox with @active_key as its key,
 * scrolling it into view if needed.
 */
void
nmt_newt_listbox_set_active_key (NmtNewtListbox *listbox,
                                 gpointer        active_key)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);
	int i;

	if (active_key == priv->active_key)
		return;

	g_return_if_fail (!priv->skip_null_keys || active_key);

	for (i = 0; i < priv->keys->len; i++) {
		if (priv->keys->pdata[i] == active_key) {
			priv->active = i;
			priv->active_key = active_key;

			g_object_notify (G_OBJECT (listbox), "active");
			g_object_notify (G_OBJECT (listbox), "active-key");
			return;
		}
	}
}

/**
 * nmt_newt_listbox_get_active:
 * @listbox: an #NmtNewtListbox
 *
 * Gets the currently-selected row in @listbox.
 *
 * Returns: the currently-selected row in @listbox.
 */
int
nmt_newt_listbox_get_active (NmtNewtListbox *listbox)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);

	return priv->active;
}

/**
 * nmt_newt_listbox_get_active_key:
 * @listbox: an #NmtNewtListbox
 *
 * Gets the key of the currently-selected row in @listbox.
 *
 * Returns: the key of the currently-selected row in @listbox.
 */
gpointer
nmt_newt_listbox_get_active_key (NmtNewtListbox *listbox)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);

	return priv->active_key;
}

/**
 * nmt_newt_listbox_set_height:
 * @listbox: an #NmtNewtListbox
 * @height: the new height, or -1 for no fixed height
 *
 * Updates @listbox's height.
 */
void
nmt_newt_listbox_set_height (NmtNewtListbox *listbox,
                             int             height)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);

	priv->height = height;
	priv->fixed_height = priv->height != 0;
	g_object_notify (G_OBJECT (listbox), "height");
}

static void
nmt_newt_listbox_init (NmtNewtListbox *listbox)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);

	priv->entries = g_ptr_array_new_with_free_func (g_free);
	priv->keys = g_ptr_array_new ();

	priv->active = -1;
}

static void
nmt_newt_listbox_finalize (GObject *object)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (object);

	g_ptr_array_unref (priv->entries);
	g_ptr_array_unref (priv->keys);

	G_OBJECT_CLASS (nmt_newt_listbox_parent_class)->finalize (object);
}

static void
nmt_newt_listbox_size_request (NmtNewtWidget *widget,
                               int           *width,
                               int           *height)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (widget);

	NMT_NEWT_WIDGET_CLASS (nmt_newt_listbox_parent_class)->
		size_request (widget, width, height);

	priv->alloc_height = -1;
	if (!priv->fixed_height)
		*height = 1;
	priv->width = *width;
}

static void
nmt_newt_listbox_size_allocate (NmtNewtWidget *widget,
                                int            x,
                                int            y,
                                int            width,
                                int            height)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (widget);

	if (width > priv->width) {
		newtListboxSetWidth (nmt_newt_component_get_component (NMT_NEWT_COMPONENT (widget)),
		                     width);
	}

	NMT_NEWT_WIDGET_CLASS (nmt_newt_listbox_parent_class)->
		size_allocate (widget, x, y, width, height);

	priv->alloc_height = height;

	if (!priv->fixed_height && height != priv->height) {
		priv->height = height;
		nmt_newt_widget_needs_rebuild (widget);
	}
}

static void
update_active_internal (NmtNewtListbox *listbox,
                        int             new_active)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);

	if (priv->active == new_active)
		return;
	if (new_active >= priv->keys->len)
		return;

	if (priv->skip_null_keys && !priv->keys->pdata[new_active]) {
		if (new_active > priv->active) {
			while (   new_active < priv->entries->len
			       && !priv->keys->pdata[new_active])
				new_active++;
		} else {
			while (   new_active >= 0
			       && !priv->keys->pdata[new_active])
				new_active--;
		}

		if (   new_active < 0
		    || new_active >= priv->entries->len
		    || !priv->keys->pdata[new_active]) {
			g_assert (priv->active >= 0 && priv->active < priv->entries->len);
			return;
		}
	}

	nmt_newt_listbox_set_active (listbox, new_active);
}

static void
selection_changed_callback (newtComponent  co,
                            void          *user_data)
{
	NmtNewtListbox *listbox = user_data;
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (listbox);
	int new_active;

	new_active = GPOINTER_TO_UINT (newtListboxGetCurrent (co));
	update_active_internal (listbox, new_active);

	if (priv->active != new_active)
		newtListboxSetCurrent (co, priv->active);
}

static guint
convert_flags (NmtNewtListboxFlags flags)
{
	guint newt_flags = NEWT_FLAG_RETURNEXIT;

	if (flags & NMT_NEWT_LISTBOX_SCROLL)
		newt_flags |= NEWT_FLAG_SCROLL;
	if (flags & NMT_NEWT_LISTBOX_BORDER)
		newt_flags |= NEWT_FLAG_BORDER;

	return newt_flags;
}

static newtComponent
nmt_newt_listbox_build_component (NmtNewtComponent *component,
                                  gboolean          sensitive)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (component);
	newtComponent co;
	int i, active;

	if (priv->active == -1)
		update_active_internal (NMT_NEWT_LISTBOX (component), 0);
	active = priv->active;

	co = newtListbox (-1, -1, priv->height, convert_flags (priv->flags));
	newtComponentAddCallback (co, selection_changed_callback, component);

	for (i = 0; i < priv->entries->len; i++) {
		newtListboxAppendEntry (co, priv->entries->pdata[i], GUINT_TO_POINTER (i));
		if (active == -1 && priv->keys->pdata[i] == priv->active_key)
			active = i;
	}

	if (active != -1)
		newtListboxSetCurrent (co, active);

	return co;
}

static void
nmt_newt_listbox_activated (NmtNewtWidget *widget)
{
	NmtNewtListbox *listbox = NMT_NEWT_LISTBOX (widget);
	newtComponent co = nmt_newt_component_get_component (NMT_NEWT_COMPONENT (widget));

	nmt_newt_listbox_set_active (listbox, GPOINTER_TO_UINT (newtListboxGetCurrent (co)));

	NMT_NEWT_WIDGET_CLASS (nmt_newt_listbox_parent_class)->activated (widget);
}

static void
nmt_newt_listbox_set_property (GObject      *object,
                               guint         prop_id,
                               const GValue *value,
                               GParamSpec   *pspec)
{
	NmtNewtListbox *listbox = NMT_NEWT_LISTBOX (object);
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (object);

	switch (prop_id) {
	case PROP_HEIGHT:
		priv->height = g_value_get_int (value);
		priv->fixed_height = (priv->height != 0);
		break;
	case PROP_FLAGS:
		priv->flags = g_value_get_uint (value);
		break;
	case PROP_ACTIVE:
		nmt_newt_listbox_set_active (listbox, g_value_get_int (value));
		break;
	case PROP_ACTIVE_KEY:
		nmt_newt_listbox_set_active_key (listbox, g_value_get_pointer (value));
		break;
	case PROP_SKIP_NULL_KEYS:
		priv->skip_null_keys = g_value_get_boolean (value);
		break;
	default:
		G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
		break;
	}
}

static void
nmt_newt_listbox_get_property (GObject    *object,
                             guint       prop_id,
                             GValue     *value,
                             GParamSpec *pspec)
{
	NmtNewtListboxPrivate *priv = NMT_NEWT_LISTBOX_GET_PRIVATE (object);

	switch (prop_id) {
	case PROP_HEIGHT:
		g_value_set_int (value, priv->height);
		break;
	case PROP_FLAGS:
		g_value_set_uint (value, priv->flags);
		break;
	case PROP_ACTIVE:
		g_value_set_int (value, priv->active);
		break;
	case PROP_ACTIVE_KEY:
		g_value_set_pointer (value, priv->active_key);
		break;
	case PROP_SKIP_NULL_KEYS:
		g_value_set_boolean (value, priv->skip_null_keys);
		break;
	default:
		G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
		break;
	}
}

static void
nmt_newt_listbox_class_init (NmtNewtListboxClass *listbox_class)
{
	GObjectClass *object_class = G_OBJECT_CLASS (listbox_class);
	NmtNewtWidgetClass *widget_class = NMT_NEWT_WIDGET_CLASS (listbox_class);
	NmtNewtComponentClass *component_class = NMT_NEWT_COMPONENT_CLASS (listbox_class);

	g_type_class_add_private (listbox_class, sizeof (NmtNewtListboxPrivate));

	/* virtual methods */
	object_class->set_property = nmt_newt_listbox_set_property;
	object_class->get_property = nmt_newt_listbox_get_property;
	object_class->finalize     = nmt_newt_listbox_finalize;

	widget_class->size_request  = nmt_newt_listbox_size_request;
	widget_class->size_allocate = nmt_newt_listbox_size_allocate;
	widget_class->activated     = nmt_newt_listbox_activated;

	component_class->build_component = nmt_newt_listbox_build_component;

	/* properties */

	/**
	 * NmtNewtListbox:height:
	 *
	 * The listbox's height, or -1 if it has no fixed height.
	 */
	g_object_class_install_property
		(object_class, PROP_HEIGHT,
		 g_param_spec_int ("height", "", "",
		                   -1, 255, -1,
		                   G_PARAM_READWRITE |
		                   G_PARAM_STATIC_STRINGS));
	/**
	 * NmtNewtListbox:flags:
	 *
	 * The listbox's #NmtNewtListboxFlags.
	 */
	g_object_class_install_property
		(object_class, PROP_FLAGS,
		 g_param_spec_uint ("flags", "", "",
		                    0, 0xFFFF, 0,
		                    G_PARAM_READWRITE |
		                    G_PARAM_CONSTRUCT_ONLY |
		                    G_PARAM_STATIC_STRINGS));
	/**
	 * NmtNewtListbox:active:
	 *
	 * The currently-selected row.
	 */
	g_object_class_install_property
		(object_class, PROP_ACTIVE,
		 g_param_spec_int ("active", "", "",
		                   0, G_MAXINT, 0,
		                   G_PARAM_READWRITE |
		                   G_PARAM_STATIC_STRINGS));
	/**
	 * NmtNewtListbox:active-key:
	 *
	 * The key of the currently-selected row.
	 */
	g_object_class_install_property
		(object_class, PROP_ACTIVE_KEY,
		 g_param_spec_pointer ("active-key", "", "",
		                       G_PARAM_READWRITE |
		                       G_PARAM_STATIC_STRINGS));
	/**
	 * NmtNewtListbox:skip-null-keys:
	 *
	 * If %TRUE, rows with %NULL key values will be skipped over when
	 * navigating the list with the arrow keys.
	 */
	g_object_class_install_property
		(object_class, PROP_SKIP_NULL_KEYS,
		 g_param_spec_boolean ("skip-null-keys", "", "",
		                       FALSE,
		                       G_PARAM_READWRITE |
		                       G_PARAM_CONSTRUCT_ONLY |
		                       G_PARAM_STATIC_STRINGS));
}