]> git.saurik.com Git - wxWidgets.git/blob - src/gtk/dataview.cpp
Route data from wxDataViewModel in a wxVariant
[wxWidgets.git] / src / gtk / dataview.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: dataview.cpp
3 // Purpose: wxDataViewCtrl GTK+2 implementation
4 // Author: Robert Roebling
5 // Id: $Id$
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
9
10 // For compilers that support precompilation, includes "wx.h".
11 #include "wx/wxprec.h"
12
13 #include "wx/defs.h"
14
15 #if wxUSE_DATAVIEWCTRL
16
17 #include "wx/dataview.h"
18 #include "wx/stockitem.h"
19
20 #include "wx/gtk/private.h"
21 #include "wx/gtk/win_gtk.h"
22
23 #include <gobject/gvaluecollector.h>
24 #include <gtk/gtktreemodel.h>
25 #include <gtk/gtktreednd.h>
26
27 //-----------------------------------------------------------------------------
28 // classes
29 //-----------------------------------------------------------------------------
30
31 class wxDataViewCtrl;
32
33 //-----------------------------------------------------------------------------
34 // idle system
35 //-----------------------------------------------------------------------------
36
37 extern void wxapp_install_idle_handler();
38 extern bool g_isIdle;
39
40 //-----------------------------------------------------------------------------
41 // data
42 //-----------------------------------------------------------------------------
43
44 extern bool g_blockEventsOnDrag;
45
46 //-----------------------------------------------------------------------------
47 // define new GTK+ class wxGtkListStore
48 //-----------------------------------------------------------------------------
49
50 extern "C" {
51
52 #define GTK_TYPE_WX_LIST_STORE (gtk_wx_list_store_get_type ())
53 #define GTK_WX_LIST_STORE(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_LIST_STORE, GtkWxListStore))
54 #define GTK_WX_LIST_STORE_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GTK_TYPE_WX_LIST_STORE, GtkWxListStoreClass))
55 #define GTK_IS_WX_LIST_STORE(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_LIST_STORE))
56 #define GTK_IS_WX_LIST_STORE_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_LIST_STORE))
57 #define GTK_WX_LIST_STORE_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_WX_LIST_STORE, GtkWxListStoreClass))
58
59 GType gtk_wx_list_store_get_type (void);
60
61 typedef struct _GtkWxListStore GtkWxListStore;
62 typedef struct _GtkWxListStoreClass GtkWxListStoreClass;
63
64 struct _GtkWxListStore
65 {
66 GObject parent;
67
68 /*< private >*/
69 gint stamp;
70 wxDataViewListModel *model;
71 };
72
73 struct _GtkWxListStoreClass
74 {
75 GObjectClass parent_class;
76
77 };
78
79 static GtkWxListStore *wxgtk_list_store_new ();
80 static void wxgtk_list_store_init (GtkWxListStore *list_store);
81 static void wxgtk_list_store_class_init (GtkWxListStoreClass *klass);
82 static void wxgtk_list_store_tree_model_init (GtkTreeModelIface *iface);
83 static void wxgtk_list_store_finalize (GObject *object);
84 static GtkTreeModelFlags wxgtk_list_store_get_flags (GtkTreeModel *tree_model);
85 static gint wxgtk_list_store_get_n_columns (GtkTreeModel *tree_model);
86 static GType wxgtk_list_store_get_column_type (GtkTreeModel *tree_model,
87 gint index);
88 static gboolean wxgtk_list_store_get_iter (GtkTreeModel *tree_model,
89 GtkTreeIter *iter,
90 GtkTreePath *path);
91 static GtkTreePath *wxgtk_list_store_get_path (GtkTreeModel *tree_model,
92 GtkTreeIter *iter);
93 static void wxgtk_list_store_get_value (GtkTreeModel *tree_model,
94 GtkTreeIter *iter,
95 gint column,
96 GValue *value);
97 static gboolean wxgtk_list_store_iter_next (GtkTreeModel *tree_model,
98 GtkTreeIter *iter);
99 static gboolean wxgtk_list_store_iter_children (GtkTreeModel *tree_model,
100 GtkTreeIter *iter,
101 GtkTreeIter *parent);
102 static gboolean wxgtk_list_store_iter_has_child (GtkTreeModel *tree_model,
103 GtkTreeIter *iter);
104 static gint wxgtk_list_store_iter_n_children (GtkTreeModel *tree_model,
105 GtkTreeIter *iter);
106 static gboolean wxgtk_list_store_iter_nth_child (GtkTreeModel *tree_model,
107 GtkTreeIter *iter,
108 GtkTreeIter *parent,
109 gint n);
110 static gboolean wxgtk_list_store_iter_parent (GtkTreeModel *tree_model,
111 GtkTreeIter *iter,
112 GtkTreeIter *child);
113
114 static GObjectClass *parent_class = NULL;
115
116 GType
117 gtk_wx_list_store_get_type (void)
118 {
119 static GType list_store_type = 0;
120
121 if (!list_store_type)
122 {
123 static const GTypeInfo list_store_info =
124 {
125 sizeof (GtkWxListStoreClass),
126 NULL, /* base_init */
127 NULL, /* base_finalize */
128 (GClassInitFunc) wxgtk_list_store_class_init,
129 NULL, /* class_finalize */
130 NULL, /* class_data */
131 sizeof (GtkWxListStore),
132 0,
133 (GInstanceInitFunc) wxgtk_list_store_init,
134 };
135
136 static const GInterfaceInfo tree_model_info =
137 {
138 (GInterfaceInitFunc) wxgtk_list_store_tree_model_init,
139 NULL,
140 NULL
141 };
142
143 list_store_type = g_type_register_static (G_TYPE_OBJECT, "GtkWxListStore",
144 &list_store_info, (GTypeFlags)0 );
145
146 g_type_add_interface_static (list_store_type,
147 GTK_TYPE_TREE_MODEL,
148 &tree_model_info);
149 }
150
151 return list_store_type;
152 }
153
154 static GtkWxListStore *
155 wxgtk_list_store_new()
156 {
157 GtkWxListStore *retval = (GtkWxListStore *) g_object_new (GTK_TYPE_WX_LIST_STORE, NULL);
158 return retval;
159 }
160
161 static void
162 wxgtk_list_store_class_init (GtkWxListStoreClass *klass)
163 {
164 GObjectClass *object_class;
165 parent_class = (GObjectClass*) g_type_class_peek_parent (klass);
166 object_class = (GObjectClass*) klass;
167 object_class->finalize = wxgtk_list_store_finalize;
168 }
169
170 static void
171 wxgtk_list_store_tree_model_init (GtkTreeModelIface *iface)
172 {
173 iface->get_flags = wxgtk_list_store_get_flags;
174 iface->get_n_columns = wxgtk_list_store_get_n_columns;
175 iface->get_column_type = wxgtk_list_store_get_column_type;
176 iface->get_iter = wxgtk_list_store_get_iter;
177 iface->get_path = wxgtk_list_store_get_path;
178 iface->get_value = wxgtk_list_store_get_value;
179 iface->iter_next = wxgtk_list_store_iter_next;
180 iface->iter_children = wxgtk_list_store_iter_children;
181 iface->iter_has_child = wxgtk_list_store_iter_has_child;
182 iface->iter_n_children = wxgtk_list_store_iter_n_children;
183 iface->iter_nth_child = wxgtk_list_store_iter_nth_child;
184 iface->iter_parent = wxgtk_list_store_iter_parent;
185 }
186
187 static void
188 wxgtk_list_store_init (GtkWxListStore *list_store)
189 {
190 list_store->model = NULL;
191 list_store->stamp = g_random_int();
192 }
193
194 static void
195 wxgtk_list_store_finalize (GObject *object)
196 {
197 /* GtkWxListStore *list_store = GTK_WX_LIST_STORE (object); */
198
199 /* we need to sort out, which class deletes what */
200 /* delete list_store->model; */
201
202 /* must chain up */
203 (* parent_class->finalize) (object);
204 }
205
206 } // extern "C"
207
208 //-----------------------------------------------------------------------------
209 // implement callbacks from wxGtkListStore class by letting
210 // them call the methods of wxWidgets' wxDataViewListModel
211 //-----------------------------------------------------------------------------
212
213 static GtkTreeModelFlags
214 wxgtk_list_store_get_flags (GtkTreeModel *tree_model)
215 {
216 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model), (GtkTreeModelFlags)0 );
217
218 // GTK+ list store uses a linked list for storing the
219 // items and a pointer to a child is used as the member
220 // field of a GtkTreeIter. This means that the iter is
221 // valid in the GtkListStore as long as the child exists.
222 // We use the index of the row and since the index of a
223 // specific row will change if a row above is deleted,
224 // the iter does not persist
225 return /* GTK_TREE_MODEL_ITERS_PERSIST | */ GTK_TREE_MODEL_LIST_ONLY;
226 }
227
228 static gint
229 wxgtk_list_store_get_n_columns (GtkTreeModel *tree_model)
230 {
231 GtkWxListStore *list_store = (GtkWxListStore *) tree_model;
232 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model), 0);
233
234 return list_store->model->GetNumberOfCols();
235 }
236
237 static GType
238 wxgtk_list_store_get_column_type (GtkTreeModel *tree_model,
239 gint index)
240 {
241 GtkWxListStore *list_store = (GtkWxListStore *) tree_model;
242 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model), G_TYPE_INVALID);
243
244 GType gtype = G_TYPE_INVALID;
245
246 wxString wxtype = list_store->model->GetColType( (size_t) index );
247
248 if (wxtype == wxT("string"))
249 gtype = G_TYPE_STRING;
250
251 return gtype;
252 }
253
254 static gboolean
255 wxgtk_list_store_get_iter (GtkTreeModel *tree_model,
256 GtkTreeIter *iter,
257 GtkTreePath *path)
258 {
259 GtkWxListStore *list_store = (GtkWxListStore *) tree_model;
260 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model), FALSE);
261 g_return_val_if_fail (gtk_tree_path_get_depth (path) > 0, FALSE);
262
263 size_t i = (size_t)gtk_tree_path_get_indices (path)[0];
264
265 if (i >= list_store->model->GetNumberOfRows())
266 return FALSE;
267
268 iter->stamp = list_store->stamp;
269 // user_data is just the index
270 iter->user_data = (gpointer) i;
271
272 return TRUE;
273 }
274
275 static GtkTreePath *
276 wxgtk_list_store_get_path (GtkTreeModel *tree_model,
277 GtkTreeIter *iter)
278 {
279 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model), NULL);
280 g_return_val_if_fail (iter->stamp == GTK_WX_LIST_STORE (tree_model)->stamp, NULL);
281
282 GtkTreePath *retval = gtk_tree_path_new ();
283 // user_data is just the index
284 int i = (int) iter->user_data;
285 gtk_tree_path_append_index (retval, i);
286 return retval;
287 }
288
289 static void
290 wxgtk_list_store_get_value (GtkTreeModel *tree_model,
291 GtkTreeIter *iter,
292 gint column,
293 GValue *value)
294 {
295 GtkWxListStore *list_store = (GtkWxListStore *) tree_model;
296 g_return_if_fail (GTK_IS_WX_LIST_STORE (tree_model) );
297
298 wxDataViewListModel *model = list_store->model;
299 wxString mtype = model->GetColType( (size_t) column );
300 if (mtype == wxT("string"))
301 {
302 g_value_init( value, G_TYPE_STRING );
303 wxVariant variant = model->GetValue( (size_t) column, (size_t) iter->user_data );
304 g_value_set_string( value, wxGTK_CONV(variant.GetString()) );
305 }
306 else
307 {
308 }
309
310 #if 0
311 GtkTreeDataList *list;
312 gint tmp_column = column;
313
314 g_return_if_fail (column < GTK_LIST_STORE (tree_model)->n_columns);
315 g_return_if_fail (GTK_LIST_STORE (tree_model)->stamp == iter->stamp);
316
317 list = G_SLIST (iter->user_data)->data;
318
319 while (tmp_column-- > 0 && list)
320 list = list->next;
321
322 if (list == NULL)
323 g_value_init (value, GTK_LIST_STORE (tree_model)->column_headers[column]);
324 else
325 _gtk_tree_data_list_node_to_value (list,
326 GTK_LIST_STORE (tree_model)->column_headers[column],
327 value);
328 #endif
329
330 }
331
332 static gboolean
333 wxgtk_list_store_iter_next (GtkTreeModel *tree_model,
334 GtkTreeIter *iter)
335 {
336 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model), FALSE);
337 GtkWxListStore *list_store = (GtkWxListStore *) tree_model;
338
339 g_return_val_if_fail (list_store->stamp == iter->stamp, FALSE);
340
341 int n = (int) iter->user_data;
342
343 if (n == -1)
344 return FALSE;
345
346 if (n >= (int) list_store->model->GetNumberOfRows())
347 return FALSE;
348
349 iter->user_data = (gpointer) ++n;
350
351 return TRUE;
352 }
353
354 static gboolean
355 wxgtk_list_store_iter_children (GtkTreeModel *tree_model,
356 GtkTreeIter *iter,
357 GtkTreeIter *parent)
358 {
359 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model), FALSE);
360 GtkWxListStore *list_store = (GtkWxListStore *) tree_model;
361
362 // this is a list, nodes have no children
363 if (parent)
364 return FALSE;
365
366 iter->stamp = list_store->stamp;
367 iter->user_data = (gpointer) -1;
368
369 return TRUE;
370 }
371
372 static gboolean
373 wxgtk_list_store_iter_has_child (GtkTreeModel *tree_model,
374 GtkTreeIter *iter)
375 {
376 return FALSE;
377 }
378
379 static gint
380 wxgtk_list_store_iter_n_children (GtkTreeModel *tree_model,
381 GtkTreeIter *iter)
382 {
383 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model), -1);
384 GtkWxListStore *list_store = (GtkWxListStore *) tree_model;
385
386 if (iter == NULL)
387 return (gint) list_store->model->GetNumberOfRows();
388
389 g_return_val_if_fail (list_store->stamp == iter->stamp, -1);
390
391 return 0;
392 }
393
394 static gboolean
395 wxgtk_list_store_iter_nth_child (GtkTreeModel *tree_model,
396 GtkTreeIter *iter,
397 GtkTreeIter *parent,
398 gint n)
399 {
400 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model), FALSE);
401 GtkWxListStore *list_store = (GtkWxListStore *) tree_model;
402
403 if (parent)
404 return FALSE;
405
406 if (n < 0)
407 return FALSE;
408
409 if (n >= (gint) list_store->model->GetNumberOfRows())
410 return FALSE;
411
412 iter->stamp = list_store->stamp;
413 iter->user_data = (gpointer) n;
414
415 return TRUE;
416 }
417
418 static gboolean
419 wxgtk_list_store_iter_parent (GtkTreeModel *tree_model,
420 GtkTreeIter *iter,
421 GtkTreeIter *child)
422 {
423 return FALSE;
424 }
425
426 // ---------------------------------------------------------
427 // wxGtkDataViewListModelNotifier
428 // ---------------------------------------------------------
429
430 class wxGtkDataViewListModelNotifier: public wxDataViewListModelNotifier
431 {
432 public:
433 wxGtkDataViewListModelNotifier( GtkWxListStore* gtk_store, wxDataViewListModel *wx_model );
434
435 virtual bool RowAppended();
436 virtual bool RowPrepended();
437 virtual bool RowInserted( size_t before );
438 virtual bool RowDeleted( size_t row );
439 virtual bool RowChanged( size_t row );
440 virtual bool ValueChanged( size_t row, size_t col );
441 virtual bool Cleared();
442
443 GtkWxListStore *m_gtk_store;
444 wxDataViewListModel *m_wx_model;
445 };
446
447 // ---------------------------------------------------------
448 // wxGtkDataViewListModelNotifier
449 // ---------------------------------------------------------
450
451 wxGtkDataViewListModelNotifier::wxGtkDataViewListModelNotifier(
452 GtkWxListStore* gtk_store, wxDataViewListModel *wx_model )
453 {
454 m_gtk_store = gtk_store;
455 m_wx_model = wx_model;
456 }
457
458 bool wxGtkDataViewListModelNotifier::RowAppended()
459 {
460 size_t pos = m_wx_model->GetNumberOfRows()-1;
461
462 GtkTreeIter iter;
463 iter.stamp = m_gtk_store->stamp;
464 iter.user_data = (gpointer) pos;
465
466 GtkTreePath *path = gtk_tree_path_new ();
467 gtk_tree_path_append_index (path, (gint) pos);
468 gtk_tree_model_row_inserted (GTK_TREE_MODEL (m_gtk_store), path, &iter);
469 gtk_tree_path_free (path);
470
471 return true;
472 }
473
474 bool wxGtkDataViewListModelNotifier::RowPrepended()
475 {
476 return false;
477 }
478
479 bool wxGtkDataViewListModelNotifier::RowInserted( size_t before )
480 {
481 return false;
482 }
483
484 bool wxGtkDataViewListModelNotifier::RowDeleted( size_t row )
485 {
486 return false;
487 }
488
489 bool wxGtkDataViewListModelNotifier::RowChanged( size_t row )
490 {
491 return false;
492 }
493
494 bool wxGtkDataViewListModelNotifier::ValueChanged( size_t row, size_t col )
495 {
496 return false;
497 }
498
499 bool wxGtkDataViewListModelNotifier::Cleared()
500 {
501 return false;
502 }
503
504 // ---------------------------------------------------------
505 // wxDataViewCell
506 // ---------------------------------------------------------
507
508 IMPLEMENT_ABSTRACT_CLASS(wxDataViewCell, wxDataViewCellBase)
509
510 wxDataViewCell::wxDataViewCell( const wxString &varianttype, wxDataViewCellMode mode ) :
511 wxDataViewCellBase( varianttype, mode )
512 {
513 m_renderer = NULL;
514 }
515
516 // ---------------------------------------------------------
517 // wxDataViewTextCell
518 // ---------------------------------------------------------
519
520 IMPLEMENT_ABSTRACT_CLASS(wxDataViewTextCell, wxDataViewCell)
521
522 wxDataViewTextCell::wxDataViewTextCell( const wxString &varianttype, wxDataViewCellMode mode ) :
523 wxDataViewCell( varianttype, mode )
524 {
525 m_renderer = (void*) gtk_cell_renderer_text_new();
526 }
527
528 bool wxDataViewTextCell::SetValue( const wxVariant &value )
529 {
530 wxString tmp = value;
531
532 GValue gvalue = { 0, };
533 g_value_init( &gvalue, G_TYPE_STRING );
534 g_value_set_string( &gvalue, wxGTK_CONV( tmp ) );
535 g_object_set_property( G_OBJECT(m_renderer), "text", &gvalue );
536 g_value_unset( &gvalue );
537
538 return true;
539 }
540
541 // ---------------------------------------------------------
542 // wxDataViewColumn
543 // ---------------------------------------------------------
544
545
546 extern "C" {
547 static void wxGtkTreeCellDataFunc( GtkTreeViewColumn *column,
548 GtkCellRenderer *cell,
549 GtkTreeModel *model,
550 GtkTreeIter *iter,
551 gpointer data );
552 }
553
554
555 static void wxGtkTreeCellDataFunc( GtkTreeViewColumn *column,
556 GtkCellRenderer *renderer,
557 GtkTreeModel *model,
558 GtkTreeIter *iter,
559 gpointer data )
560 {
561 g_return_if_fail (GTK_IS_WX_LIST_STORE (model));
562 GtkWxListStore *list_store = (GtkWxListStore *) model;
563
564 wxDataViewCell *cell = (wxDataViewCell*) data;
565
566 wxVariant value = list_store->model->GetValue( (size_t) iter->user_data,
567 cell->GetOwner()->GetModelColumn() );
568
569 if (value.GetType() != cell->GetVariantType())
570 wxPrintf( wxT("Wrong type\n") );
571
572 cell->SetValue( value );
573 }
574
575 IMPLEMENT_ABSTRACT_CLASS(wxDataViewColumn, wxDataViewColumnBase)
576
577 wxDataViewColumn::wxDataViewColumn( const wxString &title, wxDataViewCell *cell,
578 size_t model_column, int flags ) :
579 wxDataViewColumnBase( title, cell, model_column, flags )
580 {
581 GtkCellRenderer *renderer = (GtkCellRenderer *) cell->GetGtkHandle();
582
583 GtkTreeViewColumn *column = gtk_tree_view_column_new();
584
585 gtk_tree_view_column_set_title( column, wxGTK_CONV(title) );
586
587 gtk_tree_view_column_pack_start( column, renderer, TRUE );
588
589 gtk_tree_view_column_set_cell_data_func( column, renderer,
590 wxGtkTreeCellDataFunc, (gpointer) cell, NULL );
591
592 m_column = (void*) column;
593 }
594
595 wxDataViewColumn::~wxDataViewColumn()
596 {
597 }
598
599 void wxDataViewColumn::SetTitle( const wxString &title )
600 {
601 wxDataViewColumnBase::SetTitle( title );
602
603 GtkTreeViewColumn *column = (GtkTreeViewColumn *)m_column;
604 gtk_tree_view_column_set_title( column, wxGTK_CONV(title) );
605 }
606
607 //-----------------------------------------------------------------------------
608 // wxDataViewCtrl
609 //-----------------------------------------------------------------------------
610
611 IMPLEMENT_DYNAMIC_CLASS(wxDataViewCtrl, wxDataViewCtrlBase)
612
613 wxDataViewCtrl::~wxDataViewCtrl()
614 {
615 }
616
617 void wxDataViewCtrl::Init()
618 {
619 }
620
621 bool wxDataViewCtrl::Create(wxWindow *parent, wxWindowID id,
622 const wxPoint& pos, const wxSize& size,
623 long style, const wxValidator& validator )
624 {
625 Init();
626
627 m_needParent = TRUE;
628 m_acceptsFocus = TRUE;
629
630 if (!PreCreation( parent, pos, size ) ||
631 !CreateBase( parent, id, pos, size, style, validator ))
632 {
633 wxFAIL_MSG( wxT("wxDataViewCtrl creation failed") );
634 return FALSE;
635 }
636
637 m_widget = gtk_tree_view_new();
638
639 m_parent->DoAddChild( this );
640
641 PostCreation(size);
642
643 return true;
644 }
645
646 bool wxDataViewCtrl::AssociateModel( wxDataViewListModel *model )
647 {
648 if (!wxDataViewCtrlBase::AssociateModel( model ))
649 return false;
650
651 GtkWxListStore *gtk_store = wxgtk_list_store_new();
652 gtk_store->model = model;
653
654 wxGtkDataViewListModelNotifier *notifier =
655 new wxGtkDataViewListModelNotifier( gtk_store, model );
656
657 model->SetNotifier( notifier );
658
659 gtk_tree_view_set_model( GTK_TREE_VIEW(m_widget), GTK_TREE_MODEL(gtk_store) );
660
661 return true;
662 }
663
664 bool wxDataViewCtrl::AppendColumn( wxDataViewColumn *col )
665 {
666 if (!wxDataViewCtrlBase::AppendColumn(col))
667 return false;
668
669 GtkTreeViewColumn *column = (GtkTreeViewColumn *)col->GetGtkHandle();
670
671 gtk_tree_view_append_column( GTK_TREE_VIEW(m_widget), column );
672
673 return true;
674 }
675
676
677 #endif // wxUSE_DATAVIEWCTRL
678