diff options
author | Benjamin Otte <otte@redhat.com> | 2020-07-22 01:43:59 +0200 |
---|---|---|
committer | Benjamin Otte <otte@redhat.com> | 2020-07-22 14:30:49 +0200 |
commit | 5b18968867293dcb3af05f9b9a3c4109103cb664 (patch) | |
tree | d4ada0939e37a19401f124b8004aaa0c2f592632 | |
parent | e8c4e1205a51bd705c1daa1f8c069d3fc9d0144e (diff) | |
download | gtk+-5b18968867293dcb3af05f9b9a3c4109103cb664.tar.gz |
sortlistmodel: Make key generation part of the step function
SSave the missing keys as a bitset and iterate over that bitset in the
step function.
Solves the problem with a large UI block at the beginning of a sort
operation when all the keys were generated, in particular when key
generation was slow.
Benchmarks for maximum time taken by a single main loop callback:
initial sort with complex GFileInfo keys
old new
32,000 items 137ms 3ms
128,000 items 520ms 31ms
initial sort with string keys
old new
32,000 items 187ms 1ms
128,000 items 804ms 3ms
-rw-r--r-- | gtk/gtksortlistmodel.c | 83 |
1 files changed, 62 insertions, 21 deletions
diff --git a/gtk/gtksortlistmodel.c b/gtk/gtksortlistmodel.c index 49fa234e14..d3f04a5c64 100644 --- a/gtk/gtksortlistmodel.c +++ b/gtk/gtksortlistmodel.c @@ -21,6 +21,7 @@ #include "gtksortlistmodel.h" +#include "gtkbitset.h" #include "gtkintl.h" #include "gtkprivate.h" #include "gtksorterprivate.h" @@ -90,6 +91,8 @@ struct _GtkSortListModel GtkSortKeys *sort_keys; gsize key_size; gpointer keys; + GtkBitset *missing_keys; + gpointer *positions; }; @@ -199,6 +202,32 @@ gtk_sort_list_model_sort_step (GtkSortListModel *self, gpointer *start_change, *end_change; end_time += GTK_SORT_STEP_TIME_US; + + if (!gtk_bitset_is_empty (self->missing_keys)) + { + GtkBitsetIter iter; + guint pos; + + for (gtk_bitset_iter_init_first (&iter, self->missing_keys, &pos); + gtk_bitset_iter_is_valid (&iter); + gtk_bitset_iter_next (&iter, &pos)) + { + gpointer item = g_list_model_get_item (self->model, pos); + gtk_sort_keys_init_key (self->sort_keys, item, key_from_pos (self, pos)); + g_object_unref (item); + + if (g_get_monotonic_time () >= end_time && !finish) + { + gtk_bitset_remove_range_closed (self->missing_keys, 0, pos); + *out_position = 0; + *out_n_items = 0; + return TRUE; + } + } + result = TRUE; + gtk_bitset_remove_all (self->missing_keys); + } + end_change = self->positions; start_change = self->positions + self->n_items; @@ -300,17 +329,36 @@ gtk_sort_list_model_finish_sorting (GtkSortListModel *self, } static void -gtk_sort_list_model_clear_keys (GtkSortListModel *self) +gtk_sort_list_model_clear_sort_keys (GtkSortListModel *self, + guint position, + guint n_items) { + GtkBitsetIter iter; + GtkBitset *clear; + guint pos; - if (gtk_sort_keys_needs_clear_key (self->sort_keys)) - { - guint i; + if (!gtk_sort_keys_needs_clear_key (self->sort_keys)) + return; - for (i = 0; i < self->n_items; i++) - gtk_sort_keys_clear_key (self->sort_keys, key_from_pos (self, i)); + clear = gtk_bitset_new_range (position, n_items); + gtk_bitset_subtract (clear, self->missing_keys); + + for (gtk_bitset_iter_init_first (&iter, clear, &pos); + gtk_bitset_iter_is_valid (&iter); + gtk_bitset_iter_next (&iter, &pos)) + { + gtk_sort_keys_clear_key (self->sort_keys, key_from_pos (self, pos)); } + gtk_bitset_unref (clear); +} + +static void +gtk_sort_list_model_clear_keys (GtkSortListModel *self) +{ + gtk_sort_list_model_clear_sort_keys (self, 0, self->n_items); + + g_clear_pointer (&self->missing_keys, gtk_bitset_unref); g_clear_pointer (&self->keys, g_free); g_clear_pointer (&self->sort_keys, gtk_sort_keys_unref); self->key_size = 0; @@ -368,8 +416,6 @@ gtk_sort_list_model_should_sort (GtkSortListModel *self) static void gtk_sort_list_model_create_keys (GtkSortListModel *self) { - guint i; - g_assert (self->keys == NULL); g_assert (self->sort_keys == NULL); g_assert (self->key_size == 0); @@ -377,12 +423,7 @@ gtk_sort_list_model_create_keys (GtkSortListModel *self) self->sort_keys = gtk_sorter_get_keys (self->sorter); self->key_size = gtk_sort_keys_get_key_size (self->sort_keys); self->keys = g_malloc_n (self->n_items, self->key_size); - for (i = 0; i < self->n_items; i++) - { - gpointer item = g_list_model_get_item (self->model, i); - gtk_sort_keys_init_key (self->sort_keys, item, key_from_pos (self, i)); - g_object_unref (item); - } + self->missing_keys = gtk_bitset_new_range (0, self->n_items); } static void @@ -424,10 +465,7 @@ gtk_sort_list_model_update_items (GtkSortListModel *self, /* first, move the keys over */ old_keys = self->keys; - for (i = position; i < position + removed; i++) - { - gtk_sort_keys_clear_key (self->sort_keys, key_from_pos (self, i)); - } + gtk_sort_list_model_clear_sort_keys (self, position, removed); if (removed > added) { @@ -491,13 +529,16 @@ gtk_sort_list_model_update_items (GtkSortListModel *self, self->positions = g_renew (gpointer, self->positions, n_items - removed + added); + if (self->missing_keys) + { + gtk_bitset_splice (self->missing_keys, position, removed, added); + gtk_bitset_add_range (self->missing_keys, position, added); + } + self->n_items = n_items - removed + added; for (i = 0; i < added; i++) { - gpointer item = g_list_model_get_item (self->model, position + i); - gtk_sort_keys_init_key (self->sort_keys, item, key_from_pos (self, position + i)); - g_object_unref (item); self->positions[self->n_items - added + i] = key_from_pos (self, position + i); } |