Merge branch 'columnview-resizing' into 'master'

columnview: Implement interactive resizing

See merge request GNOME/gtk!2004
This commit is contained in:
Matthias Clasen 2020-06-03 20:04:17 +00:00
commit f13a596d15
9 changed files with 358 additions and 1 deletions

View File

@ -104,6 +104,7 @@
<child> <child>
<object class="GtkColumnViewColumn"> <object class="GtkColumnViewColumn">
<property name="title">Value</property> <property name="title">Value</property>
<property name="resizable">1</property>
<property name="header-menu">header_menu</property> <property name="header-menu">header_menu</property>
<property name="factory"> <property name="factory">
<object class="GtkBuilderListItemFactory"> <object class="GtkBuilderListItemFactory">
@ -131,6 +132,7 @@
<child> <child>
<object class="GtkColumnViewColumn" id="type_column"> <object class="GtkColumnViewColumn" id="type_column">
<property name="title">Type</property> <property name="title">Type</property>
<property name="resizable">1</property>
<property name="header-menu">header_menu</property> <property name="header-menu">header_menu</property>
<property name="factory"> <property name="factory">
<object class="GtkBuilderListItemFactory"> <object class="GtkBuilderListItemFactory">
@ -158,6 +160,7 @@
<child> <child>
<object class="GtkColumnViewColumn" id="default_column"> <object class="GtkColumnViewColumn" id="default_column">
<property name="title">Default</property> <property name="title">Default</property>
<property name="resizable">1</property>
<property name="header-menu">header_menu</property> <property name="header-menu">header_menu</property>
<property name="factory"> <property name="factory">
<object class="GtkBuilderListItemFactory"> <object class="GtkBuilderListItemFactory">
@ -185,6 +188,7 @@
<child> <child>
<object class="GtkColumnViewColumn" id="summary_column"> <object class="GtkColumnViewColumn" id="summary_column">
<property name="title">Summary</property> <property name="title">Summary</property>
<property name="resizable">1</property>
<property name="visible">0</property> <property name="visible">0</property>
<property name="header-menu">header_menu</property> <property name="header-menu">header_menu</property>
<property name="factory"> <property name="factory">
@ -216,6 +220,7 @@
<child> <child>
<object class="GtkColumnViewColumn" id="description_column"> <object class="GtkColumnViewColumn" id="description_column">
<property name="title">Description</property> <property name="title">Description</property>
<property name="resizable">1</property>
<property name="visible">0</property> <property name="visible">0</property>
<property name="header-menu">header_menu</property> <property name="header-menu">header_menu</property>
<property name="factory"> <property name="factory">

View File

@ -533,8 +533,12 @@ gtk_column_view_column_set_sorter
gtk_column_view_column_get_sorter gtk_column_view_column_get_sorter
gtk_column_view_column_set_visible gtk_column_view_column_set_visible
gtk_column_view_column_get_visible gtk_column_view_column_get_visible
gtk_column_view_column_set_resizable
gtk_column_view_column_get_resizable
gtk_column_view_column_set_header_menu gtk_column_view_column_set_header_menu
gtk_column_view_column_get_header_menu gtk_column_view_column_get_header_menu
gtk_column_view_column_set_fixed_width
gtk_column_view_column_get_fixed_width
<SUBSECTION Standard> <SUBSECTION Standard>
GTK_COLUMN_VIEW_COLUMN GTK_COLUMN_VIEW_COLUMN

View File

