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.
25 #include "cellareascaffold.h"
28 static void cell_area_scaffold_finalize (GObject *object);
29 static void cell_area_scaffold_dispose (GObject *object);
30 static void cell_area_scaffold_set_property (GObject *object,
34 static void cell_area_scaffold_get_property (GObject *object,
40 static void cell_area_scaffold_realize (GtkWidget *widget);
41 static void cell_area_scaffold_unrealize (GtkWidget *widget);
42 static gboolean cell_area_scaffold_draw (GtkWidget *widget,
44 static void cell_area_scaffold_size_allocate (GtkWidget *widget,
45 GtkAllocation *allocation);
46 static void cell_area_scaffold_get_preferred_width (GtkWidget *widget,
49 static void cell_area_scaffold_get_preferred_height_for_width (GtkWidget *widget,
53 static void cell_area_scaffold_get_preferred_height (GtkWidget *widget,
56 static void cell_area_scaffold_get_preferred_width_for_height (GtkWidget *widget,
60 static gint cell_area_scaffold_focus (GtkWidget *widget,
61 GtkDirectionType direction);
63 /* CellAreaScaffoldClass */
64 static void cell_area_scaffold_activate (CellAreaScaffold *scaffold);
66 /* CellArea/GtkTreeModel callbacks */
67 static void size_changed_cb (GtkCellAreaIter *iter,
69 CellAreaScaffold *scaffold);
70 static void row_changed_cb (GtkTreeModel *model,
73 CellAreaScaffold *scaffold);
74 static void row_inserted_cb (GtkTreeModel *model,
77 CellAreaScaffold *scaffold);
78 static void row_deleted_cb (GtkTreeModel *model,
80 CellAreaScaffold *scaffold);
81 static void rows_reordered_cb (GtkTreeModel *model,
85 CellAreaScaffold *scaffold);
88 gint size; /* The size of the row in the scaffold's opposing orientation */
91 struct _CellAreaScaffoldPrivate {
93 /* Window for catching events and dispatching them to the cell area */
94 GdkWindow *event_window;
96 /* The model we're showing data for */
98 gulong row_changed_id;
99 gulong row_inserted_id;
100 gulong row_deleted_id;
101 gulong rows_reordered_id;
103 /* The area rendering the data and a global iter */
105 GtkCellAreaIter *iter;
107 /* Cache some info about rows (hieghts etc) */
113 /* Check when the underlying area changes the size and
114 * we need to queue a redraw */
115 gulong size_changed_id;
130 static guint scaffold_signals[N_SIGNALS] = { 0 };
132 #define ROW_SPACING 2
134 #define DIRECTION_STR(dir) \
135 ((dir) == GTK_DIR_TAB_FORWARD ? "tab forward" : \
136 (dir) == GTK_DIR_TAB_BACKWARD ? "tab backward" : \
137 (dir) == GTK_DIR_UP ? "up" : \
138 (dir) == GTK_DIR_DOWN ? "down" : \
139 (dir) == GTK_DIR_LEFT ? "left" : \
140 (dir) == GTK_DIR_RIGHT ? "right" : "invalid")
142 G_DEFINE_TYPE_WITH_CODE (CellAreaScaffold, cell_area_scaffold, GTK_TYPE_WIDGET,
143 G_IMPLEMENT_INTERFACE (GTK_TYPE_ORIENTABLE, NULL));
147 cell_area_scaffold_init (CellAreaScaffold *scaffold)
149 CellAreaScaffoldPrivate *priv;
151 scaffold->priv = G_TYPE_INSTANCE_GET_PRIVATE (scaffold,
152 TYPE_CELL_AREA_SCAFFOLD,
153 CellAreaScaffoldPrivate);
154 priv = scaffold->priv;
156 priv->area = gtk_cell_area_box_new ();
157 priv->iter = gtk_cell_area_create_iter (priv->area);
159 priv->row_data = g_array_new (FALSE, FALSE, sizeof (RowData));
161 gtk_widget_set_has_window (GTK_WIDGET (scaffold), FALSE);
162 gtk_widget_set_can_focus (GTK_WIDGET (scaffold), TRUE);
164 priv->size_changed_id =
165 g_signal_connect (priv->iter, "notify",
166 G_CALLBACK (size_changed_cb), scaffold);
170 cell_area_scaffold_class_init (CellAreaScaffoldClass *class)
172 GObjectClass *gobject_class;
173 GtkWidgetClass *widget_class;
175 gobject_class = G_OBJECT_CLASS(class);
176 gobject_class->dispose = cell_area_scaffold_dispose;
177 gobject_class->finalize = cell_area_scaffold_finalize;
178 gobject_class->get_property = cell_area_scaffold_get_property;
179 gobject_class->set_property = cell_area_scaffold_set_property;
181 widget_class = GTK_WIDGET_CLASS(class);
182 widget_class->realize = cell_area_scaffold_realize;
183 widget_class->unrealize = cell_area_scaffold_unrealize;
184 widget_class->draw = cell_area_scaffold_draw;
185 widget_class->size_allocate = cell_area_scaffold_size_allocate;
186 widget_class->get_preferred_width = cell_area_scaffold_get_preferred_width;
187 widget_class->get_preferred_height_for_width = cell_area_scaffold_get_preferred_height_for_width;
188 widget_class->get_preferred_height = cell_area_scaffold_get_preferred_height;
189 widget_class->get_preferred_width_for_height = cell_area_scaffold_get_preferred_width_for_height;
190 widget_class->focus = cell_area_scaffold_focus;
192 class->activate = cell_area_scaffold_activate;
194 g_object_class_override_property (gobject_class, PROP_ORIENTATION, "orientation");
196 scaffold_signals[ACTIVATE] =
197 g_signal_new ("activate",
198 G_OBJECT_CLASS_TYPE (gobject_class),
199 G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION,
200 G_STRUCT_OFFSET (CellAreaScaffoldClass, activate),
202 g_cclosure_marshal_VOID__VOID,
204 widget_class->activate_signal = scaffold_signals[ACTIVATE];
207 g_type_class_add_private (gobject_class, sizeof (CellAreaScaffoldPrivate));
210 /*********************************************************
212 *********************************************************/
214 cell_area_scaffold_finalize (GObject *object)
216 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
217 CellAreaScaffoldPrivate *priv;
219 priv = scaffold->priv;
221 g_array_free (priv->row_data, TRUE);
223 G_OBJECT_CLASS (cell_area_scaffold_parent_class)->finalize (object);
227 cell_area_scaffold_dispose (GObject *object)
229 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
230 CellAreaScaffoldPrivate *priv;
232 priv = scaffold->priv;
234 cell_area_scaffold_set_model (scaffold, NULL);
238 /* Disconnect signals */
239 g_signal_handler_disconnect (priv->iter, priv->size_changed_id);
241 g_object_unref (priv->iter);
243 priv->size_changed_id = 0;
248 /* Disconnect signals */
249 g_object_unref (priv->area);
253 G_OBJECT_CLASS (cell_area_scaffold_parent_class)->dispose (object);
257 cell_area_scaffold_set_property (GObject *object,
262 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
263 CellAreaScaffoldPrivate *priv;
265 priv = scaffold->priv;
269 case PROP_ORIENTATION:
270 gtk_orientable_set_orientation (GTK_ORIENTABLE (priv->area),
271 g_value_get_enum (value));
274 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
280 cell_area_scaffold_get_property (GObject *object,
285 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
286 CellAreaScaffoldPrivate *priv;
288 priv = scaffold->priv;
292 case PROP_ORIENTATION:
293 g_value_set_enum (value,
294 gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area)));
297 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
302 /*********************************************************
304 *********************************************************/
306 cell_area_scaffold_realize (GtkWidget *widget)
308 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
309 CellAreaScaffoldPrivate *priv = scaffold->priv;
310 GtkAllocation allocation;
312 GdkWindowAttr attributes;
313 gint attributes_mask;
315 gtk_widget_get_allocation (widget, &allocation);
317 gtk_widget_set_realized (widget, TRUE);
319 attributes.window_type = GDK_WINDOW_CHILD;
320 attributes.x = allocation.x;
321 attributes.y = allocation.y;
322 attributes.width = allocation.width;
323 attributes.height = allocation.height;
324 attributes.wclass = GDK_INPUT_ONLY;
325 attributes.event_mask = gtk_widget_get_events (widget);
326 attributes.event_mask |= (GDK_BUTTON_PRESS_MASK |
327 GDK_BUTTON_RELEASE_MASK |
329 GDK_KEY_RELEASE_MASK);
331 attributes_mask = GDK_WA_X | GDK_WA_Y;
333 window = gtk_widget_get_parent_window (widget);
334 gtk_widget_set_window (widget, window);
335 g_object_ref (window);
337 priv->event_window = gdk_window_new (window,
338 &attributes, attributes_mask);
339 gdk_window_set_user_data (priv->event_window, widget);
341 gtk_widget_style_attach (widget);
345 cell_area_scaffold_unrealize (GtkWidget *widget)
347 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
348 CellAreaScaffoldPrivate *priv = scaffold->priv;
350 if (priv->event_window)
352 gdk_window_set_user_data (priv->event_window, NULL);
353 gdk_window_destroy (priv->event_window);
354 priv->event_window = NULL;
357 GTK_WIDGET_CLASS (cell_area_scaffold_parent_class)->unrealize (widget);
361 cell_area_scaffold_draw (GtkWidget *widget,
364 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
365 CellAreaScaffoldPrivate *priv = scaffold->priv;
366 GtkOrientation orientation;
369 GdkRectangle render_area;
370 GtkAllocation allocation;
373 GtkCellRendererState flags;
378 have_focus = gtk_widget_has_focus (widget);
379 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
381 gtk_widget_get_allocation (widget, &allocation);
385 render_area.width = allocation.width;
386 render_area.height = allocation.height;
388 valid = gtk_tree_model_get_iter_first (priv->model, &iter);
391 RowData *data = &g_array_index (priv->row_data, RowData, i);
393 if (have_focus && i == priv->focus_row)
394 flags = GTK_CELL_RENDERER_FOCUSED;
398 if (orientation == GTK_ORIENTATION_HORIZONTAL)
400 render_area.height = data->size;
404 render_area.width = data->size;
407 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
408 gtk_cell_area_render (priv->area, priv->iter, widget, cr, &render_area, flags);
410 if (orientation == GTK_ORIENTATION_HORIZONTAL)
412 render_area.y += data->size;
413 render_area.y += ROW_SPACING;
417 render_area.x += data->size;
418 render_area.x += ROW_SPACING;
422 valid = gtk_tree_model_iter_next (priv->model, &iter);
429 request_all_base (CellAreaScaffold *scaffold)
431 CellAreaScaffoldPrivate *priv = scaffold->priv;
432 GtkWidget *widget = GTK_WIDGET (scaffold);
433 GtkOrientation orientation;
440 g_signal_handler_block (priv->iter, priv->size_changed_id);
442 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
444 valid = gtk_tree_model_get_iter_first (priv->model, &iter);
449 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
451 if (orientation == GTK_ORIENTATION_HORIZONTAL)
452 gtk_cell_area_get_preferred_width (priv->area, priv->iter, widget, &min, &nat);
454 gtk_cell_area_get_preferred_height (priv->area, priv->iter, widget, &min, &nat);
456 valid = gtk_tree_model_iter_next (priv->model, &iter);
459 if (orientation == GTK_ORIENTATION_HORIZONTAL)
460 gtk_cell_area_iter_sum_preferred_width (priv->iter);
462 gtk_cell_area_iter_sum_preferred_height (priv->iter);
464 g_signal_handler_unblock (priv->iter, priv->size_changed_id);
468 get_row_sizes (CellAreaScaffold *scaffold,
472 CellAreaScaffoldPrivate *priv = scaffold->priv;
473 GtkWidget *widget = GTK_WIDGET (scaffold);
474 GtkOrientation orientation;
482 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
484 valid = gtk_tree_model_get_iter_first (priv->model, &iter);
487 RowData *data = &g_array_index (array, RowData, i);
489 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
491 if (orientation == GTK_ORIENTATION_HORIZONTAL)
492 gtk_cell_area_get_preferred_height_for_width (priv->area, priv->iter, widget,
493 for_size, &data->size, NULL);
495 gtk_cell_area_get_preferred_width_for_height (priv->area, priv->iter, widget,
496 for_size, &data->size, NULL);
499 valid = gtk_tree_model_iter_next (priv->model, &iter);
504 cell_area_scaffold_size_allocate (GtkWidget *widget,
505 GtkAllocation *allocation)
507 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
508 CellAreaScaffoldPrivate *priv = scaffold->priv;
509 GtkOrientation orientation;
514 gtk_widget_set_allocation (widget, allocation);
516 if (gtk_widget_get_realized (widget))
517 gdk_window_move_resize (priv->event_window,
523 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
525 /* Cache the per-row sizes and allocate the iter */
526 if (orientation == GTK_ORIENTATION_HORIZONTAL)
528 get_row_sizes (scaffold, priv->row_data, allocation->width);
529 gtk_cell_area_iter_allocate_width (priv->iter, allocation->width);
533 get_row_sizes (scaffold, priv->row_data, allocation->height);
534 gtk_cell_area_iter_allocate_height (priv->iter, allocation->height);
540 cell_area_scaffold_get_preferred_width (GtkWidget *widget,
544 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
545 CellAreaScaffoldPrivate *priv = scaffold->priv;
546 GtkOrientation orientation;
551 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
553 if (orientation == GTK_ORIENTATION_HORIZONTAL)
555 request_all_base (scaffold);
557 gtk_cell_area_iter_get_preferred_width (priv->iter, minimum_size, natural_size);
561 gint min_size, nat_size;
563 GTK_WIDGET_GET_CLASS (widget)->get_preferred_height (widget, &min_size, &nat_size);
564 GTK_WIDGET_GET_CLASS (widget)->get_preferred_width_for_height (widget, min_size,
565 minimum_size, natural_size);
570 cell_area_scaffold_get_preferred_height_for_width (GtkWidget *widget,
575 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
576 CellAreaScaffoldPrivate *priv = scaffold->priv;
577 GtkOrientation orientation;
582 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
584 if (orientation == GTK_ORIENTATION_HORIZONTAL)
586 GArray *request_array;
587 gint n_rows, i, full_size = 0;
589 n_rows = gtk_tree_model_iter_n_children (priv->model, NULL);
591 /* Get an array for the contextual request */
592 request_array = g_array_new (FALSE, FALSE, sizeof (RowData));
593 g_array_set_size (request_array, n_rows);
594 memset (request_array->data, 0x0, n_rows * sizeof (RowData));
596 /* Gather each contextual size into the request array */
597 get_row_sizes (scaffold, request_array, for_size);
599 /* Sum up the size and add some row spacing */
600 for (i = 0; i < n_rows; i++)
602 RowData *data = &g_array_index (request_array, RowData, i);
604 full_size += data->size;
607 full_size += MAX (0, n_rows -1) * ROW_SPACING;
609 g_array_free (request_array, TRUE);
611 *minimum_size = full_size;
612 *natural_size = full_size;
616 GTK_WIDGET_GET_CLASS (widget)->get_preferred_height (widget, minimum_size, natural_size);
621 cell_area_scaffold_get_preferred_height (GtkWidget *widget,
625 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
626 CellAreaScaffoldPrivate *priv = scaffold->priv;
627 GtkOrientation orientation;
632 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
634 if (orientation == GTK_ORIENTATION_VERTICAL)
636 request_all_base (scaffold);
638 gtk_cell_area_iter_get_preferred_height (priv->iter, minimum_size, natural_size);
642 gint min_size, nat_size;
644 GTK_WIDGET_GET_CLASS (widget)->get_preferred_width (widget, &min_size, &nat_size);
645 GTK_WIDGET_GET_CLASS (widget)->get_preferred_height_for_width (widget, min_size,
646 minimum_size, natural_size);
651 cell_area_scaffold_get_preferred_width_for_height (GtkWidget *widget,
656 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
657 CellAreaScaffoldPrivate *priv = scaffold->priv;
658 GtkOrientation orientation;
663 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
665 if (orientation == GTK_ORIENTATION_VERTICAL)
667 GArray *request_array;
668 gint n_rows, i, full_size = 0;
670 n_rows = gtk_tree_model_iter_n_children (priv->model, NULL);
672 /* Get an array for the contextual request */
673 request_array = g_array_new (FALSE, FALSE, sizeof (RowData));
674 g_array_set_size (request_array, n_rows);
675 memset (request_array->data, 0x0, n_rows * sizeof (RowData));
677 /* Gather each contextual size into the request array */
678 get_row_sizes (scaffold, request_array, for_size);
680 /* Sum up the size and add some row spacing */
681 for (i = 0; i < n_rows; i++)
683 RowData *data = &g_array_index (request_array, RowData, i);
685 full_size += data->size;
688 full_size += MAX (0, n_rows -1) * ROW_SPACING;
690 g_array_free (request_array, TRUE);
692 *minimum_size = full_size;
693 *natural_size = full_size;
697 GTK_WIDGET_GET_CLASS (widget)->get_preferred_width (widget, minimum_size, natural_size);
702 cell_area_scaffold_focus (GtkWidget *widget,
703 GtkDirectionType direction)
705 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
706 CellAreaScaffoldPrivate *priv = scaffold->priv;
710 GtkOrientation orientation;
712 /* Grab focus on ourself if we dont already have focus */
713 if (!gtk_widget_has_focus (widget))
714 gtk_widget_grab_focus (widget);
716 /* Move focus from cell to cell and row to row */
717 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
719 focus_row = priv->focus_row;
721 valid = gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, priv->focus_row);
724 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
726 /* If focus stays in the area we dont need to do any more */
727 if (gtk_cell_area_focus (priv->area, direction))
729 GtkCellRenderer *renderer = gtk_cell_area_get_focus_cell (priv->area);
731 priv->focus_row = focus_row;
733 g_print ("focusing in direction %s: focus set on a %s in row %d\n",
734 DIRECTION_STR (direction), G_OBJECT_TYPE_NAME (renderer), priv->focus_row);
740 if (orientation == GTK_ORIENTATION_HORIZONTAL)
742 if (direction == GTK_DIR_RIGHT ||
743 direction == GTK_DIR_LEFT)
745 else if (direction == GTK_DIR_UP ||
746 direction == GTK_DIR_TAB_BACKWARD)
752 /* XXX A real implementation should check if the
753 * previous row can focus with it's attributes setup */
755 valid = gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, focus_row);
758 else /* direction == GTK_DIR_DOWN || GTK_DIR_TAB_FORWARD */
760 if (focus_row == priv->row_data->len - 1)
764 /* XXX A real implementation should check if the
765 * previous row can focus with it's attributes setup */
767 valid = gtk_tree_model_iter_next (priv->model, &iter);
771 else /* (orientation == GTK_ORIENTATION_HORIZONTAL) */
773 if (direction == GTK_DIR_UP ||
774 direction == GTK_DIR_DOWN)
776 else if (direction == GTK_DIR_LEFT ||
777 direction == GTK_DIR_TAB_BACKWARD)
783 /* XXX A real implementation should check if the
784 * previous row can focus with it's attributes setup */
786 valid = gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, focus_row);
789 else /* direction == GTK_DIR_RIGHT || GTK_DIR_TAB_FORWARD */
791 if (focus_row == priv->row_data->len - 1)
795 /* XXX A real implementation should check if the
796 * previous row can focus with it's attributes setup */
798 valid = gtk_tree_model_iter_next (priv->model, &iter);
805 g_print ("focus leaving with no cells in focus (direction %s, focus_row %d)\n",
806 DIRECTION_STR (direction), priv->focus_row);
811 /*********************************************************
812 * CellAreaScaffoldClass *
813 *********************************************************/
815 cell_area_scaffold_activate (CellAreaScaffold *scaffold)
817 CellAreaScaffoldPrivate *priv = scaffold->priv;
818 GtkWidget *widget = GTK_WIDGET (scaffold);
819 GtkAllocation allocation;
820 GtkOrientation orientation;
821 GdkRectangle cell_area;
826 orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
827 gtk_widget_get_allocation (widget, &allocation);
831 cell_area.width = allocation.width;
832 cell_area.height = allocation.height;
834 valid = gtk_tree_model_get_iter_first (priv->model, &iter);
837 RowData *data = &g_array_index (priv->row_data, RowData, i);
839 if (i == priv->focus_row)
841 if (orientation == GTK_ORIENTATION_HORIZONTAL)
842 cell_area.height = data->size;
844 cell_area.width = data->size;
846 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
847 gtk_cell_area_activate (priv->area, priv->iter, widget, &cell_area, GTK_CELL_RENDERER_FOCUSED);
852 if (orientation == GTK_ORIENTATION_HORIZONTAL)
853 cell_area.y += data->size + ROW_SPACING;
855 cell_area.x += data->size + ROW_SPACING;
858 valid = gtk_tree_model_iter_next (priv->model, &iter);
862 /*********************************************************
863 * CellArea/GtkTreeModel callbacks *
864 *********************************************************/
866 size_changed_cb (GtkCellAreaIter *iter,
868 CellAreaScaffold *scaffold)
870 if (!strcmp (pspec->name, "minimum-width") ||
871 !strcmp (pspec->name, "natural-width") ||
872 !strcmp (pspec->name, "minimum-height") ||
873 !strcmp (pspec->name, "natural-height"))
874 gtk_widget_queue_resize (GTK_WIDGET (scaffold));
878 rebuild_and_flush_internals (CellAreaScaffold *scaffold)
880 CellAreaScaffoldPrivate *priv = scaffold->priv;
885 n_rows = gtk_tree_model_iter_n_children (priv->model, NULL);
887 /* Clear/reset the array */
888 g_array_set_size (priv->row_data, n_rows);
889 memset (priv->row_data->data, 0x0, n_rows * sizeof (RowData));
892 g_array_set_size (priv->row_data, 0);
894 /* Data changed, lets flush the iter and consequently queue resize and
895 * start everything over again (note this is definitly far from optimized) */
896 gtk_cell_area_iter_flush (priv->iter);
900 row_changed_cb (GtkTreeModel *model,
903 CellAreaScaffold *scaffold)
905 rebuild_and_flush_internals (scaffold);
909 row_inserted_cb (GtkTreeModel *model,
912 CellAreaScaffold *scaffold)
914 rebuild_and_flush_internals (scaffold);
918 row_deleted_cb (GtkTreeModel *model,
920 CellAreaScaffold *scaffold)
922 rebuild_and_flush_internals (scaffold);
926 rows_reordered_cb (GtkTreeModel *model,
930 CellAreaScaffold *scaffold)
932 rebuild_and_flush_internals (scaffold);
935 /*********************************************************
937 *********************************************************/
939 cell_area_scaffold_new (void)
941 return (GtkWidget *)g_object_new (TYPE_CELL_AREA_SCAFFOLD, NULL);
945 cell_area_scaffold_get_area (CellAreaScaffold *scaffold)
947 CellAreaScaffoldPrivate *priv;
949 g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), NULL);
951 priv = scaffold->priv;
957 cell_area_scaffold_set_model (CellAreaScaffold *scaffold,
960 CellAreaScaffoldPrivate *priv;
962 g_return_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold));
964 priv = scaffold->priv;
966 if (priv->model != model)
970 g_signal_handler_disconnect (priv->model, priv->row_changed_id);
971 g_signal_handler_disconnect (priv->model, priv->row_inserted_id);
972 g_signal_handler_disconnect (priv->model, priv->row_deleted_id);
973 g_signal_handler_disconnect (priv->model, priv->rows_reordered_id);
975 g_object_unref (priv->model);
982 g_object_ref (priv->model);
984 priv->row_changed_id =
985 g_signal_connect (priv->model, "row-changed",
986 G_CALLBACK (row_changed_cb), scaffold);
988 priv->row_inserted_id =
989 g_signal_connect (priv->model, "row-inserted",
990 G_CALLBACK (row_inserted_cb), scaffold);
992 priv->row_deleted_id =
993 g_signal_connect (priv->model, "row-deleted",
994 G_CALLBACK (row_deleted_cb), scaffold);
996 priv->rows_reordered_id =
997 g_signal_connect (priv->model, "rows-reordered",
998 G_CALLBACK (rows_reordered_cb), scaffold);
1001 rebuild_and_flush_internals (scaffold);
1006 cell_area_scaffold_get_model (CellAreaScaffold *scaffold)
1008 CellAreaScaffoldPrivate *priv;
1010 g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), NULL);
1012 priv = scaffold->priv;