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));
}