From 51938467875cf6c8a86cb24c28f9aba341ebc42c Mon Sep 17 00:00:00 2001 From: Emmanuele Bassi Date: Fri, 11 Nov 2022 11:21:17 +0000 Subject: [PATCH 1/7] docs: Add links to the list widgets table Make it easier for readers to go to the appropriate replacement for the old GtkTreeView-and-friends API. --- docs/reference/gtk/section-list-widget.md | 40 +++++++++++------------ 1 file changed, 20 insertions(+), 20 deletions(-) diff --git a/docs/reference/gtk/section-list-widget.md b/docs/reference/gtk/section-list-widget.md index 3bd3b6fe73..0be72e7f18 100644 --- a/docs/reference/gtk/section-list-widget.md +++ b/docs/reference/gtk/section-list-widget.md @@ -229,24 +229,24 @@ specialized requirements. Finally here's a quick list of equivalent functionality to look for when transitioning code for easy lookup: -| Old | New | -| -------------------- | ------------------------------------ | -| `GtkTreeModel` | `GListModel` | -| `GtkTreePath` | `guint` position, `GtkTreeListRow` | -| `GtkTreeIter` | `guint` position | -| `GtkTreeRowReference`| `GObject` item | -| `GtkListStore` | `GListStore` | -| `GtkTreeStore` | `GtkTreeListModel`, `GtkTreeExpander`| -| `GtkTreeSelection` | `GtkSelectionModel` | -| `GtkTreeViewColumn` | `GtkColumnView` | -| `GtkTreeView` | `GtkListView`, `GtkColumnView` | -| `GtkCellView` | `GtkListItemWidget` | -| `GtkComboBox` | `GtkDropDown` | -| `GtkIconView` | `GtkGridView` | -| `GtkTreeSortable` | `GtkColumnView` | -| `GtkTreeModelSort` | `GtkSortListModel` | -| `GtkTreeModelFilter` | `GtkFilterListModel` | -| `GtkCellLayout` | `GtkListItemFactory` | -| `GtkCellArea` | `GtkWidget` | -| `GtkCellRenderer` | `GtkWidget` | +| Old | New | +| -------------------- | ------------------------------------------------------- | +| `GtkTreeModel` | [`iface@Gio.ListModel`] | +| `GtkTreePath` | `guint` position, [`class@Gtk.TreeListRow`] | +| `GtkTreeIter` | `guint` position | +| `GtkTreeRowReference`| [`class@GObject.Object`] item | +| `GtkListStore` | [`class@Gio.ListStore`] | +| `GtkTreeStore` | [`class@Gtk.TreeListModel`], [`class@Gtk.TreeExpander`] | +| `GtkTreeSelection` | [`iface@Gtk.SelectionModel`] | +| `GtkTreeViewColumn` | [`class@Gtk.ColumnView`] | +| `GtkTreeView` | [`class@Gtk.ListView`], [`class@Gtk.ColumnView`] | +| `GtkCellView` | [`class@Gtk.ListItem`] | +| `GtkComboBox` | [`class@Gtk.DropDown`] | +| `GtkIconView` | [`class@Gtk.GridView`] | +| `GtkTreeSortable` | [`class@Gtk.ColumnView`] | +| `GtkTreeModelSort` | [`class@Gtk.SortListModel`] | +| `GtkTreeModelFilter` | [`class@Gtk.FilterListModel`] | +| `GtkCellLayout` | [`class@Gtk.ListItemFactory`] | +| `GtkCellArea` | [`class@Gtk.Widget`] | +| `GtkCellRenderer` | [`class@Gtk.Widget`] | From bd66198b88ff49de2dcc843cb820c7f06ea963fa Mon Sep 17 00:00:00 2001 From: Emmanuele Bassi Date: Fri, 11 Nov 2022 11:29:08 +0000 Subject: [PATCH 2/7] docs: Add more links to the list widgets description --- docs/reference/gtk/section-list-widget.md | 85 ++++++++++++----------- 1 file changed, 45 insertions(+), 40 deletions(-) diff --git a/docs/reference/gtk/section-list-widget.md b/docs/reference/gtk/section-list-widget.md index 0be72e7f18..d1a5b597d5 100644 --- a/docs/reference/gtk/section-list-widget.md +++ b/docs/reference/gtk/section-list-widget.md @@ -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. **_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 -be provided in 3 ways or combinations thereof: +Views display data from a **_model_**. A model is a [`iface@Gio.ListModel`] and +models can be provided in 3 ways or combinations thereof: * Many list models implementations already exist. There are models that provide 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 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 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 important part of a view is a **_factory_**. Each factory is -a `GtkListItemFactory` implementation that takes care of mapping the items -of the model to widgets that can be shown in the view. +a [`class@Gtk.ListItemFactory`] implementation that takes care of mapping the +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 -is currently in use. Listitems are always `GtkListItem` objects. They are only -ever created by GTK and provide information about what item they are meant -to display. +is currently in use. List items are always [`class@Gtk.ListItem`] instances. +They are only ever created by GTK and provide information about what item they +are meant to display. Different factory implementations use various different methods to allow 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. Views support selections via a **_selection model_**. A selection model is an -implementation of the `GtkSelectionModel` interface on top of the `GListModel` -interface that allows marking each item in a model as either selected or not -selected. Just like regular models, this can be implemented either by -implementing `GtkSelectionModel` directly or by wrapping a model with one of -the GTK models provided for this purposes, such as `GtkNoSelection` -or `GtkSingleSelection`. +implementation of the [`iface@Gtk.SelectionModel`] interface on top of the +[`iface@Gio.ListModel`] interface that allows marking each item in a model as either +selected or not selected. Just like regular models, this can be implemented +either by implementing `GtkSelectionModel` directly or by wrapping a model with +one of the GTK models provided for this purposes, such as [`class@Gtk.NoSelection`] +or [`class@Gtk.SingleSelection`]. 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. As such, single-selections, multi-selections or sharing selection state between 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 clicking or pressing enter while inside a focused row will cause the view -to emit and activation signal such as `GtkListView::activate`. This provides -an easy way to set up lists, but can also be turned off on listitems if undesired. +to emit and activation signal such as [`signal@Gtk.ListView::activate`]. This +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 [actions](#actions-overview). This allows developers to add widgets to their lists that cause selections to change or to trigger activation via -the `GtkActionable` interface. For a list of all supported actions see the -relevant documentation. +the [`iface@Gtk.Actionable`] interface. For a list of all supported actions see +the relevant documentation. ## 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 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 -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 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 -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 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 are not visible so they can be recycled. Views achieve that by implementing -the `GtkScrollable` interface and expecting to be placed directly into -a `GtkScrolledWindow`. +the [`iface@Gtk.Scrollable`] interface and expecting to be placed directly into +a [`class@Gtk.ScrolledWindow`]. 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 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 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 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` -and `GtkStringList`. +actual data, there are only a few ready-made choices available: [`class@Gio.ListStore`] +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 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. -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 -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 is not very dynamic. 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 -to implement. +should make a custom `GListModel` implementation. It is a small interface and +not very hard to implement. For asymptotic performance comparisons between tree- and array-based implementations, see this @@ -159,16 +163,17 @@ interface provided. 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 -the `GtkTreeListModel` model to flatten a tree into a list. The `GtkTreeExpander` -widget can then be used inside a listitem to allow users to expand and collapse -rows and provide a similar experience to `GtkTreeView`. +the [`class@Gtk.TreeListModel`] model to flatten a tree into a list. The +[`class@Gtk.TreeExpander`] widget can then be used inside a listitem to allow +users to expand and collapse rows and provide a similar experience to +`GtkTreeView`. Developers should refer to those objects' API reference for more discussion on the topic. ## 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 lot of flexibility. The themes that ship with GTK provide a few predefined 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. 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) 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) 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. ## 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 [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 -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 of custom list models. From c9df9978d27c117d3a0aa83380039e576da21b64 Mon Sep 17 00:00:00 2001 From: Matthias Clasen Date: Tue, 15 Nov 2022 07:18:41 +0100 Subject: [PATCH 3/7] Inspector: Add a missing event type We did not have a name for the new touchpad hold events. To prevent this from happening again in the future, add a static assertion. --- gtk/inspector/recorder.c | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/gtk/inspector/recorder.c b/gtk/inspector/recorder.c index e0920365ca..8d960411d6 100644 --- a/gtk/inspector/recorder.c +++ b/gtk/inspector/recorder.c @@ -1443,9 +1443,12 @@ event_type_name (GdkEventType type) "Pad Button Release", "Pad Rind", "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]; } From 5aeb6c31c00ffda1d7ee7577efb89b3b2d737264 Mon Sep 17 00:00:00 2001 From: Matthias Clasen Date: Tue, 15 Nov 2022 10:26:25 +0000 Subject: [PATCH 4/7] events: Drop an unused variable --- gdk/gdkevents.c | 4 ---- 1 file changed, 4 deletions(-) diff --git a/gdk/gdkevents.c b/gdk/gdkevents.c index 890dbf01a1..37def3897e 100644 --- a/gdk/gdkevents.c +++ b/gdk/gdkevents.c @@ -612,7 +612,6 @@ gdk_event_queue_handle_scroll_compression (GdkDisplay *display) GList *l; GdkSurface *surface = NULL; GdkDevice *device = NULL; - GdkEvent *last_event = NULL; GList *scrolls = NULL; GArray *history = NULL; GdkScrollUnit scroll_unit = GDK_SCROLL_UNIT_WHEEL; @@ -645,9 +644,6 @@ gdk_event_queue_handle_scroll_compression (GdkDisplay *display) scroll_unit != scroll_event->unit) break; - if (!last_event) - last_event = event; - surface = event->surface; device = event->device; scroll_unit = scroll_event->unit; From 4eb017904bee2420a7992d65a15bdd0b29fe1900 Mon Sep 17 00:00:00 2001 From: Matthias Clasen Date: Tue, 15 Nov 2022 10:36:32 +0000 Subject: [PATCH 5/7] Alertdialog: Cosmetics --- gtk/gtkalertdialog.h | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/gtk/gtkalertdialog.h b/gtk/gtkalertdialog.h index 5d5d9f01b7..4263eb9c92 100644 --- a/gtk/gtkalertdialog.h +++ b/gtk/gtkalertdialog.h @@ -33,8 +33,8 @@ GDK_AVAILABLE_IN_4_10 G_DECLARE_FINAL_TYPE (GtkAlertDialog, gtk_alert_dialog, GTK, ALERT_DIALOG, GObject) GDK_AVAILABLE_IN_4_10 -GtkAlertDialog *gtk_alert_dialog_new (const char *format, - ...) G_GNUC_PRINTF (1, 2); +GtkAlertDialog *gtk_alert_dialog_new (const char *format, + ...) G_GNUC_PRINTF (1, 2); GDK_AVAILABLE_IN_4_10 gboolean gtk_alert_dialog_get_modal (GtkAlertDialog *self); From adc0264dac0a32451f4963b2011879f1421631e7 Mon Sep 17 00:00:00 2001 From: robxnano <91250-robxnano@users.noreply.gitlab.gnome.org> Date: Wed, 16 Nov 2022 18:07:20 +0000 Subject: [PATCH 6/7] filechooser: Fix gtk_file_chooser_set_choice again If the choice was in the form of a GtkDropDown, the code failed to find the widget and so the choice never got set. Fixes #5346. --- gtk/gtkfilechooserwidget.c | 9 +++------ 1 file changed, 3 insertions(+), 6 deletions(-) diff --git a/gtk/gtkfilechooserwidget.c b/gtk/gtkfilechooserwidget.c index 20a8f4abb2..c2b29a2034 100644 --- a/gtk/gtkfilechooserwidget.c +++ b/gtk/gtkfilechooserwidget.c @@ -7360,20 +7360,17 @@ gtk_file_chooser_widget_set_choice (GtkFileChooser *chooser, widget = (GtkWidget *)g_hash_table_lookup (impl->choices, id); - if (GTK_IS_BOX (widget)) + if (GTK_IS_DROP_DOWN (widget)) { guint i; const char **options; - GtkWidget *dropdown; - dropdown = gtk_widget_get_last_child (widget); - - options = (const char **) g_object_get_data (G_OBJECT (dropdown), "options"); + options = (const char **) g_object_get_data (G_OBJECT (widget), "options"); for (i = 0; options[i]; i++) { 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; } } From 7adf720f08f63fcf3ae417801c90d41cecb7f408 Mon Sep 17 00:00:00 2001 From: Thomas Lange Date: Thu, 17 Nov 2022 03:46:01 +0000 Subject: [PATCH 7/7] label: Fix markup doc example --- gtk/gtklabel.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/gtk/gtklabel.c b/gtk/gtklabel.c index 5f8dd50d2d..8074cb4b1f 100644 --- a/gtk/gtklabel.c +++ b/gtk/gtklabel.c @@ -232,8 +232,8 @@ * * ```c * const char *text = - * "Go to the" - * "" + * "Go to the " + * "" * "GTK website for more..."; * GtkWidget *label = gtk_label_new (NULL); * gtk_label_set_markup (GTK_LABEL (label), text);