2010-09-27 14:44:31 +00:00
|
|
|
/* GTK - The GIMP Toolkit
|
|
|
|
* Copyright (C) 2010 Red Hat, Inc.
|
|
|
|
* Author: Matthias Clasen
|
|
|
|
*
|
|
|
|
* 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 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
|
2012-02-27 13:01:10 +00:00
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
2010-09-27 14:44:31 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2011-02-17 16:51:16 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
#include "gtkgrid.h"
|
|
|
|
|
2011-01-30 08:12:49 +00:00
|
|
|
#include "gtkorientableprivate.h"
|
2010-09-27 14:44:31 +00:00
|
|
|
#include "gtksizerequest.h"
|
|
|
|
#include "gtkprivate.h"
|
|
|
|
#include "gtkintl.h"
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SECTION:gtkgrid
|
|
|
|
* @Short_description: Pack widgets in a rows and columns
|
|
|
|
* @Title: GtkGrid
|
|
|
|
* @See_also: #GtkTable, #GtkHBox, #GtkVBox
|
|
|
|
*
|
|
|
|
* GtkGrid is a container which arranges its child widgets in
|
|
|
|
* rows and columns. It is a very similar to #GtkTable and #GtkBox,
|
2010-11-03 02:20:20 +00:00
|
|
|
* but it consistently uses #GtkWidget's #GtkWidget:margin and #GtkWidget:expand
|
|
|
|
* properties instead of custom child properties, and it fully supports
|
|
|
|
* <link linkend="geometry-management">height-for-width geometry management</link>.
|
2010-09-27 14:44:31 +00:00
|
|
|
*
|
|
|
|
* Children are added using gtk_grid_attach(). They can span multiple
|
|
|
|
* rows or columns. It is also possible to add a child next to an
|
2012-04-28 06:19:45 +00:00
|
|
|
* existing child, using gtk_grid_attach_next_to(). The behaviour of
|
|
|
|
* GtkGrid when several children occupy the same grid cell is undefined.
|
2010-09-27 14:44:31 +00:00
|
|
|
*
|
|
|
|
* GtkGrid can be used like a #GtkBox by just using gtk_container_add(),
|
|
|
|
* which will place children next to each other in the direction determined
|
2010-11-03 02:20:20 +00:00
|
|
|
* by the #GtkOrientable:orientation property.
|
2010-09-27 14:44:31 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
typedef struct _GtkGridChild GtkGridChild;
|
|
|
|
typedef struct _GtkGridChildAttach GtkGridChildAttach;
|
2013-03-21 10:09:43 +00:00
|
|
|
typedef struct _GtkGridRowProperties GtkGridRowProperties;
|
2010-09-27 14:44:31 +00:00
|
|
|
typedef struct _GtkGridLine GtkGridLine;
|
|
|
|
typedef struct _GtkGridLines GtkGridLines;
|
|
|
|
typedef struct _GtkGridLineData GtkGridLineData;
|
|
|
|
typedef struct _GtkGridRequest GtkGridRequest;
|
|
|
|
|
|
|
|
struct _GtkGridChildAttach
|
|
|
|
{
|
|
|
|
gint pos;
|
|
|
|
gint span;
|
|
|
|
};
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
struct _GtkGridRowProperties
|
|
|
|
{
|
|
|
|
gint row;
|
|
|
|
GtkBaselinePosition baseline_position;
|
|
|
|
};
|
|
|
|
|
|
|
|
static const GtkGridRowProperties gtk_grid_row_properties_default = {
|
|
|
|
0,
|
|
|
|
GTK_BASELINE_POSITION_CENTER
|
|
|
|
};
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
struct _GtkGridChild
|
|
|
|
{
|
|
|
|
GtkWidget *widget;
|
|
|
|
GtkGridChildAttach attach[2];
|
|
|
|
};
|
|
|
|
|
|
|
|
#define CHILD_LEFT(child) ((child)->attach[GTK_ORIENTATION_HORIZONTAL].pos)
|
|
|
|
#define CHILD_WIDTH(child) ((child)->attach[GTK_ORIENTATION_HORIZONTAL].span)
|
|
|
|
#define CHILD_TOP(child) ((child)->attach[GTK_ORIENTATION_VERTICAL].pos)
|
|
|
|
#define CHILD_HEIGHT(child) ((child)->attach[GTK_ORIENTATION_VERTICAL].span)
|
|
|
|
|
|
|
|
/* A GtkGridLineData struct contains row/column specific parts
|
|
|
|
* of the grid.
|
|
|
|
*/
|
|
|
|
struct _GtkGridLineData
|
|
|
|
{
|
|
|
|
gint16 spacing;
|
|
|
|
guint homogeneous : 1;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _GtkGridPrivate
|
|
|
|
{
|
|
|
|
GList *children;
|
2013-03-21 10:09:43 +00:00
|
|
|
GList *row_properties;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
GtkOrientation orientation;
|
2013-03-21 10:09:43 +00:00
|
|
|
gint baseline_row;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
GtkGridLineData linedata[2];
|
|
|
|
};
|
|
|
|
|
|
|
|
#define ROWS(priv) (&(priv)->linedata[GTK_ORIENTATION_HORIZONTAL])
|
|
|
|
#define COLUMNS(priv) (&(priv)->linedata[GTK_ORIENTATION_VERTICAL])
|
|
|
|
|
|
|
|
/* A GtkGridLine struct represents a single row or column
|
|
|
|
* during size requests
|
|
|
|
*/
|
|
|
|
struct _GtkGridLine
|
|
|
|
{
|
|
|
|
gint minimum;
|
|
|
|
gint natural;
|
2013-03-21 10:09:43 +00:00
|
|
|
gint minimum_above;
|
|
|
|
gint minimum_below;
|
|
|
|
gint natural_above;
|
|
|
|
gint natural_below;
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
gint position;
|
|
|
|
gint allocation;
|
2013-03-21 10:09:43 +00:00
|
|
|
gint allocated_baseline;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
guint need_expand : 1;
|
|
|
|
guint expand : 1;
|
|
|
|
guint empty : 1;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _GtkGridLines
|
|
|
|
{
|
|
|
|
GtkGridLine *lines;
|
|
|
|
gint min, max;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _GtkGridRequest
|
|
|
|
{
|
|
|
|
GtkGrid *grid;
|
|
|
|
GtkGridLines lines[2];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
|
|
|
PROP_ORIENTATION,
|
|
|
|
PROP_ROW_SPACING,
|
|
|
|
PROP_COLUMN_SPACING,
|
|
|
|
PROP_ROW_HOMOGENEOUS,
|
2013-03-21 10:09:43 +00:00
|
|
|
PROP_COLUMN_HOMOGENEOUS,
|
|
|
|
PROP_BASELINE_ROW
|
2010-09-27 14:44:31 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
CHILD_PROP_0,
|
|
|
|
CHILD_PROP_LEFT_ATTACH,
|
|
|
|
CHILD_PROP_TOP_ATTACH,
|
|
|
|
CHILD_PROP_WIDTH,
|
|
|
|
CHILD_PROP_HEIGHT
|
|
|
|
};
|
|
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GtkGrid, gtk_grid, GTK_TYPE_CONTAINER,
|
2013-06-27 19:02:52 +00:00
|
|
|
G_ADD_PRIVATE (GtkGrid)
|
2010-09-27 14:44:31 +00:00
|
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_ORIENTABLE, NULL))
|
|
|
|
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
static void gtk_grid_row_properties_free (GtkGridRowProperties *props);
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
static void
|
|
|
|
gtk_grid_get_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (object);
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_ORIENTATION:
|
|
|
|
g_value_set_enum (value, priv->orientation);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_ROW_SPACING:
|
2011-08-30 13:11:02 +00:00
|
|
|
g_value_set_int (value, COLUMNS (priv)->spacing);
|
2010-09-27 14:44:31 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_COLUMN_SPACING:
|
2011-08-30 13:11:02 +00:00
|
|
|
g_value_set_int (value, ROWS (priv)->spacing);
|
2010-09-27 14:44:31 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_ROW_HOMOGENEOUS:
|
2011-04-20 14:56:03 +00:00
|
|
|
g_value_set_boolean (value, COLUMNS (priv)->homogeneous);
|
2010-09-27 14:44:31 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_COLUMN_HOMOGENEOUS:
|
2011-04-20 14:56:03 +00:00
|
|
|
g_value_set_boolean (value, ROWS (priv)->homogeneous);
|
2010-09-27 14:44:31 +00:00
|
|
|
break;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
case PROP_BASELINE_ROW:
|
|
|
|
g_value_set_int (value, priv->baseline_row);
|
|
|
|
break;
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_set_orientation (GtkGrid *grid,
|
|
|
|
GtkOrientation orientation)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
|
|
|
|
if (priv->orientation != orientation)
|
|
|
|
{
|
|
|
|
priv->orientation = orientation;
|
2011-01-12 21:55:55 +00:00
|
|
|
_gtk_orientable_set_style_classes (GTK_ORIENTABLE (grid));
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
g_object_notify (G_OBJECT (grid), "orientation");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_set_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_ORIENTATION:
|
|
|
|
gtk_grid_set_orientation (grid, g_value_get_enum (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_ROW_SPACING:
|
|
|
|
gtk_grid_set_row_spacing (grid, g_value_get_int (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_COLUMN_SPACING:
|
|
|
|
gtk_grid_set_column_spacing (grid, g_value_get_int (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_ROW_HOMOGENEOUS:
|
|
|
|
gtk_grid_set_row_homogeneous (grid, g_value_get_boolean (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_COLUMN_HOMOGENEOUS:
|
|
|
|
gtk_grid_set_column_homogeneous (grid, g_value_get_boolean (value));
|
|
|
|
break;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
case PROP_BASELINE_ROW:
|
|
|
|
gtk_grid_set_baseline_row (grid, g_value_get_int (value));
|
|
|
|
break;
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkGridChild *
|
|
|
|
find_grid_child (GtkGrid *grid,
|
|
|
|
GtkWidget *widget)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GList *list;
|
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
if (child->widget == widget)
|
|
|
|
return child;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_get_child_property (GtkContainer *container,
|
|
|
|
GtkWidget *child,
|
|
|
|
guint property_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (container);
|
|
|
|
GtkGridChild *grid_child;
|
|
|
|
|
|
|
|
grid_child = find_grid_child (grid, child);
|
|
|
|
|
|
|
|
if (grid_child == NULL)
|
|
|
|
{
|
|
|
|
GTK_CONTAINER_WARN_INVALID_CHILD_PROPERTY_ID (container, property_id, pspec);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
|
|
|
case CHILD_PROP_LEFT_ATTACH:
|
|
|
|
g_value_set_int (value, CHILD_LEFT (grid_child));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHILD_PROP_TOP_ATTACH:
|
|
|
|
g_value_set_int (value, CHILD_TOP (grid_child));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHILD_PROP_WIDTH:
|
|
|
|
g_value_set_int (value, CHILD_WIDTH (grid_child));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHILD_PROP_HEIGHT:
|
|
|
|
g_value_set_int (value, CHILD_HEIGHT (grid_child));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
GTK_CONTAINER_WARN_INVALID_CHILD_PROPERTY_ID (container, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_set_child_property (GtkContainer *container,
|
|
|
|
GtkWidget *child,
|
|
|
|
guint property_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (container);
|
|
|
|
GtkGridChild *grid_child;
|
|
|
|
|
|
|
|
grid_child = find_grid_child (grid, child);
|
|
|
|
|
|
|
|
if (grid_child == NULL)
|
|
|
|
{
|
|
|
|
GTK_CONTAINER_WARN_INVALID_CHILD_PROPERTY_ID (container, property_id, pspec);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
|
|
|
case CHILD_PROP_LEFT_ATTACH:
|
|
|
|
CHILD_LEFT (grid_child) = g_value_get_int (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHILD_PROP_TOP_ATTACH:
|
|
|
|
CHILD_TOP (grid_child) = g_value_get_int (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHILD_PROP_WIDTH:
|
|
|
|
CHILD_WIDTH (grid_child) = g_value_get_int (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CHILD_PROP_HEIGHT:
|
|
|
|
CHILD_HEIGHT (grid_child) = g_value_get_int (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
GTK_CONTAINER_WARN_INVALID_CHILD_PROPERTY_ID (container, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gtk_widget_get_visible (child) &&
|
|
|
|
gtk_widget_get_visible (GTK_WIDGET (grid)))
|
|
|
|
gtk_widget_queue_resize (child);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_init (GtkGrid *grid)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
|
2013-06-27 19:02:52 +00:00
|
|
|
grid->priv = gtk_grid_get_instance_private (grid);
|
2010-09-27 14:44:31 +00:00
|
|
|
priv = grid->priv;
|
|
|
|
|
|
|
|
gtk_widget_set_has_window (GTK_WIDGET (grid), FALSE);
|
|
|
|
gtk_widget_set_redraw_on_allocate (GTK_WIDGET (grid), FALSE);
|
|
|
|
|
|
|
|
priv->children = NULL;
|
|
|
|
priv->orientation = GTK_ORIENTATION_HORIZONTAL;
|
2013-03-21 10:09:43 +00:00
|
|
|
priv->baseline_row = 0;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
priv->linedata[0].spacing = 0;
|
|
|
|
priv->linedata[1].spacing = 0;
|
|
|
|
|
|
|
|
priv->linedata[0].homogeneous = FALSE;
|
|
|
|
priv->linedata[1].homogeneous = FALSE;
|
|
|
|
}
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
static void
|
|
|
|
gtk_grid_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (object);
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
|
|
|
|
g_list_free_full (priv->row_properties, (GDestroyNotify)gtk_grid_row_properties_free);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (gtk_grid_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
static void
|
2011-09-03 00:04:06 +00:00
|
|
|
grid_attach (GtkGrid *grid,
|
|
|
|
GtkWidget *widget,
|
|
|
|
gint left,
|
|
|
|
gint top,
|
|
|
|
gint width,
|
|
|
|
gint height)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
GtkGridChild *child;
|
|
|
|
|
|
|
|
child = g_slice_new (GtkGridChild);
|
|
|
|
child->widget = widget;
|
|
|
|
CHILD_LEFT (child) = left;
|
|
|
|
CHILD_TOP (child) = top;
|
|
|
|
CHILD_WIDTH (child) = width;
|
|
|
|
CHILD_HEIGHT (child) = height;
|
|
|
|
|
|
|
|
priv->children = g_list_prepend (priv->children, child);
|
|
|
|
|
|
|
|
gtk_widget_set_parent (widget, GTK_WIDGET (grid));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Find the position 'touching' existing
|
|
|
|
* children. @orientation and @max determine
|
|
|
|
* from which direction to approach (horizontal
|
|
|
|
* + max = right, vertical + !max = top, etc).
|
|
|
|
* @op_pos, @op_span determine the rows/columns
|
|
|
|
* in which the touching has to happen.
|
|
|
|
*/
|
|
|
|
static gint
|
|
|
|
find_attach_position (GtkGrid *grid,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
gint op_pos,
|
|
|
|
gint op_span,
|
|
|
|
gboolean max)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
GtkGridChild *grid_child;
|
|
|
|
GtkGridChildAttach *attach;
|
|
|
|
GtkGridChildAttach *opposite;
|
|
|
|
GList *list;
|
|
|
|
gint pos;
|
2011-09-03 00:04:06 +00:00
|
|
|
gboolean hit;
|
|
|
|
|
|
|
|
if (max)
|
|
|
|
pos = -G_MAXINT;
|
|
|
|
else
|
|
|
|
pos = G_MAXINT;
|
|
|
|
|
|
|
|
hit = FALSE;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
grid_child = list->data;
|
|
|
|
|
2011-09-03 00:04:06 +00:00
|
|
|
attach = &grid_child->attach[orientation];
|
|
|
|
opposite = &grid_child->attach[1 - orientation];
|
|
|
|
|
|
|
|
/* check if the ranges overlap */
|
|
|
|
if (opposite->pos <= op_pos + op_span && op_pos <= opposite->pos + opposite->span)
|
|
|
|
{
|
|
|
|
hit = TRUE;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
2011-09-03 00:04:06 +00:00
|
|
|
if (max)
|
|
|
|
pos = MAX (pos, attach->pos + attach->span);
|
|
|
|
else
|
|
|
|
pos = MIN (pos, attach->pos);
|
|
|
|
}
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
2011-09-03 00:04:06 +00:00
|
|
|
if (!hit)
|
|
|
|
pos = 0;
|
|
|
|
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_add (GtkContainer *container,
|
|
|
|
GtkWidget *child)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (container);
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
gint pos[2] = { 0, 0 };
|
|
|
|
|
|
|
|
pos[priv->orientation] = find_attach_position (grid, priv->orientation, 0, 1, TRUE);
|
|
|
|
grid_attach (grid, child, pos[0], pos[1], 1, 1);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_remove (GtkContainer *container,
|
|
|
|
GtkWidget *child)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (container);
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
GtkGridChild *grid_child;
|
|
|
|
GList *list;
|
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
grid_child = list->data;
|
|
|
|
|
|
|
|
if (grid_child->widget == child)
|
|
|
|
{
|
|
|
|
gboolean was_visible = gtk_widget_get_visible (child);
|
|
|
|
|
|
|
|
gtk_widget_unparent (child);
|
|
|
|
|
|
|
|
priv->children = g_list_remove (priv->children, grid_child);
|
|
|
|
|
|
|
|
g_slice_free (GtkGridChild, grid_child);
|
|
|
|
|
|
|
|
if (was_visible && gtk_widget_get_visible (GTK_WIDGET (grid)))
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (grid));
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_forall (GtkContainer *container,
|
|
|
|
gboolean include_internals,
|
|
|
|
GtkCallback callback,
|
|
|
|
gpointer callback_data)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (container);
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GList *list;
|
|
|
|
|
2010-12-22 12:21:44 +00:00
|
|
|
list = priv->children;
|
|
|
|
while (list)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
|
|
|
child = list->data;
|
2010-12-22 12:21:44 +00:00
|
|
|
list = list->next;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
(* callback) (child->widget, callback_data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static GType
|
|
|
|
gtk_grid_child_type (GtkContainer *container)
|
|
|
|
{
|
|
|
|
return GTK_TYPE_WIDGET;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Calculates the min and max numbers for both orientations.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_count_lines (GtkGridRequest *request)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GtkGridChildAttach *attach;
|
|
|
|
GList *list;
|
|
|
|
gint min[2];
|
|
|
|
gint max[2];
|
|
|
|
|
|
|
|
min[0] = min[1] = G_MAXINT;
|
|
|
|
max[0] = max[1] = G_MININT;
|
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
attach = child->attach;
|
|
|
|
|
|
|
|
min[0] = MIN (min[0], attach[0].pos);
|
|
|
|
max[0] = MAX (max[0], attach[0].pos + attach[0].span);
|
|
|
|
min[1] = MIN (min[1], attach[1].pos);
|
|
|
|
max[1] = MAX (max[1], attach[1].pos + attach[1].span);
|
|
|
|
}
|
|
|
|
|
|
|
|
request->lines[0].min = min[0];
|
|
|
|
request->lines[0].max = max[0];
|
|
|
|
request->lines[1].min = min[1];
|
|
|
|
request->lines[1].max = max[1];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sets line sizes to 0 and marks lines as expand
|
|
|
|
* if they have a non-spanning expanding child.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_init (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GtkGridChildAttach *attach;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
GList *list;
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
|
|
|
|
for (i = 0; i < lines->max - lines->min; i++)
|
|
|
|
{
|
|
|
|
lines->lines[i].minimum = 0;
|
|
|
|
lines->lines[i].natural = 0;
|
2013-03-21 10:09:43 +00:00
|
|
|
lines->lines[i].minimum_above = -1;
|
|
|
|
lines->lines[i].minimum_below = -1;
|
|
|
|
lines->lines[i].natural_above = -1;
|
|
|
|
lines->lines[i].natural_below = -1;
|
2010-09-27 14:44:31 +00:00
|
|
|
lines->lines[i].expand = FALSE;
|
2013-03-16 03:10:39 +00:00
|
|
|
lines->lines[i].empty = TRUE;
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
attach = &child->attach[orientation];
|
2013-04-28 22:46:41 +00:00
|
|
|
if (attach->span == 1 && gtk_widget_compute_expand (child->widget, orientation))
|
2010-09-27 14:44:31 +00:00
|
|
|
lines->lines[attach->pos - lines->min].expand = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sums allocations for lines spanned by child and their spacing.
|
|
|
|
*/
|
|
|
|
static gint
|
|
|
|
compute_allocation_for_child (GtkGridRequest *request,
|
|
|
|
GtkGridChild *child,
|
|
|
|
GtkOrientation orientation)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridLineData *linedata;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
GtkGridLine *line;
|
|
|
|
GtkGridChildAttach *attach;
|
|
|
|
gint size;
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
linedata = &priv->linedata[orientation];
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
attach = &child->attach[orientation];
|
|
|
|
|
|
|
|
size = (attach->span - 1) * linedata->spacing;
|
|
|
|
for (i = 0; i < attach->span; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[attach->pos - lines->min + i];
|
|
|
|
size += line->allocation;
|
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
compute_request_for_child (GtkGridRequest *request,
|
|
|
|
GtkGridChild *child,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
gboolean contextual,
|
|
|
|
gint *minimum,
|
2013-03-21 10:09:43 +00:00
|
|
|
gint *natural,
|
|
|
|
gint *minimum_baseline,
|
|
|
|
gint *natural_baseline)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2013-03-21 10:09:43 +00:00
|
|
|
if (minimum_baseline)
|
|
|
|
*minimum_baseline = -1;
|
|
|
|
if (natural_baseline)
|
|
|
|
*natural_baseline = -1;
|
2010-09-27 14:44:31 +00:00
|
|
|
if (contextual)
|
|
|
|
{
|
|
|
|
gint size;
|
|
|
|
|
|
|
|
size = compute_allocation_for_child (request, child, 1 - orientation);
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
|
|
gtk_widget_get_preferred_width_for_height (child->widget,
|
|
|
|
size,
|
|
|
|
minimum, natural);
|
|
|
|
else
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_widget_get_preferred_height_and_baseline_for_width (child->widget,
|
|
|
|
size,
|
|
|
|
minimum, natural,
|
|
|
|
minimum_baseline, natural_baseline);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
|
|
gtk_widget_get_preferred_width (child->widget, minimum, natural);
|
|
|
|
else
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_widget_get_preferred_height_and_baseline_for_width (child->widget,
|
|
|
|
-1,
|
|
|
|
minimum, natural,
|
|
|
|
minimum_baseline, natural_baseline);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sets requisition to max. of non-spanning children.
|
|
|
|
* If contextual is TRUE, requires allocations of
|
|
|
|
* lines in the opposite orientation to be set.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_non_spanning (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
gboolean contextual)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GtkGridChildAttach *attach;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
GtkGridLine *line;
|
|
|
|
GList *list;
|
2013-03-21 10:09:43 +00:00
|
|
|
gint i;
|
|
|
|
GtkBaselinePosition baseline_pos;
|
|
|
|
gint minimum, minimum_baseline;
|
|
|
|
gint natural, natural_baseline;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
if (!gtk_widget_get_visible (child->widget))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
attach = &child->attach[orientation];
|
|
|
|
if (attach->span != 1)
|
|
|
|
continue;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
compute_request_for_child (request, child, orientation, contextual, &minimum, &natural, &minimum_baseline, &natural_baseline);
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
line = &lines->lines[attach->pos - lines->min];
|
2013-03-21 10:09:43 +00:00
|
|
|
|
|
|
|
if (minimum_baseline != -1)
|
|
|
|
{
|
|
|
|
line->minimum_above = MAX (line->minimum_above, minimum_baseline);
|
|
|
|
line->minimum_below = MAX (line->minimum_below, minimum - minimum_baseline);
|
|
|
|
line->natural_above = MAX (line->natural_above, natural_baseline);
|
|
|
|
line->natural_below = MAX (line->natural_below, natural - natural_baseline);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
line->minimum = MAX (line->minimum, minimum);
|
|
|
|
line->natural = MAX (line->natural, natural);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < lines->max - lines->min; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[i];
|
|
|
|
|
|
|
|
if (line->minimum_above != -1)
|
|
|
|
{
|
|
|
|
line->minimum = MAX (line->minimum, line->minimum_above + line->minimum_below);
|
|
|
|
line->natural = MAX (line->natural, line->natural_above + line->natural_below);
|
|
|
|
|
|
|
|
baseline_pos = gtk_grid_get_row_baseline_position (request->grid, i + lines->min);
|
|
|
|
|
|
|
|
switch (baseline_pos)
|
|
|
|
{
|
|
|
|
case GTK_BASELINE_POSITION_TOP:
|
|
|
|
line->minimum_above += 0;
|
|
|
|
line->minimum_below += line->minimum - (line->minimum_above + line->minimum_below);
|
|
|
|
line->natural_above += 0;
|
|
|
|
line->natural_below += line->natural - (line->natural_above + line->natural_below);
|
|
|
|
break;
|
|
|
|
case GTK_BASELINE_POSITION_CENTER:
|
|
|
|
line->minimum_above += (line->minimum - (line->minimum_above + line->minimum_below))/2;
|
|
|
|
line->minimum_below += (line->minimum - (line->minimum_above + line->minimum_below))/2;
|
|
|
|
line->natural_above += (line->natural - (line->natural_above + line->natural_below))/2;
|
|
|
|
line->natural_below += (line->natural - (line->natural_above + line->natural_below))/2;
|
|
|
|
break;
|
|
|
|
case GTK_BASELINE_POSITION_BOTTOM:
|
|
|
|
line->minimum_above += line->minimum - (line->minimum_above + line->minimum_below);
|
|
|
|
line->minimum_below += 0;
|
|
|
|
line->natural_above += line->natural - (line->natural_above + line->natural_below);
|
|
|
|
line->natural_below += 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enforce homogeneous sizes.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_homogeneous (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridLineData *linedata;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
gint minimum, natural;
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
linedata = &priv->linedata[orientation];
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
|
|
|
|
if (!linedata->homogeneous)
|
|
|
|
return;
|
|
|
|
|
|
|
|
minimum = 0;
|
|
|
|
natural = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < lines->max - lines->min; i++)
|
|
|
|
{
|
|
|
|
minimum = MAX (minimum, lines->lines[i].minimum);
|
|
|
|
natural = MAX (natural, lines->lines[i].natural);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < lines->max - lines->min; i++)
|
|
|
|
{
|
|
|
|
lines->lines[i].minimum = minimum;
|
|
|
|
lines->lines[i].natural = natural;
|
2013-03-21 10:09:43 +00:00
|
|
|
/* TODO: Do we want to adjust the baseline here too?
|
|
|
|
And if so, also in the homogenous resize. */
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Deals with spanning children.
|
|
|
|
* Requires expand fields of lines to be set for
|
|
|
|
* non-spanning children.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_spanning (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
gboolean contextual)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GList *list;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GtkGridChildAttach *attach;
|
|
|
|
GtkGridLineData *linedata;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
GtkGridLine *line;
|
|
|
|
gint minimum;
|
|
|
|
gint natural;
|
|
|
|
gint span_minimum;
|
|
|
|
gint span_natural;
|
|
|
|
gint span_expand;
|
|
|
|
gboolean force_expand;
|
|
|
|
gint extra;
|
|
|
|
gint expand;
|
|
|
|
gint line_extra;
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
linedata = &priv->linedata[orientation];
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
if (!gtk_widget_get_visible (child->widget))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
attach = &child->attach[orientation];
|
|
|
|
if (attach->span == 1)
|
|
|
|
continue;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
/* We ignore baselines for spanning children */
|
|
|
|
compute_request_for_child (request, child, orientation, contextual, &minimum, &natural, NULL, NULL);
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
span_minimum = (attach->span - 1) * linedata->spacing;
|
|
|
|
span_natural = (attach->span - 1) * linedata->spacing;
|
|
|
|
span_expand = 0;
|
|
|
|
force_expand = FALSE;
|
|
|
|
for (i = 0; i < attach->span; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[attach->pos - lines->min + i];
|
|
|
|
span_minimum += line->minimum;
|
|
|
|
span_natural += line->natural;
|
|
|
|
if (line->expand)
|
|
|
|
span_expand += 1;
|
|
|
|
}
|
|
|
|
if (span_expand == 0)
|
|
|
|
{
|
|
|
|
span_expand = attach->span;
|
|
|
|
force_expand = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we need to request more space for this child to fill
|
|
|
|
* its requisition, then divide up the needed space amongst the
|
|
|
|
* lines it spans, favoring expandable lines if any.
|
2012-03-04 22:19:16 +00:00
|
|
|
*
|
|
|
|
* When doing homogeneous allocation though, try to keep the
|
|
|
|
* line allocations even, since we're going to force them to
|
|
|
|
* be the same anyway, and we don't want to introduce unnecessary
|
|
|
|
* extra space.
|
2010-09-27 14:44:31 +00:00
|
|
|
*/
|
|
|
|
if (span_minimum < minimum)
|
|
|
|
{
|
2012-03-04 22:19:16 +00:00
|
|
|
if (linedata->homogeneous)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2012-03-04 22:19:16 +00:00
|
|
|
gint total, m;
|
|
|
|
|
|
|
|
total = minimum - (attach->span - 1) * linedata->spacing;
|
|
|
|
m = total / attach->span + (total % attach->span ? 1 : 0);
|
|
|
|
for (i = 0; i < attach->span; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[attach->pos - lines->min + i];
|
|
|
|
line->minimum = MAX(line->minimum, m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
extra = minimum - span_minimum;
|
|
|
|
expand = span_expand;
|
|
|
|
for (i = 0; i < attach->span; i++)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2012-03-04 22:19:16 +00:00
|
|
|
line = &lines->lines[attach->pos - lines->min + i];
|
|
|
|
if (force_expand || line->expand)
|
|
|
|
{
|
|
|
|
line_extra = extra / expand;
|
|
|
|
line->minimum += line_extra;
|
|
|
|
extra -= line_extra;
|
|
|
|
expand -= 1;
|
|
|
|
}
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (span_natural < natural)
|
|
|
|
{
|
2012-03-04 22:19:16 +00:00
|
|
|
if (linedata->homogeneous)
|
|
|
|
{
|
|
|
|
gint total, n;
|
|
|
|
|
|
|
|
total = natural - (attach->span - 1) * linedata->spacing;
|
|
|
|
n = total / attach->span + (total % attach->span ? 1 : 0);
|
|
|
|
for (i = 0; i < attach->span; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[attach->pos - lines->min + i];
|
|
|
|
line->natural = MAX(line->natural, n);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2012-03-04 22:19:16 +00:00
|
|
|
extra = natural - span_natural;
|
|
|
|
expand = span_expand;
|
|
|
|
for (i = 0; i < attach->span; i++)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2012-03-04 22:19:16 +00:00
|
|
|
line = &lines->lines[attach->pos - lines->min + i];
|
|
|
|
if (force_expand || line->expand)
|
|
|
|
{
|
|
|
|
line_extra = extra / expand;
|
|
|
|
line->natural += line_extra;
|
|
|
|
extra -= line_extra;
|
|
|
|
expand -= 1;
|
|
|
|
}
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Marks empty and expanding lines and counts them.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_compute_expand (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation,
|
2013-03-21 10:09:43 +00:00
|
|
|
gint min,
|
|
|
|
gint max,
|
2010-09-27 14:44:31 +00:00
|
|
|
gint *nonempty_lines,
|
|
|
|
gint *expand_lines)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GtkGridChildAttach *attach;
|
|
|
|
GList *list;
|
|
|
|
gint i;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
GtkGridLine *line;
|
|
|
|
gboolean has_expand;
|
|
|
|
gint expand;
|
|
|
|
gint empty;
|
|
|
|
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
min = MAX (min, lines->min);
|
|
|
|
max = MIN (max, lines->max);
|
|
|
|
|
|
|
|
for (i = min - lines->min; i < max - lines->min; i++)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
|
|
|
lines->lines[i].need_expand = FALSE;
|
|
|
|
lines->lines[i].expand = FALSE;
|
|
|
|
lines->lines[i].empty = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
if (!gtk_widget_get_visible (child->widget))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
attach = &child->attach[orientation];
|
|
|
|
if (attach->span != 1)
|
|
|
|
continue;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
if (attach->pos >= max || attach->pos < min)
|
|
|
|
continue;
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
line = &lines->lines[attach->pos - lines->min];
|
|
|
|
line->empty = FALSE;
|
|
|
|
if (gtk_widget_compute_expand (child->widget, orientation))
|
|
|
|
line->expand = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
if (!gtk_widget_get_visible (child->widget))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
attach = &child->attach[orientation];
|
|
|
|
if (attach->span == 1)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
has_expand = FALSE;
|
|
|
|
for (i = 0; i < attach->span; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[attach->pos - lines->min + i];
|
2013-03-21 10:09:43 +00:00
|
|
|
|
|
|
|
if (line->expand)
|
|
|
|
has_expand = TRUE;
|
|
|
|
|
|
|
|
if (attach->pos + i >= max || attach->pos + 1 < min)
|
|
|
|
continue;
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
line->empty = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_expand && gtk_widget_compute_expand (child->widget, orientation))
|
|
|
|
{
|
|
|
|
for (i = 0; i < attach->span; i++)
|
|
|
|
{
|
2013-03-21 10:09:43 +00:00
|
|
|
if (attach->pos + i >= max || attach->pos + 1 < min)
|
|
|
|
continue;
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
line = &lines->lines[attach->pos - lines->min + i];
|
|
|
|
line->need_expand = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
empty = 0;
|
|
|
|
expand = 0;
|
2013-03-21 10:09:43 +00:00
|
|
|
for (i = min - lines->min; i < max - lines->min; i++)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
|
|
|
line = &lines->lines[i];
|
|
|
|
|
|
|
|
if (line->need_expand)
|
|
|
|
line->expand = TRUE;
|
|
|
|
|
|
|
|
if (line->empty)
|
|
|
|
empty += 1;
|
|
|
|
|
|
|
|
if (line->expand)
|
|
|
|
expand += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nonempty_lines)
|
2013-03-21 10:09:43 +00:00
|
|
|
*nonempty_lines = max - min - empty;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
if (expand_lines)
|
|
|
|
*expand_lines = expand;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sums the minimum and natural fields of lines and their spacing.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_sum (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
gint *minimum,
|
2013-03-21 10:09:43 +00:00
|
|
|
gint *natural,
|
|
|
|
gint *minimum_baseline,
|
|
|
|
gint *natural_baseline)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridLineData *linedata;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
gint i;
|
|
|
|
gint min, nat;
|
|
|
|
gint nonempty;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_request_compute_expand (request, orientation, G_MININT, G_MAXINT, &nonempty, NULL);
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
linedata = &priv->linedata[orientation];
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
|
2011-10-21 20:23:03 +00:00
|
|
|
min = 0;
|
|
|
|
nat = 0;
|
2010-09-27 14:44:31 +00:00
|
|
|
for (i = 0; i < lines->max - lines->min; i++)
|
|
|
|
{
|
2013-03-21 10:09:43 +00:00
|
|
|
if (orientation == GTK_ORIENTATION_VERTICAL &&
|
|
|
|
lines->min + i == priv->baseline_row &&
|
|
|
|
lines->lines[i].minimum_above != -1)
|
|
|
|
{
|
|
|
|
if (minimum_baseline)
|
|
|
|
*minimum_baseline = min + lines->lines[i].minimum_above;
|
|
|
|
if (natural_baseline)
|
|
|
|
*natural_baseline = nat + lines->lines[i].natural_above;
|
|
|
|
}
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
min += lines->lines[i].minimum;
|
|
|
|
nat += lines->lines[i].natural;
|
2013-03-21 10:09:43 +00:00
|
|
|
|
|
|
|
if (!lines->lines[i].empty)
|
|
|
|
{
|
|
|
|
min += linedata->spacing;
|
|
|
|
nat += linedata->spacing;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove last spacing, if any was applied */
|
|
|
|
if (nonempty > 0)
|
|
|
|
{
|
|
|
|
min -= linedata->spacing;
|
|
|
|
nat -= linedata->spacing;
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (minimum)
|
|
|
|
*minimum = min;
|
|
|
|
|
|
|
|
if (natural)
|
|
|
|
*natural = nat;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Computes minimum and natural fields of lines.
|
|
|
|
* When contextual is TRUE, requires allocation of
|
|
|
|
* lines in the opposite orientation to be set.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_run (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
gboolean contextual)
|
|
|
|
{
|
|
|
|
gtk_grid_request_init (request, orientation);
|
|
|
|
gtk_grid_request_non_spanning (request, orientation, contextual);
|
|
|
|
gtk_grid_request_homogeneous (request, orientation);
|
|
|
|
gtk_grid_request_spanning (request, orientation, contextual);
|
|
|
|
gtk_grid_request_homogeneous (request, orientation);
|
|
|
|
}
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
static void
|
|
|
|
gtk_grid_distribute_non_homogeneous (GtkGridLines *lines,
|
|
|
|
gint nonempty,
|
|
|
|
gint expand,
|
|
|
|
gint size,
|
|
|
|
gint min,
|
|
|
|
gint max)
|
|
|
|
{
|
|
|
|
GtkRequestedSize *sizes;
|
|
|
|
GtkGridLine *line;
|
|
|
|
gint extra;
|
|
|
|
gint rest;
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
if (nonempty == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
sizes = g_newa (GtkRequestedSize, nonempty);
|
|
|
|
|
|
|
|
j = 0;
|
|
|
|
for (i = min - lines->min; i < max - lines->min; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[i];
|
|
|
|
if (line->empty)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
size -= line->minimum;
|
|
|
|
|
|
|
|
sizes[j].minimum_size = line->minimum;
|
|
|
|
sizes[j].natural_size = line->natural;
|
|
|
|
sizes[j].data = line;
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
|
|
|
|
size = gtk_distribute_natural_allocation (MAX (0, size), nonempty, sizes);
|
|
|
|
|
|
|
|
if (expand > 0)
|
|
|
|
{
|
|
|
|
extra = size / expand;
|
|
|
|
rest = size % expand;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
extra = 0;
|
|
|
|
rest = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
j = 0;
|
|
|
|
for (i = min - lines->min; i < max - lines->min; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[i];
|
|
|
|
if (line->empty)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
g_assert (line == sizes[j].data);
|
|
|
|
|
|
|
|
line->allocation = sizes[j].minimum_size;
|
|
|
|
if (line->expand)
|
|
|
|
{
|
|
|
|
line->allocation += extra;
|
|
|
|
if (rest > 0)
|
|
|
|
{
|
|
|
|
line->allocation += 1;
|
|
|
|
rest -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
/* Requires that the minimum and natural fields of lines
|
|
|
|
* have been set, computes the allocation field of lines
|
|
|
|
* by distributing total_size among lines.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_allocate (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
gint total_size)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridLineData *linedata;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
GtkGridLine *line;
|
2013-03-21 10:09:43 +00:00
|
|
|
gint nonempty1, nonempty2;
|
|
|
|
gint expand1, expand2;
|
|
|
|
gint i;
|
|
|
|
GtkBaselinePosition baseline_pos;
|
|
|
|
gint baseline;
|
|
|
|
gint extra, extra2;
|
2010-09-27 14:44:31 +00:00
|
|
|
gint rest;
|
2013-03-21 10:09:43 +00:00
|
|
|
gint size1, size2;
|
|
|
|
gint split, split_pos;
|
2012-03-24 21:16:47 +00:00
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
linedata = &priv->linedata[orientation];
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
baseline = gtk_widget_get_allocated_baseline (GTK_WIDGET (request->grid));
|
|
|
|
|
|
|
|
if (orientation == GTK_ORIENTATION_VERTICAL && baseline != -1 &&
|
|
|
|
priv->baseline_row >= lines->min && priv->baseline_row < lines->max &&
|
|
|
|
lines->lines[priv->baseline_row - lines->min].minimum_above != -1)
|
|
|
|
{
|
|
|
|
split = priv->baseline_row;
|
|
|
|
split_pos = baseline - lines->lines[priv->baseline_row - lines->min].minimum_above;
|
|
|
|
gtk_grid_request_compute_expand (request, orientation, lines->min, split, &nonempty1, &expand1);
|
|
|
|
gtk_grid_request_compute_expand (request, orientation, split, lines->max, &nonempty2, &expand2);
|
|
|
|
|
|
|
|
if (nonempty2 > 0)
|
|
|
|
{
|
|
|
|
size1 = split_pos - (nonempty1) * linedata->spacing;
|
|
|
|
size2 = (total_size - split_pos) - (nonempty2 - 1) * linedata->spacing;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
size1 = total_size - (nonempty1 - 1) * linedata->spacing;
|
|
|
|
size2 = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gtk_grid_request_compute_expand (request, orientation, lines->min, lines->max, &nonempty1, &expand1);
|
|
|
|
nonempty2 = expand2 = 0;
|
|
|
|
split = lines->max;
|
|
|
|
|
|
|
|
size1 = total_size - (nonempty1 - 1) * linedata->spacing;
|
|
|
|
size2 = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nonempty1 == 0 && nonempty2 == 0)
|
|
|
|
return;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
if (linedata->homogeneous)
|
|
|
|
{
|
2013-03-21 10:09:43 +00:00
|
|
|
if (nonempty1 > 0)
|
|
|
|
{
|
|
|
|
extra = size1 / nonempty1;
|
|
|
|
rest = size1 % nonempty1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
extra = 0;
|
|
|
|
rest = 0;
|
|
|
|
}
|
|
|
|
if (nonempty2 > 0)
|
|
|
|
{
|
|
|
|
extra2 = size2 / nonempty2;
|
|
|
|
if (extra2 < extra || nonempty1 == 0)
|
|
|
|
{
|
|
|
|
extra = extra2;
|
|
|
|
rest = size2 % nonempty2;
|
|
|
|
}
|
|
|
|
}
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
for (i = 0; i < lines->max - lines->min; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[i];
|
|
|
|
if (line->empty)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
line->allocation = extra;
|
|
|
|
if (rest > 0)
|
|
|
|
{
|
|
|
|
line->allocation += 1;
|
|
|
|
rest -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_distribute_non_homogeneous (lines,
|
|
|
|
nonempty1,
|
|
|
|
expand1,
|
|
|
|
size1,
|
|
|
|
lines->min,
|
|
|
|
split);
|
|
|
|
gtk_grid_distribute_non_homogeneous (lines,
|
|
|
|
nonempty2,
|
|
|
|
expand2,
|
|
|
|
size2,
|
|
|
|
split,
|
|
|
|
lines->max);
|
|
|
|
}
|
2010-09-27 14:44:31 +00:00
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
for (i = 0; i < lines->max - lines->min; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[i];
|
|
|
|
if (line->empty)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (line->minimum_above != -1)
|
|
|
|
{
|
|
|
|
/* Note: This is overridden in gtk_grid_request_position for the allocated baseline */
|
|
|
|
baseline_pos = gtk_grid_get_row_baseline_position (request->grid, i + lines->min);
|
|
|
|
|
|
|
|
switch (baseline_pos)
|
|
|
|
{
|
|
|
|
case GTK_BASELINE_POSITION_TOP:
|
|
|
|
line->allocated_baseline =
|
|
|
|
line->minimum_above;
|
|
|
|
break;
|
|
|
|
case GTK_BASELINE_POSITION_CENTER:
|
|
|
|
line->allocated_baseline =
|
|
|
|
line->minimum_above +
|
|
|
|
(line->allocation - (line->minimum_above + line->minimum_below)) / 2;
|
|
|
|
break;
|
|
|
|
case GTK_BASELINE_POSITION_BOTTOM:
|
|
|
|
line->allocated_baseline =
|
|
|
|
line->allocation - line->minimum_below;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2010-09-27 14:44:31 +00:00
|
|
|
else
|
2013-03-21 10:09:43 +00:00
|
|
|
line->allocated_baseline = -1;
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Computes the position fields from allocation and spacing.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
gtk_grid_request_position (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridLineData *linedata;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
GtkGridLine *line;
|
2013-03-21 10:09:43 +00:00
|
|
|
gint position, old_position;
|
|
|
|
int allocated_baseline;
|
|
|
|
gint i, j;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
linedata = &priv->linedata[orientation];
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
allocated_baseline = gtk_widget_get_allocated_baseline (GTK_WIDGET(request->grid));
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
position = 0;
|
|
|
|
for (i = 0; i < lines->max - lines->min; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[i];
|
2013-03-21 10:09:43 +00:00
|
|
|
|
|
|
|
if (orientation == GTK_ORIENTATION_VERTICAL &&
|
|
|
|
i + lines->min == priv->baseline_row &&
|
|
|
|
allocated_baseline != -1 &&
|
|
|
|
lines->lines[i].minimum_above != -1)
|
|
|
|
{
|
|
|
|
old_position = position;
|
|
|
|
position = allocated_baseline - line->minimum_above;
|
|
|
|
|
|
|
|
/* Back-patch previous rows */
|
|
|
|
for (j = 0; j < i; j++)
|
|
|
|
{
|
|
|
|
if (!lines->lines[j].empty)
|
|
|
|
lines->lines[j].position += position - old_position;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
if (!line->empty)
|
|
|
|
{
|
|
|
|
line->position = position;
|
|
|
|
position += line->allocation + linedata->spacing;
|
2013-03-21 10:09:43 +00:00
|
|
|
|
|
|
|
if (orientation == GTK_ORIENTATION_VERTICAL &&
|
|
|
|
i + lines->min == priv->baseline_row &&
|
|
|
|
allocated_baseline != -1 &&
|
|
|
|
lines->lines[i].minimum_above != -1)
|
|
|
|
line->allocated_baseline = allocated_baseline - line->position;
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_get_size (GtkGrid *grid,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
gint *minimum,
|
2013-03-21 10:09:43 +00:00
|
|
|
gint *natural,
|
|
|
|
gint *minimum_baseline,
|
|
|
|
gint *natural_baseline)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
|
|
|
GtkGridRequest request;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
|
2011-05-13 00:12:51 +00:00
|
|
|
if (minimum)
|
|
|
|
*minimum = 0;
|
|
|
|
|
|
|
|
if (natural)
|
|
|
|
*natural = 0;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
if (minimum_baseline)
|
|
|
|
*minimum_baseline = -1;
|
|
|
|
|
|
|
|
if (natural_baseline)
|
|
|
|
*natural_baseline = -1;
|
|
|
|
|
2011-05-13 00:12:51 +00:00
|
|
|
if (grid->priv->children == NULL)
|
|
|
|
return;
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
request.grid = grid;
|
|
|
|
gtk_grid_request_count_lines (&request);
|
|
|
|
lines = &request.lines[orientation];
|
|
|
|
lines->lines = g_newa (GtkGridLine, lines->max - lines->min);
|
2011-02-17 16:51:16 +00:00
|
|
|
memset (lines->lines, 0, (lines->max - lines->min) * sizeof (GtkGridLine));
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
gtk_grid_request_run (&request, orientation, FALSE);
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_request_sum (&request, orientation, minimum, natural,
|
|
|
|
minimum_baseline, natural_baseline);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_get_size_for_size (GtkGrid *grid,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
gint size,
|
|
|
|
gint *minimum,
|
2013-03-21 10:09:43 +00:00
|
|
|
gint *natural,
|
|
|
|
gint *minimum_baseline,
|
|
|
|
gint *natural_baseline)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
|
|
|
GtkGridRequest request;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
gint min_size;
|
|
|
|
|
2011-05-13 00:12:51 +00:00
|
|
|
if (minimum)
|
|
|
|
*minimum = 0;
|
|
|
|
|
|
|
|
if (natural)
|
|
|
|
*natural = 0;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
if (minimum_baseline)
|
|
|
|
*minimum_baseline = -1;
|
|
|
|
|
|
|
|
if (natural_baseline)
|
|
|
|
*natural_baseline = -1;
|
|
|
|
|
2011-05-13 00:12:51 +00:00
|
|
|
if (grid->priv->children == NULL)
|
|
|
|
return;
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
request.grid = grid;
|
|
|
|
gtk_grid_request_count_lines (&request);
|
|
|
|
lines = &request.lines[0];
|
|
|
|
lines->lines = g_newa (GtkGridLine, lines->max - lines->min);
|
2011-02-17 16:51:16 +00:00
|
|
|
memset (lines->lines, 0, (lines->max - lines->min) * sizeof (GtkGridLine));
|
2010-09-27 14:44:31 +00:00
|
|
|
lines = &request.lines[1];
|
|
|
|
lines->lines = g_newa (GtkGridLine, lines->max - lines->min);
|
2011-02-17 16:51:16 +00:00
|
|
|
memset (lines->lines, 0, (lines->max - lines->min) * sizeof (GtkGridLine));
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
gtk_grid_request_run (&request, 1 - orientation, FALSE);
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_request_sum (&request, 1 - orientation, &min_size, NULL, NULL, NULL);
|
2010-09-27 14:44:31 +00:00
|
|
|
gtk_grid_request_allocate (&request, 1 - orientation, MAX (size, min_size));
|
|
|
|
|
|
|
|
gtk_grid_request_run (&request, orientation, TRUE);
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_request_sum (&request, orientation, minimum, natural, minimum_baseline, natural_baseline);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_get_preferred_width (GtkWidget *widget,
|
|
|
|
gint *minimum,
|
|
|
|
gint *natural)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (widget);
|
|
|
|
|
2011-03-05 08:49:49 +00:00
|
|
|
if (gtk_widget_get_request_mode (widget) == GTK_SIZE_REQUEST_WIDTH_FOR_HEIGHT)
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_get_size_for_size (grid, GTK_ORIENTATION_HORIZONTAL, 0, minimum, natural, NULL, NULL);
|
2010-09-27 14:44:31 +00:00
|
|
|
else
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_get_size (grid, GTK_ORIENTATION_HORIZONTAL, minimum, natural, NULL, NULL);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_get_preferred_height (GtkWidget *widget,
|
|
|
|
gint *minimum,
|
|
|
|
gint *natural)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (widget);
|
|
|
|
|
2011-03-05 08:49:49 +00:00
|
|
|
if (gtk_widget_get_request_mode (widget) == GTK_SIZE_REQUEST_HEIGHT_FOR_WIDTH)
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_get_size_for_size (grid, GTK_ORIENTATION_VERTICAL, 0, minimum, natural, NULL, NULL);
|
2010-09-27 14:44:31 +00:00
|
|
|
else
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_get_size (grid, GTK_ORIENTATION_VERTICAL, minimum, natural, NULL, NULL);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_get_preferred_width_for_height (GtkWidget *widget,
|
|
|
|
gint height,
|
|
|
|
gint *minimum,
|
|
|
|
gint *natural)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (widget);
|
|
|
|
|
2011-03-05 08:49:49 +00:00
|
|
|
if (gtk_widget_get_request_mode (widget) == GTK_SIZE_REQUEST_WIDTH_FOR_HEIGHT)
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_get_size_for_size (grid, GTK_ORIENTATION_HORIZONTAL, height, minimum, natural, NULL, NULL);
|
2010-09-27 14:44:31 +00:00
|
|
|
else
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_get_size (grid, GTK_ORIENTATION_HORIZONTAL, minimum, natural, NULL, NULL);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_get_preferred_height_for_width (GtkWidget *widget,
|
|
|
|
gint width,
|
|
|
|
gint *minimum,
|
|
|
|
gint *natural)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (widget);
|
|
|
|
|
2011-03-05 08:49:49 +00:00
|
|
|
if (gtk_widget_get_request_mode (widget) == GTK_SIZE_REQUEST_HEIGHT_FOR_WIDTH)
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_get_size_for_size (grid, GTK_ORIENTATION_VERTICAL, width, minimum, natural, NULL, NULL);
|
|
|
|
else
|
|
|
|
gtk_grid_get_size (grid, GTK_ORIENTATION_VERTICAL, minimum, natural, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_get_preferred_height_and_baseline_for_width (GtkWidget *widget,
|
|
|
|
gint width,
|
|
|
|
gint *minimum,
|
|
|
|
gint *natural,
|
|
|
|
gint *minimum_baseline,
|
|
|
|
gint *natural_baseline)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (widget);
|
|
|
|
|
|
|
|
if (gtk_widget_get_request_mode (widget) == GTK_SIZE_REQUEST_HEIGHT_FOR_WIDTH && width != -1)
|
|
|
|
gtk_grid_get_size_for_size (grid, GTK_ORIENTATION_VERTICAL, width, minimum, natural, minimum_baseline, natural_baseline);
|
2010-09-27 14:44:31 +00:00
|
|
|
else
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_grid_get_size (grid, GTK_ORIENTATION_VERTICAL, minimum, natural, minimum_baseline, natural_baseline);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
static void
|
|
|
|
allocate_child (GtkGridRequest *request,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
GtkGridChild *child,
|
|
|
|
gint *position,
|
2013-03-21 10:09:43 +00:00
|
|
|
gint *size,
|
|
|
|
gint *baseline)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GtkGridLineData *linedata;
|
|
|
|
GtkGridLines *lines;
|
|
|
|
GtkGridLine *line;
|
|
|
|
GtkGridChildAttach *attach;
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
linedata = &priv->linedata[orientation];
|
|
|
|
lines = &request->lines[orientation];
|
|
|
|
attach = &child->attach[orientation];
|
|
|
|
|
|
|
|
*position = lines->lines[attach->pos - lines->min].position;
|
2013-03-21 10:09:43 +00:00
|
|
|
if (attach->span == 1)
|
|
|
|
*baseline = lines->lines[attach->pos - lines->min].allocated_baseline;
|
|
|
|
else
|
|
|
|
*baseline = -1;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
*size = (attach->span - 1) * linedata->spacing;
|
|
|
|
for (i = 0; i < attach->span; i++)
|
|
|
|
{
|
|
|
|
line = &lines->lines[attach->pos - lines->min + i];
|
|
|
|
*size += line->allocation;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_request_allocate_children (GtkGridRequest *request)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv = request->grid->priv;
|
|
|
|
GList *list;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GtkAllocation allocation;
|
|
|
|
GtkAllocation child_allocation;
|
2013-03-21 10:09:43 +00:00
|
|
|
gint x, y, width, height, baseline, ignore;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
gtk_widget_get_allocation (GTK_WIDGET (request->grid), &allocation);
|
|
|
|
|
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
if (!gtk_widget_get_visible (child->widget))
|
|
|
|
continue;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
allocate_child (request, GTK_ORIENTATION_HORIZONTAL, child, &x, &width, &ignore);
|
|
|
|
allocate_child (request, GTK_ORIENTATION_VERTICAL, child, &y, &height, &baseline);
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
child_allocation.x = allocation.x + x;
|
|
|
|
child_allocation.y = allocation.y + y;
|
|
|
|
child_allocation.width = MAX (1, width);
|
|
|
|
child_allocation.height = MAX (1, height);
|
|
|
|
|
2011-03-30 06:24:00 +00:00
|
|
|
if (gtk_widget_get_direction (GTK_WIDGET (request->grid)) == GTK_TEXT_DIR_RTL)
|
|
|
|
child_allocation.x = allocation.x + allocation.width
|
|
|
|
- (child_allocation.x - allocation.x) - child_allocation.width;
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
gtk_widget_size_allocate_with_baseline (child->widget, &child_allocation, baseline);
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#define GET_SIZE(allocation, orientation) (orientation == GTK_ORIENTATION_HORIZONTAL ? allocation->width : allocation->height)
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_size_allocate (GtkWidget *widget,
|
|
|
|
GtkAllocation *allocation)
|
|
|
|
{
|
|
|
|
GtkGrid *grid = GTK_GRID (widget);
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
GtkGridRequest request;
|
|
|
|
GtkGridLines *lines;
|
2011-06-07 15:05:50 +00:00
|
|
|
GtkOrientation orientation;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
2011-05-13 00:12:51 +00:00
|
|
|
if (priv->children == NULL)
|
|
|
|
{
|
|
|
|
gtk_widget_set_allocation (widget, allocation);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
request.grid = grid;
|
2011-02-17 16:51:16 +00:00
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
gtk_grid_request_count_lines (&request);
|
|
|
|
lines = &request.lines[0];
|
|
|
|
lines->lines = g_newa (GtkGridLine, lines->max - lines->min);
|
2011-02-17 16:51:16 +00:00
|
|
|
memset (lines->lines, 0, (lines->max - lines->min) * sizeof (GtkGridLine));
|
2010-09-27 14:44:31 +00:00
|
|
|
lines = &request.lines[1];
|
|
|
|
lines->lines = g_newa (GtkGridLine, lines->max - lines->min);
|
2011-02-17 16:51:16 +00:00
|
|
|
memset (lines->lines, 0, (lines->max - lines->min) * sizeof (GtkGridLine));
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
gtk_widget_set_allocation (widget, allocation);
|
|
|
|
|
2011-06-07 15:05:50 +00:00
|
|
|
if (gtk_widget_get_request_mode (widget) == GTK_SIZE_REQUEST_WIDTH_FOR_HEIGHT)
|
|
|
|
orientation = GTK_ORIENTATION_HORIZONTAL;
|
|
|
|
else
|
|
|
|
orientation = GTK_ORIENTATION_VERTICAL;
|
|
|
|
|
|
|
|
gtk_grid_request_run (&request, 1 - orientation, FALSE);
|
|
|
|
gtk_grid_request_allocate (&request, 1 - orientation, GET_SIZE (allocation, 1 - orientation));
|
|
|
|
gtk_grid_request_run (&request, orientation, TRUE);
|
2013-03-21 10:09:43 +00:00
|
|
|
|
2011-06-07 15:05:50 +00:00
|
|
|
gtk_grid_request_allocate (&request, orientation, GET_SIZE (allocation, orientation));
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
gtk_grid_request_position (&request, 0);
|
|
|
|
gtk_grid_request_position (&request, 1);
|
|
|
|
|
|
|
|
gtk_grid_request_allocate_children (&request);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_class_init (GtkGridClass *class)
|
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (class);
|
|
|
|
GtkContainerClass *container_class = GTK_CONTAINER_CLASS (class);
|
|
|
|
|
|
|
|
object_class->get_property = gtk_grid_get_property;
|
|
|
|
object_class->set_property = gtk_grid_set_property;
|
2013-03-21 10:09:43 +00:00
|
|
|
object_class->finalize = gtk_grid_finalize;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
widget_class->size_allocate = gtk_grid_size_allocate;
|
|
|
|
widget_class->get_preferred_width = gtk_grid_get_preferred_width;
|
|
|
|
widget_class->get_preferred_height = gtk_grid_get_preferred_height;
|
|
|
|
widget_class->get_preferred_width_for_height = gtk_grid_get_preferred_width_for_height;
|
|
|
|
widget_class->get_preferred_height_for_width = gtk_grid_get_preferred_height_for_width;
|
2013-03-21 10:09:43 +00:00
|
|
|
widget_class->get_preferred_height_and_baseline_for_width = gtk_grid_get_preferred_height_and_baseline_for_width;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
container_class->add = gtk_grid_add;
|
|
|
|
container_class->remove = gtk_grid_remove;
|
|
|
|
container_class->forall = gtk_grid_forall;
|
|
|
|
container_class->child_type = gtk_grid_child_type;
|
|
|
|
container_class->set_child_property = gtk_grid_set_child_property;
|
|
|
|
container_class->get_child_property = gtk_grid_get_child_property;
|
|
|
|
gtk_container_class_handle_border_width (container_class);
|
|
|
|
|
|
|
|
g_object_class_override_property (object_class, PROP_ORIENTATION, "orientation");
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_ROW_SPACING,
|
|
|
|
g_param_spec_int ("row-spacing",
|
|
|
|
P_("Row spacing"),
|
|
|
|
P_("The amount of space between two consecutive rows"),
|
|
|
|
0, G_MAXINT16, 0,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_COLUMN_SPACING,
|
|
|
|
g_param_spec_int ("column-spacing",
|
|
|
|
P_("Column spacing"),
|
|
|
|
P_("The amount of space between two consecutive columns"),
|
|
|
|
0, G_MAXINT16, 0,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_ROW_HOMOGENEOUS,
|
|
|
|
g_param_spec_boolean ("row-homogeneous",
|
|
|
|
P_("Row Homogeneous"),
|
|
|
|
P_("If TRUE, the rows are all the same height"),
|
|
|
|
FALSE,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_COLUMN_HOMOGENEOUS,
|
|
|
|
g_param_spec_boolean ("column-homogeneous",
|
|
|
|
P_("Column Homogeneous"),
|
|
|
|
P_("If TRUE, the columns are all the same width"),
|
|
|
|
FALSE,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
|
2013-03-21 10:09:43 +00:00
|
|
|
g_object_class_install_property (object_class, PROP_BASELINE_ROW,
|
|
|
|
g_param_spec_int ("baseline-row",
|
|
|
|
P_("Baseline Row"),
|
|
|
|
P_("The row to align the to the baseline when valign is GTK_ALIGN_BASELINE"),
|
|
|
|
0, G_MAXINT, 0,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
gtk_container_class_install_child_property (container_class, CHILD_PROP_LEFT_ATTACH,
|
|
|
|
g_param_spec_int ("left-attach",
|
|
|
|
P_("Left attachment"),
|
|
|
|
P_("The column number to attach the left side of the child to"),
|
|
|
|
G_MININT, G_MAXINT, 0,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
|
|
|
|
gtk_container_class_install_child_property (container_class, CHILD_PROP_TOP_ATTACH,
|
|
|
|
g_param_spec_int ("top-attach",
|
|
|
|
P_("Top attachment"),
|
|
|
|
P_("The row number to attach the top side of a child widget to"),
|
|
|
|
G_MININT, G_MAXINT, 0,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
|
|
|
|
gtk_container_class_install_child_property (container_class, CHILD_PROP_WIDTH,
|
|
|
|
g_param_spec_int ("width",
|
|
|
|
P_("Width"),
|
|
|
|
P_("The number of columns that a child spans"),
|
|
|
|
1, G_MAXINT, 1,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
|
|
|
|
gtk_container_class_install_child_property (container_class, CHILD_PROP_HEIGHT,
|
|
|
|
g_param_spec_int ("height",
|
|
|
|
P_("Height"),
|
|
|
|
P_("The number of rows that a child spans"),
|
|
|
|
1, G_MAXINT, 1,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_new:
|
|
|
|
*
|
|
|
|
* Creates a new grid widget.
|
|
|
|
*
|
|
|
|
* Returns: the new #GtkGrid
|
|
|
|
*/
|
|
|
|
GtkWidget *
|
|
|
|
gtk_grid_new (void)
|
|
|
|
{
|
|
|
|
return g_object_new (GTK_TYPE_GRID, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_attach:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @child: the widget to add
|
|
|
|
* @left: the column number to attach the left side of @child to
|
|
|
|
* @top: the row number to attach the top side of @child to
|
|
|
|
* @width: the number of columns that @child will span
|
|
|
|
* @height: the number of rows that @child will span
|
|
|
|
*
|
|
|
|
* Adds a widget to the grid.
|
|
|
|
*
|
|
|
|
* The position of @child is determined by @left and @top. The
|
|
|
|
* number of 'cells' that @child will occupy is determined by
|
|
|
|
* @width and @height.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_attach (GtkGrid *grid,
|
|
|
|
GtkWidget *child,
|
|
|
|
gint left,
|
|
|
|
gint top,
|
|
|
|
gint width,
|
|
|
|
gint height)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (child));
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (child) == NULL);
|
|
|
|
g_return_if_fail (width > 0);
|
|
|
|
g_return_if_fail (height > 0);
|
|
|
|
|
|
|
|
grid_attach (grid, child, left, top, width, height);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_attach_next_to:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @child: the widget to add
|
2011-10-15 17:37:43 +00:00
|
|
|
* @sibling: (allow-none): the child of @grid that @child will be placed
|
2011-09-03 00:04:06 +00:00
|
|
|
* next to, or %NULL to place @child at the beginning or end
|
2010-09-27 14:44:31 +00:00
|
|
|
* @side: the side of @sibling that @child is positioned next to
|
|
|
|
* @width: the number of columns that @child will span
|
|
|
|
* @height: the number of rows that @child will span
|
|
|
|
*
|
|
|
|
* Adds a widget to the grid.
|
|
|
|
*
|
|
|
|
* The widget is placed next to @sibling, on the side determined by
|
2011-09-03 00:04:06 +00:00
|
|
|
* @side. When @sibling is %NULL, the widget is placed in row (for
|
|
|
|
* left or right placement) or column 0 (for top or bottom placement),
|
|
|
|
* at the end indicated by @side.
|
2011-10-26 03:18:21 +00:00
|
|
|
*
|
|
|
|
* Attaching widgets labeled [1], [2], [3] with @sibling == %NULL and
|
|
|
|
* @side == %GTK_POS_LEFT yields a layout of [3][2][1].
|
2010-09-27 14:44:31 +00:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_attach_next_to (GtkGrid *grid,
|
|
|
|
GtkWidget *child,
|
|
|
|
GtkWidget *sibling,
|
|
|
|
GtkPositionType side,
|
|
|
|
gint width,
|
|
|
|
gint height)
|
|
|
|
{
|
|
|
|
GtkGridChild *grid_sibling;
|
|
|
|
gint left, top;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (child));
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (child) == NULL);
|
2011-09-03 00:04:06 +00:00
|
|
|
g_return_if_fail (sibling == NULL || gtk_widget_get_parent (sibling) == (GtkWidget*)grid);
|
2010-09-27 14:44:31 +00:00
|
|
|
g_return_if_fail (width > 0);
|
|
|
|
g_return_if_fail (height > 0);
|
|
|
|
|
2011-09-03 00:04:06 +00:00
|
|
|
if (sibling)
|
|
|
|
{
|
|
|
|
grid_sibling = find_grid_child (grid, sibling);
|
2010-09-27 14:44:31 +00:00
|
|
|
|
2011-09-03 00:04:06 +00:00
|
|
|
switch (side)
|
|
|
|
{
|
|
|
|
case GTK_POS_LEFT:
|
|
|
|
left = CHILD_LEFT (grid_sibling) - width;
|
|
|
|
top = CHILD_TOP (grid_sibling);
|
|
|
|
break;
|
|
|
|
case GTK_POS_RIGHT:
|
|
|
|
left = CHILD_LEFT (grid_sibling) + CHILD_WIDTH (grid_sibling);
|
|
|
|
top = CHILD_TOP (grid_sibling);
|
|
|
|
break;
|
|
|
|
case GTK_POS_TOP:
|
|
|
|
left = CHILD_LEFT (grid_sibling);
|
|
|
|
top = CHILD_TOP (grid_sibling) - height;
|
|
|
|
break;
|
|
|
|
case GTK_POS_BOTTOM:
|
|
|
|
left = CHILD_LEFT (grid_sibling);
|
|
|
|
top = CHILD_TOP (grid_sibling) + CHILD_HEIGHT (grid_sibling);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
g_assert_not_reached ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2011-09-03 00:04:06 +00:00
|
|
|
switch (side)
|
|
|
|
{
|
|
|
|
case GTK_POS_LEFT:
|
2011-10-26 03:18:21 +00:00
|
|
|
left = find_attach_position (grid, GTK_ORIENTATION_HORIZONTAL, 0, height, FALSE);
|
|
|
|
left -= width;
|
2011-09-03 00:04:06 +00:00
|
|
|
top = 0;
|
|
|
|
break;
|
|
|
|
case GTK_POS_RIGHT:
|
2011-10-26 03:18:21 +00:00
|
|
|
left = find_attach_position (grid, GTK_ORIENTATION_HORIZONTAL, 0, height, TRUE);
|
2011-09-03 00:04:06 +00:00
|
|
|
top = 0;
|
|
|
|
break;
|
|
|
|
case GTK_POS_TOP:
|
|
|
|
left = 0;
|
2011-10-26 03:18:21 +00:00
|
|
|
top = find_attach_position (grid, GTK_ORIENTATION_VERTICAL, 0, width, FALSE);
|
|
|
|
top -= height;
|
2011-09-03 00:04:06 +00:00
|
|
|
break;
|
|
|
|
case GTK_POS_BOTTOM:
|
|
|
|
left = 0;
|
2011-10-26 03:18:21 +00:00
|
|
|
top = find_attach_position (grid, GTK_ORIENTATION_VERTICAL, 0, width, TRUE);
|
2011-09-03 00:04:06 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
g_assert_not_reached ();
|
|
|
|
}
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
grid_attach (grid, child, left, top, width, height);
|
|
|
|
}
|
|
|
|
|
2011-08-29 01:20:53 +00:00
|
|
|
/**
|
|
|
|
* gtk_grid_get_child_at:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @left: the left edge of the cell
|
|
|
|
* @top: the top edge of the cell
|
|
|
|
*
|
|
|
|
* Gets the child of @grid whose area covers the grid
|
|
|
|
* cell whose upper left corner is at @left, @top.
|
|
|
|
*
|
2012-05-16 16:00:04 +00:00
|
|
|
* Returns: (transfer none): the child at the given position, or %NULL
|
2011-08-29 01:20:53 +00:00
|
|
|
*
|
|
|
|
* Since: 3.2
|
|
|
|
*/
|
|
|
|
GtkWidget *
|
|
|
|
gtk_grid_get_child_at (GtkGrid *grid,
|
|
|
|
gint left,
|
|
|
|
gint top)
|
|
|
|
{
|
2012-06-02 23:40:41 +00:00
|
|
|
GtkGridPrivate *priv;
|
2011-08-29 01:20:53 +00:00
|
|
|
GtkGridChild *child;
|
|
|
|
GList *list;
|
|
|
|
|
2012-06-02 23:40:41 +00:00
|
|
|
g_return_val_if_fail (GTK_IS_GRID (grid), NULL);
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-08-29 01:20:53 +00:00
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
if (CHILD_LEFT (child) <= left &&
|
|
|
|
CHILD_LEFT (child) + CHILD_WIDTH (child) > left &&
|
|
|
|
CHILD_TOP (child) <= top &&
|
|
|
|
CHILD_TOP (child) + CHILD_HEIGHT (child) > top)
|
|
|
|
return child->widget;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-08-08 10:06:59 +00:00
|
|
|
/**
|
|
|
|
* gtk_grid_insert_row:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @position: the position to insert the row at
|
|
|
|
*
|
|
|
|
* Inserts a row at the specified position.
|
|
|
|
*
|
|
|
|
* Children which are attached at or below this position
|
|
|
|
* are moved one row down. Children which span across this
|
|
|
|
* position are grown to span the new row.
|
|
|
|
*
|
|
|
|
* Since: 3.2
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_insert_row (GtkGrid *grid,
|
|
|
|
gint position)
|
|
|
|
{
|
2012-06-02 23:45:05 +00:00
|
|
|
GtkGridPrivate *priv;
|
2011-08-08 10:06:59 +00:00
|
|
|
GtkGridChild *child;
|
|
|
|
GList *list;
|
|
|
|
gint top, height;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
|
2012-06-02 23:45:05 +00:00
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-08-08 10:06:59 +00:00
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
top = CHILD_TOP (child);
|
|
|
|
height = CHILD_HEIGHT (child);
|
|
|
|
|
|
|
|
if (top >= position)
|
|
|
|
{
|
|
|
|
CHILD_TOP (child) = top + 1;
|
|
|
|
gtk_container_child_notify (GTK_CONTAINER (grid), child->widget, "top-attach");
|
|
|
|
}
|
|
|
|
else if (top + height > position)
|
|
|
|
{
|
|
|
|
CHILD_HEIGHT (child) = height + 1;
|
|
|
|
gtk_container_child_notify (GTK_CONTAINER (grid), child->widget, "height");
|
|
|
|
}
|
|
|
|
}
|
2013-03-21 10:09:43 +00:00
|
|
|
|
|
|
|
for (list = priv->row_properties; list != NULL; list = list->next)
|
|
|
|
{
|
|
|
|
GtkGridRowProperties *prop = list->data;
|
|
|
|
|
|
|
|
if (prop->row >= position)
|
|
|
|
prop->row += 1;
|
|
|
|
}
|
2011-08-08 10:06:59 +00:00
|
|
|
}
|
|
|
|
|
2013-03-23 19:40:44 +00:00
|
|
|
/**
|
|
|
|
* gtk_grid_remove_row:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @position: the position of the row to remove
|
|
|
|
*
|
|
|
|
* Removes a row from the grid.
|
|
|
|
*
|
|
|
|
* Children that are placed in this row are removed,
|
|
|
|
* spanning children that overlap this row have their
|
|
|
|
* height reduced by one, and children below the row
|
|
|
|
* are moved up.
|
|
|
|
*
|
|
|
|
* Since: 3.10
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_remove_row (GtkGrid *grid,
|
|
|
|
gint position)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GList *list;
|
|
|
|
gint top, height;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
|
|
|
list = priv->children;
|
|
|
|
while (list)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
list = list->next;
|
|
|
|
|
|
|
|
top = CHILD_TOP (child);
|
|
|
|
height = CHILD_HEIGHT (child);
|
|
|
|
|
|
|
|
if (top <= position && top + height > position)
|
|
|
|
height--;
|
|
|
|
if (top > position)
|
|
|
|
top--;
|
|
|
|
|
|
|
|
if (height <= 0)
|
|
|
|
gtk_container_remove (GTK_CONTAINER (grid), child->widget);
|
|
|
|
else
|
|
|
|
gtk_container_child_set (GTK_CONTAINER (grid), child->widget,
|
|
|
|
"height", height,
|
|
|
|
"top-attach", top,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-08 10:06:59 +00:00
|
|
|
/**
|
|
|
|
* gtk_grid_insert_column:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @position: the position to insert the column at
|
|
|
|
*
|
|
|
|
* Inserts a column at the specified position.
|
|
|
|
*
|
|
|
|
* Children which are attached at or to the right of this position
|
|
|
|
* are moved one column to the right. Children which span across this
|
|
|
|
* position are grown to span the new column.
|
|
|
|
*
|
|
|
|
* Since: 3.2
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_insert_column (GtkGrid *grid,
|
|
|
|
gint position)
|
|
|
|
{
|
2012-06-02 23:45:05 +00:00
|
|
|
GtkGridPrivate *priv;
|
2011-08-08 10:06:59 +00:00
|
|
|
GtkGridChild *child;
|
|
|
|
GList *list;
|
|
|
|
gint left, width;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
|
2012-06-02 23:45:05 +00:00
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-08-08 10:06:59 +00:00
|
|
|
for (list = priv->children; list; list = list->next)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
|
|
|
|
left = CHILD_LEFT (child);
|
|
|
|
width = CHILD_WIDTH (child);
|
|
|
|
|
|
|
|
if (left >= position)
|
|
|
|
{
|
|
|
|
CHILD_LEFT (child) = left + 1;
|
|
|
|
gtk_container_child_notify (GTK_CONTAINER (grid), child->widget, "left-attach");
|
|
|
|
}
|
|
|
|
else if (left + width > position)
|
|
|
|
{
|
|
|
|
CHILD_WIDTH (child) = width + 1;
|
|
|
|
gtk_container_child_notify (GTK_CONTAINER (grid), child->widget, "width");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-23 19:40:44 +00:00
|
|
|
/**
|
|
|
|
* gtk_grid_remove_column:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @position: the position of the column to remove
|
|
|
|
*
|
|
|
|
* Removes a column from the grid.
|
|
|
|
*
|
|
|
|
* Children that are placed in this column are removed,
|
|
|
|
* spanning children that overlap this column have their
|
|
|
|
* width reduced by one, and children after the column
|
|
|
|
* are moved to the left.
|
|
|
|
*
|
|
|
|
* Since: 3.10
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_remove_column (GtkGrid *grid,
|
|
|
|
gint position)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
GtkGridChild *child;
|
|
|
|
GList *list;
|
|
|
|
gint left, width;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
|
|
|
list = priv->children;
|
|
|
|
while (list)
|
|
|
|
{
|
|
|
|
child = list->data;
|
|
|
|
list = list->next;
|
|
|
|
|
|
|
|
left = CHILD_LEFT (child);
|
|
|
|
width = CHILD_WIDTH (child);
|
|
|
|
|
|
|
|
if (left <= position && left + width > position)
|
|
|
|
width--;
|
|
|
|
if (left > position)
|
|
|
|
left--;
|
|
|
|
|
|
|
|
if (width <= 0)
|
|
|
|
gtk_container_remove (GTK_CONTAINER (grid), child->widget);
|
|
|
|
else
|
|
|
|
gtk_container_child_set (GTK_CONTAINER (grid), child->widget,
|
|
|
|
"width", width,
|
|
|
|
"left-attach", left,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-08 10:06:59 +00:00
|
|
|
/**
|
|
|
|
* gtk_grid_insert_next_to:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @sibling: the child of @grid that the new row or column will be
|
|
|
|
* placed next to
|
|
|
|
* @side: the side of @sibling that @child is positioned next to
|
|
|
|
*
|
|
|
|
* Inserts a row or column at the specified position.
|
|
|
|
*
|
|
|
|
* The new row or column is placed next to @sibling, on the side
|
|
|
|
* determined by @side. If @side is %GTK_POS_TOP or %GTK_POS_BOTTOM,
|
|
|
|
* a row is inserted. If @side is %GTK_POS_LEFT of %GTK_POS_RIGHT,
|
|
|
|
* a column is inserted.
|
|
|
|
*
|
|
|
|
* Since: 3.2
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_insert_next_to (GtkGrid *grid,
|
|
|
|
GtkWidget *sibling,
|
|
|
|
GtkPositionType side)
|
|
|
|
{
|
|
|
|
GtkGridChild *child;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (sibling));
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (sibling) == (GtkWidget*)grid);
|
|
|
|
|
|
|
|
child = find_grid_child (grid, sibling);
|
|
|
|
|
|
|
|
switch (side)
|
|
|
|
{
|
|
|
|
case GTK_POS_LEFT:
|
|
|
|
gtk_grid_insert_column (grid, CHILD_LEFT (child));
|
|
|
|
break;
|
|
|
|
case GTK_POS_RIGHT:
|
|
|
|
gtk_grid_insert_column (grid, CHILD_LEFT (child) + CHILD_WIDTH (child));
|
|
|
|
break;
|
|
|
|
case GTK_POS_TOP:
|
|
|
|
gtk_grid_insert_row (grid, CHILD_TOP (child));
|
|
|
|
break;
|
|
|
|
case GTK_POS_BOTTOM:
|
|
|
|
gtk_grid_insert_row (grid, CHILD_TOP (child) + CHILD_HEIGHT (child));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
g_assert_not_reached ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-27 14:44:31 +00:00
|
|
|
/**
|
|
|
|
* gtk_grid_set_row_homogeneous:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @homogeneous: %TRUE to make rows homogeneous
|
|
|
|
*
|
|
|
|
* Sets whether all rows of @grid will have the same height.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_set_row_homogeneous (GtkGrid *grid,
|
|
|
|
gboolean homogeneous)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-04-20 14:56:03 +00:00
|
|
|
/* Yes, homogeneous rows means all the columns have the same size */
|
|
|
|
if (COLUMNS (priv)->homogeneous != homogeneous)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2011-04-20 14:56:03 +00:00
|
|
|
COLUMNS (priv)->homogeneous = homogeneous;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
if (gtk_widget_get_visible (GTK_WIDGET (grid)))
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (grid));
|
|
|
|
|
|
|
|
g_object_notify (G_OBJECT (grid), "row-homogeneous");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_get_row_homogeneous:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
*
|
|
|
|
* Returns whether all rows of @grid have the same height.
|
|
|
|
*
|
|
|
|
* Returns: whether all rows of @grid have the same height.
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
gtk_grid_get_row_homogeneous (GtkGrid *grid)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
g_return_val_if_fail (GTK_IS_GRID (grid), FALSE);
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-04-20 14:56:03 +00:00
|
|
|
return COLUMNS (priv)->homogeneous;
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_set_column_homogeneous:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @homogeneous: %TRUE to make columns homogeneous
|
|
|
|
*
|
|
|
|
* Sets whether all columns of @grid will have the same width.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_set_column_homogeneous (GtkGrid *grid,
|
|
|
|
gboolean homogeneous)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-04-20 14:56:03 +00:00
|
|
|
/* Yes, homogeneous columns means all the rows have the same size */
|
|
|
|
if (ROWS (priv)->homogeneous != homogeneous)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2011-04-20 14:56:03 +00:00
|
|
|
ROWS (priv)->homogeneous = homogeneous;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
if (gtk_widget_get_visible (GTK_WIDGET (grid)))
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (grid));
|
|
|
|
|
|
|
|
g_object_notify (G_OBJECT (grid), "column-homogeneous");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_get_column_homogeneous:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
*
|
|
|
|
* Returns whether all columns of @grid have the same width.
|
|
|
|
*
|
|
|
|
* Returns: whether all columns of @grid have the same width.
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
gtk_grid_get_column_homogeneous (GtkGrid *grid)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
g_return_val_if_fail (GTK_IS_GRID (grid), FALSE);
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-04-20 14:56:03 +00:00
|
|
|
return ROWS (priv)->homogeneous;
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_set_row_spacing:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @spacing: the amount of space to insert between rows
|
|
|
|
*
|
|
|
|
* Sets the amount of space between rows of @grid.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_set_row_spacing (GtkGrid *grid,
|
|
|
|
guint spacing)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
g_return_if_fail (spacing <= G_MAXINT16);
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-08-30 13:11:02 +00:00
|
|
|
if (COLUMNS (priv)->spacing != spacing)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2011-08-30 13:11:02 +00:00
|
|
|
COLUMNS (priv)->spacing = spacing;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
if (gtk_widget_get_visible (GTK_WIDGET (grid)))
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (grid));
|
|
|
|
|
|
|
|
g_object_notify (G_OBJECT (grid), "row-spacing");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_get_row_spacing:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
*
|
|
|
|
* Returns the amount of space between the rows of @grid.
|
|
|
|
*
|
|
|
|
* Returns: the row spacing of @grid
|
|
|
|
*/
|
|
|
|
guint
|
|
|
|
gtk_grid_get_row_spacing (GtkGrid *grid)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
g_return_val_if_fail (GTK_IS_GRID (grid), 0);
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-08-30 13:11:02 +00:00
|
|
|
return COLUMNS (priv)->spacing;
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_set_column_spacing:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @spacing: the amount of space to insert between columns
|
|
|
|
*
|
|
|
|
* Sets the amount of space between columns of @grid.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_set_column_spacing (GtkGrid *grid,
|
|
|
|
guint spacing)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
g_return_if_fail (spacing <= G_MAXINT16);
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-08-30 13:11:02 +00:00
|
|
|
if (ROWS (priv)->spacing != spacing)
|
2010-09-27 14:44:31 +00:00
|
|
|
{
|
2011-08-30 13:11:02 +00:00
|
|
|
ROWS (priv)->spacing = spacing;
|
2010-09-27 14:44:31 +00:00
|
|
|
|
|
|
|
if (gtk_widget_get_visible (GTK_WIDGET (grid)))
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (grid));
|
|
|
|
|
|
|
|
g_object_notify (G_OBJECT (grid), "column-spacing");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_get_column_spacing:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
*
|
|
|
|
* Returns the amount of space between the columns of @grid.
|
|
|
|
*
|
|
|
|
* Returns: the column spacing of @grid
|
|
|
|
*/
|
|
|
|
guint
|
|
|
|
gtk_grid_get_column_spacing (GtkGrid *grid)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GTK_IS_GRID (grid), 0);
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
2011-08-30 13:11:02 +00:00
|
|
|
return ROWS (priv)->spacing;
|
2010-09-27 14:44:31 +00:00
|
|
|
}
|
2013-03-21 10:09:43 +00:00
|
|
|
|
|
|
|
static GtkGridRowProperties *
|
|
|
|
find_row_properties (GtkGrid *grid,
|
|
|
|
gint row)
|
|
|
|
{
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
for (l = grid->priv->row_properties; l != NULL; l = l->next)
|
|
|
|
{
|
|
|
|
GtkGridRowProperties *prop = l->data;
|
|
|
|
if (prop->row == row)
|
|
|
|
return prop;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_grid_row_properties_free (GtkGridRowProperties *props)
|
|
|
|
{
|
|
|
|
g_slice_free (GtkGridRowProperties, props);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkGridRowProperties *
|
|
|
|
get_row_properties_or_create (GtkGrid *grid,
|
|
|
|
gint row)
|
|
|
|
{
|
|
|
|
GtkGridRowProperties *props;
|
|
|
|
GtkGridPrivate *priv = grid->priv;
|
|
|
|
|
|
|
|
props = find_row_properties (grid, row);
|
|
|
|
if (props)
|
|
|
|
return props;
|
|
|
|
|
|
|
|
props = g_slice_new (GtkGridRowProperties);
|
|
|
|
*props = gtk_grid_row_properties_default;
|
|
|
|
props->row = row;
|
|
|
|
|
|
|
|
priv->row_properties =
|
|
|
|
g_list_prepend (priv->row_properties, props);
|
|
|
|
|
|
|
|
return props;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const GtkGridRowProperties *
|
|
|
|
get_row_properties_or_default (GtkGrid *grid,
|
|
|
|
gint row)
|
|
|
|
{
|
|
|
|
GtkGridRowProperties *props;
|
|
|
|
|
|
|
|
props = find_row_properties (grid, row);
|
|
|
|
if (props)
|
|
|
|
return props;
|
|
|
|
return >k_grid_row_properties_default;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_set_row_baseline_position:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @row: a row index
|
|
|
|
* @pos: a #GtkBaselinePosition
|
|
|
|
*
|
|
|
|
* Sets how the baseline should be positioned on @row of the
|
|
|
|
* grid, in case that row is assigned more space than is requested.
|
2013-03-27 08:39:44 +00:00
|
|
|
*
|
|
|
|
* Since: 3.10
|
2013-03-21 10:09:43 +00:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_set_row_baseline_position (GtkGrid *grid,
|
|
|
|
gint row,
|
|
|
|
GtkBaselinePosition pos)
|
|
|
|
{
|
|
|
|
GtkGridRowProperties *props;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
|
|
|
|
props = get_row_properties_or_create (grid, row);
|
|
|
|
|
|
|
|
if (props->baseline_position != pos)
|
|
|
|
{
|
|
|
|
props->baseline_position = pos;
|
|
|
|
|
|
|
|
if (gtk_widget_get_visible (GTK_WIDGET (grid)))
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (grid));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_get_row_baseline_position:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @row: a row index
|
|
|
|
*
|
|
|
|
* Returns the baseline position of @row as set
|
|
|
|
* by gtk_grid_set_row_baseline_position() or the default value
|
|
|
|
* %GTK_BASELINE_POSITION_CENTER.
|
|
|
|
*
|
|
|
|
* Returns: the baseline position of @row
|
2013-03-27 08:39:44 +00:00
|
|
|
*
|
|
|
|
* Since: 3.10
|
2013-03-21 10:09:43 +00:00
|
|
|
*/
|
|
|
|
GtkBaselinePosition
|
|
|
|
gtk_grid_get_row_baseline_position (GtkGrid *grid,
|
|
|
|
gint row)
|
|
|
|
{
|
|
|
|
const GtkGridRowProperties *props;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GTK_IS_GRID (grid), GTK_BASELINE_POSITION_CENTER);
|
|
|
|
|
|
|
|
props = get_row_properties_or_default (grid, row);
|
|
|
|
|
|
|
|
return props->baseline_position;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_grid_set_baseline_row:
|
|
|
|
* @grid: a #GtkGrid
|
|
|
|
* @row: the row index
|
|
|
|
*
|
|
|
|
* Sets which row defines the global baseline for the entire grid.
|
|
|
|
* Each row in the grid can have its own local baseline, but only
|
|
|
|
* one of those is global, meaning it will be the baseline in the
|
|
|
|
* parent of the @grid.
|
|
|
|
*
|
|
|
|
* Returns: the row index defining the global baseline
|
2013-03-27 08:39:44 +00:00
|
|
|
*
|
|
|
|
* Since: 3.10
|
2013-03-21 10:09:43 +00:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_grid_set_baseline_row (GtkGrid *grid,
|
|
|
|
gint row)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_GRID (grid));
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
|
|
|
if (priv->baseline_row != row)
|
|
|
|
{
|
|
|
|
priv->baseline_row = row;
|
|
|
|
|
|
|
|
if (gtk_widget_get_visible (GTK_WIDGET (grid)))
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (grid));
|
|
|
|
g_object_notify (G_OBJECT (grid), "baseline-row");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-04-24 17:10:35 +00:00
|
|
|
* gtk_grid_get_baseline_row:
|
2013-03-21 10:09:43 +00:00
|
|
|
* @grid: a #GtkGrid
|
|
|
|
*
|
|
|
|
* Returns which row defines the global baseline of @grid.
|
|
|
|
*
|
|
|
|
* Returns: the row index defining the global baseline
|
2013-03-27 08:39:44 +00:00
|
|
|
*
|
|
|
|
* Since: 3.10
|
2013-03-21 10:09:43 +00:00
|
|
|
*/
|
|
|
|
gint
|
|
|
|
gtk_grid_get_baseline_row (GtkGrid *grid)
|
|
|
|
{
|
|
|
|
GtkGridPrivate *priv;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GTK_IS_GRID (grid), 0);
|
|
|
|
|
|
|
|
priv = grid->priv;
|
|
|
|
|
|
|
|
return priv->baseline_row;
|
|
|
|
}
|