forked from AuroraMiddleware/gtk
185f0181ac
Columnview reorder cells See merge request GNOME/gtk!2047
1056 lines
28 KiB
C
1056 lines
28 KiB
C
/*
|
|
* Copyright © 2019 Benjamin Otte
|
|
*
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Authors: Benjamin Otte <otte@gnome.org>
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gtkcolumnviewcolumnprivate.h"
|
|
#include "gtkcolumnviewsorterprivate.h"
|
|
|
|
#include "gtkcolumnviewprivate.h"
|
|
#include "gtkcolumnviewtitleprivate.h"
|
|
#include "gtkintl.h"
|
|
#include "gtklistbaseprivate.h"
|
|
#include "gtklistitemwidgetprivate.h"
|
|
#include "gtkmain.h"
|
|
#include "gtkprivate.h"
|
|
#include "gtkrbtreeprivate.h"
|
|
#include "gtksizegroup.h"
|
|
#include "gtkstylecontext.h"
|
|
#include "gtkwidgetprivate.h"
|
|
#include "gtksorter.h"
|
|
|
|
/**
|
|
* SECTION:gtkcolumnviewcolumn
|
|
* @title: GtkColumnViewColumn
|
|
* @short_description: The column added to GtkColumnView
|
|
* @see_also: #GtkColumnView
|
|
*
|
|
* GtkColumnViewColumn represents the columns being added to #GtkColumnView.
|
|
*/
|
|
|
|
struct _GtkColumnViewColumn
|
|
{
|
|
GObject parent_instance;
|
|
|
|
GtkListItemFactory *factory;
|
|
char *title;
|
|
GtkSorter *sorter;
|
|
|
|
/* data for the view */
|
|
GtkColumnView *view;
|
|
GtkWidget *header;
|
|
|
|
int minimum_size_request;
|
|
int natural_size_request;
|
|
int allocation_offset;
|
|
int allocation_size;
|
|
int header_position;
|
|
|
|
int fixed_width;
|
|
|
|
guint visible : 1;
|
|
guint resizable : 1;
|
|
guint expand : 1;
|
|
|
|
GMenuModel *menu;
|
|
|
|
/* This list isn't sorted - this is just caching for performance */
|
|
GtkColumnViewCell *first_cell; /* no reference, just caching */
|
|
};
|
|
|
|
struct _GtkColumnViewColumnClass
|
|
{
|
|
GObjectClass parent_class;
|
|
};
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_COLUMN_VIEW,
|
|
PROP_FACTORY,
|
|
PROP_TITLE,
|
|
PROP_SORTER,
|
|
PROP_VISIBLE,
|
|
PROP_HEADER_MENU,
|
|
PROP_RESIZABLE,
|
|
PROP_EXPAND,
|
|
PROP_FIXED_WIDTH,
|
|
|
|
N_PROPS
|
|
};
|
|
|
|
G_DEFINE_TYPE (GtkColumnViewColumn, gtk_column_view_column, G_TYPE_OBJECT)
|
|
|
|
static GParamSpec *properties[N_PROPS] = { NULL, };
|
|
|
|
static void
|
|
gtk_column_view_column_dispose (GObject *object)
|
|
{
|
|
GtkColumnViewColumn *self = GTK_COLUMN_VIEW_COLUMN (object);
|
|
|
|
g_assert (self->view == NULL); /* would hold a ref otherwise */
|
|
g_assert (self->first_cell == NULL); /* no view = no children */
|
|
|
|
g_clear_object (&self->factory);
|
|
g_clear_object (&self->sorter);
|
|
g_clear_pointer (&self->title, g_free);
|
|
g_clear_object (&self->menu);
|
|
|
|
G_OBJECT_CLASS (gtk_column_view_column_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gtk_column_view_column_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkColumnViewColumn *self = GTK_COLUMN_VIEW_COLUMN (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_COLUMN_VIEW:
|
|
g_value_set_object (value, self->view);
|
|
break;
|
|
|
|
case PROP_FACTORY:
|
|
g_value_set_object (value, self->factory);
|
|
break;
|
|
|
|
case PROP_TITLE:
|
|
g_value_set_string (value, self->title);
|
|
break;
|
|
|
|
case PROP_SORTER:
|
|
g_value_set_object (value, self->sorter);
|
|
break;
|
|
|
|
case PROP_VISIBLE:
|
|
g_value_set_boolean (value, self->visible);
|
|
break;
|
|
|
|
case PROP_HEADER_MENU:
|
|
g_value_set_object (value, self->menu);
|
|
break;
|
|
|
|
case PROP_RESIZABLE:
|
|
g_value_set_boolean (value, self->resizable);
|
|
break;
|
|
|
|
case PROP_EXPAND:
|
|
g_value_set_boolean (value, self->expand);
|
|
break;
|
|
|
|
case PROP_FIXED_WIDTH:
|
|
g_value_set_int (value, self->fixed_width);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_column_view_column_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkColumnViewColumn *self = GTK_COLUMN_VIEW_COLUMN (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_FACTORY:
|
|
gtk_column_view_column_set_factory (self, g_value_get_object (value));
|
|
break;
|
|
|
|
case PROP_TITLE:
|
|
gtk_column_view_column_set_title (self, g_value_get_string (value));
|
|
break;
|
|
|
|
case PROP_SORTER:
|
|
gtk_column_view_column_set_sorter (self, g_value_get_object (value));
|
|
break;
|
|
|
|
case PROP_VISIBLE:
|
|
gtk_column_view_column_set_visible (self, g_value_get_boolean (value));
|
|
break;
|
|
|
|
case PROP_HEADER_MENU:
|
|
gtk_column_view_column_set_header_menu (self, g_value_get_object (value));
|
|
break;
|
|
|
|
case PROP_RESIZABLE:
|
|
gtk_column_view_column_set_resizable (self, g_value_get_boolean (value));
|
|
break;
|
|
|
|
case PROP_EXPAND:
|
|
gtk_column_view_column_set_expand (self, g_value_get_boolean (value));
|
|
break;
|
|
|
|
case PROP_FIXED_WIDTH:
|
|
gtk_column_view_column_set_fixed_width (self, g_value_get_int (value));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_column_view_column_class_init (GtkColumnViewColumnClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
gobject_class->dispose = gtk_column_view_column_dispose;
|
|
gobject_class->get_property = gtk_column_view_column_get_property;
|
|
gobject_class->set_property = gtk_column_view_column_set_property;
|
|
|
|
/**
|
|
* GtkColumnViewColumn:column-view:
|
|
*
|
|
* #GtkColumnView this column is a part of
|
|
*/
|
|
properties[PROP_COLUMN_VIEW] =
|
|
g_param_spec_object ("column-view",
|
|
P_("Column view"),
|
|
P_("Column view this column is a part of"),
|
|
GTK_TYPE_COLUMN_VIEW,
|
|
G_PARAM_READABLE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
/**
|
|
* GtkColumnViewColumn:factory:
|
|
*
|
|
* Factory for populating list items
|
|
*/
|
|
properties[PROP_FACTORY] =
|
|
g_param_spec_object ("factory",
|
|
P_("Factory"),
|
|
P_("Factory for populating list items"),
|
|
GTK_TYPE_LIST_ITEM_FACTORY,
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
/**
|
|
* GtkColumnViewColumn:title:
|
|
*
|
|
* Title displayed in the header
|
|
*/
|
|
properties[PROP_TITLE] =
|
|
g_param_spec_string ("title",
|
|
P_("Title"),
|
|
P_("Title displayed in the header"),
|
|
NULL,
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
|
|
|
|
/**
|
|
* GtkColumnViewColumn:sorter:
|
|
*
|
|
* Sorter for sorting items according to this column
|
|
*/
|
|
properties[PROP_SORTER] =
|
|
g_param_spec_object ("sorter",
|
|
P_("Sorter"),
|
|
P_("Sorter for sorting items according to this column"),
|
|
GTK_TYPE_SORTER,
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
/**
|
|
* GtkColumnViewColumn:visible:
|
|
*
|
|
* Whether this column is visible
|
|
*/
|
|
properties[PROP_VISIBLE] =
|
|
g_param_spec_boolean ("visible",
|
|
P_("Visible"),
|
|
P_("Whether this column is visible"),
|
|
TRUE,
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
/**
|
|
* GtkColumnViewColumn:header-menu:
|
|
*
|
|
* Menu model used to create the context menu for the column header.
|
|
*/
|
|
properties[PROP_HEADER_MENU] =
|
|
g_param_spec_object ("header-menu",
|
|
P_("Header menu"),
|
|
P_("Menu to use on the title of this column"),
|
|
G_TYPE_MENU_MODEL,
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
/**
|
|
* GtkColumnViewColumn:resizable:
|
|
*
|
|
* Whether this column is resizable
|
|
*/
|
|
properties[PROP_RESIZABLE] =
|
|
g_param_spec_boolean ("resizable",
|
|
P_("Resizable"),
|
|
P_("Whether this column is resizable"),
|
|
FALSE,
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
/**
|
|
* GtkColumnViewColumn:expand:
|
|
*
|
|
* Column gets share of extra width allocated to the view
|
|
*/
|
|
properties[PROP_EXPAND] =
|
|
g_param_spec_boolean ("expand",
|
|
P_("Expand"),
|
|
P_("column gets share of extra width"),
|
|
FALSE,
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
/**
|
|
* GtkColumnViewColumn:fixed-width:
|
|
*
|
|
* If not -1, this is the width that the column is allocated,
|
|
* regardless of the size of its content.
|
|
*/
|
|
properties[PROP_FIXED_WIDTH] =
|
|
g_param_spec_int ("fixed-width",
|
|
P_("Fixed width"),
|
|
P_("Fixed width of this column"),
|
|
-1, G_MAXINT, -1,
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
g_object_class_install_properties (gobject_class, N_PROPS, properties);
|
|
}
|
|
|
|
static void
|
|
gtk_column_view_column_init (GtkColumnViewColumn *self)
|
|
{
|
|
self->minimum_size_request = -1;
|
|
self->natural_size_request = -1;
|
|
self->visible = TRUE;
|
|
self->resizable = FALSE;
|
|
self->expand = FALSE;
|
|
self->fixed_width = -1;
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_new:
|
|
* @title: (nullable): Title to use for this column
|
|
*
|
|
* Creates a new #GtkColumnViewColumn.
|
|
*
|
|
* You most likely want to call gtk_column_add_column() next.
|
|
*
|
|
* Returns: a new #GtkColumnViewColumn
|
|
**/
|
|
GtkColumnViewColumn *
|
|
gtk_column_view_column_new (const char *title)
|
|
{
|
|
return g_object_new (GTK_TYPE_COLUMN_VIEW_COLUMN,
|
|
"title", title,
|
|
NULL);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_new_with_factory:
|
|
* @title: (nullable): Title to use for this column
|
|
* @factory: (transfer full): The factory to populate items with
|
|
*
|
|
* Creates a new #GtkColumnViewColumn that uses the given @factory for
|
|
* mapping items to widgets.
|
|
*
|
|
* You most likely want to call gtk_column_add_column() next.
|
|
*
|
|
* The function takes ownership of the
|
|
* argument, so you can write code like
|
|
* ```
|
|
* column = gtk_column_view_column_new_with_factory (_("Name"),
|
|
* gtk_builder_list_item_factory_new_from_resource ("/name.ui"));
|
|
* ```
|
|
*
|
|
* Returns: a new #GtkColumnViewColumn using the given @factory
|
|
**/
|
|
GtkColumnViewColumn *
|
|
gtk_column_view_column_new_with_factory (const char *title,
|
|
GtkListItemFactory *factory)
|
|
{
|
|
GtkColumnViewColumn *result;
|
|
|
|
g_return_val_if_fail (GTK_IS_LIST_ITEM_FACTORY (factory), NULL);
|
|
|
|
result = g_object_new (GTK_TYPE_COLUMN_VIEW_COLUMN,
|
|
"factory", factory,
|
|
"title", title,
|
|
NULL);
|
|
|
|
g_object_unref (factory);
|
|
|
|
return result;
|
|
}
|
|
|
|
GtkColumnViewCell *
|
|
gtk_column_view_column_get_first_cell (GtkColumnViewColumn *self)
|
|
{
|
|
return self->first_cell;
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_add_cell (GtkColumnViewColumn *self,
|
|
GtkColumnViewCell *cell)
|
|
{
|
|
self->first_cell = cell;
|
|
|
|
gtk_widget_set_visible (GTK_WIDGET (cell), self->visible);
|
|
gtk_column_view_column_queue_resize (self);
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_remove_cell (GtkColumnViewColumn *self,
|
|
GtkColumnViewCell *cell)
|
|
{
|
|
if (cell == self->first_cell)
|
|
self->first_cell = gtk_column_view_cell_get_next (cell);
|
|
|
|
gtk_column_view_column_queue_resize (self);
|
|
gtk_widget_queue_resize (GTK_WIDGET (cell));
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_queue_resize (GtkColumnViewColumn *self)
|
|
{
|
|
GtkColumnViewCell *cell;
|
|
|
|
if (self->minimum_size_request < 0)
|
|
return;
|
|
|
|
self->minimum_size_request = -1;
|
|
self->natural_size_request = -1;
|
|
|
|
if (self->header)
|
|
gtk_widget_queue_resize (self->header);
|
|
|
|
for (cell = self->first_cell; cell; cell = gtk_column_view_cell_get_next (cell))
|
|
{
|
|
gtk_widget_queue_resize (GTK_WIDGET (cell));
|
|
}
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_measure (GtkColumnViewColumn *self,
|
|
int *minimum,
|
|
int *natural)
|
|
{
|
|
if (self->fixed_width > -1)
|
|
{
|
|
self->minimum_size_request = self->fixed_width;
|
|
self->natural_size_request = self->fixed_width;
|
|
}
|
|
|
|
if (self->minimum_size_request < 0)
|
|
{
|
|
GtkColumnViewCell *cell;
|
|
int min, nat, cell_min, cell_nat;
|
|
|
|
if (self->header)
|
|
{
|
|
gtk_widget_measure (self->header, GTK_ORIENTATION_HORIZONTAL, -1, &min, &nat, NULL, NULL);
|
|
}
|
|
else
|
|
{
|
|
min = 0;
|
|
nat = 0;
|
|
}
|
|
|
|
for (cell = self->first_cell; cell; cell = gtk_column_view_cell_get_next (cell))
|
|
{
|
|
gtk_widget_measure (GTK_WIDGET (cell),
|
|
GTK_ORIENTATION_HORIZONTAL,
|
|
-1,
|
|
&cell_min, &cell_nat,
|
|
NULL, NULL);
|
|
|
|
min = MAX (min, cell_min);
|
|
nat = MAX (nat, cell_nat);
|
|
}
|
|
|
|
self->minimum_size_request = min;
|
|
self->natural_size_request = nat;
|
|
}
|
|
|
|
*minimum = self->minimum_size_request;
|
|
*natural = self->natural_size_request;
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_allocate (GtkColumnViewColumn *self,
|
|
int offset,
|
|
int size)
|
|
{
|
|
self->allocation_offset = offset;
|
|
self->allocation_size = size;
|
|
self->header_position = offset;
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_get_allocation (GtkColumnViewColumn *self,
|
|
int *offset,
|
|
int *size)
|
|
{
|
|
if (offset)
|
|
*offset = self->allocation_offset;
|
|
if (size)
|
|
*size = self->allocation_size;
|
|
}
|
|
|
|
static void
|
|
gtk_column_view_column_create_cells (GtkColumnViewColumn *self)
|
|
{
|
|
GtkWidget *row;
|
|
|
|
if (self->first_cell)
|
|
return;
|
|
|
|
for (row = gtk_widget_get_first_child (GTK_WIDGET (self->view));
|
|
row != NULL;
|
|
row = gtk_widget_get_next_sibling (row))
|
|
{
|
|
GtkListItemWidget *list_item;
|
|
GtkWidget *cell;
|
|
|
|
if (!gtk_widget_get_root (row))
|
|
continue;
|
|
|
|
list_item = GTK_LIST_ITEM_WIDGET (row);
|
|
cell = gtk_column_view_cell_new (self);
|
|
gtk_list_item_widget_add_child (list_item, cell);
|
|
gtk_list_item_widget_update (GTK_LIST_ITEM_WIDGET (cell),
|
|
gtk_list_item_widget_get_position (list_item),
|
|
gtk_list_item_widget_get_item (list_item),
|
|
gtk_list_item_widget_get_selected (list_item));
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_column_view_column_remove_cells (GtkColumnViewColumn *self)
|
|
{
|
|
while (self->first_cell)
|
|
gtk_column_view_cell_remove (self->first_cell);
|
|
}
|
|
|
|
static void
|
|
gtk_column_view_column_create_header (GtkColumnViewColumn *self)
|
|
{
|
|
if (self->header != NULL)
|
|
return;
|
|
|
|
self->header = gtk_column_view_title_new (self);
|
|
gtk_widget_set_visible (self->header, self->visible);
|
|
gtk_list_item_widget_add_child (gtk_column_view_get_header_widget (self->view),
|
|
self->header);
|
|
gtk_column_view_column_queue_resize (self);
|
|
}
|
|
|
|
static void
|
|
gtk_column_view_column_remove_header (GtkColumnViewColumn *self)
|
|
{
|
|
if (self->header == NULL)
|
|
return;
|
|
|
|
gtk_list_item_widget_remove_child (gtk_column_view_get_header_widget (self->view),
|
|
self->header);
|
|
self->header = NULL;
|
|
gtk_column_view_column_queue_resize (self);
|
|
}
|
|
|
|
static void
|
|
gtk_column_view_column_ensure_cells (GtkColumnViewColumn *self)
|
|
{
|
|
if (self->view && gtk_widget_get_root (GTK_WIDGET (self->view)))
|
|
gtk_column_view_column_create_cells (self);
|
|
else
|
|
gtk_column_view_column_remove_cells (self);
|
|
|
|
if (self->view)
|
|
gtk_column_view_column_create_header (self);
|
|
else
|
|
gtk_column_view_column_remove_header (self);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_get_column_view:
|
|
* @self: a #GtkColumnViewColumn
|
|
*
|
|
* Gets the column view that's currently displaying this column.
|
|
*
|
|
* If @self has not been added to a column view yet, %NULL is returned.
|
|
*
|
|
* Returns: (nullable) (transfer none): The column view displaying @self.
|
|
**/
|
|
GtkColumnView *
|
|
gtk_column_view_column_get_column_view (GtkColumnViewColumn *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self), NULL);
|
|
|
|
return self->view;
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_set_column_view (GtkColumnViewColumn *self,
|
|
GtkColumnView *view)
|
|
{
|
|
if (self->view == view)
|
|
return;
|
|
|
|
gtk_column_view_column_remove_cells (self);
|
|
gtk_column_view_column_remove_header (self);
|
|
|
|
self->view = view;
|
|
|
|
gtk_column_view_column_ensure_cells (self);
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_COLUMN_VIEW]);
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_set_position (GtkColumnViewColumn *self,
|
|
guint position)
|
|
{
|
|
GtkColumnViewCell *cell;
|
|
|
|
gtk_list_item_widget_reorder_child (gtk_column_view_get_header_widget (self->view),
|
|
self->header,
|
|
position);
|
|
|
|
for (cell = self->first_cell; cell; cell = gtk_column_view_cell_get_next (cell))
|
|
{
|
|
GtkListItemWidget *list_item;
|
|
|
|
list_item = GTK_LIST_ITEM_WIDGET (gtk_widget_get_parent (GTK_WIDGET (cell)));
|
|
gtk_list_item_widget_reorder_child (list_item, GTK_WIDGET (cell), position);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_get_factory:
|
|
* @self: a #GtkColumnViewColumn
|
|
*
|
|
* Gets the factory that's currently used to populate list items for
|
|
* this column.
|
|
*
|
|
* Returns: (nullable) (transfer none): The factory in use
|
|
**/
|
|
GtkListItemFactory *
|
|
gtk_column_view_column_get_factory (GtkColumnViewColumn *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self), NULL);
|
|
|
|
return self->factory;
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_set_factory:
|
|
* @self: a #GtkColumnViewColumn
|
|
* @factory: (allow-none) (transfer none): the factory to use or %NULL for none
|
|
*
|
|
* Sets the #GtkListItemFactory to use for populating list items for this
|
|
* column.
|
|
**/
|
|
void
|
|
gtk_column_view_column_set_factory (GtkColumnViewColumn *self,
|
|
GtkListItemFactory *factory)
|
|
{
|
|
g_return_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self));
|
|
g_return_if_fail (factory == NULL || GTK_LIST_ITEM_FACTORY (factory));
|
|
|
|
if (!g_set_object (&self->factory, factory))
|
|
return;
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_FACTORY]);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_set_title:
|
|
* @self: a #GtkColumnViewColumn
|
|
* @title: (nullable): Title to use for this column
|
|
*
|
|
* Sets the title of this column. The title is displayed in the header of a
|
|
* #GtkColumnView for this column and is therefor user-facing text that should
|
|
* be translated.
|
|
*/
|
|
void
|
|
gtk_column_view_column_set_title (GtkColumnViewColumn *self,
|
|
const char *title)
|
|
{
|
|
g_return_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self));
|
|
|
|
if (g_strcmp0 (self->title, title) == 0)
|
|
return;
|
|
|
|
g_free (self->title);
|
|
self->title = g_strdup (title);
|
|
|
|
if (self->header)
|
|
gtk_column_view_title_update (GTK_COLUMN_VIEW_TITLE (self->header));
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_TITLE]);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_get_title:
|
|
* @self: a #GtkColumnViewColumn
|
|
*
|
|
* Returns the title set with gtk_column_view_column_set_title().
|
|
*
|
|
* Returns: (nullable): The column's title
|
|
*/
|
|
const char *
|
|
gtk_column_view_column_get_title (GtkColumnViewColumn *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self), FALSE);
|
|
|
|
return self->title;
|
|
}
|
|
|
|
#if 0
|
|
static void
|
|
gtk_column_view_column_add_to_sorter (GtkColumnViewColumn *self)
|
|
{
|
|
if (self->view == NULL)
|
|
return;
|
|
|
|
gtk_column_view_sorter_add_column (GTK_COLUMN_VIEW_SORTER (gtk_column_view_get_sorter (self->view)), self);
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
gtk_column_view_column_remove_from_sorter (GtkColumnViewColumn *self)
|
|
{
|
|
if (self->view == NULL)
|
|
return;
|
|
|
|
gtk_column_view_sorter_remove_column (GTK_COLUMN_VIEW_SORTER (gtk_column_view_get_sorter (self->view)), self);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_set_sorter:
|
|
* @self: a #GtkColumnViewColumn
|
|
* @sorter: (nullable): the #GtkSorter to associate with @column
|
|
*
|
|
* Associates a sorter with the column.
|
|
*
|
|
* This sorter can be made active by clicking on the column
|
|
* header, or by calling gtk_column_view_sort_by_column().
|
|
*/
|
|
void
|
|
gtk_column_view_column_set_sorter (GtkColumnViewColumn *self,
|
|
GtkSorter *sorter)
|
|
{
|
|
g_return_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self));
|
|
g_return_if_fail (sorter == NULL || GTK_IS_SORTER (sorter));
|
|
|
|
if (!g_set_object (&self->sorter, sorter))
|
|
return;
|
|
|
|
gtk_column_view_column_remove_from_sorter (self);
|
|
|
|
if (self->header)
|
|
gtk_column_view_title_update (GTK_COLUMN_VIEW_TITLE (self->header));
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_SORTER]);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_get_sorter:
|
|
* @self: a #GtkColumnViewColumn
|
|
*
|
|
* Returns the sorter that is associated with the column.
|
|
*
|
|
* Returns: (transfer none): the #GtkSorter of @self
|
|
*/
|
|
GtkSorter *
|
|
gtk_column_view_column_get_sorter (GtkColumnViewColumn *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self), NULL);
|
|
|
|
return self->sorter;
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_notify_sort (GtkColumnViewColumn *self)
|
|
{
|
|
if (self->header)
|
|
gtk_column_view_title_update (GTK_COLUMN_VIEW_TITLE (self->header));
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_set_visible:
|
|
* @self: a #GtkColumnViewColumn
|
|
* @visible: whether this column should be visible
|
|
*
|
|
* Sets whether this column should be visible in views.
|
|
*/
|
|
void
|
|
gtk_column_view_column_set_visible (GtkColumnViewColumn *self,
|
|
gboolean visible)
|
|
{
|
|
GtkColumnViewCell *cell;
|
|
|
|
g_return_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self));
|
|
|
|
if (self->visible == visible)
|
|
return;
|
|
|
|
self->visible = visible;
|
|
|
|
self->minimum_size_request = -1;
|
|
self->natural_size_request = -1;
|
|
|
|
if (self->header)
|
|
gtk_widget_set_visible (GTK_WIDGET (self->header), visible);
|
|
|
|
for (cell = self->first_cell; cell; cell = gtk_column_view_cell_get_next (cell))
|
|
{
|
|
gtk_widget_set_visible (GTK_WIDGET (cell), visible);
|
|
}
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_VISIBLE]);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_get_visible:
|
|
* @self: a #GtkColumnViewColumn
|
|
*
|
|
* Returns whether this column is visible.
|
|
*
|
|
* Returns: %TRUE if this column is visible
|
|
*/
|
|
gboolean
|
|
gtk_column_view_column_get_visible (GtkColumnViewColumn *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self), TRUE);
|
|
|
|
return self->visible;
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_set_header_menu:
|
|
* @self: a #GtkColumnViewColumn
|
|
* @menu: (allow-none): a #GMenuModel, or %NULL
|
|
*
|
|
* Sets the menu model that is used to create the context menu
|
|
* for the column header.
|
|
*/
|
|
void
|
|
gtk_column_view_column_set_header_menu (GtkColumnViewColumn *self,
|
|
GMenuModel *menu)
|
|
{
|
|
g_return_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self));
|
|
g_return_if_fail (menu == NULL || G_IS_MENU_MODEL (menu));
|
|
|
|
if (!g_set_object (&self->menu, menu))
|
|
return;
|
|
|
|
if (self->header)
|
|
gtk_column_view_title_update (GTK_COLUMN_VIEW_TITLE (self->header));
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_HEADER_MENU]);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_get_header_menu:
|
|
* @self: a #GtkColumnViewColumn
|
|
*
|
|
* Gets the menu model that is used to create the context menu
|
|
* for the column header.
|
|
*
|
|
* Returns: (transfer none) (nullable): the #GMenuModel, or %NULL
|
|
*/
|
|
GMenuModel *
|
|
gtk_column_view_column_get_header_menu (GtkColumnViewColumn *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self), NULL);
|
|
|
|
return self->menu;
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_set_expand:
|
|
* @self: a #GtkColumnViewColumn
|
|
* @expand: %TRUE if this column should expand to fill available sace
|
|
*
|
|
* Sets the column to take available extra space.
|
|
*
|
|
* The extra space is shared equally amongst all columns that
|
|
* have the expand set to %TRUE.
|
|
*/
|
|
void
|
|
gtk_column_view_column_set_expand (GtkColumnViewColumn *self,
|
|
gboolean expand)
|
|
{
|
|
g_return_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self));
|
|
|
|
if (self->expand == expand)
|
|
return;
|
|
|
|
self->expand = expand;
|
|
|
|
if (self->visible && self->view)
|
|
gtk_widget_queue_resize (GTK_WIDGET (self->view));
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_EXPAND]);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_get_expand:
|
|
* @self: a #GtkColumnViewColumn
|
|
*
|
|
* Returns whether this column should expand.
|
|
*
|
|
* Returns: %TRUE if this column expands
|
|
*/
|
|
gboolean
|
|
gtk_column_view_column_get_expand (GtkColumnViewColumn *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self), TRUE);
|
|
|
|
return self->expand;
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_set_resizable:
|
|
* @self: a #GtkColumnViewColumn
|
|
* @resizable: whether this column should be resizable
|
|
*
|
|
* Sets whether this column should be resizable by dragging.
|
|
*/
|
|
void
|
|
gtk_column_view_column_set_resizable (GtkColumnViewColumn *self,
|
|
gboolean resizable)
|
|
{
|
|
g_return_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self));
|
|
|
|
if (self->resizable == resizable)
|
|
return;
|
|
|
|
self->resizable = resizable;
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_RESIZABLE]);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_get_resizable:
|
|
* @self: a #GtkColumnView
|
|
*
|
|
* Returns whether this column is resizable.
|
|
*
|
|
* Returns: %TRUE if this column is resizable
|
|
*/
|
|
gboolean
|
|
gtk_column_view_column_get_resizable (GtkColumnViewColumn *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self), TRUE);
|
|
|
|
return self->resizable;
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_set_fixed_width:
|
|
* @self: a #GtkColumnViewColumn
|
|
* @fixed_width: the new fixed width, or -1
|
|
*
|
|
* If @fixed_width is not -1, sets the fixed width of @column;
|
|
* otherwise unsets it.
|
|
*
|
|
* Setting a fixed width overrides the automatically calculated
|
|
* width. Interactive resizing also sets the “fixed-width” property.
|
|
*/
|
|
void
|
|
gtk_column_view_column_set_fixed_width (GtkColumnViewColumn *self,
|
|
int fixed_width)
|
|
{
|
|
GtkOverflow overflow;
|
|
|
|
g_return_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self));
|
|
g_return_if_fail (fixed_width >= -1);
|
|
|
|
if (self->fixed_width == fixed_width)
|
|
return;
|
|
|
|
self->fixed_width = fixed_width;
|
|
|
|
if (fixed_width > -1)
|
|
overflow = GTK_OVERFLOW_HIDDEN;
|
|
else
|
|
overflow = GTK_OVERFLOW_VISIBLE;
|
|
|
|
if (self->header &&
|
|
overflow != gtk_widget_get_overflow (GTK_WIDGET (self->header)))
|
|
{
|
|
GtkColumnViewCell *cell;
|
|
|
|
gtk_widget_set_overflow (GTK_WIDGET (self->header), overflow);
|
|
|
|
for (cell = self->first_cell; cell; cell = gtk_column_view_cell_get_next (cell))
|
|
gtk_widget_set_overflow (GTK_WIDGET (cell), overflow);
|
|
}
|
|
|
|
gtk_column_view_column_queue_resize (self);
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_FIXED_WIDTH]);
|
|
}
|
|
|
|
/**
|
|
* gtk_column_view_column_get_fixed_width:
|
|
* @self: a #GtkColumnViewColumn
|
|
*
|
|
* Gets the fixed width of the column.
|
|
*
|
|
* Returns: the fixed with of the column
|
|
*/
|
|
int
|
|
gtk_column_view_column_get_fixed_width (GtkColumnViewColumn *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_COLUMN_VIEW_COLUMN (self), -1);
|
|
|
|
return self->fixed_width;
|
|
}
|
|
|
|
GtkWidget *
|
|
gtk_column_view_column_get_header (GtkColumnViewColumn *self)
|
|
{
|
|
return self->header;
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_set_header_position (GtkColumnViewColumn *self,
|
|
int offset)
|
|
{
|
|
self->header_position = offset;
|
|
}
|
|
|
|
void
|
|
gtk_column_view_column_get_header_allocation (GtkColumnViewColumn *self,
|
|
int *offset,
|
|
int *size)
|
|
{
|
|
if (offset)
|
|
*offset = self->header_position;
|
|
|
|
if (size)
|
|
*size = self->allocation_size;
|
|
}
|