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
|
|
|
|
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
|
|
|
*/
|
|
|
|
|
2010-03-09 22:32:53 +00:00
|
|
|
/**
|
|
|
|
* SECTION:gtkbin
|
|
|
|
* @Short_description: A container with just one child
|
|
|
|
* @Title: GtkBin
|
|
|
|
*
|
|
|
|
* The #GtkBin widget is a container with just one child.
|
|
|
|
* It is not very useful itself, but it is useful for deriving subclasses,
|
|
|
|
* since it provides common code needed for handling a single child widget.
|
|
|
|
*
|
|
|
|
* Many GTK+ widgets are subclasses of #GtkBin, including #GtkWindow,
|
2016-10-03 18:02:21 +00:00
|
|
|
* #GtkButton, #GtkFrame or #GtkScrolledWindow.
|
2010-03-09 22:32:53 +00:00
|
|
|
*/
|
|
|
|
|
2008-06-22 14:28:52 +00:00
|
|
|
#include "config.h"
|
1997-11-24 22:37:52 +00:00
|
|
|
#include "gtkbin.h"
|
2010-07-09 17:22:23 +00:00
|
|
|
|
1997-11-24 22:37:52 +00:00
|
|
|
|
2010-08-26 17:15:37 +00:00
|
|
|
struct _GtkBinPrivate
|
2010-05-24 19:51:49 +00:00
|
|
|
{
|
|
|
|
GtkWidget *child;
|
|
|
|
};
|
|
|
|
|
2002-10-04 23:50:27 +00:00
|
|
|
static void gtk_bin_add (GtkContainer *container,
|
|
|
|
GtkWidget *widget);
|
|
|
|
static void gtk_bin_remove (GtkContainer *container,
|
|
|
|
GtkWidget *widget);
|
|
|
|
static void gtk_bin_forall (GtkContainer *container,
|
|
|
|
GtkCallback callback,
|
|
|
|
gpointer callback_data);
|
|
|
|
static GType gtk_bin_child_type (GtkContainer *container);
|
1997-11-24 22:37:52 +00:00
|
|
|
|
2016-10-22 14:06:14 +00:00
|
|
|
static void gtk_bin_measure (GtkWidget *widget,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
int for_size,
|
|
|
|
int *minimum,
|
|
|
|
int *natural,
|
|
|
|
int *minimum_baseline,
|
|
|
|
int *natural_baseline);
|
2012-11-08 23:21:01 +00:00
|
|
|
static void gtk_bin_size_allocate (GtkWidget *widget,
|
|
|
|
GtkAllocation *allocation);
|
2010-09-21 14:35:17 +00:00
|
|
|
|
2013-06-27 19:02:52 +00:00
|
|
|
G_DEFINE_ABSTRACT_TYPE_WITH_PRIVATE (GtkBin, gtk_bin, GTK_TYPE_CONTAINER)
|
1997-11-24 22:37:52 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_bin_class_init (GtkBinClass *class)
|
|
|
|
{
|
2010-09-21 14:35:17 +00:00
|
|
|
GtkWidgetClass *widget_class = (GtkWidgetClass*) class;
|
|
|
|
GtkContainerClass *container_class = (GtkContainerClass*) class;
|
1997-11-24 22:37:52 +00:00
|
|
|
|
2016-10-22 14:06:14 +00:00
|
|
|
widget_class->measure = gtk_bin_measure;
|
2012-11-08 23:21:01 +00:00
|
|
|
widget_class->size_allocate = gtk_bin_size_allocate;
|
1997-11-24 22:37:52 +00:00
|
|
|
|
|
|
|
container_class->add = gtk_bin_add;
|
|
|
|
container_class->remove = gtk_bin_remove;
|
GTK_MENU_DIR_CHILD: check for the existance of
Thu Sep 3 04:22:20 1998 Tim Janik <timj@gtk.org>
* gtk/gtkmenushell.c (gtk_real_menu_shell_move_current):
GTK_MENU_DIR_CHILD: check for the existance of
menu_shell->active_menu_item before accessing its child.
GTK_MENU_DIR_PREV:
GTK_MENU_DIR_NEXT: if we haven't had an active item and still
don't, make a default selection.
Wed Sep 2 00:28:58 1998 Tim Janik <timj@gtk.org>
* gtk/gtkwidget.c (gtk_widget_propagate_state): iterate
the children with _forall for sensitivity changes and with
_foreach on pure state changes. this fixes a lot of the
old inclusions of internal widgets into _foreach calls.
* gtk/gtktree.c: removed gtk_tree_foreach, let gtk_tree_forall
do the work. don't walk the subtrees of first level children.
* gtk/gtktreeitem.c: provide a _forall implementation,
which walks the subtrees as well for include_internals.
* gtk/gtkmenuitem.c: provide a _forall implementation, which walks
the submenus as well for include_internals.
* gtk/gtkscrolledwindow.c: removed gtk_scrolled_window_foreach and
implemented gtk_scrolled_window_forall, which will iterate over
the viewport and the scrollbars for gtk_container_forall or
iterate over the viewports children for gtk_container_foreach.
* gtk/gtktoolbar.c:
* gtk/gtktable.c:
* gtk/gtkpaned.c:
* gtk/gtkpacker.c:
* gtk/gtkmenushell.c:
* gtk/gtklist.c:
* gtk/gtkfixed.c:
* gtk/gtkclist.c:
* gtk/gtkbox.c:
* gtk/gtkbin.c:
* gtk/gtknotebook.c:
removed the old gtk_*_foreach functions and provided gtk_*_forall.
* gtk/gtknotebook.c:
(gtk_notebook_real_switch_page): expose tabs.
(gtk_notebook_page_num): new function to return the page number
of a distinct child.
(gtk_notebook_focus): minor fixups. foxus handling is still screwed
under some circumstances.
* gtk/gtktreeitem.c:
(gtk_real_tree_item_select):
(gtk_real_tree_item_deselect): major fixes.
some general fixups wrt queue_redraw, and tree items not being
NO_WINDOW widgets.
* gtk/gtklistitem.c:
(gtk_real_list_item_select):
(gtk_real_list_item_deselect):
(gtk_real_list_item_toggle):
removed unneccessary queue_redraw calls.
Wed Aug 30 09:42:07 1998 Tim Janik <timj@gtk.org>
* gtk/gtkoptionmenu.c: allow optionmenus to have the focus and
automatically popup the menu on space bar.
Wed Aug 26 06:40:34 1998 Tim Janik <timj@gtk.org>
* gtk/gtkcontainer.h:
* gtk/gtkcontainer.c: implemented gtk_container_forall() (as a class
method), which acts similar to gtk_container_foreach(), but iterates
over internal children. the GtkContainer::foreach signal vanished in
favour of a new class method ->forall() that optionally includes
internal widgets.
* gtk/gtkclist.c (gtk_clist_init): provide no _foreach implementation
but a _forall implementation, since all child widgets we have are
internal ones.
(column_button_create): set the parent window prior
to gtk_widget_set_parent().
* gtk/gtkwidget.c:
exchanged all calls to gtk_container_foreach() with
gtk_container_forall().
* gtk/gtkwidget.h:
* gtk/gtkwidget.c: added the GTK_COMPOSITE_CHILD, exported through
the GtkWidget::composite_child argument. to have a widget created
with the flag initially, two new functions got added to wrap a widgets
creation:
gtk_widget_push_composite_flag() and gtk_widget_pop_composite_flag().
Wed Aug 25 23:37:39 1998 Tim Janik <timj@gtk.org>
* gtk/gtktooltips.h:
* gtk/gtktooltips.c: exported gtk_tooltips_create_window() as
gtk_tooltips_force_window(), so tooltips->tip_window can be accessed
prior to the first tip being set.
don't put an extra reference on the window, since it is a toplevel,
it wont get destroyed from anywhere else.
* overall macro and GtkType fixups.
1998-09-03 02:38:53 +00:00
|
|
|
container_class->forall = gtk_bin_forall;
|
1998-06-16 05:20:05 +00:00
|
|
|
container_class->child_type = gtk_bin_child_type;
|
1997-11-24 22:37:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_bin_init (GtkBin *bin)
|
|
|
|
{
|
2010-03-06 10:29:31 +00:00
|
|
|
gtk_widget_set_has_window (GTK_WIDGET (bin), FALSE);
|
1997-11-24 22:37:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-10-04 23:50:27 +00:00
|
|
|
static GType
|
1998-06-16 05:20:05 +00:00
|
|
|
gtk_bin_child_type (GtkContainer *container)
|
|
|
|
{
|
2017-05-25 15:15:38 +00:00
|
|
|
GtkBinPrivate *priv = gtk_bin_get_instance_private (GTK_BIN (container));
|
2010-05-24 19:51:49 +00:00
|
|
|
|
|
|
|
if (!priv->child)
|
1998-06-16 05:20:05 +00:00
|
|
|
return GTK_TYPE_WIDGET;
|
|
|
|
else
|
2002-10-04 23:50:27 +00:00
|
|
|
return G_TYPE_NONE;
|
1998-06-16 05:20:05 +00:00
|
|
|
}
|
|
|
|
|
1997-11-24 22:37:52 +00:00
|
|
|
static void
|
|
|
|
gtk_bin_add (GtkContainer *container,
|
1998-07-07 01:25:27 +00:00
|
|
|
GtkWidget *child)
|
1997-11-24 22:37:52 +00:00
|
|
|
{
|
2002-01-30 22:29:03 +00:00
|
|
|
GtkBin *bin = GTK_BIN (container);
|
2017-05-25 15:15:38 +00:00
|
|
|
GtkBinPrivate *priv = gtk_bin_get_instance_private (bin);
|
1997-11-24 22:37:52 +00:00
|
|
|
|
2010-05-24 19:51:49 +00:00
|
|
|
if (priv->child != NULL)
|
2001-02-09 00:40:48 +00:00
|
|
|
{
|
|
|
|
g_warning ("Attempting to add a widget with type %s to a %s, "
|
|
|
|
"but as a GtkBin subclass a %s can only contain one widget at a time; "
|
|
|
|
"it already contains a widget of type %s",
|
2001-03-07 14:49:21 +00:00
|
|
|
g_type_name (G_OBJECT_TYPE (child)),
|
|
|
|
g_type_name (G_OBJECT_TYPE (bin)),
|
|
|
|
g_type_name (G_OBJECT_TYPE (bin)),
|
2010-05-24 19:51:49 +00:00
|
|
|
g_type_name (G_OBJECT_TYPE (priv->child)));
|
2001-02-09 00:40:48 +00:00
|
|
|
return;
|
|
|
|
}
|
1997-11-24 22:37:52 +00:00
|
|
|
|
1998-07-07 01:25:27 +00:00
|
|
|
gtk_widget_set_parent (child, GTK_WIDGET (bin));
|
2010-05-24 19:51:49 +00:00
|
|
|
priv->child = child;
|
1997-11-24 22:37:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_bin_remove (GtkContainer *container,
|
1998-07-07 01:25:27 +00:00
|
|
|
GtkWidget *child)
|
1997-11-24 22:37:52 +00:00
|
|
|
{
|
2002-01-30 22:29:03 +00:00
|
|
|
GtkBin *bin = GTK_BIN (container);
|
2017-05-25 15:15:38 +00:00
|
|
|
GtkBinPrivate *priv = gtk_bin_get_instance_private (bin);
|
1998-07-07 01:25:27 +00:00
|
|
|
gboolean widget_was_visible;
|
1997-11-24 22:37:52 +00:00
|
|
|
|
2010-05-24 19:51:49 +00:00
|
|
|
g_return_if_fail (priv->child == child);
|
1997-11-24 22:37:52 +00:00
|
|
|
|
2010-03-01 06:47:38 +00:00
|
|
|
widget_was_visible = gtk_widget_get_visible (child);
|
1998-07-07 01:25:27 +00:00
|
|
|
|
|
|
|
gtk_widget_unparent (child);
|
2010-05-24 19:51:49 +00:00
|
|
|
priv->child = NULL;
|
1998-07-07 01:25:27 +00:00
|
|
|
|
2010-03-01 06:47:38 +00:00
|
|
|
/* queue resize regardless of gtk_widget_get_visible (container),
|
1998-07-07 01:25:27 +00:00
|
|
|
* since that's what is needed by toplevels, which derive from GtkBin.
|
|
|
|
*/
|
|
|
|
if (widget_was_visible)
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (container));
|
1997-11-24 22:37:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
GTK_MENU_DIR_CHILD: check for the existance of
Thu Sep 3 04:22:20 1998 Tim Janik <timj@gtk.org>
* gtk/gtkmenushell.c (gtk_real_menu_shell_move_current):
GTK_MENU_DIR_CHILD: check for the existance of
menu_shell->active_menu_item before accessing its child.
GTK_MENU_DIR_PREV:
GTK_MENU_DIR_NEXT: if we haven't had an active item and still
don't, make a default selection.
Wed Sep 2 00:28:58 1998 Tim Janik <timj@gtk.org>
* gtk/gtkwidget.c (gtk_widget_propagate_state): iterate
the children with _forall for sensitivity changes and with
_foreach on pure state changes. this fixes a lot of the
old inclusions of internal widgets into _foreach calls.
* gtk/gtktree.c: removed gtk_tree_foreach, let gtk_tree_forall
do the work. don't walk the subtrees of first level children.
* gtk/gtktreeitem.c: provide a _forall implementation,
which walks the subtrees as well for include_internals.
* gtk/gtkmenuitem.c: provide a _forall implementation, which walks
the submenus as well for include_internals.
* gtk/gtkscrolledwindow.c: removed gtk_scrolled_window_foreach and
implemented gtk_scrolled_window_forall, which will iterate over
the viewport and the scrollbars for gtk_container_forall or
iterate over the viewports children for gtk_container_foreach.
* gtk/gtktoolbar.c:
* gtk/gtktable.c:
* gtk/gtkpaned.c:
* gtk/gtkpacker.c:
* gtk/gtkmenushell.c:
* gtk/gtklist.c:
* gtk/gtkfixed.c:
* gtk/gtkclist.c:
* gtk/gtkbox.c:
* gtk/gtkbin.c:
* gtk/gtknotebook.c:
removed the old gtk_*_foreach functions and provided gtk_*_forall.
* gtk/gtknotebook.c:
(gtk_notebook_real_switch_page): expose tabs.
(gtk_notebook_page_num): new function to return the page number
of a distinct child.
(gtk_notebook_focus): minor fixups. foxus handling is still screwed
under some circumstances.
* gtk/gtktreeitem.c:
(gtk_real_tree_item_select):
(gtk_real_tree_item_deselect): major fixes.
some general fixups wrt queue_redraw, and tree items not being
NO_WINDOW widgets.
* gtk/gtklistitem.c:
(gtk_real_list_item_select):
(gtk_real_list_item_deselect):
(gtk_real_list_item_toggle):
removed unneccessary queue_redraw calls.
Wed Aug 30 09:42:07 1998 Tim Janik <timj@gtk.org>
* gtk/gtkoptionmenu.c: allow optionmenus to have the focus and
automatically popup the menu on space bar.
Wed Aug 26 06:40:34 1998 Tim Janik <timj@gtk.org>
* gtk/gtkcontainer.h:
* gtk/gtkcontainer.c: implemented gtk_container_forall() (as a class
method), which acts similar to gtk_container_foreach(), but iterates
over internal children. the GtkContainer::foreach signal vanished in
favour of a new class method ->forall() that optionally includes
internal widgets.
* gtk/gtkclist.c (gtk_clist_init): provide no _foreach implementation
but a _forall implementation, since all child widgets we have are
internal ones.
(column_button_create): set the parent window prior
to gtk_widget_set_parent().
* gtk/gtkwidget.c:
exchanged all calls to gtk_container_foreach() with
gtk_container_forall().
* gtk/gtkwidget.h:
* gtk/gtkwidget.c: added the GTK_COMPOSITE_CHILD, exported through
the GtkWidget::composite_child argument. to have a widget created
with the flag initially, two new functions got added to wrap a widgets
creation:
gtk_widget_push_composite_flag() and gtk_widget_pop_composite_flag().
Wed Aug 25 23:37:39 1998 Tim Janik <timj@gtk.org>
* gtk/gtktooltips.h:
* gtk/gtktooltips.c: exported gtk_tooltips_create_window() as
gtk_tooltips_force_window(), so tooltips->tip_window can be accessed
prior to the first tip being set.
don't put an extra reference on the window, since it is a toplevel,
it wont get destroyed from anywhere else.
* overall macro and GtkType fixups.
1998-09-03 02:38:53 +00:00
|
|
|
gtk_bin_forall (GtkContainer *container,
|
|
|
|
GtkCallback callback,
|
|
|
|
gpointer callback_data)
|
1997-11-24 22:37:52 +00:00
|
|
|
{
|
2002-01-30 22:29:03 +00:00
|
|
|
GtkBin *bin = GTK_BIN (container);
|
2017-05-25 15:15:38 +00:00
|
|
|
GtkBinPrivate *priv = gtk_bin_get_instance_private (bin);
|
1997-11-24 22:37:52 +00:00
|
|
|
|
2010-05-24 19:51:49 +00:00
|
|
|
if (priv->child)
|
|
|
|
(* callback) (priv->child, callback_data);
|
1997-11-24 22:37:52 +00:00
|
|
|
}
|
2001-05-25 03:42:40 +00:00
|
|
|
|
2012-11-08 23:21:01 +00:00
|
|
|
static void
|
2016-10-22 14:06:14 +00:00
|
|
|
gtk_bin_measure (GtkWidget *widget,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
int for_size,
|
|
|
|
int *minimum,
|
|
|
|
int *natural,
|
|
|
|
int *minimum_baseline,
|
|
|
|
int *natural_baseline)
|
2012-11-08 23:21:01 +00:00
|
|
|
{
|
2016-10-22 14:06:14 +00:00
|
|
|
GtkBinPrivate *priv = gtk_bin_get_instance_private (GTK_BIN (widget));
|
2013-03-04 15:22:00 +00:00
|
|
|
|
2016-10-22 14:06:14 +00:00
|
|
|
if (priv->child != NULL && gtk_widget_get_visible (priv->child))
|
2012-11-08 23:21:01 +00:00
|
|
|
{
|
2016-10-22 14:06:14 +00:00
|
|
|
gtk_widget_measure (priv->child,
|
|
|
|
orientation,
|
|
|
|
for_size,
|
|
|
|
minimum, natural,
|
|
|
|
minimum_baseline, natural_baseline);
|
2012-11-08 23:21:01 +00:00
|
|
|
}
|
2016-10-22 14:06:14 +00:00
|
|
|
else
|
2010-04-09 04:19:42 +00:00
|
|
|
{
|
2016-10-22 14:06:14 +00:00
|
|
|
*minimum = 0;
|
|
|
|
*natural = 0;
|
2010-04-09 04:19:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-08 23:21:01 +00:00
|
|
|
static void
|
|
|
|
gtk_bin_size_allocate (GtkWidget *widget,
|
|
|
|
GtkAllocation *allocation)
|
|
|
|
{
|
|
|
|
GtkBin *bin = GTK_BIN (widget);
|
2017-05-25 15:15:38 +00:00
|
|
|
GtkBinPrivate *priv = gtk_bin_get_instance_private (bin);
|
2012-11-08 23:21:01 +00:00
|
|
|
|
2012-11-11 18:30:11 +00:00
|
|
|
if (priv->child && gtk_widget_get_visible (priv->child))
|
2012-11-11 19:24:58 +00:00
|
|
|
{
|
2017-05-04 09:27:45 +00:00
|
|
|
GtkAllocation clip = *allocation;
|
|
|
|
|
2016-10-09 00:32:00 +00:00
|
|
|
gtk_widget_size_allocate (priv->child, allocation);
|
2017-05-04 09:27:45 +00:00
|
|
|
gtk_widget_get_clip (priv->child, &clip);
|
|
|
|
|
|
|
|
gtk_widget_set_clip (widget, &clip);
|
2012-11-11 19:24:58 +00:00
|
|
|
}
|
2012-11-08 23:21:01 +00:00
|
|
|
}
|
2010-04-09 04:19:42 +00:00
|
|
|
|
2001-05-25 03:42:40 +00:00
|
|
|
/**
|
|
|
|
* gtk_bin_get_child:
|
|
|
|
* @bin: a #GtkBin
|
|
|
|
*
|
|
|
|
* Gets the child of the #GtkBin, or %NULL if the bin contains
|
|
|
|
* no child widget. The returned widget does not have a reference
|
|
|
|
* added, so you do not need to unref it.
|
2009-12-10 10:23:40 +00:00
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
* Returns: (transfer none): pointer to child of the #GtkBin
|
2001-05-25 03:42:40 +00:00
|
|
|
**/
|
|
|
|
GtkWidget*
|
|
|
|
gtk_bin_get_child (GtkBin *bin)
|
|
|
|
{
|
2017-05-25 15:15:38 +00:00
|
|
|
GtkBinPrivate *priv = gtk_bin_get_instance_private (bin);
|
|
|
|
|
2001-05-25 03:42:40 +00:00
|
|
|
g_return_val_if_fail (GTK_IS_BIN (bin), NULL);
|
|
|
|
|
2017-05-25 15:15:38 +00:00
|
|
|
return priv->child;
|
2001-05-25 03:42:40 +00:00
|
|
|
}
|
2010-05-28 03:49:30 +00:00
|
|
|
|
2014-06-16 21:31:56 +00:00
|
|
|
void
|
|
|
|
_gtk_bin_set_child (GtkBin *bin,
|
|
|
|
GtkWidget *widget)
|
|
|
|
{
|
2017-05-25 15:15:38 +00:00
|
|
|
GtkBinPrivate *priv = gtk_bin_get_instance_private (bin);
|
|
|
|
|
|
|
|
priv->child = widget;
|
2014-06-16 21:31:56 +00:00
|
|
|
}
|