2019-06-26 03:35:10 +00:00
|
|
|
/* Constraints/Interactive
|
|
|
|
*
|
2020-06-07 02:05:52 +00:00
|
|
|
* Demonstrate how constraints can be updates during user interaction.
|
|
|
|
* The vertical edge between the buttons can be dragged with the mouse.
|
2019-06-26 03:35:10 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <glib/gi18n.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
|
|
|
|
G_DECLARE_FINAL_TYPE (InteractiveGrid, interactive_grid, INTERACTIVE, GRID, GtkWidget)
|
|
|
|
|
|
|
|
struct _InteractiveGrid
|
|
|
|
{
|
|
|
|
GtkWidget parent_instance;
|
|
|
|
|
|
|
|
GtkWidget *button1, *button2;
|
|
|
|
GtkWidget *button3;
|
|
|
|
GtkConstraintGuide *guide;
|
|
|
|
GtkConstraint *constraint;
|
|
|
|
};
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (InteractiveGrid, interactive_grid, GTK_TYPE_WIDGET)
|
|
|
|
|
|
|
|
static void
|
2020-05-05 20:57:28 +00:00
|
|
|
interactive_grid_dispose (GObject *object)
|
2019-06-26 03:35:10 +00:00
|
|
|
{
|
2020-05-05 20:57:28 +00:00
|
|
|
InteractiveGrid *self = INTERACTIVE_GRID (object);
|
2019-06-26 03:35:10 +00:00
|
|
|
|
2020-05-09 14:33:02 +00:00
|
|
|
g_clear_pointer (&self->button1, gtk_widget_unparent);
|
|
|
|
g_clear_pointer (&self->button2, gtk_widget_unparent);
|
|
|
|
g_clear_pointer (&self->button3, gtk_widget_unparent);
|
2019-06-26 03:35:10 +00:00
|
|
|
|
2020-05-05 20:57:28 +00:00
|
|
|
G_OBJECT_CLASS (interactive_grid_parent_class)->dispose (object);
|
2019-06-26 03:35:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
interactive_grid_class_init (InteractiveGridClass *klass)
|
|
|
|
{
|
2020-05-05 20:57:28 +00:00
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
2019-06-26 03:35:10 +00:00
|
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
|
2020-05-05 20:57:28 +00:00
|
|
|
object_class->dispose = interactive_grid_dispose;
|
2019-06-26 03:35:10 +00:00
|
|
|
|
|
|
|
gtk_widget_class_set_layout_manager_type (widget_class, GTK_TYPE_CONSTRAINT_LAYOUT);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
build_constraints (InteractiveGrid *self,
|
|
|
|
GtkConstraintLayout *manager)
|
|
|
|
{
|
|
|
|
self->guide = g_object_new (GTK_TYPE_CONSTRAINT_GUIDE, NULL);
|
|
|
|
gtk_constraint_layout_add_guide (manager, self->guide);
|
|
|
|
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new_constant (GTK_CONSTRAINT_TARGET (self->guide),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_WIDTH,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
0.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (NULL,
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_START,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
GTK_CONSTRAINT_TARGET (self->button1),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_START,
|
|
|
|
1.0,
|
|
|
|
-8.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (GTK_CONSTRAINT_TARGET (self->button1),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_END,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
GTK_CONSTRAINT_TARGET (self->guide),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_START,
|
|
|
|
1.0,
|
|
|
|
0.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (GTK_CONSTRAINT_TARGET (self->button2),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_START,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
GTK_CONSTRAINT_TARGET (self->guide),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_END,
|
|
|
|
1.0,
|
|
|
|
0.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (GTK_CONSTRAINT_TARGET (self->button2),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_END,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
NULL,
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_END,
|
|
|
|
1.0,
|
|
|
|
-8.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (NULL,
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_START,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
GTK_CONSTRAINT_TARGET (self->button3),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_START,
|
|
|
|
1.0,
|
|
|
|
-8.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (GTK_CONSTRAINT_TARGET (self->button3),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_END,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
GTK_CONSTRAINT_TARGET (self->guide),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_START,
|
|
|
|
1.0,
|
|
|
|
0.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (NULL,
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_TOP,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
GTK_CONSTRAINT_TARGET (self->button1),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_TOP,
|
|
|
|
1.0,
|
|
|
|
-8.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (GTK_CONSTRAINT_TARGET (self->button2),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_TOP,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
GTK_CONSTRAINT_TARGET (self->button1),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_BOTTOM,
|
|
|
|
1.0,
|
|
|
|
0.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (GTK_CONSTRAINT_TARGET (self->button3),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_TOP,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
GTK_CONSTRAINT_TARGET (self->button2),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_BOTTOM,
|
|
|
|
1.0,
|
|
|
|
0.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
gtk_constraint_layout_add_constraint (manager,
|
|
|
|
gtk_constraint_new (GTK_CONSTRAINT_TARGET (self->button3),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_BOTTOM,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
NULL,
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_BOTTOM,
|
|
|
|
1.0,
|
|
|
|
-8.0,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
drag_cb (GtkGestureDrag *drag,
|
|
|
|
double offset_x,
|
|
|
|
double offset_y,
|
|
|
|
InteractiveGrid *self)
|
|
|
|
{
|
|
|
|
GtkConstraintLayout *layout = GTK_CONSTRAINT_LAYOUT (gtk_widget_get_layout_manager (GTK_WIDGET (self)));
|
|
|
|
double x, y;
|
|
|
|
|
|
|
|
if (self->constraint)
|
|
|
|
{
|
|
|
|
gtk_constraint_layout_remove_constraint (layout, self->constraint);
|
|
|
|
g_clear_object (&self->constraint);
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_gesture_drag_get_start_point (drag, &x, &y);
|
|
|
|
self->constraint = gtk_constraint_new_constant (GTK_CONSTRAINT_TARGET (self->guide),
|
|
|
|
GTK_CONSTRAINT_ATTRIBUTE_LEFT,
|
|
|
|
GTK_CONSTRAINT_RELATION_EQ,
|
|
|
|
x + offset_x,
|
|
|
|
GTK_CONSTRAINT_STRENGTH_REQUIRED);
|
|
|
|
gtk_constraint_layout_add_constraint (layout, g_object_ref (self->constraint));
|
|
|
|
gtk_widget_queue_allocate (GTK_WIDGET (self));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
interactive_grid_init (InteractiveGrid *self)
|
|
|
|
{
|
|
|
|
GtkWidget *widget = GTK_WIDGET (self);
|
|
|
|
GtkGesture *drag;
|
|
|
|
|
|
|
|
self->button1 = gtk_button_new_with_label ("Child 1");
|
|
|
|
gtk_widget_set_parent (self->button1, widget);
|
|
|
|
gtk_widget_set_name (self->button1, "button1");
|
|
|
|
|
|
|
|
self->button2 = gtk_button_new_with_label ("Child 2");
|
|
|
|
gtk_widget_set_parent (self->button2, widget);
|
|
|
|
gtk_widget_set_name (self->button2, "button2");
|
|
|
|
|
|
|
|
self->button3 = gtk_button_new_with_label ("Child 3");
|
|
|
|
gtk_widget_set_parent (self->button3, widget);
|
|
|
|
gtk_widget_set_name (self->button3, "button3");
|
|
|
|
|
|
|
|
GtkLayoutManager *manager = gtk_widget_get_layout_manager (GTK_WIDGET (self));
|
|
|
|
build_constraints (self, GTK_CONSTRAINT_LAYOUT (manager));
|
|
|
|
|
|
|
|
drag = gtk_gesture_drag_new ();
|
|
|
|
g_signal_connect (drag, "drag-update", G_CALLBACK (drag_cb), self);
|
|
|
|
gtk_widget_add_controller (GTK_WIDGET (self), GTK_EVENT_CONTROLLER (drag));
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
do_constraints2 (GtkWidget *do_widget)
|
|
|
|
{
|
|
|
|
static GtkWidget *window;
|
|
|
|
|
|
|
|
if (!window)
|
|
|
|
{
|
|
|
|
GtkWidget *header, *box, *grid, *button;
|
|
|
|
|
2020-02-14 19:55:36 +00:00
|
|
|
window = gtk_window_new ();
|
2019-06-26 03:35:10 +00:00
|
|
|
gtk_window_set_display (GTK_WINDOW (window), gtk_widget_get_display (do_widget));
|
2020-04-30 23:04:08 +00:00
|
|
|
gtk_window_set_title (GTK_WINDOW (window), "Constraints");
|
2019-06-26 03:35:10 +00:00
|
|
|
|
|
|
|
header = gtk_header_bar_new ();
|
|
|
|
gtk_header_bar_set_show_title_buttons (GTK_HEADER_BAR (header), FALSE);
|
|
|
|
gtk_window_set_titlebar (GTK_WINDOW (window), header);
|
2020-05-09 16:03:11 +00:00
|
|
|
g_object_add_weak_pointer (G_OBJECT (window), (gpointer *)&window);
|
2019-06-26 03:35:10 +00:00
|
|
|
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12);
|
2020-05-02 21:26:54 +00:00
|
|
|
gtk_window_set_child (GTK_WINDOW (window), box);
|
2019-06-26 03:35:10 +00:00
|
|
|
|
|
|
|
grid = g_object_new (interactive_grid_get_type (), NULL);
|
|
|
|
gtk_widget_set_hexpand (grid, TRUE);
|
|
|
|
gtk_widget_set_vexpand (grid, TRUE);
|
2020-05-09 12:26:52 +00:00
|
|
|
gtk_box_append (GTK_BOX (box), grid);
|
2019-06-26 03:35:10 +00:00
|
|
|
|
|
|
|
button = gtk_button_new_with_label ("Close");
|
2020-05-09 12:26:52 +00:00
|
|
|
gtk_box_append (GTK_BOX (box), button);
|
2019-06-26 03:35:10 +00:00
|
|
|
gtk_widget_set_hexpand (grid, TRUE);
|
|
|
|
g_signal_connect_swapped (button, "clicked",
|
2020-05-09 14:26:22 +00:00
|
|
|
G_CALLBACK (gtk_window_destroy), window);
|
2019-06-26 03:35:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!gtk_widget_get_visible (window))
|
|
|
|
gtk_widget_show (window);
|
|
|
|
else
|
2020-05-09 14:26:22 +00:00
|
|
|
gtk_window_destroy (GTK_WINDOW (window));
|
2019-06-26 03:35:10 +00:00
|
|
|
|
|
|
|
return window;
|
|
|
|
}
|