mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-15 06:40:08 +00:00
Merge branch 'main' into ccook/doc-spelling-fixes
This commit is contained in:
commit
2a14753739
@ -24,10 +24,10 @@ you should be aware of what they refer to. These are often generic terms that
|
|||||||
have a specific meaning in this context.
|
have a specific meaning in this context.
|
||||||
|
|
||||||
**_Views_** or **_list widgets_** are the widgets that hold and manage the lists.
|
**_Views_** or **_list widgets_** are the widgets that hold and manage the lists.
|
||||||
Examples of these widgets would be `GtkListView` or `GtkGridView`.
|
Examples of these widgets would be [`class@Gtk.ListView`] or [`class@Gtk.GridView`].
|
||||||
|
|
||||||
Views display data from a **_model_**. A model is a `GListModel` and models can
|
Views display data from a **_model_**. A model is a [`iface@Gio.ListModel`] and
|
||||||
be provided in 3 ways or combinations thereof:
|
models can be provided in 3 ways or combinations thereof:
|
||||||
|
|
||||||
* Many list models implementations already exist. There are models that provide
|
* Many list models implementations already exist. There are models that provide
|
||||||
specific data, like `GtkDirectoryList`. And there are models like `GListStore`
|
specific data, like `GtkDirectoryList`. And there are models like `GListStore`
|
||||||
@ -42,7 +42,8 @@ be provided in 3 ways or combinations thereof:
|
|||||||
The same model can be used in multiple different views and wrapped with
|
The same model can be used in multiple different views and wrapped with
|
||||||
multiple different models at once.
|
multiple different models at once.
|
||||||
|
|
||||||
The elements in a model are called **_items_**. All items are `GObjects`.
|
The elements in a model are called **_items_**. All items are
|
||||||
|
[`class@GObject.Object`] instances.
|
||||||
|
|
||||||
Every item in a model has a **_position_** which is the unsigned integer that
|
Every item in a model has a **_position_** which is the unsigned integer that
|
||||||
describes where in the model the item is located. The first item in a model is
|
describes where in the model the item is located. The first item in a model is
|
||||||
@ -56,13 +57,13 @@ with models. Oftentimes some things are really hard to do one way but very easy
|
|||||||
the other way.
|
the other way.
|
||||||
|
|
||||||
The other important part of a view is a **_factory_**. Each factory is
|
The other important part of a view is a **_factory_**. Each factory is
|
||||||
a `GtkListItemFactory` implementation that takes care of mapping the items
|
a [`class@Gtk.ListItemFactory`] implementation that takes care of mapping the
|
||||||
of the model to widgets that can be shown in the view.
|
items of the model to widgets that can be shown in the view.
|
||||||
|
|
||||||
The way factories do this is by creating a **_listitem_** for each item that
|
The way factories do this is by creating a **_listitem_** for each item that
|
||||||
is currently in use. Listitems are always `GtkListItem` objects. They are only
|
is currently in use. List items are always [`class@Gtk.ListItem`] instances.
|
||||||
ever created by GTK and provide information about what item they are meant
|
They are only ever created by GTK and provide information about what item they
|
||||||
to display.
|
are meant to display.
|
||||||
|
|
||||||
Different factory implementations use various different methods to allow
|
Different factory implementations use various different methods to allow
|
||||||
developers to add the right widgets to listitems and to link those widgets
|
developers to add the right widgets to listitems and to link those widgets
|
||||||
@ -71,29 +72,30 @@ for the data displayed, the programming language and development environment
|
|||||||
is an important task that can simplify setting up the view tremendously.
|
is an important task that can simplify setting up the view tremendously.
|
||||||
|
|
||||||
Views support selections via a **_selection model_**. A selection model is an
|
Views support selections via a **_selection model_**. A selection model is an
|
||||||
implementation of the `GtkSelectionModel` interface on top of the `GListModel`
|
implementation of the [`iface@Gtk.SelectionModel`] interface on top of the
|
||||||
interface that allows marking each item in a model as either selected or not
|
[`iface@Gio.ListModel`] interface that allows marking each item in a model as either
|
||||||
selected. Just like regular models, this can be implemented either by
|
selected or not selected. Just like regular models, this can be implemented
|
||||||
implementing `GtkSelectionModel` directly or by wrapping a model with one of
|
either by implementing `GtkSelectionModel` directly or by wrapping a model with
|
||||||
the GTK models provided for this purposes, such as `GtkNoSelection`
|
one of the GTK models provided for this purposes, such as [`class@Gtk.NoSelection`]
|
||||||
or `GtkSingleSelection`.
|
or [`class@Gtk.SingleSelection`].
|
||||||
|
|
||||||
The behavior of selection models - ie which items they allow selecting and
|
The behavior of selection models - ie which items they allow selecting and
|
||||||
what effect this has on other items - is completely up to the selection model.
|
what effect this has on other items - is completely up to the selection model.
|
||||||
As such, single-selections, multi-selections or sharing selection state between
|
As such, single-selections, multi-selections or sharing selection state between
|
||||||
different selection models and/or views is possible. The selection state of an
|
different selection models and/or views is possible. The selection state of an
|
||||||
item is exposed in the listitem via the `GtkListItem:selected` property.
|
item is exposed in the listitem via the [`property@Gtk.ListItem:selected`] property.
|
||||||
|
|
||||||
Views and listitems also support activation. Activation means that double
|
Views and listitems also support activation. Activation means that double
|
||||||
clicking or pressing enter while inside a focused row will cause the view
|
clicking or pressing enter while inside a focused row will cause the view
|
||||||
to emit and activation signal such as `GtkListView::activate`. This provides
|
to emit and activation signal such as [`signal@Gtk.ListView::activate`]. This
|
||||||
an easy way to set up lists, but can also be turned off on listitems if undesired.
|
provides an easy way to set up lists, but can also be turned off on listitems
|
||||||
|
if undesired.
|
||||||
|
|
||||||
Both selections and activation are supported among other things via widget
|
Both selections and activation are supported among other things via widget
|
||||||
[actions](#actions-overview). This allows developers to add widgets to their
|
[actions](#actions-overview). This allows developers to add widgets to their
|
||||||
lists that cause selections to change or to trigger activation via
|
lists that cause selections to change or to trigger activation via
|
||||||
the `GtkActionable` interface. For a list of all supported actions see the
|
the [`iface@Gtk.Actionable`] interface. For a list of all supported actions see
|
||||||
relevant documentation.
|
the relevant documentation.
|
||||||
|
|
||||||
## Behind the scenes
|
## Behind the scenes
|
||||||
|
|
||||||
@ -101,12 +103,14 @@ While for short lists it is not a problem to instantiate widgets for every
|
|||||||
item in the model, once lists grow to thousands or millions of elements, this
|
item in the model, once lists grow to thousands or millions of elements, this
|
||||||
gets less feasible. Because of this, the views only create a limited amount of
|
gets less feasible. Because of this, the views only create a limited amount of
|
||||||
listitems and recycle them by binding them to new items. In general, views try
|
listitems and recycle them by binding them to new items. In general, views try
|
||||||
to keep listitems available only for the items that can actually be seen on screen.
|
to keep listitems available only for the items that can actually be seen on
|
||||||
|
screen.
|
||||||
|
|
||||||
While this behavior allows views to scale effortlessly to huge lists, it has a
|
While this behavior allows views to scale effortlessly to huge lists, it has a
|
||||||
few implication on what can be done with views. For example, it is not possible
|
few implication on what can be done with views. For example, it is not possible
|
||||||
to query a view for a listitem used for a certain position - there might not be
|
to query a view for a listitem used for a certain position - there might not be
|
||||||
one and even if there is, that listitem might soon be recycled for a new position.
|
one and even if there is, that listitem might soon be recycled for a new
|
||||||
|
position.
|
||||||
|
|
||||||
It is also important that developers save state they care about in the item and
|
It is also important that developers save state they care about in the item and
|
||||||
do not rely on the widgets they created as those widgets can be recycled for a
|
do not rely on the widgets they created as those widgets can be recycled for a
|
||||||
@ -114,13 +118,13 @@ new position at any time causing any state to be lost.
|
|||||||
|
|
||||||
Another important requirement for views is that they need to know which items
|
Another important requirement for views is that they need to know which items
|
||||||
are not visible so they can be recycled. Views achieve that by implementing
|
are not visible so they can be recycled. Views achieve that by implementing
|
||||||
the `GtkScrollable` interface and expecting to be placed directly into
|
the [`iface@Gtk.Scrollable`] interface and expecting to be placed directly into
|
||||||
a `GtkScrolledWindow`.
|
a [`class@Gtk.ScrolledWindow`].
|
||||||
|
|
||||||
Of course, if you are only using models with few items, this is not important
|
Of course, if you are only using models with few items, this is not important
|
||||||
and you can treat views like any other widget. But if you use large lists and
|
and you can treat views like any other widget. But if you use large lists and
|
||||||
your performance suffers, you should be aware of this. Views also allow tuning
|
your performance suffers, you should be aware of this. Views also allow tuning
|
||||||
the number of listitems they create such as with gtk_grid_view_set_max_columns(),
|
the number of listitems they create such as with [`method@Gtk.GridView.set_max_columns`],
|
||||||
and developers running into performance problems should definitely study the
|
and developers running into performance problems should definitely study the
|
||||||
tradeoffs of those and experiment with them.
|
tradeoffs of those and experiment with them.
|
||||||
|
|
||||||
@ -128,23 +132,23 @@ tradeoffs of those and experiment with them.
|
|||||||
|
|
||||||
GTK offers a wide variety of wrapping models which change or supplement an
|
GTK offers a wide variety of wrapping models which change or supplement an
|
||||||
existing model (or models) in some way. But when it comes to storing your
|
existing model (or models) in some way. But when it comes to storing your
|
||||||
actual data, there are only a few ready-made choices available: `GListStore`
|
actual data, there are only a few ready-made choices available: [`class@Gio.ListStore`]
|
||||||
and `GtkStringList`.
|
and [`class@Gtk.StringList`].
|
||||||
|
|
||||||
GListStore is backed by a balanced tree and has performance characteristics
|
`GListStore` is backed by a balanced tree and has performance characteristics
|
||||||
that are expected for that data structure. It works reasonably well for dataset
|
that are expected for that data structure. It works reasonably well for dataset
|
||||||
sizes in the 1,000,000 range, and can handle insertions and deletions. It uses
|
sizes in the 1,000,000 range, and can handle insertions and deletions. It uses
|
||||||
a cached iter to make linear access to the items fast.
|
a cached iter to make linear access to the items fast.
|
||||||
|
|
||||||
GtkStringList is not a general store - it can only handle strings. It is
|
`GtkStringList` is not a general store - it can only handle strings. It is
|
||||||
backed by an dynamically allocated array and has performance characteristics
|
backed by an dynamically allocated array and has performance characteristics
|
||||||
that are expected for that data structure. GtkStringList is a good fit for any
|
that are expected for that data structure. `GtkStringList` is a good fit for any
|
||||||
place where you would otherwise use `char*[]` and works best if the dataset
|
place where you would otherwise use `char*[]` and works best if the dataset
|
||||||
is not very dynamic.
|
is not very dynamic.
|
||||||
|
|
||||||
If these models don't fit your use case or scalability requirements, you
|
If these models don't fit your use case or scalability requirements, you
|
||||||
should make a custom `GListModel`. It is a small interface and not very hard
|
should make a custom `GListModel` implementation. It is a small interface and
|
||||||
to implement.
|
not very hard to implement.
|
||||||
|
|
||||||
For asymptotic performance comparisons between tree- and array-based
|
For asymptotic performance comparisons between tree- and array-based
|
||||||
implementations, see this
|
implementations, see this
|
||||||
@ -159,16 +163,17 @@ interface provided.
|
|||||||
|
|
||||||
However, GTK provides functionality to make trees look and behave like lists
|
However, GTK provides functionality to make trees look and behave like lists
|
||||||
for the people who still want to display lists. This is achieved by using
|
for the people who still want to display lists. This is achieved by using
|
||||||
the `GtkTreeListModel` model to flatten a tree into a list. The `GtkTreeExpander`
|
the [`class@Gtk.TreeListModel`] model to flatten a tree into a list. The
|
||||||
widget can then be used inside a listitem to allow users to expand and collapse
|
[`class@Gtk.TreeExpander`] widget can then be used inside a listitem to allow
|
||||||
rows and provide a similar experience to `GtkTreeView`.
|
users to expand and collapse rows and provide a similar experience to
|
||||||
|
`GtkTreeView`.
|
||||||
|
|
||||||
Developers should refer to those objects' API reference for more discussion
|
Developers should refer to those objects' API reference for more discussion
|
||||||
on the topic.
|
on the topic.
|
||||||
|
|
||||||
## List styles
|
## List styles
|
||||||
|
|
||||||
One of the advantages of the new list widgets over `GtkTreeViews` and cell
|
One of the advantages of the new list widgets over `GtkTreeView` and cell
|
||||||
renderers is that they are fully themable using GTK CSS. This provides a
|
renderers is that they are fully themable using GTK CSS. This provides a
|
||||||
lot of flexibility. The themes that ship with GTK provide a few predefined
|
lot of flexibility. The themes that ship with GTK provide a few predefined
|
||||||
list styles that can be used in many situations:
|
list styles that can be used in many situations:
|
||||||
@ -177,17 +182,17 @@ list styles that can be used in many situations:
|
|||||||
|
|
||||||
This style of list is low density, spacious and uses an outline focus ring.
|
This style of list is low density, spacious and uses an outline focus ring.
|
||||||
It is suitable for lists of controls, e.g. in preference dialogs or
|
It is suitable for lists of controls, e.g. in preference dialogs or
|
||||||
settings panels. Use the .rich-list style class.
|
settings panels. Use the `.rich-list` style class.
|
||||||
|
|
||||||
![Navigation sidebar](navigation-sidebar.png)
|
![Navigation sidebar](navigation-sidebar.png)
|
||||||
|
|
||||||
This style of list is medium density, using a full background to indicate
|
This style of list is medium density, using a full background to indicate
|
||||||
focus and selection. Use the .navigation-sidebar style class.
|
focus and selection. Use the `.navigation-sidebar` style class.
|
||||||
|
|
||||||
![Data table](data-table.png)
|
![Data table](data-table.png)
|
||||||
|
|
||||||
This style of list is a high density table, similar in style to a traditional
|
This style of list is a high density table, similar in style to a traditional
|
||||||
treeview. Individual cells can be selectable and editable. Use the .data-table
|
treeview. Individual cells can be selectable and editable. Use the `.data-table`
|
||||||
style class.
|
style class.
|
||||||
|
|
||||||
## Comparison to GtkTreeView
|
## Comparison to GtkTreeView
|
||||||
@ -210,7 +215,7 @@ was a rather complex interface for a tree data structure and `GListModel` was
|
|||||||
deliberately designed to be a simple data structure for lists only. (See
|
deliberately designed to be a simple data structure for lists only. (See
|
||||||
[above](#displaying-trees)) for how to still do trees with this new model.)
|
[above](#displaying-trees)) for how to still do trees with this new model.)
|
||||||
Another big change is that the new model allows for bulk changes via
|
Another big change is that the new model allows for bulk changes via
|
||||||
the `GListModel:items-changed` signal while `GtkTreeModel` only allows a single
|
the `GListModel::items-changed` signal while `GtkTreeModel` only allows a single
|
||||||
item to change at once. The goal here is of course to encourage implementation
|
item to change at once. The goal here is of course to encourage implementation
|
||||||
of custom list models.
|
of custom list models.
|
||||||
|
|
||||||
@ -229,24 +234,24 @@ specialized requirements.
|
|||||||
Finally here's a quick list of equivalent functionality to look for when
|
Finally here's a quick list of equivalent functionality to look for when
|
||||||
transitioning code for easy lookup:
|
transitioning code for easy lookup:
|
||||||
|
|
||||||
| Old | New |
|
| Old | New |
|
||||||
| -------------------- | ------------------------------------ |
|
| -------------------- | ------------------------------------------------------- |
|
||||||
| `GtkTreeModel` | `GListModel` |
|
| `GtkTreeModel` | [`iface@Gio.ListModel`] |
|
||||||
| `GtkTreePath` | `guint` position, `GtkTreeListRow` |
|
| `GtkTreePath` | `guint` position, [`class@Gtk.TreeListRow`] |
|
||||||
| `GtkTreeIter` | `guint` position |
|
| `GtkTreeIter` | `guint` position |
|
||||||
| `GtkTreeRowReference`| `GObject` item |
|
| `GtkTreeRowReference`| [`class@GObject.Object`] item |
|
||||||
| `GtkListStore` | `GListStore` |
|
| `GtkListStore` | [`class@Gio.ListStore`] |
|
||||||
| `GtkTreeStore` | `GtkTreeListModel`, `GtkTreeExpander`|
|
| `GtkTreeStore` | [`class@Gtk.TreeListModel`], [`class@Gtk.TreeExpander`] |
|
||||||
| `GtkTreeSelection` | `GtkSelectionModel` |
|
| `GtkTreeSelection` | [`iface@Gtk.SelectionModel`] |
|
||||||
| `GtkTreeViewColumn` | `GtkColumnView` |
|
| `GtkTreeViewColumn` | [`class@Gtk.ColumnView`] |
|
||||||
| `GtkTreeView` | `GtkListView`, `GtkColumnView` |
|
| `GtkTreeView` | [`class@Gtk.ListView`], [`class@Gtk.ColumnView`] |
|
||||||
| `GtkCellView` | `GtkListItemWidget` |
|
| `GtkCellView` | [`class@Gtk.ListItem`] |
|
||||||
| `GtkComboBox` | `GtkDropDown` |
|
| `GtkComboBox` | [`class@Gtk.DropDown`] |
|
||||||
| `GtkIconView` | `GtkGridView` |
|
| `GtkIconView` | [`class@Gtk.GridView`] |
|
||||||
| `GtkTreeSortable` | `GtkColumnView` |
|
| `GtkTreeSortable` | [`class@Gtk.ColumnView`] |
|
||||||
| `GtkTreeModelSort` | `GtkSortListModel` |
|
| `GtkTreeModelSort` | [`class@Gtk.SortListModel`] |
|
||||||
| `GtkTreeModelFilter` | `GtkFilterListModel` |
|
| `GtkTreeModelFilter` | [`class@Gtk.FilterListModel`] |
|
||||||
| `GtkCellLayout` | `GtkListItemFactory` |
|
| `GtkCellLayout` | [`class@Gtk.ListItemFactory`] |
|
||||||
| `GtkCellArea` | `GtkWidget` |
|
| `GtkCellArea` | [`class@Gtk.Widget`] |
|
||||||
| `GtkCellRenderer` | `GtkWidget` |
|
| `GtkCellRenderer` | [`class@Gtk.Widget`] |
|
||||||
|
|
||||||
|
@ -612,7 +612,6 @@ gdk_event_queue_handle_scroll_compression (GdkDisplay *display)
|
|||||||
GList *l;
|
GList *l;
|
||||||
GdkSurface *surface = NULL;
|
GdkSurface *surface = NULL;
|
||||||
GdkDevice *device = NULL;
|
GdkDevice *device = NULL;
|
||||||
GdkEvent *last_event = NULL;
|
|
||||||
GList *scrolls = NULL;
|
GList *scrolls = NULL;
|
||||||
GArray *history = NULL;
|
GArray *history = NULL;
|
||||||
GdkScrollUnit scroll_unit = GDK_SCROLL_UNIT_WHEEL;
|
GdkScrollUnit scroll_unit = GDK_SCROLL_UNIT_WHEEL;
|
||||||
@ -645,9 +644,6 @@ gdk_event_queue_handle_scroll_compression (GdkDisplay *display)
|
|||||||
scroll_unit != scroll_event->unit)
|
scroll_unit != scroll_event->unit)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
if (!last_event)
|
|
||||||
last_event = event;
|
|
||||||
|
|
||||||
surface = event->surface;
|
surface = event->surface;
|
||||||
device = event->device;
|
device = event->device;
|
||||||
scroll_unit = scroll_event->unit;
|
scroll_unit = scroll_event->unit;
|
||||||
|
@ -33,8 +33,8 @@ GDK_AVAILABLE_IN_4_10
|
|||||||
G_DECLARE_FINAL_TYPE (GtkAlertDialog, gtk_alert_dialog, GTK, ALERT_DIALOG, GObject)
|
G_DECLARE_FINAL_TYPE (GtkAlertDialog, gtk_alert_dialog, GTK, ALERT_DIALOG, GObject)
|
||||||
|
|
||||||
GDK_AVAILABLE_IN_4_10
|
GDK_AVAILABLE_IN_4_10
|
||||||
GtkAlertDialog *gtk_alert_dialog_new (const char *format,
|
GtkAlertDialog *gtk_alert_dialog_new (const char *format,
|
||||||
...) G_GNUC_PRINTF (1, 2);
|
...) G_GNUC_PRINTF (1, 2);
|
||||||
|
|
||||||
GDK_AVAILABLE_IN_4_10
|
GDK_AVAILABLE_IN_4_10
|
||||||
gboolean gtk_alert_dialog_get_modal (GtkAlertDialog *self);
|
gboolean gtk_alert_dialog_get_modal (GtkAlertDialog *self);
|
||||||
|
@ -7360,20 +7360,17 @@ gtk_file_chooser_widget_set_choice (GtkFileChooser *chooser,
|
|||||||
|
|
||||||
widget = (GtkWidget *)g_hash_table_lookup (impl->choices, id);
|
widget = (GtkWidget *)g_hash_table_lookup (impl->choices, id);
|
||||||
|
|
||||||
if (GTK_IS_BOX (widget))
|
if (GTK_IS_DROP_DOWN (widget))
|
||||||
{
|
{
|
||||||
guint i;
|
guint i;
|
||||||
const char **options;
|
const char **options;
|
||||||
GtkWidget *dropdown;
|
|
||||||
|
|
||||||
dropdown = gtk_widget_get_last_child (widget);
|
options = (const char **) g_object_get_data (G_OBJECT (widget), "options");
|
||||||
|
|
||||||
options = (const char **) g_object_get_data (G_OBJECT (dropdown), "options");
|
|
||||||
for (i = 0; options[i]; i++)
|
for (i = 0; options[i]; i++)
|
||||||
{
|
{
|
||||||
if (strcmp (option, options[i]) == 0)
|
if (strcmp (option, options[i]) == 0)
|
||||||
{
|
{
|
||||||
gtk_drop_down_set_selected (GTK_DROP_DOWN (dropdown), i);
|
gtk_drop_down_set_selected (GTK_DROP_DOWN (widget), i);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -232,8 +232,8 @@
|
|||||||
*
|
*
|
||||||
* ```c
|
* ```c
|
||||||
* const char *text =
|
* const char *text =
|
||||||
* "Go to the"
|
* "Go to the "
|
||||||
* "<a href=\"http://www.gtk.org title=\"<i>Our</i> website\">"
|
* "<a href=\"https://www.gtk.org\" title=\"<i>Our</i> website\">"
|
||||||
* "GTK website</a> for more...";
|
* "GTK website</a> for more...";
|
||||||
* GtkWidget *label = gtk_label_new (NULL);
|
* GtkWidget *label = gtk_label_new (NULL);
|
||||||
* gtk_label_set_markup (GTK_LABEL (label), text);
|
* gtk_label_set_markup (GTK_LABEL (label), text);
|
||||||
|
@ -1443,9 +1443,12 @@ event_type_name (GdkEventType type)
|
|||||||
"Pad Button Release",
|
"Pad Button Release",
|
||||||
"Pad Rind",
|
"Pad Rind",
|
||||||
"Pad Strip",
|
"Pad Strip",
|
||||||
"Pad Group Mode"
|
"Pad Group Mode",
|
||||||
|
"Touchpad Hold",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
G_STATIC_ASSERT (G_N_ELEMENTS (event_name) == GDK_EVENT_LAST);
|
||||||
|
|
||||||
return event_name[type];
|
return event_name[type];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user