3 * Copyright (C) 2010 Openismus GmbH
6 * Tristan Van Berkom <tristanvb@openismus.com>
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Library General Public
10 * License as published by the Free Software Foundation; either
11 * version 2 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Library General Public License for more details.
18 * You should have received a copy of the GNU Library General Public
19 * License along with this library; if not, write to the
20 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
21 * Boston, MA 02111-1307, USA.
26 #include "gtkmarshalers.h"
27 #include "gtkorientable.h"
28 #include "gtkcelllayout.h"
29 #include "gtkcellareaiter.h"
32 static void gtk_cell_area_iter_finalize (GObject *object);
33 static void gtk_cell_area_iter_get_property (GObject *object,
38 /* GtkCellAreaIterClass */
39 static void gtk_cell_area_iter_real_flush_preferred_width (GtkCellAreaIter *iter);
40 static void gtk_cell_area_iter_real_flush_preferred_height_for_width (GtkCellAreaIter *iter,
42 static void gtk_cell_area_iter_real_flush_preferred_height (GtkCellAreaIter *iter);
43 static void gtk_cell_area_iter_real_flush_preferred_width_for_height (GtkCellAreaIter *iter,
46 /* CachedSize management */
52 static CachedSize *cached_size_new (gint min_size, gint nat_size);
53 static void cached_size_free (CachedSize *size);
55 struct _GtkCellAreaIterPrivate
76 SIGNAL_HEIGHT_CHANGED,
80 static guint cell_area_iter_signals[LAST_SIGNAL] = { 0 };
82 G_DEFINE_TYPE (GtkCellAreaIter, gtk_cell_area_iter, G_TYPE_OBJECT);
85 gtk_cell_area_iter_init (GtkCellAreaIter *iter)
87 GtkCellAreaIterPrivate *priv;
89 iter->priv = G_TYPE_INSTANCE_GET_PRIVATE (iter,
90 GTK_TYPE_CELL_AREA_ITER,
91 GtkCellAreaIterPrivate);
96 priv->min_height = -1;
97 priv->nat_height = -1;
98 priv->widths = g_hash_table_new_full (g_direct_hash, g_direct_equal,
99 NULL, (GDestroyNotify)cached_size_free);
100 priv->heights = g_hash_table_new_full (g_direct_hash, g_direct_equal,
101 NULL, (GDestroyNotify)cached_size_free);
105 gtk_cell_area_iter_class_init (GtkCellAreaIterClass *class)
107 GObjectClass *object_class = G_OBJECT_CLASS (class);
110 object_class->finalize = gtk_cell_area_iter_finalize;
111 object_class->get_property = gtk_cell_area_iter_get_property;
113 class->flush_preferred_width = gtk_cell_area_iter_real_flush_preferred_width;
114 class->flush_preferred_height_for_width = gtk_cell_area_iter_real_flush_preferred_height_for_width;
115 class->flush_preferred_height = gtk_cell_area_iter_real_flush_preferred_height;
116 class->flush_preferred_width_for_height = gtk_cell_area_iter_real_flush_preferred_width_for_height;
118 cell_area_iter_signals[SIGNAL_HEIGHT_CHANGED] =
119 g_signal_new (I_("height-changed"),
120 G_TYPE_FROM_CLASS (object_class),
122 0, /* Class offset (just a notification, no class handler) */
124 _gtk_marshal_VOID__INT_INT_INT,
126 G_TYPE_INT, G_TYPE_INT, G_TYPE_INT);
128 cell_area_iter_signals[SIGNAL_WIDTH_CHANGED] =
129 g_signal_new (I_("width-changed"),
130 G_TYPE_FROM_CLASS (object_class),
132 0, /* Class offset (just a notification, no class handler) */
134 _gtk_marshal_VOID__INT_INT_INT,
136 G_TYPE_INT, G_TYPE_INT, G_TYPE_INT);
138 g_object_class_install_property (object_class,
140 g_param_spec_int ("minimum-width",
142 P_("Minimum cached width"),
148 g_object_class_install_property (object_class,
150 g_param_spec_int ("natural-width",
152 P_("Minimum cached width"),
158 g_object_class_install_property (object_class,
160 g_param_spec_int ("minimum-height",
161 P_("Minimum Height"),
162 P_("Minimum cached height"),
168 g_object_class_install_property (object_class,
170 g_param_spec_int ("natural-height",
171 P_("Minimum Height"),
172 P_("Minimum cached height"),
178 g_type_class_add_private (object_class, sizeof (GtkCellAreaIterPrivate));
183 /*************************************************************
185 *************************************************************/
187 cached_size_new (gint min_size,
190 CachedSize *size = g_slice_new (CachedSize);
192 size->min_size = min_size;
193 size->nat_size = nat_size;
199 cached_size_free (CachedSize *size)
201 g_slice_free (CachedSize, size);
204 /*************************************************************
206 *************************************************************/
208 gtk_cell_area_iter_finalize (GObject *object)
210 GtkCellAreaIter *iter = GTK_CELL_AREA_ITER (object);
211 GtkCellAreaIterPrivate *priv = iter->priv;
213 g_hash_table_destroy (priv->widths);
214 g_hash_table_destroy (priv->heights);
216 G_OBJECT_CLASS (gtk_cell_area_iter_parent_class)->finalize (object);
220 gtk_cell_area_iter_get_property (GObject *object,
225 GtkCellAreaIter *iter = GTK_CELL_AREA_ITER (object);
226 GtkCellAreaIterPrivate *priv = iter->priv;
231 g_value_set_int (value, priv->min_width);
234 g_value_set_int (value, priv->nat_width);
236 case PROP_MIN_HEIGHT:
237 g_value_set_int (value, priv->min_height);
239 case PROP_NAT_HEIGHT:
240 g_value_set_int (value, priv->nat_height);
243 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
248 /*************************************************************
249 * GtkCellAreaIterClass *
250 *************************************************************/
252 gtk_cell_area_iter_real_flush_preferred_width (GtkCellAreaIter *iter)
254 GtkCellAreaIterPrivate *priv = iter->priv;
256 priv->min_width = -1;
257 priv->nat_width = -1;
259 g_object_freeze_notify (G_OBJECT (iter));
260 g_object_notify (G_OBJECT (iter), "minimum-width");
261 g_object_notify (G_OBJECT (iter), "natural-width");
262 g_object_thaw_notify (G_OBJECT (iter));
266 gtk_cell_area_iter_real_flush_preferred_height_for_width (GtkCellAreaIter *iter,
269 GtkCellAreaIterPrivate *priv = iter->priv;
271 /* Flush all sizes for special -1 value */
273 g_hash_table_remove_all (priv->heights);
275 g_hash_table_remove (priv->heights, GINT_TO_POINTER (width));
277 /* XXX Should we bother signalling removed values as "size-changed" signals ? */
281 gtk_cell_area_iter_real_flush_preferred_height (GtkCellAreaIter *iter)
283 GtkCellAreaIterPrivate *priv = iter->priv;
285 priv->min_height = -1;
286 priv->nat_height = -1;
288 g_object_freeze_notify (G_OBJECT (iter));
289 g_object_notify (G_OBJECT (iter), "minimum-height");
290 g_object_notify (G_OBJECT (iter), "natural-height");
291 g_object_thaw_notify (G_OBJECT (iter));
295 gtk_cell_area_iter_real_flush_preferred_width_for_height (GtkCellAreaIter *iter,
298 GtkCellAreaIterPrivate *priv = iter->priv;
300 /* Flush all sizes for special -1 value */
302 g_hash_table_remove_all (priv->widths);
304 g_hash_table_remove (priv->widths, GINT_TO_POINTER (height));
306 /* XXX Should we bother signalling removed values as "size-changed" signals ? */
309 /*************************************************************
311 *************************************************************/
314 gtk_cell_area_iter_get_preferred_width (GtkCellAreaIter *iter,
318 GtkCellAreaIterPrivate *priv;
320 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
325 *minimum_width = priv->min_width;
328 *natural_width = priv->nat_width;
332 gtk_cell_area_iter_get_preferred_height_for_width (GtkCellAreaIter *iter,
334 gint *minimum_height,
335 gint *natural_height)
337 GtkCellAreaIterPrivate *priv;
340 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
344 size = g_hash_table_lookup (priv->heights, GINT_TO_POINTER (for_width));
349 *minimum_height = size->min_size;
352 *natural_height = size->nat_size;
357 *minimum_height = -1;
360 *natural_height = -1;
365 gtk_cell_area_iter_get_preferred_height (GtkCellAreaIter *iter,
366 gint *minimum_height,
367 gint *natural_height)
369 GtkCellAreaIterPrivate *priv;
371 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
376 *minimum_height = priv->min_height;
379 *natural_height = priv->nat_height;
383 gtk_cell_area_iter_get_preferred_width_for_height (GtkCellAreaIter *iter,
388 GtkCellAreaIterPrivate *priv;
391 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
395 size = g_hash_table_lookup (priv->widths, GINT_TO_POINTER (for_height));
400 *minimum_width = size->min_size;
403 *natural_width = size->nat_size;
417 gtk_cell_area_iter_push_preferred_width (GtkCellAreaIter *iter,
421 GtkCellAreaIterPrivate *priv;
423 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
427 g_object_freeze_notify (G_OBJECT (iter));
429 if (minimum_width > priv->min_width)
431 priv->min_width = minimum_width;
433 g_object_notify (G_OBJECT (iter), "minimum-width");
436 if (natural_width > priv->nat_width)
438 priv->nat_width = natural_width;
440 g_object_notify (G_OBJECT (iter), "natural-width");
443 g_object_thaw_notify (G_OBJECT (iter));
447 gtk_cell_area_iter_push_preferred_height_for_width (GtkCellAreaIter *iter,
452 GtkCellAreaIterPrivate *priv;
454 gboolean changed = FALSE;
456 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
460 size = g_hash_table_lookup (priv->heights, GINT_TO_POINTER (for_width));
464 size = cached_size_new (minimum_height, natural_height);
466 g_hash_table_insert (priv->heights, GINT_TO_POINTER (for_width), size);
472 if (minimum_height > size->min_size)
474 size->min_size = minimum_height;
478 if (natural_height > size->nat_size)
480 size->nat_size = natural_height;
486 g_signal_emit (iter, cell_area_iter_signals[SIGNAL_HEIGHT_CHANGED], 0,
487 for_width, size->min_size, size->nat_size);
491 gtk_cell_area_iter_push_preferred_height (GtkCellAreaIter *iter,
495 GtkCellAreaIterPrivate *priv;
497 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
501 g_object_freeze_notify (G_OBJECT (iter));
503 if (minimum_height > priv->min_height)
505 priv->min_height = minimum_height;
507 g_object_notify (G_OBJECT (iter), "minimum-height");
510 if (natural_height > priv->nat_height)
512 priv->nat_height = natural_height;
514 g_object_notify (G_OBJECT (iter), "natural-height");
517 g_object_thaw_notify (G_OBJECT (iter));
521 gtk_cell_area_iter_push_preferred_width_for_height (GtkCellAreaIter *iter,
526 GtkCellAreaIterPrivate *priv;
528 gboolean changed = FALSE;
530 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
534 size = g_hash_table_lookup (priv->widths, GINT_TO_POINTER (for_height));
538 size = cached_size_new (minimum_width, natural_width);
540 g_hash_table_insert (priv->widths, GINT_TO_POINTER (for_height), size);
546 if (minimum_width > size->min_size)
548 size->min_size = minimum_width;
552 if (natural_width > size->nat_size)
554 size->nat_size = natural_width;
560 g_signal_emit (iter, cell_area_iter_signals[SIGNAL_WIDTH_CHANGED], 0,
561 for_height, size->min_size, size->nat_size);
565 gtk_cell_area_iter_flush (GtkCellAreaIter *iter)
567 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
569 gtk_cell_area_iter_flush_preferred_width (iter);
570 gtk_cell_area_iter_flush_preferred_height_for_width (iter, -1);
571 gtk_cell_area_iter_flush_preferred_height (iter);
572 gtk_cell_area_iter_flush_preferred_width_for_height (iter, -1);
576 gtk_cell_area_iter_flush_preferred_width (GtkCellAreaIter *iter)
578 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
580 GTK_CELL_AREA_ITER_GET_CLASS (iter)->flush_preferred_width (iter);
584 gtk_cell_area_iter_flush_preferred_height_for_width (GtkCellAreaIter *iter,
587 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
589 GTK_CELL_AREA_ITER_GET_CLASS (iter)->flush_preferred_height_for_width (iter, for_width);
593 gtk_cell_area_iter_flush_preferred_height (GtkCellAreaIter *iter)
595 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
597 GTK_CELL_AREA_ITER_GET_CLASS (iter)->flush_preferred_height (iter);
601 gtk_cell_area_iter_flush_preferred_width_for_height (GtkCellAreaIter *iter,
604 g_return_if_fail (GTK_IS_CELL_AREA_ITER (iter));
606 GTK_CELL_AREA_ITER_GET_CLASS (iter)->flush_preferred_width_for_height (iter, for_height);