/* * Copyright © 2019 Matthias Clasen * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . * * Authors: Matthias Clasen */ #include "config.h" #include "gtkcolumnviewsorterprivate.h" #include "gtkcolumnviewcolumnprivate.h" #include "gtkintl.h" #include "gtktypebuiltins.h" typedef struct { GtkColumnViewColumn *column; GtkSorter *sorter; gboolean inverted; gulong changed_id; } Sorter; static void free_sorter (gpointer data) { Sorter *s = data; g_signal_handler_disconnect (s->sorter, s->changed_id); g_object_unref (s->sorter); g_object_unref (s->column); g_free (s); } struct _GtkColumnViewSorter { GtkSorter parent_instance; GSequence *sorters; }; G_DEFINE_TYPE (GtkColumnViewSorter, gtk_column_view_sorter, GTK_TYPE_SORTER) static GtkOrdering gtk_column_view_sorter_compare (GtkSorter *sorter, gpointer item1, gpointer item2) { GtkColumnViewSorter *self = GTK_COLUMN_VIEW_SORTER (sorter); GtkOrdering result = GTK_ORDERING_EQUAL; GSequenceIter *iter; for (iter = g_sequence_get_begin_iter (self->sorters); !g_sequence_iter_is_end (iter); iter = g_sequence_iter_next (iter)) { Sorter *s = g_sequence_get (iter); result = gtk_sorter_compare (s->sorter, item1, item2); if (s->inverted) result = - result; if (result != GTK_ORDERING_EQUAL) break; } return result; } static GtkSorterOrder gtk_column_view_sorter_get_order (GtkSorter *sorter) { GtkColumnViewSorter *self = GTK_COLUMN_VIEW_SORTER (sorter); GtkSorterOrder result = GTK_SORTER_ORDER_NONE; GSequenceIter *iter; for (iter = g_sequence_get_begin_iter (self->sorters); !g_sequence_iter_is_end (iter); iter = g_sequence_iter_next (iter)) { Sorter *s = g_sequence_get (iter); switch (gtk_sorter_get_order (s->sorter)) { case GTK_SORTER_ORDER_PARTIAL: result = GTK_SORTER_ORDER_PARTIAL; break; case GTK_SORTER_ORDER_NONE: break; case GTK_SORTER_ORDER_TOTAL: return GTK_SORTER_ORDER_TOTAL; default: g_assert_not_reached (); break; } } return result; } static void gtk_column_view_sorter_dispose (GObject *object) { GtkColumnViewSorter *self = GTK_COLUMN_VIEW_SORTER (object); /* The sorter is owned by the columview and is unreffed * after the columns, so the sequence must be empty at * this point. * The sorter can outlive the columview it comes from * (the model might still have a ref), but that does * not change the fact that all columns will be gone. */ g_assert (g_sequence_is_empty (self->sorters)); g_clear_pointer (&self->sorters, g_sequence_free); G_OBJECT_CLASS (gtk_column_view_sorter_parent_class)->dispose (object); } static void gtk_column_view_sorter_class_init (GtkColumnViewSorterClass *class) { GtkSorterClass *sorter_class = GTK_SORTER_CLASS (class); GObjectClass *object_class = G_OBJECT_CLASS (class); sorter_class->compare = gtk_column_view_sorter_compare; sorter_class->get_order = gtk_column_view_sorter_get_order; object_class->dispose = gtk_column_view_sorter_dispose; } static void gtk_column_view_sorter_init (GtkColumnViewSorter *self) { self->sorters = g_sequence_new (free_sorter); } GtkSorter * gtk_column_view_sorter_new (void) { return g_object_new (GTK_TYPE_COLUMN_VIEW_SORTER, NULL); } static void gtk_column_view_sorter_changed_cb (GtkSorter *sorter, int change, gpointer data) { gtk_sorter_changed (GTK_SORTER (data), GTK_SORTER_CHANGE_DIFFERENT); } static gboolean remove_column (GtkColumnViewSorter *self, GtkColumnViewColumn *column) { GSequenceIter *iter; for (iter = g_sequence_get_begin_iter (self->sorters); !g_sequence_iter_is_end (iter); iter = g_sequence_iter_next (iter)) { Sorter *s = g_sequence_get (iter); if (s->column == column) { g_sequence_remove (iter); return TRUE; } } return FALSE; } gboolean gtk_column_view_sorter_add_column (GtkColumnViewSorter *self, GtkColumnViewColumn *column) { GSequenceIter *iter; GtkSorter *sorter; Sorter *s, *first; g_return_val_if_fail (GTK_IS_COLUMN_VIEW_SORTER (self), FALSE); g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (column), FALSE); sorter = gtk_column_view_column_get_sorter (column); if (sorter == NULL) return FALSE; iter = g_sequence_get_begin_iter (self->sorters); if (!g_sequence_iter_is_end (iter)) { first = g_sequence_get (iter); if (first->column == column) { first->inverted = !first->inverted; goto out; } } else first = NULL; remove_column (self, column); s = g_new (Sorter, 1); s->column = g_object_ref (column); s->sorter = g_object_ref (sorter); s->changed_id = g_signal_connect (sorter, "changed", G_CALLBACK (gtk_column_view_sorter_changed_cb), self); s->inverted = FALSE; g_sequence_insert_before (iter, s); /* notify the previous first column to stop drawing an arrow */ if (first) gtk_column_view_column_notify_sort (first->column); out: gtk_sorter_changed (GTK_SORTER (self), GTK_SORTER_CHANGE_DIFFERENT); gtk_column_view_column_notify_sort (column); return TRUE; } gboolean gtk_column_view_sorter_remove_column (GtkColumnViewSorter *self, GtkColumnViewColumn *column) { g_return_val_if_fail (GTK_IS_COLUMN_VIEW_SORTER (self), FALSE); g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (column), FALSE); if (remove_column (self, column)) { gtk_sorter_changed (GTK_SORTER (self), GTK_SORTER_CHANGE_DIFFERENT); gtk_column_view_column_notify_sort (column); return TRUE; } return FALSE; } gboolean gtk_column_view_sorter_set_column (GtkColumnViewSorter *self, GtkColumnViewColumn *column, gboolean inverted) { GtkSorter *sorter; Sorter *s; g_return_val_if_fail (GTK_IS_COLUMN_VIEW_SORTER (self), FALSE); g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (column), FALSE); sorter = gtk_column_view_column_get_sorter (column); if (sorter == NULL) return FALSE; g_object_ref (column); g_sequence_remove_range (g_sequence_get_begin_iter (self->sorters), g_sequence_get_end_iter (self->sorters)); s = g_new (Sorter, 1); s->column = g_object_ref (column); s->sorter = g_object_ref (sorter); s->changed_id = g_signal_connect (sorter, "changed", G_CALLBACK (gtk_column_view_sorter_changed_cb), self); s->inverted = inverted; g_sequence_prepend (self->sorters, s); gtk_sorter_changed (GTK_SORTER (self), GTK_SORTER_CHANGE_DIFFERENT); gtk_column_view_column_notify_sort (column); g_object_unref (column); return TRUE; } void gtk_column_view_sorter_clear (GtkColumnViewSorter *self) { GSequenceIter *iter; Sorter *s; GtkColumnViewColumn *column; g_return_if_fail (GTK_IS_COLUMN_VIEW_SORTER (self)); if (g_sequence_is_empty (self->sorters)) return; iter = g_sequence_get_begin_iter (self->sorters); s = g_sequence_get (iter); column = g_object_ref (s->column); g_sequence_remove_range (iter, g_sequence_get_end_iter (self->sorters)); gtk_sorter_changed (GTK_SORTER (self), GTK_SORTER_CHANGE_DIFFERENT); gtk_column_view_column_notify_sort (column); g_object_unref (column); } GtkColumnViewColumn * gtk_column_view_sorter_get_sort_column (GtkColumnViewSorter *self, gboolean *inverted) { GSequenceIter *iter; Sorter *s; g_return_val_if_fail (GTK_IS_COLUMN_VIEW_SORTER (self), NULL); if (g_sequence_is_empty (self->sorters)) return NULL; iter = g_sequence_get_begin_iter (self->sorters); s = g_sequence_get (iter); *inverted = s->inverted; return s->column; }