Blame gtk/gtksizegroup.c

Packit 98cdb6
/* GTK - The GIMP Toolkit
Packit 98cdb6
 * gtksizegroup.c: 
Packit 98cdb6
 * Copyright (C) 2001 Red Hat Software
Packit 98cdb6
 *
Packit 98cdb6
 * This library is free software; you can redistribute it and/or
Packit 98cdb6
 * modify it under the terms of the GNU Lesser General Public
Packit 98cdb6
 * License as published by the Free Software Foundation; either
Packit 98cdb6
 * version 2 of the License, or (at your option) any later version.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is distributed in the hope that it will be useful,
Packit 98cdb6
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 98cdb6
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 98cdb6
 * Lesser General Public License for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * You should have received a copy of the GNU Lesser General Public
Packit 98cdb6
 * License along with this library; if not, write to the
Packit 98cdb6
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit 98cdb6
 * Boston, MA 02111-1307, USA.
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
#include "config.h"
Packit 98cdb6
#include <string.h>
Packit 98cdb6
#include "gtkcontainer.h"
Packit 98cdb6
#include "gtkintl.h"
Packit 98cdb6
#include "gtkprivate.h"
Packit 98cdb6
#include "gtksizegroup.h"
Packit 98cdb6
#include "gtkbuildable.h"
Packit 98cdb6
#include "gtkalias.h"
Packit 98cdb6
Packit 98cdb6
enum {
Packit 98cdb6
  PROP_0,
Packit 98cdb6
  PROP_MODE,
Packit 98cdb6
  PROP_IGNORE_HIDDEN
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
static void gtk_size_group_set_property (GObject      *object,
Packit 98cdb6
					 guint         prop_id,
Packit 98cdb6
					 const GValue *value,
Packit 98cdb6
					 GParamSpec   *pspec);
Packit 98cdb6
static void gtk_size_group_get_property (GObject      *object,
Packit 98cdb6
					 guint         prop_id,
Packit 98cdb6
					 GValue       *value,
Packit 98cdb6
					 GParamSpec   *pspec);
Packit 98cdb6
Packit 98cdb6
static void add_group_to_closure  (GtkSizeGroup      *group,
Packit 98cdb6
				   GtkSizeGroupMode   mode,
Packit 98cdb6
				   GSList           **groups,
Packit 98cdb6
				   GSList           **widgets);
Packit 98cdb6
static void add_widget_to_closure (GtkWidget         *widget,
Packit 98cdb6
				   GtkSizeGroupMode   mode,
Packit 98cdb6
				   GSList           **groups,
Packit 98cdb6
				   GSList           **widgets);
Packit 98cdb6
Packit 98cdb6
/* GtkBuildable */
Packit 98cdb6
static void gtk_size_group_buildable_init (GtkBuildableIface *iface);
Packit 98cdb6
static gboolean gtk_size_group_buildable_custom_tag_start (GtkBuildable  *buildable,
Packit 98cdb6
							   GtkBuilder    *builder,
Packit 98cdb6
							   GObject       *child,
Packit 98cdb6
							   const gchar   *tagname,
Packit 98cdb6
							   GMarkupParser *parser,
Packit 98cdb6
							   gpointer      *data);
Packit 98cdb6
static void gtk_size_group_buildable_custom_finished (GtkBuildable  *buildable,
Packit 98cdb6
						      GtkBuilder    *builder,
Packit 98cdb6
						      GObject       *child,
Packit 98cdb6
						      const gchar   *tagname,
Packit 98cdb6
						      gpointer       user_data);