@ -36,6 +36,8 @@
#include "gtkwidgetprivate.h" #include "gtkwidgetprivate.h"
#include "gtksizerequest.h" #include "gtksizerequest.h"
#include "gtkadjustment.h" #include "gtkadjustment.h"
#include "gtkgesturedrag.h"
#include "gtkeventcontrollermotion.h"
/** /**
* SECTION:gtkcolumnview * SECTION:gtkcolumnview
@ -67,6 +69,10 @@ struct _GtkColumnView
GtkSorter *sorter; GtkSorter *sorter;
GtkAdjustment *hadjustment; GtkAdjustment *hadjustment;
gboolean in_column_resize;
int drag_pos;
int drag_x;
}; };
struct _GtkColumnViewClass struct _GtkColumnViewClass
@ -551,9 +557,144 @@ gtk_column_view_class_init (GtkColumnViewClass *klass)
gtk_widget_class_set_css_name (widget_class, I_("treeview")); gtk_widget_class_set_css_name (widget_class, I_("treeview"));
} }
#define DRAG_WIDTH 6
static gboolean
gtk_column_view_in_resize_rect (GtkColumnView *self,
GtkColumnViewColumn *column,
double x,
double y)
{
GtkWidget *header;
graphene_rect_t rect;
header = gtk_column_view_column_get_header (column);
if (!gtk_widget_compute_bounds (header, self->header, &rect))
return FALSE;
rect.origin.x += rect.size.width - DRAG_WIDTH / 2;
rect.size.width = DRAG_WIDTH;
return graphene_rect_contains_point (&rect, &(graphene_point_t) { x, y});
}
static void
header_drag_begin (GtkGestureDrag *gesture,
double start_x,
double start_y,
GtkColumnView *self)
{
int i, n;
n = g_list_model_get_n_items (G_LIST_MODEL (self->columns));
for (i = 0; !self->in_column_resize && i < n; i++)
{
GtkColumnViewColumn *column = g_list_model_get_item (G_LIST_MODEL (self->columns), i);
if (!gtk_column_view_column_get_visible (column))
{
g_object_unref (column);
continue;
}
if (i + 1 < n &&
gtk_column_view_column_get_resizable (column) &&
gtk_column_view_in_resize_rect (self, column, start_x, start_y))
{
int size;
gtk_gesture_set_state (GTK_GESTURE (gesture), GTK_EVENT_SEQUENCE_CLAIMED);
gtk_column_view_column_get_allocation (column, NULL, &size);
gtk_column_view_column_set_fixed_width (column, size);
self->drag_pos = i;
self->drag_x = start_x - size;
self->in_column_resize = TRUE;
g_object_unref (column);
break;
}
g_object_unref (column);
}
}
static void
header_drag_end (GtkGestureDrag *gesture,
double offset_x,
double offset_y,
GtkColumnView *self)
{
self->in_column_resize = FALSE;
}
static void
update_column_resize (GtkColumnView *self,
double x)
{
GtkColumnViewColumn *column;
column = g_list_model_get_item (G_LIST_MODEL (self->columns), self->drag_pos);
gtk_column_view_column_set_fixed_width (column, MAX (x - self->drag_x, 0));
g_object_unref (column);
}
static void
header_drag_update (GtkGestureDrag *gesture,
double offset_x,
double offset_y,
GtkColumnView *self)
{
double start_x, x;
gtk_gesture_drag_get_start_point (gesture, &start_x, NULL);
x = start_x + offset_x;
if (self->in_column_resize)
update_column_resize (self, x);
}
static void
header_motion (GtkEventControllerMotion *controller,
double x,
double y,
GtkColumnView *self)
{
gboolean cursor_set = FALSE;
int i, n;
n = g_list_model_get_n_items (G_LIST_MODEL (self->columns));
for (i = 0; i < n; i++)
{
GtkColumnViewColumn *column = g_list_model_get_item (G_LIST_MODEL (self->columns), i);
if (!gtk_column_view_column_get_visible (column))
{
g_object_unref (column);
continue;
}
if (i + 1 < n &&
gtk_column_view_column_get_resizable (column) &&
gtk_column_view_in_resize_rect (self, column, x, y))
{
gtk_widget_set_cursor_from_name (self->header, "col-resize");
cursor_set = TRUE;
}
g_object_unref (column);
}
if (!cursor_set)
gtk_widget_set_cursor (self->header, NULL);
}
static void static void
gtk_column_view_init (GtkColumnView *self) gtk_column_view_init (GtkColumnView *self)
{ {
GtkEventController *controller;
self->columns = g_list_store_new (GTK_TYPE_COLUMN_VIEW_COLUMN); self->columns = g_list_store_new (GTK_TYPE_COLUMN_VIEW_COLUMN);
self->header = gtk_list_item_widget_new (NULL, "header"); self->header = gtk_list_item_widget_new (NULL, "header");
@ -561,6 +702,17 @@ gtk_column_view_init (GtkColumnView *self)
gtk_widget_set_layout_manager (self->header, gtk_column_view_layout_new (self)); gtk_widget_set_layout_manager (self->header, gtk_column_view_layout_new (self));
gtk_widget_set_parent (self->header, GTK_WIDGET (self)); gtk_widget_set_parent (self->header, GTK_WIDGET (self));
controller = GTK_EVENT_CONTROLLER (gtk_gesture_drag_new ());
g_signal_connect (controller, "drag-begin", G_CALLBACK (header_drag_begin), self);
g_signal_connect (controller, "drag-update", G_CALLBACK (header_drag_update), self);
g_signal_connect (controller, "drag-end", G_CALLBACK (header_drag_end), self);
gtk_event_controller_set_propagation_phase (controller, GTK_PHASE_CAPTURE);
gtk_widget_add_controller (self->header, controller);
controller = gtk_event_controller_motion_new ();
g_signal_connect (controller, "motion", G_CALLBACK (header_motion), self);
gtk_widget_add_controller (self->header, controller);
self->sorter = gtk_column_view_sorter_new (); self->sorter = gtk_column_view_sorter_new ();
self->factory = gtk_column_list_item_factory_new (self); self->factory = gtk_column_list_item_factory_new (self);
self->listview = GTK_LIST_VIEW (gtk_list_view_new_with_factory ( self->listview = GTK_LIST_VIEW (gtk_list_view_new_with_factory (
@ -826,6 +978,12 @@ gtk_column_view_get_header_widget (GtkColumnView *self)
return GTK_LIST_ITEM_WIDGET (self->header); return GTK_LIST_ITEM_WIDGET (self->header);
} }
GtkListView *
gtk_column_view_get_list_view (GtkColumnView *self)
{
return GTK_LIST_VIEW (self->listview);
}
/** /**
* gtk_column_view_get_sorter: * gtk_column_view_get_sorter:
* @self: a #GtkColumnView * @self: a #GtkColumnView

View File

@ -53,10 +53,18 @@ gtk_column_view_cell_measure (GtkWidget *widget,
int *minimum_baseline, int *minimum_baseline,
int *natural_baseline) int *natural_baseline)
{ {
GtkColumnViewCell *cell = GTK_COLUMN_VIEW_CELL (widget);
GtkWidget *child = gtk_widget_get_first_child (widget); GtkWidget *child = gtk_widget_get_first_child (widget);
if (child) if (child)
gtk_widget_measure (child, orientation, for_size, minimum, natural, minimum_baseline, natural_baseline); gtk_widget_measure (child, orientation, for_size, minimum, natural, minimum_baseline, natural_baseline);
if (orientation == GTK_ORIENTATION_HORIZONTAL)
{
int fixed_width = gtk_column_view_column_get_fixed_width (cell->column);
if (fixed_width > -1)
*minimum = *natural = fixed_width;
}
} }
static void static void

View File

@ -61,7 +61,10 @@ struct _GtkColumnViewColumn
int allocation_offset; int allocation_offset;
int allocation_size; int allocation_size;
gboolean visible; int fixed_width;
guint visible : 1;
guint resizable : 1;
GMenuModel *menu; GMenuModel *menu;
@ -83,6 +86,8 @@ enum
PROP_SORTER, PROP_SORTER,
PROP_VISIBLE, PROP_VISIBLE,
PROP_HEADER_MENU, PROP_HEADER_MENU,
PROP_RESIZABLE,
PROP_FIXED_WIDTH,
N_PROPS N_PROPS
}; };
@ -141,6 +146,14 @@ gtk_column_view_column_get_property (GObject *object,
g_value_set_object (value, self->menu); g_value_set_object (value, self->menu);
break; break;
case PROP_RESIZABLE:
g_value_set_boolean (value, self->resizable);
break;
case PROP_FIXED_WIDTH:
g_value_set_int (value, self->fixed_width);
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break; break;
@ -177,6 +190,14 @@ gtk_column_view_column_set_property (GObject *object,
gtk_column_view_column_set_header_menu (self, g_value_get_object (value)); gtk_column_view_column_set_header_menu (self, g_value_get_object (value));
break; break;
case PROP_RESIZABLE:
gtk_column_view_column_set_resizable (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: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break; break;
@ -264,6 +285,31 @@ gtk_column_view_column_class_init (GtkColumnViewColumnClass *klass)
G_TYPE_MENU_MODEL, G_TYPE_MENU_MODEL,
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS); 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: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); g_object_class_install_properties (gobject_class, N_PROPS, properties);
} }
@ -273,6 +319,8 @@ gtk_column_view_column_init (GtkColumnViewColumn *self)
self->minimum_size_request = -1; self->minimum_size_request = -1;
self->natural_size_request = -1; self->natural_size_request = -1;
self->visible = TRUE; self->visible = TRUE;
self->resizable = FALSE;
self->fixed_width = -1;
} }
/** /**
@ -382,6 +430,12 @@ gtk_column_view_column_measure (GtkColumnViewColumn *self,
int *minimum, int *minimum,
int *natural) 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) if (self->minimum_size_request < 0)
{ {
GtkColumnViewCell *cell; GtkColumnViewCell *cell;
@ -788,3 +842,108 @@ gtk_column_view_column_get_header_menu (GtkColumnViewColumn *self)
return self->menu; return self->menu;
} }
/**
* 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;
}

View File

@ -79,11 +79,23 @@ GDK_AVAILABLE_IN_ALL
gboolean gtk_column_view_column_get_visible (GtkColumnViewColumn *self); gboolean gtk_column_view_column_get_visible (GtkColumnViewColumn *self);
GDK_AVAILABLE_IN_ALL GDK_AVAILABLE_IN_ALL
void gtk_column_view_column_set_header_menu (GtkColumnViewColumn *self, void gtk_column_view_column_set_header_menu (GtkColumnViewColumn *self,
GMenuModel *menu); GMenuModel *menu);
GDK_AVAILABLE_IN_ALL GDK_AVAILABLE_IN_ALL
GMenuModel * gtk_column_view_column_get_header_menu (GtkColumnViewColumn *self); GMenuModel * gtk_column_view_column_get_header_menu (GtkColumnViewColumn *self);
void gtk_column_view_column_set_fixed_width (GtkColumnViewColumn *self,
int fixed_width);
GDK_AVAILABLE_IN_ALL
int gtk_column_view_column_get_fixed_width (GtkColumnViewColumn *self);
GDK_AVAILABLE_IN_ALL
void gtk_column_view_column_set_resizable (GtkColumnViewColumn *self,
gboolean resizable);
GDK_AVAILABLE_IN_ALL
gboolean gtk_column_view_column_get_resizable (GtkColumnViewColumn *self);
G_END_DECLS G_END_DECLS
#endif /* __GTK_COLUMN_VIEW_COLUMN_H__ */ #endif /* __GTK_COLUMN_VIEW_COLUMN_H__ */

