GtkSignalListItemFactory

GtkSignalListItemFactory and GtkBulderListItemFactory

GtkBuilderlistItemFactory is convenient when GtkListView just shows the contents of a list. Its binding direction is always from an item of a list to a child of GtkListItem.

When it comes to dynamic connection, it’s not enough. For example, you want to edit the contents of a list. You set a child of GtkListItem to a GtkText instance so a user can edit a text with it. You need to bind an item in the list with the buffer of the GtkText. The direction is opposite from the one with GtkBuilderListItemFactory. It is from the GtkText instance to the item in the list. You can implement this with GtkSignalListItemFactory, which is more flexible than GtkBuilderListItemFactory.

Two things are shown in this section.

This section shows just a part of the source file listeditor.c. If you want to see the whole codes, see src/listeditor directory of the Gtk4 tutorial repository.

A list editor

The sample program is a list editor and data of the list are strings. It’s the same as a line editor. It reads a text file line by line. Each line is an item of the list. The list is displayed with GtkColumnView. There are two columns. The one is a button, which makes the line be a current line. If the line is the current line, the button is colored with red. The other is a string which is the contents of the corresponding item of the list.

List editor

The source files are located at src/listeditor directory. You can compile end execute it as follows.

$ meson _build
$ ninja -C _build
$ _build/listeditor

The current line number (zero-based) is shown at the left of the tool bar. The file name is shown at the right of the write button.

Connect a GtkText instance and an item in the list

The second column (GtkColumnViewColumn) sets its factory property to GtkSignalListItemFactory. It uses three signals setup, bind and unbind. The following is their sgnal handlers.

static void
setup2_cb (GtkListItemFactory *factory, GtkListItem *listitem) {
  GtkWidget *text = gtk_text_new ();
  gtk_list_item_set_child (listitem, GTK_WIDGET (text));
  gtk_editable_set_alignment (GTK_EDITABLE (text), 0.0);
}

static void
bind2_cb (GtkListItemFactory *factory, GtkListItem *listitem) {
  GtkWidget *text = gtk_list_item_get_child (listitem);
  GtkEntryBuffer *buffer = gtk_text_get_buffer (GTK_TEXT (text));
  LeData *data = LE_DATA (gtk_list_item_get_item (listitem));
  GBinding *bind;

  gtk_editable_set_text (GTK_EDITABLE (text), le_data_look_string (data));
  gtk_editable_set_position (GTK_EDITABLE (text), 0);

  bind = g_object_bind_property (buffer, "text", data, "string", G_BINDING_DEFAULT);
  g_object_set_data (G_OBJECT (listitem), "bind", bind);
}

static void
unbind2_cb (GtkListItemFactory *factory, GtkListItem *listitem) {
  GBinding *bind = G_BINDING (g_object_get_data (G_OBJECT (listitem), "bind"));

  g_binding_unbind(bind);
  g_object_set_data (G_OBJECT (listitem), "bind", NULL);
}

This technique is not so complicated. You can use it when you make a cell editable application.

Change the cell of GtkColumnView dynamically

Next topic is to change the GtkColumnView (or GtkListView) cells dynamically. The example changes the color of the buttons, which are children of GtkListItem instances, as the current line position moves.

The line editor has the current position of the list.

The button of the current line is colored with red and otherwise white.

The current line has no relationship to GtkSingleSelection object. GtkSingleSelection selects a line on the display. The current line doesn’t need to be on the display. It is possible to be on the line out of the Window (GtkScrolledWindow). Actually, the program doesn’t use GtkSingleSelection.

It is necessary to know the corresponding GtkListItem instance from the item in the list. It is the opposite direction from gtk_list_item_get_item function. To accomplish this, we set a listitem element of LeData to point the corresponding GtkListItem instance. Therefore, items (LeData) in the list always know the GtkListItem. If there’s no GtkListItem bound to the item, NULL is assigned.

void
select_cb (GtkButton *btn, GtkListItem *listitem) {
  LeWindow *win = LE_WINDOW (gtk_widget_get_ancestor (GTK_WIDGET (btn), LE_TYPE_WINDOW));

  update_current (win, gtk_list_item_get_position (listitem));
}

