[StOverflowBox] Vertical box which skips painting underallocated children
https://bugzilla.gnome.org/show_bug.cgi?id=603523
This commit is contained in:
parent
c2fe9e57b1
commit
14df7cd62c
@ -78,6 +78,7 @@ st_source_h = \
|
|||||||
st/st-entry.h \
|
st/st-entry.h \
|
||||||
st/st-im-text.h \
|
st/st-im-text.h \
|
||||||
st/st-label.h \
|
st/st-label.h \
|
||||||
|
st/st-overflow-box.h \
|
||||||
st/st-private.h \
|
st/st-private.h \
|
||||||
st/st-scrollable.h \
|
st/st-scrollable.h \
|
||||||
st/st-scroll-bar.h \
|
st/st-scroll-bar.h \
|
||||||
@ -116,6 +117,7 @@ st_source_c = \
|
|||||||
st/st-entry.c \
|
st/st-entry.c \
|
||||||
st/st-im-text.c \
|
st/st-im-text.c \
|
||||||
st/st-label.c \
|
st/st-label.c \
|
||||||
|
st/st-overflow-box.c \
|
||||||
st/st-private.c \
|
st/st-private.c \
|
||||||
st/st-scrollable.c \
|
st/st-scrollable.c \
|
||||||
st/st-scroll-bar.c \
|
st/st-scroll-bar.c \
|
||||||
|
706
src/st/st-overflow-box.c
Normal file
706
src/st/st-overflow-box.c
Normal file
@ -0,0 +1,706 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
||||||
|
/*
|
||||||
|
* Portions derived from st-box-layout.c, which is
|
||||||
|
* Copyright 2009 Intel Corporation.
|
||||||
|
* Modified into -overflow-box, by Colin Walters <walters@verbum.org>, which is
|
||||||
|
* Copyright 2009 Red Hat, Inc.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms and conditions of the GNU Lesser General Public License,
|
||||||
|
* version 2.1, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope 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 License
|
||||||
|
* along with this program; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SECTION:st-overflow-box
|
||||||
|
* @short_description: A vertical box which paints as many actors as it can fit
|
||||||
|
*
|
||||||
|
* This is a "flexible" box which will paint as many actors as it can within
|
||||||
|
* its given allocation; its minimum height request will be the sum of the
|
||||||
|
* mimimum size for the #StOverflowBox:min-children property, which is
|
||||||
|
* by default 0.
|
||||||
|
*
|
||||||
|
* Every child will be allocated the full width of the box, and always be
|
||||||
|
* given its preferred height. Even if not actually painted, every child
|
||||||
|
* is counted for overall preferred width/height.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "st-overflow-box.h"
|
||||||
|
|
||||||
|
#include "st-private.h"
|
||||||
|
#include "st-box-layout-child.h"
|
||||||
|
|
||||||
|
static void st_overflow_box_container_iface_init (ClutterContainerIface *iface);
|
||||||
|
|
||||||
|
G_DEFINE_TYPE_WITH_CODE (StOverflowBox, st_overflow_box, ST_TYPE_WIDGET,
|
||||||
|
G_IMPLEMENT_INTERFACE (CLUTTER_TYPE_CONTAINER,
|
||||||
|
st_overflow_box_container_iface_init));
|
||||||
|
|
||||||
|
#define OVERFLOW_BOX_LAYOUT_PRIVATE(o) \
|
||||||
|
(G_TYPE_INSTANCE_GET_PRIVATE ((o), ST_TYPE_OVERFLOW_BOX, StOverflowBoxPrivate))
|
||||||
|
|
||||||
|
enum {
|
||||||
|
PROP_0,
|
||||||
|
|
||||||
|
PROP_MIN_CHILDREN
|
||||||
|
};
|
||||||
|
|
||||||
|
struct _StOverflowBoxPrivate
|
||||||
|
{
|
||||||
|
GList *children;
|
||||||
|
guint min_children;
|
||||||
|
guint n_visible;
|
||||||
|
|
||||||
|
guint spacing;
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* ClutterContainer Implementation
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
st_overflow_box_add_actor (ClutterContainer *container,
|
||||||
|
ClutterActor *actor)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (container)->priv;
|
||||||
|
|
||||||
|
clutter_actor_set_parent (actor, CLUTTER_ACTOR (container));
|
||||||
|
|
||||||
|
priv->children = g_list_append (priv->children, actor);
|
||||||
|
|
||||||
|
clutter_actor_queue_relayout (CLUTTER_ACTOR (container));
|
||||||
|
|
||||||
|
g_signal_emit_by_name (container, "actor-added", actor);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_remove_actor (ClutterContainer *container,
|
||||||
|
ClutterActor *actor)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (container)->priv;
|
||||||
|
|
||||||
|
GList *item = NULL;
|
||||||
|
|
||||||
|
item = g_list_find (priv->children, actor);
|
||||||
|
|
||||||
|
if (item == NULL)
|
||||||
|
{
|
||||||
|
g_warning ("Actor of type '%s' is not a child of container of type '%s'",
|
||||||
|
g_type_name (G_OBJECT_TYPE (actor)),
|
||||||
|
g_type_name (G_OBJECT_TYPE (container)));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_object_ref (actor);
|
||||||
|
|
||||||
|
priv->children = g_list_delete_link (priv->children, item);
|
||||||
|
clutter_actor_unparent (actor);
|
||||||
|
|
||||||
|
g_signal_emit_by_name (container, "actor-removed", actor);
|
||||||
|
|
||||||
|
g_object_unref (actor);
|
||||||
|
|
||||||
|
clutter_actor_queue_relayout ((ClutterActor*) container);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_foreach (ClutterContainer *container,
|
||||||
|
ClutterCallback callback,
|
||||||
|
gpointer callback_data)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (container)->priv;
|
||||||
|
|
||||||
|
g_list_foreach (priv->children, (GFunc) callback, callback_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_lower (ClutterContainer *container,
|
||||||
|
ClutterActor *actor,
|
||||||
|
ClutterActor *sibling)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (container)->priv;
|
||||||
|
|
||||||
|
/* copied from clutter/clutter/clutter-group.c */
|
||||||
|
|
||||||
|
priv->children = g_list_remove (priv->children, actor);
|
||||||
|
|
||||||
|
/* Push to bottom */
|
||||||
|
if (!sibling)
|
||||||
|
{
|
||||||
|
GList *last_item;
|
||||||
|
|
||||||
|
last_item = g_list_first (priv->children);
|
||||||
|
|
||||||
|
if (last_item)
|
||||||
|
sibling = last_item->data;
|
||||||
|
|
||||||
|
priv->children = g_list_prepend (priv->children, actor);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
gint pos;
|
||||||
|
|
||||||
|
pos = g_list_index (priv->children, sibling);
|
||||||
|
|
||||||
|
priv->children = g_list_insert (priv->children, actor, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* See comment in group_raise for this */
|
||||||
|
if (sibling &&
|
||||||
|
clutter_actor_get_depth (sibling) != clutter_actor_get_depth (actor))
|
||||||
|
{
|
||||||
|
clutter_actor_set_depth (actor, clutter_actor_get_depth (sibling));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (CLUTTER_ACTOR_IS_VISIBLE (container))
|
||||||
|
clutter_actor_queue_redraw (CLUTTER_ACTOR (container));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_raise (ClutterContainer *container,
|
||||||
|
ClutterActor *actor,
|
||||||
|
ClutterActor *sibling)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (container)->priv;
|
||||||
|
|
||||||
|
priv->children = g_list_remove (priv->children, actor);
|
||||||
|
|
||||||
|
/* copied from clutter/clutter/clutter-group.c */
|
||||||
|
|
||||||
|
/* Raise at the top */
|
||||||
|
if (!sibling)
|
||||||
|
{
|
||||||
|
GList *last_item;
|
||||||
|
|
||||||
|
last_item = g_list_last (priv->children);
|
||||||
|
|
||||||
|
if (last_item)
|
||||||
|
sibling = last_item->data;
|
||||||
|
|
||||||
|
priv->children = g_list_append (priv->children, actor);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
gint pos;
|
||||||
|
|
||||||
|
pos = g_list_index (priv->children, sibling) + 1;
|
||||||
|
|
||||||
|
priv->children = g_list_insert (priv->children, actor, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* set Z ordering a value below, this will then call sort
|
||||||
|
* as values are equal ordering shouldn't change but Z
|
||||||
|
* values will be correct.
|
||||||
|
*
|
||||||
|
* FIXME: optimise
|
||||||
|
*/
|
||||||
|
if (sibling &&
|
||||||
|
clutter_actor_get_depth (sibling) != clutter_actor_get_depth (actor))
|
||||||
|
{
|
||||||
|
clutter_actor_set_depth (actor, clutter_actor_get_depth (sibling));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (CLUTTER_ACTOR_IS_VISIBLE (container))
|
||||||
|
clutter_actor_queue_redraw (CLUTTER_ACTOR (container));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_sort_depth_order (ClutterContainer *container)
|
||||||
|
{
|
||||||
|
/* XXX: not yet implemented */
|
||||||
|
g_warning ("%s() not yet implemented", __FUNCTION__);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_container_iface_init (ClutterContainerIface *iface)
|
||||||
|
{
|
||||||
|
iface->add = st_overflow_box_add_actor;
|
||||||
|
iface->remove = st_overflow_box_remove_actor;
|
||||||
|
iface->foreach = st_overflow_box_foreach;
|
||||||
|
iface->lower = st_overflow_box_lower;
|
||||||
|
iface->raise = st_overflow_box_raise;
|
||||||
|
iface->sort_depth_order = st_overflow_box_sort_depth_order;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_get_property (GObject *object,
|
||||||
|
guint property_id,
|
||||||
|
GValue *value,
|
||||||
|
GParamSpec *pspec)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (object)->priv;
|
||||||
|
|
||||||
|
switch (property_id)
|
||||||
|
{
|
||||||
|
case PROP_MIN_CHILDREN:
|
||||||
|
g_value_set_uint (value, priv->min_children);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_set_property (GObject *object,
|
||||||
|
guint property_id,
|
||||||
|
const GValue *value,
|
||||||
|
GParamSpec *pspec)
|
||||||
|
{
|
||||||
|
StOverflowBox *box = ST_OVERFLOW_BOX (object);
|
||||||
|
|
||||||
|
switch (property_id)
|
||||||
|
{
|
||||||
|
case PROP_MIN_CHILDREN:
|
||||||
|
st_overflow_box_set_min_children (box, g_value_get_uint (value));
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_dispose (GObject *object)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (object)->priv;
|
||||||
|
|
||||||
|
while (priv->children)
|
||||||
|
clutter_actor_destroy (priv->children->data);
|
||||||
|
|
||||||
|
G_OBJECT_CLASS (st_overflow_box_parent_class)->dispose (object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
get_content_preferred_width (StOverflowBox *self,
|
||||||
|
gfloat for_height,
|
||||||
|
gfloat *min_width_p,
|
||||||
|
gfloat *natural_width_p)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = self->priv;
|
||||||
|
gint n_children = 0;
|
||||||
|
gint n_fixed = 0;
|
||||||
|
gfloat min_width, natural_width;
|
||||||
|
GList *l;
|
||||||
|
|
||||||
|
min_width = 0;
|
||||||
|
natural_width = 0;
|
||||||
|
|
||||||
|
for (l = priv->children; l; l = g_list_next (l))
|
||||||
|
{
|
||||||
|
ClutterActor *child = l->data;
|
||||||
|
gfloat child_min = 0, child_nat = 0;
|
||||||
|
|
||||||
|
if (!CLUTTER_ACTOR_IS_VISIBLE (child))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
n_children++;
|
||||||
|
|
||||||
|
if (clutter_actor_get_fixed_position_set (child))
|
||||||
|
{
|
||||||
|
n_fixed++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
clutter_actor_get_preferred_width (child,
|
||||||
|
-1,
|
||||||
|
&child_min,
|
||||||
|
&child_nat);
|
||||||
|
|
||||||
|
min_width = MAX (child_min, min_width);
|
||||||
|
natural_width = MAX (child_nat, natural_width);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((n_children - n_fixed) > 1)
|
||||||
|
{
|
||||||
|
min_width += priv->spacing * (n_children - n_fixed - 1);
|
||||||
|
natural_width += priv->spacing * (n_children - n_fixed - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (min_width_p)
|
||||||
|
*min_width_p = min_width;
|
||||||
|
|
||||||
|
if (natural_width_p)
|
||||||
|
*natural_width_p = natural_width;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_get_preferred_width (ClutterActor *actor,
|
||||||
|
gfloat for_height,
|
||||||
|
gfloat *min_width_p,
|
||||||
|
gfloat *natural_width_p)
|
||||||
|
{
|
||||||
|
StThemeNode *theme_node = st_widget_get_theme_node (ST_WIDGET (actor));
|
||||||
|
|
||||||
|
st_theme_node_adjust_for_height (theme_node, &for_height);
|
||||||
|
|
||||||
|
get_content_preferred_width (ST_OVERFLOW_BOX (actor), for_height,
|
||||||
|
min_width_p, natural_width_p);
|
||||||
|
|
||||||
|
st_theme_node_adjust_preferred_width (theme_node,
|
||||||
|
min_width_p, natural_width_p);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
get_content_preferred_height (StOverflowBox *self,
|
||||||
|
gfloat for_width,
|
||||||
|
gfloat *min_height_p,
|
||||||
|
gfloat *natural_height_p)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = self->priv;
|
||||||
|
gint n_min_children = 0;
|
||||||
|
gint n_children = 0;
|
||||||
|
gint n_fixed = 0;
|
||||||
|
gfloat min_height, natural_height;
|
||||||
|
GList *l;
|
||||||
|
|
||||||
|
min_height = 0;
|
||||||
|
natural_height = 0;
|
||||||
|
|
||||||
|
for (l = priv->children; l; l = g_list_next (l))
|
||||||
|
{
|
||||||
|
ClutterActor *child = l->data;
|
||||||
|
gfloat child_min = 0, child_nat = 0;
|
||||||
|
|
||||||
|
if (!CLUTTER_ACTOR_IS_VISIBLE (child))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
n_children++;
|
||||||
|
|
||||||
|
if (clutter_actor_get_fixed_position_set (child))
|
||||||
|
{
|
||||||
|
n_fixed++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
clutter_actor_get_preferred_height (child,
|
||||||
|
for_width,
|
||||||
|
&child_min,
|
||||||
|
&child_nat);
|
||||||
|
|
||||||
|
if (n_children < priv->min_children)
|
||||||
|
{
|
||||||
|
n_min_children++;
|
||||||
|
min_height += child_min;
|
||||||
|
}
|
||||||
|
natural_height += child_nat;
|
||||||
|
}
|
||||||
|
|
||||||
|
min_height += priv->spacing * MAX(0, n_min_children - 1);
|
||||||
|
natural_height += priv->spacing * MAX(0, n_children - n_fixed - 1);
|
||||||
|
|
||||||
|
if (min_height_p)
|
||||||
|
*min_height_p = min_height;
|
||||||
|
|
||||||
|
if (natural_height_p)
|
||||||
|
*natural_height_p = natural_height;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_get_preferred_height (ClutterActor *actor,
|
||||||
|
gfloat for_width,
|
||||||
|
gfloat *min_height_p,
|
||||||
|
gfloat *natural_height_p)
|
||||||
|
{
|
||||||
|
StThemeNode *theme_node = st_widget_get_theme_node (ST_WIDGET (actor));
|
||||||
|
|
||||||
|
st_theme_node_adjust_for_width (theme_node, &for_width);
|
||||||
|
|
||||||
|
get_content_preferred_height (ST_OVERFLOW_BOX (actor), for_width,
|
||||||
|
min_height_p, natural_height_p);
|
||||||
|
|
||||||
|
st_theme_node_adjust_preferred_height (theme_node,
|
||||||
|
min_height_p, natural_height_p);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_allocate (ClutterActor *actor,
|
||||||
|
const ClutterActorBox *box,
|
||||||
|
ClutterAllocationFlags flags)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (actor)->priv;
|
||||||
|
StThemeNode *theme_node = st_widget_get_theme_node (ST_WIDGET (actor));
|
||||||
|
ClutterActorBox content_box;
|
||||||
|
gfloat position;
|
||||||
|
float avail_width, avail_height;
|
||||||
|
GList *l;
|
||||||
|
int i;
|
||||||
|
gboolean done_non_fixed;
|
||||||
|
|
||||||
|
CLUTTER_ACTOR_CLASS (st_overflow_box_parent_class)->allocate (actor, box,
|
||||||
|
flags);
|
||||||
|
|
||||||
|
if (priv->children == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
st_theme_node_get_content_box (theme_node, box, &content_box);
|
||||||
|
|
||||||
|
avail_width = content_box.x2 - content_box.x1;
|
||||||
|
avail_height = content_box.y2 - content_box.y1;
|
||||||
|
|
||||||
|
position = content_box.y1;
|
||||||
|
priv->n_visible = 0;
|
||||||
|
|
||||||
|
done_non_fixed = FALSE;
|
||||||
|
for (l = priv->children, i = 0; l; l = l->next, i++)
|
||||||
|
{
|
||||||
|
ClutterActor *child = (ClutterActor*) l->data;
|
||||||
|
ClutterActorBox child_box;
|
||||||
|
gfloat child_min, child_nat;
|
||||||
|
gboolean fixed;
|
||||||
|
|
||||||
|
if (!CLUTTER_ACTOR_IS_VISIBLE (child))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
fixed = clutter_actor_get_fixed_position_set (child);
|
||||||
|
if (fixed)
|
||||||
|
{
|
||||||
|
clutter_actor_allocate_preferred_size (child, flags);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
else if (done_non_fixed)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
clutter_actor_get_preferred_height (child, avail_width,
|
||||||
|
&child_min, &child_nat);
|
||||||
|
|
||||||
|
if (position + child_nat > content_box.y2)
|
||||||
|
{
|
||||||
|
done_non_fixed = TRUE; /* Continue iterating on non fixed */
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
priv->n_visible++;
|
||||||
|
child_box.y1 = (int)(0.5 + position);
|
||||||
|
child_box.y2 = child_box.y1 + (int)(0.5 + child_nat);
|
||||||
|
child_box.x1 = content_box.x1;
|
||||||
|
child_box.x2 = content_box.x2;
|
||||||
|
|
||||||
|
position += child_nat + priv->spacing;
|
||||||
|
|
||||||
|
clutter_actor_allocate (child, &child_box, flags);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_internal_paint (StOverflowBox *box)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = box->priv;
|
||||||
|
GList *l;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
i = 0;
|
||||||
|
for (l = priv->children; i < priv->n_visible && l; l = l->next)
|
||||||
|
{
|
||||||
|
ClutterActor *child = (ClutterActor*) l->data;
|
||||||
|
|
||||||
|
if (!CLUTTER_ACTOR_IS_VISIBLE (child))
|
||||||
|
continue;
|
||||||
|
if (!clutter_actor_get_fixed_position_set (child))
|
||||||
|
i++;
|
||||||
|
|
||||||
|
clutter_actor_paint (child);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (;l; l = l->next)
|
||||||
|
{
|
||||||
|
ClutterActor *child = (ClutterActor*) l->data;
|
||||||
|
|
||||||
|
if (!CLUTTER_ACTOR_IS_VISIBLE (child))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (clutter_actor_get_fixed_position_set (child))
|
||||||
|
clutter_actor_paint (child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_paint (ClutterActor *actor)
|
||||||
|
{
|
||||||
|
CLUTTER_ACTOR_CLASS (st_overflow_box_parent_class)->paint (actor);
|
||||||
|
|
||||||
|
st_overflow_box_internal_paint (ST_OVERFLOW_BOX (actor));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_pick (ClutterActor *actor,
|
||||||
|
const ClutterColor *color)
|
||||||
|
{
|
||||||
|
CLUTTER_ACTOR_CLASS (st_overflow_box_parent_class)->pick (actor, color);
|
||||||
|
|
||||||
|
st_overflow_box_internal_paint (ST_OVERFLOW_BOX (actor));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_style_changed (StWidget *self)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (self)->priv;
|
||||||
|
StThemeNode *theme_node = st_widget_get_theme_node (self);
|
||||||
|
int old_spacing = priv->spacing;
|
||||||
|
double spacing = 0;
|
||||||
|
|
||||||
|
st_theme_node_get_length (theme_node, "spacing", FALSE, &spacing);
|
||||||
|
priv->spacing = (int)(spacing + 0.5);
|
||||||
|
if (priv->spacing != old_spacing)
|
||||||
|
clutter_actor_queue_relayout (CLUTTER_ACTOR (self));
|
||||||
|
|
||||||
|
ST_WIDGET_CLASS (st_overflow_box_parent_class)->style_changed (self);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_class_init (StOverflowBoxClass *klass)
|
||||||
|
{
|
||||||
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||||||
|
ClutterActorClass *actor_class = CLUTTER_ACTOR_CLASS (klass);
|
||||||
|
StWidgetClass *widget_class = ST_WIDGET_CLASS (klass);
|
||||||
|
GParamSpec *pspec;
|
||||||
|
|
||||||
|
g_type_class_add_private (klass, sizeof (StOverflowBoxPrivate));
|
||||||
|
|
||||||
|
object_class->get_property = st_overflow_box_get_property;
|
||||||
|
object_class->set_property = st_overflow_box_set_property;
|
||||||
|
object_class->dispose = st_overflow_box_dispose;
|
||||||
|
|
||||||
|
actor_class->allocate = st_overflow_box_allocate;
|
||||||
|
actor_class->get_preferred_width = st_overflow_box_get_preferred_width;
|
||||||
|
actor_class->get_preferred_height = st_overflow_box_get_preferred_height;
|
||||||
|
actor_class->paint = st_overflow_box_paint;
|
||||||
|
actor_class->pick = st_overflow_box_pick;
|
||||||
|
|
||||||
|
widget_class->style_changed = st_overflow_box_style_changed;
|
||||||
|
|
||||||
|
pspec = g_param_spec_uint ("min-children",
|
||||||
|
"Min Children",
|
||||||
|
"The actor will request a minimum size large enough to include this many children",
|
||||||
|
0, G_MAXUINT, 0,
|
||||||
|
ST_PARAM_READWRITE);
|
||||||
|
g_object_class_install_property (object_class, PROP_MIN_CHILDREN, pspec);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_init (StOverflowBox *self)
|
||||||
|
{
|
||||||
|
self->priv = OVERFLOW_BOX_LAYOUT_PRIVATE (self);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* st_overflow_box_get_min_children:
|
||||||
|
* @box: A #StOverflowBox
|
||||||
|
*
|
||||||
|
* Get the value of the #StOverflowBox::pack-start property.
|
||||||
|
*
|
||||||
|
* Returns: #TRUE if pack-start is enabled
|
||||||
|
*/
|
||||||
|
gboolean
|
||||||
|
st_overflow_box_get_min_children (StOverflowBox *box)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (ST_IS_OVERFLOW_BOX (box), FALSE);
|
||||||
|
|
||||||
|
return box->priv->min_children;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* st_box_layout_set_min_children:
|
||||||
|
* @box: A #StOverflowBox
|
||||||
|
* @min_children: Minimum children value
|
||||||
|
*
|
||||||
|
* Set the minimum number of children to be visible.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
st_overflow_box_set_min_children (StOverflowBox *box,
|
||||||
|
guint min_children)
|
||||||
|
{
|
||||||
|
g_return_if_fail (ST_IS_OVERFLOW_BOX (box));
|
||||||
|
|
||||||
|
if (box->priv->min_children != min_children)
|
||||||
|
{
|
||||||
|
box->priv->min_children = min_children;
|
||||||
|
clutter_actor_queue_relayout ((ClutterActor*) box);
|
||||||
|
|
||||||
|
g_object_notify (G_OBJECT (box), "min-children");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
|
st_overflow_box_internal_remove_all (StOverflowBox *self,
|
||||||
|
gboolean destroy)
|
||||||
|
{
|
||||||
|
StOverflowBoxPrivate *priv = ST_OVERFLOW_BOX (self)->priv;
|
||||||
|
ClutterActor *child;
|
||||||
|
|
||||||
|
while (priv->children)
|
||||||
|
{
|
||||||
|
child = priv->children->data;
|
||||||
|
|
||||||
|
g_object_ref (child);
|
||||||
|
priv->children = g_list_delete_link (priv->children, priv->children);
|
||||||
|
clutter_actor_unparent (child);
|
||||||
|
g_signal_emit_by_name (self, "actor-removed", child);
|
||||||
|
if (destroy)
|
||||||
|
clutter_actor_destroy (child);
|
||||||
|
g_object_unref (child);
|
||||||
|
}
|
||||||
|
|
||||||
|
clutter_actor_queue_relayout ((ClutterActor*) self);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* st_overflow_box_remove_all:
|
||||||
|
* @self:
|
||||||
|
*
|
||||||
|
* Efficiently unparent all children currently in this box.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
st_overflow_box_remove_all (StOverflowBox *self)
|
||||||
|
{
|
||||||
|
st_overflow_box_internal_remove_all (self, FALSE);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* st_overflow_box_destroy_children:
|
||||||
|
* @self:
|
||||||
|
*
|
||||||
|
* Efficiently unparent and destroy all children currently in this box.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
st_overflow_box_destroy_children (StOverflowBox *self)
|
||||||
|
{
|
||||||
|
st_overflow_box_internal_remove_all (self, TRUE);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* st_overflow_box_get_n_children:
|
||||||
|
* @self: a #StOverflowBox
|
||||||
|
*
|
||||||
|
* Returns the number of children in this box.
|
||||||
|
*/
|
||||||
|
guint
|
||||||
|
st_overflow_box_get_n_children (StOverflowBox *self)
|
||||||
|
{
|
||||||
|
return g_list_length (self->priv->children);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* st_overflow_box_get_n_visible:
|
||||||
|
* @self: a #StOverflowBox
|
||||||
|
*
|
||||||
|
* Returns the number of children we will paint. Only valid
|
||||||
|
* after the actor has been allocated.
|
||||||
|
*/
|
||||||
|
guint
|
||||||
|
st_overflow_box_get_n_visible (StOverflowBox *self)
|
||||||
|
{
|
||||||
|
return self->priv->n_visible;
|
||||||
|
}
|
75
src/st/st-overflow-box.h
Normal file
75
src/st/st-overflow-box.h
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
||||||
|
/*
|
||||||
|
* st-overflow-box.h: box which hides actors that don't fit
|
||||||
|
*
|
||||||
|
* Copyright 2009 Red Hat, Inc.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms and conditions of the GNU Lesser General Public License,
|
||||||
|
* version 2.1, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope 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 License
|
||||||
|
* along with this program; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if !defined(ST_H_INSIDE) && !defined(ST_COMPILATION)
|
||||||
|
#error "Only <st/st.h> can be included directly.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef _ST_OVERFLOW_BOX_H
|
||||||
|
#define _ST_OVERFLOW_BOX_H
|
||||||
|
|
||||||
|
#include <st/st-box-layout.h>
|
||||||
|
|
||||||
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
#define ST_TYPE_OVERFLOW_BOX st_overflow_box_get_type()
|
||||||
|
|
||||||
|
#define ST_OVERFLOW_BOX(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), ST_TYPE_OVERFLOW_BOX, StOverflowBox))
|
||||||
|
#define ST_OVERFLOW_BOX_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), ST_TYPE_OVERFLOW_BOX, StOverflowBoxClass))
|
||||||
|
#define ST_IS_OVERFLOW_BOX(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), ST_TYPE_OVERFLOW_BOX))
|
||||||
|
#define ST_IS_OVERFLOW_BOX_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), ST_TYPE_OVERFLOW_BOX))
|
||||||
|
#define ST_OVERFLOW_BOX_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), ST_TYPE_OVERFLOW_BOX, StOverflowBoxClass))
|
||||||
|
|
||||||
|
typedef struct _StOverflowBox StOverflowBox;
|
||||||
|
typedef struct _StOverflowBoxClass StOverflowBoxClass;
|
||||||
|
typedef struct _StOverflowBoxPrivate StOverflowBoxPrivate;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* StOverflowBox:
|
||||||
|
*
|
||||||
|
* The contents of this structure are private and should only be accessed
|
||||||
|
* through the public API.
|
||||||
|
*/
|
||||||
|
struct _StOverflowBox
|
||||||
|
{
|
||||||
|
/*< private >*/
|
||||||
|
StWidget parent;
|
||||||
|
|
||||||
|
StOverflowBoxPrivate *priv;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct _StOverflowBoxClass
|
||||||
|
{
|
||||||
|
StWidgetClass parent_class;
|
||||||
|
};
|
||||||
|
|
||||||
|
GType st_overflow_box_get_type (void);
|
||||||
|
|
||||||
|
void st_overflow_box_set_min_children (StOverflowBox *self, guint min_children);
|
||||||
|
void st_overflow_box_remove_all (StOverflowBox *box);
|
||||||
|
void st_overflow_box_destroy_children (StOverflowBox *box);
|
||||||
|
guint st_overflow_box_get_n_children (StOverflowBox *box);
|
||||||
|
guint st_overflow_box_get_n_visible (StOverflowBox *box);
|
||||||
|
gboolean st_overflow_box_get_min_children (StOverflowBox *box);
|
||||||
|
|
||||||
|
G_END_DECLS
|
||||||
|
|
||||||
|
#endif /* _ST_OVERFLOW_BOX_H */
|
Loading…
Reference in New Issue
Block a user