2008-07-01 22:57:50 +00:00
|
|
|
|
/* GTK - The GIMP Toolkit
|
1997-11-24 22:37:52 +00:00
|
|
|
|
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
|
|
|
|
*
|
|
|
|
|
* This library is free software; you can redistribute it and/or
|
2000-07-26 11:33:08 +00:00
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
1997-11-24 22:37:52 +00:00
|
|
|
|
* 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
|
2000-07-26 11:33:08 +00:00
|
|
|
|
* Lesser General Public License for more details.
|
1997-11-24 22:37:52 +00:00
|
|
|
|
*
|
2000-07-26 11:33:08 +00:00
|
|
|
|
* 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/>.
|
1997-11-24 22:37:52 +00:00
|
|
|
|
*/
|
1999-02-24 07:37:18 +00:00
|
|
|
|
|
|
|
|
|
/*
|
2000-07-26 11:33:08 +00:00
|
|
|
|
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
1999-02-24 07:37:18 +00:00
|
|
|
|
* file for a list of people on the GTK+ Team. See the ChangeLog
|
|
|
|
|
* files for a list of changes. These files are distributed with
|
2008-10-07 07:44:06 +00:00
|
|
|
|
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
1999-02-24 07:37:18 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2010-03-09 22:51:03 +00:00
|
|
|
|
/**
|
|
|
|
|
* SECTION:gtkbox
|
Box, Grid: Improve various bits of documentation
Issue #1495 showed that the docs of GtkGrid retain outdated implications
that (as was once, but is no longer, the case) it is intended to replace
GtkBox, by discussing HfW and widget properties in a way that suggests
GtkBox can't handle them. But of course it does, and it's preferable for
simple single-row/column cases. Worse, we said GtkGrid “provides exactly
the same functionality” for the latter case, but the original point of
that Issues was that it doesn’t, at least for CSS positional selectors!
Box:
• Use an actually meaningful @Short_description.
• Remove unhelpful @See_also references to unrelated containers.
• Remove references to “rectangular area”: it might be another shape
via CSS, or “rectangular” might falsely imply 2 dimensions of children.
• Mention Orientable:orientation.
• Emphasise usefulness of :[hv]align for allocating in the other axis.
• Don’t say that Grid “provides exactly the same functionality” for a
single row or column, since (A) it is overkill for that case and (B)
said Issue proved that it *doesn’t* for CSS child order, for example.
Grid:
• Don’t dwell on widget properties and height-for-width in a way that
wrongly implies that Box can’t handle those (or Grid can better). In
fact, just get rid of that bit altogether: Box handles them fine, and
such wording was only needed years ago for migration from GTK+ 2 to 3.
• Point to GtkBox as being preferred for the simple row/column use case.
2018-12-04 20:37:08 +00:00
|
|
|
|
* @Short_description: A container for packing widgets in a single row or column
|
2010-03-09 22:51:03 +00:00
|
|
|
|
* @Title: GtkBox
|
Box, Grid: Improve various bits of documentation
Issue #1495 showed that the docs of GtkGrid retain outdated implications
that (as was once, but is no longer, the case) it is intended to replace
GtkBox, by discussing HfW and widget properties in a way that suggests
GtkBox can't handle them. But of course it does, and it's preferable for
simple single-row/column cases. Worse, we said GtkGrid “provides exactly
the same functionality” for the latter case, but the original point of
that Issues was that it doesn’t, at least for CSS positional selectors!
Box:
• Use an actually meaningful @Short_description.
• Remove unhelpful @See_also references to unrelated containers.
• Remove references to “rectangular area”: it might be another shape
via CSS, or “rectangular” might falsely imply 2 dimensions of children.
• Mention Orientable:orientation.
• Emphasise usefulness of :[hv]align for allocating in the other axis.
• Don’t say that Grid “provides exactly the same functionality” for a
single row or column, since (A) it is overkill for that case and (B)
said Issue proved that it *doesn’t* for CSS child order, for example.
Grid:
• Don’t dwell on widget properties and height-for-width in a way that
wrongly implies that Box can’t handle those (or Grid can better). In
fact, just get rid of that bit altogether: Box handles them fine, and
such wording was only needed years ago for migration from GTK+ 2 to 3.
• Point to GtkBox as being preferred for the simple row/column use case.
2018-12-04 20:37:08 +00:00
|
|
|
|
* @See_also: #GtkGrid
|
2010-03-09 22:51:03 +00:00
|
|
|
|
*
|
Box, Grid: Improve various bits of documentation
Issue #1495 showed that the docs of GtkGrid retain outdated implications
that (as was once, but is no longer, the case) it is intended to replace
GtkBox, by discussing HfW and widget properties in a way that suggests
GtkBox can't handle them. But of course it does, and it's preferable for
simple single-row/column cases. Worse, we said GtkGrid “provides exactly
the same functionality” for the latter case, but the original point of
that Issues was that it doesn’t, at least for CSS positional selectors!
Box:
• Use an actually meaningful @Short_description.
• Remove unhelpful @See_also references to unrelated containers.
• Remove references to “rectangular area”: it might be another shape
via CSS, or “rectangular” might falsely imply 2 dimensions of children.
• Mention Orientable:orientation.
• Emphasise usefulness of :[hv]align for allocating in the other axis.
• Don’t say that Grid “provides exactly the same functionality” for a
single row or column, since (A) it is overkill for that case and (B)
said Issue proved that it *doesn’t* for CSS child order, for example.
Grid:
• Don’t dwell on widget properties and height-for-width in a way that
wrongly implies that Box can’t handle those (or Grid can better). In
fact, just get rid of that bit altogether: Box handles them fine, and
such wording was only needed years ago for migration from GTK+ 2 to 3.
• Point to GtkBox as being preferred for the simple row/column use case.
2018-12-04 20:37:08 +00:00
|
|
|
|
* The GtkBox widget arranges child widgets into a single row or column,
|
|
|
|
|
* depending upon the value of its #GtkOrientable:orientation property. Within
|
|
|
|
|
* the other dimension, all children are allocated the same size. Of course,
|
|
|
|
|
* the #GtkWidget:halign and #GtkWidget:valign properties can be used on
|
|
|
|
|
* the children to influence their allocation.
|
2010-03-09 22:51:03 +00:00
|
|
|
|
*
|
2020-05-08 16:21:30 +00:00
|
|
|
|
* Use repeated calls to gtk_box_append() to pack widgets into a GtkBox
|
|
|
|
|
* from start to end. Use gtk_box_remove() to remove widgets from the GtkBox.
|
|
|
|
|
* gtk_box_insert_child_after() can be used to add a child at a particular position.
|
2010-03-09 22:51:03 +00:00
|
|
|
|
*
|
|
|
|
|
* Use gtk_box_set_homogeneous() to specify whether or not all children
|
|
|
|
|
* of the GtkBox are forced to get the same amount of space.
|
|
|
|
|
*
|
|
|
|
|
* Use gtk_box_set_spacing() to determine how much space will be
|
2010-10-15 14:58:35 +00:00
|
|
|
|
* minimally placed between all children in the GtkBox. Note that
|
2016-10-02 15:37:22 +00:00
|
|
|
|
* spacing is added between the children.
|
2010-03-09 22:51:03 +00:00
|
|
|
|
*
|
2019-01-22 23:44:34 +00:00
|
|
|
|
* Use gtk_box_reorder_child_after() to move a child to a different
|
2010-03-09 22:51:03 +00:00
|
|
|
|
* place in the box.
|
|
|
|
|
*
|
2015-12-06 06:53:29 +00:00
|
|
|
|
* # CSS nodes
|
2015-11-19 12:07:26 +00:00
|
|
|
|
*
|
2016-01-13 05:39:53 +00:00
|
|
|
|
* GtkBox uses a single CSS node with name box.
|
2020-11-12 00:36:32 +00:00
|
|
|
|
*
|
|
|
|
|
* # Accessibility
|
|
|
|
|
*
|
|
|
|
|
* GtkBox uses the %GTK_ACCESSIBLE_ROLE_GROUP role.
|
2010-03-09 22:51:03 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2008-06-22 14:28:52 +00:00
|
|
|
|
#include "config.h"
|
2008-10-07 07:44:06 +00:00
|
|
|
|
|
1997-11-24 22:37:52 +00:00
|
|
|
|
#include "gtkbox.h"
|
2019-01-25 12:19:34 +00:00
|
|
|
|
#include "gtkboxlayout.h"
|
2020-05-08 19:31:02 +00:00
|
|
|
|
#include "gtkbuildable.h"
|
2012-03-16 19:01:50 +00:00
|
|
|
|
#include "gtkintl.h"
|
2008-10-07 07:44:06 +00:00
|
|
|
|
#include "gtkorientable.h"
|
2005-03-22 02:14:55 +00:00
|
|
|
|
#include "gtkprivate.h"
|
2012-03-16 19:01:50 +00:00
|
|
|
|
#include "gtktypebuiltins.h"
|
|
|
|
|
#include "gtksizerequest.h"
|
2016-12-10 02:47:53 +00:00
|
|
|
|
#include "gtkstylecontextprivate.h"
|
2012-03-21 08:37:09 +00:00
|
|
|
|
#include "gtkwidgetprivate.h"
|
2010-07-09 17:22:23 +00:00
|
|
|
|
|
1997-11-24 22:37:52 +00:00
|
|
|
|
|
1998-01-17 07:52:38 +00:00
|
|
|
|
enum {
|
2001-03-23 23:39:24 +00:00
|
|
|
|
PROP_0,
|
|
|
|
|
PROP_SPACING,
|
2013-03-05 14:20:20 +00:00
|
|
|
|
PROP_HOMOGENEOUS,
|
2014-06-07 14:49:38 +00:00
|
|
|
|
PROP_BASELINE_POSITION,
|
|
|
|
|
|
|
|
|
|
/* orientable */
|
2014-06-07 14:59:14 +00:00
|
|
|
|
PROP_ORIENTATION,
|
|
|
|
|
LAST_PROP = PROP_ORIENTATION
|
1998-01-17 07:52:38 +00:00
|
|
|
|
};
|
1997-11-24 22:37:52 +00:00
|
|
|
|
|
2018-12-28 16:17:16 +00:00
|
|
|
|
typedef struct
|
2008-10-07 07:44:06 +00:00
|
|
|
|
{
|
2011-04-12 16:24:56 +00:00
|
|
|
|
GtkOrientation orientation;
|
2010-05-24 23:11:54 +00:00
|
|
|
|
gint16 spacing;
|
|
|
|
|
|
|
|
|
|
guint homogeneous : 1;
|
2013-03-05 14:20:20 +00:00
|
|
|
|
guint baseline_pos : 2;
|
2018-12-28 16:17:16 +00:00
|
|
|
|
} GtkBoxPrivate;
|
2008-10-07 07:44:06 +00:00
|
|
|
|
|
2014-06-07 14:59:14 +00:00
|
|
|
|
static GParamSpec *props[LAST_PROP] = { NULL, };
|
|
|
|
|
|
2020-05-08 19:31:02 +00:00
|
|
|
|
static void gtk_box_buildable_iface_init (GtkBuildableIface *iface);
|
|
|
|
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GtkBox, gtk_box, GTK_TYPE_WIDGET,
|
2013-06-27 19:02:52 +00:00
|
|
|
|
G_ADD_PRIVATE (GtkBox)
|
2020-05-08 19:31:02 +00:00
|
|
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_ORIENTABLE, NULL)
|
|
|
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_BUILDABLE,
|
|
|
|
|
gtk_box_buildable_iface_init))
|
1997-11-24 22:37:52 +00:00
|
|
|
|
|
|
|
|
|
|
2008-10-07 07:44:06 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_box_set_property (GObject *object,
|
|
|
|
|
guint prop_id,
|
|
|
|
|
const GValue *value,
|
|
|
|
|
GParamSpec *pspec)
|
1998-01-17 07:52:38 +00:00
|
|
|
|
{
|
2008-10-07 07:44:06 +00:00
|
|
|
|
GtkBox *box = GTK_BOX (object);
|
2018-02-07 19:03:25 +00:00
|
|
|
|
GtkBoxPrivate *priv = gtk_box_get_instance_private (box);
|
2019-01-25 12:19:34 +00:00
|
|
|
|
GtkLayoutManager *box_layout = gtk_widget_get_layout_manager (GTK_WIDGET (box));
|
1998-06-28 07:46:10 +00:00
|
|
|
|
|
2001-03-23 23:39:24 +00:00
|
|
|
|
switch (prop_id)
|
1998-01-17 07:52:38 +00:00
|
|
|
|
{
|
2008-10-07 07:44:06 +00:00
|
|
|
|
case PROP_ORIENTATION:
|
2014-06-07 14:59:14 +00:00
|
|
|
|
{
|
|
|
|
|
GtkOrientation orientation = g_value_get_enum (value);
|
2018-02-07 19:03:25 +00:00
|
|
|
|
if (priv->orientation != orientation)
|
2014-06-07 14:59:14 +00:00
|
|
|
|
{
|
2018-02-07 19:03:25 +00:00
|
|
|
|
priv->orientation = orientation;
|
2019-01-25 12:19:34 +00:00
|
|
|
|
gtk_orientable_set_orientation (GTK_ORIENTABLE (box_layout),
|
|
|
|
|
priv->orientation);
|
2020-04-23 16:48:29 +00:00
|
|
|
|
gtk_widget_update_orientation (GTK_WIDGET (box), priv->orientation);
|
2014-06-07 14:59:14 +00:00
|
|
|
|
g_object_notify (object, "orientation");
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-10-07 07:44:06 +00:00
|
|
|
|
break;
|
2001-03-23 23:39:24 +00:00
|
|
|
|
case PROP_SPACING:
|
|
|
|
|
gtk_box_set_spacing (box, g_value_get_int (value));
|
1998-01-17 07:52:38 +00:00
|
|
|
|
break;
|
2013-03-05 14:20:20 +00:00
|
|
|
|
case PROP_BASELINE_POSITION:
|
|
|
|
|
gtk_box_set_baseline_position (box, g_value_get_enum (value));
|
|
|
|
|
break;
|
2001-03-23 23:39:24 +00:00
|
|
|
|
case PROP_HOMOGENEOUS:
|
|
|
|
|
gtk_box_set_homogeneous (box, g_value_get_boolean (value));
|
1998-01-17 07:52:38 +00:00
|
|
|
|
break;
|
1998-01-21 23:03:11 +00:00
|
|
|
|
default:
|
2001-03-23 23:39:24 +00:00
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
1998-01-21 23:03:11 +00:00
|
|
|
|
break;
|
1998-01-17 07:52:38 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-07 07:44:06 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_box_get_property (GObject *object,
|
|
|
|
|
guint prop_id,
|
|
|
|
|
GValue *value,
|
|
|
|
|
GParamSpec *pspec)
|
1998-01-17 07:52:38 +00:00
|
|
|
|
{
|
2008-10-07 07:44:06 +00:00
|
|
|
|
GtkBox *box = GTK_BOX (object);
|
2018-02-07 19:03:25 +00:00
|
|
|
|
GtkBoxPrivate *priv = gtk_box_get_instance_private (box);
|
2019-01-25 12:19:34 +00:00
|
|
|
|
GtkBoxLayout *box_layout = GTK_BOX_LAYOUT (gtk_widget_get_layout_manager (GTK_WIDGET (box)));
|
1998-06-28 07:46:10 +00:00
|
|
|
|
|
2001-03-23 23:39:24 +00:00
|
|
|
|
switch (prop_id)
|
1998-01-17 07:52:38 +00:00
|
|
|
|
{
|
2008-10-07 07:44:06 +00:00
|
|
|
|
case PROP_ORIENTATION:
|
2018-02-07 19:03:25 +00:00
|
|
|
|
g_value_set_enum (value, priv->orientation);
|
2008-10-07 07:44:06 +00:00
|
|
|
|
break;
|
2001-03-23 23:39:24 +00:00
|
|
|
|
case PROP_SPACING:
|
2019-01-25 12:19:34 +00:00
|
|
|
|
g_value_set_int (value, gtk_box_layout_get_spacing (box_layout));
|
1998-01-17 07:52:38 +00:00
|
|
|
|
break;
|
2013-03-05 14:20:20 +00:00
|
|
|
|
case PROP_BASELINE_POSITION:
|
2019-01-25 12:19:34 +00:00
|
|
|
|
g_value_set_enum (value, gtk_box_layout_get_baseline_position (box_layout));
|
2013-03-05 14:20:20 +00:00
|
|
|
|
break;
|
2001-03-23 23:39:24 +00:00
|
|
|
|
case PROP_HOMOGENEOUS:
|
2019-01-25 12:19:34 +00:00
|
|
|
|
g_value_set_boolean (value, gtk_box_layout_get_homogeneous (box_layout));
|
1998-01-17 07:52:38 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2001-03-23 23:39:24 +00:00
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
1998-01-17 07:52:38 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-05 07:08:45 +00:00
|
|
|
|
static void
|
2020-05-08 19:31:02 +00:00
|
|
|
|
gtk_box_compute_expand (GtkWidget *widget,
|
|
|
|
|
gboolean *hexpand_p,
|
|
|
|
|
gboolean *vexpand_p)
|
2020-02-05 07:08:45 +00:00
|
|
|
|
{
|
2020-05-08 19:31:02 +00:00
|
|
|
|
GtkWidget *w;
|
|
|
|
|
gboolean hexpand = FALSE;
|
|
|
|
|
gboolean vexpand = FALSE;
|
2020-02-05 07:08:45 +00:00
|
|
|
|
|
2020-05-08 19:31:02 +00:00
|
|
|
|
for (w = gtk_widget_get_first_child (widget);
|
|
|
|
|
w != NULL;
|
|
|
|
|
w = gtk_widget_get_next_sibling (w))
|
|
|
|
|
{
|
|
|
|
|
hexpand = hexpand || gtk_widget_compute_expand (w, GTK_ORIENTATION_HORIZONTAL);
|
|
|
|
|
vexpand = vexpand || gtk_widget_compute_expand (w, GTK_ORIENTATION_VERTICAL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*hexpand_p = hexpand;
|
|
|
|
|
*vexpand_p = vexpand;
|
2020-02-05 07:08:45 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-08 19:31:02 +00:00
|
|
|
|
static GtkSizeRequestMode
|
|
|
|
|
gtk_box_get_request_mode (GtkWidget *widget)
|
2020-02-05 07:08:45 +00:00
|
|
|
|
{
|
2020-05-08 19:31:02 +00:00
|
|
|
|
GtkWidget *w;
|
|
|
|
|
int wfh = 0, hfw = 0;
|
2020-02-05 07:08:45 +00:00
|
|
|
|
|
2020-05-08 19:31:02 +00:00
|
|
|
|
for (w = gtk_widget_get_first_child (widget);
|
|
|
|
|
w != NULL;
|
|
|
|
|
w = gtk_widget_get_next_sibling (w))
|
2020-02-05 07:08:45 +00:00
|
|
|
|
{
|
2020-05-08 19:31:02 +00:00
|
|
|
|
GtkSizeRequestMode mode = gtk_widget_get_request_mode (w);
|
|
|
|
|
|
|
|
|
|
switch (mode)
|
|
|
|
|
{
|
|
|
|
|
case GTK_SIZE_REQUEST_HEIGHT_FOR_WIDTH:
|
|
|
|
|
hfw ++;
|
|
|
|
|
break;
|
|
|
|
|
case GTK_SIZE_REQUEST_WIDTH_FOR_HEIGHT:
|
|
|
|
|
wfh ++;
|
|
|
|
|
break;
|
|
|
|
|
case GTK_SIZE_REQUEST_CONSTANT_SIZE:
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-02-05 07:08:45 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-08 19:31:02 +00:00
|
|
|
|
if (hfw == 0 && wfh == 0)
|
|
|
|
|
return GTK_SIZE_REQUEST_CONSTANT_SIZE;
|
|
|
|
|
else
|
|
|
|
|
return wfh > hfw ?
|
|
|
|
|
GTK_SIZE_REQUEST_WIDTH_FOR_HEIGHT :
|
|
|
|
|
GTK_SIZE_REQUEST_HEIGHT_FOR_WIDTH;
|
2020-02-05 07:08:45 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-08 19:31:02 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_box_dispose (GObject *object)
|
1998-06-16 05:20:05 +00:00
|
|
|
|
{
|
2020-05-08 19:31:02 +00:00
|
|
|
|
GtkWidget *child;
|
|
|
|
|
|
|
|
|
|
while ((child = gtk_widget_get_first_child (GTK_WIDGET (object))))
|
|
|
|
|
gtk_widget_unparent (child);
|
|
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (gtk_box_parent_class)->dispose (object);
|
1998-06-16 05:20:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-05 07:08:45 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_box_class_init (GtkBoxClass *class)
|
|
|
|
|
{
|
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
|
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (class);
|
|
|
|
|
|
|
|
|
|
object_class->set_property = gtk_box_set_property;
|
|
|
|
|
object_class->get_property = gtk_box_get_property;
|
2020-05-08 19:31:02 +00:00
|
|
|
|
object_class->dispose = gtk_box_dispose;
|
2020-02-05 07:08:45 +00:00
|
|
|
|
|
2020-05-08 19:31:02 +00:00
|
|
|
|
widget_class->focus = gtk_widget_focus_child;
|
|
|
|
|
widget_class->compute_expand = gtk_box_compute_expand;
|
|
|
|
|
widget_class->get_request_mode = gtk_box_get_request_mode;
|
2020-02-05 07:08:45 +00:00
|
|
|
|
|
|
|
|
|
g_object_class_override_property (object_class,
|
|
|
|
|
PROP_ORIENTATION,
|
|
|
|
|
"orientation");
|
|
|
|
|
|
|
|
|
|
props[PROP_SPACING] =
|
|
|
|
|
g_param_spec_int ("spacing",
|
|
|
|
|
P_("Spacing"),
|
|
|
|
|
P_("The amount of space between children"),
|
|
|
|
|
0, G_MAXINT, 0,
|
|
|
|
|
GTK_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY);
|
|
|
|
|
|
|
|
|
|
props[PROP_HOMOGENEOUS] =
|
|
|
|
|
g_param_spec_boolean ("homogeneous",
|
|
|
|
|
P_("Homogeneous"),
|
|
|
|
|
P_("Whether the children should all be the same size"),
|
|
|
|
|
FALSE,
|
|
|
|
|
GTK_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY);
|
|
|
|
|
|
|
|
|
|
props[PROP_BASELINE_POSITION] =
|
|
|
|
|
g_param_spec_enum ("baseline-position",
|
|
|
|
|
P_("Baseline position"),
|
|
|
|
|
P_("The position of the baseline aligned widgets if extra space is available"),
|
|
|
|
|
GTK_TYPE_BASELINE_POSITION,
|
|
|
|
|
GTK_BASELINE_POSITION_CENTER,
|
|
|
|
|
GTK_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY);
|
|
|
|
|
|
|
|
|
|
g_object_class_install_properties (object_class, LAST_PROP, props);
|
|
|
|
|
|
|
|
|
|
gtk_widget_class_set_layout_manager_type (widget_class, GTK_TYPE_BOX_LAYOUT);
|
|
|
|
|
gtk_widget_class_set_css_name (widget_class, I_("box"));
|
2020-11-11 17:54:57 +00:00
|
|
|
|
gtk_widget_class_set_accessible_role (widget_class, GTK_ACCESSIBLE_ROLE_GROUP);
|
2020-02-05 07:08:45 +00:00
|
|
|
|
}
|
2015-11-18 00:47:53 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_box_init (GtkBox *box)
|
|
|
|
|
{
|
2018-02-07 19:03:25 +00:00
|
|
|
|
GtkBoxPrivate *priv = gtk_box_get_instance_private (box);
|
2015-11-18 00:47:53 +00:00
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
priv->orientation = GTK_ORIENTATION_HORIZONTAL;
|
2020-04-23 16:48:29 +00:00
|
|
|
|
gtk_widget_update_orientation (GTK_WIDGET (box), priv->orientation);
|
2015-11-18 00:47:53 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-08 19:31:02 +00:00
|
|
|
|
static GtkBuildableIface *parent_buildable_iface;
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtk_box_buildable_add_child (GtkBuildable *buildable,
|
|
|
|
|
GtkBuilder *builder,
|
|
|
|
|
GObject *child,
|
2020-07-24 18:40:36 +00:00
|
|
|
|
const char *type)
|
2020-05-08 19:31:02 +00:00
|
|
|
|
{
|
|
|
|
|
if (GTK_IS_WIDGET (child))
|
|
|
|
|
gtk_box_append (GTK_BOX (buildable), GTK_WIDGET (child));
|
|
|
|
|
else
|
|
|
|
|
parent_buildable_iface->add_child (buildable, builder, child, type);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtk_box_buildable_iface_init (GtkBuildableIface *iface)
|
|
|
|
|
{
|
|
|
|
|
parent_buildable_iface = g_type_interface_peek_parent (iface);
|
|
|
|
|
|
|
|
|
|
iface->add_child = gtk_box_buildable_add_child;
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-07 07:44:06 +00:00
|
|
|
|
/**
|
|
|
|
|
* gtk_box_new:
|
2020-12-21 14:42:01 +00:00
|
|
|
|
* @orientation: the box’s orientation
|
|
|
|
|
* @spacing: the number of pixels to place by default between children
|
2008-10-07 07:44:06 +00:00
|
|
|
|
*
|
2010-05-30 03:31:50 +00:00
|
|
|
|
* Creates a new #GtkBox.
|
2008-10-07 07:44:06 +00:00
|
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
|
* Returns: a new #GtkBox.
|
2008-10-07 07:44:06 +00:00
|
|
|
|
**/
|
2008-10-07 09:07:27 +00:00
|
|
|
|
GtkWidget*
|
2010-05-25 22:55:15 +00:00
|
|
|
|
gtk_box_new (GtkOrientation orientation,
|
2020-07-24 13:54:49 +00:00
|
|
|
|
int spacing)
|
2008-10-07 07:44:06 +00:00
|
|
|
|
{
|
|
|
|
|
return g_object_new (GTK_TYPE_BOX,
|
|
|
|
|
"orientation", orientation,
|
2019-01-25 12:19:34 +00:00
|
|
|
|
"spacing", spacing,
|
2008-10-07 07:44:06 +00:00
|
|
|
|
NULL);
|
|
|
|
|
}
|
|
|
|
|
|
2007-06-10 02:53:17 +00:00
|
|
|
|
/**
|
|
|
|
|
* gtk_box_set_homogeneous:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
* @homogeneous: a boolean value, %TRUE to create equal allotments,
|
|
|
|
|
* %FALSE for variable allotments
|
2010-11-02 09:43:42 +00:00
|
|
|
|
*
|
|
|
|
|
* Sets the #GtkBox:homogeneous property of @box, controlling
|
|
|
|
|
* whether or not all children of @box are given equal space
|
2007-06-10 02:53:17 +00:00
|
|
|
|
* in the box.
|
|
|
|
|
*/
|
1997-11-24 22:37:52 +00:00
|
|
|
|
void
|
1998-06-16 05:20:05 +00:00
|
|
|
|
gtk_box_set_homogeneous (GtkBox *box,
|
2020-12-27 17:14:57 +00:00
|
|
|
|
gboolean homogeneous)
|
1997-11-24 22:37:52 +00:00
|
|
|
|
{
|
2019-01-25 12:19:34 +00:00
|
|
|
|
GtkBoxLayout *box_layout;
|
2010-05-24 23:11:54 +00:00
|
|
|
|
|
1997-11-24 22:37:52 +00:00
|
|
|
|
g_return_if_fail (GTK_IS_BOX (box));
|
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
homogeneous = !!homogeneous;
|
2014-06-07 14:59:14 +00:00
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
box_layout = GTK_BOX_LAYOUT (gtk_widget_get_layout_manager (GTK_WIDGET (box)));
|
|
|
|
|
if (homogeneous == gtk_box_layout_get_homogeneous (box_layout))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
gtk_box_layout_set_homogeneous (box_layout, homogeneous);
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (box), props[PROP_HOMOGENEOUS]);
|
1997-11-24 22:37:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2001-06-24 15:34:48 +00:00
|
|
|
|
/**
|
|
|
|
|
* gtk_box_get_homogeneous:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
*
|
|
|
|
|
* Returns whether the box is homogeneous (all children are the
|
2007-05-26 06:59:36 +00:00
|
|
|
|
* same size). See gtk_box_set_homogeneous().
|
2001-06-24 15:34:48 +00:00
|
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
|
* Returns: %TRUE if the box is homogeneous.
|
2001-06-24 15:34:48 +00:00
|
|
|
|
**/
|
|
|
|
|
gboolean
|
|
|
|
|
gtk_box_get_homogeneous (GtkBox *box)
|
|
|
|
|
{
|
2019-01-25 12:19:34 +00:00
|
|
|
|
GtkLayoutManager *box_layout;
|
2017-10-06 12:24:52 +00:00
|
|
|
|
|
2001-06-24 15:34:48 +00:00
|
|
|
|
g_return_val_if_fail (GTK_IS_BOX (box), FALSE);
|
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
box_layout = gtk_widget_get_layout_manager (GTK_WIDGET (box));
|
|
|
|
|
|
|
|
|
|
return gtk_box_layout_get_homogeneous (GTK_BOX_LAYOUT (box_layout));
|
2001-06-24 15:34:48 +00:00
|
|
|
|
}
|
|
|
|
|
|
2007-06-10 02:53:17 +00:00
|
|
|
|
/**
|
|
|
|
|
* gtk_box_set_spacing:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
* @spacing: the number of pixels to put between children
|
|
|
|
|
*
|
2010-11-02 09:43:42 +00:00
|
|
|
|
* Sets the #GtkBox:spacing property of @box, which is the
|
2007-06-10 02:53:17 +00:00
|
|
|
|
* number of pixels to place between children of @box.
|
|
|
|
|
*/
|
1997-11-24 22:37:52 +00:00
|
|
|
|
void
|
|
|
|
|
gtk_box_set_spacing (GtkBox *box,
|
2020-12-27 17:14:57 +00:00
|
|
|
|
int spacing)
|
1997-11-24 22:37:52 +00:00
|
|
|
|
{
|
2019-01-25 12:19:34 +00:00
|
|
|
|
GtkBoxLayout *box_layout;
|
2010-05-24 23:11:54 +00:00
|
|
|
|
|
1997-11-24 22:37:52 +00:00
|
|
|
|
g_return_if_fail (GTK_IS_BOX (box));
|
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
box_layout = GTK_BOX_LAYOUT (gtk_widget_get_layout_manager (GTK_WIDGET (box)));
|
|
|
|
|
if (spacing == gtk_box_layout_get_spacing (box_layout))
|
|
|
|
|
return;
|
2008-09-24 08:41:46 +00:00
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
gtk_box_layout_set_spacing (box_layout, spacing);
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (box), props[PROP_SPACING]);
|
1997-11-24 22:37:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2001-03-07 21:10:44 +00:00
|
|
|
|
/**
|
|
|
|
|
* gtk_box_get_spacing:
|
|
|
|
|
* @box: a #GtkBox
|
2010-11-02 09:43:42 +00:00
|
|
|
|
*
|
2001-03-07 21:10:44 +00:00
|
|
|
|
* Gets the value set by gtk_box_set_spacing().
|
2010-11-02 09:43:42 +00:00
|
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
|
* Returns: spacing between children
|
2001-03-07 21:10:44 +00:00
|
|
|
|
**/
|
2020-07-24 13:54:49 +00:00
|
|
|
|
int
|
2001-03-07 21:10:44 +00:00
|
|
|
|
gtk_box_get_spacing (GtkBox *box)
|
|
|
|
|
{
|
2019-01-25 12:19:34 +00:00
|
|
|
|
GtkLayoutManager *box_layout;
|
|
|
|
|
|
2001-03-09 14:49:00 +00:00
|
|
|
|
g_return_val_if_fail (GTK_IS_BOX (box), 0);
|
2001-03-07 21:10:44 +00:00
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
box_layout = gtk_widget_get_layout_manager (GTK_WIDGET (box));
|
|
|
|
|
|
|
|
|
|
return gtk_box_layout_get_spacing (GTK_BOX_LAYOUT (box_layout));
|
2001-03-07 21:10:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-03-05 14:20:20 +00:00
|
|
|
|
/**
|
|
|
|
|
* gtk_box_set_baseline_position:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
* @position: a #GtkBaselinePosition
|
|
|
|
|
*
|
|
|
|
|
* Sets the baseline position of a box. This affects
|
|
|
|
|
* only horizontal boxes with at least one baseline aligned
|
2014-01-28 04:30:28 +00:00
|
|
|
|
* child. If there is more vertical space available than requested,
|
2013-03-05 14:20:20 +00:00
|
|
|
|
* and the baseline is not allocated by the parent then
|
|
|
|
|
* @position is used to allocate the baseline wrt the
|
|
|
|
|
* extra space available.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
gtk_box_set_baseline_position (GtkBox *box,
|
2020-12-27 17:14:57 +00:00
|
|
|
|
GtkBaselinePosition position)
|
2013-03-05 14:20:20 +00:00
|
|
|
|
{
|
2019-01-25 12:19:34 +00:00
|
|
|
|
GtkBoxLayout *box_layout;
|
2013-03-05 14:20:20 +00:00
|
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_BOX (box));
|
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
box_layout = GTK_BOX_LAYOUT (gtk_widget_get_layout_manager (GTK_WIDGET (box)));
|
|
|
|
|
if (position == gtk_box_layout_get_baseline_position (box_layout))
|
|
|
|
|
return;
|
2013-03-05 14:20:20 +00:00
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
gtk_box_layout_set_baseline_position (box_layout, position);
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (box), props[PROP_BASELINE_POSITION]);
|
2013-03-05 14:20:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* gtk_box_get_baseline_position:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
*
|
|
|
|
|
* Gets the value set by gtk_box_set_baseline_position().
|
|
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
|
* Returns: the baseline position
|
2013-03-05 14:20:20 +00:00
|
|
|
|
**/
|
|
|
|
|
GtkBaselinePosition
|
2014-06-07 14:59:14 +00:00
|
|
|
|
gtk_box_get_baseline_position (GtkBox *box)
|
2013-03-05 14:20:20 +00:00
|
|
|
|
{
|
2019-01-25 12:19:34 +00:00
|
|
|
|
GtkLayoutManager *box_layout;
|
2017-10-06 12:24:52 +00:00
|
|
|
|
|
2013-03-05 14:20:20 +00:00
|
|
|
|
g_return_val_if_fail (GTK_IS_BOX (box), GTK_BASELINE_POSITION_CENTER);
|
|
|
|
|
|
2019-01-25 12:19:34 +00:00
|
|
|
|
box_layout = gtk_widget_get_layout_manager (GTK_WIDGET (box));
|
|
|
|
|
|
|
|
|
|
return gtk_box_layout_get_baseline_position (GTK_BOX_LAYOUT (box_layout));
|
2013-03-05 14:20:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-22 23:44:34 +00:00
|
|
|
|
/**
|
|
|
|
|
* gtk_box_insert_child_after:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
* @child: the #GtkWidget to insert
|
2020-12-21 14:42:01 +00:00
|
|
|
|
* @sibling: (nullable): the sibling after which to insert @child
|
2019-01-22 23:44:34 +00:00
|
|
|
|
*
|
|
|
|
|
* Inserts @child in the position after @sibling in the list
|
2019-08-28 07:26:29 +00:00
|
|
|
|
* of @box children. If @sibling is %NULL, insert @child at
|
2019-01-22 23:44:34 +00:00
|
|
|
|
* the first position.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
gtk_box_insert_child_after (GtkBox *box,
|
|
|
|
|
GtkWidget *child,
|
|
|
|
|
GtkWidget *sibling)
|
|
|
|
|
{
|
2019-08-28 07:26:29 +00:00
|
|
|
|
GtkWidget *widget;
|
2019-01-22 23:44:34 +00:00
|
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_BOX (box));
|
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (child));
|
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (child) == NULL);
|
2019-08-28 07:26:29 +00:00
|
|
|
|
|
|
|
|
|
widget = GTK_WIDGET (box);
|
|
|
|
|
|
2019-01-22 23:44:34 +00:00
|
|
|
|
if (sibling)
|
|
|
|
|
{
|
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (sibling));
|
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (sibling) == widget);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (child == sibling)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
gtk_widget_insert_after (child, widget, sibling);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* gtk_box_reorder_child_after:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
* @child: the #GtkWidget to move, must be a child of @box
|
|
|
|
|
* @sibling: (nullable): the sibling to move @child after, or %NULL
|
|
|
|
|
*
|
|
|
|
|
* Moves @child to the position after @sibling in the list
|
|
|
|
|
* of @box children. If @sibling is %NULL, move @child to
|
|
|
|
|
* the first position.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
gtk_box_reorder_child_after (GtkBox *box,
|
|
|
|
|
GtkWidget *child,
|
|
|
|
|
GtkWidget *sibling)
|
|
|
|
|
{
|
2019-08-28 07:26:29 +00:00
|
|
|
|
GtkWidget *widget;
|
2019-01-22 23:44:34 +00:00
|
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_BOX (box));
|
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (child));
|
2019-08-28 07:26:29 +00:00
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (child) == (GtkWidget *)box);
|
|
|
|
|
|
|
|
|
|
widget = GTK_WIDGET (box);
|
|
|
|
|
|
2019-01-22 23:44:34 +00:00
|
|
|
|
if (sibling)
|
|
|
|
|
{
|
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (sibling));
|
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (sibling) == widget);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (child == sibling)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
gtk_widget_insert_after (child, widget, sibling);
|
|
|
|
|
}
|
2020-05-08 16:21:30 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* gtk_box_append:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
* @child: the #GtkWidget to append
|
|
|
|
|
*
|
|
|
|
|
* Adds @child as the last child to @box.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
gtk_box_append (GtkBox *box,
|
|
|
|
|
GtkWidget *child)
|
|
|
|
|
{
|
|
|
|
|
g_return_if_fail (GTK_IS_BOX (box));
|
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (child));
|
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (child) == NULL);
|
|
|
|
|
|
|
|
|
|
gtk_widget_insert_before (child, GTK_WIDGET (box), NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* gtk_box_prepend:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
* @child: the #GtkWidget to prepend
|
|
|
|
|
*
|
|
|
|
|
* Adds @child as the first child to @box.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
gtk_box_prepend (GtkBox *box,
|
|
|
|
|
GtkWidget *child)
|
|
|
|
|
{
|
|
|
|
|
g_return_if_fail (GTK_IS_BOX (box));
|
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (child));
|
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (child) == NULL);
|
|
|
|
|
|
|
|
|
|
gtk_widget_insert_after (child, GTK_WIDGET (box), NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* gtk_box_remove:
|
|
|
|
|
* @box: a #GtkBox
|
|
|
|
|
* @child: the child to remove
|
|
|
|
|
*
|
|
|
|
|
* Removes a child widget from @box, after it has been
|
|
|
|
|
* added with gtk_box_append(), gtk_box_prepend(), or
|
|
|
|
|
* gtk_box_insert_child_after().
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
gtk_box_remove (GtkBox *box,
|
|
|
|
|
GtkWidget *child)
|
|
|
|
|
{
|
|
|
|
|
g_return_if_fail (GTK_IS_BOX (box));
|
|
|
|
|
g_return_if_fail (GTK_IS_WIDGET (child));
|
|
|
|
|
g_return_if_fail (gtk_widget_get_parent (child) == (GtkWidget *)box);
|
|
|
|
|
|
|
|
|
|
gtk_widget_unparent (child);
|
|
|
|
|
}
|