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);
32 static void cell_area_scaffold_realize (GtkWidget *widget);
33 static void cell_area_scaffold_unrealize (GtkWidget *widget);
34 static gboolean cell_area_scaffold_draw (GtkWidget *widget,
36 static void cell_area_scaffold_size_allocate (GtkWidget *widget,
37 GtkAllocation *allocation);
38 static void cell_area_scaffold_get_preferred_width (GtkWidget *widget,
41 static void cell_area_scaffold_get_preferred_height_for_width (GtkWidget *widget,
45 static void cell_area_scaffold_get_preferred_height (GtkWidget *widget,
48 static void cell_area_scaffold_get_preferred_width_for_height (GtkWidget *widget,
52 static void cell_area_scaffold_map (GtkWidget *widget);
53 static void cell_area_scaffold_unmap (GtkWidget *widget);
54 static gint cell_area_scaffold_focus (GtkWidget *widget,
55 GtkDirectionType direction);
56 static gboolean cell_area_scaffold_button_press (GtkWidget *widget,
57 GdkEventButton *event);
59 /* GtkContainerClass */
60 static void cell_area_scaffold_forall (GtkContainer *container,
61 gboolean include_internals,
63 gpointer callback_data);
64 static void cell_area_scaffold_remove (GtkContainer *container,
66 static void cell_area_scaffold_put_edit_widget (CellAreaScaffold *scaffold,
67 GtkWidget *edit_widget,
73 /* CellAreaScaffoldClass */
74 static void cell_area_scaffold_activate (CellAreaScaffold *scaffold);
76 /* CellArea/GtkTreeModel callbacks */
77 static void size_changed_cb (GtkCellAreaContext *context,
79 CellAreaScaffold *scaffold);
80 static void focus_changed_cb (GtkCellArea *area,
81 GtkCellRenderer *renderer,
83 CellAreaScaffold *scaffold);
84 static void add_editable_cb (GtkCellArea *area,
85 GtkCellRenderer *renderer,
86 GtkCellEditable *edit_widget,
87 GdkRectangle *cell_area,
89 CellAreaScaffold *scaffold);
90 static void remove_editable_cb (GtkCellArea *area,
91 GtkCellRenderer *renderer,
92 GtkCellEditable *edit_widget,
93 CellAreaScaffold *scaffold);
94 static void row_changed_cb (GtkTreeModel *model,
97 CellAreaScaffold *scaffold);
98 static void row_inserted_cb (GtkTreeModel *model,
101 CellAreaScaffold *scaffold);
102 static void row_deleted_cb (GtkTreeModel *model,
104 CellAreaScaffold *scaffold);
105 static void rows_reordered_cb (GtkTreeModel *model,
109 CellAreaScaffold *scaffold);
112 gint size; /* The height of rows in the scaffold's */
115 struct _CellAreaScaffoldPrivate {
117 /* Window for catching events and dispatching them to the cell area */
118 GdkWindow *event_window;
120 /* The model we're showing data for */
122 gulong row_changed_id;
123 gulong row_inserted_id;
124 gulong row_deleted_id;
125 gulong rows_reordered_id;
127 /* The area rendering the data and a global context */
129 GtkCellAreaContext *context;
131 /* Cache some info about rows (hieghts etc) */
136 gulong focus_changed_id;
138 /* Check when the underlying area changes the size and
139 * we need to queue a redraw */
140 gulong size_changed_id;
142 /* Currently edited widget */
143 GtkWidget *edit_widget;
144 GdkRectangle edit_rect;
145 gulong add_editable_id;
146 gulong remove_editable_id;
158 static guint scaffold_signals[N_SIGNALS] = { 0 };
160 #define DIRECTION_STR(dir) \
161 ((dir) == GTK_DIR_TAB_FORWARD ? "tab forward" : \
162 (dir) == GTK_DIR_TAB_BACKWARD ? "tab backward" : \
163 (dir) == GTK_DIR_UP ? "up" : \
164 (dir) == GTK_DIR_DOWN ? "down" : \
165 (dir) == GTK_DIR_LEFT ? "left" : \
166 (dir) == GTK_DIR_RIGHT ? "right" : "invalid")
168 G_DEFINE_TYPE_WITH_CODE (CellAreaScaffold, cell_area_scaffold, GTK_TYPE_CONTAINER,
169 G_IMPLEMENT_INTERFACE (GTK_TYPE_ORIENTABLE, NULL));
173 cell_area_scaffold_init (CellAreaScaffold *scaffold)
175 CellAreaScaffoldPrivate *priv;
177 scaffold->priv = G_TYPE_INSTANCE_GET_PRIVATE (scaffold,
178 TYPE_CELL_AREA_SCAFFOLD,
179 CellAreaScaffoldPrivate);
180 priv = scaffold->priv;
182 priv->area = gtk_cell_area_box_new ();
183 priv->context = gtk_cell_area_create_context (priv->area);
185 priv->row_data = g_array_new (FALSE, FALSE, sizeof (RowData));
187 gtk_widget_set_has_window (GTK_WIDGET (scaffold), FALSE);
188 gtk_widget_set_can_focus (GTK_WIDGET (scaffold), TRUE);
190 priv->size_changed_id =
191 g_signal_connect (priv->context, "notify",
192 G_CALLBACK (size_changed_cb), scaffold);
194 priv->focus_changed_id =
195 g_signal_connect (priv->area, "focus-changed",
196 G_CALLBACK (focus_changed_cb), scaffold);
198 priv->add_editable_id =
199 g_signal_connect (priv->area, "add-editable",
200 G_CALLBACK (add_editable_cb), scaffold);
202 priv->remove_editable_id =
203 g_signal_connect (priv->area, "remove-editable",
204 G_CALLBACK (remove_editable_cb), scaffold);
208 cell_area_scaffold_class_init (CellAreaScaffoldClass *class)
210 GObjectClass *gobject_class;
211 GtkWidgetClass *widget_class;
212 GtkContainerClass *container_class;
214 gobject_class = G_OBJECT_CLASS (class);
215 gobject_class->dispose = cell_area_scaffold_dispose;
216 gobject_class->finalize = cell_area_scaffold_finalize;
218 widget_class = GTK_WIDGET_CLASS (class);
219 widget_class->realize = cell_area_scaffold_realize;
220 widget_class->unrealize = cell_area_scaffold_unrealize;
221 widget_class->draw = cell_area_scaffold_draw;
222 widget_class->size_allocate = cell_area_scaffold_size_allocate;
223 widget_class->get_preferred_width = cell_area_scaffold_get_preferred_width;
224 widget_class->get_preferred_height_for_width = cell_area_scaffold_get_preferred_height_for_width;
225 widget_class->get_preferred_height = cell_area_scaffold_get_preferred_height;
226 widget_class->get_preferred_width_for_height = cell_area_scaffold_get_preferred_width_for_height;
227 widget_class->map = cell_area_scaffold_map;
228 widget_class->unmap = cell_area_scaffold_unmap;
229 widget_class->focus = cell_area_scaffold_focus;
230 widget_class->button_press_event = cell_area_scaffold_button_press;
232 container_class = GTK_CONTAINER_CLASS (class);
233 container_class->forall = cell_area_scaffold_forall;
234 container_class->remove = cell_area_scaffold_remove;
236 class->activate = cell_area_scaffold_activate;
238 scaffold_signals[ACTIVATE] =
239 g_signal_new ("activate",
240 G_OBJECT_CLASS_TYPE (gobject_class),
241 G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION,
242 G_STRUCT_OFFSET (CellAreaScaffoldClass, activate),
244 g_cclosure_marshal_VOID__VOID,
246 widget_class->activate_signal = scaffold_signals[ACTIVATE];
249 g_type_class_add_private (gobject_class, sizeof (CellAreaScaffoldPrivate));
252 /*********************************************************
254 *********************************************************/
256 cell_area_scaffold_finalize (GObject *object)
258 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
259 CellAreaScaffoldPrivate *priv;
261 priv = scaffold->priv;
263 g_array_free (priv->row_data, TRUE);
265 G_OBJECT_CLASS (cell_area_scaffold_parent_class)->finalize (object);
269 cell_area_scaffold_dispose (GObject *object)
271 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
272 CellAreaScaffoldPrivate *priv;
274 priv = scaffold->priv;
276 cell_area_scaffold_set_model (scaffold, NULL);
280 /* Disconnect signals */
281 g_signal_handler_disconnect (priv->context, priv->size_changed_id);
283 g_object_unref (priv->context);
284 priv->context = NULL;
285 priv->size_changed_id = 0;
290 /* Disconnect signals */
291 g_signal_handler_disconnect (priv->area, priv->focus_changed_id);
292 g_signal_handler_disconnect (priv->area, priv->add_editable_id);
293 g_signal_handler_disconnect (priv->area, priv->remove_editable_id);
295 g_object_unref (priv->area);
297 priv->focus_changed_id = 0;
300 G_OBJECT_CLASS (cell_area_scaffold_parent_class)->dispose (object);
303 /*********************************************************
305 *********************************************************/
307 cell_area_scaffold_realize (GtkWidget *widget)
309 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
310 CellAreaScaffoldPrivate *priv = scaffold->priv;
311 GtkAllocation allocation;
313 GdkWindowAttr attributes;
314 gint attributes_mask;
316 gtk_widget_get_allocation (widget, &allocation);
318 gtk_widget_set_realized (widget, TRUE);
320 attributes.window_type = GDK_WINDOW_CHILD;
321 attributes.x = allocation.x;
322 attributes.y = allocation.y;
323 attributes.width = allocation.width;
324 attributes.height = allocation.height;
325 attributes.wclass = GDK_INPUT_ONLY;
326 attributes.event_mask = gtk_widget_get_events (widget);
327 attributes.event_mask |= (GDK_BUTTON_PRESS_MASK |
328 GDK_BUTTON_RELEASE_MASK |
330 GDK_KEY_RELEASE_MASK);
332 attributes_mask = GDK_WA_X | GDK_WA_Y;
334 window = gtk_widget_get_parent_window (widget);
335 gtk_widget_set_window (widget, window);
336 g_object_ref (window);
338 priv->event_window = gdk_window_new (window, &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;
368 GdkRectangle background_area;
369 GdkRectangle render_area;
370 GtkAllocation allocation;
373 GtkCellRendererState flags;
378 have_focus = gtk_widget_has_focus (widget);
380 gtk_widget_get_allocation (widget, &allocation);
384 render_area.width = allocation.width;
385 render_area.height = allocation.height;
387 background_area = render_area;
389 render_area.x = priv->indent;
390 render_area.width -= priv->indent;
392 valid = gtk_tree_model_get_iter_first (priv->model, &iter);
395 RowData *data = &g_array_index (priv->row_data, RowData, i);
397 if (have_focus && i == priv->focus_row)
398 flags = GTK_CELL_RENDERER_FOCUSED;
402 render_area.height = data->size;
404 background_area.height = render_area.height;
405 background_area.y = render_area.y;
409 background_area.height += priv->row_spacing / 2;
410 background_area.height += priv->row_spacing % 2;
412 else if (i == priv->row_data->len - 1)
414 background_area.y -= priv->row_spacing / 2;
415 background_area.height += priv->row_spacing / 2;
419 background_area.y -= priv->row_spacing / 2;
420 background_area.height += priv->row_spacing;
423 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
424 gtk_cell_area_render (priv->area, priv->context, widget, cr,
425 &background_area, &render_area, flags,
426 (have_focus && i == priv->focus_row));
428 render_area.y += data->size;
429 render_area.y += priv->row_spacing;
432 valid = gtk_tree_model_iter_next (priv->model, &iter);
435 /* Draw the edit widget after drawing everything else */
436 GTK_WIDGET_CLASS (cell_area_scaffold_parent_class)->draw (widget, cr);
442 request_all_base (CellAreaScaffold *scaffold)
444 CellAreaScaffoldPrivate *priv = scaffold->priv;
445 GtkWidget *widget = GTK_WIDGET (scaffold);
452 g_signal_handler_block (priv->context, priv->size_changed_id);
454 valid = gtk_tree_model_get_iter_first (priv->model, &iter);
459 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
460 gtk_cell_area_get_preferred_width (priv->area, priv->context, widget, &min, &nat);
462 valid = gtk_tree_model_iter_next (priv->model, &iter);
465 g_signal_handler_unblock (priv->context, priv->size_changed_id);
469 get_row_sizes (CellAreaScaffold *scaffold,
473 CellAreaScaffoldPrivate *priv = scaffold->priv;
474 GtkWidget *widget = GTK_WIDGET (scaffold);
482 valid = gtk_tree_model_get_iter_first (priv->model, &iter);
485 RowData *data = &g_array_index (array, RowData, i);
487 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
488 gtk_cell_area_get_preferred_height_for_width (priv->area, priv->context, widget,
489 for_size, &data->size, NULL);
492 valid = gtk_tree_model_iter_next (priv->model, &iter);
497 cell_area_scaffold_size_allocate (GtkWidget *widget,
498 GtkAllocation *allocation)
500 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
501 CellAreaScaffoldPrivate *priv = scaffold->priv;
503 gtk_widget_set_allocation (widget, allocation);
505 if (gtk_widget_get_realized (widget))
506 gdk_window_move_resize (priv->event_window,
512 /* Allocate the child GtkCellEditable widget if one is currently editing a row */
513 if (priv->edit_widget)
514 gtk_widget_size_allocate (priv->edit_widget, &priv->edit_rect);
519 /* Cache the per-row sizes and allocate the context */
520 gtk_cell_area_context_allocate (priv->context, allocation->width - priv->indent, -1);
521 get_row_sizes (scaffold, priv->row_data, allocation->width - priv->indent);
526 cell_area_scaffold_get_preferred_width (GtkWidget *widget,
530 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
531 CellAreaScaffoldPrivate *priv = scaffold->priv;
536 request_all_base (scaffold);
537 gtk_cell_area_context_get_preferred_width (priv->context, minimum_size, natural_size);
539 *minimum_size += priv->indent;
540 *natural_size += priv->indent;
544 cell_area_scaffold_get_preferred_height_for_width (GtkWidget *widget,
549 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
550 CellAreaScaffoldPrivate *priv = scaffold->priv;
551 GArray *request_array;
552 gint n_rows, i, full_size = 0;
557 n_rows = gtk_tree_model_iter_n_children (priv->model, NULL);
559 /* Get an array for the contextual request */
560 request_array = g_array_new (FALSE, FALSE, sizeof (RowData));
561 g_array_set_size (request_array, n_rows);
562 memset (request_array->data, 0x0, n_rows * sizeof (RowData));
564 /* Gather each contextual size into the request array */
565 get_row_sizes (scaffold, request_array, for_size - priv->indent);
567 /* Sum up the size and add some row spacing */
568 for (i = 0; i < n_rows; i++)
570 RowData *data = &g_array_index (request_array, RowData, i);
572 full_size += data->size;
575 full_size += MAX (0, n_rows -1) * priv->row_spacing;
577 g_array_free (request_array, TRUE);
579 *minimum_size = full_size;
580 *natural_size = full_size;
584 cell_area_scaffold_get_preferred_height (GtkWidget *widget,
588 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
589 CellAreaScaffoldPrivate *priv = scaffold->priv;
590 gint min_size, nat_size;
595 GTK_WIDGET_GET_CLASS (widget)->get_preferred_width (widget, &min_size, &nat_size);
596 GTK_WIDGET_GET_CLASS (widget)->get_preferred_height_for_width (widget, min_size,
597 minimum_size, natural_size);
601 cell_area_scaffold_get_preferred_width_for_height (GtkWidget *widget,
606 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
607 CellAreaScaffoldPrivate *priv = scaffold->priv;
612 GTK_WIDGET_GET_CLASS (widget)->get_preferred_width (widget, minimum_size, natural_size);
616 cell_area_scaffold_map (GtkWidget *widget)
618 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
619 CellAreaScaffoldPrivate *priv = scaffold->priv;
621 GTK_WIDGET_CLASS (cell_area_scaffold_parent_class)->map (widget);
623 if (priv->event_window)
624 gdk_window_show (priv->event_window);
628 cell_area_scaffold_unmap (GtkWidget *widget)
630 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
631 CellAreaScaffoldPrivate *priv = scaffold->priv;
633 GTK_WIDGET_CLASS (cell_area_scaffold_parent_class)->unmap (widget);
635 if (priv->event_window)
636 gdk_window_hide (priv->event_window);
641 cell_area_scaffold_focus (GtkWidget *widget,
642 GtkDirectionType direction)
644 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
645 CellAreaScaffoldPrivate *priv = scaffold->priv;
649 gboolean changed = FALSE;
651 /* Grab focus on ourself if we dont already have focus */
652 if (!gtk_widget_has_focus (widget))
653 gtk_widget_grab_focus (widget);
655 /* Move focus from cell to cell and row to row */
656 focus_row = priv->focus_row;
658 g_signal_handler_block (priv->area, priv->focus_changed_id);
660 valid = gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, priv->focus_row);
663 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
665 /* If focus stays in the area we dont need to do any more */
666 if (gtk_cell_area_focus (priv->area, direction))
668 priv->focus_row = focus_row;
670 /* XXX A smarter implementation would only invalidate the rectangles where
671 * focus was removed from and new focus was placed */
672 gtk_widget_queue_draw (widget);
678 if (direction == GTK_DIR_RIGHT ||
679 direction == GTK_DIR_LEFT)
681 else if (direction == GTK_DIR_UP ||
682 direction == GTK_DIR_TAB_BACKWARD)
688 /* XXX A real implementation should check if the
689 * previous row can focus with it's attributes setup */
691 valid = gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, focus_row);
694 else /* direction == GTK_DIR_DOWN || GTK_DIR_TAB_FORWARD */
696 if (focus_row == priv->row_data->len - 1)
700 /* XXX A real implementation should check if the
701 * previous row can focus with it's attributes setup */
703 valid = gtk_tree_model_iter_next (priv->model, &iter);
709 g_signal_handler_unblock (priv->area, priv->focus_changed_id);
711 /* XXX A smarter implementation would only invalidate the rectangles where
712 * focus was removed from and new focus was placed */
713 gtk_widget_queue_draw (widget);
719 cell_area_scaffold_button_press (GtkWidget *widget,
720 GdkEventButton *event)
722 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
723 CellAreaScaffoldPrivate *priv = scaffold->priv;
727 GdkRectangle event_area;
728 GtkAllocation allocation;
729 gboolean handled = FALSE;
731 gtk_widget_get_allocation (widget, &allocation);
735 event_area.width = allocation.width;
736 event_area.height = allocation.height;
738 event_area.x = priv->indent;
739 event_area.width -= priv->indent;
741 valid = gtk_tree_model_get_iter_first (priv->model, &iter);
744 RowData *data = &g_array_index (priv->row_data, RowData, i);
746 event_area.height = data->size;
748 if (event->y >= event_area.y &&
749 event->y <= event_area.y + event_area.height)
751 /* XXX A real implementation would assemble GtkCellRendererState flags here */
752 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
753 handled = gtk_cell_area_event (priv->area, priv->context, GTK_WIDGET (scaffold),
754 (GdkEvent *)event, &event_area, 0);
758 event_area.y += data->size;
759 event_area.y += priv->row_spacing;
762 valid = gtk_tree_model_iter_next (priv->model, &iter);
769 /*********************************************************
770 * GtkContainerClass *
771 *********************************************************/
773 cell_area_scaffold_put_edit_widget (CellAreaScaffold *scaffold,
774 GtkWidget *edit_widget,
780 CellAreaScaffoldPrivate *priv = scaffold->priv;
782 priv->edit_rect.x = x;
783 priv->edit_rect.y = y;
784 priv->edit_rect.width = width;
785 priv->edit_rect.height = height;
786 priv->edit_widget = edit_widget;
788 gtk_widget_set_parent (edit_widget, GTK_WIDGET (scaffold));
792 cell_area_scaffold_forall (GtkContainer *container,
793 gboolean include_internals,
794 GtkCallback callback,
795 gpointer callback_data)
797 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (container);
798 CellAreaScaffoldPrivate *priv = scaffold->priv;
800 if (priv->edit_widget)
801 (* callback) (priv->edit_widget, callback_data);
805 cell_area_scaffold_remove (GtkContainer *container,
808 CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (container);
809 CellAreaScaffoldPrivate *priv = scaffold->priv;
811 g_return_if_fail (child == priv->edit_widget);
813 gtk_widget_unparent (priv->edit_widget);
814 priv->edit_widget = NULL;
817 /*********************************************************
818 * CellAreaScaffoldClass *
819 *********************************************************/
821 cell_area_scaffold_activate (CellAreaScaffold *scaffold)
823 CellAreaScaffoldPrivate *priv = scaffold->priv;
824 GtkWidget *widget = GTK_WIDGET (scaffold);
825 GtkAllocation allocation;
826 GdkRectangle cell_area;
831 gtk_widget_get_allocation (widget, &allocation);
835 cell_area.width = allocation.width;
836 cell_area.height = allocation.height;
838 cell_area.x = priv->indent;
839 cell_area.width -= priv->indent;
841 valid = gtk_tree_model_get_iter_first (priv->model, &iter);
844 RowData *data = &g_array_index (priv->row_data, RowData, i);
846 if (i == priv->focus_row)
848 cell_area.height = data->size;
850 gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
851 gtk_cell_area_activate (priv->area, priv->context, widget, &cell_area,
852 GTK_CELL_RENDERER_FOCUSED, FALSE);
857 cell_area.y += data->size + priv->row_spacing;
860 valid = gtk_tree_model_iter_next (priv->model, &iter);
864 /*********************************************************
865 * CellArea/GtkTreeModel callbacks *
866 *********************************************************/
868 size_changed_cb (GtkCellAreaContext *context,
870 CellAreaScaffold *scaffold)
872 if (!strcmp (pspec->name, "minimum-width") ||
873 !strcmp (pspec->name, "natural-width") ||
874 !strcmp (pspec->name, "minimum-height") ||
875 !strcmp (pspec->name, "natural-height"))
876 gtk_widget_queue_resize (GTK_WIDGET (scaffold));
880 focus_changed_cb (GtkCellArea *area,
881 GtkCellRenderer *renderer,
883 CellAreaScaffold *scaffold)
885 CellAreaScaffoldPrivate *priv = scaffold->priv;
886 GtkWidget *widget = GTK_WIDGET (scaffold);
887 GtkTreePath *treepath;
893 /* We can be signaled that a renderer lost focus, here
898 treepath = gtk_tree_path_new_from_string (path);
899 indices = gtk_tree_path_get_indices (treepath);
901 priv->focus_row = indices[0];
903 gtk_tree_path_free (treepath);
905 /* Make sure we have focus now */
906 if (!gtk_widget_has_focus (widget))
907 gtk_widget_grab_focus (widget);
909 gtk_widget_queue_draw (widget);
913 add_editable_cb (GtkCellArea *area,
914 GtkCellRenderer *renderer,
915 GtkCellEditable *edit_widget,
916 GdkRectangle *cell_area,
918 CellAreaScaffold *scaffold)
920 GtkAllocation allocation;
922 gtk_widget_get_allocation (GTK_WIDGET (scaffold), &allocation);
924 cell_area_scaffold_put_edit_widget (scaffold, GTK_WIDGET (edit_widget),
925 allocation.x + cell_area->x,
926 allocation.y + cell_area->y,
927 cell_area->width, cell_area->height);
931 remove_editable_cb (GtkCellArea *area,
932 GtkCellRenderer *renderer,
933 GtkCellEditable *edit_widget,
934 CellAreaScaffold *scaffold)
936 gtk_container_remove (GTK_CONTAINER (scaffold), GTK_WIDGET (edit_widget));
938 gtk_widget_grab_focus (GTK_WIDGET (scaffold));
942 rebuild_and_reset_internals (CellAreaScaffold *scaffold)
944 CellAreaScaffoldPrivate *priv = scaffold->priv;
949 n_rows = gtk_tree_model_iter_n_children (priv->model, NULL);
951 /* Clear/reset the array */
952 g_array_set_size (priv->row_data, n_rows);
953 memset (priv->row_data->data, 0x0, n_rows * sizeof (RowData));
956 g_array_set_size (priv->row_data, 0);
958 /* Data changed, lets reset the context and consequently queue resize and
959 * start everything over again (note this is definitly far from optimized) */
960 gtk_cell_area_context_reset (priv->context);
964 row_changed_cb (GtkTreeModel *model,
967 CellAreaScaffold *scaffold)
969 rebuild_and_reset_internals (scaffold);
973 row_inserted_cb (GtkTreeModel *model,
976 CellAreaScaffold *scaffold)
978 rebuild_and_reset_internals (scaffold);
982 row_deleted_cb (GtkTreeModel *model,
984 CellAreaScaffold *scaffold)
986 rebuild_and_reset_internals (scaffold);
990 rows_reordered_cb (GtkTreeModel *model,
994 CellAreaScaffold *scaffold)
996 rebuild_and_reset_internals (scaffold);
999 /*********************************************************
1001 *********************************************************/
1003 cell_area_scaffold_new (void)
1005 return (GtkWidget *)g_object_new (TYPE_CELL_AREA_SCAFFOLD, NULL);
1009 cell_area_scaffold_get_area (CellAreaScaffold *scaffold)
1011 CellAreaScaffoldPrivate *priv;
1013 g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), NULL);
1015 priv = scaffold->priv;
1021 cell_area_scaffold_set_model (CellAreaScaffold *scaffold,
1022 GtkTreeModel *model)
1024 CellAreaScaffoldPrivate *priv;
1026 g_return_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold));
1028 priv = scaffold->priv;
1030 if (priv->model != model)
1034 g_signal_handler_disconnect (priv->model, priv->row_changed_id);
1035 g_signal_handler_disconnect (priv->model, priv->row_inserted_id);
1036 g_signal_handler_disconnect (priv->model, priv->row_deleted_id);
1037 g_signal_handler_disconnect (priv->model, priv->rows_reordered_id);
1039 g_object_unref (priv->model);
1042 priv->model = model;
1046 g_object_ref (priv->model);
1048 priv->row_changed_id =
1049 g_signal_connect (priv->model, "row-changed",
1050 G_CALLBACK (row_changed_cb), scaffold);
1052 priv->row_inserted_id =
1053 g_signal_connect (priv->model, "row-inserted",
1054 G_CALLBACK (row_inserted_cb), scaffold);
1056 priv->row_deleted_id =
1057 g_signal_connect (priv->model, "row-deleted",
1058 G_CALLBACK (row_deleted_cb), scaffold);
1060 priv->rows_reordered_id =
1061 g_signal_connect (priv->model, "rows-reordered",
1062 G_CALLBACK (rows_reordered_cb), scaffold);
1065 rebuild_and_reset_internals (scaffold);
1070 cell_area_scaffold_get_model (CellAreaScaffold *scaffold)
1072 CellAreaScaffoldPrivate *priv;
1074 g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), NULL);
1076 priv = scaffold->priv;
1083 cell_area_scaffold_set_row_spacing (CellAreaScaffold *scaffold,
1086 CellAreaScaffoldPrivate *priv;
1088 g_return_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold));
1090 priv = scaffold->priv;
1092 if (priv->row_spacing != spacing)
1094 priv->row_spacing = spacing;
1095 gtk_widget_queue_resize (GTK_WIDGET (scaffold));
1100 cell_area_scaffold_get_row_spacing (CellAreaScaffold *scaffold)
1102 CellAreaScaffoldPrivate *priv;
1104 g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), 0);
1106 priv = scaffold->priv;
1108 return priv->row_spacing;
1112 cell_area_scaffold_set_indentation (CellAreaScaffold *scaffold,
1115 CellAreaScaffoldPrivate *priv;
1117 g_return_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold));
1119 priv = scaffold->priv;
1121 if (priv->indent != indent)
1123 priv->indent = indent;
1124 gtk_widget_queue_resize (GTK_WIDGET (scaffold));
1129 cell_area_scaffold_get_indentation (CellAreaScaffold *scaffold)
1131 CellAreaScaffoldPrivate *priv;
1133 g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), 0);
1135 priv = scaffold->priv;
1137 return priv->indent;