Packit 98cdb6
Packit 98cdb6
static GQuark size_groups_quark;
Packit 98cdb6
static const gchar size_groups_tag[] = "gtk-size-groups";
Packit 98cdb6
Packit 98cdb6
static GQuark visited_quark;
Packit 98cdb6
static const gchar visited_tag[] = "gtk-size-group-visited";
Packit 98cdb6
Packit 98cdb6
static GSList *
Packit 98cdb6
get_size_groups (GtkWidget *widget)
Packit 98cdb6
{
Packit 98cdb6
  return g_object_get_qdata (G_OBJECT (widget), size_groups_quark);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
set_size_groups (GtkWidget *widget,
Packit 98cdb6
		 GSList    *groups)
Packit 98cdb6
{
Packit 98cdb6
  g_object_set_qdata (G_OBJECT (widget), size_groups_quark, groups);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
mark_visited (gpointer object)
Packit 98cdb6
{
Packit 98cdb6
  g_object_set_qdata (object, visited_quark, "visited");
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
mark_unvisited (gpointer object)
Packit 98cdb6
{
Packit 98cdb6
  g_object_set_qdata (object, visited_quark, NULL);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
is_visited (gpointer object)
Packit 98cdb6
{
Packit 98cdb6
  return g_object_get_qdata (object, visited_quark) != NULL;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
add_group_to_closure (GtkSizeGroup    *group,
Packit 98cdb6
		      GtkSizeGroupMode mode,
Packit 98cdb6
		      GSList         **groups,
Packit 98cdb6
		      GSList         **widgets)
Packit 98cdb6
{
Packit 98cdb6
  GSList *tmp_widgets;
Packit 98cdb6
  
Packit 98cdb6
  *groups = g_slist_prepend (*groups, group);
Packit 98cdb6
  mark_visited (group);
Packit 98cdb6
Packit 98cdb6
  tmp_widgets = group->widgets;
Packit 98cdb6
  while (tmp_widgets)
Packit 98cdb6
    {
Packit 98cdb6
      GtkWidget *tmp_widget = tmp_widgets->data;
Packit 98cdb6
      
Packit 98cdb6
      if (!is_visited (tmp_widget))
Packit 98cdb6
	add_widget_to_closure (tmp_widget, mode, groups, widgets);
Packit 98cdb6
      
Packit 98cdb6
      tmp_widgets = tmp_widgets->next;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
add_widget_to_closure (GtkWidget       *widget,
Packit 98cdb6
		       GtkSizeGroupMode mode,
Packit 98cdb6
		       GSList         **groups,
Packit 98cdb6
		       GSList         **widgets)
Packit 98cdb6
{
Packit 98cdb6
  GSList *tmp_groups;
Packit 98cdb6
Packit 98cdb6
  *widgets = g_slist_prepend (*widgets, widget);
Packit 98cdb6
  mark_visited (widget);
Packit 98cdb6
Packit 98cdb6
  tmp_groups = get_size_groups (widget);
Packit 98cdb6
  while (tmp_groups)
Packit 98cdb6
    {
Packit 98cdb6
      GtkSizeGroup *tmp_group = tmp_groups->data;
Packit 98cdb6
      
Packit 98cdb6
      if ((tmp_group->mode == GTK_SIZE_GROUP_BOTH || tmp_group->mode == mode) &&
Packit 98cdb6
	  !is_visited (tmp_group))
Packit 98cdb6
	add_group_to_closure (tmp_group, mode, groups, widgets);
Packit 98cdb6
Packit 98cdb6
      tmp_groups = tmp_groups->next;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
real_queue_resize (GtkWidget *widget)
Packit 98cdb6
{
Packit 98cdb6
  GTK_PRIVATE_SET_FLAG (widget, GTK_ALLOC_NEEDED);
Packit 98cdb6
  GTK_PRIVATE_SET_FLAG (widget, GTK_REQUEST_NEEDED);
Packit 98cdb6
  
Packit 98cdb6
  if (widget->parent)
Packit 98cdb6
    _gtk_container_queue_resize (GTK_CONTAINER (widget->parent));
Packit 98cdb6
  else if (gtk_widget_is_toplevel (widget) && GTK_IS_CONTAINER (widget))
Packit 98cdb6
    _gtk_container_queue_resize (GTK_CONTAINER (widget));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
reset_group_sizes (GSList *groups)
Packit 98cdb6
{
Packit 98cdb6
  GSList *tmp_list = groups;
Packit 98cdb6
  while (tmp_list)
Packit 98cdb6
    {
Packit 98cdb6
      GtkSizeGroup *tmp_group = tmp_list->data;
Packit 98cdb6
Packit 98cdb6
      tmp_group->have_width = FALSE;
Packit 98cdb6
      tmp_group->have_height = FALSE;
Packit 98cdb6
      
Packit 98cdb6
      tmp_list = tmp_list->next;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
queue_resize_on_widget (GtkWidget *widget,
Packit 98cdb6
			gboolean   check_siblings)
Packit 98cdb6
{
Packit 98cdb6
  GtkWidget *parent = widget;
Packit 98cdb6
  GSList *tmp_list;
Packit 98cdb6
Packit 98cdb6
  while (parent)
Packit 98cdb6
    {
Packit 98cdb6
      GSList *widget_groups;
Packit 98cdb6
      GSList *groups;
Packit 98cdb6
      GSList *widgets;
Packit 98cdb6
      
Packit 98cdb6
      if (widget == parent && !check_siblings)
Packit 98cdb6
	{
Packit 98cdb6
	  real_queue_resize (widget);
Packit 98cdb6
	  parent = parent->parent;
Packit 98cdb6
	  continue;
Packit 98cdb6
	}
Packit 98cdb6
      
Packit 98cdb6
      widget_groups = get_size_groups (parent);
Packit 98cdb6
      if (!widget_groups)
Packit 98cdb6
	{
Packit 98cdb6
	  if (widget == parent)
Packit 98cdb6
	    real_queue_resize (widget);
Packit 98cdb6
Packit 98cdb6
	  parent = parent->parent;
Packit 98cdb6
	  continue;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      groups = NULL;
Packit 98cdb6
      widgets = NULL;
Packit 98cdb6
	  
Packit 98cdb6
      add_widget_to_closure (parent, GTK_SIZE_GROUP_HORIZONTAL, &groups, &widgets);
Packit 98cdb6
      g_slist_foreach (widgets, (GFunc)mark_unvisited, NULL);
Packit 98cdb6
      g_slist_foreach (groups, (GFunc)mark_unvisited, NULL);
Packit 98cdb6
Packit 98cdb6
      reset_group_sizes (groups);
Packit 98cdb6
	      
Packit 98cdb6
      tmp_list = widgets;
Packit 98cdb6
      while (tmp_list)
Packit 98cdb6
	{
Packit 98cdb6
	  if (tmp_list->data == parent)
Packit 98cdb6
	    {
Packit 98cdb6
	      if (widget == parent)
Packit 98cdb6
		real_queue_resize (parent);
Packit 98cdb6
	    }
Packit 98cdb6
	  else if (tmp_list->data == widget)
Packit 98cdb6
            {
Packit 98cdb6
              g_warning ("A container and its child are part of this SizeGroup");
Packit 98cdb6
            }
Packit 98cdb6
	  else
Packit 98cdb6
	    queue_resize_on_widget (tmp_list->data, FALSE);
Packit 98cdb6
Packit 98cdb6
	  tmp_list = tmp_list->next;
Packit 98cdb6
	}
Packit 98cdb6
      
Packit 98cdb6
      g_slist_free (widgets);
Packit 98cdb6
      g_slist_free (groups);
Packit 98cdb6
	      
Packit 98cdb6
      groups = NULL;
Packit 98cdb6
      widgets = NULL;
Packit 98cdb6
	      
Packit 98cdb6
      add_widget_to_closure (parent, GTK_SIZE_GROUP_VERTICAL, &groups, &widgets);
Packit 98cdb6
      g_slist_foreach (widgets, (GFunc)mark_unvisited, NULL);
Packit 98cdb6
      g_slist_foreach (groups, (GFunc)mark_unvisited, NULL);
Packit 98cdb6
Packit 98cdb6
      reset_group_sizes (groups);
Packit 98cdb6
	      
Packit 98cdb6
      tmp_list = widgets;
Packit 98cdb6
      while (tmp_list)
Packit 98cdb6
	{
Packit 98cdb6
	  if (tmp_list->data == parent)
Packit 98cdb6
	    {
Packit 98cdb6
	      if (widget == parent)
Packit 98cdb6
		real_queue_resize (parent);
Packit 98cdb6
	    }
Packit 98cdb6
	  else if (tmp_list->data == widget)
Packit 98cdb6
            {
Packit 98cdb6
              g_warning ("A container and its child are part of this SizeGroup");
Packit 98cdb6
            }
Packit 98cdb6
	  else
Packit 98cdb6
	    queue_resize_on_widget (tmp_list->data, FALSE);
Packit 98cdb6
Packit 98cdb6
	  tmp_list = tmp_list->next;
Packit 98cdb6
	}
Packit 98cdb6
      
Packit 98cdb6
      g_slist_free (widgets);
Packit 98cdb6
      g_slist_free (groups);
Packit 98cdb6
      
Packit 98cdb6
      parent = parent->parent;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
queue_resize_on_group (GtkSizeGroup *size_group)
Packit 98cdb6
{
Packit 98cdb6
  if (size_group->widgets)
Packit 98cdb6
    queue_resize_on_widget (size_group->widgets->data, TRUE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
initialize_size_group_quarks (void)
Packit 98cdb6
{
Packit 98cdb6
  if (!size_groups_quark)
Packit 98cdb6
    {
Packit 98cdb6
      size_groups_quark = g_quark_from_static_string (size_groups_tag);
Packit 98cdb6
      visited_quark = g_quark_from_static_string (visited_tag);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_size_group_class_init (GtkSizeGroupClass *klass)
Packit 98cdb6
{
Packit 98cdb6
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Packit 98cdb6
Packit 98cdb6
  gobject_class->set_property = gtk_size_group_set_property;
Packit 98cdb6
  gobject_class->get_property = gtk_size_group_get_property;
Packit 98cdb6
  
Packit 98cdb6
  g_object_class_install_property (gobject_class,
Packit 98cdb6
				   PROP_MODE,
Packit 98cdb6
				   g_param_spec_enum ("mode",
Packit 98cdb6
						      P_("Mode"),
Packit 98cdb6
						      P_("The directions in which the size group affects the requested sizes"
Packit 98cdb6
							" of its component widgets"),
Packit 98cdb6
						      GTK_TYPE_SIZE_GROUP_MODE,
Packit 98cdb6
						      GTK_SIZE_GROUP_HORIZONTAL,						      GTK_PARAM_READWRITE));
Packit 98cdb6
Packit 98cdb6
  /**
Packit 98cdb6
   * GtkSizeGroup:ignore-hidden:
Packit 98cdb6
   *
Packit 98cdb6
   * If %TRUE, unmapped widgets are ignored when determining 
Packit 98cdb6
   * the size of the group.
Packit 98cdb6
   *
Packit 98cdb6
   * Since: 2.8
Packit 98cdb6
   */
Packit 98cdb6
  g_object_class_install_property (gobject_class,
Packit 98cdb6
				   PROP_IGNORE_HIDDEN,
Packit 98cdb6
				   g_param_spec_boolean ("ignore-hidden",
Packit 98cdb6
							 P_("Ignore hidden"),
Packit 98cdb6
							 P_("If TRUE, unmapped widgets are ignored "
Packit 98cdb6
							    "when determining the size of the group"),
Packit 98cdb6
							 FALSE,
Packit 98cdb6
							 GTK_PARAM_READWRITE));
Packit 98cdb6
  
Packit 98cdb6
  initialize_size_group_quarks ();
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_size_group_init (GtkSizeGroup *size_group)
Packit 98cdb6
{
Packit 98cdb6
  size_group->widgets = NULL;
Packit 98cdb6
  size_group->mode = GTK_SIZE_GROUP_HORIZONTAL;
Packit 98cdb6
  size_group->have_width = 0;
Packit 98cdb6
  size_group->have_height = 0;
Packit 98cdb6
  size_group->ignore_hidden = 0;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_size_group_buildable_init (GtkBuildableIface *iface)
Packit 98cdb6
{
Packit 98cdb6
  iface->custom_tag_start = gtk_size_group_buildable_custom_tag_start;
Packit 98cdb6
  iface->custom_finished = gtk_size_group_buildable_custom_finished;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
G_DEFINE_TYPE_WITH_CODE (GtkSizeGroup, gtk_size_group, G_TYPE_OBJECT,
Packit 98cdb6
			 G_IMPLEMENT_INTERFACE (GTK_TYPE_BUILDABLE,
Packit 98cdb6
						gtk_size_group_buildable_init))
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_size_group_set_property (GObject      *object,
Packit 98cdb6
			     guint         prop_id,
Packit 98cdb6
			     const GValue *value,
Packit 98cdb6
			     GParamSpec   *pspec)
Packit 98cdb6
{
Packit 98cdb6
  GtkSizeGroup *size_group = GTK_SIZE_GROUP (object);
Packit 98cdb6
Packit 98cdb6
  switch (prop_id)
Packit 98cdb6
    {
Packit 98cdb6
    case PROP_MODE:
Packit 98cdb6
      gtk_size_group_set_mode (size_group, g_value_get_enum (value));
Packit 98cdb6
      break;
Packit 98cdb6
    case PROP_IGNORE_HIDDEN:
Packit 98cdb6
      gtk_size_group_set_ignore_hidden (size_group, g_value_get_boolean (value));
Packit 98cdb6
      break;
Packit 98cdb6
    default:
Packit 98cdb6
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit 98cdb6
      break;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_size_group_get_property (GObject      *object,
Packit 98cdb6
			     guint         prop_id,
Packit 98cdb6
			     GValue       *value,
Packit 98cdb6
			     GParamSpec   *pspec)
Packit 98cdb6
{
Packit 98cdb6
  GtkSizeGroup *size_group = GTK_SIZE_GROUP (object);
Packit 98cdb6
Packit 98cdb6
  switch (prop_id)
Packit 98cdb6
    {
Packit 98cdb6
    case PROP_MODE:
Packit 98cdb6
      g_value_set_enum (value, size_group->mode);
Packit 98cdb6
      break;
Packit 98cdb6
    case PROP_IGNORE_HIDDEN:
Packit 98cdb6
      g_value_set_boolean (value, size_group->ignore_hidden);
Packit 98cdb6
      break;
Packit 98cdb6
    default:
Packit 98cdb6
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit 98cdb6
      break;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_size_group_new:
Packit 98cdb6
 * @mode: the mode for the new size group.
Packit 98cdb6
 * 
Packit 98cdb6
 * Create a new #GtkSizeGroup.
Packit 98cdb6
 
Packit 98cdb6
 * Return value: a newly created #GtkSizeGroup
Packit 98cdb6
 **/
Packit 98cdb6
GtkSizeGroup *
Packit 98cdb6
gtk_size_group_new (GtkSizeGroupMode mode)
Packit 98cdb6
{
Packit 98cdb6
  GtkSizeGroup *size_group = g_object_new (GTK_TYPE_SIZE_GROUP, NULL);
Packit 98cdb6
Packit 98cdb6
  size_group->mode = mode;
Packit 98cdb6
Packit 98cdb6
  return size_group;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_size_group_set_mode:
Packit 98cdb6
 * @size_group: a #GtkSizeGroup
Packit 98cdb6
 * @mode: the mode to set for the size group.
Packit 98cdb6
 * 
Packit 98cdb6
 * Sets the #GtkSizeGroupMode of the size group. The mode of the size
Packit 98cdb6
 * group determines whether the widgets in the size group should
Packit 98cdb6
 * all have the same horizontal requisition (%GTK_SIZE_GROUP_MODE_HORIZONTAL)
Packit 98cdb6
 * all have the same vertical requisition (%GTK_SIZE_GROUP_MODE_VERTICAL),
Packit 98cdb6
 * or should all have the same requisition in both directions
Packit 98cdb6
 * (%GTK_SIZE_GROUP_MODE_BOTH).
Packit 98cdb6
 **/
Packit 98cdb6
void
Packit 98cdb6
gtk_size_group_set_mode (GtkSizeGroup     *size_group,
Packit 98cdb6
			 GtkSizeGroupMode  mode)
Packit 98cdb6
{
Packit 98cdb6
  g_return_if_fail (GTK_IS_SIZE_GROUP (size_group));
Packit 98cdb6
Packit 98cdb6
  if (size_group->mode != mode)
Packit 98cdb6
    {
Packit 98cdb6
      if (size_group->mode != GTK_SIZE_GROUP_NONE)
Packit 98cdb6
	queue_resize_on_group (size_group);
Packit 98cdb6
      size_group->mode = mode;
Packit 98cdb6
      if (size_group->mode != GTK_SIZE_GROUP_NONE)
Packit 98cdb6
	queue_resize_on_group (size_group);
Packit 98cdb6
Packit 98cdb6
      g_object_notify (G_OBJECT (size_group), "mode");
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_size_group_get_mode:
Packit 98cdb6
 * @size_group: a #GtkSizeGroup
Packit 98cdb6
 * 
Packit 98cdb6
 * Gets the current mode of the size group. See gtk_size_group_set_mode().
Packit 98cdb6
 * 
Packit 98cdb6
 * Return value: the current mode of the size group.
Packit 98cdb6
 **/
Packit 98cdb6
GtkSizeGroupMode
Packit 98cdb6
gtk_size_group_get_mode (GtkSizeGroup *size_group)
Packit 98cdb6
{
Packit 98cdb6
  g_return_val_if_fail (GTK_IS_SIZE_GROUP (size_group), GTK_SIZE_GROUP_BOTH);
Packit 98cdb6
Packit 98cdb6
  return size_group->mode;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_size_group_set_ignore_hidden:
Packit 98cdb6
 * @size_group: a #GtkSizeGroup
Packit 98cdb6
 * @ignore_hidden: whether unmapped widgets should be ignored
Packit 98cdb6
 *   when calculating the size
Packit 98cdb6
 * 
Packit 98cdb6
 * Sets whether unmapped widgets should be ignored when
Packit 98cdb6
 * calculating the size.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.8 
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_size_group_set_ignore_hidden (GtkSizeGroup *size_group,
Packit 98cdb6
				  gboolean      ignore_hidden)
Packit 98cdb6
{
Packit 98cdb6
  g_return_if_fail (GTK_IS_SIZE_GROUP (size_group));
Packit 98cdb6
  
Packit 98cdb6
  ignore_hidden = ignore_hidden != FALSE;
Packit 98cdb6
Packit 98cdb6
  if (size_group->ignore_hidden != ignore_hidden)
Packit 98cdb6
    {
Packit 98cdb6
      size_group->ignore_hidden = ignore_hidden;
Packit 98cdb6
Packit 98cdb6
      g_object_notify (G_OBJECT (size_group), "ignore-hidden");
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_size_group_get_ignore_hidden:
Packit 98cdb6
 * @size_group: a #GtkSizeGroup
Packit 98cdb6
 *
Packit 98cdb6
 * Returns if invisible widgets are ignored when calculating the size.
Packit 98cdb6
 *
Packit 98cdb6
 * Returns: %TRUE if invisible widgets are ignored.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.8
Packit 98cdb6
 */
Packit 98cdb6
gboolean
Packit 98cdb6
gtk_size_group_get_ignore_hidden (GtkSizeGroup *size_group)
Packit 98cdb6
{
Packit 98cdb6
  g_return_val_if_fail (GTK_IS_SIZE_GROUP (size_group), FALSE);
Packit 98cdb6
Packit 98cdb6
  return size_group->ignore_hidden;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_size_group_widget_destroyed (GtkWidget    *widget,
Packit 98cdb6
				 GtkSizeGroup *size_group)
Packit 98cdb6
{
Packit 98cdb6
  gtk_size_group_remove_widget (size_group, widget);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_size_group_add_widget:
Packit 98cdb6
 * @size_group: a #GtkSizeGroup
Packit 98cdb6
 * @widget: the #GtkWidget to add
Packit 98cdb6
 * 
Packit 98cdb6
 * Adds a widget to a #GtkSizeGroup. In the future, the requisition
Packit 98cdb6
 * of the widget will be determined as the maximum of its requisition
Packit 98cdb6
 * and the requisition of the other widgets in the size group.
Packit 98cdb6
 * Whether this applies horizontally, vertically, or in both directions
Packit 98cdb6
 * depends on the mode of the size group. See gtk_size_group_set_mode().
Packit 98cdb6
 *
Packit 98cdb6
 * When the widget is destroyed or no longer referenced elsewhere, it will 
Packit 98cdb6
 * be removed from the size group.
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_size_group_add_widget (GtkSizeGroup     *size_group,
Packit 98cdb6
			   GtkWidget        *widget)
Packit 98cdb6
{
Packit 98cdb6
  GSList *groups;
Packit 98cdb6
  
Packit 98cdb6
  g_return_if_fail (GTK_IS_SIZE_GROUP (size_group));
Packit 98cdb6
  g_return_if_fail (GTK_IS_WIDGET (widget));
Packit 98cdb6
  
Packit 98cdb6
  groups = get_size_groups (widget);
Packit 98cdb6
Packit 98cdb6
  if (!g_slist_find (groups, size_group))
Packit 98cdb6
    {
Packit 98cdb6
      groups = g_slist_prepend (groups, size_group);
Packit 98cdb6
      set_size_groups (widget, groups);
Packit 98cdb6
Packit 98cdb6
      size_group->widgets = g_slist_prepend (size_group->widgets, widget);
Packit 98cdb6
Packit 98cdb6
      g_signal_connect (widget, "destroy",
Packit 98cdb6
			G_CALLBACK (gtk_size_group_widget_destroyed),
Packit 98cdb6
			size_group);
Packit 98cdb6
Packit 98cdb6
      g_object_ref (size_group);
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  queue_resize_on_group (size_group);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_size_group_remove_widget:
Packit 98cdb6
 * @size_group: a #GtkSizeGrup
Packit 98cdb6
 * @widget: the #GtkWidget to remove
Packit 98cdb6
 * 
Packit 98cdb6
 * Removes a widget from a #GtkSizeGroup.
Packit 98cdb6
 **/
Packit 98cdb6
void
Packit 98cdb6
gtk_size_group_remove_widget (GtkSizeGroup *size_group,
Packit 98cdb6
			      GtkWidget    *widget)
Packit 98cdb6
{
Packit 98cdb6
  GSList *groups;
Packit 98cdb6
  
Packit 98cdb6
  g_return_if_fail (GTK_IS_SIZE_GROUP (size_group));
Packit 98cdb6
  g_return_if_fail (GTK_IS_WIDGET (widget));
Packit 98cdb6
  g_return_if_fail (g_slist_find (size_group->widgets, widget));
Packit 98cdb6
Packit 98cdb6
  g_signal_handlers_disconnect_by_func (widget,
Packit 98cdb6
					gtk_size_group_widget_destroyed,
Packit 98cdb6
					size_group);
Packit 98cdb6
  
Packit 98cdb6
  groups = get_size_groups (widget);
Packit 98cdb6
  groups = g_slist_remove (groups, size_group);
Packit 98cdb6
  set_size_groups (widget, groups);
Packit 98cdb6
Packit 98cdb6
  size_group->widgets = g_slist_remove (size_group->widgets, widget);
Packit 98cdb6
  queue_resize_on_group (size_group);
Packit 98cdb6
  gtk_widget_queue_resize (widget);
Packit 98cdb6
Packit 98cdb6
  g_object_unref (size_group);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_size_group_get_widgets:
Packit 98cdb6
 * @size_group: a #GtkSizeGrup
Packit 98cdb6
 * 
Packit 98cdb6
 * Returns the list of widgets associated with @size_group.
Packit 98cdb6
 *
Packit 98cdb6
 * Return value:  (element-type GtkWidget) (transfer none): a #GSList of
Packit 98cdb6
 *   widgets. The list is owned by GTK+ and should not be modified.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.10
Packit 98cdb6
 **/
Packit 98cdb6
GSList *
Packit 98cdb6
gtk_size_group_get_widgets (GtkSizeGroup *size_group)
Packit 98cdb6
{
Packit 98cdb6
  return size_group->widgets;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gint
Packit 98cdb6
get_base_dimension (GtkWidget        *widget,
Packit 98cdb6
		    GtkSizeGroupMode  mode)
Packit 98cdb6
{
Packit 98cdb6
  GtkWidgetAuxInfo *aux_info = _gtk_widget_get_aux_info (widget, FALSE);
Packit 98cdb6
Packit 98cdb6
  if (mode == GTK_SIZE_GROUP_HORIZONTAL)
Packit 98cdb6
    {
Packit 98cdb6
      if (aux_info && aux_info->width > 0)
Packit 98cdb6
	return aux_info->width;
Packit 98cdb6
      else
Packit 98cdb6
	return widget->requisition.width;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      if (aux_info && aux_info->height > 0)
Packit 98cdb6
	return aux_info->height;
Packit 98cdb6
      else
Packit 98cdb6
	return widget->requisition.height;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
do_size_request (GtkWidget *widget)
Packit 98cdb6
{
Packit 98cdb6
  if (GTK_WIDGET_REQUEST_NEEDED (widget))
Packit 98cdb6
    {
Packit 98cdb6
      gtk_widget_ensure_style (widget);      
Packit 98cdb6
      GTK_PRIVATE_UNSET_FLAG (widget, GTK_REQUEST_NEEDED);
Packit 98cdb6
      g_signal_emit_by_name (widget,
Packit 98cdb6
			     "size-request",
Packit 98cdb6
			     &widget->requisition);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gint
Packit 98cdb6
compute_base_dimension (GtkWidget        *widget,
Packit 98cdb6
			GtkSizeGroupMode  mode)
Packit 98cdb6
{
Packit 98cdb6
  do_size_request (widget);
Packit 98cdb6
Packit 98cdb6
  return get_base_dimension (widget, mode);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gint
Packit 98cdb6
compute_dimension (GtkWidget        *widget,
Packit 98cdb6
		   GtkSizeGroupMode  mode)
Packit 98cdb6
{
Packit 98cdb6
  GSList *widgets = NULL;
Packit 98cdb6
  GSList *groups = NULL;
Packit 98cdb6
  GSList *tmp_list;
Packit 98cdb6
  gint result = 0;
Packit 98cdb6
Packit 98cdb6
  add_widget_to_closure (widget, mode, &groups, &widgets);
Packit 98cdb6
Packit 98cdb6
  g_slist_foreach (widgets, (GFunc)mark_unvisited, NULL);
Packit 98cdb6
  g_slist_foreach (groups, (GFunc)mark_unvisited, NULL);
Packit 98cdb6
  
Packit 98cdb6
  g_slist_foreach (widgets, (GFunc)g_object_ref, NULL);
Packit 98cdb6
  
Packit 98cdb6
  if (!groups)
Packit 98cdb6
    {
Packit 98cdb6
      result = compute_base_dimension (widget, mode);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      GtkSizeGroup *group = groups->data;
Packit 98cdb6
Packit 98cdb6
      if (mode == GTK_SIZE_GROUP_HORIZONTAL && group->have_width)
Packit 98cdb6
	result = group->requisition.width;
Packit 98cdb6
      else if (mode == GTK_SIZE_GROUP_VERTICAL && group->have_height)
Packit 98cdb6
	result = group->requisition.height;
Packit 98cdb6
      else
Packit 98cdb6
	{
Packit 98cdb6
	  tmp_list = widgets;
Packit 98cdb6
	  while (tmp_list)
Packit 98cdb6
	    {
Packit 98cdb6
	      GtkWidget *tmp_widget = tmp_list->data;
Packit 98cdb6
Packit 98cdb6
	      gint dimension = compute_base_dimension (tmp_widget, mode);
Packit 98cdb6
Packit 98cdb6
	      if (gtk_widget_get_mapped (tmp_widget) || !group->ignore_hidden)
Packit 98cdb6
		{
Packit 98cdb6
		  if (dimension > result)
Packit 98cdb6
		    result = dimension;
Packit 98cdb6
		}
Packit 98cdb6
Packit 98cdb6
	      tmp_list = tmp_list->next;
Packit 98cdb6
	    }
Packit 98cdb6
Packit 98cdb6
	  tmp_list = groups;
Packit 98cdb6
	  while (tmp_list)
Packit 98cdb6
	    {
Packit 98cdb6
	      GtkSizeGroup *tmp_group = tmp_list->data;
Packit 98cdb6
Packit 98cdb6
	      if (mode == GTK_SIZE_GROUP_HORIZONTAL)
Packit 98cdb6
		{
Packit 98cdb6
		  tmp_group->have_width = TRUE;
Packit 98cdb6
		  tmp_group->requisition.width = result;
Packit 98cdb6
		}
Packit 98cdb6
	      else
Packit 98cdb6
		{
Packit 98cdb6
		  tmp_group->have_height = TRUE;
Packit 98cdb6
		  tmp_group->requisition.height = result;
Packit 98cdb6
		}
Packit 98cdb6
	      
Packit 98cdb6
	      tmp_list = tmp_list->next;
Packit 98cdb6
	    }
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  g_slist_foreach (widgets, (GFunc)g_object_unref, NULL);
Packit 98cdb6
Packit 98cdb6
  g_slist_free (widgets);
Packit 98cdb6
  g_slist_free (groups);
Packit 98cdb6
Packit 98cdb6
  return result;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gint
Packit 98cdb6
get_dimension (GtkWidget        *widget,
Packit 98cdb6
	       GtkSizeGroupMode  mode)
Packit 98cdb6
{
Packit 98cdb6
  GSList *widgets = NULL;
Packit 98cdb6
  GSList *groups = NULL;
Packit 98cdb6
  gint result = 0;
Packit 98cdb6
Packit 98cdb6
  add_widget_to_closure (widget, mode, &groups, &widgets);
Packit 98cdb6
Packit 98cdb6
  g_slist_foreach (widgets, (GFunc)mark_unvisited, NULL);
Packit 98cdb6
  g_slist_foreach (groups, (GFunc)mark_unvisited, NULL);  
Packit 98cdb6
Packit 98cdb6
  if (!groups)
Packit 98cdb6
    {
Packit 98cdb6
      result = get_base_dimension (widget, mode);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      GtkSizeGroup *group = groups->data;
Packit 98cdb6
Packit 98cdb6
      if (mode == GTK_SIZE_GROUP_HORIZONTAL && group->have_width)
Packit 98cdb6
	result = group->requisition.width;
Packit 98cdb6
      else if (mode == GTK_SIZE_GROUP_VERTICAL && group->have_height)
Packit 98cdb6
	result = group->requisition.height;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  g_slist_free (widgets);
Packit 98cdb6
  g_slist_free (groups);
Packit 98cdb6
Packit 98cdb6
  return result;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
get_fast_child_requisition (GtkWidget      *widget,
Packit 98cdb6
			    GtkRequisition *requisition)
Packit 98cdb6
{
Packit 98cdb6
  GtkWidgetAuxInfo *aux_info = _gtk_widget_get_aux_info (widget, FALSE);
Packit 98cdb6
  
Packit 98cdb6
  *requisition = widget->requisition;
Packit 98cdb6
  
Packit 98cdb6
  if (aux_info)
Packit 98cdb6
    {
Packit 98cdb6
      if (aux_info->width > 0)
Packit 98cdb6
	requisition->width = aux_info->width;
Packit 98cdb6
      if (aux_info && aux_info->height > 0)
Packit 98cdb6
	requisition->height = aux_info->height;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gtk_size_group_get_child_requisition:
Packit 98cdb6
 * @widget: a #GtkWidget
Packit 98cdb6
 * @requisition: location to store computed requisition.
Packit 98cdb6
 * 
Packit 98cdb6
 * Retrieve the "child requisition" of the widget, taking account grouping
Packit 98cdb6
 * of the widget's requisition with other widgets.
Packit 98cdb6
 **/
Packit 98cdb6
void
Packit 98cdb6
_gtk_size_group_get_child_requisition (GtkWidget      *widget,
Packit 98cdb6
				       GtkRequisition *requisition)
Packit 98cdb6
{
Packit 98cdb6
  initialize_size_group_quarks ();
Packit 98cdb6
Packit 98cdb6
  if (requisition)
Packit 98cdb6
    {
Packit 98cdb6
      if (get_size_groups (widget))
Packit 98cdb6
	{
Packit 98cdb6
	  requisition->width = get_dimension (widget, GTK_SIZE_GROUP_HORIZONTAL);
Packit 98cdb6
	  requisition->height = get_dimension (widget, GTK_SIZE_GROUP_VERTICAL);
Packit 98cdb6
Packit 98cdb6
	  /* Only do the full computation if we actually have size groups */
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	get_fast_child_requisition (widget, requisition);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gtk_size_group_compute_requisition:
Packit 98cdb6
 * @widget: a #GtkWidget
Packit 98cdb6
 * @requisition: location to store computed requisition.
Packit 98cdb6
 * 
Packit 98cdb6
 * Compute the requisition of a widget taking into account grouping of
Packit 98cdb6
 * the widget's requisition with other widgets.
Packit 98cdb6
 **/
Packit 98cdb6
void
Packit 98cdb6
_gtk_size_group_compute_requisition (GtkWidget      *widget,
Packit 98cdb6
				     GtkRequisition *requisition)
Packit 98cdb6
{
Packit 98cdb6
  gint width;
Packit 98cdb6
  gint height;
Packit 98cdb6
Packit 98cdb6
  initialize_size_group_quarks ();
Packit 98cdb6
Packit 98cdb6
  if (get_size_groups (widget))
Packit 98cdb6
    {
Packit 98cdb6
      /* Only do the full computation if we actually have size groups */
Packit 98cdb6
      
Packit 98cdb6
      width = compute_dimension (widget, GTK_SIZE_GROUP_HORIZONTAL);
Packit 98cdb6
      height = compute_dimension (widget, GTK_SIZE_GROUP_VERTICAL);
Packit 98cdb6
Packit 98cdb6
      if (requisition)
Packit 98cdb6
	{
Packit 98cdb6
	  requisition->width = width;
Packit 98cdb6
	  requisition->height = height;
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      do_size_request (widget);
Packit 98cdb6
      
Packit 98cdb6
      if (requisition)
Packit 98cdb6
	get_fast_child_requisition (widget, requisition);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gtk_size_group_queue_resize:
Packit 98cdb6
 * @widget: a #GtkWidget
Packit 98cdb6
 * 
Packit 98cdb6
 * Queue a resize on a widget, and on all other widgets grouped with this widget.
Packit 98cdb6
 **/
Packit 98cdb6
void
Packit 98cdb6
_gtk_size_group_queue_resize (GtkWidget *widget)
Packit 98cdb6
{
Packit 98cdb6
  initialize_size_group_quarks ();
Packit 98cdb6
Packit 98cdb6
  queue_resize_on_widget (widget, TRUE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
typedef struct {
Packit 98cdb6
  GObject *object;
Packit 98cdb6
  GSList *items;
Packit 98cdb6
} GSListSubParserData;
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
size_group_start_element (GMarkupParseContext *context,
Packit 98cdb6
			  const gchar         *element_name,
Packit 98cdb6
			  const gchar        **names,
Packit 98cdb6
			  const gchar        **values,
Packit 98cdb6
			  gpointer            user_data,
Packit 98cdb6
			  GError            **error)
Packit 98cdb6
{
Packit 98cdb6
  guint i;
Packit 98cdb6
  GSListSubParserData *data = (GSListSubParserData*)user_data;
Packit 98cdb6
Packit 98cdb6
  if (strcmp (element_name, "widget") == 0)
Packit 98cdb6
    for (i = 0; names[i]; i++)
Packit 98cdb6
      if (strcmp (names[i], "name") == 0)
Packit 98cdb6
	data->items = g_slist_prepend (data->items, g_strdup (values[i]));
Packit 98cdb6
  else if (strcmp (element_name, "widgets") == 0)
Packit 98cdb6
    return;
Packit 98cdb6
  else
Packit 98cdb6
    g_warning ("Unsupported type tag for GtkSizeGroup: %s\n",
Packit 98cdb6
	       element_name);
Packit 98cdb6
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static const GMarkupParser size_group_parser =
Packit 98cdb6
  {
Packit 98cdb6
    size_group_start_element
Packit 98cdb6
  };
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_size_group_buildable_custom_tag_start (GtkBuildable  *buildable,
Packit 98cdb6
					   GtkBuilder    *builder,
Packit 98cdb6
					   GObject       *child,
Packit 98cdb6
					   const gchar   *tagname,
Packit 98cdb6
					   GMarkupParser *parser,
Packit 98cdb6
					   gpointer      *data)
Packit 98cdb6
{
Packit 98cdb6
  GSListSubParserData *parser_data;
Packit 98cdb6
Packit 98cdb6
  if (child)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  if (strcmp (tagname, "widgets") == 0)
Packit 98cdb6
    {
Packit 98cdb6
      parser_data = g_slice_new0 (GSListSubParserData);
Packit 98cdb6
      parser_data->items = NULL;
Packit 98cdb6
      parser_data->object = G_OBJECT (buildable);
Packit 98cdb6
Packit 98cdb6
      *parser = size_group_parser;
Packit 98cdb6
      *data = parser_data;
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return FALSE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_size_group_buildable_custom_finished (GtkBuildable  *buildable,
Packit 98cdb6
					  GtkBuilder    *builder,
Packit 98cdb6
					  GObject       *child,
Packit 98cdb6
					  const gchar   *tagname,
Packit 98cdb6
					  gpointer       user_data)
Packit 98cdb6
{
Packit 98cdb6
  GSList *l;
Packit 98cdb6
  GSListSubParserData *data;
Packit 98cdb6
  GObject *object;
Packit 98cdb6
Packit 98cdb6
  if (strcmp (tagname, "widgets"))
Packit 98cdb6
    return;
Packit 98cdb6
  
Packit 98cdb6
  data = (GSListSubParserData*)user_data;
Packit 98cdb6
  data->items = g_slist_reverse (data->items);
Packit 98cdb6
Packit 98cdb6
  for (l = data->items; l; l = l->next)
Packit 98cdb6
    {
Packit 98cdb6
      object = gtk_builder_get_object (builder, l->data);
Packit 98cdb6
      if (!object)
Packit 98cdb6
	{
Packit 98cdb6
	  g_warning ("Unknown object %s specified in sizegroup %s",
Packit 98cdb6
		     (const gchar*)l->data,
Packit 98cdb6
		     gtk_buildable_get_name (GTK_BUILDABLE (data->object)));
Packit 98cdb6
	  continue;
Packit 98cdb6
	}
Packit 98cdb6
      gtk_size_group_add_widget (GTK_SIZE_GROUP (data->object),
Packit 98cdb6
				 GTK_WIDGET (object));
Packit 98cdb6
      g_free (l->data);
Packit 98cdb6
    }
Packit 98cdb6
  g_slist_free (data->items);
Packit 98cdb6
  g_slice_free (GSListSubParserData, data);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
#define __GTK_SIZE_GROUP_C__
Packit 98cdb6
#include "gtkaliasdef.c"