]> Pileus Git - ~andy/gtk/blob - tests/testtreeview.c
We no longer need to create the signals here ... they come from the
[~andy/gtk] / tests / testtreeview.c
1 #include <string.h>
2 #include "prop-editor.h"
3 #include <gtk/gtk.h>
4
5
6 /* Don't copy this bad example; inline RGB data is always a better
7  * idea than inline XPMs.
8  */
9 static char  *book_closed_xpm[] = {
10 "16 16 6 1",
11 "       c None s None",
12 ".      c black",
13 "X      c red",
14 "o      c yellow",
15 "O      c #808080",
16 "#      c white",
17 "                ",
18 "       ..       ",
19 "     ..XX.      ",
20 "   ..XXXXX.     ",
21 " ..XXXXXXXX.    ",
22 ".ooXXXXXXXXX.   ",
23 "..ooXXXXXXXXX.  ",
24 ".X.ooXXXXXXXXX. ",
25 ".XX.ooXXXXXX..  ",
26 " .XX.ooXXX..#O  ",
27 "  .XX.oo..##OO. ",
28 "   .XX..##OO..  ",
29 "    .X.#OO..    ",
30 "     ..O..      ",
31 "      ..        ",
32 "                "
33 };
34
35 static void run_automated_tests (void);
36
37 /* This custom model is to test custom model use. */
38
39 #define GTK_TYPE_MODEL_TYPES                    (gtk_tree_model_types_get_type ())
40 #define GTK_TREE_MODEL_TYPES(obj)               (GTK_CHECK_CAST ((obj), GTK_TYPE_MODEL_TYPES, GtkTreeModelTypes))
41 #define GTK_TREE_MODEL_TYPES_CLASS(klass)       (GTK_CHECK_CLASS_CAST ((klass), GTK_TYPE_MODEL_TYPES, GtkTreeModelTypesClass))
42 #define GTK_IS_TREE_MODEL_TYPES(obj)            (GTK_CHECK_TYPE ((obj), GTK_TYPE_MODEL_TYPES))
43 #define GTK_IS_TREE_MODEL_TYPES_CLASS(klass)    (GTK_CHECK_CLASS_TYPE ((obj), GTK_TYPE_MODEL_TYPES))
44
45 typedef struct _GtkTreeModelTypes       GtkTreeModelTypes;
46 typedef struct _GtkTreeModelTypesClass  GtkTreeModelTypesClass;
47
48 struct _GtkTreeModelTypes
49 {
50   GtkObject parent;
51
52   gint stamp;
53 };
54
55 struct _GtkTreeModelTypesClass
56 {
57   GtkObjectClass parent_class;
58
59   guint        (* get_flags)       (GtkTreeModel *tree_model);   
60   gint         (* get_n_columns)   (GtkTreeModel *tree_model);
61   GType        (* get_column_type) (GtkTreeModel *tree_model,
62                                     gint          index);
63   gboolean     (* get_iter)        (GtkTreeModel *tree_model,
64                                     GtkTreeIter  *iter,
65                                     GtkTreePath  *path);
66   GtkTreePath *(* get_path)        (GtkTreeModel *tree_model,
67                                     GtkTreeIter  *iter);
68   void         (* get_value)       (GtkTreeModel *tree_model,
69                                     GtkTreeIter  *iter,
70                                     gint          column,
71                                     GValue       *value);
72   gboolean     (* iter_next)       (GtkTreeModel *tree_model,
73                                     GtkTreeIter  *iter);
74   gboolean     (* iter_children)   (GtkTreeModel *tree_model,
75                                     GtkTreeIter  *iter,
76                                     GtkTreeIter  *parent);
77   gboolean     (* iter_has_child)  (GtkTreeModel *tree_model,
78                                     GtkTreeIter  *iter);
79   gint         (* iter_n_children) (GtkTreeModel *tree_model,
80                                     GtkTreeIter  *iter);
81   gboolean     (* iter_nth_child)  (GtkTreeModel *tree_model,
82                                     GtkTreeIter  *iter,
83                                     GtkTreeIter  *parent,
84                                     gint          n);
85   gboolean     (* iter_parent)     (GtkTreeModel *tree_model,
86                                     GtkTreeIter  *iter,
87                                     GtkTreeIter  *child);
88   void         (* ref_iter)        (GtkTreeModel *tree_model,
89                                     GtkTreeIter  *iter);
90   void         (* unref_iter)      (GtkTreeModel *tree_model,
91                                     GtkTreeIter  *iter);
92
93   /* These will be moved into the GtkTreeModelIface eventually */
94   void         (* changed)         (GtkTreeModel *tree_model,
95                                     GtkTreePath  *path,
96                                     GtkTreeIter  *iter);
97   void         (* inserted)        (GtkTreeModel *tree_model,
98                                     GtkTreePath  *path,
99                                     GtkTreeIter  *iter);
100   void         (* child_toggled)   (GtkTreeModel *tree_model,
101                                     GtkTreePath  *path,
102                                     GtkTreeIter  *iter);
103   void         (* deleted)         (GtkTreeModel *tree_model,
104                                     GtkTreePath  *path);
105 };
106
107 GtkType             gtk_tree_model_types_get_type      (void);
108 GtkTreeModelTypes *gtk_tree_model_types_new           (void);
109
110 typedef enum
111 {
112   COLUMNS_NONE,
113   COLUMNS_ONE,
114   COLUMNS_LOTS,
115   COLUMNS_LAST
116 } ColumnsType;
117
118 static gchar *column_type_names[] = {
119   "No columns",
120   "One column",
121   "Many columns"
122 };
123
124 #define N_COLUMNS 9
125
126 static GType*
127 get_model_types (void)
128 {
129   static GType column_types[N_COLUMNS] = { 0 };
130   
131   if (column_types[0] == 0)
132     {
133       column_types[0] = G_TYPE_STRING;
134       column_types[1] = G_TYPE_STRING;
135       column_types[2] = GDK_TYPE_PIXBUF;
136       column_types[3] = G_TYPE_FLOAT;
137       column_types[4] = G_TYPE_UINT;
138       column_types[5] = G_TYPE_UCHAR;
139       column_types[6] = G_TYPE_CHAR;
140 #define BOOL_COLUMN 7
141       column_types[BOOL_COLUMN] = G_TYPE_BOOLEAN;
142       column_types[8] = G_TYPE_INT;
143     }
144
145   return column_types;
146 }
147
148 static void
149 col_clicked_cb (GtkTreeViewColumn *col, gpointer data)
150 {
151   GtkWindow *win;
152
153   win = GTK_WINDOW (create_prop_editor (G_OBJECT (col), GTK_TYPE_TREE_VIEW_COLUMN));
154
155   gtk_window_set_title (win, gtk_tree_view_column_get_title (col));
156 }
157
158 static void
159 setup_column (GtkTreeViewColumn *col)
160 {
161   gtk_tree_view_column_set_clickable (col, TRUE);
162   g_signal_connect (G_OBJECT (col),
163                     "clicked",
164                     (GCallback) col_clicked_cb,
165                     NULL);
166 }
167
168 static void
169 toggled_callback (GtkCellRendererToggle *celltoggle,
170                   gchar                 *path_string,
171                   GtkTreeView           *tree_view)
172 {
173   GtkTreeModel *model = NULL;
174   GtkTreeModelSort *sort_model = NULL;
175   GtkTreePath *path;
176   GtkTreeIter iter;
177   gboolean active = FALSE;
178   
179   g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
180
181   model = gtk_tree_view_get_model (tree_view);
182   
183   if (GTK_IS_TREE_MODEL_SORT (model))
184     {
185       sort_model = GTK_TREE_MODEL_SORT (model);
186       model = gtk_tree_model_sort_get_model (sort_model);
187     }
188
189   if (model == NULL)
190     return;
191
192   if (sort_model)
193     {
194       g_warning ("FIXME implement conversion from TreeModelSort iter to child model iter");
195       return;
196     }
197       
198   path = gtk_tree_path_new_from_string (path_string);
199   if (!gtk_tree_model_get_iter (model,
200                                 &iter, path))
201     {
202       g_warning ("%s: bad path?", G_STRLOC);
203       return;
204     }
205   gtk_tree_path_free (path);
206   
207   if (GTK_IS_LIST_STORE (model))
208     {
209       gtk_tree_model_get (GTK_TREE_MODEL (model),
210                           &iter,
211                           BOOL_COLUMN,
212                           &active,
213                           -1);
214       
215       gtk_list_store_set (GTK_LIST_STORE (model),
216                           &iter,
217                           BOOL_COLUMN,
218                           !active,
219                           -1);
220     }
221   else if (GTK_IS_TREE_STORE (model))
222     {
223       gtk_tree_model_get (GTK_TREE_MODEL (model),
224                           &iter,
225                           BOOL_COLUMN,
226                           &active,
227                           -1);
228             
229       gtk_tree_store_set (GTK_TREE_STORE (model),
230                           &iter,
231                           BOOL_COLUMN,
232                           !active,
233                           -1);
234     }
235   else
236     g_warning ("don't know how to actually toggle value for model type %s",
237                g_type_name (G_TYPE_FROM_INSTANCE (model)));
238 }
239
240
241 static ColumnsType current_column_type = COLUMNS_LOTS;
242
243 static void
244 set_columns_type (GtkTreeView *tree_view, ColumnsType type)
245 {
246   GtkTreeViewColumn *col;
247   GtkCellRenderer *rend;
248   GdkPixbuf *pixbuf;
249   GtkWidget *image;
250   
251   current_column_type = type;
252   
253   col = gtk_tree_view_get_column (tree_view, 0);
254   while (col)
255     {
256       gtk_tree_view_remove_column (tree_view, col);
257
258       col = gtk_tree_view_get_column (tree_view, 0);
259     }
260
261   gtk_tree_view_set_rules_hint (tree_view, FALSE);
262   
263   switch (type)
264     {
265     case COLUMNS_NONE:
266       break;
267
268     case COLUMNS_LOTS:
269       /* with lots of columns we need to turn on rules */
270       gtk_tree_view_set_rules_hint (tree_view, TRUE);
271       
272       rend = gtk_cell_renderer_text_new ();
273       
274       col = gtk_tree_view_column_new_with_attributes ("Column 1",
275                                                       rend,
276                                                       "text", 1,
277                                                       NULL);
278       setup_column (col);
279       
280       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
281       
282       col = gtk_tree_view_column_new();
283       gtk_tree_view_column_set_title (col, "Column 2");
284       
285       rend = gtk_cell_renderer_pixbuf_new ();
286       gtk_tree_view_column_pack_start (col, rend, FALSE);
287       gtk_tree_view_column_add_attribute (col, rend, "pixbuf", 2);
288       rend = gtk_cell_renderer_text_new ();
289       gtk_tree_view_column_pack_start (col, rend, TRUE);
290       gtk_tree_view_column_add_attribute (col, rend, "text", 0);
291
292       setup_column (col);
293       
294       
295       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
296       gtk_tree_view_set_expander_column (tree_view, col);
297       
298       rend = gtk_cell_renderer_toggle_new ();
299
300       g_signal_connect (G_OBJECT (rend), "toggled",
301                         GTK_SIGNAL_FUNC (toggled_callback), tree_view);
302       
303       col = gtk_tree_view_column_new_with_attributes ("Column 3",
304                                                       rend,
305                                                       "active", BOOL_COLUMN,
306                                                       NULL);
307
308       setup_column (col);
309       
310       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
311
312       pixbuf = gdk_pixbuf_new_from_xpm_data ((const char **)book_closed_xpm);
313
314       image = gtk_image_new_from_pixbuf (pixbuf);
315
316       g_object_unref (G_OBJECT (pixbuf));
317       
318       gtk_widget_show (image);
319       
320       gtk_tree_view_column_set_widget (col, image);
321       
322       rend = gtk_cell_renderer_toggle_new ();
323
324       /* you could also set this per-row by tying it to a column
325        * in the model of course.
326        */
327       g_object_set (G_OBJECT (rend), "radio", TRUE, NULL);
328       
329       g_signal_connect (G_OBJECT (rend), "toggled",
330                         G_CALLBACK (toggled_callback), tree_view);
331       
332       col = gtk_tree_view_column_new_with_attributes ("Column 4",
333                                                       rend,
334                                                       "active", BOOL_COLUMN,
335                                                       NULL);
336
337       setup_column (col);
338       
339       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
340       
341 #if 0
342
343       rend = gtk_cell_renderer_text_new ();
344       
345       col = gtk_tree_view_column_new_with_attributes ("Column 5",
346                                                       rend,
347                                                       "text", 3,
348                                                       NULL);
349
350       setup_column (col);
351       
352       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
353       
354       rend = gtk_cell_renderer_text_new ();
355       
356       col = gtk_tree_view_column_new_with_attributes ("Column 6",
357                                                       rend,
358                                                       "text", 4,
359                                                       NULL);
360
361       setup_column (col);
362       
363       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
364       
365       rend = gtk_cell_renderer_text_new ();
366       
367       col = gtk_tree_view_column_new_with_attributes ("Column 7",
368                                                       rend,
369                                                       "text", 5,
370                                                       NULL);
371
372       setup_column (col);
373       
374       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
375       
376       rend = gtk_cell_renderer_text_new ();
377       
378       col = gtk_tree_view_column_new_with_attributes ("Column 8",
379                                                       rend,
380                                                       "text", 6,
381                                                       NULL);
382
383       setup_column (col);
384       
385       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
386       
387       rend = gtk_cell_renderer_text_new ();
388       
389       col = gtk_tree_view_column_new_with_attributes ("Column 9",
390                                                       rend,
391                                                       "text", 7,
392                                                       NULL);
393
394       setup_column (col);
395       
396       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
397       
398       rend = gtk_cell_renderer_text_new ();
399       
400       col = gtk_tree_view_column_new_with_attributes ("Column 10",
401                                                       rend,
402                                                       "text", 8,
403                                                       NULL);
404
405       setup_column (col);
406       
407       gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), col);
408       
409 #endif
410       
411       /* FALL THRU */
412       
413     case COLUMNS_ONE:
414       rend = gtk_cell_renderer_text_new ();
415       
416       col = gtk_tree_view_column_new_with_attributes ("Column 0",
417                                                       rend,
418                                                       "text", 0,
419                                                       NULL);
420
421       setup_column (col);
422       
423       gtk_tree_view_insert_column (GTK_TREE_VIEW (tree_view), col, 0);
424     default:
425       break;
426     }
427 }
428
429 static ColumnsType
430 get_columns_type (void)
431 {
432   return current_column_type;
433 }
434
435 static GdkPixbuf *our_pixbuf;
436   
437 typedef enum
438 {
439   /*   MODEL_TYPES, */
440   MODEL_TREE,
441   MODEL_LIST,
442   MODEL_SORTED_TREE,
443   MODEL_SORTED_LIST,
444   MODEL_EMPTY_LIST,
445   MODEL_EMPTY_TREE,
446   MODEL_NULL,
447   MODEL_LAST
448 } ModelType;
449
450 /* FIXME add a custom model to test */
451 static GtkTreeModel *models[MODEL_LAST];
452 static const char *model_names[MODEL_LAST] = {
453   "GtkTreeStore",
454   "GtkListStore",
455   "GtkTreeModelSort wrapping GtkTreeStore",
456   "GtkTreeModelSort wrapping GtkListStore",
457   "Empty GtkListStore",
458   "Empty GtkTreeStore",
459   "NULL (no model)"
460 };
461
462 static GtkTreeModel*
463 create_list_model (void)
464 {
465   GtkListStore *store;
466   GtkTreeIter iter;
467   gint i;
468   GType *t;
469
470   t = get_model_types ();
471   
472   store = gtk_list_store_new (N_COLUMNS,
473                               t[0], t[1], t[2],
474                               t[3], t[4], t[5],
475                               t[6], t[7], t[8]);
476
477   i = 0;
478   while (i < 200)
479     {
480       char *msg;
481       
482       gtk_list_store_append (store, &iter);
483
484       msg = g_strdup_printf ("%d", i);
485       
486       gtk_list_store_set (store, &iter, 0, msg, 1, "Foo! Foo! Foo!",
487                           2, our_pixbuf,
488                           3, 7.0, 4, (guint) 9000,
489                           5, 'f', 6, 'g',
490                           7, TRUE, 8, 23245454,
491                           -1);
492
493       g_free (msg);
494       
495       ++i;
496     }
497
498   return GTK_TREE_MODEL (store);
499 }
500
501 static void
502 typesystem_recurse (GType        type,
503                     GtkTreeIter *parent_iter,
504                     GtkTreeStore *store)
505 {
506   GType* children;
507   guint n_children = 0;
508   gint i;
509   GtkTreeIter iter;
510   gchar *str;
511   
512   gtk_tree_store_append (store, &iter, parent_iter);
513
514   str = g_strdup_printf ("%d", type);
515   gtk_tree_store_set (store, &iter, 0, str, 1, g_type_name (type),
516                       2, our_pixbuf,
517                       3, 7.0, 4, (guint) 9000,
518                       5, 'f', 6, 'g',
519                       7, TRUE, 8, 23245454,
520                       -1);
521   g_free (str);
522   
523   children = g_type_children (type, &n_children);
524
525   i = 0;
526   while (i < n_children)
527     {
528       typesystem_recurse (children[i], &iter, store);
529
530       ++i;
531     }
532   
533   g_free (children);
534 }
535
536 static GtkTreeModel*
537 create_tree_model (void)
538 {
539   GtkTreeStore *store;
540   gint i;
541   GType *t;
542   volatile GType dummy; /* G_GNUC_CONST makes the optimizer remove
543                          * get_type calls if you don't do something
544                          * like this
545                          */
546   
547   /* Make the tree more interesting */
548   dummy = gtk_scrolled_window_get_type ();
549   dummy = gtk_label_get_type ();
550   dummy = gtk_hscrollbar_get_type ();
551   dummy = gtk_vscrollbar_get_type ();
552   dummy = pango_layout_get_type ();
553
554   t = get_model_types ();
555   
556   store = gtk_tree_store_new (N_COLUMNS,
557                               t[0], t[1], t[2],
558                               t[3], t[4], t[5],
559                               t[6], t[7], t[8]);
560
561   i = 0;
562   while (i < G_TYPE_RESERVED_LAST_FUNDAMENTAL)
563     {
564       typesystem_recurse (i, NULL, store);
565       
566       ++i;
567     }
568
569   return GTK_TREE_MODEL (store);
570 }
571
572 static void
573 model_selected (GtkOptionMenu *om, gpointer data)
574 {
575   GtkTreeView *tree_view = GTK_TREE_VIEW (data);
576   gint hist;
577
578   hist = gtk_option_menu_get_history (om);
579
580   if (models[hist] != gtk_tree_view_get_model (tree_view))
581     {
582       gtk_tree_view_set_model (tree_view, models[hist]);
583     }
584 }
585
586 static void
587 columns_selected (GtkOptionMenu *om, gpointer data)
588 {
589   GtkTreeView *tree_view = GTK_TREE_VIEW (data);
590   gint hist;
591
592   hist = gtk_option_menu_get_history (om);
593
594   if (hist != get_columns_type ())
595     {
596       set_columns_type (tree_view, hist);
597     }
598 }
599
600
601 enum
602 {
603   TARGET_GTK_TREE_MODEL_ROW
604 };
605
606 static GtkTargetEntry row_targets[] = {
607   { "GTK_TREE_MODEL_ROW", GTK_TARGET_SAME_APP,
608     TARGET_GTK_TREE_MODEL_ROW }
609 };
610
611 int
612 main (int    argc,
613       char **argv)
614 {
615   GtkWidget *window;
616   GtkWidget *sw;
617   GtkWidget *tv;
618   GtkWidget *table;
619   GtkWidget *om;
620   GtkWidget *menu;
621   GtkTreeModel *model;
622   gint i;
623   
624   gtk_init (&argc, &argv);
625
626   our_pixbuf = gdk_pixbuf_new_from_xpm_data ((const char **) book_closed_xpm);  
627   
628 #if 0
629   models[MODEL_TYPES] = GTK_TREE_MODEL (gtk_tree_model_types_new ());
630 #endif
631   models[MODEL_LIST] = create_list_model ();
632   models[MODEL_TREE] = create_tree_model ();
633
634   model = create_list_model ();
635   models[MODEL_SORTED_LIST] = gtk_tree_model_sort_new_with_model (model);
636   g_object_unref (G_OBJECT (model));
637
638   model = create_tree_model ();
639   models[MODEL_SORTED_TREE] = gtk_tree_model_sort_new_with_model (model);
640   g_object_unref (G_OBJECT (model));
641
642   models[MODEL_EMPTY_LIST] = GTK_TREE_MODEL (gtk_list_store_new (1, G_TYPE_INT));
643   models[MODEL_EMPTY_TREE] = GTK_TREE_MODEL (gtk_tree_store_new (1, G_TYPE_INT));
644   
645   models[MODEL_NULL] = NULL;
646
647   run_automated_tests ();
648   
649   window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
650
651   gtk_window_set_default_size (GTK_WINDOW (window), 400, 400);
652
653   table = gtk_table_new (3, 1, FALSE);
654
655   gtk_container_add (GTK_CONTAINER (window), table);
656
657   tv = gtk_tree_view_new_with_model (models[0]);
658   
659   gtk_tree_view_enable_model_drag_source (GTK_TREE_VIEW (tv),
660                                           GDK_BUTTON1_MASK,
661                                           row_targets,
662                                           G_N_ELEMENTS (row_targets),
663                                           GDK_ACTION_MOVE | GDK_ACTION_COPY);
664
665   gtk_tree_view_enable_model_drag_dest (GTK_TREE_VIEW (tv),
666                                         row_targets,
667                                         G_N_ELEMENTS (row_targets),
668                                         GDK_ACTION_MOVE | GDK_ACTION_COPY);
669   
670   /* Model menu */
671
672   menu = gtk_menu_new ();
673   
674   i = 0;
675   while (i < MODEL_LAST)
676     {
677       GtkWidget *mi;
678       const char *name;
679
680       name = model_names[i];
681       
682       mi = gtk_menu_item_new_with_label (name);
683
684       gtk_menu_shell_append (GTK_MENU_SHELL (menu), mi);
685
686 #if 0
687       window = create_prop_editor (G_OBJECT (models[i]));
688
689       gtk_window_set_title (GTK_WINDOW (window),                            
690                             name);
691 #endif
692
693       ++i;
694     }
695   gtk_widget_show_all (menu);
696   
697   om = gtk_option_menu_new ();
698   gtk_option_menu_set_menu (GTK_OPTION_MENU (om), menu);
699   
700   gtk_table_attach (GTK_TABLE (table), om,
701                     0, 1, 0, 1,
702                     0, 0, 
703                     0, 0);
704
705   gtk_signal_connect (GTK_OBJECT (om),
706                       "changed",
707                       GTK_SIGNAL_FUNC (model_selected),
708                       tv);
709   
710   /* Columns menu */
711
712   menu = gtk_menu_new ();
713   
714   i = 0;
715   while (i < COLUMNS_LAST)
716     {
717       GtkWidget *mi;
718       const char *name;
719
720       name = column_type_names[i];
721       
722       mi = gtk_menu_item_new_with_label (name);
723
724       gtk_menu_shell_append (GTK_MENU_SHELL (menu), mi);
725
726       ++i;
727     }
728   gtk_widget_show_all (menu);
729   
730   om = gtk_option_menu_new ();
731   gtk_option_menu_set_menu (GTK_OPTION_MENU (om), menu);
732   
733   gtk_table_attach (GTK_TABLE (table), om,
734                     0, 1, 1, 2,
735                     0, 0, 
736                     0, 0);
737
738   set_columns_type (GTK_TREE_VIEW (tv), COLUMNS_LOTS);
739   gtk_option_menu_set_history (GTK_OPTION_MENU (om), COLUMNS_LOTS);
740   
741   gtk_signal_connect (GTK_OBJECT (om),
742                       "changed",
743                       GTK_SIGNAL_FUNC (columns_selected),
744                       tv);
745   
746   sw = gtk_scrolled_window_new (NULL, NULL);
747   gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw),
748                                   GTK_POLICY_AUTOMATIC,
749                                   GTK_POLICY_AUTOMATIC);
750   
751   gtk_table_attach (GTK_TABLE (table), sw,
752                     0, 1, 2, 3,
753                     GTK_EXPAND | GTK_FILL,
754                     GTK_EXPAND | GTK_FILL,
755                     0, 0);
756   
757   gtk_container_add (GTK_CONTAINER (sw), tv);
758   
759   gtk_widget_show_all (window);
760   
761   gtk_main ();
762
763   return 0;
764 }
765
766 /*
767  * GtkTreeModelTypes
768  */
769
770 static void         gtk_tree_model_types_init                 (GtkTreeModelTypes      *model_types);
771 static void         gtk_tree_model_types_tree_model_init      (GtkTreeModelIface   *iface);
772 static gint         gtk_real_model_types_get_n_columns   (GtkTreeModel        *tree_model);
773 static GType        gtk_real_model_types_get_column_type (GtkTreeModel        *tree_model,
774                                                            gint                 index);
775 static GtkTreePath *gtk_real_model_types_get_path        (GtkTreeModel        *tree_model,
776                                                            GtkTreeIter         *iter);
777 static void         gtk_real_model_types_get_value       (GtkTreeModel        *tree_model,
778                                                            GtkTreeIter         *iter,
779                                                            gint                 column,
780                                                            GValue              *value);
781 static gboolean     gtk_real_model_types_iter_next       (GtkTreeModel        *tree_model,
782                                                            GtkTreeIter         *iter);
783 static gboolean     gtk_real_model_types_iter_children   (GtkTreeModel        *tree_model,
784                                                            GtkTreeIter         *iter,
785                                                            GtkTreeIter         *parent);
786 static gboolean     gtk_real_model_types_iter_has_child  (GtkTreeModel        *tree_model,
787                                                            GtkTreeIter         *iter);
788 static gint         gtk_real_model_types_iter_n_children (GtkTreeModel        *tree_model,
789                                                            GtkTreeIter         *iter);
790 static gboolean     gtk_real_model_types_iter_nth_child  (GtkTreeModel        *tree_model,
791                                                            GtkTreeIter         *iter,
792                                                            GtkTreeIter         *parent,
793                                                            gint                 n);
794 static gboolean     gtk_real_model_types_iter_parent     (GtkTreeModel        *tree_model,
795                                                            GtkTreeIter         *iter,
796                                                            GtkTreeIter         *child);
797
798
799 GtkType
800 gtk_tree_model_types_get_type (void)
801 {
802   static GtkType model_types_type = 0;
803
804   if (!model_types_type)
805     {
806       static const GTypeInfo model_types_info =
807       {
808         sizeof (GtkTreeModelTypesClass),
809         NULL,           /* base_init */
810         NULL,           /* base_finalize */
811         NULL,           /* class_init */
812         NULL,           /* class_finalize */
813         NULL,           /* class_data */
814         sizeof (GtkTreeModelTypes),
815         0,
816         (GInstanceInitFunc) gtk_tree_model_types_init
817       };
818
819       static const GInterfaceInfo tree_model_info =
820       {
821         (GInterfaceInitFunc) gtk_tree_model_types_tree_model_init,
822         NULL,
823         NULL
824       };
825
826       model_types_type = g_type_register_static (GTK_TYPE_OBJECT, "GtkTreeModelTypes", &model_types_info, 0);
827       g_type_add_interface_static (model_types_type,
828                                    GTK_TYPE_TREE_MODEL,
829                                    &tree_model_info);
830     }
831
832   return model_types_type;
833 }
834
835 GtkTreeModelTypes *
836 gtk_tree_model_types_new (void)
837 {
838   GtkTreeModelTypes *retval;
839
840   retval = GTK_TREE_MODEL_TYPES (g_object_new (GTK_TYPE_MODEL_TYPES, NULL));
841
842   return retval;
843 }
844
845 static void
846 gtk_tree_model_types_tree_model_init (GtkTreeModelIface *iface)
847 {
848   iface->get_n_columns = gtk_real_model_types_get_n_columns;
849   iface->get_column_type = gtk_real_model_types_get_column_type;
850   iface->get_path = gtk_real_model_types_get_path;
851   iface->get_value = gtk_real_model_types_get_value;
852   iface->iter_next = gtk_real_model_types_iter_next;
853   iface->iter_children = gtk_real_model_types_iter_children;
854   iface->iter_has_child = gtk_real_model_types_iter_has_child;
855   iface->iter_n_children = gtk_real_model_types_iter_n_children;
856   iface->iter_nth_child = gtk_real_model_types_iter_nth_child;
857   iface->iter_parent = gtk_real_model_types_iter_parent;
858 }
859
860 static void
861 gtk_tree_model_types_init (GtkTreeModelTypes *model_types)
862 {
863   model_types->stamp = g_random_int ();
864 }
865
866 static GType column_types[] = {
867   G_TYPE_STRING, /* GType */
868   G_TYPE_STRING  /* type name */
869 };
870   
871 static gint
872 gtk_real_model_types_get_n_columns (GtkTreeModel *tree_model)
873 {
874   return G_N_ELEMENTS (column_types);
875 }
876
877 static GType
878 gtk_real_model_types_get_column_type (GtkTreeModel *tree_model,
879                                       gint          index)
880 {
881   g_return_val_if_fail (index < G_N_ELEMENTS (column_types), G_TYPE_INVALID);
882   
883   return column_types[index];
884 }
885
886 #if 0
887 /* Use default implementation of this */
888 static gboolean
889 gtk_real_model_types_get_iter (GtkTreeModel *tree_model,
890                                GtkTreeIter  *iter,
891                                GtkTreePath  *path)
892 {
893   
894 }
895 #endif
896
897 /* The toplevel nodes of the tree are the reserved types, G_TYPE_NONE through
898  * G_TYPE_RESERVED_FUNDAMENTAL.
899  */
900
901 static GtkTreePath *
902 gtk_real_model_types_get_path (GtkTreeModel *tree_model,
903                                GtkTreeIter  *iter)
904 {
905   GtkTreePath *retval;
906   GType type;
907   GType parent;
908   
909   g_return_val_if_fail (GTK_IS_TREE_MODEL_TYPES (tree_model), NULL);
910   g_return_val_if_fail (iter != NULL, NULL);
911
912   type = GPOINTER_TO_INT (iter->user_data);
913   
914   retval = gtk_tree_path_new ();
915   
916   parent = g_type_parent (type);
917   while (parent != G_TYPE_INVALID)
918     {
919       GType* children = g_type_children (parent, NULL);
920       gint i = 0;
921
922       if (!children || children[0] == G_TYPE_INVALID)
923         {
924           g_warning ("bad iterator?");
925           return NULL;
926         }
927       
928       while (children[i] != type)
929         ++i;
930
931       gtk_tree_path_prepend_index (retval, i);
932
933       g_free (children);
934       
935       type = parent;
936       parent = g_type_parent (parent);
937     }
938
939   /* The fundamental type itself is the index on the toplevel */
940   gtk_tree_path_prepend_index (retval, type);
941
942   return retval;
943 }
944
945 static void
946 gtk_real_model_types_get_value (GtkTreeModel *tree_model,
947                                 GtkTreeIter  *iter,
948                                 gint          column,
949                                 GValue       *value)
950 {
951   GType type;
952
953   type = GPOINTER_TO_INT (iter->user_data);
954
955   switch (column)
956     {
957     case 0:
958       {
959         gchar *str;
960         
961         g_value_init (value, G_TYPE_STRING);
962
963         str = g_strdup_printf ("%d", type);
964         g_value_set_string (value, str);
965         g_free (str);
966       }
967       break;
968
969     case 1:
970       g_value_init (value, G_TYPE_STRING);
971       g_value_set_string (value, g_type_name (type));
972       break;
973
974     default:
975       g_warning ("Bad column %d requested", column);
976     }
977 }
978
979 static gboolean
980 gtk_real_model_types_iter_next (GtkTreeModel  *tree_model,
981                                 GtkTreeIter   *iter)
982 {
983   
984   GType parent;
985   GType type;
986
987   type = GPOINTER_TO_INT (iter->user_data);
988
989   parent = g_type_parent (type);
990
991   if (parent == G_TYPE_INVALID)
992     {
993       /* fundamental type, add 1 */
994       if ((type + 1) < G_TYPE_RESERVED_LAST_FUNDAMENTAL)
995         {
996           iter->user_data = GINT_TO_POINTER (type + 1);
997           return TRUE;
998         }
999       else
1000         return FALSE;
1001     }
1002   else
1003     {
1004       GType* children = g_type_children (parent, NULL);
1005       gint i = 0;
1006
1007       g_assert (children != NULL);
1008       
1009       while (children[i] != type)
1010         ++i;
1011   
1012       ++i;
1013
1014       if (children[i] != G_TYPE_INVALID)
1015         {
1016           g_free (children);
1017           iter->user_data = GINT_TO_POINTER (children[i]);
1018           return TRUE;
1019         }
1020       else
1021         {
1022           g_free (children);
1023           return FALSE;
1024         }
1025     }
1026 }
1027
1028 static gboolean
1029 gtk_real_model_types_iter_children (GtkTreeModel *tree_model,
1030                                     GtkTreeIter  *iter,
1031                                     GtkTreeIter  *parent)
1032 {
1033   GType type;
1034   GType* children;
1035   
1036   type = GPOINTER_TO_INT (parent->user_data);
1037
1038   children = g_type_children (type, NULL);
1039
1040   if (!children || children[0] == G_TYPE_INVALID)
1041     {
1042       g_free (children);
1043       return FALSE;
1044     }
1045   else
1046     {
1047       iter->user_data = GINT_TO_POINTER (children[0]);
1048       g_free (children);
1049       return TRUE;
1050     }
1051 }
1052
1053 static gboolean
1054 gtk_real_model_types_iter_has_child (GtkTreeModel *tree_model,
1055                                      GtkTreeIter  *iter)
1056 {
1057   GType type;
1058   GType* children;
1059   
1060   type = GPOINTER_TO_INT (iter->user_data);
1061   
1062   children = g_type_children (type, NULL);
1063
1064   if (!children || children[0] == G_TYPE_INVALID)
1065     {
1066       g_free (children);
1067       return FALSE;
1068     }
1069   else
1070     {
1071       g_free (children);
1072       return TRUE;
1073     }
1074 }
1075
1076 static gint
1077 gtk_real_model_types_iter_n_children (GtkTreeModel *tree_model,
1078                                       GtkTreeIter  *iter)
1079 {
1080   if (iter == NULL)
1081     {
1082       return G_TYPE_RESERVED_LAST_FUNDAMENTAL - 1;
1083     }
1084   else
1085     {
1086       GType type;
1087       GType* children;
1088       guint n_children = 0;
1089
1090       type = GPOINTER_TO_INT (iter->user_data);
1091       
1092       children = g_type_children (type, &n_children);
1093       
1094       g_free (children);
1095       
1096       return n_children;
1097     }
1098 }
1099
1100 static gboolean
1101 gtk_real_model_types_iter_nth_child (GtkTreeModel *tree_model,
1102                                      GtkTreeIter  *iter,
1103                                      GtkTreeIter  *parent,
1104                                      gint          n)
1105 {  
1106   if (parent == NULL)
1107     {
1108       /* fundamental type */
1109       if (n < G_TYPE_RESERVED_LAST_FUNDAMENTAL)
1110         {
1111           iter->user_data = GINT_TO_POINTER (n);
1112           return TRUE;
1113         }
1114       else
1115         return FALSE;
1116     }
1117   else
1118     {
1119       GType type = GPOINTER_TO_INT (parent->user_data);      
1120       guint n_children = 0;
1121       GType* children = g_type_children (type, &n_children);
1122
1123       if (n_children == 0)
1124         {
1125           g_free (children);
1126           return FALSE;
1127         }
1128       else if (n >= n_children)
1129         {
1130           g_free (children);
1131           return FALSE;
1132         }
1133       else
1134         {
1135           iter->user_data = GINT_TO_POINTER (children[n]);
1136           g_free (children);
1137
1138           return TRUE;
1139         }
1140     }
1141 }
1142
1143 static gboolean
1144 gtk_real_model_types_iter_parent (GtkTreeModel *tree_model,
1145                                   GtkTreeIter  *iter,
1146                                   GtkTreeIter  *child)
1147 {
1148   GType type;
1149   GType parent;
1150   
1151   type = GPOINTER_TO_INT (child->user_data);
1152   
1153   parent = g_type_parent (type);
1154   
1155   if (parent == G_TYPE_INVALID)
1156     {
1157       if (type >= G_TYPE_RESERVED_LAST_FUNDAMENTAL)
1158         g_warning ("no parent for %d %s\n", type, g_type_name (type));
1159       return FALSE;
1160     }
1161   else
1162     {
1163       iter->user_data = GINT_TO_POINTER (parent);
1164       
1165       return TRUE;
1166     }
1167 }
1168
1169 /*
1170  * Automated testing
1171  */
1172
1173 static void
1174 treestore_torture_recurse (GtkTreeStore *store,
1175                            GtkTreeIter  *root,
1176                            gint          depth)
1177 {
1178   GtkTreeModel *model;
1179   gint i;
1180   GtkTreeIter iter;  
1181   
1182   model = GTK_TREE_MODEL (store);    
1183
1184   if (depth > 2)
1185     return;
1186
1187   ++depth;
1188
1189   gtk_tree_store_append (store, &iter, root);
1190   
1191   gtk_tree_model_iter_children (model, &iter, root);
1192   
1193   i = 0;
1194   while (i < 100)
1195     {
1196       gtk_tree_store_append (store, &iter, root);
1197       ++i;
1198     }
1199
1200   while (gtk_tree_model_iter_children (model, &iter, root))
1201     gtk_tree_store_remove (store, &iter);
1202
1203   gtk_tree_store_append (store, &iter, root);
1204
1205   /* inserts before last node in tree */
1206   i = 0;
1207   while (i < 100)
1208     {
1209       gtk_tree_store_insert_before (store, &iter, root, &iter);
1210       ++i;
1211     }
1212
1213   /* inserts after the node before the last node */
1214   i = 0;
1215   while (i < 100)
1216     {
1217       gtk_tree_store_insert_after (store, &iter, root, &iter);
1218       ++i;
1219     }
1220
1221   /* inserts after the last node */
1222   gtk_tree_store_append (store, &iter, root);
1223     
1224   i = 0;
1225   while (i < 100)
1226     {
1227       gtk_tree_store_insert_after (store, &iter, root, &iter);
1228       ++i;
1229     }
1230
1231   /* remove everything again */
1232   while (gtk_tree_model_iter_children (model, &iter, root))
1233     gtk_tree_store_remove (store, &iter);
1234
1235
1236     /* Prepends */
1237   gtk_tree_store_prepend (store, &iter, root);
1238     
1239   i = 0;
1240   while (i < 100)
1241     {
1242       gtk_tree_store_prepend (store, &iter, root);
1243       ++i;
1244     }
1245
1246   /* remove everything again */
1247   while (gtk_tree_model_iter_children (model, &iter, root))
1248     gtk_tree_store_remove (store, &iter);
1249
1250   gtk_tree_store_append (store, &iter, root);
1251   gtk_tree_store_append (store, &iter, root);
1252   gtk_tree_store_append (store, &iter, root);
1253   gtk_tree_store_append (store, &iter, root);
1254
1255   while (gtk_tree_model_iter_children (model, &iter, root))
1256     {
1257       treestore_torture_recurse (store, &iter, depth);
1258       gtk_tree_store_remove (store, &iter);
1259     }
1260 }
1261
1262 static void
1263 run_automated_tests (void)
1264 {
1265   g_print ("Running automated tests...\n");
1266   
1267   /* FIXME TreePath basic verification */
1268
1269   /* FIXME generic consistency checks on the models */
1270
1271   {
1272     /* Make sure list store mutations don't crash anything */
1273     GtkListStore *store;
1274     GtkTreeModel *model;
1275     gint i;
1276     GtkTreeIter iter;
1277     
1278     store = gtk_list_store_new (1, G_TYPE_INT);
1279
1280     model = GTK_TREE_MODEL (store);
1281     
1282     i = 0;
1283     while (i < 100)
1284       {
1285         gtk_list_store_append (store, &iter);
1286         ++i;
1287       }
1288
1289     while (gtk_tree_model_get_iter_root (model, &iter))
1290       gtk_list_store_remove (store, &iter);
1291
1292     gtk_list_store_append (store, &iter);
1293
1294     /* inserts before last node in list */
1295     i = 0;
1296     while (i < 100)
1297       {
1298         gtk_list_store_insert_before (store, &iter, &iter);
1299         ++i;
1300       }
1301
1302     /* inserts after the node before the last node */
1303     i = 0;
1304     while (i < 100)
1305       {
1306         gtk_list_store_insert_after (store, &iter, &iter);
1307         ++i;
1308       }
1309
1310     /* inserts after the last node */
1311     gtk_list_store_append (store, &iter);
1312     
1313     i = 0;
1314     while (i < 100)
1315       {
1316         gtk_list_store_insert_after (store, &iter, &iter);
1317         ++i;
1318       }
1319
1320     /* remove everything again */
1321     while (gtk_tree_model_get_iter_root (model, &iter))
1322       gtk_list_store_remove (store, &iter);
1323
1324
1325     /* Prepends */
1326     gtk_list_store_prepend (store, &iter);
1327     
1328     i = 0;
1329     while (i < 100)
1330       {
1331         gtk_list_store_prepend (store, &iter);
1332         ++i;
1333       }
1334
1335     /* remove everything again */
1336     while (gtk_tree_model_get_iter_root (model, &iter))
1337       gtk_list_store_remove (store, &iter);
1338     
1339     g_object_unref (G_OBJECT (store));
1340   }
1341
1342   {
1343     /* Make sure tree store mutations don't crash anything */
1344     GtkTreeStore *store;
1345     GtkTreeIter root;
1346
1347     store = gtk_tree_store_new (1, G_TYPE_INT);
1348     gtk_tree_store_append (GTK_TREE_STORE (store), &root, NULL);
1349     /* Remove test until it is rewritten to work */
1350     /*    treestore_torture_recurse (store, &root, 0);*/
1351     
1352     g_object_unref (G_OBJECT (store));
1353   }
1354
1355   g_print ("Passed.\n");
1356 }