2019-12-09 01:22:06 +00:00
|
|
|
/*
|
|
|
|
* Copyright © 2019 Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*
|
|
|
|
* Authors: Matthias Clasen <mclasen@redhat.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "gtkdropdown.h"
|
|
|
|
|
|
|
|
#include "gtkbuiltiniconprivate.h"
|
|
|
|
#include "gtkintl.h"
|
|
|
|
#include "gtklistview.h"
|
|
|
|
#include "gtklistitemfactory.h"
|
|
|
|
#include "gtksignallistitemfactory.h"
|
|
|
|
#include "gtklistitemwidgetprivate.h"
|
|
|
|
#include "gtkpopover.h"
|
|
|
|
#include "gtkprivate.h"
|
|
|
|
#include "gtksingleselection.h"
|
|
|
|
#include "gtkfilterlistmodel.h"
|
|
|
|
#include "gtkstringfilter.h"
|
|
|
|
#include "gtkmultifilter.h"
|
|
|
|
#include "gtkstylecontext.h"
|
|
|
|
#include "gtkwidgetprivate.h"
|
|
|
|
#include "gtknative.h"
|
|
|
|
#include "gtktogglebutton.h"
|
|
|
|
#include "gtkexpression.h"
|
|
|
|
#include "gtkstack.h"
|
|
|
|
#include "gtksearchentry.h"
|
|
|
|
#include "gtklabel.h"
|
|
|
|
#include "gtklistitem.h"
|
|
|
|
#include "gtkbuildable.h"
|
|
|
|
#include "gtkbuilderprivate.h"
|
2020-06-21 22:00:29 +00:00
|
|
|
#include "gtkstringlist.h"
|
2019-12-09 01:22:06 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* SECTION:gtkdropdown
|
|
|
|
* @Title: GtkDropDown
|
|
|
|
* @Short_description: Choose an item from a list
|
|
|
|
* @See_also: #GtkComboBox
|
|
|
|
*
|
|
|
|
* GtkDropDown is a widget that allows the user to choose an item
|
|
|
|
* from a list of options. The GtkDropDown displays the selected
|
|
|
|
* choice.
|
|
|
|
*
|
|
|
|
* The options are given to GtkDropDown in the form of #GListModel,
|
|
|
|
* and how the individual options are represented is determined by
|
|
|
|
* a #GtkListItemFactory. The default factory displays simple strings,
|
|
|
|
* and expects to obtain these from the model by evaluating an expression
|
|
|
|
* that has to be provided via gtk_drop_down_set_expression().
|
|
|
|
*
|
|
|
|
* The convenience method gtk_drop_down_set_from_strings() can be used
|
|
|
|
* to set up a model that is populated from an array of strings and
|
|
|
|
* an expression for obtaining those strings.
|
|
|
|
*
|
|
|
|
* GtkDropDown can optionally allow search in the popup, which is
|
|
|
|
* useful if the list of options is long. To enable the search entry,
|
|
|
|
* use gtk_drop_down_set_enable_search().
|
|
|
|
*
|
2020-07-17 01:58:52 +00:00
|
|
|
* # CSS nodes
|
2020-06-07 13:01:20 +00:00
|
|
|
*
|
|
|
|
* GtkDropDown has a single CSS node with name dropdown,
|
|
|
|
* with the button and popover nodes as children.
|
2019-12-09 01:22:06 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
struct _GtkDropDown
|
|
|
|
{
|
|
|
|
GtkWidget parent_instance;
|
|
|
|
|
|
|
|
GtkListItemFactory *factory;
|
|
|
|
GtkListItemFactory *list_factory;
|
|
|
|
GListModel *model;
|
|
|
|
GListModel *selection;
|
|
|
|
GListModel *filter_model;
|
|
|
|
GListModel *popup_selection;
|
|
|
|
|
|
|
|
GtkWidget *popup;
|
|
|
|
GtkWidget *button;
|
|
|
|
|
|
|
|
GtkWidget *popup_list;
|
|
|
|
GtkWidget *button_stack;
|
|
|
|
GtkWidget *button_item;
|
|
|
|
GtkWidget *button_placeholder;
|
|
|
|
GtkWidget *search_entry;
|
|
|
|
|
|
|
|
gboolean enable_search;
|
|
|
|
GtkExpression *expression;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _GtkDropDownClass
|
|
|
|
{
|
|
|
|
GtkWidgetClass parent_class;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
|
|
|
PROP_FACTORY,
|
|
|
|
PROP_LIST_FACTORY,
|
|
|
|
PROP_MODEL,
|
|
|
|
PROP_SELECTED,
|
2020-06-20 17:11:48 +00:00
|
|
|
PROP_SELECTED_ITEM,
|
2019-12-09 01:22:06 +00:00
|
|
|
PROP_ENABLE_SEARCH,
|
|
|
|
PROP_EXPRESSION,
|
|
|
|
|
|
|
|
N_PROPS
|
|
|
|
};
|
|
|
|
|
2020-06-21 22:00:29 +00:00
|
|
|
G_DEFINE_TYPE (GtkDropDown, gtk_drop_down, GTK_TYPE_WIDGET)
|
2019-12-09 01:22:06 +00:00
|
|
|
|
|
|
|
static GParamSpec *properties[N_PROPS] = { NULL, };
|
|
|
|
|
|
|
|
static void
|
|
|
|
button_toggled (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = data;
|
|
|
|
|
|
|
|
if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (widget)))
|
|
|
|
gtk_popover_popup (GTK_POPOVER (self->popup));
|
|
|
|
else
|
|
|
|
gtk_popover_popdown (GTK_POPOVER (self->popup));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
popover_closed (GtkPopover *popover,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = data;
|
|
|
|
|
|
|
|
gtk_editable_set_text (GTK_EDITABLE (self->search_entry), "");
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (self->button), FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
row_activated (GtkListView *listview,
|
|
|
|
guint position,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = data;
|
|
|
|
GtkFilter *filter;
|
|
|
|
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (self->button), FALSE);
|
|
|
|
gtk_popover_popdown (GTK_POPOVER (self->popup));
|
|
|
|
|
|
|
|
/* reset the filter so positions are 1-1 */
|
|
|
|
filter = gtk_filter_list_model_get_filter (GTK_FILTER_LIST_MODEL (self->filter_model));
|
|
|
|
if (GTK_IS_STRING_FILTER (filter))
|
|
|
|
gtk_string_filter_set_search (GTK_STRING_FILTER (filter), "");
|
|
|
|
gtk_drop_down_set_selected (self, gtk_single_selection_get_selected (GTK_SINGLE_SELECTION (self->popup_selection)));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
selection_changed (GtkSingleSelection *selection,
|
|
|
|
GParamSpec *pspec,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = data;
|
|
|
|
guint selected;
|
|
|
|
gpointer item;
|
|
|
|
GtkFilter *filter;
|
|
|
|
|
|
|
|
selected = gtk_single_selection_get_selected (GTK_SINGLE_SELECTION (self->selection));
|
|
|
|
item = gtk_single_selection_get_selected_item (GTK_SINGLE_SELECTION (self->selection));
|
|
|
|
|
|
|
|
if (selected == GTK_INVALID_LIST_POSITION)
|
|
|
|
{
|
|
|
|
gtk_stack_set_visible_child_name (GTK_STACK (self->button_stack), "empty");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gtk_stack_set_visible_child_name (GTK_STACK (self->button_stack), "item");
|
|
|
|
gtk_list_item_widget_update (GTK_LIST_ITEM_WIDGET (self->button_item), selected, item, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* reset the filter so positions are 1-1 */
|
|
|
|
filter = gtk_filter_list_model_get_filter (GTK_FILTER_LIST_MODEL (self->filter_model));
|
|
|
|
if (GTK_IS_STRING_FILTER (filter))
|
|
|
|
gtk_string_filter_set_search (GTK_STRING_FILTER (filter), "");
|
|
|
|
gtk_single_selection_set_selected (GTK_SINGLE_SELECTION (self->popup_selection), selected);
|
2020-06-21 00:17:47 +00:00
|
|
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_SELECTED]);
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_SELECTED_ITEM]);
|
2019-12-09 01:22:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
update_filter (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
if (self->filter_model)
|
|
|
|
{
|
|
|
|
GtkFilter *filter;
|
|
|
|
|
|
|
|
if (self->expression)
|
|
|
|
{
|
2020-07-12 05:02:04 +00:00
|
|
|
filter = gtk_string_filter_new (gtk_expression_ref (self->expression));
|
2019-12-09 01:22:06 +00:00
|
|
|
gtk_string_filter_set_match_mode (GTK_STRING_FILTER (filter), GTK_STRING_FILTER_MATCH_MODE_PREFIX);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
filter = gtk_every_filter_new ();
|
|
|
|
gtk_filter_list_model_set_filter (GTK_FILTER_LIST_MODEL (self->filter_model), filter);
|
|
|
|
g_object_unref (filter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
search_changed (GtkSearchEntry *entry, gpointer data)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = data;
|
|
|
|
const char *text;
|
|
|
|
GtkFilter *filter;
|
|
|
|
|
|
|
|
text = gtk_editable_get_text (GTK_EDITABLE (entry));
|
|
|
|
|
|
|
|
filter = gtk_filter_list_model_get_filter (GTK_FILTER_LIST_MODEL (self->filter_model));
|
|
|
|
if (GTK_IS_STRING_FILTER (filter))
|
|
|
|
gtk_string_filter_set_search (GTK_STRING_FILTER (filter), text);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
search_stop (GtkSearchEntry *entry, gpointer data)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = data;
|
|
|
|
GtkFilter *filter;
|
|
|
|
|
|
|
|
filter = gtk_filter_list_model_get_filter (GTK_FILTER_LIST_MODEL (self->filter_model));
|
|
|
|
if (GTK_IS_STRING_FILTER (filter))
|
|
|
|
{
|
|
|
|
if (gtk_string_filter_get_search (GTK_STRING_FILTER (filter)))
|
|
|
|
gtk_string_filter_set_search (GTK_STRING_FILTER (filter), NULL);
|
|
|
|
else
|
|
|
|
gtk_popover_popdown (GTK_POPOVER (self->popup));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_drop_down_dispose (GObject *object)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = GTK_DROP_DOWN (object);
|
|
|
|
|
|
|
|
g_clear_pointer (&self->popup, gtk_widget_unparent);
|
|
|
|
g_clear_pointer (&self->button, gtk_widget_unparent);
|
|
|
|
|
|
|
|
g_clear_object (&self->model);
|
|
|
|
if (self->selection)
|
|
|
|
g_signal_handlers_disconnect_by_func (self->selection, selection_changed, self);
|
|
|
|
g_clear_object (&self->filter_model);
|
|
|
|
g_clear_pointer (&self->expression, gtk_expression_unref);
|
|
|
|
g_clear_object (&self->selection);
|
|
|
|
g_clear_object (&self->popup_selection);
|
|
|
|
g_clear_object (&self->factory);
|
|
|
|
g_clear_object (&self->list_factory);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (gtk_drop_down_parent_class)->dispose (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_drop_down_get_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = GTK_DROP_DOWN (object);
|
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
|
|
|
case PROP_FACTORY:
|
|
|
|
g_value_set_object (value, self->factory);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_LIST_FACTORY:
|
|
|
|
g_value_set_object (value, self->list_factory);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_MODEL:
|
|
|
|
g_value_set_object (value, self->model);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_SELECTED:
|
|
|
|
g_value_set_uint (value, gtk_drop_down_get_selected (self));
|
|
|
|
break;
|
|
|
|
|
2020-06-20 17:11:48 +00:00
|
|
|
case PROP_SELECTED_ITEM:
|
|
|
|
g_value_set_object (value, gtk_drop_down_get_selected_item (self));
|
|
|
|
break;
|
|
|
|
|
2019-12-09 01:22:06 +00:00
|
|
|
case PROP_ENABLE_SEARCH:
|
|
|
|
g_value_set_boolean (value, self->enable_search);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_EXPRESSION:
|
2020-06-01 20:07:53 +00:00
|
|
|
gtk_value_set_expression (value, self->expression);
|
2019-12-09 01:22:06 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_drop_down_set_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = GTK_DROP_DOWN (object);
|
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
|
|
|
case PROP_FACTORY:
|
|
|
|
gtk_drop_down_set_factory (self, g_value_get_object (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_LIST_FACTORY:
|
|
|
|
gtk_drop_down_set_list_factory (self, g_value_get_object (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_MODEL:
|
|
|
|
gtk_drop_down_set_model (self, g_value_get_object (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_SELECTED:
|
|
|
|
gtk_drop_down_set_selected (self, g_value_get_uint (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_ENABLE_SEARCH:
|
|
|
|
gtk_drop_down_set_enable_search (self, g_value_get_boolean (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_EXPRESSION:
|
2020-06-01 20:07:53 +00:00
|
|
|
gtk_drop_down_set_expression (self, gtk_value_get_expression (value));
|
2019-12-09 01:22:06 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_drop_down_measure (GtkWidget *widget,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
int size,
|
|
|
|
int *minimum,
|
|
|
|
int *natural,
|
|
|
|
int *minimum_baseline,
|
|
|
|
int *natural_baseline)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = GTK_DROP_DOWN (widget);
|
|
|
|
|
|
|
|
gtk_widget_measure (self->button,
|
|
|
|
orientation,
|
|
|
|
size,
|
|
|
|
minimum, natural,
|
|
|
|
minimum_baseline, natural_baseline);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_drop_down_size_allocate (GtkWidget *widget,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
int baseline)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = GTK_DROP_DOWN (widget);
|
|
|
|
|
|
|
|
gtk_widget_size_allocate (self->button, &(GtkAllocation) { 0, 0, width, height }, baseline);
|
|
|
|
|
2020-06-11 22:29:58 +00:00
|
|
|
gtk_widget_set_size_request (self->popup, width, -1);
|
|
|
|
|
2019-12-09 01:22:06 +00:00
|
|
|
gtk_native_check_resize (GTK_NATIVE (self->popup));
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gtk_drop_down_focus (GtkWidget *widget,
|
|
|
|
GtkDirectionType direction)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = GTK_DROP_DOWN (widget);
|
|
|
|
|
|
|
|
if (self->popup && gtk_widget_get_visible (self->popup))
|
|
|
|
return gtk_widget_child_focus (self->popup, direction);
|
|
|
|
else
|
|
|
|
return gtk_widget_child_focus (self->button, direction);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gtk_drop_down_grab_focus (GtkWidget *widget)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = GTK_DROP_DOWN (widget);
|
|
|
|
|
|
|
|
return gtk_widget_grab_focus (self->button);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_drop_down_class_init (GtkDropDownClass *klass)
|
|
|
|
{
|
|
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
gobject_class->dispose = gtk_drop_down_dispose;
|
|
|
|
gobject_class->get_property = gtk_drop_down_get_property;
|
|
|
|
gobject_class->set_property = gtk_drop_down_set_property;
|
|
|
|
|
|
|
|
widget_class->measure = gtk_drop_down_measure;
|
|
|
|
widget_class->size_allocate = gtk_drop_down_size_allocate;
|
|
|
|
widget_class->focus = gtk_drop_down_focus;
|
|
|
|
widget_class->grab_focus = gtk_drop_down_grab_focus;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GtkDropDown:factory:
|
|
|
|
*
|
|
|
|
* Factory for populating list items.
|
|
|
|
*/
|
|
|
|
properties[PROP_FACTORY] =
|
|
|
|
g_param_spec_object ("factory",
|
|
|
|
P_("Factory"),
|
|
|
|
P_("Factory for populating list items"),
|
|
|
|
GTK_TYPE_LIST_ITEM_FACTORY,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GtkDropDown:list-factory:
|
|
|
|
*
|
|
|
|
* The factory for populating list items in the popup.
|
|
|
|
*
|
|
|
|
* If this is not set, #GtkDropDown:factory is used.
|
|
|
|
*/
|
|
|
|
properties[PROP_LIST_FACTORY] =
|
|
|
|
g_param_spec_object ("list-factory",
|
|
|
|
P_("List Factory"),
|
|
|
|
P_("Factory for populating list items"),
|
|
|
|
GTK_TYPE_LIST_ITEM_FACTORY,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GtkDropDown:model:
|
|
|
|
*
|
|
|
|
* Model for the displayed items.
|
|
|
|
*/
|
|
|
|
properties[PROP_MODEL] =
|
|
|
|
g_param_spec_object ("model",
|
|
|
|
P_("Model"),
|
|
|
|
P_("Model for the displayed items"),
|
|
|
|
G_TYPE_LIST_MODEL,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GtkDropDown:selected:
|
|
|
|
*
|
|
|
|
* The position of the selected item in #GtkDropDown:model,
|
|
|
|
* or #GTK_INVALID_LIST_POSITION if no item is selected.
|
|
|
|
*/
|
|
|
|
properties[PROP_SELECTED] =
|
|
|
|
g_param_spec_uint ("selected",
|
|
|
|
P_("Selected"),
|
|
|
|
P_("Position of the selected item"),
|
|
|
|
0, G_MAXUINT, GTK_INVALID_LIST_POSITION,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
|
2020-06-20 17:11:48 +00:00
|
|
|
/**
|
|
|
|
* GtkDropDown:selected-item:
|
|
|
|
*
|
|
|
|
* The selected item.
|
|
|
|
*/
|
|
|
|
properties[PROP_SELECTED_ITEM] =
|
|
|
|
g_param_spec_object ("selected-item",
|
|
|
|
P_("Selected Item"),
|
|
|
|
P_("The selected item"),
|
|
|
|
G_TYPE_OBJECT,
|
|
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
|
|
|
|
|
2019-12-09 01:22:06 +00:00
|
|
|
/**
|
|
|
|
* GtkDropDown:enable-search:
|
|
|
|
*
|
|
|
|
* Whether to show a search entry in the popup.
|
|
|
|
*
|
|
|
|
* Note that search requires #GtkDropDown:expression to be set.
|
|
|
|
*/
|
|
|
|
properties[PROP_ENABLE_SEARCH] =
|
|
|
|
g_param_spec_boolean ("enable-search",
|
|
|
|
P_("Enable search"),
|
|
|
|
P_("Whether to show a search entry in the popup"),
|
|
|
|
FALSE,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
|
|
|
|
/**
|
2020-06-01 20:07:53 +00:00
|
|
|
* GtkDropDown:expression: (type GtkExpression)
|
2019-12-09 01:22:06 +00:00
|
|
|
*
|
|
|
|
* An expression to evaluate to obtain strings to match against the search
|
|
|
|
* term (see #GtkDropDown:enable-search). If #GtkDropDown:factory is not set,
|
|
|
|
* the expression is also used to bind strings to labels produced by a
|
|
|
|
* default factory.
|
|
|
|
*/
|
|
|
|
properties[PROP_EXPRESSION] =
|
2020-06-01 20:07:53 +00:00
|
|
|
gtk_param_spec_expression ("expression",
|
|
|
|
P_("Expression"),
|
|
|
|
P_("Expression to determine strings to search for"),
|
|
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
2019-12-09 01:22:06 +00:00
|
|
|
|
|
|
|
g_object_class_install_properties (gobject_class, N_PROPS, properties);
|
|
|
|
|
|
|
|
gtk_widget_class_set_template_from_resource (widget_class, "/org/gtk/libgtk/ui/gtkdropdown.ui");
|
|
|
|
gtk_widget_class_bind_template_child (widget_class, GtkDropDown, button);
|
|
|
|
gtk_widget_class_bind_template_child (widget_class, GtkDropDown, button_stack);
|
|
|
|
gtk_widget_class_bind_template_child (widget_class, GtkDropDown, button_item);
|
|
|
|
gtk_widget_class_bind_template_child (widget_class, GtkDropDown, popup);
|
|
|
|
gtk_widget_class_bind_template_child (widget_class, GtkDropDown, popup_list);
|
|
|
|
gtk_widget_class_bind_template_child (widget_class, GtkDropDown, search_entry);
|
|
|
|
|
|
|
|
gtk_widget_class_bind_template_callback (widget_class, row_activated);
|
|
|
|
gtk_widget_class_bind_template_callback (widget_class, button_toggled);
|
|
|
|
gtk_widget_class_bind_template_callback (widget_class, popover_closed);
|
|
|
|
gtk_widget_class_bind_template_callback (widget_class, search_changed);
|
|
|
|
gtk_widget_class_bind_template_callback (widget_class, search_stop);
|
|
|
|
|
2020-06-07 13:01:20 +00:00
|
|
|
gtk_widget_class_set_css_name (widget_class, I_("dropdown"));
|
2019-12-09 01:22:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
setup_item (GtkSignalListItemFactory *factory,
|
|
|
|
GtkListItem *list_item,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkWidget *label;
|
|
|
|
|
|
|
|
label = gtk_label_new (NULL);
|
|
|
|
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
|
|
|
gtk_list_item_set_child (list_item, label);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
bind_item (GtkSignalListItemFactory *factory,
|
|
|
|
GtkListItem *list_item,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkDropDown *self = data;
|
|
|
|
gpointer item;
|
|
|
|
GtkWidget *label;
|
|
|
|
GValue value = G_VALUE_INIT;
|
|
|
|
|
|
|
|
item = gtk_list_item_get_item (list_item);
|
|
|
|
label = gtk_list_item_get_child (list_item);
|
|
|
|
|
2020-06-21 22:00:29 +00:00
|
|
|
if (self->expression &&
|
|
|
|
gtk_expression_evaluate (self->expression, item, &value))
|
2019-12-09 01:22:06 +00:00
|
|
|
{
|
|
|
|
gtk_label_set_label (GTK_LABEL (label), g_value_get_string (&value));
|
|
|
|
g_value_unset (&value);
|
|
|
|
}
|
2020-06-21 22:00:29 +00:00
|
|
|
else if (GTK_IS_STRING_OBJECT (item))
|
|
|
|
{
|
|
|
|
const char *string;
|
|
|
|
|
|
|
|
string = gtk_string_object_get_string (GTK_STRING_OBJECT (item));
|
|
|
|
gtk_label_set_label (GTK_LABEL (label), string);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_critical ("Either GtkDropDown:factory or GtkDropDown:expression must be set");
|
|
|
|
}
|
2019-12-09 01:22:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_default_factory (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
GtkListItemFactory *factory;
|
|
|
|
|
|
|
|
factory = gtk_signal_list_item_factory_new ();
|
|
|
|
|
|
|
|
g_signal_connect (factory, "setup", G_CALLBACK (setup_item), self);
|
|
|
|
g_signal_connect (factory, "bind", G_CALLBACK (bind_item), self);
|
|
|
|
|
|
|
|
gtk_drop_down_set_factory (self, factory);
|
|
|
|
|
|
|
|
g_object_unref (factory);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_drop_down_init (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
g_type_ensure (GTK_TYPE_BUILTIN_ICON);
|
|
|
|
g_type_ensure (GTK_TYPE_LIST_ITEM_WIDGET);
|
|
|
|
|
|
|
|
gtk_widget_init_template (GTK_WIDGET (self));
|
|
|
|
|
|
|
|
set_default_factory (self);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_new:
|
|
|
|
*
|
|
|
|
* Creates a new empty #GtkDropDown.
|
|
|
|
*
|
|
|
|
* You most likely want to call gtk_drop_down_set_factory() to
|
|
|
|
* set up a way to map its items to widgets and gtk_drop_down_set_model()
|
|
|
|
* to set a model to provide items next.
|
|
|
|
*
|
|
|
|
* Returns: a new #GtkDropDown
|
|
|
|
**/
|
|
|
|
GtkWidget *
|
|
|
|
gtk_drop_down_new (void)
|
|
|
|
{
|
|
|
|
return g_object_new (GTK_TYPE_DROP_DOWN, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_get_model:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
*
|
|
|
|
* Gets the model that provides the displayed items.
|
|
|
|
*
|
|
|
|
* Returns: (nullable) (transfer none): The model in use
|
|
|
|
**/
|
|
|
|
GListModel *
|
|
|
|
gtk_drop_down_get_model (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_DROP_DOWN (self), NULL);
|
|
|
|
|
|
|
|
return self->model;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_set_model:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
* @model: (allow-none) (transfer none): the model to use or %NULL for none
|
|
|
|
*
|
|
|
|
* Sets the #GListModel to use.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_drop_down_set_model (GtkDropDown *self,
|
|
|
|
GListModel *model)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_DROP_DOWN (self));
|
|
|
|
g_return_if_fail (model == NULL || G_IS_LIST_MODEL (model));
|
|
|
|
|
|
|
|
if (!g_set_object (&self->model, model))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (model == NULL)
|
|
|
|
{
|
|
|
|
gtk_list_view_set_model (GTK_LIST_VIEW (self->popup_list), NULL);
|
|
|
|
|
|
|
|
if (self->selection)
|
|
|
|
g_signal_handlers_disconnect_by_func (self->selection, selection_changed, self);
|
|
|
|
|
|
|
|
g_clear_object (&self->selection);
|
|
|
|
g_clear_object (&self->filter_model);
|
|
|
|
g_clear_object (&self->popup_selection);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GListModel *filter_model;
|
|
|
|
GListModel *selection;
|
|
|
|
|
|
|
|
filter_model = G_LIST_MODEL (gtk_filter_list_model_new (model, NULL));
|
|
|
|
g_set_object (&self->filter_model, filter_model);
|
|
|
|
g_object_unref (filter_model);
|
|
|
|
|
|
|
|
update_filter (self);
|
|
|
|
|
|
|
|
selection = G_LIST_MODEL (gtk_single_selection_new (filter_model));
|
|
|
|
g_set_object (&self->popup_selection, selection);
|
|
|
|
gtk_list_view_set_model (GTK_LIST_VIEW (self->popup_list), selection);
|
|
|
|
g_object_unref (selection);
|
|
|
|
|
|
|
|
selection = G_LIST_MODEL (gtk_single_selection_new (model));
|
|
|
|
g_set_object (&self->selection, selection);
|
|
|
|
g_object_unref (selection);
|
|
|
|
|
|
|
|
g_signal_connect (self->selection, "notify::selected", G_CALLBACK (selection_changed), self);
|
|
|
|
selection_changed (GTK_SINGLE_SELECTION (self->selection), NULL, self);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_MODEL]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_get_factory:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
*
|
|
|
|
* Gets the factory that's currently used to populate list items.
|
|
|
|
*
|
|
|
|
* The factory returned by this function is always used for the
|
|
|
|
* item in the button. It is also used for items in the popup
|
|
|
|
* if #GtkDropDown:list-factory is not set.
|
|
|
|
*
|
|
|
|
* Returns: (nullable) (transfer none): The factory in use
|
|
|
|
**/
|
|
|
|
GtkListItemFactory *
|
|
|
|
gtk_drop_down_get_factory (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_DROP_DOWN (self), NULL);
|
|
|
|
|
|
|
|
return self->factory;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_set_factory:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
* @factory: (allow-none) (transfer none): the factory to use or %NULL for none
|
|
|
|
*
|
|
|
|
* Sets the #GtkListItemFactory to use for populating list items.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
gtk_drop_down_set_factory (GtkDropDown *self,
|
|
|
|
GtkListItemFactory *factory)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_DROP_DOWN (self));
|
|
|
|
g_return_if_fail (factory == NULL || GTK_LIST_ITEM_FACTORY (factory));
|
|
|
|
|
|
|
|
if (!g_set_object (&self->factory, factory))
|
|
|
|
return;
|
|
|
|
|
|
|
|
gtk_list_item_widget_set_factory (GTK_LIST_ITEM_WIDGET (self->button_item), factory);
|
|
|
|
if (self->list_factory == NULL)
|
|
|
|
gtk_list_view_set_factory (GTK_LIST_VIEW (self->popup_list), factory);
|
|
|
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_FACTORY]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_get_list_factory:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
*
|
|
|
|
* Gets the factory that's currently used to populate list items in the popup.
|
|
|
|
*
|
|
|
|
* Returns: (nullable) (transfer none): The factory in use
|
|
|
|
**/
|
|
|
|
GtkListItemFactory *
|
|
|
|
gtk_drop_down_get_list_factory (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_DROP_DOWN (self), NULL);
|
|
|
|
|
|
|
|
return self->list_factory;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_set_list_factory:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
* @factory: (allow-none) (transfer none): the factory to use or %NULL for none
|
|
|
|
*
|
|
|
|
* Sets the #GtkListItemFactory to use for populating list items in the popup.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
gtk_drop_down_set_list_factory (GtkDropDown *self,
|
|
|
|
GtkListItemFactory *factory)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_DROP_DOWN (self));
|
|
|
|
g_return_if_fail (factory == NULL || GTK_LIST_ITEM_FACTORY (factory));
|
|
|
|
|
|
|
|
if (!g_set_object (&self->list_factory, factory))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (self->list_factory != NULL)
|
|
|
|
gtk_list_view_set_factory (GTK_LIST_VIEW (self->popup_list), self->list_factory);
|
|
|
|
else
|
|
|
|
gtk_list_view_set_factory (GTK_LIST_VIEW (self->popup_list), self->factory);
|
|
|
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_LIST_FACTORY]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_set_selected:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
* @position: the position of the item to select, or #GTK_INVALID_LIST_POSITION
|
|
|
|
*
|
|
|
|
* Selects the item at the given position.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
gtk_drop_down_set_selected (GtkDropDown *self,
|
|
|
|
guint position)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_DROP_DOWN (self));
|
|
|
|
|
|
|
|
if (self->selection == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (gtk_single_selection_get_selected (GTK_SINGLE_SELECTION (self->selection)) == position)
|
|
|
|
return;
|
|
|
|
|
|
|
|
gtk_single_selection_set_selected (GTK_SINGLE_SELECTION (self->selection), position);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_get_selected:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
*
|
|
|
|
* Gets the position of the selected item.
|
|
|
|
*
|
|
|
|
* Returns: the position of the selected item, or #GTK_INVALID_LIST_POSITION
|
|
|
|
* if not item is selected
|
|
|
|
**/
|
|
|
|
guint
|
|
|
|
gtk_drop_down_get_selected (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_DROP_DOWN (self), GTK_INVALID_LIST_POSITION);
|
|
|
|
|
|
|
|
if (self->selection == NULL)
|
|
|
|
return GTK_INVALID_LIST_POSITION;
|
|
|
|
|
|
|
|
return gtk_single_selection_get_selected (GTK_SINGLE_SELECTION (self->selection));
|
|
|
|
}
|
|
|
|
|
2020-06-20 17:11:48 +00:00
|
|
|
/**
|
|
|
|
* gtk_drop_down_get_selected_item:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
*
|
|
|
|
* Gets the selected item. If no item is selected, %NULL is returned.
|
|
|
|
*
|
2020-06-26 07:59:12 +00:00
|
|
|
* Returns: (transfer none) (type GObject) (nullable): The selected item
|
2020-06-20 17:11:48 +00:00
|
|
|
*/
|
|
|
|
gpointer
|
|
|
|
gtk_drop_down_get_selected_item (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_DROP_DOWN (self), NULL);
|
|
|
|
|
|
|
|
if (self->selection == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return gtk_single_selection_get_selected_item (GTK_SINGLE_SELECTION (self->selection));
|
|
|
|
}
|
|
|
|
|
2019-12-09 01:22:06 +00:00
|
|
|
/**
|
|
|
|
* gtk_drop_down_set_enable_search:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
* @enable_search: whether to enable search
|
|
|
|
*
|
|
|
|
* Sets whether a search entry will be shown in the popup that
|
|
|
|
* allows to search for items in the list.
|
|
|
|
*
|
|
|
|
* Note that #GtkDropDown:expression must be set for search to work.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
gtk_drop_down_set_enable_search (GtkDropDown *self,
|
|
|
|
gboolean enable_search)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_DROP_DOWN (self));
|
|
|
|
|
|
|
|
if (self->enable_search == enable_search)
|
|
|
|
return;
|
|
|
|
|
|
|
|
self->enable_search = enable_search;
|
|
|
|
|
|
|
|
gtk_editable_set_text (GTK_EDITABLE (self->search_entry), "");
|
|
|
|
gtk_widget_set_visible (self->search_entry, enable_search);
|
|
|
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_ENABLE_SEARCH]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_get_enable_search:
|
2020-06-06 01:36:27 +00:00
|
|
|
* @self: a #GtkDropDown
|
2019-12-09 01:22:06 +00:00
|
|
|
*
|
|
|
|
* Returns whether search is enabled.
|
|
|
|
*
|
|
|
|
* Returns: %TRUE if the popup includes a search entry
|
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
gtk_drop_down_get_enable_search (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_DROP_DOWN (self), FALSE);
|
|
|
|
|
|
|
|
return self->enable_search;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_set_expression:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
* @expression: (nullable): a #GtkExpression, or %NULL
|
|
|
|
*
|
|
|
|
* Sets the expression that gets evaluated to obtain strings from items
|
|
|
|
* when searching in the popup. The expression must have a value type of
|
|
|
|
* #GTK_TYPE_STRING.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_drop_down_set_expression (GtkDropDown *self,
|
|
|
|
GtkExpression *expression)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_DROP_DOWN (self));
|
|
|
|
g_return_if_fail (expression == NULL ||
|
|
|
|
gtk_expression_get_value_type (expression) == G_TYPE_STRING);
|
|
|
|
|
|
|
|
if (self->expression == expression)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (self->expression)
|
|
|
|
gtk_expression_unref (self->expression);
|
|
|
|
self->expression = expression;
|
|
|
|
if (self->expression)
|
|
|
|
gtk_expression_ref (self->expression);
|
|
|
|
|
|
|
|
update_filter (self);
|
|
|
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_EXPRESSION]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_get_expression:
|
|
|
|
* @self: a #GtkDropDown
|
|
|
|
*
|
|
|
|
* Gets the expression set with gtk_drop_down_set_expression().
|
|
|
|
*
|
|
|
|
* Returns: (nullable) (transfer none): a #GtkExpression or %NULL
|
|
|
|
*/
|
|
|
|
GtkExpression *
|
|
|
|
gtk_drop_down_get_expression (GtkDropDown *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_DROP_DOWN (self), NULL);
|
|
|
|
|
|
|
|
return self->expression;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_drop_down_set_from_strings:
|
|
|
|
* @self: a #GtkDropDown
|
2020-07-07 11:04:13 +00:00
|
|
|
* @texts: (array zero-terminated=1) (element-type utf8): a %NULL-terminated string array
|
2019-12-09 01:22:06 +00:00
|
|
|
*
|
|
|
|
* Populates @self with the strings in @text,
|
|
|
|
* by creating a suitable model and factory.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_drop_down_set_from_strings (GtkDropDown *self,
|
2020-06-06 01:36:27 +00:00
|
|
|
const char *const *texts)
|
2019-12-09 01:22:06 +00:00
|
|
|
{
|
|
|
|
GListModel *model;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_DROP_DOWN (self));
|
2020-06-06 01:36:27 +00:00
|
|
|
g_return_if_fail (texts != NULL);
|
2019-12-09 01:22:06 +00:00
|
|
|
|
|
|
|
set_default_factory (self);
|
|
|
|
|
2020-06-27 04:31:18 +00:00
|
|
|
model = G_LIST_MODEL (gtk_string_list_new (texts));
|
2019-12-09 01:22:06 +00:00
|
|
|
gtk_drop_down_set_model (self, model);
|
|
|
|
g_object_unref (model);
|
|
|
|
}
|