1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/dataview.cpp
3 // Purpose: wxDataViewCtrl GTK+2 implementation
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
10 // For compilers that support precompilation, includes "wx.h".
11 #include "wx/wxprec.h"
13 #if wxUSE_DATAVIEWCTRL
15 #include "wx/dataview.h"
17 #ifndef wxUSE_GENERICDATAVIEWCTRL
21 #include "wx/dcclient.h"
25 #include "wx/stockitem.h"
26 #include "wx/calctrl.h"
27 #include "wx/popupwin.h"
31 #include "wx/gtk/private.h"
32 #include "wx/gtk/win_gtk.h"
34 #include <gobject/gvaluecollector.h>
35 #include <gtk/gtktreemodel.h>
36 #include <gtk/gtktreesortable.h>
37 #include <gtk/gtktreednd.h>
39 #include <gdk/gdkkeysyms.h>
41 //-----------------------------------------------------------------------------
42 //-----------------------------------------------------------------------------
44 wxDataViewModel
*g_model
= NULL
;
46 class wxGtkTreeModelNode
;
49 typedef struct _GtkWxTreeModel GtkWxTreeModel
;
52 //-----------------------------------------------------------------------------
53 // wxDataViewCtrlInternal
54 //-----------------------------------------------------------------------------
56 class wxDataViewCtrlInternal
59 wxDataViewCtrlInternal( wxDataViewCtrl
*owner
, wxDataViewModel
*wx_model
, GtkWxTreeModel
*owner
);
60 ~wxDataViewCtrlInternal();
62 gboolean
get_iter( GtkTreeIter
*iter
, GtkTreePath
*path
);
63 GtkTreePath
*get_path( GtkTreeIter
*iter
);
64 GtkTreePath
*get_path_safe( GtkTreeIter
*iter
);
65 gboolean
iter_next( GtkTreeIter
*iter
);
66 gboolean
iter_children( GtkTreeIter
*iter
, GtkTreeIter
*parent
);
67 gboolean
iter_has_child( GtkTreeIter
*iter
);
68 gint
iter_n_children( GtkTreeIter
*iter
);
69 gboolean
iter_nth_child( GtkTreeIter
*iter
, GtkTreeIter
*parent
, gint n
);
70 gboolean
iter_parent( GtkTreeIter
*iter
, GtkTreeIter
*child
);
72 wxDataViewModel
* GetDataViewModel() { return m_wx_model
; }
73 wxDataViewCtrl
* GetOwner() { return m_owner
; }
74 GtkWxTreeModel
* GetGtkModel() { return m_gtk_model
; }
76 bool ItemAdded( const wxDataViewItem
&parent
, const wxDataViewItem
&item
);
77 bool ItemDeleted( const wxDataViewItem
&item
);
83 wxGtkTreeModelNode
*FindNode( const wxDataViewItem
&item
);
84 wxGtkTreeModelNode
*FindNode( GtkTreeIter
*iter
);
85 wxGtkTreeModelNode
*FindParentNode( const wxDataViewItem
&item
);
86 wxGtkTreeModelNode
*FindParentNode( GtkTreeIter
*iter
);
87 void BuildBranch( wxGtkTreeModelNode
*branch
);
90 wxGtkTreeModelNode
*m_root
;
91 wxDataViewModel
*m_wx_model
;
92 GtkWxTreeModel
*m_gtk_model
;
93 wxDataViewCtrl
*m_owner
;
97 //-----------------------------------------------------------------------------
99 //-----------------------------------------------------------------------------
101 int LINKAGEMODE
wxGtkTreeModelNodeCmp( void *id1
, void *id2
);
103 WX_DEFINE_ARRAY_PTR( wxGtkTreeModelNode
*, wxGtkTreeModelNodes
);
104 WX_DEFINE_SORTED_ARRAY( void* , wxGtkTreeModelChildren
);
106 class wxGtkTreeModelNode
109 wxGtkTreeModelNode( wxGtkTreeModelNode
* parent
, const wxDataViewItem
&item
,
110 wxDataViewCtrlInternal
*internal
)
114 m_internal
= internal
;
115 g_model
= internal
->GetDataViewModel();
116 m_children
= new wxGtkTreeModelChildren( wxGtkTreeModelNodeCmp
);
119 ~wxGtkTreeModelNode()
121 g_model
= m_internal
->GetDataViewModel();
122 size_t count
= m_children
->GetCount();
124 for (i
= 0; i
< count
; i
++)
126 wxGtkTreeModelNode
*child
= m_nodes
.Item( i
);
132 unsigned int AddNode( wxGtkTreeModelNode
* child
)
134 g_model
= m_internal
->GetDataViewModel();
135 m_nodes
.Add( child
);
136 return m_children
->Add( child
->GetItem().GetID() );
139 unsigned int AddLeave( void* id
)
141 g_model
= m_internal
->GetDataViewModel();
142 return m_children
->Add( id
);
145 void DeleteChild( void* id
)
147 g_model
= m_internal
->GetDataViewModel();
149 size_t count
= m_children
->GetCount();
150 for (pos
= 0; pos
< count
; pos
++)
152 if (m_children
->Item( pos
) == id
)
154 m_children
->RemoveAt( pos
);
158 count
= m_nodes
.GetCount();
159 for (pos
= 0; pos
< count
; pos
++)
161 wxGtkTreeModelNode
*node
= m_nodes
.Item( pos
);
162 if (node
->GetItem().GetID() == id
)
164 m_nodes
.RemoveAt( pos
);
172 wxGtkTreeModelNode
* GetParent()
174 wxGtkTreeModelNodes
&GetNodes()
176 wxGtkTreeModelChildren
&GetChildren()
177 { return *m_children
; }
179 unsigned int GetChildCount() { return m_children
->GetCount(); }
180 unsigned int GetNodesCount() { return m_nodes
.GetCount(); }
182 wxDataViewItem
&GetItem() { return m_item
; }
183 wxDataViewCtrlInternal
*GetInternal() { return m_internal
; }
188 wxGtkTreeModelNode
*m_parent
;
189 wxGtkTreeModelNodes m_nodes
;
190 wxGtkTreeModelChildren
*m_children
;
191 wxDataViewItem m_item
;
192 wxDataViewCtrlInternal
*m_internal
;
196 int LINKAGEMODE
wxGtkTreeModelNodeCmp( void* id1
, void* id2
)
198 return g_model
->Compare( id1
, id2
);
201 //-----------------------------------------------------------------------------
203 //-----------------------------------------------------------------------------
205 extern bool g_blockEventsOnDrag
;
207 //-----------------------------------------------------------------------------
208 // define new GTK+ class wxGtkTreeModel
209 //-----------------------------------------------------------------------------
213 #define GTK_TYPE_WX_TREE_MODEL (gtk_wx_tree_model_get_type ())
214 #define GTK_WX_TREE_MODEL(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_TREE_MODEL, GtkWxTreeModel))
215 #define GTK_WX_TREE_MODEL_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GTK_TYPE_WX_TREE_MODEL, GtkWxTreeModelClass))
216 #define GTK_IS_WX_TREE_MODEL(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_TREE_MODEL))
217 #define GTK_IS_WX_TREE_MODEL_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_TREE_MODEL))
218 #define GTK_WX_TREE_MODEL_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_WX_TREE_MODEL, GtkWxTreeModelClass))
220 GType
gtk_wx_tree_model_get_type (void);
222 typedef struct _GtkWxTreeModelClass GtkWxTreeModelClass
;
224 struct _GtkWxTreeModel
230 wxDataViewCtrlInternal
*internal
;
233 struct _GtkWxTreeModelClass
235 GObjectClass list_parent_class
;
238 static GtkWxTreeModel
*wxgtk_tree_model_new (void);
239 static void wxgtk_tree_model_init (GtkWxTreeModel
*tree_model
);
240 static void wxgtk_tree_model_class_init (GtkWxTreeModelClass
*klass
);
241 static void wxgtk_tree_model_tree_model_init (GtkTreeModelIface
*iface
);
242 static void wxgtk_tree_model_sortable_init (GtkTreeSortableIface
*iface
);
243 static void wxgtk_tree_model_finalize (GObject
*object
);
244 static GtkTreeModelFlags
wxgtk_tree_model_get_flags (GtkTreeModel
*tree_model
);
245 static gint
wxgtk_tree_model_get_n_columns (GtkTreeModel
*tree_model
);
246 static GType
wxgtk_tree_model_get_column_type (GtkTreeModel
*tree_model
,
248 static gboolean
wxgtk_tree_model_get_iter (GtkTreeModel
*tree_model
,
251 static GtkTreePath
*wxgtk_tree_model_get_path (GtkTreeModel
*tree_model
,
253 static GtkTreePath
*wxgtk_tree_model_get_path_safe (GtkTreeModel
*tree_model
,
255 static void wxgtk_tree_model_get_value (GtkTreeModel
*tree_model
,
259 static gboolean
wxgtk_tree_model_iter_next (GtkTreeModel
*tree_model
,
261 static gboolean
wxgtk_tree_model_iter_children (GtkTreeModel
*tree_model
,
263 GtkTreeIter
*parent
);
264 static gboolean
wxgtk_tree_model_iter_has_child (GtkTreeModel
*tree_model
,
266 static gint
wxgtk_tree_model_iter_n_children (GtkTreeModel
*tree_model
,
268 static gboolean
wxgtk_tree_model_iter_nth_child (GtkTreeModel
*tree_model
,
272 static gboolean
wxgtk_tree_model_iter_parent (GtkTreeModel
*tree_model
,
277 static gboolean
wxgtk_tree_model_get_sort_column_id (GtkTreeSortable
*sortable
,
278 gint
*sort_column_id
,
280 static void wxgtk_tree_model_set_sort_column_id (GtkTreeSortable
*sortable
,
283 static void wxgtk_tree_model_set_sort_func (GtkTreeSortable
*sortable
,
285 GtkTreeIterCompareFunc func
,
287 GtkDestroyNotify destroy
);
288 static void wxgtk_tree_model_set_default_sort_func (GtkTreeSortable
*sortable
,
289 GtkTreeIterCompareFunc func
,
291 GtkDestroyNotify destroy
);
292 static gboolean
wxgtk_tree_model_has_default_sort_func (GtkTreeSortable
*sortable
);
296 static GObjectClass
*list_parent_class
= NULL
;
299 gtk_wx_tree_model_get_type (void)
301 static GType tree_model_type
= 0;
303 if (!tree_model_type
)
305 const GTypeInfo tree_model_info
=
307 sizeof (GtkWxTreeModelClass
),
308 NULL
, /* base_init */
309 NULL
, /* base_finalize */
310 (GClassInitFunc
) wxgtk_tree_model_class_init
,
311 NULL
, /* class_finalize */
312 NULL
, /* class_data */
313 sizeof (GtkWxTreeModel
),
315 (GInstanceInitFunc
) wxgtk_tree_model_init
,
318 static const GInterfaceInfo tree_model_iface_info
=
320 (GInterfaceInitFunc
) wxgtk_tree_model_tree_model_init
,
325 static const GInterfaceInfo sortable_iface_info
=
327 (GInterfaceInitFunc
) wxgtk_tree_model_sortable_init
,
332 tree_model_type
= g_type_register_static (G_TYPE_OBJECT
, "GtkWxTreeModel",
333 &tree_model_info
, (GTypeFlags
)0 );
335 g_type_add_interface_static (tree_model_type
,
337 &tree_model_iface_info
);
338 g_type_add_interface_static (tree_model_type
,
339 GTK_TYPE_TREE_SORTABLE
,
340 &sortable_iface_info
);
343 return tree_model_type
;
346 static GtkWxTreeModel
*
347 wxgtk_tree_model_new(void)
349 GtkWxTreeModel
*retval
= (GtkWxTreeModel
*) g_object_new (GTK_TYPE_WX_TREE_MODEL
, NULL
);
354 wxgtk_tree_model_class_init (GtkWxTreeModelClass
*klass
)
356 list_parent_class
= (GObjectClass
*) g_type_class_peek_parent (klass
);
357 GObjectClass
*object_class
= (GObjectClass
*) klass
;
358 object_class
->finalize
= wxgtk_tree_model_finalize
;
362 wxgtk_tree_model_tree_model_init (GtkTreeModelIface
*iface
)
364 iface
->get_flags
= wxgtk_tree_model_get_flags
;
365 iface
->get_n_columns
= wxgtk_tree_model_get_n_columns
;
366 iface
->get_column_type
= wxgtk_tree_model_get_column_type
;
367 iface
->get_iter
= wxgtk_tree_model_get_iter
;
368 iface
->get_path
= wxgtk_tree_model_get_path
;
369 iface
->get_value
= wxgtk_tree_model_get_value
;
370 iface
->iter_next
= wxgtk_tree_model_iter_next
;
371 iface
->iter_children
= wxgtk_tree_model_iter_children
;
372 iface
->iter_has_child
= wxgtk_tree_model_iter_has_child
;
373 iface
->iter_n_children
= wxgtk_tree_model_iter_n_children
;
374 iface
->iter_nth_child
= wxgtk_tree_model_iter_nth_child
;
375 iface
->iter_parent
= wxgtk_tree_model_iter_parent
;
379 wxgtk_tree_model_sortable_init (GtkTreeSortableIface
*iface
)
381 iface
->get_sort_column_id
= wxgtk_tree_model_get_sort_column_id
;
382 iface
->set_sort_column_id
= wxgtk_tree_model_set_sort_column_id
;
383 iface
->set_sort_func
= wxgtk_tree_model_set_sort_func
;
384 iface
->set_default_sort_func
= wxgtk_tree_model_set_default_sort_func
;
385 iface
->has_default_sort_func
= wxgtk_tree_model_has_default_sort_func
;
389 wxgtk_tree_model_init (GtkWxTreeModel
*tree_model
)
391 tree_model
->internal
= NULL
;
392 tree_model
->stamp
= g_random_int();
396 wxgtk_tree_model_finalize (GObject
*object
)
399 (* list_parent_class
->finalize
) (object
);
404 //-----------------------------------------------------------------------------
405 // implement callbacks from wxGtkTreeModel class by letting
406 // them call the methods of wxWidgets' wxDataViewModel
407 //-----------------------------------------------------------------------------
409 static GtkTreeModelFlags
410 wxgtk_tree_model_get_flags (GtkTreeModel
*tree_model
)
412 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (tree_model
), (GtkTreeModelFlags
)0 );
414 return GTK_TREE_MODEL_ITERS_PERSIST
;
418 wxgtk_tree_model_get_n_columns (GtkTreeModel
*tree_model
)
420 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
421 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), 0);
423 return wxtree_model
->internal
->GetDataViewModel()->GetColumnCount();
427 wxgtk_tree_model_get_column_type (GtkTreeModel
*tree_model
,
430 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
431 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), G_TYPE_INVALID
);
433 GType gtype
= G_TYPE_INVALID
;
435 wxString wxtype
= wxtree_model
->internal
->GetDataViewModel()->GetColumnType( (unsigned int) index
);
437 if (wxtype
== wxT("string"))
438 gtype
= G_TYPE_STRING
;
441 wxFAIL_MSG( _T("non-string columns not supported yet") );
448 wxgtk_tree_model_get_iter (GtkTreeModel
*tree_model
,
452 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
453 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
454 g_return_val_if_fail (gtk_tree_path_get_depth (path
) > 0, FALSE
);
456 return wxtree_model
->internal
->get_iter( iter
, path
);
460 wxgtk_tree_model_get_path (GtkTreeModel
*tree_model
,
463 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
464 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), NULL
);
465 g_return_val_if_fail (iter
->stamp
== GTK_WX_TREE_MODEL (wxtree_model
)->stamp
, NULL
);
467 return wxtree_model
->internal
->get_path( iter
);
471 wxgtk_tree_model_get_path_safe (GtkTreeModel
*tree_model
,
474 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
475 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), NULL
);
476 g_return_val_if_fail (iter
->stamp
== GTK_WX_TREE_MODEL (wxtree_model
)->stamp
, NULL
);
478 return wxtree_model
->internal
->get_path_safe( iter
);
482 wxgtk_tree_model_get_value (GtkTreeModel
*tree_model
,
487 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
488 g_return_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
) );
490 wxDataViewModel
*model
= wxtree_model
->internal
->GetDataViewModel();
491 wxString mtype
= model
->GetColumnType( (unsigned int) column
);
492 if (mtype
== wxT("string"))
495 g_value_init( value
, G_TYPE_STRING
);
496 wxDataViewItem
item( (void*) iter
->user_data
);
497 model
->GetValue( variant
, item
, (unsigned int) column
);
499 g_value_set_string( value
, variant
.GetString().utf8_str() );
503 wxFAIL_MSG( _T("non-string columns not supported yet") );
508 wxgtk_tree_model_iter_next (GtkTreeModel
*tree_model
,
511 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
512 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
513 g_return_val_if_fail (wxtree_model
->stamp
== iter
->stamp
, FALSE
);
515 return wxtree_model
->internal
->iter_next( iter
);
519 wxgtk_tree_model_iter_children (GtkTreeModel
*tree_model
,
523 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
524 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
525 g_return_val_if_fail (wxtree_model
->stamp
== parent
->stamp
, FALSE
);
527 return wxtree_model
->internal
->iter_children( iter
, parent
);
531 wxgtk_tree_model_iter_has_child (GtkTreeModel
*tree_model
,
534 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
535 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
536 g_return_val_if_fail (wxtree_model
->stamp
== iter
->stamp
, FALSE
);
538 return wxtree_model
->internal
->iter_has_child( iter
);
542 wxgtk_tree_model_iter_n_children (GtkTreeModel
*tree_model
,
545 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
546 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
547 g_return_val_if_fail (wxtree_model
->stamp
== iter
->stamp
, 0);
549 return wxtree_model
->internal
->iter_n_children( iter
);
553 wxgtk_tree_model_iter_nth_child (GtkTreeModel
*tree_model
,
558 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
559 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
561 return wxtree_model
->internal
->iter_nth_child( iter
, parent
, n
);
565 wxgtk_tree_model_iter_parent (GtkTreeModel
*tree_model
,
569 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
570 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
571 g_return_val_if_fail (wxtree_model
->stamp
== child
->stamp
, FALSE
);
573 return wxtree_model
->internal
->iter_parent( iter
, child
);
577 gboolean
wxgtk_tree_model_get_sort_column_id (GtkTreeSortable
*sortable
,
578 gint
*sort_column_id
,
581 GtkWxTreeModel
*tree_model
= (GtkWxTreeModel
*) sortable
;
583 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (sortable
), FALSE
);
586 *sort_column_id
= tree_model
->internal
->GetDataViewModel()->GetSortingColumn();
590 bool ascending
= tree_model
->internal
->GetDataViewModel()->GetSortOrderAscending();
592 *order
= GTK_SORT_ASCENDING
;
594 *order
= GTK_SORT_DESCENDING
;
600 void wxgtk_tree_model_set_sort_column_id (GtkTreeSortable
*sortable
,
604 GtkWxTreeModel
*tree_model
= (GtkWxTreeModel
*) sortable
;
605 g_return_if_fail (GTK_IS_WX_TREE_MODEL (sortable
) );
607 bool ascending
= TRUE
;
608 if (order
!= GTK_SORT_ASCENDING
)
611 if ((sort_column_id
== (gint
) tree_model
->internal
->GetDataViewModel()->GetSortingColumn()) &&
612 (ascending
== tree_model
->internal
->GetDataViewModel()->GetSortOrderAscending()))
615 tree_model
->internal
->GetDataViewModel()->SetSortingColumn( sort_column_id
);
617 tree_model
->internal
->GetDataViewModel()->SetSortOrderAscending( ascending
);
619 gtk_tree_sortable_sort_column_changed (sortable
);
621 tree_model
->internal
->GetDataViewModel()->Resort();
624 void wxgtk_tree_model_set_sort_func (GtkTreeSortable
*sortable
,
626 GtkTreeIterCompareFunc func
,
628 GtkDestroyNotify destroy
)
630 g_return_if_fail (GTK_IS_WX_TREE_MODEL (sortable
) );
631 g_return_if_fail (func
!= NULL
);
634 void wxgtk_tree_model_set_default_sort_func (GtkTreeSortable
*sortable
,
635 GtkTreeIterCompareFunc func
,
637 GtkDestroyNotify destroy
)
639 g_return_if_fail (GTK_IS_WX_TREE_MODEL (sortable
) );
640 g_return_if_fail (func
!= NULL
);
642 wxPrintf( "wxgtk_tree_model_set_default_sort_func\n" );
645 gboolean
wxgtk_tree_model_has_default_sort_func (GtkTreeSortable
*sortable
)
650 //-----------------------------------------------------------------------------
651 // define new GTK+ class wxGtkRendererRenderer
652 //-----------------------------------------------------------------------------
656 #define GTK_TYPE_WX_CELL_RENDERER (gtk_wx_cell_renderer_get_type ())
657 #define GTK_WX_CELL_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRenderer))
658 #define GTK_WX_CELL_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRendererClass))
659 #define GTK_IS_WX_CELL_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_CELL_RENDERER))
660 #define GTK_IS_WX_CELL_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_CELL_RENDERER))
661 #define GTK_WX_CELL_RENDERER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRendererClass))
663 GType
gtk_wx_cell_renderer_get_type (void);
665 typedef struct _GtkWxCellRenderer GtkWxCellRenderer
;
666 typedef struct _GtkWxCellRendererClass GtkWxCellRendererClass
;
668 struct _GtkWxCellRenderer
670 GtkCellRenderer parent
;
673 wxDataViewCustomRenderer
*cell
;
677 struct _GtkWxCellRendererClass
679 GtkCellRendererClass cell_parent_class
;
683 static GtkCellRenderer
*gtk_wx_cell_renderer_new (void);
684 static void gtk_wx_cell_renderer_init (
685 GtkWxCellRenderer
*cell
);
686 static void gtk_wx_cell_renderer_class_init(
687 GtkWxCellRendererClass
*klass
);
688 static void gtk_wx_cell_renderer_finalize (
690 static void gtk_wx_cell_renderer_get_size (
691 GtkCellRenderer
*cell
,
693 GdkRectangle
*rectangle
,
698 static void gtk_wx_cell_renderer_render (
699 GtkCellRenderer
*cell
,
702 GdkRectangle
*background_area
,
703 GdkRectangle
*cell_area
,
704 GdkRectangle
*expose_area
,
705 GtkCellRendererState flags
);
706 static gboolean
gtk_wx_cell_renderer_activate(
707 GtkCellRenderer
*cell
,
711 GdkRectangle
*background_area
,
712 GdkRectangle
*cell_area
,
713 GtkCellRendererState flags
);
714 static GtkCellEditable
*gtk_wx_cell_renderer_start_editing(
715 GtkCellRenderer
*cell
,
719 GdkRectangle
*background_area
,
720 GdkRectangle
*cell_area
,
721 GtkCellRendererState flags
);
724 static GObjectClass
*cell_parent_class
= NULL
;
729 gtk_wx_cell_renderer_get_type (void)
731 static GType cell_wx_type
= 0;
735 const GTypeInfo cell_wx_info
=
737 sizeof (GtkWxCellRendererClass
),
738 NULL
, /* base_init */
739 NULL
, /* base_finalize */
740 (GClassInitFunc
) gtk_wx_cell_renderer_class_init
,
741 NULL
, /* class_finalize */
742 NULL
, /* class_data */
743 sizeof (GtkWxCellRenderer
),
745 (GInstanceInitFunc
) gtk_wx_cell_renderer_init
,
748 cell_wx_type
= g_type_register_static( GTK_TYPE_CELL_RENDERER
,
749 "GtkWxCellRenderer", &cell_wx_info
, (GTypeFlags
)0 );
756 gtk_wx_cell_renderer_init (GtkWxCellRenderer
*cell
)
759 cell
->last_click
= 0;
763 gtk_wx_cell_renderer_class_init (GtkWxCellRendererClass
*klass
)
765 GObjectClass
*object_class
= G_OBJECT_CLASS (klass
);
766 GtkCellRendererClass
*cell_class
= GTK_CELL_RENDERER_CLASS (klass
);
768 cell_parent_class
= (GObjectClass
*) g_type_class_peek_parent (klass
);
770 object_class
->finalize
= gtk_wx_cell_renderer_finalize
;
772 cell_class
->get_size
= gtk_wx_cell_renderer_get_size
;
773 cell_class
->render
= gtk_wx_cell_renderer_render
;
774 cell_class
->activate
= gtk_wx_cell_renderer_activate
;
775 cell_class
->start_editing
= gtk_wx_cell_renderer_start_editing
;
779 gtk_wx_cell_renderer_finalize (GObject
*object
)
782 (* G_OBJECT_CLASS (cell_parent_class
)->finalize
) (object
);
786 gtk_wx_cell_renderer_new (void)
788 return (GtkCellRenderer
*) g_object_new (GTK_TYPE_WX_CELL_RENDERER
, NULL
);
793 static GtkCellEditable
*gtk_wx_cell_renderer_start_editing(
794 GtkCellRenderer
*renderer
,
798 GdkRectangle
*background_area
,
799 GdkRectangle
*cell_area
,
800 GtkCellRendererState flags
)
802 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
803 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
804 if (!cell
->HasEditorCtrl())
808 gtk_wx_cell_renderer_get_size (renderer
, widget
, cell_area
,
814 rect
.x
+= cell_area
->x
;
815 rect
.y
+= cell_area
->y
;
816 // rect.width -= renderer->xpad * 2;
817 // rect.height -= renderer->ypad * 2;
819 // wxRect renderrect( rect.x, rect.y, rect.width, rect.height );
820 wxRect
renderrect( cell_area
->x
, cell_area
->y
, cell_area
->width
, cell_area
->height
);
822 GtkTreePath
*treepath
= gtk_tree_path_new_from_string( path
);
824 cell
->GetOwner()->GetOwner()->GtkGetInternal()->get_iter( &iter
, treepath
);
825 wxDataViewItem
item( (void*) iter
.user_data
);
826 gtk_tree_path_free( treepath
);
828 cell
->StartEditing( item
, renderrect
);
834 gtk_wx_cell_renderer_get_size (GtkCellRenderer
*renderer
,
836 GdkRectangle
*cell_area
,
842 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
843 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
845 wxSize size
= cell
->GetSize();
847 gint calc_width
= (gint
) renderer
->xpad
* 2 + size
.x
;
848 gint calc_height
= (gint
) renderer
->ypad
* 2 + size
.y
;
855 if (cell_area
&& size
.x
> 0 && size
.y
> 0)
859 *x_offset
= (gint
)((renderer
->xalign
*
860 (cell_area
->width
- calc_width
- 2 * renderer
->xpad
)));
861 *x_offset
= MAX (*x_offset
, 0) + renderer
->xpad
;
865 *y_offset
= (gint
)((renderer
->yalign
*
866 (cell_area
->height
- calc_height
- 2 * renderer
->ypad
)));
867 *y_offset
= MAX (*y_offset
, 0) + renderer
->ypad
;
875 *height
= calc_height
;
879 gtk_wx_cell_renderer_render (GtkCellRenderer
*renderer
,
882 GdkRectangle
*background_area
,
883 GdkRectangle
*cell_area
,
884 GdkRectangle
*expose_area
,
885 GtkCellRendererState flags
)
888 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
889 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
892 gtk_wx_cell_renderer_get_size (renderer
, widget
, cell_area
,
898 rect
.x
+= cell_area
->x
;
899 rect
.y
+= cell_area
->y
;
900 rect
.width
-= renderer
->xpad
* 2;
901 rect
.height
-= renderer
->ypad
* 2;
904 if (gdk_rectangle_intersect (expose_area
, &rect
, &dummy
))
906 wxRect
renderrect( rect
.x
, rect
.y
, rect
.width
, rect
.height
);
907 wxWindowDC
* dc
= (wxWindowDC
*) cell
->GetDC();
908 if (dc
->m_window
== NULL
)
910 dc
->m_window
= window
;
915 if (flags
& GTK_CELL_RENDERER_SELECTED
)
916 state
|= wxDATAVIEW_CELL_SELECTED
;
917 if (flags
& GTK_CELL_RENDERER_PRELIT
)
918 state
|= wxDATAVIEW_CELL_PRELIT
;
919 if (flags
& GTK_CELL_RENDERER_INSENSITIVE
)
920 state
|= wxDATAVIEW_CELL_INSENSITIVE
;
921 if (flags
& GTK_CELL_RENDERER_INSENSITIVE
)
922 state
|= wxDATAVIEW_CELL_INSENSITIVE
;
923 if (flags
& GTK_CELL_RENDERER_FOCUSED
)
924 state
|= wxDATAVIEW_CELL_FOCUSED
;
925 cell
->Render( renderrect
, dc
, state
);
930 gtk_wx_cell_renderer_activate(
931 GtkCellRenderer
*renderer
,
935 GdkRectangle
*background_area
,
936 GdkRectangle
*cell_area
,
937 GtkCellRendererState flags
)
939 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
940 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
943 gtk_wx_cell_renderer_get_size (renderer
, widget
, cell_area
,
949 rect
.x
+= cell_area
->x
;
950 rect
.y
+= cell_area
->y
;
951 rect
.width
-= renderer
->xpad
* 2;
952 rect
.height
-= renderer
->ypad
* 2;
954 wxRect
renderrect( rect
.x
, rect
.y
, rect
.width
, rect
.height
);
956 wxDataViewModel
*model
= cell
->GetOwner()->GetOwner()->GetModel();
958 GtkTreePath
*treepath
= gtk_tree_path_new_from_string( path
);
961 gtk_tree_path_free( treepath
);
963 unsigned int model_col
= cell
->GetOwner()->GetModelColumn();
969 // activated by <ENTER>
970 if (cell
->Activate( renderrect
, model
, item
, model_col
))
975 else if (event
->type
== GDK_BUTTON_PRESS
)
977 GdkEventButton
*button_event
= (GdkEventButton
*) event
;
978 wxPoint
pt( ((int) button_event
->x
) - renderrect
.x
,
979 ((int) button_event
->y
) - renderrect
.y
);
982 if (button_event
->button
== 1)
984 if (cell
->LeftClick( pt
, renderrect
, model
, item
, model_col
))
986 // TODO: query system double-click time
987 if (button_event
->time
- wxrenderer
->last_click
< 400)
988 if (cell
->Activate( renderrect
, model
, item
, model_col
))
991 if (button_event
->button
== 3)
993 if (cell
->RightClick( pt
, renderrect
, model
, item
, model_col
))
997 wxrenderer
->last_click
= button_event
->time
;
1005 // ---------------------------------------------------------
1006 // wxGtkDataViewModelNotifier
1007 // ---------------------------------------------------------
1009 class wxGtkDataViewModelNotifier
: public wxDataViewModelNotifier
1012 wxGtkDataViewModelNotifier( GtkWxTreeModel
*wxgtk_model
,
1013 wxDataViewModel
*wx_model
,
1014 wxDataViewCtrl
*ctrl
);
1015 ~wxGtkDataViewModelNotifier();
1017 virtual bool ItemAdded( const wxDataViewItem
&parent
, const wxDataViewItem
&item
);
1018 virtual bool ItemDeleted( const wxDataViewItem
&item
);
1019 virtual bool ItemChanged( const wxDataViewItem
&item
);
1020 virtual bool ValueChanged( const wxDataViewItem
&item
, unsigned int col
);
1021 virtual bool Cleared();
1022 virtual void Resort();
1024 GtkWxTreeModel
*m_wxgtk_model
;
1025 wxDataViewModel
*m_wx_model
;
1026 wxDataViewCtrl
*m_owner
;
1029 // ---------------------------------------------------------
1030 // wxGtkDataViewListModelNotifier
1031 // ---------------------------------------------------------
1033 wxGtkDataViewModelNotifier::wxGtkDataViewModelNotifier(
1034 GtkWxTreeModel
* wxgtk_model
, wxDataViewModel
*wx_model
,
1035 wxDataViewCtrl
*ctrl
)
1037 m_wxgtk_model
= wxgtk_model
;
1038 m_wx_model
= wx_model
;
1042 wxGtkDataViewModelNotifier::~wxGtkDataViewModelNotifier()
1045 m_wxgtk_model
= NULL
;
1048 bool wxGtkDataViewModelNotifier::ItemAdded( const wxDataViewItem
&parent
, const wxDataViewItem
&item
)
1050 m_owner
->GtkGetInternal()->ItemAdded( parent
, item
);
1053 iter
.stamp
= m_wxgtk_model
->stamp
;
1054 iter
.user_data
= (gpointer
) item
.GetID();
1056 GtkTreePath
*path
= wxgtk_tree_model_get_path(
1057 GTK_TREE_MODEL(m_wxgtk_model
), &iter
);
1058 gtk_tree_model_row_inserted(
1059 GTK_TREE_MODEL(m_wxgtk_model
), path
, &iter
);
1060 gtk_tree_path_free (path
);
1065 bool wxGtkDataViewModelNotifier::ItemDeleted( const wxDataViewItem
&item
)
1068 iter
.stamp
= m_wxgtk_model
->stamp
;
1069 iter
.user_data
= (gpointer
) item
.GetID();
1071 GtkTreePath
*path
= wxgtk_tree_model_get_path_safe(
1072 GTK_TREE_MODEL(m_wxgtk_model
), &iter
);
1073 gtk_tree_model_row_deleted(
1074 GTK_TREE_MODEL(m_wxgtk_model
), path
);
1075 gtk_tree_path_free (path
);
1077 m_owner
->GtkGetInternal()->ItemDeleted( item
);
1082 void wxGtkDataViewModelNotifier::Resort()
1084 m_owner
->GtkGetInternal()->Resort();
1087 bool wxGtkDataViewModelNotifier::ItemChanged( const wxDataViewItem
&item
)
1090 iter
.stamp
= m_wxgtk_model
->stamp
;
1091 iter
.user_data
= (gpointer
) item
.GetID();
1093 GtkTreePath
*path
= wxgtk_tree_model_get_path(
1094 GTK_TREE_MODEL(m_wxgtk_model
), &iter
);
1095 gtk_tree_model_row_changed(
1096 GTK_TREE_MODEL(m_wxgtk_model
), path
, &iter
);
1097 gtk_tree_path_free (path
);
1102 bool wxGtkDataViewModelNotifier::ValueChanged( const wxDataViewItem
&item
, unsigned int model_col
)
1104 // This adds GTK+'s missing MVC logic for ValueChanged
1106 for (index
= 0; index
< m_owner
->GetColumnCount(); index
++)
1108 wxDataViewColumn
*column
= m_owner
->GetColumn( index
);
1109 if (column
->GetModelColumn() == model_col
)
1111 GtkTreeView
*widget
= GTK_TREE_VIEW(m_owner
->m_treeview
);
1112 GtkTreeViewColumn
*gcolumn
= GTK_TREE_VIEW_COLUMN(column
->GetGtkHandle());
1116 iter
.stamp
= m_wxgtk_model
->stamp
;
1117 iter
.user_data
= (gpointer
) item
.GetID();
1118 GtkTreePath
*path
= wxgtk_tree_model_get_path(
1119 GTK_TREE_MODEL(m_wxgtk_model
), &iter
);
1120 GdkRectangle cell_area
;
1121 gtk_tree_view_get_cell_area( widget
, path
, gcolumn
, &cell_area
);
1122 gtk_tree_path_free( path
);
1124 GtkAdjustment
* hadjust
= gtk_tree_view_get_hadjustment( widget
);
1125 double d
= gtk_adjustment_get_value( hadjust
);
1126 int xdiff
= (int) d
;
1128 int ydiff
= gcolumn
->button
->allocation
.height
;
1130 gtk_widget_queue_draw_area( GTK_WIDGET(widget
),
1131 cell_area
.x
- xdiff
, ydiff
+ cell_area
.y
, cell_area
.width
, cell_area
.height
);
1138 bool wxGtkDataViewModelNotifier::Cleared()
1143 // ---------------------------------------------------------
1144 // wxDataViewRenderer
1145 // ---------------------------------------------------------
1147 IMPLEMENT_ABSTRACT_CLASS(wxDataViewRenderer
, wxDataViewRendererBase
)
1149 wxDataViewRenderer::wxDataViewRenderer( const wxString
&varianttype
, wxDataViewCellMode mode
,
1151 wxDataViewRendererBase( varianttype
, mode
, align
)
1155 // NOTE: SetMode() and SetAlignment() needs to be called in the renderer's ctor,
1156 // after the m_renderer pointer has been initialized
1159 void wxDataViewRenderer::SetMode( wxDataViewCellMode mode
)
1161 GtkCellRendererMode gtkMode
;
1164 case wxDATAVIEW_CELL_INERT
:
1165 gtkMode
= GTK_CELL_RENDERER_MODE_INERT
;
1167 case wxDATAVIEW_CELL_ACTIVATABLE
:
1168 gtkMode
= GTK_CELL_RENDERER_MODE_ACTIVATABLE
;
1170 case wxDATAVIEW_CELL_EDITABLE
:
1171 gtkMode
= GTK_CELL_RENDERER_MODE_EDITABLE
;
1175 GValue gvalue
= { 0, };
1176 g_value_init( &gvalue
, gtk_cell_renderer_mode_get_type() );
1177 g_value_set_enum( &gvalue
, gtkMode
);
1178 g_object_set_property( G_OBJECT(m_renderer
), "mode", &gvalue
);
1179 g_value_unset( &gvalue
);
1182 wxDataViewCellMode
wxDataViewRenderer::GetMode() const
1184 wxDataViewCellMode ret
;
1187 g_object_get( G_OBJECT(m_renderer
), "mode", &gvalue
, NULL
);
1189 switch (g_value_get_enum(&gvalue
))
1191 case GTK_CELL_RENDERER_MODE_INERT
:
1192 ret
= wxDATAVIEW_CELL_INERT
;
1194 case GTK_CELL_RENDERER_MODE_ACTIVATABLE
:
1195 ret
= wxDATAVIEW_CELL_ACTIVATABLE
;
1197 case GTK_CELL_RENDERER_MODE_EDITABLE
:
1198 ret
= wxDATAVIEW_CELL_EDITABLE
;
1202 g_value_unset( &gvalue
);
1207 void wxDataViewRenderer::SetAlignment( int align
)
1209 // horizontal alignment:
1211 gfloat xalign
= 0.0;
1212 if (align
& wxALIGN_RIGHT
)
1214 else if (align
& wxALIGN_CENTER_HORIZONTAL
)
1217 GValue gvalue
= { 0, };
1218 g_value_init( &gvalue
, G_TYPE_FLOAT
);
1219 g_value_set_float( &gvalue
, xalign
);
1220 g_object_set_property( G_OBJECT(m_renderer
), "xalign", &gvalue
);
1221 g_value_unset( &gvalue
);
1223 // vertical alignment:
1225 gfloat yalign
= 0.0;
1226 if (align
& wxALIGN_BOTTOM
)
1228 else if (align
& wxALIGN_CENTER_VERTICAL
)
1231 GValue gvalue2
= { 0, };
1232 g_value_init( &gvalue2
, G_TYPE_FLOAT
);
1233 g_value_set_float( &gvalue2
, yalign
);
1234 g_object_set_property( G_OBJECT(m_renderer
), "yalign", &gvalue2
);
1235 g_value_unset( &gvalue2
);
1238 int wxDataViewRenderer::GetAlignment() const
1243 // horizontal alignment:
1245 g_object_get( G_OBJECT(m_renderer
), "xalign", &gvalue
, NULL
);
1246 float xalign
= g_value_get_float( &gvalue
);
1248 ret
|= wxALIGN_LEFT
;
1249 else if (xalign
== 0.5)
1250 ret
|= wxALIGN_CENTER_HORIZONTAL
;
1252 ret
|= wxALIGN_RIGHT
;
1253 g_value_unset( &gvalue
);
1256 // vertical alignment:
1258 g_object_get( G_OBJECT(m_renderer
), "yalign", &gvalue
, NULL
);
1259 float yalign
= g_value_get_float( &gvalue
);
1262 else if (yalign
== 0.5)
1263 ret
|= wxALIGN_CENTER_VERTICAL
;
1265 ret
|= wxALIGN_BOTTOM
;
1266 g_value_unset( &gvalue
);
1273 // ---------------------------------------------------------
1274 // wxDataViewTextRenderer
1275 // ---------------------------------------------------------
1278 static void wxGtkTextRendererEditedCallback( GtkCellRendererText
*renderer
,
1279 gchar
*arg1
, gchar
*arg2
, gpointer user_data
);
1282 static void wxGtkTextRendererEditedCallback( GtkCellRendererText
*renderer
,
1283 gchar
*arg1
, gchar
*arg2
, gpointer user_data
)
1285 wxDataViewTextRenderer
*cell
= (wxDataViewTextRenderer
*) user_data
;
1287 wxString tmp
= wxGTK_CONV_BACK_FONT(arg2
, cell
->GetOwner()->GetOwner()->GetFont());
1288 wxVariant value
= tmp
;
1289 if (!cell
->Validate( value
))
1292 wxDataViewModel
*model
= cell
->GetOwner()->GetOwner()->GetModel();
1294 GtkTreePath
*path
= gtk_tree_path_new_from_string( arg1
);
1296 cell
->GetOwner()->GetOwner()->GtkGetInternal()->get_iter( &iter
, path
);
1297 wxDataViewItem
item( (void*) iter
.user_data
);;
1298 gtk_tree_path_free( path
);
1300 unsigned int model_col
= cell
->GetOwner()->GetModelColumn();
1302 model
->SetValue( value
, item
, model_col
);
1303 model
->ValueChanged( item
, model_col
);
1306 IMPLEMENT_CLASS(wxDataViewTextRenderer
, wxDataViewRenderer
)
1308 wxDataViewTextRenderer::wxDataViewTextRenderer( const wxString
&varianttype
, wxDataViewCellMode mode
,
1310 wxDataViewRenderer( varianttype
, mode
, align
)
1312 m_renderer
= (GtkCellRenderer
*) gtk_cell_renderer_text_new();
1314 if (mode
& wxDATAVIEW_CELL_EDITABLE
)
1316 GValue gvalue
= { 0, };
1317 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1318 g_value_set_boolean( &gvalue
, true );
1319 g_object_set_property( G_OBJECT(m_renderer
), "editable", &gvalue
);
1320 g_value_unset( &gvalue
);
1322 g_signal_connect_after( m_renderer
, "edited", G_CALLBACK(wxGtkTextRendererEditedCallback
), this );
1326 SetAlignment(align
);
1329 bool wxDataViewTextRenderer::SetValue( const wxVariant
&value
)
1331 wxString tmp
= value
;
1333 GValue gvalue
= { 0, };
1334 g_value_init( &gvalue
, G_TYPE_STRING
);
1335 g_value_set_string( &gvalue
, wxGTK_CONV_FONT( tmp
, GetOwner()->GetOwner()->GetFont() ) );
1336 g_object_set_property( G_OBJECT(m_renderer
), "text", &gvalue
);
1337 g_value_unset( &gvalue
);
1342 bool wxDataViewTextRenderer::GetValue( wxVariant
&value
) const
1344 GValue gvalue
= { 0, };
1345 g_value_init( &gvalue
, G_TYPE_STRING
);
1346 g_object_get_property( G_OBJECT(m_renderer
), "text", &gvalue
);
1347 wxString tmp
= wxGTK_CONV_BACK_FONT( g_value_get_string( &gvalue
),
1348 wx_const_cast(wxDataViewTextRenderer
*, this)->GetOwner()->GetOwner()->GetFont() );
1349 g_value_unset( &gvalue
);
1356 void wxDataViewTextRenderer::SetAlignment( int align
)
1358 wxDataViewRenderer::SetAlignment(align
);
1360 if (gtk_check_version(2,10,0))
1363 // horizontal alignment:
1364 PangoAlignment pangoAlign
= PANGO_ALIGN_LEFT
;
1365 if (align
& wxALIGN_RIGHT
)
1366 pangoAlign
= PANGO_ALIGN_RIGHT
;
1367 else if (align
& wxALIGN_CENTER_HORIZONTAL
)
1368 pangoAlign
= PANGO_ALIGN_CENTER
;
1370 GValue gvalue
= { 0, };
1371 g_value_init( &gvalue
, gtk_cell_renderer_mode_get_type() );
1372 g_value_set_enum( &gvalue
, pangoAlign
);
1373 g_object_set_property( G_OBJECT(m_renderer
), "alignment", &gvalue
);
1374 g_value_unset( &gvalue
);
1377 // ---------------------------------------------------------
1378 // wxDataViewBitmapRenderer
1379 // ---------------------------------------------------------
1381 IMPLEMENT_CLASS(wxDataViewBitmapRenderer
, wxDataViewRenderer
)
1383 wxDataViewBitmapRenderer::wxDataViewBitmapRenderer( const wxString
&varianttype
, wxDataViewCellMode mode
,
1385 wxDataViewRenderer( varianttype
, mode
, align
)
1387 m_renderer
= (GtkCellRenderer
*) gtk_cell_renderer_pixbuf_new();
1390 SetAlignment(align
);
1393 bool wxDataViewBitmapRenderer::SetValue( const wxVariant
&value
)
1395 if (value
.GetType() == wxT("wxBitmap"))
1400 // This may create a Pixbuf representation in the
1401 // wxBitmap object (and it will stay there)
1402 GdkPixbuf
*pixbuf
= bitmap
.GetPixbuf();
1404 GValue gvalue
= { 0, };
1405 g_value_init( &gvalue
, G_TYPE_OBJECT
);
1406 g_value_set_object( &gvalue
, pixbuf
);
1407 g_object_set_property( G_OBJECT(m_renderer
), "pixbuf", &gvalue
);
1408 g_value_unset( &gvalue
);
1413 if (value
.GetType() == wxT("wxIcon"))
1418 // This may create a Pixbuf representation in the
1419 // wxBitmap object (and it will stay there)
1420 GdkPixbuf
*pixbuf
= bitmap
.GetPixbuf();
1422 GValue gvalue
= { 0, };
1423 g_value_init( &gvalue
, G_TYPE_OBJECT
);
1424 g_value_set_object( &gvalue
, pixbuf
);
1425 g_object_set_property( G_OBJECT(m_renderer
), "pixbuf", &gvalue
);
1426 g_value_unset( &gvalue
);
1434 bool wxDataViewBitmapRenderer::GetValue( wxVariant
&value
) const
1439 // ---------------------------------------------------------
1440 // wxDataViewToggleRenderer
1441 // ---------------------------------------------------------
1444 static void wxGtkToggleRendererToggledCallback( GtkCellRendererToggle
*renderer
,
1445 gchar
*path
, gpointer user_data
);
1448 static void wxGtkToggleRendererToggledCallback( GtkCellRendererToggle
*renderer
,
1449 gchar
*path
, gpointer user_data
)
1451 wxDataViewToggleRenderer
*cell
= (wxDataViewToggleRenderer
*) user_data
;
1454 GValue gvalue
= { 0, };
1455 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1456 g_object_get_property( G_OBJECT(renderer
), "active", &gvalue
);
1457 bool tmp
= g_value_get_boolean( &gvalue
);
1458 g_value_unset( &gvalue
);
1462 wxVariant value
= tmp
;
1463 if (!cell
->Validate( value
))
1466 wxDataViewModel
*model
= cell
->GetOwner()->GetOwner()->GetModel();
1468 GtkTreePath
*gtk_path
= gtk_tree_path_new_from_string( path
);
1470 cell
->GetOwner()->GetOwner()->GtkGetInternal()->get_iter( &iter
, gtk_path
);
1471 wxDataViewItem
item( (void*) iter
.user_data
);;
1472 gtk_tree_path_free( gtk_path
);
1474 unsigned int model_col
= cell
->GetOwner()->GetModelColumn();
1476 model
->SetValue( value
, item
, model_col
);
1477 model
->ValueChanged( item
, model_col
);
1480 IMPLEMENT_CLASS(wxDataViewToggleRenderer
, wxDataViewRenderer
)
1482 wxDataViewToggleRenderer::wxDataViewToggleRenderer( const wxString
&varianttype
,
1483 wxDataViewCellMode mode
, int align
) :
1484 wxDataViewRenderer( varianttype
, mode
, align
)
1486 m_renderer
= (GtkCellRenderer
*) gtk_cell_renderer_toggle_new();
1488 if (mode
& wxDATAVIEW_CELL_ACTIVATABLE
)
1490 g_signal_connect_after( m_renderer
, "toggled",
1491 G_CALLBACK(wxGtkToggleRendererToggledCallback
), this );
1495 GValue gvalue
= { 0, };
1496 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1497 g_value_set_boolean( &gvalue
, false );
1498 g_object_set_property( G_OBJECT(m_renderer
), "activatable", &gvalue
);
1499 g_value_unset( &gvalue
);
1503 SetAlignment(align
);
1506 bool wxDataViewToggleRenderer::SetValue( const wxVariant
&value
)
1510 GValue gvalue
= { 0, };
1511 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1512 g_value_set_boolean( &gvalue
, tmp
);
1513 g_object_set_property( G_OBJECT(m_renderer
), "active", &gvalue
);
1514 g_value_unset( &gvalue
);
1519 bool wxDataViewToggleRenderer::GetValue( wxVariant
&value
) const
1521 GValue gvalue
= { 0, };
1522 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1523 g_object_get_property( G_OBJECT(m_renderer
), "active", &gvalue
);
1524 bool tmp
= g_value_get_boolean( &gvalue
);
1525 g_value_unset( &gvalue
);
1532 // ---------------------------------------------------------
1533 // wxDataViewCustomRenderer
1534 // ---------------------------------------------------------
1536 class wxDataViewCtrlDC
: public wxWindowDC
1539 wxDataViewCtrlDC( wxDataViewCtrl
*window
)
1541 GtkWidget
*widget
= window
->m_treeview
;
1545 m_context
= window
->GtkGetPangoDefaultContext();
1546 m_layout
= pango_layout_new( m_context
);
1547 m_fontdesc
= pango_font_description_copy( widget
->style
->font_desc
);
1549 m_cmap
= gtk_widget_get_colormap( widget
? widget
: window
->m_widget
);
1551 // Set m_window later
1553 // m_owner = window;
1557 // ---------------------------------------------------------
1558 // wxDataViewCustomRenderer
1559 // ---------------------------------------------------------
1561 IMPLEMENT_CLASS(wxDataViewCustomRenderer
, wxDataViewRenderer
)
1563 wxDataViewCustomRenderer::wxDataViewCustomRenderer( const wxString
&varianttype
,
1564 wxDataViewCellMode mode
, int align
,
1566 wxDataViewRenderer( varianttype
, mode
, align
)
1576 bool wxDataViewCustomRenderer::Init(wxDataViewCellMode mode
, int align
)
1578 GtkWxCellRenderer
*renderer
= (GtkWxCellRenderer
*) gtk_wx_cell_renderer_new();
1579 renderer
->cell
= this;
1581 m_renderer
= (GtkCellRenderer
*) renderer
;
1584 SetAlignment(align
);
1589 wxDataViewCustomRenderer::~wxDataViewCustomRenderer()
1595 wxDC
*wxDataViewCustomRenderer::GetDC()
1599 if (GetOwner() == NULL
)
1601 if (GetOwner()->GetOwner() == NULL
)
1603 m_dc
= new wxDataViewCtrlDC( GetOwner()->GetOwner() );
1609 // ---------------------------------------------------------
1610 // wxDataViewProgressRenderer
1611 // ---------------------------------------------------------
1613 IMPLEMENT_CLASS(wxDataViewProgressRenderer
, wxDataViewCustomRenderer
)
1615 wxDataViewProgressRenderer::wxDataViewProgressRenderer( const wxString
&label
,
1616 const wxString
&varianttype
, wxDataViewCellMode mode
, int align
) :
1617 wxDataViewCustomRenderer( varianttype
, mode
, align
, true )
1623 if (!gtk_check_version(2,6,0))
1625 m_renderer
= (GtkCellRenderer
*) gtk_cell_renderer_progress_new();
1627 GValue gvalue
= { 0, };
1628 g_value_init( &gvalue
, G_TYPE_STRING
);
1630 // FIXME: font encoding support
1631 g_value_set_string( &gvalue
, wxGTK_CONV_SYS(m_label
) );
1632 g_object_set_property( G_OBJECT(m_renderer
), "text", &gvalue
);
1633 g_value_unset( &gvalue
);
1636 SetAlignment(align
);
1641 // Use custom cell code
1642 wxDataViewCustomRenderer::Init(mode
, align
);
1646 wxDataViewProgressRenderer::~wxDataViewProgressRenderer()
1650 bool wxDataViewProgressRenderer::SetValue( const wxVariant
&value
)
1653 if (!gtk_check_version(2,6,0))
1655 gint tmp
= (long) value
;
1656 GValue gvalue
= { 0, };
1657 g_value_init( &gvalue
, G_TYPE_INT
);
1658 g_value_set_int( &gvalue
, tmp
);
1659 g_object_set_property( G_OBJECT(m_renderer
), "value", &gvalue
);
1660 g_value_unset( &gvalue
);
1665 m_value
= (long) value
;
1667 if (m_value
< 0) m_value
= 0;
1668 if (m_value
> 100) m_value
= 100;
1674 bool wxDataViewProgressRenderer::GetValue( wxVariant
&value
) const
1679 bool wxDataViewProgressRenderer::Render( wxRect cell
, wxDC
*dc
, int state
)
1681 double pct
= (double)m_value
/ 100.0;
1683 bar
.width
= (int)(cell
.width
* pct
);
1684 dc
->SetPen( *wxTRANSPARENT_PEN
);
1685 dc
->SetBrush( *wxBLUE_BRUSH
);
1686 dc
->DrawRectangle( bar
);
1688 dc
->SetBrush( *wxTRANSPARENT_BRUSH
);
1689 dc
->SetPen( *wxBLACK_PEN
);
1690 dc
->DrawRectangle( cell
);
1695 wxSize
wxDataViewProgressRenderer::GetSize() const
1697 return wxSize(40,12);
1700 // ---------------------------------------------------------
1701 // wxDataViewDateRenderer
1702 // ---------------------------------------------------------
1704 class wxDataViewDateRendererPopupTransient
: public wxPopupTransientWindow
1707 wxDataViewDateRendererPopupTransient( wxWindow
* parent
, wxDateTime
*value
,
1708 wxDataViewModel
*model
, const wxDataViewItem
&item
, unsigned int col
) :
1709 wxPopupTransientWindow( parent
, wxBORDER_SIMPLE
)
1714 m_cal
= new wxCalendarCtrl( this, -1, *value
);
1715 wxBoxSizer
*sizer
= new wxBoxSizer( wxHORIZONTAL
);
1716 sizer
->Add( m_cal
, 1, wxGROW
);
1721 virtual void OnDismiss()
1725 void OnCalendar( wxCalendarEvent
&event
);
1727 wxCalendarCtrl
*m_cal
;
1728 wxDataViewModel
*m_model
;
1729 wxDataViewItem m_item
;
1733 DECLARE_EVENT_TABLE()
1736 BEGIN_EVENT_TABLE(wxDataViewDateRendererPopupTransient
,wxPopupTransientWindow
)
1737 EVT_CALENDAR( -1, wxDataViewDateRendererPopupTransient::OnCalendar
)
1740 void wxDataViewDateRendererPopupTransient::OnCalendar( wxCalendarEvent
&event
)
1742 wxDateTime date
= event
.GetDate();
1743 wxVariant value
= date
;
1744 m_model
->SetValue( value
, m_item
, m_col
);
1745 m_model
->ValueChanged( m_item
, m_col
);
1749 IMPLEMENT_CLASS(wxDataViewDateRenderer
, wxDataViewCustomRenderer
)
1751 wxDataViewDateRenderer::wxDataViewDateRenderer( const wxString
&varianttype
,
1752 wxDataViewCellMode mode
, int align
) :
1753 wxDataViewCustomRenderer( varianttype
, mode
, align
)
1756 SetAlignment(align
);
1759 bool wxDataViewDateRenderer::SetValue( const wxVariant
&value
)
1761 m_date
= value
.GetDateTime();
1766 bool wxDataViewDateRenderer::GetValue( wxVariant
&value
) const
1771 bool wxDataViewDateRenderer::Render( wxRect cell
, wxDC
*dc
, int state
)
1773 dc
->SetFont( GetOwner()->GetOwner()->GetFont() );
1774 wxString tmp
= m_date
.FormatDate();
1775 dc
->DrawText( tmp
, cell
.x
, cell
.y
);
1780 wxSize
wxDataViewDateRenderer::GetSize() const
1782 wxString tmp
= m_date
.FormatDate();
1784 GetView()->GetTextExtent( tmp
, &x
, &y
, &d
);
1785 return wxSize(x
,y
+d
);
1788 bool wxDataViewDateRenderer::Activate( wxRect cell
, wxDataViewModel
*model
,
1789 const wxDataViewItem
&item
, unsigned int col
)
1792 model
->GetValue( variant
, item
, col
);
1793 wxDateTime value
= variant
.GetDateTime();
1795 wxDataViewDateRendererPopupTransient
*popup
= new wxDataViewDateRendererPopupTransient(
1796 GetOwner()->GetOwner()->GetParent(), &value
, model
, item
, col
);
1797 wxPoint pos
= wxGetMousePosition();
1800 popup
->Popup( popup
->m_cal
);
1805 // ---------------------------------------------------------
1807 // ---------------------------------------------------------
1811 gtk_dataview_header_button_press_callback( GtkWidget
*widget
,
1812 GdkEventButton
*gdk_event
,
1813 wxDataViewColumn
*column
)
1815 if (gdk_event
->type
!= GDK_BUTTON_PRESS
)
1818 if (gdk_event
->button
== 1)
1820 wxDataViewCtrl
*dv
= column
->GetOwner();
1821 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_COLUMN_HEADER_CLICK
, dv
->GetId() );
1822 event
.SetDataViewColumn( column
);
1823 event
.SetModel( dv
->GetModel() );
1824 if (dv
->GetEventHandler()->ProcessEvent( event
))
1832 static void wxGtkTreeCellDataFunc( GtkTreeViewColumn
*column
,
1833 GtkCellRenderer
*cell
,
1834 GtkTreeModel
*model
,
1840 static void wxGtkTreeCellDataFunc( GtkTreeViewColumn
*column
,
1841 GtkCellRenderer
*renderer
,
1842 GtkTreeModel
*model
,
1846 g_return_if_fail (GTK_IS_WX_TREE_MODEL (model
));
1847 GtkWxTreeModel
*tree_model
= (GtkWxTreeModel
*) model
;
1849 wxDataViewRenderer
*cell
= (wxDataViewRenderer
*) data
;
1851 wxDataViewItem
item( (void*) iter
->user_data
);
1854 tree_model
->internal
->GetDataViewModel()->GetValue( value
, item
, cell
->GetOwner()->GetModelColumn() );
1856 if (value
.GetType() != cell
->GetVariantType())
1857 wxLogError( wxT("Wrong type, required: %s but: %s"),
1858 value
.GetType().c_str(),
1859 cell
->GetVariantType().c_str() );
1861 cell
->SetValue( value
);
1864 wxListItemAttr attr
;
1865 tree_model
->model
->GetAttr( attr
, cell
->GetOwner()->GetModelColumn(), model_row
);
1867 if (attr
.HasBackgroundColour())
1869 wxColour colour
= attr
.GetBackgroundColour();
1870 const GdkColor
* const gcol
= colour
.GetColor();
1872 GValue gvalue
= { 0, };
1873 g_value_init( &gvalue
, GDK_TYPE_COLOR
);
1874 g_value_set_boxed( &gvalue
, gcol
);
1875 g_object_set_property( G_OBJECT(renderer
), "cell-background_gdk", &gvalue
);
1876 g_value_unset( &gvalue
);
1880 GValue gvalue
= { 0, };
1881 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1882 g_value_set_boolean( &gvalue
, FALSE
);
1883 g_object_set_property( G_OBJECT(renderer
), "cell-background-set", &gvalue
);
1884 g_value_unset( &gvalue
);
1890 IMPLEMENT_CLASS(wxDataViewColumn
, wxDataViewColumnBase
)
1892 wxDataViewColumn::wxDataViewColumn( const wxString
&title
, wxDataViewRenderer
*cell
,
1893 unsigned int model_column
, int width
,
1894 wxAlignment align
, int flags
) :
1895 wxDataViewColumnBase( title
, cell
, model_column
, width
, align
, flags
)
1897 Init( align
, flags
, width
);
1899 gtk_tree_view_column_set_clickable( GTK_TREE_VIEW_COLUMN(m_column
), TRUE
);
1903 wxDataViewColumn::wxDataViewColumn( const wxBitmap
&bitmap
, wxDataViewRenderer
*cell
,
1904 unsigned int model_column
, int width
,
1905 wxAlignment align
, int flags
) :
1906 wxDataViewColumnBase( bitmap
, cell
, model_column
, width
, align
, flags
)
1908 Init( align
, flags
, width
);
1910 SetBitmap( bitmap
);
1913 void wxDataViewColumn::Init(wxAlignment align
, int flags
, int width
)
1915 m_isConnected
= false;
1917 GtkCellRenderer
*renderer
= (GtkCellRenderer
*) GetRenderer()->GetGtkHandle();
1918 GtkTreeViewColumn
*column
= gtk_tree_view_column_new();
1919 m_column
= (GtkWidget
*) column
;
1922 SetAlignment( align
);
1924 // NOTE: we prefer not to call SetMinWidth(wxDVC_DEFAULT_MINWIDTH);
1925 // as GTK+ is smart and unless explicitely told, will set the minimal
1926 // width to the title's lenght, which is a better default
1928 // the GTK_TREE_VIEW_COLUMN_FIXED is required by the "fixed height" mode
1929 // that we use for the wxDataViewCtrl
1930 gtk_tree_view_column_set_fixed_width( column
, width
< 0 ? wxDVC_DEFAULT_WIDTH
: width
);
1931 gtk_tree_view_column_set_sizing( column
, GTK_TREE_VIEW_COLUMN_FIXED
);
1933 gtk_tree_view_column_pack_end( column
, renderer
, TRUE
);
1935 gtk_tree_view_column_set_cell_data_func( column
, renderer
,
1936 wxGtkTreeCellDataFunc
, (gpointer
) GetRenderer(), NULL
);
1939 wxDataViewColumn::~wxDataViewColumn()
1943 void wxDataViewColumn::OnInternalIdle()
1948 if (GTK_WIDGET_REALIZED(GetOwner()->m_treeview
))
1950 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1953 g_signal_connect(column
->button
, "button_press_event",
1954 G_CALLBACK (gtk_dataview_header_button_press_callback
), this);
1956 m_isConnected
= true;
1961 void wxDataViewColumn::SetOwner( wxDataViewCtrl
*owner
)
1963 wxDataViewColumnBase::SetOwner( owner
);
1965 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1967 gtk_tree_view_column_set_title( column
, wxGTK_CONV_FONT(GetTitle(), GetOwner()->GetFont() ) );
1970 void wxDataViewColumn::SetTitle( const wxString
&title
)
1972 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1976 // disconnect before column->button gets recreated
1977 g_signal_handlers_disconnect_by_func( column
->button
,
1978 (GtkWidget
*) gtk_dataview_header_button_press_callback
, this);
1980 m_isConnected
= false;
1983 // FIXME: can it really happen that we don't have the owner here??
1984 wxDataViewCtrl
*ctrl
= GetOwner();
1985 gtk_tree_view_column_set_title( column
, ctrl
? wxGTK_CONV_FONT(title
, ctrl
->GetFont())
1986 : wxGTK_CONV_SYS(title
) );
1988 gtk_tree_view_column_set_widget( column
, NULL
);
1991 wxString
wxDataViewColumn::GetTitle() const
1993 const gchar
*str
= gtk_tree_view_column_get_title( GTK_TREE_VIEW_COLUMN(m_column
) );
1994 return wxConvFileName
->cMB2WX(str
);
1997 void wxDataViewColumn::SetBitmap( const wxBitmap
&bitmap
)
1999 wxDataViewColumnBase::SetBitmap( bitmap
);
2001 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
2004 GtkImage
*gtk_image
= GTK_IMAGE( gtk_image_new() );
2006 GdkBitmap
*mask
= (GdkBitmap
*) NULL
;
2007 if (bitmap
.GetMask())
2008 mask
= bitmap
.GetMask()->GetBitmap();
2010 if (bitmap
.HasPixbuf())
2012 gtk_image_set_from_pixbuf(GTK_IMAGE(gtk_image
),
2013 bitmap
.GetPixbuf());
2017 gtk_image_set_from_pixmap(GTK_IMAGE(gtk_image
),
2018 bitmap
.GetPixmap(), mask
);
2020 gtk_widget_show( GTK_WIDGET(gtk_image
) );
2022 gtk_tree_view_column_set_widget( column
, GTK_WIDGET(gtk_image
) );
2026 gtk_tree_view_column_set_widget( column
, NULL
);
2030 void wxDataViewColumn::SetHidden( bool hidden
)
2032 gtk_tree_view_column_set_visible( GTK_TREE_VIEW_COLUMN(m_column
), !hidden
);
2035 void wxDataViewColumn::SetResizeable( bool resizeable
)
2037 gtk_tree_view_column_set_resizable( GTK_TREE_VIEW_COLUMN(m_column
), resizeable
);
2040 void wxDataViewColumn::SetAlignment( wxAlignment align
)
2042 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
2044 gfloat xalign
= 0.0;
2045 if (align
== wxALIGN_RIGHT
)
2047 if (align
== wxALIGN_CENTER_HORIZONTAL
||
2048 align
== wxALIGN_CENTER
)
2051 gtk_tree_view_column_set_alignment( column
, xalign
);
2054 wxAlignment
wxDataViewColumn::GetAlignment() const
2056 gfloat xalign
= gtk_tree_view_column_get_alignment( GTK_TREE_VIEW_COLUMN(m_column
) );
2059 return wxALIGN_RIGHT
;
2061 return wxALIGN_CENTER_HORIZONTAL
;
2063 return wxALIGN_LEFT
;
2066 void wxDataViewColumn::SetSortable( bool sortable
)
2068 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
2071 gtk_tree_view_column_set_sort_column_id( column
, GetModelColumn() );
2073 gtk_tree_view_column_set_sort_column_id( column
, -1 );
2076 bool wxDataViewColumn::IsSortable() const
2078 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
2079 return (gtk_tree_view_column_get_sort_column_id( column
) != -1);
2082 bool wxDataViewColumn::IsResizeable() const
2084 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
2085 return gtk_tree_view_column_get_resizable( column
);
2088 bool wxDataViewColumn::IsHidden() const
2090 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
2091 return !gtk_tree_view_column_get_visible( column
);
2094 void wxDataViewColumn::SetSortOrder( bool ascending
)
2096 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
2099 gtk_tree_view_column_set_sort_order( column
, GTK_SORT_ASCENDING
);
2101 gtk_tree_view_column_set_sort_order( column
, GTK_SORT_DESCENDING
);
2104 bool wxDataViewColumn::IsSortOrderAscending() const
2106 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
2108 return (gtk_tree_view_column_get_sort_order( column
) != GTK_SORT_DESCENDING
);
2111 void wxDataViewColumn::SetMinWidth( int width
)
2113 gtk_tree_view_column_set_min_width( GTK_TREE_VIEW_COLUMN(m_column
), width
);
2116 int wxDataViewColumn::GetMinWidth() const
2118 return gtk_tree_view_column_get_min_width( GTK_TREE_VIEW_COLUMN(m_column
) );
2121 int wxDataViewColumn::GetWidth() const
2123 return gtk_tree_view_column_get_width( GTK_TREE_VIEW_COLUMN(m_column
) );
2126 void wxDataViewColumn::SetWidth( int width
)
2128 gtk_tree_view_column_set_fixed_width( GTK_TREE_VIEW_COLUMN(m_column
), width
);
2132 //-----------------------------------------------------------------------------
2133 // wxGtkTreeModelNode
2134 //-----------------------------------------------------------------------------
2136 void wxGtkTreeModelNode::Resort()
2138 g_model
= m_internal
->GetDataViewModel();
2140 size_t child_count
= GetChildCount();
2141 if (child_count
== 0)
2144 size_t node_count
= GetNodesCount();
2146 if (child_count
== 1)
2148 if (node_count
== 1)
2150 wxGtkTreeModelNode
*node
= m_nodes
.Item( 0 );
2156 wxGtkTreeModelChildren
*new_array
= new wxGtkTreeModelChildren( wxGtkTreeModelNodeCmp
);
2159 for (pos
= 0; pos
< child_count
; pos
++)
2160 new_array
->Add( m_children
->Item( pos
) );
2162 gint
*new_order
= new gint
[child_count
];
2164 for (pos
= 0; pos
< child_count
; pos
++)
2166 void *id
= new_array
->Item( pos
);
2168 for (old_pos
= 0; old_pos
< child_count
; old_pos
++)
2170 if (id
== m_children
->Item(old_pos
))
2172 new_order
[pos
] = old_pos
;
2178 // for (pos = 0; pos < count; pos++)
2179 // m_children->Clear();
2182 m_children
= new_array
;
2184 GtkTreeModel
*gtk_tree_model
= GTK_TREE_MODEL( m_internal
->GetGtkModel() );
2186 GtkTreePath
*path
= gtk_tree_path_new ();
2187 wxGtkTreeModelNode
*parent
= GetParent();
2188 void *id
= GetItem().GetID();
2192 int pos
= parent
->GetChildren().Index( id
);
2193 gtk_tree_path_prepend_index( path
, pos
);
2194 id
= parent
->GetItem().GetID();
2195 parent
= parent
->GetParent();
2199 iter
.user_data
= id
;
2200 iter
.stamp
= m_internal
->GetGtkModel()->stamp
;
2201 gtk_tree_model_rows_reordered( gtk_tree_model
, path
, &iter
, new_order
);
2203 gtk_tree_path_free (path
);
2205 delete [] new_order
;
2207 for (pos
= 0; pos
< node_count
; pos
++)
2209 wxGtkTreeModelNode
*node
= m_nodes
.Item( pos
);
2214 //-----------------------------------------------------------------------------
2215 // wxDataViewCtrlInternal
2216 //-----------------------------------------------------------------------------
2218 wxDataViewCtrlInternal::wxDataViewCtrlInternal( wxDataViewCtrl
*owner
,
2219 wxDataViewModel
*wx_model
, GtkWxTreeModel
*gtk_model
)
2222 m_wx_model
= wx_model
;
2223 m_gtk_model
= gtk_model
;
2228 wxDataViewCtrlInternal::~wxDataViewCtrlInternal()
2230 g_object_unref( m_gtk_model
);
2233 void wxDataViewCtrlInternal::InitTree()
2235 wxDataViewItem item
;
2236 m_root
= new wxGtkTreeModelNode( NULL
, item
, this );
2238 BuildBranch( m_root
);
2241 void wxDataViewCtrlInternal::BuildBranch( wxGtkTreeModelNode
*node
)
2243 if (node
->GetChildCount() == 0)
2245 wxDataViewItem child
= m_wx_model
->GetFirstChild( node
->GetItem() );
2246 while (child
.IsOk())
2248 if (m_wx_model
->IsContainer( child
))
2249 node
->AddNode( new wxGtkTreeModelNode( node
, child
, this ) );
2251 node
->AddLeave( child
.GetID() );
2252 child
= m_wx_model
->GetNextSibling( child
);
2257 void wxDataViewCtrlInternal::Resort()
2262 bool wxDataViewCtrlInternal::ItemAdded( const wxDataViewItem
&parent
, const wxDataViewItem
&item
)
2264 wxGtkTreeModelNode
*parent_node
= FindNode( parent
);
2265 if (m_wx_model
->IsContainer( item
))
2266 parent_node
->AddNode( new wxGtkTreeModelNode( parent_node
, item
, this ) );
2268 parent_node
->AddLeave( item
.GetID() );
2272 bool wxDataViewCtrlInternal::ItemDeleted( const wxDataViewItem
&item
)
2274 wxGtkTreeModelNode
*parent
= FindParentNode( item
);
2275 parent
->DeleteChild( item
.GetID() );
2279 gboolean
wxDataViewCtrlInternal::get_iter( GtkTreeIter
*iter
, GtkTreePath
*path
)
2281 g_model
= m_wx_model
;
2283 int depth
= gtk_tree_path_get_depth( path
);
2285 wxGtkTreeModelNode
*node
= m_root
;
2288 for (i
= 0; i
< depth
; i
++)
2290 BuildBranch( node
);
2292 gint pos
= gtk_tree_path_get_indices (path
)[i
];
2293 if (pos
< 0) return FALSE
;
2294 if ((size_t)pos
>= node
->GetChildCount()) return FALSE
;
2296 void* id
= node
->GetChildren().Item( (size_t) pos
);
2300 iter
->stamp
= m_gtk_model
->stamp
;
2301 iter
->user_data
= id
;
2305 size_t count
= node
->GetNodes().GetCount();
2307 for (pos2
= 0; pos2
< count
; pos2
++)
2309 wxGtkTreeModelNode
*child_node
= node
->GetNodes().Item( pos2
);
2310 if (child_node
->GetItem().GetID() == id
)
2321 GtkTreePath
*wxDataViewCtrlInternal::get_path( GtkTreeIter
*iter
)
2323 g_model
= m_wx_model
;
2325 GtkTreePath
*retval
= gtk_tree_path_new ();
2326 void *id
= iter
->user_data
;
2328 wxGtkTreeModelNode
*node
= FindParentNode( iter
);
2331 int pos
= node
->GetChildren().Index( id
);
2332 gtk_tree_path_prepend_index( retval
, pos
);
2334 id
= node
->GetItem().GetID();
2335 node
= node
->GetParent();
2341 GtkTreePath
*wxDataViewCtrlInternal::get_path_safe( GtkTreeIter
*iter
)
2343 g_model
= m_wx_model
;
2345 GtkTreePath
*retval
= gtk_tree_path_new ();
2346 void *id
= iter
->user_data
;
2348 wxGtkTreeModelNode
*node
= FindParentNode( iter
);
2352 for (pos
= 0; pos
< node
->GetChildren().GetCount(); pos
++)
2354 if (id
== node
->GetChildren().Item( pos
))
2356 gtk_tree_path_prepend_index( retval
, (int) pos
);
2361 id
= node
->GetItem().GetID();
2362 node
= node
->GetParent();
2368 gboolean
wxDataViewCtrlInternal::iter_next( GtkTreeIter
*iter
)
2370 g_model
= m_wx_model
;
2372 wxGtkTreeModelNode
*parent
= FindParentNode( iter
);
2373 unsigned int pos
= parent
->GetChildren().Index( iter
->user_data
);
2375 if (pos
== parent
->GetChildCount()-1)
2378 iter
->stamp
= m_gtk_model
->stamp
;
2379 iter
->user_data
= parent
->GetChildren().Item( pos
+1 );
2384 gboolean
wxDataViewCtrlInternal::iter_children( GtkTreeIter
*iter
, GtkTreeIter
*parent
)
2386 g_model
= m_wx_model
;
2388 wxDataViewItem
item( (void*) parent
->user_data
);
2390 if (!m_wx_model
->IsContainer( item
))
2393 wxGtkTreeModelNode
*parent_node
= FindNode( parent
);
2394 BuildBranch( parent_node
);
2396 if (parent_node
->GetChildCount() == 0)
2399 iter
->stamp
= m_gtk_model
->stamp
;
2400 iter
->user_data
= (gpointer
) parent_node
->GetChildren().Item( 0 );
2405 gboolean
wxDataViewCtrlInternal::iter_has_child( GtkTreeIter
*iter
)
2407 g_model
= m_wx_model
;
2409 wxDataViewItem
item( (void*) iter
->user_data
);
2410 bool is_container
= m_wx_model
->IsContainer( item
);
2415 wxGtkTreeModelNode
*node
= FindNode( iter
);
2416 BuildBranch( node
);
2418 return (node
->GetChildCount() > 0);
2421 gint
wxDataViewCtrlInternal::iter_n_children( GtkTreeIter
*iter
)
2423 g_model
= m_wx_model
;
2425 wxDataViewItem
item( (void*) iter
->user_data
);
2427 if (!m_wx_model
->IsContainer( item
))
2430 wxGtkTreeModelNode
*parent_node
= FindNode( iter
);
2431 BuildBranch( parent_node
);
2433 // wxPrintf( "iter_n_children %d\n", parent_node->GetChildCount() );
2435 return parent_node
->GetChildCount();
2438 gboolean
wxDataViewCtrlInternal::iter_nth_child( GtkTreeIter
*iter
, GtkTreeIter
*parent
, gint n
)
2440 g_model
= m_wx_model
;
2443 if (parent
) id
= (void*) parent
->user_data
;
2444 wxDataViewItem
item( id
);
2446 if (!m_wx_model
->IsContainer( item
))
2449 wxGtkTreeModelNode
*parent_node
= FindNode( parent
);
2450 BuildBranch( parent_node
);
2452 // wxPrintf( "iter_nth_child %d\n", n );
2454 iter
->stamp
= m_gtk_model
->stamp
;
2455 iter
->user_data
= parent_node
->GetChildren().Item( n
);
2460 gboolean
wxDataViewCtrlInternal::iter_parent( GtkTreeIter
*iter
, GtkTreeIter
*child
)
2462 g_model
= m_wx_model
;
2464 wxGtkTreeModelNode
*node
= FindParentNode( child
);
2468 iter
->stamp
= m_gtk_model
->stamp
;
2469 iter
->user_data
= (gpointer
) node
->GetItem().GetID();
2474 static wxGtkTreeModelNode
*
2475 wxDataViewCtrlInternal_FindNode( wxGtkTreeModelNode
*node
, const wxDataViewItem
&item
)
2477 if (!node
) return NULL
;
2479 size_t count
= node
->GetNodesCount();
2481 for (i
= 0; i
< count
; i
++)
2483 wxGtkTreeModelNode
*child
= node
->GetNodes().Item( i
);
2484 if (child
->GetItem().GetID() == item
.GetID())
2486 // wxPrintf( "leave findnode at %d\n", i );
2490 wxGtkTreeModelNode
*node2
= wxDataViewCtrlInternal_FindNode( child
, item
);
2493 // wxPrintf( "branch findnode at %d\n", i );
2501 wxGtkTreeModelNode
*wxDataViewCtrlInternal::FindNode( GtkTreeIter
*iter
)
2506 wxDataViewItem
item( (void*) iter
->user_data
);
2510 wxGtkTreeModelNode
*result
= wxDataViewCtrlInternal_FindNode( m_root
, item
);
2514 wxPrintf( "Not found %d\n", (int) iter
->user_data
);
2522 wxGtkTreeModelNode
*wxDataViewCtrlInternal::FindNode( const wxDataViewItem
&item
)
2527 wxGtkTreeModelNode
*result
= wxDataViewCtrlInternal_FindNode( m_root
, item
);
2531 wxPrintf( "Not found %d\n", (int) item
.GetID() );
2539 static wxGtkTreeModelNode
*
2540 wxDataViewCtrlInternal_FindParentNode( wxGtkTreeModelNode
*node
, const wxDataViewItem
&item
)
2542 size_t child_count
= node
->GetChildCount();
2543 void *id
= item
.GetID();
2544 const wxGtkTreeModelChildren
&children
= node
->GetChildren();
2546 for (pos
= 0; pos
< child_count
; pos
++)
2548 if (children
.Item( pos
) == id
)
2552 size_t node_count
= node
->GetNodesCount();
2553 for (pos
= 0; pos
< node_count
; pos
++)
2555 wxGtkTreeModelNode
*child
= node
->GetNodes().Item( pos
);
2557 wxGtkTreeModelNode
*node2
= wxDataViewCtrlInternal_FindParentNode( child
, item
);
2560 // wxPrintf( "branch findnode at %d\n", i );
2568 wxGtkTreeModelNode
*wxDataViewCtrlInternal::FindParentNode( GtkTreeIter
*iter
)
2573 wxDataViewItem
item( (void*) iter
->user_data
);
2577 return wxDataViewCtrlInternal_FindParentNode( m_root
, item
);
2580 wxGtkTreeModelNode
*wxDataViewCtrlInternal::FindParentNode( const wxDataViewItem
&item
)
2585 return wxDataViewCtrlInternal_FindParentNode( m_root
, item
);
2588 //-----------------------------------------------------------------------------
2589 // wxDataViewCtrl signal callbacks
2590 //-----------------------------------------------------------------------------
2593 wxdataview_selection_changed_callback( GtkTreeSelection
* selection
, wxDataViewCtrl
*dv
)
2595 if (!GTK_WIDGET_REALIZED(dv
->m_widget
))
2598 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_SELECTED
, dv
->GetId() );
2600 event
.SetModel( dv
->GetModel() );
2601 dv
->GetEventHandler()->ProcessEvent( event
);
2605 wxdataview_row_activated_callback( GtkTreeView
* treeview
, GtkTreePath
*path
,
2606 GtkTreeViewColumn
*column
, wxDataViewCtrl
*dv
)
2608 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_ACTIVATED
, dv
->GetId() );
2611 dv
->GtkGetInternal()->get_iter( &iter
, path
);
2612 wxDataViewItem
item( (void*) iter
.user_data
);;
2613 event
.SetItem( item
);
2614 event
.SetModel( dv
->GetModel() );
2615 dv
->GetEventHandler()->ProcessEvent( event
);
2618 //-----------------------------------------------------------------------------
2620 //-----------------------------------------------------------------------------
2622 //-----------------------------------------------------------------------------
2623 // InsertChild for wxDataViewCtrl
2624 //-----------------------------------------------------------------------------
2626 static void wxInsertChildInDataViewCtrl( wxWindowGTK
* parent
, wxWindowGTK
* child
)
2628 wxDataViewCtrl
* dvc
= (wxDataViewCtrl
*) parent
;
2629 GtkWidget
*treeview
= dvc
->GtkGetTreeView();
2631 // Insert widget in GtkTreeView
2632 if (GTK_WIDGET_REALIZED(treeview
))
2633 gtk_widget_set_parent_window( child
->m_widget
,
2634 gtk_tree_view_get_bin_window( GTK_TREE_VIEW(treeview
) ) );
2635 gtk_widget_set_parent( child
->m_widget
, treeview
);
2639 void gtk_dataviewctrl_size_callback( GtkWidget
*WXUNUSED(widget
),
2640 GtkAllocation
*alloc
,
2641 wxDataViewCtrl
*win
)
2644 wxWindowList::Node
*node
= win
->GetChildren().GetFirst();
2647 wxWindow
*child
= node
->GetData();
2650 gtk_widget_size_request( child
->m_widget
, &req
);
2652 GtkAllocation alloc
;
2653 alloc
.x
= child
->m_x
;
2654 alloc
.y
= child
->m_y
;
2655 alloc
.width
= child
->m_width
;
2656 alloc
.height
= child
->m_height
;
2657 gtk_widget_size_allocate( child
->m_widget
, &alloc
);
2659 node
= node
->GetNext();
2665 IMPLEMENT_DYNAMIC_CLASS(wxDataViewCtrl
, wxDataViewCtrlBase
)
2667 wxDataViewCtrl::~wxDataViewCtrl()
2670 GetModel()->RemoveNotifier( m_notifier
);
2672 // remove the model from the GtkTreeView before it gets destroyed by the
2673 // wxDataViewCtrlBase's dtor
2674 gtk_tree_view_set_model( GTK_TREE_VIEW(m_treeview
), NULL
);
2679 void wxDataViewCtrl::Init()
2684 bool wxDataViewCtrl::Create(wxWindow
*parent
, wxWindowID id
,
2685 const wxPoint
& pos
, const wxSize
& size
,
2686 long style
, const wxValidator
& validator
)
2690 if (!PreCreation( parent
, pos
, size
) ||
2691 !CreateBase( parent
, id
, pos
, size
, style
, validator
))
2693 wxFAIL_MSG( wxT("wxDataViewCtrl creation failed") );
2697 m_insertCallback
= wxInsertChildInDataViewCtrl
;
2699 m_widget
= gtk_scrolled_window_new (NULL
, NULL
);
2701 GtkScrolledWindowSetBorder(m_widget
, style
);
2703 m_treeview
= gtk_tree_view_new();
2704 gtk_container_add (GTK_CONTAINER (m_widget
), m_treeview
);
2706 g_signal_connect (m_treeview
, "size_allocate",
2707 G_CALLBACK (gtk_dataviewctrl_size_callback
), this);
2710 if (!gtk_check_version(2,6,0))
2711 gtk_tree_view_set_fixed_height_mode( GTK_TREE_VIEW(m_treeview
), TRUE
);
2714 if (style
& wxDV_MULTIPLE
)
2716 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
2717 gtk_tree_selection_set_mode( selection
, GTK_SELECTION_MULTIPLE
);
2720 gtk_tree_view_set_headers_visible( GTK_TREE_VIEW(m_treeview
), (style
& wxDV_NO_HEADER
) == 0 );
2723 if (!gtk_check_version(2,10,0))
2725 GtkTreeViewGridLines grid
= GTK_TREE_VIEW_GRID_LINES_NONE
;
2727 if ((style
& wxDV_HORIZ_RULES
) != 0 &&
2728 (style
& wxDV_VERT_RULES
) != 0)
2729 grid
= GTK_TREE_VIEW_GRID_LINES_BOTH
;
2730 else if (style
& wxDV_VERT_RULES
)
2731 grid
= GTK_TREE_VIEW_GRID_LINES_VERTICAL
;
2732 else if (style
& wxDV_HORIZ_RULES
)
2733 grid
= GTK_TREE_VIEW_GRID_LINES_HORIZONTAL
;
2735 gtk_tree_view_set_grid_lines( GTK_TREE_VIEW(m_treeview
), grid
);
2740 gtk_tree_view_set_rules_hint( GTK_TREE_VIEW(m_treeview
), (style
& wxDV_HORIZ_RULES
) != 0 );
2743 gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (m_widget
),
2744 GTK_POLICY_AUTOMATIC
, GTK_POLICY_ALWAYS
);
2745 gtk_widget_show (m_treeview
);
2747 m_parent
->DoAddChild( this );
2751 GtkEnableSelectionEvents();
2753 g_signal_connect_after (m_treeview
, "row_activated",
2754 G_CALLBACK (wxdataview_row_activated_callback
), this);
2759 void wxDataViewCtrl::OnInternalIdle()
2761 wxWindow::OnInternalIdle();
2763 unsigned int cols
= GetColumnCount();
2765 for (i
= 0; i
< cols
; i
++)
2767 wxDataViewColumn
*col
= GetColumn( i
);
2768 col
->OnInternalIdle();
2772 bool wxDataViewCtrl::AssociateModel( wxDataViewModel
*model
)
2774 if (!wxDataViewCtrlBase::AssociateModel( model
))
2777 GtkWxTreeModel
*gtk_model
= wxgtk_tree_model_new();
2778 m_internal
= new wxDataViewCtrlInternal( this, model
, gtk_model
);
2779 gtk_model
->internal
= m_internal
;
2781 m_notifier
= new wxGtkDataViewModelNotifier( gtk_model
, model
, this );
2783 model
->AddNotifier( m_notifier
);
2785 gtk_tree_view_set_model( GTK_TREE_VIEW(m_treeview
), GTK_TREE_MODEL(gtk_model
) );
2787 // unref in wxDataViewCtrlInternal
2788 // g_object_unref( gtk_model );
2793 bool wxDataViewCtrl::AppendColumn( wxDataViewColumn
*col
)
2795 if (!wxDataViewCtrlBase::AppendColumn(col
))
2798 GtkTreeViewColumn
*column
= (GtkTreeViewColumn
*)col
->GetGtkHandle();
2800 gtk_tree_view_append_column( GTK_TREE_VIEW(m_treeview
), column
);
2805 wxDataViewItem
wxDataViewCtrl::GetSelection()
2807 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
2809 if (m_windowStyle
& wxDV_MULTIPLE
)
2815 if (gtk_tree_selection_get_selected( selection
, NULL
, &iter
))
2817 wxDataViewItem
item( (void*) iter
.user_data
);
2822 return wxDataViewItem(0);
2825 void wxDataViewCtrl::DoSetExpanderColumn()
2829 void wxDataViewCtrl::DoSetIndent()
2833 void wxDataViewCtrl::GtkDisableSelectionEvents()
2835 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
2836 g_signal_connect_after (selection
, "changed",
2837 G_CALLBACK (wxdataview_selection_changed_callback
), this);
2840 void wxDataViewCtrl::GtkEnableSelectionEvents()
2842 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
2843 g_signal_handlers_disconnect_by_func( selection
,
2844 (gpointer
) (wxdataview_selection_changed_callback
), this);
2849 wxDataViewCtrl::GetClassDefaultAttributes(wxWindowVariant
WXUNUSED(variant
))
2851 return GetDefaultAttributesFromGTKWidget(gtk_tree_view_new
);
2856 // !wxUSE_GENERICDATAVIEWCTRL
2859 // wxUSE_DATAVIEWCTRL