/* * e-table-sort-info.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. * * 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 General Public License * for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, see . * */ #include "e-table-sort-info.h" #include #include "e-table-specification.h" #include "e-xml-utils.h" #define E_TABLE_SORT_INFO_GET_PRIVATE(obj) \ (G_TYPE_INSTANCE_GET_PRIVATE \ ((obj), E_TYPE_TABLE_SORT_INFO, ETableSortInfoPrivate)) typedef struct _ColumnData ColumnData; struct _ETableSortInfoPrivate { GWeakRef specification; GArray *groupings; GArray *sortings; gboolean can_group; }; struct _ColumnData { ETableColumnSpecification *column_spec; GtkSortType sort_type; }; enum { PROP_0, PROP_SPECIFICATION }; enum { SORT_INFO_CHANGED, GROUP_INFO_CHANGED, LAST_SIGNAL }; static guint signals[LAST_SIGNAL]; G_DEFINE_TYPE (ETableSortInfo , e_table_sort_info, G_TYPE_OBJECT) static void column_data_clear (ColumnData *data) { g_clear_object (&data->column_spec); } static void table_sort_info_parser_start_group (GMarkupParseContext *context, const gchar *element_name, const gchar **attribute_names, const gchar **attribute_values, ETableSortInfo *sort_info, GPtrArray *columns, GError **error) { const gchar *index_str; gboolean ascending; gboolean success; success = g_markup_collect_attributes ( element_name, attribute_names, attribute_values, error, G_MARKUP_COLLECT_STRING, "column", &index_str, G_MARKUP_COLLECT_BOOLEAN | G_MARKUP_COLLECT_OPTIONAL, "ascending", &ascending, G_MARKUP_COLLECT_INVALID); if (success) { ETableColumnSpecification *column_spec; ColumnData column_data; gint64 index; g_return_if_fail (index_str != NULL); index = g_ascii_strtoll (index_str, NULL, 10); g_return_if_fail (index < columns->len); column_spec = g_ptr_array_index (columns, index); column_data.column_spec = g_object_ref (column_spec); if (ascending) column_data.sort_type = GTK_SORT_ASCENDING; else column_data.sort_type = GTK_SORT_DESCENDING; g_array_append_val (sort_info->priv->groupings, column_data); } } static void table_sort_info_parser_start_leaf (GMarkupParseContext *context, const gchar *element_name, const gchar **attribute_names, const gchar **attribute_values, ETableSortInfo *sort_info, GPtrArray *columns, GError **error) { const gchar *index_str; gboolean ascending; gboolean success; success = g_markup_collect_attributes ( element_name, attribute_names, attribute_values, error, G_MARKUP_COLLECT_STRING, "column", &index_str, G_MARKUP_COLLECT_BOOLEAN | G_MARKUP_COLLECT_OPTIONAL, "ascending", &ascending, G_MARKUP_COLLECT_INVALID); if (success) { ETableColumnSpecification *column_spec; ColumnData column_data; gint64 index; g_return_if_fail (index_str != NULL); index = g_ascii_strtoll (index_str, NULL, 10); g_return_if_fail (index < columns->len); column_spec = g_ptr_array_index (columns, index); column_data.column_spec = g_object_ref (column_spec); if (ascending) column_data.sort_type = GTK_SORT_ASCENDING; else column_data.sort_type = GTK_SORT_DESCENDING; g_array_append_val (sort_info->priv->sortings, column_data); } } static void table_sort_info_parser_start_element (GMarkupParseContext *context, const gchar *element_name, const gchar **attribute_names, const gchar **attribute_values, gpointer user_data, GError **error) { ETableSpecification *specification; ETableSortInfo *sort_info; GPtrArray *columns; sort_info = E_TABLE_SORT_INFO (user_data); specification = e_table_sort_info_ref_specification (sort_info); columns = e_table_specification_ref_columns (specification); if (g_str_equal (element_name, "group")) table_sort_info_parser_start_group ( context, element_name, attribute_names, attribute_values, sort_info, columns, error); if (g_str_equal (element_name, "leaf")) table_sort_info_parser_start_leaf ( context, element_name, attribute_names, attribute_values, sort_info, columns, error); g_object_unref (specification); g_ptr_array_unref (columns); } static void table_sort_info_parser_error (GMarkupParseContext *context, GError *error, gpointer user_data) { g_object_unref (E_TABLE_SORT_INFO (user_data)); } static const GMarkupParser table_sort_info_parser = { table_sort_info_parser_start_element, NULL, NULL, NULL, table_sort_info_parser_error }; static void table_sort_info_set_specification (ETableSortInfo *sort_info, ETableSpecification *specification) { g_return_if_fail (E_IS_TABLE_SPECIFICATION (specification)); g_weak_ref_set (&sort_info->priv->specification, specification); } static void table_sort_info_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) { switch (property_id) { case PROP_SPECIFICATION: table_sort_info_set_specification ( E_TABLE_SORT_INFO (object), g_value_get_object (value)); return; } G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } static void table_sort_info_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) { switch (property_id) { case PROP_SPECIFICATION: g_value_take_object ( value, e_table_sort_info_ref_specification ( E_TABLE_SORT_INFO (object))); return; } G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } static void table_sort_info_dispose (GObject *object) { ETableSortInfoPrivate *priv; priv = E_TABLE_SORT_INFO_GET_PRIVATE (object); g_weak_ref_set (&priv->specification, NULL); g_array_set_size (priv->groupings, 0); g_array_set_size (priv->sortings, 0); /* Chain up to parent's dispose() method. */ G_OBJECT_CLASS (e_table_sort_info_parent_class)->dispose (object); } static void table_sort_info_finalize (GObject *object) { ETableSortInfoPrivate *priv; priv = E_TABLE_SORT_INFO_GET_PRIVATE (object); g_array_free (priv->groupings, TRUE); g_array_free (priv->sortings, TRUE); /* Chain up to parent's finalize() method. */ G_OBJECT_CLASS (e_table_sort_info_parent_class)->finalize (object); } static void e_table_sort_info_class_init (ETableSortInfoClass *class) { GObjectClass * object_class; g_type_class_add_private (class, sizeof (ETableSortInfoPrivate)); object_class = G_OBJECT_CLASS (class); object_class->set_property = table_sort_info_set_property; object_class->get_property = table_sort_info_get_property; object_class->dispose = table_sort_info_dispose; object_class->finalize = table_sort_info_finalize; g_object_class_install_property ( object_class, PROP_SPECIFICATION, g_param_spec_object ( "specification", "Table Specification", "Specification for the table state", E_TYPE_TABLE_SPECIFICATION, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS)); signals[SORT_INFO_CHANGED] = g_signal_new ( "sort_info_changed", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (ETableSortInfoClass, sort_info_changed), (GSignalAccumulator) NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[GROUP_INFO_CHANGED] = g_signal_new ( "group_info_changed", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (ETableSortInfoClass, group_info_changed), (GSignalAccumulator) NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); } static void e_table_sort_info_init (ETableSortInfo *sort_info) { sort_info->priv = E_TABLE_SORT_INFO_GET_PRIVATE (sort_info); sort_info->priv->groupings = g_array_new ( FALSE, TRUE, sizeof (ColumnData)); g_array_set_clear_func ( sort_info->priv->groupings, (GDestroyNotify) column_data_clear); sort_info->priv->sortings = g_array_new ( FALSE, TRUE, sizeof (ColumnData)); g_array_set_clear_func ( sort_info->priv->sortings, (GDestroyNotify) column_data_clear); sort_info->priv->can_group = TRUE; } /** * e_table_sort_info_new: * @specification: an #ETableSpecification * * This creates a new #ETableSortInfo object that contains no * grouping and no sorting defined as of yet. This object is used * to keep track of multi-level sorting and multi-level grouping of * an #ETable. * * Returns: A new #ETableSortInfo object */ ETableSortInfo * e_table_sort_info_new (ETableSpecification *specification) { g_return_val_if_fail (E_IS_TABLE_SPECIFICATION (specification), NULL); return g_object_new ( E_TYPE_TABLE_SORT_INFO, "specification", specification, NULL); } /** * e_table_sort_info_parse_context_push: * @context: a #GMarkupParseContext * @specification: an #ETableSpecification * * Creates a new #ETableSortInfo from a segment of XML data being fed to * @context. Call this function for the appropriate opening tag from the * start_element callback of a #GMarkupParser, * then call e_table_sort_info_parse_context_pop() for the corresponding * closing tag from the end_element callback. **/ void e_table_sort_info_parse_context_push (GMarkupParseContext *context, ETableSpecification *specification) { g_return_if_fail (context != NULL); g_return_if_fail (E_IS_TABLE_SPECIFICATION (specification)); g_markup_parse_context_push ( context, &table_sort_info_parser, e_table_sort_info_new (specification)); } /** * e_table_sort_info_parse_context_pop: * @context: a #GMarkupParseContext * * Creates a new #ETableSortInfo from a segment of XML data being fed to * @context. Call e_table_sort_info_parse_context_push() for the appropriate * opening tag from the start_element callback of a * #GMarkupParser, then call this function for the corresponding closing tag * from the end_element callback. * * Unreference the newly-created #ETableSortInfo with g_object_unref() when * finished with it. * * Returns: an #ETableSortInfo **/ ETableSortInfo * e_table_sort_info_parse_context_pop (GMarkupParseContext *context) { gpointer user_data; g_return_val_if_fail (context != NULL, NULL); user_data = g_markup_parse_context_pop (context); return E_TABLE_SORT_INFO (user_data); } /** * e_table_sort_info_ref_specification: * @sort_info: an #ETableSortInfo * * Returns the #ETableSpecification passed to e_table_sort_info_new(). * * The returned #ETableSpecification is referenced for thread-safety and must * be unreferenced with g_object_unref() when finished with it. * * Returns: an #ETableSpecification **/ ETableSpecification * e_table_sort_info_ref_specification (ETableSortInfo *sort_info) { g_return_val_if_fail (E_IS_TABLE_SORT_INFO (sort_info), NULL); return g_weak_ref_get (&sort_info->priv->specification); } gboolean e_table_sort_info_get_can_group (ETableSortInfo *sort_info) { g_return_val_if_fail (E_IS_TABLE_SORT_INFO (sort_info), FALSE); return sort_info->priv->can_group; } void e_table_sort_info_set_can_group (ETableSortInfo *sort_info, gboolean can_group) { g_return_if_fail (E_IS_TABLE_SORT_INFO (sort_info)); sort_info->priv->can_group = can_group; } /** * e_table_sort_info_grouping_get_count: * @sort_info: an #ETableSortInfo * * Returns: the number of grouping criteria in the object. */ guint e_table_sort_info_grouping_get_count (ETableSortInfo *sort_info) { guint count = 0; g_return_val_if_fail (E_IS_TABLE_SORT_INFO (sort_info), 0); if (e_table_sort_info_get_can_group (sort_info)) count = sort_info->priv->groupings->len; return count; } /** * e_table_sort_info_grouping_truncate: * @sort_info: an #ETableSortInfo * @length: position where the truncation happens. * * This routine can be used to reduce or grow the number of grouping * criteria in the object. */ void e_table_sort_info_grouping_truncate (ETableSortInfo *sort_info, guint length) { g_return_if_fail (E_IS_TABLE_SORT_INFO (sort_info)); g_array_set_size (sort_info->priv->groupings, length); g_signal_emit (sort_info, signals[GROUP_INFO_CHANGED], 0); } /** * e_table_sort_info_grouping_get_nth: * @sort_info: an #ETableSortInfo * @n: Item information to fetch. * @out_sort_type: return location for a #GtkSortType value, or %NULL * * Returns: the description of the @n-th grouping criteria in the @info object. */ ETableColumnSpecification * e_table_sort_info_grouping_get_nth (ETableSortInfo *sort_info, guint n, GtkSortType *out_sort_type) { ETableColumnSpecification *column_spec = NULL; GArray *array; gboolean can_group; g_return_val_if_fail (E_IS_TABLE_SORT_INFO (sort_info), NULL); array = sort_info->priv->groupings; can_group = e_table_sort_info_get_can_group (sort_info); if (can_group && n < array->len) { ColumnData *column_data; column_data = &g_array_index (array, ColumnData, n); if (out_sort_type != NULL) *out_sort_type = column_data->sort_type; column_spec = column_data->column_spec; } return column_spec; } /** * e_table_sort_info_grouping_set_nth: * @sort_info: an #ETableSortInfo * @n: Item information to fetch. * @spec: an #ETableColumnSpecification * @sort_type: a #GtkSortType * * Sets the grouping criteria for index @n to @spec and @sort_type. */ void e_table_sort_info_grouping_set_nth (ETableSortInfo *sort_info, guint n, ETableColumnSpecification *spec, GtkSortType sort_type) { GArray *array; ColumnData *column_data; g_return_if_fail (E_IS_TABLE_SORT_INFO (sort_info)); g_return_if_fail (E_IS_TABLE_COLUMN_SPECIFICATION (spec)); array = sort_info->priv->groupings; g_array_set_size (array, MAX (n + 1, array->len)); column_data = &g_array_index (array, ColumnData, n); column_data_clear (column_data); column_data->column_spec = g_object_ref (spec); column_data->sort_type = sort_type; g_signal_emit (sort_info, signals[GROUP_INFO_CHANGED], 0); } /** * e_table_sort_info_get_count: * @sort_info: an #ETableSortInfo * * Returns: the number of sorting criteria in the object. */ guint e_table_sort_info_sorting_get_count (ETableSortInfo *sort_info) { g_return_val_if_fail (E_IS_TABLE_SORT_INFO (sort_info), 0); return sort_info->priv->sortings->len; } /** * e_table_sort_info_sorting_remove: * @sort_info: an #ETableSortInfo * @n: the index of the element to remove * * Removes the sorting element at the given index. The following sorting * elements are moved down one place. **/ void e_table_sort_info_sorting_remove (ETableSortInfo *sort_info, guint n) { g_return_if_fail (E_IS_TABLE_SORT_INFO (sort_info)); g_array_remove_index (sort_info->priv->sortings, n); g_signal_emit (sort_info, signals[SORT_INFO_CHANGED], 0); } /** * e_table_sort_info_sorting_truncate: * @sort_info: an #ETableSortInfo * @length: position where the truncation happens. * * This routine can be used to reduce or grow the number of sort * criteria in the object. */ void e_table_sort_info_sorting_truncate (ETableSortInfo *sort_info, guint length) { g_return_if_fail (E_IS_TABLE_SORT_INFO (sort_info)); g_array_set_size (sort_info->priv->sortings, length); g_signal_emit (sort_info, signals[SORT_INFO_CHANGED], 0); } /** * e_table_sort_info_sorting_get_nth: * @sort_info: an #ETableSortInfo * @n: Item information to fetch. * @out_sort_type: return location for a #GtkSortType value, or %NULL * * Returns: the description of the @n-th grouping criteria in the @info object. */ ETableColumnSpecification * e_table_sort_info_sorting_get_nth (ETableSortInfo *sort_info, guint n, GtkSortType *out_sort_type) { ETableColumnSpecification *column_spec = NULL; GArray *array; g_return_val_if_fail (E_IS_TABLE_SORT_INFO (sort_info), NULL); array = sort_info->priv->sortings; if (n < array->len) { ColumnData *column_data; column_data = &g_array_index (array, ColumnData, n); if (out_sort_type != NULL) *out_sort_type = column_data->sort_type; column_spec = column_data->column_spec; } return column_spec; } /** * e_table_sort_info_sorting_set_nth: * @sort_info: an #ETableSortInfo * @n: Item information to fetch. * @spec: an #ETableColumnSpecification * @sort_type: a #GtkSortType * * Sets the sorting criteria for index @n to @spec and @sort_type. */ void e_table_sort_info_sorting_set_nth (ETableSortInfo *sort_info, guint n, ETableColumnSpecification *spec, GtkSortType sort_type) { GArray *array; ColumnData *column_data; g_return_if_fail (E_IS_TABLE_SORT_INFO (sort_info)); g_return_if_fail (E_IS_TABLE_COLUMN_SPECIFICATION (spec)); array = sort_info->priv->sortings; g_array_set_size (array, MAX (n + 1, array->len)); column_data = &g_array_index (array, ColumnData, n); column_data_clear (column_data); column_data->column_spec = g_object_ref (spec); column_data->sort_type = sort_type; g_signal_emit (sort_info, signals[SORT_INFO_CHANGED], 0); } /** * e_table_sort_info_load_from_node: * @sort_info: an #ETableSortInfo * @node: pointer to the xmlNode that describes the sorting and grouping information * @state_version: * * This loads the state for the #ETableSortInfo object @info from the * xml node @node. */ void e_table_sort_info_load_from_node (ETableSortInfo *sort_info, xmlNode *node, gdouble state_version) { ETableSpecification *specification; GPtrArray *columns; xmlNode *grouping; guint gcnt = 0; guint scnt = 0; g_return_if_fail (E_IS_TABLE_SORT_INFO (sort_info)); g_return_if_fail (node != NULL); specification = e_table_sort_info_ref_specification (sort_info); columns = e_table_specification_ref_columns (specification); for (grouping = node->children; grouping; grouping = grouping->next) { if (grouping->type != XML_ELEMENT_NODE) continue; if (g_str_equal ((gchar *) grouping->name, "group")) { GtkSortType sort_type; gboolean ascending; guint index; index = e_xml_get_integer_prop_by_name ( grouping, (guchar *) "column"); ascending = e_xml_get_bool_prop_by_name ( grouping, (guchar *) "ascending"); if (ascending) sort_type = GTK_SORT_ASCENDING; else sort_type = GTK_SORT_DESCENDING; if (index < columns->len) e_table_sort_info_grouping_set_nth ( sort_info, gcnt++, columns->pdata[index], sort_type); } if (g_str_equal ((gchar *) grouping->name, "leaf")) { GtkSortType sort_type; gboolean ascending; gint index;; index = e_xml_get_integer_prop_by_name ( grouping, (guchar *) "column"); ascending = e_xml_get_bool_prop_by_name ( grouping, (guchar *) "ascending"); if (ascending) sort_type = GTK_SORT_ASCENDING; else sort_type = GTK_SORT_DESCENDING; if (index < columns->len) e_table_sort_info_sorting_set_nth ( sort_info, scnt++, columns->pdata[index], sort_type); } } g_object_unref (specification); g_ptr_array_unref (columns); g_signal_emit (sort_info, signals[SORT_INFO_CHANGED], 0); } /** * e_table_sort_info_save_to_node: * @sort_info: an #ETableSortInfo * @parent: xmlNode that will be hosting the saved state of the @info object. * * This function is used * * Returns: the node that has been appended to @parent as a child containing * the sorting and grouping information for this ETableSortInfo object. */ xmlNode * e_table_sort_info_save_to_node (ETableSortInfo *sort_info, xmlNode *parent) { ETableSpecification *specification; xmlNode *grouping; guint sort_count; guint group_count; guint ii; g_return_val_if_fail (E_IS_TABLE_SORT_INFO (sort_info), NULL); sort_count = e_table_sort_info_sorting_get_count (sort_info); group_count = e_table_sort_info_grouping_get_count (sort_info); grouping = xmlNewChild (parent, NULL, (guchar *) "grouping", NULL); specification = e_table_sort_info_ref_specification (sort_info); for (ii = 0; ii < group_count; ii++) { ETableColumnSpecification *column_spec; GtkSortType sort_type; xmlNode *new_node; gint index; column_spec = e_table_sort_info_grouping_get_nth ( sort_info, ii, &sort_type); index = e_table_specification_get_column_index ( specification, column_spec); if (index < 0) { g_warn_if_reached (); continue; } new_node = xmlNewChild ( grouping, NULL, (guchar *) "group", NULL); e_xml_set_integer_prop_by_name ( new_node, (guchar *) "column", index); e_xml_set_bool_prop_by_name ( new_node, (guchar *) "ascending", (sort_type == GTK_SORT_ASCENDING)); } for (ii = 0; ii < sort_count; ii++) { ETableColumnSpecification *column_spec; GtkSortType sort_type; xmlNode *new_node; gint index; column_spec = e_table_sort_info_sorting_get_nth ( sort_info, ii, &sort_type); index = e_table_specification_get_column_index ( specification, column_spec); if (index < 0) { g_warn_if_reached (); continue; } new_node = xmlNewChild ( grouping, NULL, (guchar *) "leaf", NULL); e_xml_set_integer_prop_by_name ( new_node, (guchar *) "column", index); e_xml_set_bool_prop_by_name ( new_node, (guchar *) "ascending", (sort_type == GTK_SORT_ASCENDING)); } g_object_unref (specification); return grouping; } ETableSortInfo * e_table_sort_info_duplicate (ETableSortInfo *sort_info) { ETableSpecification *specification; ETableSortInfo *new_info; gint ii; g_return_val_if_fail (E_IS_TABLE_SORT_INFO (sort_info), NULL); specification = e_table_sort_info_ref_specification (sort_info); new_info = e_table_sort_info_new (specification); g_object_unref (specification); g_array_set_size ( new_info->priv->groupings, sort_info->priv->groupings->len); memmove ( new_info->priv->groupings->data, sort_info->priv->groupings->data, sort_info->priv->groupings->len * g_array_get_element_size (sort_info->priv->groupings)); for (ii = 0; ii < new_info->priv->groupings->len; ii++) { ColumnData *column_data; column_data = &g_array_index (new_info->priv->groupings, ColumnData, ii); g_object_ref (column_data->column_spec); } g_array_set_size ( new_info->priv->sortings, sort_info->priv->sortings->len); memmove ( new_info->priv->sortings->data, sort_info->priv->sortings->data, sort_info->priv->sortings->len * g_array_get_element_size (sort_info->priv->sortings)); for (ii = 0; ii < new_info->priv->sortings->len; ii++) { ColumnData *column_data; column_data = &g_array_index (new_info->priv->sortings, ColumnData, ii); g_object_ref (column_data->column_spec); } new_info->priv->can_group = sort_info->priv->can_group; return new_info; }