static void
setup1_cb (GtkListItemFactory *factory, GtkListItem *listitem) {
  GtkWidget *button = gtk_button_new ();
  gtk_list_item_set_child (listitem, button);
  gtk_widget_set_focusable (GTK_WIDGET (button), FALSE);
  g_signal_connect (button, "clicked", G_CALLBACK (select_cb), listitem);
}

static void
bind1_cb (GtkListItemFactory *factory, GtkListItem *listitem, gpointer user_data) {
  LeWindow *win = LE_WINDOW (user_data);
  GtkWidget *button = gtk_list_item_get_child (listitem);
  const char *non_current[1] = {NULL};
  const char *current[2] = {"current", NULL};
  LeData *data = LE_DATA (gtk_list_item_get_item (listitem));

  if (data) {
    le_data_set_listitem (data, listitem);
    if (win->position == gtk_list_item_get_position (listitem))
      gtk_widget_set_css_classes (GTK_WIDGET (button), current);
    else
      gtk_widget_set_css_classes (GTK_WIDGET (button), non_current);
  }
}

static void
unbind1_cb (GtkListItemFactory *factory, GtkListItem *listitem) {
  LeData *data = LE_DATA (gtk_list_item_get_item (listitem));
  if (data)
    le_data_set_listitem (data, NULL);
}
static void
update_current (LeWindow *win, int new) {
  char *s;
  LeData *data;
  GtkListItem *listitem;
  GtkButton *button;
  const char *non_current[1] = {NULL};
  const char *current[2] = {"current", NULL};

  if (new >= 0)
    s = g_strdup_printf ("%d", new);
  else
    s = "";
  gtk_label_set_text (GTK_LABEL (win->position_label), s);
  if (*s) // s isn't an empty string
    g_free (s);

  if (win->position >=0) {
    data = LE_DATA (g_list_model_get_item (G_LIST_MODEL (win->liststore), win->position));
    if ((listitem = le_data_get_listitem (data)) != NULL) {
      button = GTK_BUTTON (gtk_list_item_get_child (listitem));
      gtk_widget_set_css_classes (GTK_WIDGET (button), non_current);
      g_object_unref (listitem);
    }
    g_object_unref (data);
  }
  win->position = new;
  if (win->position >=0) {
    data = LE_DATA (g_list_model_get_item (G_LIST_MODEL (win->liststore), win->position));
    if ((listitem = le_data_get_listitem (data)) != NULL) {
      button = GTK_BUTTON (gtk_list_item_get_child (listitem));
      gtk_widget_set_css_classes (GTK_WIDGET (button), current);
      g_object_unref (listitem);
    }
    g_object_unref (data);
  }
}

The function update_current does several things.

The color of buttons are determined by the “background” CSS style. The following CSS is applied to the default GdkDisplay in advance (in the startup handler of the application).

columnview listview row button.current {background: red;}

The selectors “columnview listview row” is needed before “button” selector. Otherwise the buttons in the GtkColumnview won’t be found. The button selector has “current” class. So, the only “current” class button is colored with red. Other buttons are not colored, which means they are white.

Gtk_widget_dispose_template function

The function gtk_widget_dispose_template clears the template children for the given widget. This is the opposite of gtk_widget_init_template(). It is a new function of GTK 4.8 version. If your GTK version is lower than 4.8, you need to modify the program.

A waring from GtkText

If your program has the following two, a warning message can be issued.

GtkText - unexpected blinking selection. Removing

I don’t have an exact idea why this happens. But if GtkText “focusable” property is FALSE, the warning doesn’t happen. So it probably comes from focus and scroll.

You can avoid this by unsetting any focus widget under the main window. When scroll begins, the “value-changed” signal on the vertical adjustment of the scrolled window is emitted.

The following is extracted from the ui file and C source file.

... ... ...
<object class="GtkScrolledWindow">
  <property name="hexpand">TRUE</property>
  <property name="vexpand">TRUE</property>
  <property name="vadjustment">
    <object class="GtkAdjustment">
      <signal name="value-changed" handler="adjustment_value_changed_cb" swapped="no" object="LeWindow"/>
    </object>
  </property>
... ... ...  
static void
adjustment_value_changed_cb (GtkAdjustment *adjustment, gpointer user_data) {
  GtkWidget *win = GTK_WIDGET (user_data);

  gtk_window_set_focus (GTK_WINDOW (win), NULL);
}