313 lines
6.6 KiB
C
313 lines
6.6 KiB
C
/*
|
|
* This program 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 of the License, or (at your option) version 3.
|
|
*
|
|
* This program 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 the program; if not, see <http://www.gnu.org/licenses/>
|
|
*
|
|
*
|
|
* Authors:
|
|
* Chris Lahey <clahey@ximian.com>
|
|
*
|
|
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "e-sorter-array.h"
|
|
|
|
#include "e-misc-utils.h"
|
|
|
|
#define d(x)
|
|
|
|
#define INCREMENT_AMOUNT 100
|
|
|
|
G_DEFINE_TYPE (
|
|
ESorterArray,
|
|
e_sorter_array,
|
|
E_TYPE_SORTER)
|
|
|
|
static void esa_sort (ESorterArray *esa);
|
|
static void esa_backsort (ESorterArray *esa);
|
|
|
|
static gint esa_model_to_sorted (ESorter *sorter, gint row);
|
|
static gint esa_sorted_to_model (ESorter *sorter, gint row);
|
|
static void esa_get_model_to_sorted_array (ESorter *sorter,
|
|
gint **array,
|
|
gint *count);
|
|
static void esa_get_sorted_to_model_array (ESorter *sorter,
|
|
gint **array,
|
|
gint *count);
|
|
static gboolean esa_needs_sorting (ESorter *esa);
|
|
|
|
#define ESA_NEEDS_SORTING(esa) (((ESorterArray *) (esa))->compare != NULL)
|
|
|
|
static gint
|
|
esort_callback (gconstpointer data1,
|
|
gconstpointer data2,
|
|
gpointer user_data)
|
|
{
|
|
ESorterArray *esa = user_data;
|
|
gint ret_val;
|
|
gint int1, int2;
|
|
|
|
int1 = *(gint *) data1;
|
|
int2 = *(gint *) data2;
|
|
|
|
ret_val = esa->compare (int1, int2, esa->cmp_cache, esa->closure);
|
|
if (ret_val != 0)
|
|
return ret_val;
|
|
|
|
if (int1 < int2)
|
|
return -1;
|
|
if (int1 > int2)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
esa_sort (ESorterArray *esa)
|
|
{
|
|
gint rows;
|
|
gint i;
|
|
|
|
if (esa->sorted)
|
|
return;
|
|
|
|
rows = esa->rows;
|
|
|
|
esa->sorted = g_new (int, rows);
|
|
for (i = 0; i < rows; i++)
|
|
esa->sorted[i] = i;
|
|
|
|
if (esa->compare) {
|
|
if (esa->create_cmp_cache)
|
|
esa->cmp_cache = esa->create_cmp_cache (esa->closure);
|
|
|
|
g_qsort_with_data (
|
|
esa->sorted, rows, sizeof (gint),
|
|
esort_callback, esa);
|
|
|
|
if (esa->cmp_cache) {
|
|
g_hash_table_destroy (esa->cmp_cache);
|
|
esa->cmp_cache = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
esa_backsort (ESorterArray *esa)
|
|
{
|
|
gint i, rows;
|
|
|
|
if (esa->backsorted)
|
|
return;
|
|
|
|
esa_sort (esa);
|
|
|
|
rows = esa->rows;
|
|
|
|
esa->backsorted = g_new0 (int, rows);
|
|
|
|
for (i = 0; i < rows; i++) {
|
|
esa->backsorted[esa->sorted[i]] = i;
|
|
}
|
|
}
|
|
|
|
static gint
|
|
esa_model_to_sorted (ESorter *es,
|
|
gint row)
|
|
{
|
|
ESorterArray *esa = E_SORTER_ARRAY (es);
|
|
|
|
g_return_val_if_fail (row >= 0, -1);
|
|
g_return_val_if_fail (row < esa->rows, -1);
|
|
|
|
if (ESA_NEEDS_SORTING (es))
|
|
esa_backsort (esa);
|
|
|
|
if (esa->backsorted)
|
|
return esa->backsorted[row];
|
|
else
|
|
return row;
|
|
}
|
|
|
|
static gint
|
|
esa_sorted_to_model (ESorter *es,
|
|
gint row)
|
|
{
|
|
ESorterArray *esa = (ESorterArray *) es;
|
|
|
|
g_return_val_if_fail (row >= 0, -1);
|
|
g_return_val_if_fail (row < esa->rows, -1);
|
|
|
|
if (ESA_NEEDS_SORTING (es))
|
|
esa_sort (esa);
|
|
|
|
if (esa->sorted)
|
|
return esa->sorted[row];
|
|
else
|
|
return row;
|
|
}
|
|
|
|
static void
|
|
esa_get_model_to_sorted_array (ESorter *es,
|
|
gint **array,
|
|
gint *count)
|
|
{
|
|
ESorterArray *esa = E_SORTER_ARRAY (es);
|
|
if (array || count) {
|
|
esa_backsort (esa);
|
|
|
|
if (array)
|
|
*array = esa->backsorted;
|
|
if (count)
|
|
*count = esa->rows;
|
|
}
|
|
}
|
|
|
|
static void
|
|
esa_get_sorted_to_model_array (ESorter *es,
|
|
gint **array,
|
|
gint *count)
|
|
{
|
|
ESorterArray *esa = E_SORTER_ARRAY (es);
|
|
if (array || count) {
|
|
esa_sort (esa);
|
|
|
|
if (array)
|
|
*array = esa->sorted;
|
|
if (count)
|
|
*count = esa->rows;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
esa_needs_sorting (ESorter *es)
|
|
{
|
|
ESorterArray *esa = E_SORTER_ARRAY (es);
|
|
return esa->compare != NULL;
|
|
}
|
|
|
|
void
|
|
e_sorter_array_clean (ESorterArray *esa)
|
|
{
|
|
g_free (esa->sorted);
|
|
esa->sorted = NULL;
|
|
|
|
g_free (esa->backsorted);
|
|
esa->backsorted = NULL;
|
|
}
|
|
|
|
void
|
|
e_sorter_array_set_count (ESorterArray *esa,
|
|
gint count)
|
|
{
|
|
e_sorter_array_clean (esa);
|
|
esa->rows = count;
|
|
}
|
|
|
|
void
|
|
e_sorter_array_append (ESorterArray *esa,
|
|
gint count)
|
|
{
|
|
gint i;
|
|
g_free (esa->backsorted);
|
|
esa->backsorted = NULL;
|
|
|
|
if (esa->sorted) {
|
|
esa->sorted = g_renew (int, esa->sorted, esa->rows + count);
|
|
for (i = 0; i < count; i++) {
|
|
gint value = esa->rows;
|
|
gsize pos;
|
|
|
|
e_bsearch (
|
|
&value, esa->sorted, esa->rows,
|
|
sizeof (gint), esort_callback, esa, &pos, NULL);
|
|
memmove (
|
|
esa->sorted + pos + 1,
|
|
esa->sorted + pos,
|
|
sizeof (gint) * (esa->rows - pos));
|
|
esa->sorted[pos] = value;
|
|
esa->rows++;
|
|
}
|
|
} else {
|
|
esa->rows += count;
|
|
}
|
|
}
|
|
|
|
static void
|
|
esa_finalize (GObject *object)
|
|
{
|
|
ESorterArray *esa = E_SORTER_ARRAY (object);
|
|
|
|
if (esa)
|
|
e_sorter_array_clean (esa);
|
|
|
|
/* Chain up to parent's finalize() method. */
|
|
G_OBJECT_CLASS (e_sorter_array_parent_class)->finalize (object);
|
|
}
|
|
|
|
ESorterArray *
|
|
e_sorter_array_construct (ESorterArray *esa,
|
|
ECreateCmpCacheFunc create_cmp_cache,
|
|
ECompareRowsFunc compare,
|
|
gpointer closure)
|
|
{
|
|
esa->create_cmp_cache = create_cmp_cache;
|
|
esa->compare = compare;
|
|
esa->closure = closure;
|
|
return esa;
|
|
}
|
|
|
|
ESorterArray *
|
|
e_sorter_array_new (ECreateCmpCacheFunc create_cmp_cache,
|
|
ECompareRowsFunc compare,
|
|
gpointer closure)
|
|
{
|
|
ESorterArray *esa = g_object_new (E_TYPE_SORTER_ARRAY, NULL);
|
|
|
|
return e_sorter_array_construct (esa, create_cmp_cache, compare, closure);
|
|
}
|
|
|
|
static void
|
|
e_sorter_array_class_init (ESorterArrayClass *class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
ESorterClass *sorter_class = E_SORTER_CLASS (class);
|
|
|
|
object_class->finalize = esa_finalize;
|
|
|
|
sorter_class->model_to_sorted = esa_model_to_sorted;
|
|
sorter_class->sorted_to_model = esa_sorted_to_model;
|
|
sorter_class->get_model_to_sorted_array = esa_get_model_to_sorted_array;
|
|
sorter_class->get_sorted_to_model_array = esa_get_sorted_to_model_array;
|
|
sorter_class->needs_sorting = esa_needs_sorting;
|
|
}
|
|
|
|
static void
|
|
e_sorter_array_init (ESorterArray *esa)
|
|
{
|
|
esa->rows = 0;
|
|
esa->cmp_cache = NULL;
|
|
esa->create_cmp_cache = NULL;
|
|
esa->compare = NULL;
|
|
esa->closure = NULL;
|
|
esa->sorted = NULL;
|
|
esa->backsorted = NULL;
|
|
}
|
|
|