View File

@ -33,6 +33,7 @@ void gtk_column_view_column_add_cell (GtkColu
void gtk_column_view_column_remove_cell (GtkColumnViewColumn *self, void gtk_column_view_column_remove_cell (GtkColumnViewColumn *self,
GtkColumnViewCell *cell); GtkColumnViewCell *cell);
GtkColumnViewCell * gtk_column_view_column_get_first_cell (GtkColumnViewColumn *self); GtkColumnViewCell * gtk_column_view_column_get_first_cell (GtkColumnViewColumn *self);
GtkWidget * gtk_column_view_column_get_header (GtkColumnViewColumn *self);
void gtk_column_view_column_queue_resize (GtkColumnViewColumn *self); void gtk_column_view_column_queue_resize (GtkColumnViewColumn *self);
void gtk_column_view_column_measure (GtkColumnViewColumn *self, void gtk_column_view_column_measure (GtkColumnViewColumn *self,

View File

@ -21,11 +21,13 @@
#define __GTK_COLUMN_VIEW_PRIVATE_H__ #define __GTK_COLUMN_VIEW_PRIVATE_H__
#include "gtk/gtkcolumnview.h" #include "gtk/gtkcolumnview.h"
#include "gtk/gtklistview.h"
#include "gtk/gtkcolumnviewsorterprivate.h" #include "gtk/gtkcolumnviewsorterprivate.h"
#include "gtk/gtklistitemwidgetprivate.h" #include "gtk/gtklistitemwidgetprivate.h"
GtkListItemWidget * gtk_column_view_get_header_widget (GtkColumnView *self); GtkListItemWidget * gtk_column_view_get_header_widget (GtkColumnView *self);
GtkListView * gtk_column_view_get_list_view (GtkColumnView *self);
void gtk_column_view_measure_across (GtkColumnView *self, void gtk_column_view_measure_across (GtkColumnView *self,
int *minimum, int *minimum,

View File

@ -61,10 +61,18 @@ gtk_column_view_title_measure (GtkWidget *widget,
int *minimum_baseline, int *minimum_baseline,
int *natural_baseline) int *natural_baseline)
{ {
GtkColumnViewTitle *self = GTK_COLUMN_VIEW_TITLE (widget);
GtkWidget *child = gtk_widget_get_first_child (widget); GtkWidget *child = gtk_widget_get_first_child (widget);
if (child) if (child)
gtk_widget_measure (child, orientation, for_size, minimum, natural, minimum_baseline, natural_baseline); gtk_widget_measure (child, orientation, for_size, minimum, natural, minimum_baseline, natural_baseline);
if (orientation == GTK_ORIENTATION_HORIZONTAL)
{
int fixed_width = gtk_column_view_column_get_fixed_width (self->column);
if (fixed_width > -1)
*minimum = *natural = fixed_width;
}
} }
static void static void