mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2024-12-26 21:51:08 +00:00
listitemmanager: Add GtkListItemChange
... for tracking widgets during changes. This just pulls all the different disjointed parts into one struct with a sensible API.
This commit is contained in:
parent
38844fef4d
commit
76d601631d
@ -26,6 +26,8 @@
|
|||||||
#include "gtksectionmodel.h"
|
#include "gtksectionmodel.h"
|
||||||
#include "gtkwidgetprivate.h"
|
#include "gtkwidgetprivate.h"
|
||||||
|
|
||||||
|
typedef struct _GtkListItemChange GtkListItemChange;
|
||||||
|
|
||||||
struct _GtkListItemManager
|
struct _GtkListItemManager
|
||||||
{
|
{
|
||||||
GObject parent_instance;
|
GObject parent_instance;
|
||||||
@ -56,26 +58,82 @@ struct _GtkListItemTracker
|
|||||||
guint n_after;
|
guint n_after;
|
||||||
};
|
};
|
||||||
|
|
||||||
static GtkWidget * gtk_list_item_manager_acquire_list_item (GtkListItemManager *self,
|
struct _GtkListItemChange
|
||||||
guint position,
|
{
|
||||||
GtkWidget *prev_sibling);
|
GHashTable *deleted_items;
|
||||||
static GtkWidget * gtk_list_item_manager_try_reacquire_list_item
|
GQueue recycled_items;
|
||||||
(GtkListItemManager *self,
|
};
|
||||||
GHashTable *change,
|
|
||||||
guint position,
|
|
||||||
GtkWidget *prev_sibling);
|
|
||||||
static void gtk_list_item_manager_update_list_item (GtkListItemManager *self,
|
|
||||||
GtkWidget *item,
|
|
||||||
guint position);
|
|
||||||
static void gtk_list_item_manager_move_list_item (GtkListItemManager *self,
|
|
||||||
GtkWidget *list_item,
|
|
||||||
guint position,
|
|
||||||
GtkWidget *prev_sibling);
|
|
||||||
static void gtk_list_item_manager_release_list_item (GtkListItemManager *self,
|
|
||||||
GHashTable *change,
|
|
||||||
GtkWidget *widget);
|
|
||||||
G_DEFINE_TYPE (GtkListItemManager, gtk_list_item_manager, G_TYPE_OBJECT)
|
G_DEFINE_TYPE (GtkListItemManager, gtk_list_item_manager, G_TYPE_OBJECT)
|
||||||
|
|
||||||
|
static void
|
||||||
|
gtk_list_item_change_init (GtkListItemChange *change)
|
||||||
|
{
|
||||||
|
change->deleted_items = NULL;
|
||||||
|
g_queue_init (&change->recycled_items);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gtk_list_item_change_finish (GtkListItemChange *change)
|
||||||
|
{
|
||||||
|
GtkWidget *widget;
|
||||||
|
|
||||||
|
g_clear_pointer (&change->deleted_items, g_hash_table_destroy);
|
||||||
|
|
||||||
|
while ((widget = g_queue_pop_head (&change->recycled_items)))
|
||||||
|
gtk_widget_unparent (widget);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gtk_list_item_change_recycle (GtkListItemChange *change,
|
||||||
|
GtkListItemBase *widget)
|
||||||
|
{
|
||||||
|
g_queue_push_tail (&change->recycled_items, widget);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gtk_list_item_change_release (GtkListItemChange *change,
|
||||||
|
GtkListItemBase *widget)
|
||||||
|
{
|
||||||
|
if (change->deleted_items == NULL)
|
||||||
|
change->deleted_items = g_hash_table_new_full (g_direct_hash, g_direct_equal, NULL, (GDestroyNotify) gtk_widget_unparent);
|
||||||
|
|
||||||
|
if (!g_hash_table_replace (change->deleted_items, gtk_list_item_base_get_item (GTK_LIST_ITEM_BASE (widget)), widget))
|
||||||
|
{
|
||||||
|
g_warning ("Duplicate item detected in list. Picking one randomly.");
|
||||||
|
gtk_list_item_change_recycle (change, widget);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static GtkListItemBase *
|
||||||
|
gtk_list_item_change_find (GtkListItemChange *change,
|
||||||
|
gpointer item)
|
||||||
|
{
|
||||||
|
gpointer result;
|
||||||
|
|
||||||
|
if (change->deleted_items && g_hash_table_steal_extended (change->deleted_items, item, NULL, &result))
|
||||||
|
return result;
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static GtkListItemBase *
|
||||||
|
gtk_list_item_change_get (GtkListItemChange *change,
|
||||||
|
gpointer item)
|
||||||
|
{
|
||||||
|
GtkListItemBase *result;
|
||||||
|
|
||||||
|
result = gtk_list_item_change_find (change, item);
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
result = g_queue_pop_head (&change->recycled_items);
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
potentially_empty_rectangle_union (cairo_rectangle_int_t *self,
|
potentially_empty_rectangle_union (cairo_rectangle_int_t *self,
|
||||||
const cairo_rectangle_int_t *area)
|
const cairo_rectangle_int_t *area)
|
||||||
@ -750,7 +808,7 @@ gtk_list_item_manager_ensure_split (GtkListItemManager *self,
|
|||||||
|
|
||||||
static void
|
static void
|
||||||
gtk_list_item_manager_remove_items (GtkListItemManager *self,
|
gtk_list_item_manager_remove_items (GtkListItemManager *self,
|
||||||
GHashTable *change,
|
GtkListItemChange *change,
|
||||||
guint position,
|
guint position,
|
||||||
guint n_items)
|
guint n_items)
|
||||||
{
|
{
|
||||||
@ -794,7 +852,7 @@ gtk_list_item_manager_remove_items (GtkListItemManager *self,
|
|||||||
g_assert (tile->n_items <= n_items);
|
g_assert (tile->n_items <= n_items);
|
||||||
}
|
}
|
||||||
if (tile->widget)
|
if (tile->widget)
|
||||||
gtk_list_item_manager_release_list_item (self, change, tile->widget);
|
gtk_list_item_change_release (change, GTK_LIST_ITEM_BASE (tile->widget));
|
||||||
tile->widget = NULL;
|
tile->widget = NULL;
|
||||||
n_items -= tile->n_items;
|
n_items -= tile->n_items;
|
||||||
tile->n_items = 0;
|
tile->n_items = 0;
|
||||||
@ -1044,7 +1102,7 @@ gtk_list_tile_gc (GtkListItemManager *self,
|
|||||||
|
|
||||||
static void
|
static void
|
||||||
gtk_list_item_manager_release_items (GtkListItemManager *self,
|
gtk_list_item_manager_release_items (GtkListItemManager *self,
|
||||||
GQueue *released)
|
GtkListItemChange *change)
|
||||||
{
|
{
|
||||||
GtkListTile *tile;
|
GtkListTile *tile;
|
||||||
guint position, i, n_items, query_n_items;
|
guint position, i, n_items, query_n_items;
|
||||||
@ -1073,7 +1131,7 @@ gtk_list_item_manager_release_items (GtkListItemManager *self,
|
|||||||
case GTK_LIST_TILE_ITEM:
|
case GTK_LIST_TILE_ITEM:
|
||||||
if (tile->widget)
|
if (tile->widget)
|
||||||
{
|
{
|
||||||
g_queue_push_tail (released, tile->widget);
|
gtk_list_item_change_recycle (change, GTK_LIST_ITEM_BASE (tile->widget));
|
||||||
tile->widget = NULL;
|
tile->widget = NULL;
|
||||||
}
|
}
|
||||||
i += tile->n_items;
|
i += tile->n_items;
|
||||||
@ -1158,13 +1216,12 @@ gtk_list_item_manager_insert_section (GtkListItemManager *self,
|
|||||||
|
|
||||||
static void
|
static void
|
||||||
gtk_list_item_manager_ensure_items (GtkListItemManager *self,
|
gtk_list_item_manager_ensure_items (GtkListItemManager *self,
|
||||||
GHashTable *change,
|
GtkListItemChange *change,
|
||||||
guint update_start)
|
guint update_start)
|
||||||
{
|
{
|
||||||
GtkListTile *tile, *other_tile, *header;
|
GtkListTile *tile, *other_tile, *header;
|
||||||
GtkWidget *widget, *insert_after;
|
GtkWidget *insert_after;
|
||||||
guint position, i, n_items, query_n_items, offset;
|
guint position, i, n_items, query_n_items, offset;
|
||||||
GQueue released = G_QUEUE_INIT;
|
|
||||||
gboolean tracked, has_sections;
|
gboolean tracked, has_sections;
|
||||||
|
|
||||||
if (self->model == NULL)
|
if (self->model == NULL)
|
||||||
@ -1174,7 +1231,7 @@ gtk_list_item_manager_ensure_items (GtkListItemManager *self,
|
|||||||
position = 0;
|
position = 0;
|
||||||
has_sections = gtk_list_item_manager_has_sections (self);
|
has_sections = gtk_list_item_manager_has_sections (self);
|
||||||
|
|
||||||
gtk_list_item_manager_release_items (self, &released);
|
gtk_list_item_manager_release_items (self, change);
|
||||||
|
|
||||||
while (position < n_items)
|
while (position < n_items)
|
||||||
{
|
{
|
||||||
@ -1225,35 +1282,25 @@ gtk_list_item_manager_ensure_items (GtkListItemManager *self,
|
|||||||
|
|
||||||
if (tile->widget == NULL)
|
if (tile->widget == NULL)
|
||||||
{
|
{
|
||||||
if (change)
|
gpointer item = g_list_model_get_item (G_LIST_MODEL (self->model), position + i);
|
||||||
{
|
tile->widget = GTK_WIDGET (gtk_list_item_change_get (change, item));
|
||||||
tile->widget = gtk_list_item_manager_try_reacquire_list_item (self,
|
|
||||||
change,
|
|
||||||
position + i,
|
|
||||||
insert_after);
|
|
||||||
}
|
|
||||||
if (tile->widget == NULL)
|
if (tile->widget == NULL)
|
||||||
{
|
tile->widget = GTK_WIDGET (self->create_widget (self->widget));
|
||||||
tile->widget = g_queue_pop_head (&released);
|
gtk_list_item_base_update (GTK_LIST_ITEM_BASE (tile->widget),
|
||||||
if (tile->widget)
|
position + i,
|
||||||
{
|
item,
|
||||||
gtk_list_item_manager_move_list_item (self,
|
gtk_selection_model_is_selected (self->model, position + i));
|
||||||
tile->widget,
|
gtk_widget_insert_after (tile->widget, self->widget, insert_after);
|
||||||
position + i,
|
|
||||||
insert_after);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
tile->widget = gtk_list_item_manager_acquire_list_item (self,
|
|
||||||
position + i,
|
|
||||||
insert_after);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (update_start <= position + i)
|
if (update_start <= position + i)
|
||||||
gtk_list_item_manager_update_list_item (self, tile->widget, position + i);
|
{
|
||||||
|
gtk_list_item_base_update (GTK_LIST_ITEM_BASE (tile->widget),
|
||||||
|
position + i,
|
||||||
|
gtk_list_item_base_get_item (GTK_LIST_ITEM_BASE (tile->widget)),
|
||||||
|
gtk_selection_model_is_selected (self->model, position + i));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
insert_after = tile->widget;
|
insert_after = tile->widget;
|
||||||
i++;
|
i++;
|
||||||
@ -1288,9 +1335,6 @@ gtk_list_item_manager_ensure_items (GtkListItemManager *self,
|
|||||||
|
|
||||||
position += query_n_items;
|
position += query_n_items;
|
||||||
}
|
}
|
||||||
|
|
||||||
while ((widget = g_queue_pop_head (&released)))
|
|
||||||
gtk_list_item_manager_release_list_item (self, NULL, widget);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -1300,14 +1344,14 @@ gtk_list_item_manager_model_items_changed_cb (GListModel *model,
|
|||||||
guint added,
|
guint added,
|
||||||
GtkListItemManager *self)
|
GtkListItemManager *self)
|
||||||
{
|
{
|
||||||
GHashTable *change;
|
GtkListItemChange change;
|
||||||
GSList *l;
|
GSList *l;
|
||||||
guint n_items;
|
guint n_items;
|
||||||
|
|
||||||
|
gtk_list_item_change_init (&change);
|
||||||
n_items = g_list_model_get_n_items (G_LIST_MODEL (self->model));
|
n_items = g_list_model_get_n_items (G_LIST_MODEL (self->model));
|
||||||
change = g_hash_table_new_full (g_direct_hash, g_direct_equal, NULL, (GDestroyNotify )gtk_widget_unparent);
|
|
||||||
|
|
||||||
gtk_list_item_manager_remove_items (self, change, position, removed);
|
gtk_list_item_manager_remove_items (self, &change, position, removed);
|
||||||
gtk_list_item_manager_add_items (self, position, added);
|
gtk_list_item_manager_add_items (self, position, added);
|
||||||
|
|
||||||
/* Check if any tracked item was removed */
|
/* Check if any tracked item was removed */
|
||||||
@ -1318,7 +1362,7 @@ gtk_list_item_manager_model_items_changed_cb (GListModel *model,
|
|||||||
if (tracker->widget == NULL)
|
if (tracker->widget == NULL)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (g_hash_table_lookup (change, gtk_list_item_base_get_item (tracker->widget)))
|
if (tracker->position >= position && tracker->position < position + removed)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1342,12 +1386,14 @@ gtk_list_item_manager_model_items_changed_cb (GListModel *model,
|
|||||||
|
|
||||||
for (i = 0; i < added; i++)
|
for (i = 0; i < added; i++)
|
||||||
{
|
{
|
||||||
GtkWidget *widget;
|
GtkListItemBase *widget;
|
||||||
|
gpointer item;
|
||||||
|
|
||||||
|
/* XXX: can we avoid temporarily allocating items on failure? */
|
||||||
|
item = g_list_model_get_item (G_LIST_MODEL (self->model), position + i);
|
||||||
|
widget = gtk_list_item_change_find (&change, item);
|
||||||
|
g_object_unref (item);
|
||||||
|
|
||||||
widget = gtk_list_item_manager_try_reacquire_list_item (self,
|
|
||||||
change,
|
|
||||||
position + i,
|
|
||||||
insert_after);
|
|
||||||
if (widget == NULL)
|
if (widget == NULL)
|
||||||
{
|
{
|
||||||
offset++;
|
offset++;
|
||||||
@ -1371,8 +1417,13 @@ gtk_list_item_manager_model_items_changed_cb (GListModel *model,
|
|||||||
else
|
else
|
||||||
tile = gtk_list_item_manager_ensure_split (self, tile, 1);
|
tile = gtk_list_item_manager_ensure_split (self, tile, 1);
|
||||||
|
|
||||||
new_tile->widget = widget;
|
new_tile->widget = GTK_WIDGET (widget);
|
||||||
insert_after = widget;
|
gtk_list_item_base_update (widget,
|
||||||
|
position + i,
|
||||||
|
item,
|
||||||
|
gtk_selection_model_is_selected (self->model, position + i));
|
||||||
|
gtk_widget_insert_after (new_tile->widget, self->widget, insert_after);
|
||||||
|
insert_after = new_tile->widget;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1396,9 +1447,13 @@ gtk_list_item_manager_model_items_changed_cb (GListModel *model,
|
|||||||
}
|
}
|
||||||
else if (tracker->position >= position)
|
else if (tracker->position >= position)
|
||||||
{
|
{
|
||||||
if (g_hash_table_lookup (change, gtk_list_item_base_get_item (tracker->widget)))
|
GtkListItemBase *widget = gtk_list_item_change_find (&change, gtk_list_item_base_get_item (tracker->widget));
|
||||||
|
if (widget)
|
||||||
{
|
{
|
||||||
/* The item is gone. Guess a good new position */
|
/* The item is still in the recycling pool, which means it got deleted.
|
||||||
|
* Put the widget back and then guess a good new position */
|
||||||
|
gtk_list_item_change_release (&change, widget);
|
||||||
|
|
||||||
tracker->position = position + (tracker->position - position) * added / removed;
|
tracker->position = position + (tracker->position - position) * added / removed;
|
||||||
if (tracker->position >= n_items)
|
if (tracker->position >= n_items)
|
||||||
{
|
{
|
||||||
@ -1423,7 +1478,7 @@ gtk_list_item_manager_model_items_changed_cb (GListModel *model,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
gtk_list_item_manager_ensure_items (self, change, position + added);
|
gtk_list_item_manager_ensure_items (self, &change, position + added);
|
||||||
|
|
||||||
/* final loop through the trackers: Grab the missing widgets.
|
/* final loop through the trackers: Grab the missing widgets.
|
||||||
* For items that had been removed and a new position was set, grab
|
* For items that had been removed and a new position was set, grab
|
||||||
@ -1444,7 +1499,7 @@ gtk_list_item_manager_model_items_changed_cb (GListModel *model,
|
|||||||
tracker->widget = GTK_LIST_ITEM_BASE (tile->widget);
|
tracker->widget = GTK_LIST_ITEM_BASE (tile->widget);
|
||||||
}
|
}
|
||||||
|
|
||||||
g_hash_table_unref (change);
|
gtk_list_item_change_finish (&change);
|
||||||
|
|
||||||
gtk_widget_queue_resize (self->widget);
|
gtk_widget_queue_resize (self->widget);
|
||||||
}
|
}
|
||||||
@ -1472,24 +1527,32 @@ gtk_list_item_manager_model_selection_changed_cb (GListModel *model,
|
|||||||
|
|
||||||
while (n_items > 0)
|
while (n_items > 0)
|
||||||
{
|
{
|
||||||
if (tile->widget)
|
if (tile->widget && tile->type == GTK_LIST_TILE_ITEM)
|
||||||
gtk_list_item_manager_update_list_item (self, tile->widget, position);
|
{
|
||||||
|
gtk_list_item_base_update (GTK_LIST_ITEM_BASE (tile->widget),
|
||||||
|
position,
|
||||||
|
gtk_list_item_base_get_item (GTK_LIST_ITEM_BASE (tile->widget)),
|
||||||
|
gtk_selection_model_is_selected (self->model, position));
|
||||||
|
}
|
||||||
position += tile->n_items;
|
position += tile->n_items;
|
||||||
n_items -= MIN (n_items, tile->n_items);
|
n_items -= MIN (n_items, tile->n_items);
|
||||||
tile = gtk_rb_tree_node_get_next (tile);
|
tile = gtk_list_tile_get_next_skip (tile);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gtk_list_item_manager_clear_model (GtkListItemManager *self)
|
gtk_list_item_manager_clear_model (GtkListItemManager *self)
|
||||||
{
|
{
|
||||||
|
GtkListItemChange change;
|
||||||
GtkListTile *tile;
|
GtkListTile *tile;
|
||||||
GSList *l;
|
GSList *l;
|
||||||
|
|
||||||
if (self->model == NULL)
|
if (self->model == NULL)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
gtk_list_item_manager_remove_items (self, NULL, 0, g_list_model_get_n_items (G_LIST_MODEL (self->model)));
|
gtk_list_item_change_init (&change);
|
||||||
|
gtk_list_item_manager_remove_items (self, &change, 0, g_list_model_get_n_items (G_LIST_MODEL (self->model)));
|
||||||
|
gtk_list_item_change_finish (&change);
|
||||||
for (l = self->trackers; l; l = l->next)
|
for (l = self->trackers; l; l = l->next)
|
||||||
{
|
{
|
||||||
gtk_list_item_tracker_unset_position (self, l->data);
|
gtk_list_item_tracker_unset_position (self, l->data);
|
||||||
@ -1579,6 +1642,7 @@ void
|
|||||||
gtk_list_item_manager_set_has_sections (GtkListItemManager *self,
|
gtk_list_item_manager_set_has_sections (GtkListItemManager *self,
|
||||||
gboolean has_sections)
|
gboolean has_sections)
|
||||||
{
|
{
|
||||||
|
GtkListItemChange change;
|
||||||
GtkListTile *tile;
|
GtkListTile *tile;
|
||||||
gboolean had_sections;
|
gboolean had_sections;
|
||||||
|
|
||||||
@ -1642,7 +1706,9 @@ gtk_list_item_manager_set_has_sections (GtkListItemManager *self,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
gtk_list_item_manager_ensure_items (self, NULL, G_MAXUINT);
|
gtk_list_item_change_init (&change);
|
||||||
|
gtk_list_item_manager_ensure_items (self, &change, G_MAXUINT);
|
||||||
|
gtk_list_item_change_finish (&change);
|
||||||
|
|
||||||
gtk_widget_queue_resize (self->widget);
|
gtk_widget_queue_resize (self->widget);
|
||||||
}
|
}
|
||||||
@ -1653,187 +1719,6 @@ gtk_list_item_manager_get_has_sections (GtkListItemManager *self)
|
|||||||
return self->has_sections;
|
return self->has_sections;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* gtk_list_item_manager_acquire_list_item:
|
|
||||||
* @self: a `GtkListItemManager`
|
|
||||||
* @position: the row in the model to create a list item for
|
|
||||||
* @prev_sibling: the widget this widget should be inserted before or %NULL
|
|
||||||
* if it should be the first widget
|
|
||||||
*
|
|
||||||
* Creates a list item widget to use for @position. No widget may
|
|
||||||
* yet exist that is used for @position.
|
|
||||||
*
|
|
||||||
* When the returned item is no longer needed, the caller is responsible
|
|
||||||
* for calling gtk_list_item_manager_release_list_item().
|
|
||||||
* A particular case is when the row at @position is removed. In that case,
|
|
||||||
* all list items in the removed range must be released before
|
|
||||||
* gtk_list_item_manager_model_changed() is called.
|
|
||||||
*
|
|
||||||
* Returns: a properly setup widget to use in @position
|
|
||||||
**/
|
|
||||||
static GtkWidget *
|
|
||||||
gtk_list_item_manager_acquire_list_item (GtkListItemManager *self,
|
|
||||||
guint position,
|
|
||||||
GtkWidget *prev_sibling)
|
|
||||||
{
|
|
||||||
GtkListItemBase *result;
|
|
||||||
gpointer item;
|
|
||||||
gboolean selected;
|
|
||||||
|
|
||||||
g_return_val_if_fail (GTK_IS_LIST_ITEM_MANAGER (self), NULL);
|
|
||||||
g_return_val_if_fail (prev_sibling == NULL || GTK_IS_WIDGET (prev_sibling), NULL);
|
|
||||||
|
|
||||||
result = self->create_widget (self->widget);
|
|
||||||
|
|
||||||
item = g_list_model_get_item (G_LIST_MODEL (self->model), position);
|
|
||||||
selected = gtk_selection_model_is_selected (self->model, position);
|
|
||||||
gtk_list_item_base_update (result, position, item, selected);
|
|
||||||
g_object_unref (item);
|
|
||||||
gtk_widget_insert_after (GTK_WIDGET (result), self->widget, prev_sibling);
|
|
||||||
|
|
||||||
return GTK_WIDGET (result);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* gtk_list_item_manager_try_acquire_list_item_from_change:
|
|
||||||
* @self: a `GtkListItemManager`
|
|
||||||
* @position: the row in the model to create a list item for
|
|
||||||
* @prev_sibling: the widget this widget should be inserted after or %NULL
|
|
||||||
* if it should be the first widget
|
|
||||||
*
|
|
||||||
* Like gtk_list_item_manager_acquire_list_item(), but only tries to acquire list
|
|
||||||
* items from those previously released as part of @change.
|
|
||||||
* If no matching list item is found, %NULL is returned and the caller should use
|
|
||||||
* gtk_list_item_manager_acquire_list_item().
|
|
||||||
*
|
|
||||||
* Returns: (nullable): a properly setup widget to use in @position or %NULL if
|
|
||||||
* no item for reuse existed
|
|
||||||
**/
|
|
||||||
static GtkWidget *
|
|
||||||
gtk_list_item_manager_try_reacquire_list_item (GtkListItemManager *self,
|
|
||||||
GHashTable *change,
|
|
||||||
guint position,
|
|
||||||
GtkWidget *prev_sibling)
|
|
||||||
{
|
|
||||||
GtkWidget *result;
|
|
||||||
gpointer item;
|
|
||||||
|
|
||||||
g_return_val_if_fail (GTK_IS_LIST_ITEM_MANAGER (self), NULL);
|
|
||||||
g_return_val_if_fail (prev_sibling == NULL || GTK_IS_WIDGET (prev_sibling), NULL);
|
|
||||||
|
|
||||||
/* XXX: can we avoid temporarily allocating items on failure? */
|
|
||||||
item = g_list_model_get_item (G_LIST_MODEL (self->model), position);
|
|
||||||
if (g_hash_table_steal_extended (change, item, NULL, (gpointer *) &result))
|
|
||||||
{
|
|
||||||
GtkListItemBase *list_item = GTK_LIST_ITEM_BASE (result);
|
|
||||||
gtk_list_item_base_update (list_item,
|
|
||||||
position,
|
|
||||||
gtk_list_item_base_get_item (list_item),
|
|
||||||
gtk_selection_model_is_selected (self->model, position));
|
|
||||||
gtk_widget_insert_after (result, self->widget, prev_sibling);
|
|
||||||
/* XXX: Should we let the listview do this? */
|
|
||||||
gtk_widget_queue_resize (result);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
result = NULL;
|
|
||||||
}
|
|
||||||
g_object_unref (item);
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* gtk_list_item_manager_move_list_item:
|
|
||||||
* @self: a `GtkListItemManager`
|
|
||||||
* @list_item: an acquired `GtkListItem` that should be moved to represent
|
|
||||||
* a different row
|
|
||||||
* @position: the new position of that list item
|
|
||||||
* @prev_sibling: the new previous sibling
|
|
||||||
*
|
|
||||||
* Moves the widget to represent a new position in the listmodel without
|
|
||||||
* releasing the item.
|
|
||||||
*
|
|
||||||
* This is most useful when scrolling.
|
|
||||||
**/
|
|
||||||
static void
|
|
||||||
gtk_list_item_manager_move_list_item (GtkListItemManager *self,
|
|
||||||
GtkWidget *list_item,
|
|
||||||
guint position,
|
|
||||||
GtkWidget *prev_sibling)
|
|
||||||
{
|
|
||||||
gpointer item;
|
|
||||||
gboolean selected;
|
|
||||||
|
|
||||||
item = g_list_model_get_item (G_LIST_MODEL (self->model), position);
|
|
||||||
selected = gtk_selection_model_is_selected (self->model, position);
|
|
||||||
gtk_list_item_base_update (GTK_LIST_ITEM_BASE (list_item),
|
|
||||||
position,
|
|
||||||
item,
|
|
||||||
selected);
|
|
||||||
gtk_widget_insert_after (list_item, _gtk_widget_get_parent (list_item), prev_sibling);
|
|
||||||
g_object_unref (item);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* gtk_list_item_manager_update_list_item:
|
|
||||||
* @self: a `GtkListItemManager`
|
|
||||||
* @item: a `GtkListItem` that has been acquired
|
|
||||||
* @position: the new position of that list item
|
|
||||||
*
|
|
||||||
* Updates the position of the given @item. This function must be called whenever
|
|
||||||
* the position of an item changes, like when new items are added before it.
|
|
||||||
**/
|
|
||||||
static void
|
|
||||||
gtk_list_item_manager_update_list_item (GtkListItemManager *self,
|
|
||||||
GtkWidget *item,
|
|
||||||
guint position)
|
|
||||||
{
|
|
||||||
GtkListItemBase *list_item = GTK_LIST_ITEM_BASE (item);
|
|
||||||
gboolean selected;
|
|
||||||
|
|
||||||
g_return_if_fail (GTK_IS_LIST_ITEM_MANAGER (self));
|
|
||||||
g_return_if_fail (GTK_IS_LIST_ITEM_BASE (item));
|
|
||||||
|
|
||||||
selected = gtk_selection_model_is_selected (self->model, position);
|
|
||||||
gtk_list_item_base_update (list_item,
|
|
||||||
position,
|
|
||||||
gtk_list_item_base_get_item (list_item),
|
|
||||||
selected);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* gtk_list_item_manager_release_list_item:
|
|
||||||
* @self: a `GtkListItemManager`
|
|
||||||
* @change: (nullable): The change associated with this release or
|
|
||||||
* %NULL if this is a final removal
|
|
||||||
* @item: an item previously acquired with
|
|
||||||
* gtk_list_item_manager_acquire_list_item()
|
|
||||||
*
|
|
||||||
* Releases an item that was previously acquired via
|
|
||||||
* gtk_list_item_manager_acquire_list_item() and is no longer in use.
|
|
||||||
**/
|
|
||||||
static void
|
|
||||||
gtk_list_item_manager_release_list_item (GtkListItemManager *self,
|
|
||||||
GHashTable *change,
|
|
||||||
GtkWidget *item)
|
|
||||||
{
|
|
||||||
g_return_if_fail (GTK_IS_LIST_ITEM_MANAGER (self));
|
|
||||||
g_return_if_fail (GTK_IS_LIST_ITEM_BASE (item));
|
|
||||||
|
|
||||||
if (change != NULL)
|
|
||||||
{
|
|
||||||
if (!g_hash_table_replace (change, gtk_list_item_base_get_item (GTK_LIST_ITEM_BASE (item)), item))
|
|
||||||
{
|
|
||||||
g_warning ("Duplicate item detected in list. Picking one randomly.");
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
gtk_widget_unparent (item);
|
|
||||||
}
|
|
||||||
|
|
||||||
GtkListItemTracker *
|
GtkListItemTracker *
|
||||||
gtk_list_item_tracker_new (GtkListItemManager *self)
|
gtk_list_item_tracker_new (GtkListItemManager *self)
|
||||||
{
|
{
|
||||||
@ -1854,13 +1739,17 @@ void
|
|||||||
gtk_list_item_tracker_free (GtkListItemManager *self,
|
gtk_list_item_tracker_free (GtkListItemManager *self,
|
||||||
GtkListItemTracker *tracker)
|
GtkListItemTracker *tracker)
|
||||||
{
|
{
|
||||||
|
GtkListItemChange change;
|
||||||
|
|
||||||
gtk_list_item_tracker_unset_position (self, tracker);
|
gtk_list_item_tracker_unset_position (self, tracker);
|
||||||
|
|
||||||
self->trackers = g_slist_remove (self->trackers, tracker);
|
self->trackers = g_slist_remove (self->trackers, tracker);
|
||||||
|
|
||||||
g_free (tracker);
|
g_free (tracker);
|
||||||
|
|
||||||
gtk_list_item_manager_ensure_items (self, NULL, G_MAXUINT);
|
gtk_list_item_change_init (&change);
|
||||||
|
gtk_list_item_manager_ensure_items (self, &change, G_MAXUINT);
|
||||||
|
gtk_list_item_change_finish (&change);
|
||||||
|
|
||||||
gtk_widget_queue_resize (self->widget);
|
gtk_widget_queue_resize (self->widget);
|
||||||
}
|
}
|
||||||
@ -1872,6 +1761,7 @@ gtk_list_item_tracker_set_position (GtkListItemManager *self,
|
|||||||
guint n_before,
|
guint n_before,
|
||||||
guint n_after)
|
guint n_after)
|
||||||
{
|
{
|
||||||
|
GtkListItemChange change;
|
||||||
GtkListTile *tile;
|
GtkListTile *tile;
|
||||||
guint n_items;
|
guint n_items;
|
||||||
|
|
||||||
@ -1888,7 +1778,9 @@ gtk_list_item_tracker_set_position (GtkListItemManager *self,
|
|||||||
tracker->n_before = n_before;
|
tracker->n_before = n_before;
|
||||||
tracker->n_after = n_after;
|
tracker->n_after = n_after;
|
||||||
|
|
||||||
gtk_list_item_manager_ensure_items (self, NULL, G_MAXUINT);
|
gtk_list_item_change_init (&change);
|
||||||
|
gtk_list_item_manager_ensure_items (self, &change, G_MAXUINT);
|
||||||
|
gtk_list_item_change_finish (&change);
|
||||||
|
|
||||||
tile = gtk_list_item_manager_get_nth (self, position, NULL);
|
tile = gtk_list_item_manager_get_nth (self, position, NULL);
|
||||||
if (tile)
|
if (tile)
|
||||||
|
Loading…
Reference in New Issue
Block a user