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"
23 #include "wx/settings.h"
27 #include "wx/stockitem.h"
28 #include "wx/popupwin.h"
29 #include "wx/listimpl.cpp"
31 #include "wx/gtk/private.h"
32 #include "wx/gtk/dc.h"
33 #include "wx/gtk/dcclient.h"
35 #include "wx/gtk/private/gdkconv.h"
36 #include "wx/gtk/private/list.h"
37 #include "wx/gtk/private/event.h"
38 using namespace wxGTKImpl
;
40 class wxGtkDataViewModelNotifier
;
42 //-----------------------------------------------------------------------------
43 //-----------------------------------------------------------------------------
45 static wxDataViewCtrlInternal
*gs_internal
= NULL
;
47 class wxGtkTreeModelNode
;
50 typedef struct _GtkWxTreeModel GtkWxTreeModel
;
53 // ----------------------------------------------------------------------------
54 // wxGtkTreePath: self-destroying GtkTreePath
55 // ----------------------------------------------------------------------------
57 // Usually this object is initialized with the associated GtkTreePath
58 // immediately when it's constructed but it can also be changed later either by
59 // using Assign() or by getting the pointer to the internally stored pointer
60 // value using ByRef(). The latter should be avoided but is very convenient
61 // when using GTK functions with GtkTreePath output parameters.
65 // Ctor takes ownership of the given path and will free it if non-NULL.
66 wxGtkTreePath(GtkTreePath
*path
= NULL
) : m_path(path
) { }
68 // Creates a tree path for the given string path.
69 wxGtkTreePath(const gchar
*strpath
)
70 : m_path(gtk_tree_path_new_from_string(strpath
))
74 // Set the stored pointer if not done by ctor.
75 void Assign(GtkTreePath
*path
)
77 wxASSERT_MSG( !m_path
, "shouldn't be already initialized" );
82 // Return the pointer to the internally stored pointer. This should only be
83 // used to initialize the object by passing it to some GTK function.
86 wxASSERT_MSG( !m_path
, "shouldn't be already initialized" );
92 operator GtkTreePath
*() const { return m_path
; }
94 ~wxGtkTreePath() { if ( m_path
) gtk_tree_path_free(m_path
); }
99 wxDECLARE_NO_COPY_CLASS(wxGtkTreePath
);
102 // ----------------------------------------------------------------------------
103 // wxGtkTreePathList: self-destroying list of GtkTreePath objects.
104 // ----------------------------------------------------------------------------
106 class wxGtkTreePathList
: public wxGtkList
109 // Ctor takes ownership of the list.
110 explicit wxGtkTreePathList(GList
* list
)
117 // Delete the list contents, wxGtkList will delete the list itself.
118 g_list_foreach(m_list
, (GFunc
)gtk_tree_path_free
, NULL
);
122 // ----------------------------------------------------------------------------
123 // wxGtkTreeSelectionLock: prevent selection from changing during the
124 // lifetime of this object
125 // ----------------------------------------------------------------------------
127 // Implementation note: it could be expected that setting the selection
128 // function in this class ctor and resetting it back to the old value in its
129 // dtor would work. However currently gtk_tree_selection_get_select_function()
130 // can't be passed NULL (see https://bugzilla.gnome.org/show_bug.cgi?id=626276)
131 // so we can't do this. Instead, we always use the selection function (which
132 // imposes extra overhead, albeit minimal one, on all selection operations) and
133 // just set/reset the flag telling it whether it should allow or forbid the
136 // Also notice that currently only a single object of this class may exist at
137 // any given moment. It's just simpler like this and we don't need anything
141 gboolean
wxdataview_selection_func(GtkTreeSelection
* WXUNUSED(selection
),
142 GtkTreeModel
* WXUNUSED(model
),
143 GtkTreePath
* WXUNUSED(path
),
144 gboolean
WXUNUSED(path_currently_selected
),
150 class wxGtkTreeSelectionLock
153 wxGtkTreeSelectionLock(GtkTreeSelection
*selection
)
154 : m_selection(selection
)
156 wxASSERT_MSG( !ms_instance
, "this class is not reentrant currently" );
160 CheckCurrentSelectionFunc(NULL
);
162 // Pass some non-NULL pointer as "data" for the callback, it doesn't
163 // matter what it is as long as it's non-NULL.
164 gtk_tree_selection_set_select_function(selection
,
165 wxdataview_selection_func
,
170 ~wxGtkTreeSelectionLock()
172 CheckCurrentSelectionFunc(wxdataview_selection_func
);
174 gtk_tree_selection_set_select_function(m_selection
,
175 wxdataview_selection_func
,
183 void CheckCurrentSelectionFunc(GtkTreeSelectionFunc func
)
185 // We can only use gtk_tree_selection_get_select_function() with 2.14+
186 // so check for its availability both during compile- and run-time.
187 #if GTK_CHECK_VERSION(2, 14, 0)
188 if ( gtk_check_version(2, 14, 0) != NULL
)
191 // If this assert is triggered, it means the code elsewhere has called
192 // gtk_tree_selection_set_select_function() but currently doing this
193 // breaks this class so the code here needs to be changed.
196 gtk_tree_selection_get_select_function(m_selection
) == func
,
197 "selection function has changed unexpectedly, review this code!"
204 static wxGtkTreeSelectionLock
*ms_instance
;
206 GtkTreeSelection
* const m_selection
;
208 wxDECLARE_NO_COPY_CLASS(wxGtkTreeSelectionLock
);
211 wxGtkTreeSelectionLock
*wxGtkTreeSelectionLock::ms_instance
= NULL
;
213 //-----------------------------------------------------------------------------
214 // wxDataViewCtrlInternal
215 //-----------------------------------------------------------------------------
217 WX_DECLARE_LIST(wxDataViewItem
, ItemList
);
218 WX_DEFINE_LIST(ItemList
)
220 class wxDataViewCtrlInternal
223 wxDataViewCtrlInternal( wxDataViewCtrl
*owner
, wxDataViewModel
*wx_model
);
224 ~wxDataViewCtrlInternal();
227 GtkTreeModelFlags
get_flags();
228 gboolean
get_iter( GtkTreeIter
*iter
, GtkTreePath
*path
);
229 GtkTreePath
*get_path( GtkTreeIter
*iter
);
230 gboolean
iter_next( GtkTreeIter
*iter
);
231 gboolean
iter_children( GtkTreeIter
*iter
, GtkTreeIter
*parent
);
232 gboolean
iter_has_child( GtkTreeIter
*iter
);
233 gint
iter_n_children( GtkTreeIter
*iter
);
234 gboolean
iter_nth_child( GtkTreeIter
*iter
, GtkTreeIter
*parent
, gint n
);
235 gboolean
iter_parent( GtkTreeIter
*iter
, GtkTreeIter
*child
);
239 bool EnableDragSource( const wxDataFormat
&format
);
240 bool EnableDropTarget( const wxDataFormat
&format
);
242 gboolean
row_draggable( GtkTreeDragSource
*drag_source
, GtkTreePath
*path
);
243 gboolean
drag_data_delete( GtkTreeDragSource
*drag_source
, GtkTreePath
* path
);
244 gboolean
drag_data_get( GtkTreeDragSource
*drag_source
, GtkTreePath
*path
,
245 GtkSelectionData
*selection_data
);
246 gboolean
drag_data_received( GtkTreeDragDest
*drag_dest
, GtkTreePath
*dest
,
247 GtkSelectionData
*selection_data
);
248 gboolean
row_drop_possible( GtkTreeDragDest
*drag_dest
, GtkTreePath
*dest_path
,
249 GtkSelectionData
*selection_data
);
251 // notifactions from wxDataViewModel
252 bool ItemAdded( const wxDataViewItem
&parent
, const wxDataViewItem
&item
);
253 bool ItemDeleted( const wxDataViewItem
&parent
, const wxDataViewItem
&item
);
254 bool ItemChanged( const wxDataViewItem
&item
);
255 bool ValueChanged( const wxDataViewItem
&item
, unsigned int model_column
);
262 void SetSortOrder( GtkSortType sort_order
) { m_sort_order
= sort_order
; }
263 GtkSortType
GetSortOrder() const { return m_sort_order
; }
265 void SetSortColumn( int column
) { m_sort_column
= column
; }
266 int GetSortColumn() const { return m_sort_column
; }
268 void SetDataViewSortColumn( wxDataViewColumn
*column
) { m_dataview_sort_column
= column
; }
269 wxDataViewColumn
*GetDataViewSortColumn() { return m_dataview_sort_column
; }
271 bool IsSorted() { return (m_sort_column
>= 0); }
274 wxDataViewModel
* GetDataViewModel() { return m_wx_model
; }
275 const wxDataViewModel
* GetDataViewModel() const { return m_wx_model
; }
276 wxDataViewCtrl
* GetOwner() { return m_owner
; }
277 GtkWxTreeModel
* GetGtkModel() { return m_gtk_model
; }
279 // item can be deleted already in the model
280 int GetIndexOf( const wxDataViewItem
&parent
, const wxDataViewItem
&item
);
282 void OnInternalIdle();
286 void ScheduleRefresh();
288 wxGtkTreeModelNode
*FindNode( const wxDataViewItem
&item
);
289 wxGtkTreeModelNode
*FindNode( GtkTreeIter
*iter
);
290 wxGtkTreeModelNode
*FindParentNode( const wxDataViewItem
&item
);
291 wxGtkTreeModelNode
*FindParentNode( GtkTreeIter
*iter
);
292 void BuildBranch( wxGtkTreeModelNode
*branch
);
295 wxGtkTreeModelNode
*m_root
;
296 wxDataViewModel
*m_wx_model
;
297 GtkWxTreeModel
*m_gtk_model
;
298 wxDataViewCtrl
*m_owner
;
299 GtkSortType m_sort_order
;
300 wxDataViewColumn
*m_dataview_sort_column
;
303 GtkTargetEntry m_dragSourceTargetEntry
;
304 wxCharBuffer m_dragSourceTargetEntryTarget
;
305 wxDataObject
*m_dragDataObject
;
307 GtkTargetEntry m_dropTargetTargetEntry
;
308 wxCharBuffer m_dropTargetTargetEntryTarget
;
309 wxDataObject
*m_dropDataObject
;
311 wxGtkDataViewModelNotifier
*m_notifier
;
317 //-----------------------------------------------------------------------------
318 // wxGtkTreeModelNode
319 //-----------------------------------------------------------------------------
322 int LINKAGEMODE
wxGtkTreeModelChildCmp( void** id1
, void** id2
)
324 int ret
= gs_internal
->GetDataViewModel()->Compare( wxDataViewItem(*id1
), wxDataViewItem(*id2
),
325 gs_internal
->GetSortColumn(), (gs_internal
->GetSortOrder() == GTK_SORT_ASCENDING
) );
330 WX_DEFINE_ARRAY_PTR( wxGtkTreeModelNode
*, wxGtkTreeModelNodes
);
331 WX_DEFINE_ARRAY_PTR( void*, wxGtkTreeModelChildren
);
333 class wxGtkTreeModelNode
336 wxGtkTreeModelNode( wxGtkTreeModelNode
* parent
, const wxDataViewItem
&item
,
337 wxDataViewCtrlInternal
*internal
)
341 m_internal
= internal
;
344 ~wxGtkTreeModelNode()
346 size_t count
= m_nodes
.GetCount();
348 for (i
= 0; i
< count
; i
++)
350 wxGtkTreeModelNode
*child
= m_nodes
.Item( i
);
355 void AddNode( wxGtkTreeModelNode
* child
)
357 m_nodes
.Add( child
);
359 void *id
= child
->GetItem().GetID();
361 m_children
.Add( id
);
363 if (m_internal
->IsSorted() || m_internal
->GetDataViewModel()->HasDefaultCompare())
365 gs_internal
= m_internal
;
366 m_children
.Sort( &wxGtkTreeModelChildCmp
);
370 void InsertNode( wxGtkTreeModelNode
* child
, unsigned pos
)
372 if (m_internal
->IsSorted() || m_internal
->GetDataViewModel()->HasDefaultCompare())
378 void *id
= child
->GetItem().GetID();
380 // Insert into m_nodes so that the order of nodes in m_nodes is the
381 // same as the order of their corresponding IDs in m_children:
382 const unsigned int count
= m_nodes
.GetCount();
383 bool inserted
= false;
384 for (unsigned i
= 0; i
< count
; i
++)
386 wxGtkTreeModelNode
*node
= m_nodes
[i
];
387 int posInChildren
= m_children
.Index(node
->GetItem().GetID());
388 if ( (unsigned)posInChildren
>= pos
)
390 m_nodes
.Insert(child
, i
);
398 m_children
.Insert( id
, pos
);
401 void AddLeaf( void* id
)
403 InsertLeaf(id
, m_children
.size());
406 void InsertLeaf( void* id
, unsigned pos
)
408 m_children
.Insert( id
, pos
);
410 if (m_internal
->IsSorted() || m_internal
->GetDataViewModel()->HasDefaultCompare())
412 gs_internal
= m_internal
;
413 m_children
.Sort( &wxGtkTreeModelChildCmp
);
417 void DeleteChild( void* id
)
419 m_children
.Remove( id
);
421 unsigned int count
= m_nodes
.GetCount();
423 for (pos
= 0; pos
< count
; pos
++)
425 wxGtkTreeModelNode
*node
= m_nodes
.Item( pos
);
426 if (node
->GetItem().GetID() == id
)
428 m_nodes
.RemoveAt( pos
);
435 // returns position of child node for given item in children list or wxNOT_FOUND
436 int FindChildByItem(const wxDataViewItem
& item
) const
438 const void* itemId
= item
.GetID();
439 const wxGtkTreeModelChildren
& nodes
= m_children
;
440 const int len
= nodes
.size();
441 for ( int i
= 0; i
< len
; i
++ )
443 if ( nodes
[i
] == itemId
)
449 wxGtkTreeModelNode
* GetParent()
451 wxGtkTreeModelNodes
&GetNodes()
453 wxGtkTreeModelChildren
&GetChildren()
454 { return m_children
; }
456 unsigned int GetChildCount() const { return m_children
.GetCount(); }
457 unsigned int GetNodesCount() const { return m_nodes
.GetCount(); }
459 wxDataViewItem
&GetItem() { return m_item
; }
460 wxDataViewCtrlInternal
*GetInternal() { return m_internal
; }
465 wxGtkTreeModelNode
*m_parent
;
466 wxGtkTreeModelNodes m_nodes
;
467 wxGtkTreeModelChildren m_children
;
468 wxDataViewItem m_item
;
469 wxDataViewCtrlInternal
*m_internal
;
473 //-----------------------------------------------------------------------------
475 //-----------------------------------------------------------------------------
477 extern bool g_blockEventsOnDrag
;
479 //-----------------------------------------------------------------------------
480 // define new GTK+ class wxGtkTreeModel
481 //-----------------------------------------------------------------------------
485 #define GTK_TYPE_WX_TREE_MODEL (gtk_wx_tree_model_get_type ())
486 #define GTK_WX_TREE_MODEL(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_TREE_MODEL, GtkWxTreeModel))
487 #define GTK_IS_WX_TREE_MODEL(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_TREE_MODEL))
488 #define GTK_IS_WX_TREE_MODEL_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_TREE_MODEL))
490 GType
gtk_wx_tree_model_get_type (void);
492 struct _GtkWxTreeModel
498 wxDataViewCtrlInternal
*internal
;
501 static GtkWxTreeModel
*wxgtk_tree_model_new (void);
502 static void wxgtk_tree_model_init (GtkWxTreeModel
*tree_model
);
504 static void wxgtk_tree_model_tree_model_init (GtkTreeModelIface
*iface
);
505 static void wxgtk_tree_model_sortable_init (GtkTreeSortableIface
*iface
);
506 static void wxgtk_tree_model_drag_source_init(GtkTreeDragSourceIface
*iface
);
507 static void wxgtk_tree_model_drag_dest_init (GtkTreeDragDestIface
*iface
);
509 static GtkTreeModelFlags
wxgtk_tree_model_get_flags (GtkTreeModel
*tree_model
);
510 static gint
wxgtk_tree_model_get_n_columns (GtkTreeModel
*tree_model
);
511 static GType
wxgtk_tree_model_get_column_type (GtkTreeModel
*tree_model
,
513 static gboolean
wxgtk_tree_model_get_iter (GtkTreeModel
*tree_model
,
516 static GtkTreePath
*wxgtk_tree_model_get_path (GtkTreeModel
*tree_model
,
518 static void wxgtk_tree_model_get_value (GtkTreeModel
*tree_model
,
522 static gboolean
wxgtk_tree_model_iter_next (GtkTreeModel
*tree_model
,
524 static gboolean
wxgtk_tree_model_iter_children (GtkTreeModel
*tree_model
,
526 GtkTreeIter
*parent
);
527 static gboolean
wxgtk_tree_model_iter_has_child (GtkTreeModel
*tree_model
,
529 static gint
wxgtk_tree_model_iter_n_children (GtkTreeModel
*tree_model
,
531 static gboolean
wxgtk_tree_model_iter_nth_child (GtkTreeModel
*tree_model
,
535 static gboolean
wxgtk_tree_model_iter_parent (GtkTreeModel
*tree_model
,
540 static gboolean
wxgtk_tree_model_get_sort_column_id (GtkTreeSortable
*sortable
,
541 gint
*sort_column_id
,
543 static void wxgtk_tree_model_set_sort_column_id (GtkTreeSortable
*sortable
,
546 static void wxgtk_tree_model_set_sort_func (GtkTreeSortable
*sortable
,
548 GtkTreeIterCompareFunc func
,
550 GDestroyNotify destroy
);
551 static void wxgtk_tree_model_set_default_sort_func (GtkTreeSortable
*sortable
,
552 GtkTreeIterCompareFunc func
,
554 GDestroyNotify destroy
);
555 static gboolean
wxgtk_tree_model_has_default_sort_func (GtkTreeSortable
*sortable
);
558 static gboolean
wxgtk_tree_model_row_draggable (GtkTreeDragSource
*drag_source
,
560 static gboolean
wxgtk_tree_model_drag_data_delete (GtkTreeDragSource
*drag_source
,
562 static gboolean
wxgtk_tree_model_drag_data_get (GtkTreeDragSource
*drag_source
,
564 GtkSelectionData
*selection_data
);
565 static gboolean
wxgtk_tree_model_drag_data_received (GtkTreeDragDest
*drag_dest
,
567 GtkSelectionData
*selection_data
);
568 static gboolean
wxgtk_tree_model_row_drop_possible (GtkTreeDragDest
*drag_dest
,
569 GtkTreePath
*dest_path
,
570 GtkSelectionData
*selection_data
);
573 gtk_wx_tree_model_get_type (void)
575 static GType tree_model_type
= 0;
577 if (!tree_model_type
)
579 const GTypeInfo tree_model_info
=
581 sizeof (GObjectClass
),
582 NULL
, /* base_init */
583 NULL
, /* base_finalize */
585 NULL
, /* class_finalize */
586 NULL
, /* class_data */
587 sizeof (GtkWxTreeModel
),
589 (GInstanceInitFunc
) wxgtk_tree_model_init
,
592 static const GInterfaceInfo tree_model_iface_info
=
594 (GInterfaceInitFunc
) wxgtk_tree_model_tree_model_init
,
599 static const GInterfaceInfo sortable_iface_info
=
601 (GInterfaceInitFunc
) wxgtk_tree_model_sortable_init
,
606 static const GInterfaceInfo drag_source_iface_info
=
608 (GInterfaceInitFunc
) wxgtk_tree_model_drag_source_init
,
613 static const GInterfaceInfo drag_dest_iface_info
=
615 (GInterfaceInitFunc
) wxgtk_tree_model_drag_dest_init
,
620 tree_model_type
= g_type_register_static (G_TYPE_OBJECT
, "GtkWxTreeModel",
621 &tree_model_info
, (GTypeFlags
)0 );
623 g_type_add_interface_static (tree_model_type
,
625 &tree_model_iface_info
);
626 g_type_add_interface_static (tree_model_type
,
627 GTK_TYPE_TREE_SORTABLE
,
628 &sortable_iface_info
);
629 g_type_add_interface_static (tree_model_type
,
630 GTK_TYPE_TREE_DRAG_DEST
,
631 &drag_dest_iface_info
);
632 g_type_add_interface_static (tree_model_type
,
633 GTK_TYPE_TREE_DRAG_SOURCE
,
634 &drag_source_iface_info
);
637 return tree_model_type
;
640 static GtkWxTreeModel
*
641 wxgtk_tree_model_new(void)
643 GtkWxTreeModel
*retval
= (GtkWxTreeModel
*) g_object_new (GTK_TYPE_WX_TREE_MODEL
, NULL
);
648 wxgtk_tree_model_tree_model_init (GtkTreeModelIface
*iface
)
650 iface
->get_flags
= wxgtk_tree_model_get_flags
;
651 iface
->get_n_columns
= wxgtk_tree_model_get_n_columns
;
652 iface
->get_column_type
= wxgtk_tree_model_get_column_type
;
653 iface
->get_iter
= wxgtk_tree_model_get_iter
;
654 iface
->get_path
= wxgtk_tree_model_get_path
;
655 iface
->get_value
= wxgtk_tree_model_get_value
;
656 iface
->iter_next
= wxgtk_tree_model_iter_next
;
657 iface
->iter_children
= wxgtk_tree_model_iter_children
;
658 iface
->iter_has_child
= wxgtk_tree_model_iter_has_child
;
659 iface
->iter_n_children
= wxgtk_tree_model_iter_n_children
;
660 iface
->iter_nth_child
= wxgtk_tree_model_iter_nth_child
;
661 iface
->iter_parent
= wxgtk_tree_model_iter_parent
;
665 wxgtk_tree_model_sortable_init (GtkTreeSortableIface
*iface
)
667 iface
->get_sort_column_id
= wxgtk_tree_model_get_sort_column_id
;
668 iface
->set_sort_column_id
= wxgtk_tree_model_set_sort_column_id
;
669 iface
->set_sort_func
= wxgtk_tree_model_set_sort_func
;
670 iface
->set_default_sort_func
= wxgtk_tree_model_set_default_sort_func
;
671 iface
->has_default_sort_func
= wxgtk_tree_model_has_default_sort_func
;
675 wxgtk_tree_model_drag_source_init(GtkTreeDragSourceIface
*iface
)
677 iface
->row_draggable
= wxgtk_tree_model_row_draggable
;
678 iface
->drag_data_delete
= wxgtk_tree_model_drag_data_delete
;
679 iface
->drag_data_get
= wxgtk_tree_model_drag_data_get
;
683 wxgtk_tree_model_drag_dest_init (GtkTreeDragDestIface
*iface
)
685 iface
->drag_data_received
= wxgtk_tree_model_drag_data_received
;
686 iface
->row_drop_possible
= wxgtk_tree_model_row_drop_possible
;
690 wxgtk_tree_model_init (GtkWxTreeModel
*tree_model
)
692 tree_model
->internal
= NULL
;
693 tree_model
->stamp
= g_random_int();
698 //-----------------------------------------------------------------------------
699 // implement callbacks from wxGtkTreeModel class by letting
700 // them call the methods of wxWidgets' wxDataViewModel
701 //-----------------------------------------------------------------------------
703 static GtkTreeModelFlags
704 wxgtk_tree_model_get_flags (GtkTreeModel
*tree_model
)
706 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
707 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), (GtkTreeModelFlags
)0 );
709 return wxtree_model
->internal
->get_flags();
713 wxgtk_tree_model_get_n_columns (GtkTreeModel
*tree_model
)
715 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
716 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), 0);
718 return wxtree_model
->internal
->GetDataViewModel()->GetColumnCount();
722 wxgtk_tree_model_get_column_type (GtkTreeModel
*tree_model
,
725 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
726 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), G_TYPE_INVALID
);
728 GType gtype
= G_TYPE_INVALID
;
730 wxString wxtype
= wxtree_model
->internal
->GetDataViewModel()->GetColumnType( (unsigned int) index
);
732 if (wxtype
== wxT("string"))
733 gtype
= G_TYPE_STRING
;
736 gtype
= G_TYPE_POINTER
;
737 // wxFAIL_MSG( wxT("non-string columns not supported for searching yet") );
744 wxgtk_tree_model_get_iter (GtkTreeModel
*tree_model
,
748 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
749 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
750 g_return_val_if_fail (gtk_tree_path_get_depth (path
) > 0, FALSE
);
752 return wxtree_model
->internal
->get_iter( iter
, path
);
756 wxgtk_tree_model_get_path (GtkTreeModel
*tree_model
,
759 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
760 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), NULL
);
761 g_return_val_if_fail (iter
->stamp
== GTK_WX_TREE_MODEL (wxtree_model
)->stamp
, NULL
);
763 return wxtree_model
->internal
->get_path( iter
);
767 wxgtk_tree_model_get_value (GtkTreeModel
*tree_model
,
772 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
773 g_return_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
) );
775 wxDataViewModel
*model
= wxtree_model
->internal
->GetDataViewModel();
776 wxString mtype
= model
->GetColumnType( (unsigned int) column
);
777 if (mtype
== wxT("string"))
780 g_value_init( value
, G_TYPE_STRING
);
781 wxDataViewItem
item( (void*) iter
->user_data
);
782 model
->GetValue( variant
, item
, (unsigned int) column
);
784 g_value_set_string( value
, variant
.GetString().utf8_str() );
788 wxFAIL_MSG( wxT("non-string columns not supported yet") );
793 wxgtk_tree_model_iter_next (GtkTreeModel
*tree_model
,
796 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
798 // This happens when clearing the view by calling .._set_model( NULL );
799 if (iter
->stamp
== 0) return FALSE
;
801 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
802 g_return_val_if_fail (wxtree_model
->stamp
== iter
->stamp
, FALSE
);
804 return wxtree_model
->internal
->iter_next( iter
);
808 wxgtk_tree_model_iter_children (GtkTreeModel
*tree_model
,
812 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
813 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
816 g_return_val_if_fail (wxtree_model
->stamp
== parent
->stamp
, FALSE
);
819 return wxtree_model
->internal
->iter_children( iter
, parent
);
823 wxgtk_tree_model_iter_has_child (GtkTreeModel
*tree_model
,
826 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
827 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
828 g_return_val_if_fail (wxtree_model
->stamp
== iter
->stamp
, FALSE
);
830 return wxtree_model
->internal
->iter_has_child( iter
);
834 wxgtk_tree_model_iter_n_children (GtkTreeModel
*tree_model
,
837 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
838 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), 0);
839 g_return_val_if_fail ( !iter
|| wxtree_model
->stamp
== iter
->stamp
, 0);
841 return wxtree_model
->internal
->iter_n_children( iter
);
845 wxgtk_tree_model_iter_nth_child (GtkTreeModel
*tree_model
,
850 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
851 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
853 return wxtree_model
->internal
->iter_nth_child( iter
, parent
, n
);
857 wxgtk_tree_model_iter_parent (GtkTreeModel
*tree_model
,
861 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) tree_model
;
862 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
863 g_return_val_if_fail (wxtree_model
->stamp
== child
->stamp
, FALSE
);
865 return wxtree_model
->internal
->iter_parent( iter
, child
);
868 /* drag'n'drop iface */
870 wxgtk_tree_model_row_draggable (GtkTreeDragSource
*drag_source
,
873 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) drag_source
;
874 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
876 return wxtree_model
->internal
->row_draggable( drag_source
, path
);
880 wxgtk_tree_model_drag_data_delete (GtkTreeDragSource
*drag_source
,
883 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) drag_source
;
884 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
886 return wxtree_model
->internal
->drag_data_delete( drag_source
, path
);
890 wxgtk_tree_model_drag_data_get (GtkTreeDragSource
*drag_source
,
892 GtkSelectionData
*selection_data
)
894 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) drag_source
;
895 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
898 wxPrintf( "drag_get_data\n");
900 wxGtkString
atom_selection(gdk_atom_name(selection_data
->selection
));
901 wxPrintf( "selection %s\n", wxString::FromAscii(atom_selection
) );
903 wxGtkString
atom_target(gdk_atom_name(selection_data
->target
));
904 wxPrintf( "target %s\n", wxString::FromAscii(atom_target
) );
906 wxGtkString
atom_type(gdk_atom_name(selection_data
->type
));
907 wxPrintf( "type %s\n", wxString::FromAscii(atom_type
) );
909 wxPrintf( "format %d\n", selection_data
->format
);
912 return wxtree_model
->internal
->drag_data_get( drag_source
, path
, selection_data
);
916 wxgtk_tree_model_drag_data_received (GtkTreeDragDest
*drag_dest
,
918 GtkSelectionData
*selection_data
)
920 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) drag_dest
;
921 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
923 return wxtree_model
->internal
->drag_data_received( drag_dest
, dest
, selection_data
);
927 wxgtk_tree_model_row_drop_possible (GtkTreeDragDest
*drag_dest
,
928 GtkTreePath
*dest_path
,
929 GtkSelectionData
*selection_data
)
931 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) drag_dest
;
932 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (wxtree_model
), FALSE
);
934 return wxtree_model
->internal
->row_drop_possible( drag_dest
, dest_path
, selection_data
);
939 wxgtk_tree_model_get_sort_column_id (GtkTreeSortable
*sortable
,
940 gint
*sort_column_id
,
943 GtkWxTreeModel
*wxtree_model
= (GtkWxTreeModel
*) sortable
;
945 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (sortable
), FALSE
);
947 if (!wxtree_model
->internal
->IsSorted())
950 *sort_column_id
= -1;
957 *sort_column_id
= wxtree_model
->internal
->GetSortColumn();
960 *order
= wxtree_model
->internal
->GetSortOrder();
965 wxDataViewColumn
*gs_lastLeftClickHeader
= NULL
;
968 wxgtk_tree_model_set_sort_column_id (GtkTreeSortable
*sortable
,
972 GtkWxTreeModel
*tree_model
= (GtkWxTreeModel
*) sortable
;
973 g_return_if_fail (GTK_IS_WX_TREE_MODEL (sortable
) );
975 tree_model
->internal
->SetDataViewSortColumn( gs_lastLeftClickHeader
);
977 if ((sort_column_id
!= (gint
) tree_model
->internal
->GetSortColumn()) ||
978 (order
!= tree_model
->internal
->GetSortOrder()))
980 tree_model
->internal
->SetSortColumn( sort_column_id
);
981 tree_model
->internal
->SetSortOrder( order
);
983 gtk_tree_sortable_sort_column_changed (sortable
);
985 tree_model
->internal
->GetDataViewModel()->Resort();
988 if (gs_lastLeftClickHeader
)
990 wxDataViewCtrl
*dv
= tree_model
->internal
->GetOwner();
991 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_COLUMN_SORTED
, dv
->GetId() );
992 event
.SetDataViewColumn( gs_lastLeftClickHeader
);
993 event
.SetModel( dv
->GetModel() );
994 dv
->HandleWindowEvent( event
);
997 gs_lastLeftClickHeader
= NULL
;
1001 wxgtk_tree_model_set_sort_func (GtkTreeSortable
*sortable
,
1002 gint
WXUNUSED(sort_column_id
),
1003 GtkTreeIterCompareFunc func
,
1004 gpointer
WXUNUSED(data
),
1005 GDestroyNotify
WXUNUSED(destroy
))
1007 g_return_if_fail (GTK_IS_WX_TREE_MODEL (sortable
) );
1008 g_return_if_fail (func
!= NULL
);
1012 wxgtk_tree_model_set_default_sort_func (GtkTreeSortable
*sortable
,
1013 GtkTreeIterCompareFunc func
,
1014 gpointer
WXUNUSED(data
),
1015 GDestroyNotify
WXUNUSED(destroy
))
1017 g_return_if_fail (GTK_IS_WX_TREE_MODEL (sortable
) );
1018 g_return_if_fail (func
!= NULL
);
1020 //wxPrintf( "wxgtk_tree_model_set_default_sort_func\n" );
1021 // TODO: remove this code
1025 wxgtk_tree_model_has_default_sort_func (GtkTreeSortable
*sortable
)
1027 g_return_val_if_fail (GTK_IS_WX_TREE_MODEL (sortable
), FALSE
);
1032 //-----------------------------------------------------------------------------
1033 // define new GTK+ class GtkWxRendererText
1034 //-----------------------------------------------------------------------------
1038 #define GTK_TYPE_WX_CELL_RENDERER_TEXT (gtk_wx_cell_renderer_text_get_type ())
1039 #define GTK_WX_CELL_RENDERER_TEXT(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_CELL_RENDERER_TEXT, GtkWxCellRendererText))
1040 #define GTK_IS_WX_CELL_RENDERER_TEXT(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_CELL_RENDERER_TEXT))
1041 #define GTK_IS_WX_CELL_RENDERER_TEXT_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_CELL_RENDERER_TEXT))
1043 GType
gtk_wx_cell_renderer_text_get_type (void);
1045 typedef struct _GtkWxCellRendererText GtkWxCellRendererText
;
1047 struct _GtkWxCellRendererText
1049 GtkCellRendererText parent
;
1051 wxDataViewRenderer
*wx_renderer
;
1054 static GtkWxCellRendererText
*gtk_wx_cell_renderer_text_new (void);
1055 static void gtk_wx_cell_renderer_text_init (
1056 GtkWxCellRendererText
*cell
);
1057 static void gtk_wx_cell_renderer_text_class_init(
1058 void* klass
, void*);
1059 static GtkCellEditable
*gtk_wx_cell_renderer_text_start_editing(
1060 GtkCellRenderer
*cell
,
1064 GdkRectangle
*background_area
,
1065 GdkRectangle
*cell_area
,
1066 GtkCellRendererState flags
);
1069 static GObjectClass
*text_cell_parent_class
= NULL
;
1074 gtk_wx_cell_renderer_text_get_type (void)
1076 static GType cell_wx_type
= 0;
1080 const GTypeInfo cell_wx_info
=
1082 sizeof (GtkCellRendererTextClass
),
1083 NULL
, /* base_init */
1084 NULL
, /* base_finalize */
1085 gtk_wx_cell_renderer_text_class_init
,
1086 NULL
, /* class_finalize */
1087 NULL
, /* class_data */
1088 sizeof (GtkWxCellRendererText
),
1089 0, /* n_preallocs */
1090 (GInstanceInitFunc
) gtk_wx_cell_renderer_text_init
,
1093 cell_wx_type
= g_type_register_static( GTK_TYPE_CELL_RENDERER_TEXT
,
1094 "GtkWxCellRendererText", &cell_wx_info
, (GTypeFlags
)0 );
1097 return cell_wx_type
;
1101 gtk_wx_cell_renderer_text_init (GtkWxCellRendererText
*cell
)
1103 cell
->wx_renderer
= NULL
;
1107 gtk_wx_cell_renderer_text_class_init(void* klass
, void*)
1109 GtkCellRendererClass
*cell_class
= GTK_CELL_RENDERER_CLASS (klass
);
1111 text_cell_parent_class
= (GObjectClass
*) g_type_class_peek_parent (klass
);
1113 cell_class
->start_editing
= gtk_wx_cell_renderer_text_start_editing
;
1116 GtkWxCellRendererText
*
1117 gtk_wx_cell_renderer_text_new (void)
1119 return (GtkWxCellRendererText
*) g_object_new (GTK_TYPE_WX_CELL_RENDERER_TEXT
, NULL
);
1122 static GtkCellEditable
*gtk_wx_cell_renderer_text_start_editing(
1123 GtkCellRenderer
*gtk_renderer
,
1124 GdkEvent
*gdk_event
,
1127 GdkRectangle
*background_area
,
1128 GdkRectangle
*cell_area
,
1129 GtkCellRendererState flags
)
1131 GtkWxCellRendererText
*wxgtk_renderer
= (GtkWxCellRendererText
*) gtk_renderer
;
1132 wxDataViewRenderer
*wx_renderer
= wxgtk_renderer
->wx_renderer
;
1133 wxDataViewColumn
*column
= wx_renderer
->GetOwner();
1136 item(column
->GetOwner()->GTKPathToItem(wxGtkTreePath(path
)));
1138 wxDataViewCtrl
*dv
= column
->GetOwner();
1139 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_START_EDITING
, dv
->GetId() );
1140 event
.SetDataViewColumn( column
);
1141 event
.SetModel( dv
->GetModel() );
1142 event
.SetColumn( column
->GetModelColumn() );
1143 event
.SetItem( item
);
1144 dv
->HandleWindowEvent( event
);
1146 if (event
.IsAllowed())
1147 return GTK_CELL_RENDERER_CLASS(text_cell_parent_class
)->
1148 start_editing( gtk_renderer
, gdk_event
, widget
, path
, background_area
, cell_area
, flags
);
1153 //-----------------------------------------------------------------------------
1154 // define new GTK+ class GtkWxCellRenderer
1155 //-----------------------------------------------------------------------------
1159 #define GTK_TYPE_WX_CELL_RENDERER (gtk_wx_cell_renderer_get_type ())
1160 #define GTK_WX_CELL_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRenderer))
1161 #define GTK_IS_WX_CELL_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_CELL_RENDERER))
1162 #define GTK_IS_WX_CELL_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_CELL_RENDERER))
1164 GType
gtk_wx_cell_renderer_get_type (void);
1166 typedef struct _GtkWxCellRenderer GtkWxCellRenderer
;
1168 struct _GtkWxCellRenderer
1170 GtkCellRenderer parent
;
1173 wxDataViewCustomRenderer
*cell
;
1176 static GtkCellRenderer
*gtk_wx_cell_renderer_new (void);
1177 static void gtk_wx_cell_renderer_init (
1178 GtkWxCellRenderer
*cell
);
1179 static void gtk_wx_cell_renderer_class_init(
1180 void* klass
, void*);
1181 static void gtk_wx_cell_renderer_get_size (
1182 GtkCellRenderer
*cell
,
1184 GdkRectangle
*rectangle
,
1189 static void gtk_wx_cell_renderer_render (
1190 GtkCellRenderer
*cell
,
1193 GdkRectangle
*background_area
,
1194 GdkRectangle
*cell_area
,
1195 GdkRectangle
*expose_area
,
1196 GtkCellRendererState flags
);
1197 static gboolean
gtk_wx_cell_renderer_activate(
1198 GtkCellRenderer
*cell
,
1202 GdkRectangle
*background_area
,
1203 GdkRectangle
*cell_area
,
1204 GtkCellRendererState flags
);
1205 static GtkCellEditable
*gtk_wx_cell_renderer_start_editing(
1206 GtkCellRenderer
*cell
,
1210 GdkRectangle
*background_area
,
1211 GdkRectangle
*cell_area
,
1212 GtkCellRendererState flags
);
1217 gtk_wx_cell_renderer_get_type (void)
1219 static GType cell_wx_type
= 0;
1223 const GTypeInfo cell_wx_info
=
1225 sizeof (GtkCellRendererClass
),
1226 NULL
, /* base_init */
1227 NULL
, /* base_finalize */
1228 gtk_wx_cell_renderer_class_init
,
1229 NULL
, /* class_finalize */
1230 NULL
, /* class_data */
1231 sizeof (GtkWxCellRenderer
),
1232 0, /* n_preallocs */
1233 (GInstanceInitFunc
) gtk_wx_cell_renderer_init
,
1236 cell_wx_type
= g_type_register_static( GTK_TYPE_CELL_RENDERER
,
1237 "GtkWxCellRenderer", &cell_wx_info
, (GTypeFlags
)0 );
1240 return cell_wx_type
;
1244 gtk_wx_cell_renderer_init (GtkWxCellRenderer
*cell
)
1250 gtk_wx_cell_renderer_class_init(void* klass
, void*)
1252 GtkCellRendererClass
*cell_class
= GTK_CELL_RENDERER_CLASS (klass
);
1254 cell_class
->get_size
= gtk_wx_cell_renderer_get_size
;
1255 cell_class
->render
= gtk_wx_cell_renderer_render
;
1256 cell_class
->activate
= gtk_wx_cell_renderer_activate
;
1257 cell_class
->start_editing
= gtk_wx_cell_renderer_start_editing
;
1261 gtk_wx_cell_renderer_new (void)
1263 return (GtkCellRenderer
*) g_object_new (GTK_TYPE_WX_CELL_RENDERER
, NULL
);
1266 static GtkCellEditable
*gtk_wx_cell_renderer_start_editing(
1267 GtkCellRenderer
*renderer
,
1268 GdkEvent
*WXUNUSED(event
),
1271 GdkRectangle
*WXUNUSED(background_area
),
1272 GdkRectangle
*cell_area
,
1273 GtkCellRendererState
WXUNUSED(flags
) )
1275 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
1276 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
1278 // Renderer doesn't support in-place editing
1279 if (!cell
->HasEditorCtrl())
1282 // An in-place editing control is still around
1283 if (cell
->GetEditorCtrl())
1287 gtk_wx_cell_renderer_get_size (renderer
, widget
, cell_area
,
1293 rect
.x
+= cell_area
->x
;
1294 rect
.y
+= cell_area
->y
;
1295 // rect.width -= renderer->xpad * 2;
1296 // rect.height -= renderer->ypad * 2;
1298 // wxRect renderrect(wxRectFromGDKRect(&rect));
1299 wxRect
renderrect(wxRectFromGDKRect(cell_area
));
1302 item(cell
->GetOwner()->GetOwner()->GTKPathToItem(wxGtkTreePath(path
)));
1304 cell
->StartEditing( item
, renderrect
);
1310 gtk_wx_cell_renderer_get_size (GtkCellRenderer
*renderer
,
1311 GtkWidget
*WXUNUSED(widget
),
1312 GdkRectangle
*cell_area
,
1318 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
1319 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
1321 wxSize size
= cell
->GetSize();
1323 wxDataViewCtrl
* const ctrl
= cell
->GetOwner()->GetOwner();
1325 // Uniform row height, if specified, overrides the value returned by the
1327 if ( !ctrl
->HasFlag(wxDV_VARIABLE_LINE_HEIGHT
) )
1329 const int uniformHeight
= ctrl
->GTKGetUniformRowHeight();
1330 if ( uniformHeight
> 0 )
1331 size
.y
= uniformHeight
;
1335 gtk_cell_renderer_get_padding(renderer
, &xpad
, &ypad
);
1336 int calc_width
= xpad
* 2 + size
.x
;
1337 int calc_height
= ypad
* 2 + size
.y
;
1344 if (cell_area
&& size
.x
> 0 && size
.y
> 0)
1346 float xalign
, yalign
;
1347 gtk_cell_renderer_get_alignment(renderer
, &xalign
, &yalign
);
1350 *x_offset
= int(xalign
* (cell_area
->width
- calc_width
- 2 * xpad
));
1351 *x_offset
= MAX(*x_offset
, 0) + xpad
;
1355 *y_offset
= int(yalign
* (cell_area
->height
- calc_height
- 2 * ypad
));
1356 *y_offset
= MAX(*y_offset
, 0) + ypad
;
1361 *width
= calc_width
;
1364 *height
= calc_height
;
1368 gtk_wx_cell_renderer_render (GtkCellRenderer
*renderer
,
1371 GdkRectangle
*background_area
,
1372 GdkRectangle
*cell_area
,
1373 GdkRectangle
*expose_area
,
1374 GtkCellRendererState flags
)
1377 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
1378 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
1380 cell
->GTKStashRenderParams(window
, widget
,
1381 background_area
, expose_area
, flags
);
1383 wxRect
rect(wxRectFromGDKRect(cell_area
));
1385 gtk_cell_renderer_get_padding(renderer
, &xpad
, &ypad
);
1386 rect
= rect
.Deflate(xpad
, ypad
);
1388 wxWindowDC
* dc
= (wxWindowDC
*) cell
->GetDC();
1389 wxWindowDCImpl
*impl
= (wxWindowDCImpl
*) dc
->GetImpl();
1391 // Reinitialize wxWindowDC's GDK window if drawing occurs into a different
1392 // window such as a DnD drop window.
1393 if (window
!= impl
->m_gdkwindow
)
1396 impl
->m_gdkwindow
= window
;
1401 if (flags
& GTK_CELL_RENDERER_SELECTED
)
1402 state
|= wxDATAVIEW_CELL_SELECTED
;
1403 if (flags
& GTK_CELL_RENDERER_PRELIT
)
1404 state
|= wxDATAVIEW_CELL_PRELIT
;
1405 if (flags
& GTK_CELL_RENDERER_INSENSITIVE
)
1406 state
|= wxDATAVIEW_CELL_INSENSITIVE
;
1407 if (flags
& GTK_CELL_RENDERER_INSENSITIVE
)
1408 state
|= wxDATAVIEW_CELL_INSENSITIVE
;
1409 if (flags
& GTK_CELL_RENDERER_FOCUSED
)
1410 state
|= wxDATAVIEW_CELL_FOCUSED
;
1411 cell
->WXCallRender( rect
, dc
, state
);
1415 gtk_wx_cell_renderer_activate(
1416 GtkCellRenderer
*renderer
,
1420 GdkRectangle
*WXUNUSED(background_area
),
1421 GdkRectangle
*cell_area
,
1422 GtkCellRendererState
WXUNUSED(flags
) )
1424 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
1425 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
1428 gtk_wx_cell_renderer_get_size (renderer
, widget
, cell_area
,
1434 rect
.x
+= cell_area
->x
;
1435 rect
.y
+= cell_area
->y
;
1437 gtk_cell_renderer_get_padding(renderer
, &xpad
, &ypad
);
1438 rect
.width
-= xpad
* 2;
1439 rect
.height
-= ypad
* 2;
1441 wxRect
renderrect(wxRectFromGDKRect(&rect
));
1443 wxDataViewCtrl
* const ctrl
= cell
->GetOwner()->GetOwner();
1444 wxDataViewModel
*model
= ctrl
->GetModel();
1446 wxDataViewItem
item(ctrl
->GTKPathToItem(wxGtkTreePath(path
)));
1448 unsigned int model_col
= cell
->GetOwner()->GetModelColumn();
1452 // activated by <ENTER>
1453 return cell
->ActivateCell(renderrect
, model
, item
, model_col
, NULL
);
1455 else if ( event
->type
== GDK_BUTTON_PRESS
)
1457 GdkEventButton
*button_event
= (GdkEventButton
*)event
;
1458 if ( button_event
->button
== 1 )
1460 wxMouseEvent
mouse_event(wxEVT_LEFT_DOWN
);
1461 InitMouseEvent(ctrl
, mouse_event
, button_event
);
1463 mouse_event
.m_x
-= renderrect
.x
;
1464 mouse_event
.m_y
-= renderrect
.y
;
1466 return cell
->ActivateCell(renderrect
, model
, item
, model_col
, &mouse_event
);
1470 wxLogDebug("unexpected event type in gtk_wx_cell_renderer_activate()");
1474 // ---------------------------------------------------------
1475 // wxGtkDataViewModelNotifier
1476 // ---------------------------------------------------------
1478 class wxGtkDataViewModelNotifier
: public wxDataViewModelNotifier
1481 wxGtkDataViewModelNotifier( wxDataViewModel
*wx_model
, wxDataViewCtrlInternal
*internal
);
1482 ~wxGtkDataViewModelNotifier();
1484 virtual bool ItemAdded( const wxDataViewItem
&parent
, const wxDataViewItem
&item
);
1485 virtual bool ItemDeleted( const wxDataViewItem
&parent
, const wxDataViewItem
&item
);
1486 virtual bool ItemChanged( const wxDataViewItem
&item
);
1487 virtual bool ValueChanged( const wxDataViewItem
&item
, unsigned int model_column
);
1488 virtual bool Cleared();
1489 virtual void Resort();
1490 virtual bool BeforeReset();
1491 virtual bool AfterReset();
1493 void UpdateLastCount();
1496 wxDataViewModel
*m_wx_model
;
1497 wxDataViewCtrlInternal
*m_internal
;
1500 // ---------------------------------------------------------
1501 // wxGtkDataViewListModelNotifier
1502 // ---------------------------------------------------------
1504 wxGtkDataViewModelNotifier::wxGtkDataViewModelNotifier(
1505 wxDataViewModel
*wx_model
, wxDataViewCtrlInternal
*internal
)
1507 m_wx_model
= wx_model
;
1508 m_internal
= internal
;
1511 wxGtkDataViewModelNotifier::~wxGtkDataViewModelNotifier()
1517 bool wxGtkDataViewModelNotifier::ItemAdded( const wxDataViewItem
&parent
, const wxDataViewItem
&item
)
1519 m_internal
->ItemAdded( parent
, item
);
1520 GtkWxTreeModel
*wxgtk_model
= m_internal
->GetGtkModel();
1523 iter
.stamp
= wxgtk_model
->stamp
;
1524 iter
.user_data
= item
.GetID();
1526 wxGtkTreePath
path(wxgtk_tree_model_get_path(
1527 GTK_TREE_MODEL(wxgtk_model
), &iter
));
1528 gtk_tree_model_row_inserted(
1529 GTK_TREE_MODEL(wxgtk_model
), path
, &iter
);
1534 bool wxGtkDataViewModelNotifier::ItemDeleted( const wxDataViewItem
&parent
, const wxDataViewItem
&item
)
1536 GtkWxTreeModel
*wxgtk_model
= m_internal
->GetGtkModel();
1538 // using _get_path for a deleted item cannot be
1541 iter
.stamp
= wxgtk_model
->stamp
;
1542 iter
.user_data
= (gpointer
) item
.GetID();
1543 wxGtkTreePath
path(wxgtk_tree_model_get_path(
1544 GTK_TREE_MODEL(wxgtk_model
), &iter
));
1546 // so get the path from the parent
1547 GtkTreeIter parentIter
;
1548 parentIter
.stamp
= wxgtk_model
->stamp
;
1549 parentIter
.user_data
= (gpointer
) parent
.GetID();
1550 wxGtkTreePath
parentPath(wxgtk_tree_model_get_path(
1551 GTK_TREE_MODEL(wxgtk_model
), &parentIter
));
1553 // and add the final index ourselves
1554 wxGtkTreePath
path(gtk_tree_path_copy(parentPath
));
1555 int index
= m_internal
->GetIndexOf( parent
, item
);
1556 gtk_tree_path_append_index( path
, index
);
1559 gtk_tree_model_row_deleted(
1560 GTK_TREE_MODEL(wxgtk_model
), path
);
1562 m_internal
->ItemDeleted( parent
, item
);
1564 // Did we remove the last child, causing 'parent' to become a leaf?
1565 if ( !m_wx_model
->IsContainer(parent
) )
1567 gtk_tree_model_row_has_child_toggled
1569 GTK_TREE_MODEL(wxgtk_model
),
1578 void wxGtkDataViewModelNotifier::Resort()
1580 m_internal
->Resort();
1583 bool wxGtkDataViewModelNotifier::ItemChanged( const wxDataViewItem
&item
)
1585 GtkWxTreeModel
*wxgtk_model
= m_internal
->GetGtkModel();
1588 iter
.stamp
= wxgtk_model
->stamp
;
1589 iter
.user_data
= (gpointer
) item
.GetID();
1591 wxGtkTreePath
path(wxgtk_tree_model_get_path(
1592 GTK_TREE_MODEL(wxgtk_model
), &iter
));
1593 gtk_tree_model_row_changed(
1594 GTK_TREE_MODEL(wxgtk_model
), path
, &iter
);
1596 m_internal
->ItemChanged( item
);
1601 bool wxGtkDataViewModelNotifier::ValueChanged( const wxDataViewItem
&item
, unsigned int model_column
)
1603 GtkWxTreeModel
*wxgtk_model
= m_internal
->GetGtkModel();
1604 wxDataViewCtrl
*ctrl
= m_internal
->GetOwner();
1606 // This adds GTK+'s missing MVC logic for ValueChanged
1608 for (index
= 0; index
< ctrl
->GetColumnCount(); index
++)
1610 wxDataViewColumn
*column
= ctrl
->GetColumn( index
);
1611 if (column
->GetModelColumn() == model_column
)
1613 GtkTreeView
*widget
= GTK_TREE_VIEW(ctrl
->GtkGetTreeView());
1614 GtkTreeViewColumn
*gcolumn
= GTK_TREE_VIEW_COLUMN(column
->GetGtkHandle());
1616 // Don't attempt to refresh not yet realized tree, it is useless
1617 // and results in GTK errors.
1618 if ( gtk_widget_get_realized(ctrl
->GtkGetTreeView()) )
1622 iter
.stamp
= wxgtk_model
->stamp
;
1623 iter
.user_data
= (gpointer
) item
.GetID();
1624 wxGtkTreePath
path(wxgtk_tree_model_get_path(
1625 GTK_TREE_MODEL(wxgtk_model
), &iter
));
1626 GdkRectangle cell_area
;
1627 gtk_tree_view_get_cell_area( widget
, path
, gcolumn
, &cell_area
);
1629 GtkAdjustment
* hadjust
= gtk_tree_view_get_hadjustment( widget
);
1630 double d
= gtk_adjustment_get_value( hadjust
);
1631 int xdiff
= (int) d
;
1633 int ydiff
= gcolumn
->button
->allocation
.height
;
1635 gtk_widget_queue_draw_area( GTK_WIDGET(widget
),
1636 cell_area
.x
- xdiff
, ydiff
+ cell_area
.y
, cell_area
.width
, cell_area
.height
);
1639 m_internal
->ValueChanged( item
, model_column
);
1648 bool wxGtkDataViewModelNotifier::BeforeReset()
1650 GtkWidget
*treeview
= m_internal
->GetOwner()->GtkGetTreeView();
1651 gtk_tree_view_set_model( GTK_TREE_VIEW(treeview
), NULL
);
1656 bool wxGtkDataViewModelNotifier::AfterReset()
1658 GtkWidget
*treeview
= m_internal
->GetOwner()->GtkGetTreeView();
1659 GtkWxTreeModel
*wxgtk_model
= m_internal
->GetGtkModel();
1661 m_internal
->Cleared();
1663 gtk_tree_view_set_model( GTK_TREE_VIEW(treeview
), GTK_TREE_MODEL(wxgtk_model
) );
1668 bool wxGtkDataViewModelNotifier::Cleared()
1670 GtkWxTreeModel
*wxgtk_model
= m_internal
->GetGtkModel();
1672 // There is no call to tell the model that everything
1673 // has been deleted so call row_deleted() for every
1676 int count
= m_internal
->iter_n_children( NULL
); // number of children of root
1678 GtkTreePath
*path
= gtk_tree_path_new_first(); // points to root
1681 for (i
= 0; i
< count
; i
++)
1682 gtk_tree_model_row_deleted( GTK_TREE_MODEL(wxgtk_model
), path
);
1684 gtk_tree_path_free( path
);
1686 m_internal
->Cleared();
1691 // ---------------------------------------------------------
1692 // wxDataViewRenderer
1693 // ---------------------------------------------------------
1695 static gpointer s_user_data
= NULL
;
1698 wxgtk_cell_editable_editing_done( GtkCellEditable
*WXUNUSED(editable
),
1699 wxDataViewRenderer
*wxrenderer
)
1701 wxDataViewColumn
*column
= wxrenderer
->GetOwner();
1702 wxDataViewCtrl
*dv
= column
->GetOwner();
1703 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_EDITING_DONE
, dv
->GetId() );
1704 event
.SetDataViewColumn( column
);
1705 event
.SetModel( dv
->GetModel() );
1706 wxDataViewItem
item( s_user_data
);
1707 event
.SetItem( item
);
1708 dv
->HandleWindowEvent( event
);
1712 wxgtk_renderer_editing_started( GtkCellRenderer
*WXUNUSED(cell
), GtkCellEditable
*editable
,
1713 gchar
*path
, wxDataViewRenderer
*wxrenderer
)
1718 wxDataViewColumn
*column
= wxrenderer
->GetOwner();
1719 wxDataViewCtrl
*dv
= column
->GetOwner();
1720 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_EDITING_STARTED
, dv
->GetId() );
1721 event
.SetDataViewColumn( column
);
1722 event
.SetModel( dv
->GetModel() );
1723 wxDataViewItem
item(dv
->GTKPathToItem(wxGtkTreePath(path
)));
1724 event
.SetItem( item
);
1725 dv
->HandleWindowEvent( event
);
1727 if (GTK_IS_CELL_EDITABLE(editable
))
1729 s_user_data
= item
.GetID();
1731 g_signal_connect (GTK_CELL_EDITABLE (editable
), "editing_done",
1732 G_CALLBACK (wxgtk_cell_editable_editing_done
),
1733 (gpointer
) wxrenderer
);
1739 IMPLEMENT_ABSTRACT_CLASS(wxDataViewRenderer
, wxDataViewRendererBase
)
1741 wxDataViewRenderer::wxDataViewRenderer( const wxString
&varianttype
, wxDataViewCellMode mode
,
1743 wxDataViewRendererBase( varianttype
, mode
, align
)
1748 // we haven't changed them yet
1749 m_usingDefaultAttrs
= true;
1751 // NOTE: SetMode() and SetAlignment() needs to be called in the renderer's ctor,
1752 // after the m_renderer pointer has been initialized
1755 void wxDataViewRenderer::GtkPackIntoColumn(GtkTreeViewColumn
*column
)
1757 gtk_tree_view_column_pack_end( column
, m_renderer
, TRUE
/* expand */);
1760 void wxDataViewRenderer::GtkInitHandlers()
1762 if (!gtk_check_version(2,6,0))
1764 g_signal_connect (GTK_CELL_RENDERER(m_renderer
), "editing_started",
1765 G_CALLBACK (wxgtk_renderer_editing_started
),
1770 void wxDataViewRenderer::SetMode( wxDataViewCellMode mode
)
1772 GtkCellRendererMode gtkMode
;
1775 case wxDATAVIEW_CELL_INERT
:
1776 gtkMode
= GTK_CELL_RENDERER_MODE_INERT
;
1779 case wxDATAVIEW_CELL_ACTIVATABLE
:
1780 gtkMode
= GTK_CELL_RENDERER_MODE_ACTIVATABLE
;
1783 case wxDATAVIEW_CELL_EDITABLE
:
1784 gtkMode
= GTK_CELL_RENDERER_MODE_EDITABLE
;
1788 wxFAIL_MSG( "unknown wxDataViewCellMode value" );
1794 // This value is most often ignored in GtkTreeView
1795 GValue gvalue
= { 0, };
1796 g_value_init( &gvalue
, gtk_cell_renderer_mode_get_type() );
1797 g_value_set_enum( &gvalue
, gtkMode
);
1798 g_object_set_property( G_OBJECT(m_renderer
), "mode", &gvalue
);
1799 g_value_unset( &gvalue
);
1802 wxDataViewCellMode
wxDataViewRenderer::GetMode() const
1804 wxDataViewCellMode ret
;
1807 g_object_get( G_OBJECT(m_renderer
), "mode", &gvalue
, NULL
);
1809 switch (g_value_get_enum(&gvalue
))
1812 wxFAIL_MSG( "unknown GtkCellRendererMode value" );
1813 // fall through (we have to return something)
1815 case GTK_CELL_RENDERER_MODE_INERT
:
1816 ret
= wxDATAVIEW_CELL_INERT
;
1819 case GTK_CELL_RENDERER_MODE_ACTIVATABLE
:
1820 ret
= wxDATAVIEW_CELL_ACTIVATABLE
;
1823 case GTK_CELL_RENDERER_MODE_EDITABLE
:
1824 ret
= wxDATAVIEW_CELL_EDITABLE
;
1828 g_value_unset( &gvalue
);
1833 void wxDataViewRenderer::GtkApplyAlignment(GtkCellRenderer
*renderer
)
1835 int align
= m_alignment
;
1837 // query alignment from column ?
1841 if (GetOwner() == NULL
)
1844 align
= GetOwner()->GetAlignment();
1845 align
|= wxALIGN_CENTRE_VERTICAL
;
1848 // horizontal alignment:
1850 gfloat xalign
= 0.0;
1851 if (align
& wxALIGN_RIGHT
)
1853 else if (align
& wxALIGN_CENTER_HORIZONTAL
)
1856 GValue gvalue
= { 0, };
1857 g_value_init( &gvalue
, G_TYPE_FLOAT
);
1858 g_value_set_float( &gvalue
, xalign
);
1859 g_object_set_property( G_OBJECT(renderer
), "xalign", &gvalue
);
1860 g_value_unset( &gvalue
);
1862 // vertical alignment:
1864 gfloat yalign
= 0.0;
1865 if (align
& wxALIGN_BOTTOM
)
1867 else if (align
& wxALIGN_CENTER_VERTICAL
)
1870 GValue gvalue2
= { 0, };
1871 g_value_init( &gvalue2
, G_TYPE_FLOAT
);
1872 g_value_set_float( &gvalue2
, yalign
);
1873 g_object_set_property( G_OBJECT(renderer
), "yalign", &gvalue2
);
1874 g_value_unset( &gvalue2
);
1877 void wxDataViewRenderer::SetAlignment( int align
)
1879 m_alignment
= align
;
1880 GtkUpdateAlignment();
1883 int wxDataViewRenderer::GetAlignment() const
1888 void wxDataViewRenderer::EnableEllipsize(wxEllipsizeMode mode
)
1891 if ( gtk_check_version(2, 6, 0) != NULL
)
1894 GtkCellRendererText
* const rend
= GtkGetTextRenderer();
1898 // we use the same values in wxEllipsizeMode as PangoEllipsizeMode so we
1899 // can just cast between them
1900 GValue gvalue
= { 0, };
1901 g_value_init( &gvalue
, PANGO_TYPE_ELLIPSIZE_MODE
);
1902 g_value_set_enum( &gvalue
, static_cast<PangoEllipsizeMode
>(mode
) );
1903 g_object_set_property( G_OBJECT(rend
), "ellipsize", &gvalue
);
1904 g_value_unset( &gvalue
);
1907 #endif // GTK 2.6/before
1910 wxEllipsizeMode
wxDataViewRenderer::GetEllipsizeMode() const
1913 if ( gtk_check_version(2, 6, 0) != NULL
)
1914 return wxELLIPSIZE_NONE
;
1916 GtkCellRendererText
* const rend
= GtkGetTextRenderer();
1918 return wxELLIPSIZE_NONE
;
1920 GValue gvalue
= { 0, };
1921 g_value_init( &gvalue
, PANGO_TYPE_ELLIPSIZE_MODE
);
1922 g_object_get_property( G_OBJECT(rend
), "ellipsize", &gvalue
);
1924 mode
= static_cast<wxEllipsizeMode
>(g_value_get_enum( &gvalue
));
1925 g_value_unset( &gvalue
);
1929 return wxELLIPSIZE_NONE
;
1930 #endif // GTK 2.6/before
1934 wxDataViewRenderer::GtkOnTextEdited(const char *itempath
, const wxString
& str
)
1936 wxVariant
value(str
);
1937 if (!Validate( value
))
1941 item(GetOwner()->GetOwner()->GTKPathToItem(wxGtkTreePath(itempath
)));
1943 GtkOnCellChanged(value
, item
, GetOwner()->GetModelColumn());
1947 wxDataViewRenderer::GtkOnCellChanged(const wxVariant
& value
,
1948 const wxDataViewItem
& item
,
1951 wxDataViewModel
*model
= GetOwner()->GetOwner()->GetModel();
1952 model
->ChangeValue( value
, item
, col
);
1955 // ---------------------------------------------------------
1956 // wxDataViewTextRenderer
1957 // ---------------------------------------------------------
1962 static void wxGtkTextRendererEditedCallback( GtkCellRendererText
*WXUNUSED(renderer
),
1963 gchar
*arg1
, gchar
*arg2
, gpointer user_data
)
1965 wxDataViewRenderer
*cell
= (wxDataViewRenderer
*) user_data
;
1967 cell
->GtkOnTextEdited(arg1
, wxGTK_CONV_BACK_FONT(
1968 arg2
, cell
->GetOwner()->GetOwner()->GetFont()));
1976 // helper function used by wxDataViewTextRenderer and
1977 // wxDataViewCustomRenderer::RenderText(): it applies the attributes to the
1978 // given text renderer and returns true if anything was done
1979 bool GtkApplyAttr(GtkCellRendererText
*renderer
, const wxDataViewItemAttr
& attr
)
1981 bool usingDefaultAttrs
= true;
1982 if (attr
.HasColour())
1984 const GdkColor
* const gcol
= attr
.GetColour().GetColor();
1986 GValue gvalue
= { 0, };
1987 g_value_init( &gvalue
, GDK_TYPE_COLOR
);
1988 g_value_set_boxed( &gvalue
, gcol
);
1989 g_object_set_property( G_OBJECT(renderer
), "foreground_gdk", &gvalue
);
1990 g_value_unset( &gvalue
);
1992 usingDefaultAttrs
= false;
1996 GValue gvalue
= { 0, };
1997 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1998 g_value_set_boolean( &gvalue
, FALSE
);
1999 g_object_set_property( G_OBJECT(renderer
), "foreground-set", &gvalue
);
2000 g_value_unset( &gvalue
);
2003 if (attr
.GetItalic())
2005 GValue gvalue
= { 0, };
2006 g_value_init( &gvalue
, PANGO_TYPE_STYLE
);
2007 g_value_set_enum( &gvalue
, PANGO_STYLE_ITALIC
);
2008 g_object_set_property( G_OBJECT(renderer
), "style", &gvalue
);
2009 g_value_unset( &gvalue
);
2011 usingDefaultAttrs
= false;
2015 GValue gvalue
= { 0, };
2016 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2017 g_value_set_boolean( &gvalue
, FALSE
);
2018 g_object_set_property( G_OBJECT(renderer
), "style-set", &gvalue
);
2019 g_value_unset( &gvalue
);
2025 GValue gvalue
= { 0, };
2026 g_value_init( &gvalue
, PANGO_TYPE_WEIGHT
);
2027 g_value_set_enum( &gvalue
, PANGO_WEIGHT_BOLD
);
2028 g_object_set_property( G_OBJECT(renderer
), "weight", &gvalue
);
2029 g_value_unset( &gvalue
);
2031 usingDefaultAttrs
= false;
2035 GValue gvalue
= { 0, };
2036 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2037 g_value_set_boolean( &gvalue
, FALSE
);
2038 g_object_set_property( G_OBJECT(renderer
), "weight-set", &gvalue
);
2039 g_value_unset( &gvalue
);
2043 if (attr
.HasBackgroundColour())
2045 wxColour colour
= attr
.GetBackgroundColour();
2046 const GdkColor
* const gcol
= colour
.GetColor();
2048 GValue gvalue
= { 0, };
2049 g_value_init( &gvalue
, GDK_TYPE_COLOR
);
2050 g_value_set_boxed( &gvalue
, gcol
);
2051 g_object_set_property( G_OBJECT(renderer
), "cell-background_gdk", &gvalue
);
2052 g_value_unset( &gvalue
);
2056 GValue gvalue
= { 0, };
2057 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2058 g_value_set_boolean( &gvalue
, FALSE
);
2059 g_object_set_property( G_OBJECT(renderer
), "cell-background-set", &gvalue
);
2060 g_value_unset( &gvalue
);
2064 return !usingDefaultAttrs
;
2067 } // anonymous namespace
2069 IMPLEMENT_CLASS(wxDataViewTextRenderer
, wxDataViewRenderer
)
2071 wxDataViewTextRenderer::wxDataViewTextRenderer( const wxString
&varianttype
, wxDataViewCellMode mode
,
2073 wxDataViewRenderer( varianttype
, mode
, align
)
2075 GtkWxCellRendererText
*text_renderer
= gtk_wx_cell_renderer_text_new();
2076 text_renderer
->wx_renderer
= this;
2077 m_renderer
= (GtkCellRenderer
*) text_renderer
;
2079 if (mode
& wxDATAVIEW_CELL_EDITABLE
)
2081 GValue gvalue
= { 0, };
2082 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2083 g_value_set_boolean( &gvalue
, true );
2084 g_object_set_property( G_OBJECT(m_renderer
), "editable", &gvalue
);
2085 g_value_unset( &gvalue
);
2087 g_signal_connect_after( m_renderer
, "edited", G_CALLBACK(wxGtkTextRendererEditedCallback
), this );
2093 SetAlignment(align
);
2096 bool wxDataViewTextRenderer::SetTextValue(const wxString
& str
)
2098 GValue gvalue
= { 0, };
2099 g_value_init( &gvalue
, G_TYPE_STRING
);
2100 g_value_set_string( &gvalue
, wxGTK_CONV_FONT( str
, GetOwner()->GetOwner()->GetFont() ) );
2101 g_object_set_property( G_OBJECT(m_renderer
), "text", &gvalue
);
2102 g_value_unset( &gvalue
);
2107 bool wxDataViewTextRenderer::GetTextValue(wxString
& str
) const
2109 GValue gvalue
= { 0, };
2110 g_value_init( &gvalue
, G_TYPE_STRING
);
2111 g_object_get_property( G_OBJECT(m_renderer
), "text", &gvalue
);
2112 str
= wxGTK_CONV_BACK_FONT( g_value_get_string( &gvalue
), const_cast<wxDataViewTextRenderer
*>(this)->GetOwner()->GetOwner()->GetFont() );
2113 g_value_unset( &gvalue
);
2118 void wxDataViewTextRenderer::SetAlignment( int align
)
2120 wxDataViewRenderer::SetAlignment(align
);
2122 if (gtk_check_version(2,10,0))
2125 // horizontal alignment:
2126 PangoAlignment pangoAlign
= PANGO_ALIGN_LEFT
;
2127 if (align
& wxALIGN_RIGHT
)
2128 pangoAlign
= PANGO_ALIGN_RIGHT
;
2129 else if (align
& wxALIGN_CENTER_HORIZONTAL
)
2130 pangoAlign
= PANGO_ALIGN_CENTER
;
2132 GValue gvalue
= { 0, };
2133 g_value_init( &gvalue
, gtk_cell_renderer_mode_get_type() );
2134 g_value_set_enum( &gvalue
, pangoAlign
);
2135 g_object_set_property( G_OBJECT(m_renderer
), "alignment", &gvalue
);
2136 g_value_unset( &gvalue
);
2139 bool wxDataViewTextRenderer::GtkSetAttr(const wxDataViewItemAttr
& attr
)
2141 return GtkApplyAttr(GtkGetTextRenderer(), attr
);
2144 GtkCellRendererText
*wxDataViewTextRenderer::GtkGetTextRenderer() const
2146 return GTK_CELL_RENDERER_TEXT(m_renderer
);
2149 // ---------------------------------------------------------
2150 // wxDataViewBitmapRenderer
2151 // ---------------------------------------------------------
2156 // set "pixbuf" property on the given renderer
2157 void SetPixbufProp(GtkCellRenderer
*renderer
, GdkPixbuf
*pixbuf
)
2159 GValue gvalue
= { 0, };
2160 g_value_init( &gvalue
, G_TYPE_OBJECT
);
2161 g_value_set_object( &gvalue
, pixbuf
);
2162 g_object_set_property( G_OBJECT(renderer
), "pixbuf", &gvalue
);
2163 g_value_unset( &gvalue
);
2166 } // anonymous namespace
2168 IMPLEMENT_CLASS(wxDataViewBitmapRenderer
, wxDataViewRenderer
)
2170 wxDataViewBitmapRenderer::wxDataViewBitmapRenderer( const wxString
&varianttype
, wxDataViewCellMode mode
,
2172 wxDataViewRenderer( varianttype
, mode
, align
)
2174 m_renderer
= gtk_cell_renderer_pixbuf_new();
2177 SetAlignment(align
);
2180 bool wxDataViewBitmapRenderer::SetValue( const wxVariant
&value
)
2182 if (value
.GetType() == wxT("wxBitmap"))
2187 // GetPixbuf() may create a Pixbuf representation in the wxBitmap
2188 // object (and it will stay there and remain owned by wxBitmap)
2189 SetPixbufProp(m_renderer
, bitmap
.GetPixbuf());
2191 else if (value
.GetType() == wxT("wxIcon"))
2196 SetPixbufProp(m_renderer
, icon
.GetPixbuf());
2206 bool wxDataViewBitmapRenderer::GetValue( wxVariant
&WXUNUSED(value
) ) const
2211 // ---------------------------------------------------------
2212 // wxDataViewToggleRenderer
2213 // ---------------------------------------------------------
2216 static void wxGtkToggleRendererToggledCallback( GtkCellRendererToggle
*renderer
,
2217 gchar
*path
, gpointer user_data
);
2220 static void wxGtkToggleRendererToggledCallback( GtkCellRendererToggle
*renderer
,
2221 gchar
*path
, gpointer user_data
)
2223 wxDataViewToggleRenderer
*cell
= (wxDataViewToggleRenderer
*) user_data
;
2226 GValue gvalue
= { 0, };
2227 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2228 g_object_get_property( G_OBJECT(renderer
), "active", &gvalue
);
2229 bool tmp
= g_value_get_boolean( &gvalue
);
2230 g_value_unset( &gvalue
);
2234 wxVariant value
= tmp
;
2235 if (!cell
->Validate( value
))
2238 wxDataViewCtrl
* const ctrl
= cell
->GetOwner()->GetOwner();
2239 wxDataViewModel
*model
= ctrl
->GetModel();
2241 wxDataViewItem
item(ctrl
->GTKPathToItem(wxGtkTreePath(path
)));
2243 unsigned int model_col
= cell
->GetOwner()->GetModelColumn();
2245 model
->ChangeValue( value
, item
, model_col
);
2248 IMPLEMENT_CLASS(wxDataViewToggleRenderer
, wxDataViewRenderer
)
2250 wxDataViewToggleRenderer::wxDataViewToggleRenderer( const wxString
&varianttype
,
2251 wxDataViewCellMode mode
, int align
) :
2252 wxDataViewRenderer( varianttype
, mode
, align
)
2254 m_renderer
= (GtkCellRenderer
*) gtk_cell_renderer_toggle_new();
2256 if (mode
& wxDATAVIEW_CELL_ACTIVATABLE
)
2258 g_signal_connect_after( m_renderer
, "toggled",
2259 G_CALLBACK(wxGtkToggleRendererToggledCallback
), this );
2263 GValue gvalue
= { 0, };
2264 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2265 g_value_set_boolean( &gvalue
, false );
2266 g_object_set_property( G_OBJECT(m_renderer
), "activatable", &gvalue
);
2267 g_value_unset( &gvalue
);
2271 SetAlignment(align
);
2274 bool wxDataViewToggleRenderer::SetValue( const wxVariant
&value
)
2278 GValue gvalue
= { 0, };
2279 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2280 g_value_set_boolean( &gvalue
, tmp
);
2281 g_object_set_property( G_OBJECT(m_renderer
), "active", &gvalue
);
2282 g_value_unset( &gvalue
);
2287 bool wxDataViewToggleRenderer::GetValue( wxVariant
&value
) const
2289 GValue gvalue
= { 0, };
2290 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2291 g_object_get_property( G_OBJECT(m_renderer
), "active", &gvalue
);
2292 bool tmp
= g_value_get_boolean( &gvalue
);
2293 g_value_unset( &gvalue
);
2300 // ---------------------------------------------------------
2301 // wxDataViewCustomRenderer
2302 // ---------------------------------------------------------
2304 class wxDataViewCtrlDCImpl
: public wxWindowDCImpl
2307 wxDataViewCtrlDCImpl( wxDC
*owner
, wxDataViewCtrl
*window
) :
2308 wxWindowDCImpl( owner
)
2310 GtkWidget
*widget
= window
->m_treeview
;
2316 m_context
= window
->GTKGetPangoDefaultContext();
2317 m_layout
= pango_layout_new( m_context
);
2318 m_fontdesc
= pango_font_description_copy(gtk_widget_get_style(widget
)->font_desc
);
2320 m_cmap
= gtk_widget_get_colormap( widget
? widget
: window
->m_widget
);
2322 // Set m_gdkwindow later
2327 class wxDataViewCtrlDC
: public wxWindowDC
2330 wxDataViewCtrlDC( wxDataViewCtrl
*window
) :
2331 wxWindowDC( new wxDataViewCtrlDCImpl( this, window
) )
2336 // ---------------------------------------------------------
2337 // wxDataViewCustomRenderer
2338 // ---------------------------------------------------------
2340 IMPLEMENT_CLASS(wxDataViewCustomRenderer
, wxDataViewRenderer
)
2342 wxDataViewCustomRenderer::wxDataViewCustomRenderer( const wxString
&varianttype
,
2343 wxDataViewCellMode mode
,
2346 : wxDataViewCustomRendererBase( varianttype
, mode
, align
)
2349 m_text_renderer
= NULL
;
2357 void wxDataViewCustomRenderer::GtkInitTextRenderer()
2359 m_text_renderer
= GTK_CELL_RENDERER_TEXT(gtk_cell_renderer_text_new());
2360 g_object_ref_sink(m_text_renderer
);
2362 GtkApplyAlignment(GTK_CELL_RENDERER(m_text_renderer
));
2365 GtkCellRendererText
*wxDataViewCustomRenderer::GtkGetTextRenderer() const
2367 if ( !m_text_renderer
)
2369 // we create it on demand so need to do it even from a const function
2370 const_cast<wxDataViewCustomRenderer
*>(this)->GtkInitTextRenderer();
2373 return m_text_renderer
;
2376 void wxDataViewCustomRenderer::RenderText( const wxString
&text
,
2380 int WXUNUSED(state
) )
2383 GtkCellRendererText
* const textRenderer
= GtkGetTextRenderer();
2385 GValue gvalue
= { 0, };
2386 g_value_init( &gvalue
, G_TYPE_STRING
);
2387 g_value_set_string( &gvalue
, wxGTK_CONV_FONT( text
, GetOwner()->GetOwner()->GetFont() ) );
2388 g_object_set_property( G_OBJECT(textRenderer
), "text", &gvalue
);
2389 g_value_unset( &gvalue
);
2391 GtkApplyAttr(textRenderer
, GetAttr());
2393 GdkRectangle cell_area
;
2394 wxRectToGDKRect(cell
, cell_area
);
2395 cell_area
.x
+= xoffset
;
2396 cell_area
.width
-= xoffset
;
2398 gtk_cell_renderer_render( GTK_CELL_RENDERER(textRenderer
),
2399 m_renderParams
.window
,
2400 m_renderParams
.widget
,
2401 m_renderParams
.background_area
,
2403 m_renderParams
.expose_area
,
2404 (GtkCellRendererState
) m_renderParams
.flags
);
2407 bool wxDataViewCustomRenderer::Init(wxDataViewCellMode mode
, int align
)
2409 GtkWxCellRenderer
*renderer
= (GtkWxCellRenderer
*) gtk_wx_cell_renderer_new();
2410 renderer
->cell
= this;
2412 m_renderer
= (GtkCellRenderer
*) renderer
;
2415 SetAlignment(align
);
2422 wxDataViewCustomRenderer::~wxDataViewCustomRenderer()
2427 if (m_text_renderer
)
2428 g_object_unref(m_text_renderer
);
2431 wxDC
*wxDataViewCustomRenderer::GetDC()
2435 if (GetOwner() == NULL
)
2437 if (GetOwner()->GetOwner() == NULL
)
2439 m_dc
= new wxDataViewCtrlDC( GetOwner()->GetOwner() );
2445 // ---------------------------------------------------------
2446 // wxDataViewProgressRenderer
2447 // ---------------------------------------------------------
2449 IMPLEMENT_CLASS(wxDataViewProgressRenderer
, wxDataViewCustomRenderer
)
2451 wxDataViewProgressRenderer::wxDataViewProgressRenderer( const wxString
&label
,
2452 const wxString
&varianttype
, wxDataViewCellMode mode
, int align
) :
2453 wxDataViewCustomRenderer( varianttype
, mode
, align
, true )
2459 if (!gtk_check_version(2,6,0))
2461 m_renderer
= (GtkCellRenderer
*) gtk_cell_renderer_progress_new();
2464 SetAlignment(align
);
2467 // We can't initialize the renderer just yet because we don't have the
2468 // pointer to the column that uses this renderer yet and so attempt to
2469 // dereference GetOwner() to get the font that is used as a source of
2470 // encoding in multibyte-to-Unicode conversion in GTKSetLabel() in
2471 // non-Unicode builds would crash. So simply remember to do it later.
2472 if ( !m_label
.empty() )
2473 m_needsToSetLabel
= true;
2475 #endif // !wxUSE_UNICODE
2481 // Use custom cell code
2482 wxDataViewCustomRenderer::Init(mode
, align
);
2486 wxDataViewProgressRenderer::~wxDataViewProgressRenderer()
2490 void wxDataViewProgressRenderer::GTKSetLabel()
2492 GValue gvalue
= { 0, };
2493 g_value_init( &gvalue
, G_TYPE_STRING
);
2495 // Take care to not use GetOwner() here if the label is empty, we can be
2496 // called from ctor when GetOwner() is still NULL in this case.
2497 wxScopedCharBuffer buf
;
2498 if ( m_label
.empty() )
2499 buf
= wxScopedCharBuffer::CreateNonOwned("");
2501 buf
= wxGTK_CONV_FONT(m_label
, GetOwner()->GetOwner()->GetFont());
2503 g_value_set_string( &gvalue
, buf
);
2504 g_object_set_property( G_OBJECT(m_renderer
), "text", &gvalue
);
2505 g_value_unset( &gvalue
);
2508 m_needsToSetLabel
= false;
2509 #endif // !wxUSE_UNICODE
2512 bool wxDataViewProgressRenderer::SetValue( const wxVariant
&value
)
2515 if (!gtk_check_version(2,6,0))
2518 if ( m_needsToSetLabel
)
2520 #endif // !wxUSE_UNICODE
2522 gint tmp
= (long) value
;
2523 GValue gvalue
= { 0, };
2524 g_value_init( &gvalue
, G_TYPE_INT
);
2525 g_value_set_int( &gvalue
, tmp
);
2526 g_object_set_property( G_OBJECT(m_renderer
), "value", &gvalue
);
2527 g_value_unset( &gvalue
);
2532 m_value
= (long) value
;
2534 if (m_value
< 0) m_value
= 0;
2535 if (m_value
> 100) m_value
= 100;
2541 bool wxDataViewProgressRenderer::GetValue( wxVariant
&WXUNUSED(value
) ) const
2546 bool wxDataViewProgressRenderer::Render( wxRect cell
, wxDC
*dc
, int WXUNUSED(state
) )
2548 double pct
= (double)m_value
/ 100.0;
2550 bar
.width
= (int)(cell
.width
* pct
);
2551 dc
->SetPen( *wxTRANSPARENT_PEN
);
2552 dc
->SetBrush( *wxBLUE_BRUSH
);
2553 dc
->DrawRectangle( bar
);
2555 dc
->SetBrush( *wxTRANSPARENT_BRUSH
);
2556 dc
->SetPen( *wxBLACK_PEN
);
2557 dc
->DrawRectangle( cell
);
2562 wxSize
wxDataViewProgressRenderer::GetSize() const
2564 return wxSize(40,12);
2567 // -------------------------------------
2568 // wxDataViewChoiceRenderer
2569 // -------------------------------------
2571 wxDataViewChoiceRenderer::wxDataViewChoiceRenderer( const wxArrayString
&choices
,
2572 wxDataViewCellMode mode
, int alignment
) :
2573 wxDataViewCustomRenderer( "string", mode
, alignment
, true )
2575 m_choices
= choices
;
2578 if (!gtk_check_version(2,6,0))
2580 m_renderer
= (GtkCellRenderer
*) gtk_cell_renderer_combo_new();
2582 GtkListStore
*store
= gtk_list_store_new( 1, G_TYPE_STRING
);
2583 for (size_t n
= 0; n
< m_choices
.GetCount(); n
++)
2585 gtk_list_store_insert_with_values(
2587 static_cast<const char *>(m_choices
[n
].utf8_str()), -1 );
2590 g_object_set (m_renderer
,
2596 bool editable
= (mode
& wxDATAVIEW_CELL_EDITABLE
);
2597 g_object_set (m_renderer
, "editable", editable
, NULL
);
2599 SetAlignment(alignment
);
2601 g_signal_connect_after( m_renderer
, "edited", G_CALLBACK(wxGtkTextRendererEditedCallback
), this );
2608 // Use custom cell code
2609 wxDataViewCustomRenderer::Init(mode
, alignment
);
2613 bool wxDataViewChoiceRenderer::Render( wxRect rect
, wxDC
*dc
, int state
)
2615 RenderText( m_data
, 0, rect
, dc
, state
);
2619 wxSize
wxDataViewChoiceRenderer::GetSize() const
2621 return wxSize(70,20);
2624 bool wxDataViewChoiceRenderer::SetValue( const wxVariant
&value
)
2628 if (!gtk_check_version(2,6,0))
2630 GValue gvalue
= { 0, };
2631 g_value_init( &gvalue
, G_TYPE_STRING
);
2632 g_value_set_string(&gvalue
,
2633 wxGTK_CONV_FONT(value
.GetString(),
2634 GetOwner()->GetOwner()->GetFont()));
2635 g_object_set_property( G_OBJECT(m_renderer
), "text", &gvalue
);
2636 g_value_unset( &gvalue
);
2640 m_data
= value
.GetString();
2645 bool wxDataViewChoiceRenderer::GetValue( wxVariant
&value
) const
2648 if (!gtk_check_version(2,6,0))
2650 GValue gvalue
= { 0, };
2651 g_value_init( &gvalue
, G_TYPE_STRING
);
2652 g_object_get_property( G_OBJECT(m_renderer
), "text", &gvalue
);
2653 wxString temp
= wxGTK_CONV_BACK_FONT(g_value_get_string(&gvalue
),
2654 GetOwner()->GetOwner()->GetFont());
2655 g_value_unset( &gvalue
);
2658 //wxPrintf( "temp %s\n", temp );
2659 // TODO: remove this code
2668 void wxDataViewChoiceRenderer::SetAlignment( int align
)
2670 wxDataViewCustomRenderer::SetAlignment(align
);
2672 if (gtk_check_version(2,10,0))
2675 // horizontal alignment:
2676 PangoAlignment pangoAlign
= PANGO_ALIGN_LEFT
;
2677 if (align
& wxALIGN_RIGHT
)
2678 pangoAlign
= PANGO_ALIGN_RIGHT
;
2679 else if (align
& wxALIGN_CENTER_HORIZONTAL
)
2680 pangoAlign
= PANGO_ALIGN_CENTER
;
2682 GValue gvalue
= { 0, };
2683 g_value_init( &gvalue
, gtk_cell_renderer_mode_get_type() );
2684 g_value_set_enum( &gvalue
, pangoAlign
);
2685 g_object_set_property( G_OBJECT(m_renderer
), "alignment", &gvalue
);
2686 g_value_unset( &gvalue
);
2689 // ----------------------------------------------------------------------------
2690 // wxDataViewChoiceByIndexRenderer
2691 // ----------------------------------------------------------------------------
2693 wxDataViewChoiceByIndexRenderer::wxDataViewChoiceByIndexRenderer( const wxArrayString
&choices
,
2694 wxDataViewCellMode mode
, int alignment
) :
2695 wxDataViewChoiceRenderer( choices
, mode
, alignment
)
2699 void wxDataViewChoiceByIndexRenderer::GtkOnTextEdited(const char *itempath
, const wxString
& str
)
2701 wxVariant
value( (long) GetChoices().Index( str
) );
2703 if (!Validate( value
))
2707 item(GetOwner()->GetOwner()->GTKPathToItem(wxGtkTreePath(itempath
)));
2709 GtkOnCellChanged(value
, item
, GetOwner()->GetModelColumn());
2712 bool wxDataViewChoiceByIndexRenderer::SetValue( const wxVariant
&value
)
2714 wxVariant string_value
= GetChoice( value
.GetLong() );
2715 return wxDataViewChoiceRenderer::SetValue( string_value
);
2718 bool wxDataViewChoiceByIndexRenderer::GetValue( wxVariant
&value
) const
2720 wxVariant string_value
;
2721 if (!wxDataViewChoiceRenderer::GetValue( string_value
))
2724 value
= (long) GetChoices().Index( string_value
.GetString() );
2728 // ---------------------------------------------------------
2729 // wxDataViewIconTextRenderer
2730 // ---------------------------------------------------------
2732 IMPLEMENT_CLASS(wxDataViewIconTextRenderer
, wxDataViewCustomRenderer
)
2734 wxDataViewIconTextRenderer::wxDataViewIconTextRenderer
2736 const wxString
&varianttype
,
2737 wxDataViewCellMode mode
,
2740 : wxDataViewTextRenderer(varianttype
, mode
, align
)
2742 m_rendererIcon
= gtk_cell_renderer_pixbuf_new();
2745 wxDataViewIconTextRenderer::~wxDataViewIconTextRenderer()
2749 void wxDataViewIconTextRenderer::GtkPackIntoColumn(GtkTreeViewColumn
*column
)
2751 // add the icon renderer first
2752 gtk_tree_view_column_pack_start(column
, m_rendererIcon
, FALSE
/* !expand */);
2754 // add the text renderer too
2755 wxDataViewRenderer::GtkPackIntoColumn(column
);
2758 bool wxDataViewIconTextRenderer::SetValue( const wxVariant
&value
)
2762 SetTextValue(m_value
.GetText());
2763 SetPixbufProp(m_rendererIcon
, m_value
.GetIcon().GetPixbuf());
2768 bool wxDataViewIconTextRenderer::GetValue(wxVariant
& value
) const
2771 if ( !GetTextValue(str
) )
2774 // user doesn't have any way to edit the icon so leave it unchanged
2775 value
<< wxDataViewIconText(str
, m_value
.GetIcon());
2781 wxDataViewIconTextRenderer::GtkOnCellChanged(const wxVariant
& value
,
2782 const wxDataViewItem
& item
,
2785 // we receive just the text part of our value as it's the only one which
2786 // can be edited, but we need the full wxDataViewIconText value for the
2788 wxVariant valueIconText
;
2789 valueIconText
<< wxDataViewIconText(value
.GetString(), m_value
.GetIcon());
2790 wxDataViewTextRenderer::GtkOnCellChanged(valueIconText
, item
, col
);
2793 // ---------------------------------------------------------
2795 // ---------------------------------------------------------
2799 gtk_dataview_header_button_press_callback( GtkWidget
*WXUNUSED(widget
),
2800 GdkEventButton
*gdk_event
,
2801 wxDataViewColumn
*column
)
2803 if (gdk_event
->type
!= GDK_BUTTON_PRESS
)
2806 if (gdk_event
->button
== 1)
2808 gs_lastLeftClickHeader
= column
;
2810 wxDataViewCtrl
*dv
= column
->GetOwner();
2811 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_COLUMN_HEADER_CLICK
, dv
->GetId() );
2812 event
.SetDataViewColumn( column
);
2813 event
.SetModel( dv
->GetModel() );
2814 if (dv
->HandleWindowEvent( event
))
2818 if (gdk_event
->button
== 3)
2820 wxDataViewCtrl
*dv
= column
->GetOwner();
2821 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_COLUMN_HEADER_RIGHT_CLICK
, dv
->GetId() );
2822 event
.SetDataViewColumn( column
);
2823 event
.SetModel( dv
->GetModel() );
2824 if (dv
->HandleWindowEvent( event
))
2834 static void wxGtkTreeCellDataFunc( GtkTreeViewColumn
*WXUNUSED(column
),
2835 GtkCellRenderer
*renderer
,
2836 GtkTreeModel
*model
,
2840 g_return_if_fail (GTK_IS_WX_TREE_MODEL (model
));
2841 GtkWxTreeModel
*tree_model
= (GtkWxTreeModel
*) model
;
2843 wxDataViewRenderer
*cell
= (wxDataViewRenderer
*) data
;
2845 wxDataViewItem
item( (void*) iter
->user_data
);
2847 wxDataViewModel
*wx_model
= tree_model
->internal
->GetDataViewModel();
2849 if (!wx_model
->IsVirtualListModel())
2852 if (wx_model
->IsContainer( item
))
2854 visible
= wx_model
->HasContainerColumns( item
) ||
2855 (cell
->GetOwner()->GetModelColumn() == 0);
2862 GValue gvalue
= { 0, };
2863 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2864 g_value_set_boolean( &gvalue
, visible
);
2865 g_object_set_property( G_OBJECT(renderer
), "visible", &gvalue
);
2866 g_value_unset( &gvalue
);
2873 wx_model
->GetValue( value
, item
, cell
->GetOwner()->GetModelColumn() );
2875 if (value
.GetType() != cell
->GetVariantType())
2877 wxLogError( wxT("Wrong type, required: %s but: %s"),
2878 value
.GetType().c_str(),
2879 cell
->GetVariantType().c_str() );
2882 cell
->SetValue( value
);
2884 // deal with disabled items
2885 bool enabled
= wx_model
->IsEnabled( item
, cell
->GetOwner()->GetModelColumn() );
2887 // a) this sets the appearance to disabled grey
2888 GValue gvalue
= { 0, };
2889 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
2890 g_value_set_boolean( &gvalue
, enabled
);
2891 g_object_set_property( G_OBJECT(renderer
), "sensitive", &gvalue
);
2892 g_value_unset( &gvalue
);
2894 // b) this actually disables the control/renderer
2896 cell
->SetMode( cell
->GtkGetMode() );
2898 cell
->SetMode( wxDATAVIEW_CELL_INERT
);
2901 // deal with attributes: if the renderer doesn't support them at all, we
2902 // don't even need to query the model for them
2903 if ( !cell
->GtkSupportsAttrs() )
2906 // it can support attributes so check if this item has any
2907 wxDataViewItemAttr attr
;
2908 if ( wx_model
->GetAttr( item
, cell
->GetOwner()->GetModelColumn(), attr
)
2909 || !cell
->GtkIsUsingDefaultAttrs() )
2911 bool usingDefaultAttrs
= !cell
->GtkSetAttr(attr
);
2912 cell
->GtkSetUsingDefaultAttrs(usingDefaultAttrs
);
2914 // else: no custom attributes specified and we're already using the default
2915 // ones -- nothing to do
2921 #include <wx/listimpl.cpp>
2922 WX_DEFINE_LIST(wxDataViewColumnList
)
2924 wxDataViewColumn::wxDataViewColumn( const wxString
&title
, wxDataViewRenderer
*cell
,
2925 unsigned int model_column
, int width
,
2926 wxAlignment align
, int flags
)
2927 : wxDataViewColumnBase( cell
, model_column
)
2929 Init( align
, flags
, width
);
2934 wxDataViewColumn::wxDataViewColumn( const wxBitmap
&bitmap
, wxDataViewRenderer
*cell
,
2935 unsigned int model_column
, int width
,
2936 wxAlignment align
, int flags
)
2937 : wxDataViewColumnBase( bitmap
, cell
, model_column
)
2939 Init( align
, flags
, width
);
2941 SetBitmap( bitmap
);
2944 void wxDataViewColumn::Init(wxAlignment align
, int flags
, int width
)
2946 m_isConnected
= false;
2948 GtkTreeViewColumn
*column
= gtk_tree_view_column_new();
2949 m_column
= (GtkWidget
*) column
;
2952 SetAlignment( align
);
2956 // Create container for icon and label
2957 GtkWidget
*box
= gtk_hbox_new( FALSE
, 1 );
2958 gtk_widget_show( box
);
2959 // gtk_container_set_border_width((GtkContainer*)box, 2);
2960 m_image
= gtk_image_new();
2961 gtk_box_pack_start(GTK_BOX(box
), m_image
, FALSE
, FALSE
, 1);
2962 m_label
= gtk_label_new("");
2963 gtk_box_pack_end( GTK_BOX(box
), GTK_WIDGET(m_label
), FALSE
, FALSE
, 1 );
2964 gtk_tree_view_column_set_widget( column
, box
);
2966 wxDataViewRenderer
* const colRenderer
= GetRenderer();
2967 GtkCellRenderer
* const cellRenderer
= colRenderer
->GetGtkHandle();
2969 colRenderer
->GtkPackIntoColumn(column
);
2971 gtk_tree_view_column_set_cell_data_func( column
, cellRenderer
,
2972 wxGtkTreeCellDataFunc
, (gpointer
) colRenderer
, NULL
);
2975 void wxDataViewColumn::OnInternalIdle()
2980 if (gtk_widget_get_realized(GetOwner()->m_treeview
))
2982 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
2985 g_signal_connect(column
->button
, "button_press_event",
2986 G_CALLBACK (gtk_dataview_header_button_press_callback
), this);
2988 // otherwise the event will be blocked by GTK+
2989 gtk_tree_view_column_set_clickable( column
, TRUE
);
2991 m_isConnected
= true;
2996 void wxDataViewColumn::SetOwner( wxDataViewCtrl
*owner
)
2998 wxDataViewColumnBase::SetOwner( owner
);
3000 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
3002 gtk_tree_view_column_set_title( column
, wxGTK_CONV_FONT(GetTitle(), GetOwner()->GetFont() ) );
3005 void wxDataViewColumn::SetTitle( const wxString
&title
)
3007 wxDataViewCtrl
*ctrl
= GetOwner();
3008 gtk_label_set_text( GTK_LABEL(m_label
), ctrl
? wxGTK_CONV_FONT(title
, ctrl
->GetFont())
3009 : wxGTK_CONV_SYS(title
) );
3011 gtk_widget_hide( m_label
);
3013 gtk_widget_show( m_label
);
3016 wxString
wxDataViewColumn::GetTitle() const
3018 return wxGTK_CONV_BACK_FONT(
3019 gtk_label_get_text( GTK_LABEL(m_label
) ),
3020 GetOwner()->GetFont()
3024 void wxDataViewColumn::SetBitmap( const wxBitmap
&bitmap
)
3026 wxDataViewColumnBase::SetBitmap( bitmap
);
3030 GtkImage
*gtk_image
= GTK_IMAGE(m_image
);
3032 GdkBitmap
*mask
= NULL
;
3033 if (bitmap
.GetMask())
3034 mask
= bitmap
.GetMask()->GetBitmap();
3036 if (bitmap
.HasPixbuf())
3038 gtk_image_set_from_pixbuf(GTK_IMAGE(gtk_image
),
3039 bitmap
.GetPixbuf());
3043 gtk_image_set_from_pixmap(GTK_IMAGE(gtk_image
),
3044 bitmap
.GetPixmap(), mask
);
3046 gtk_widget_show( m_image
);
3050 gtk_widget_hide( m_image
);
3054 void wxDataViewColumn::SetHidden( bool hidden
)
3056 gtk_tree_view_column_set_visible( GTK_TREE_VIEW_COLUMN(m_column
), !hidden
);
3059 void wxDataViewColumn::SetResizeable( bool resizable
)
3061 gtk_tree_view_column_set_resizable( GTK_TREE_VIEW_COLUMN(m_column
), resizable
);
3064 void wxDataViewColumn::SetAlignment( wxAlignment align
)
3066 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
3068 gfloat xalign
= 0.0;
3069 if (align
== wxALIGN_RIGHT
)
3071 if (align
== wxALIGN_CENTER_HORIZONTAL
||
3072 align
== wxALIGN_CENTER
)
3075 gtk_tree_view_column_set_alignment( column
, xalign
);
3077 if (m_renderer
&& m_renderer
->GetAlignment() == -1)
3078 m_renderer
->GtkUpdateAlignment();
3081 wxAlignment
wxDataViewColumn::GetAlignment() const
3083 gfloat xalign
= gtk_tree_view_column_get_alignment( GTK_TREE_VIEW_COLUMN(m_column
) );
3086 return wxALIGN_RIGHT
;
3088 return wxALIGN_CENTER_HORIZONTAL
;
3090 return wxALIGN_LEFT
;
3093 void wxDataViewColumn::SetSortable( bool sortable
)
3095 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
3099 gtk_tree_view_column_set_sort_column_id( column
, GetModelColumn() );
3103 gtk_tree_view_column_set_sort_column_id( column
, -1 );
3104 gtk_tree_view_column_set_sort_indicator( column
, FALSE
);
3105 gtk_tree_view_column_set_clickable( column
, FALSE
);
3109 bool wxDataViewColumn::IsSortable() const
3111 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
3112 return gtk_tree_view_column_get_clickable( column
);
3115 bool wxDataViewColumn::IsSortKey() const
3117 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
3118 return gtk_tree_view_column_get_sort_indicator( column
);
3121 bool wxDataViewColumn::IsResizeable() const
3123 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
3124 return gtk_tree_view_column_get_resizable( column
);
3127 bool wxDataViewColumn::IsHidden() const
3129 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
3130 return !gtk_tree_view_column_get_visible( column
);
3133 void wxDataViewColumn::SetSortOrder( bool ascending
)
3135 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
3138 gtk_tree_view_column_set_sort_order( column
, GTK_SORT_ASCENDING
);
3140 gtk_tree_view_column_set_sort_order( column
, GTK_SORT_DESCENDING
);
3142 gtk_tree_view_column_set_sort_indicator( column
, TRUE
);
3145 bool wxDataViewColumn::IsSortOrderAscending() const
3147 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
3149 return (gtk_tree_view_column_get_sort_order( column
) != GTK_SORT_DESCENDING
);
3152 void wxDataViewColumn::SetMinWidth( int width
)
3154 gtk_tree_view_column_set_min_width( GTK_TREE_VIEW_COLUMN(m_column
), width
);
3157 int wxDataViewColumn::GetMinWidth() const
3159 return gtk_tree_view_column_get_min_width( GTK_TREE_VIEW_COLUMN(m_column
) );
3162 int wxDataViewColumn::GetWidth() const
3164 return gtk_tree_view_column_get_width( GTK_TREE_VIEW_COLUMN(m_column
) );
3167 void wxDataViewColumn::SetWidth( int width
)
3169 if ( width
== wxCOL_WIDTH_AUTOSIZE
)
3171 // NB: this disables user resizing
3172 gtk_tree_view_column_set_sizing( GTK_TREE_VIEW_COLUMN(m_column
), GTK_TREE_VIEW_COLUMN_AUTOSIZE
);
3176 if ( width
== wxCOL_WIDTH_DEFAULT
)
3178 // TODO find a better calculation
3179 width
= wxDVC_DEFAULT_WIDTH
;
3182 gtk_tree_view_column_set_sizing( GTK_TREE_VIEW_COLUMN(m_column
), GTK_TREE_VIEW_COLUMN_FIXED
);
3183 gtk_tree_view_column_set_fixed_width( GTK_TREE_VIEW_COLUMN(m_column
), width
);
3187 void wxDataViewColumn::SetReorderable( bool reorderable
)
3189 gtk_tree_view_column_set_reorderable( GTK_TREE_VIEW_COLUMN(m_column
), reorderable
);
3192 bool wxDataViewColumn::IsReorderable() const
3194 return gtk_tree_view_column_get_reorderable( GTK_TREE_VIEW_COLUMN(m_column
) );
3197 //-----------------------------------------------------------------------------
3198 // wxGtkTreeModelNode
3199 //-----------------------------------------------------------------------------
3202 class wxGtkTreeModelChildWithPos
3210 int wxGtkTreeModelChildWithPosCmp( const void* data1
, const void* data2
, const void* user_data
)
3212 const wxGtkTreeModelChildWithPos
* child1
= (const wxGtkTreeModelChildWithPos
*) data1
;
3213 const wxGtkTreeModelChildWithPos
* child2
= (const wxGtkTreeModelChildWithPos
*) data2
;
3214 const wxDataViewCtrlInternal
*internal
= (const wxDataViewCtrlInternal
*) user_data
;
3215 int ret
= internal
->GetDataViewModel()->Compare( child1
->id
, child2
->id
,
3216 internal
->GetSortColumn(), (internal
->GetSortOrder() == GTK_SORT_DESCENDING
) );
3222 int LINKAGEMODE
wxGtkTreeModelChildPtrCmp( void*** data1
, void*** data2
)
3224 return gs_internal
->GetDataViewModel()->Compare( wxDataViewItem(**data1
), wxDataViewItem(**data2
),
3225 gs_internal
->GetSortColumn(), (gs_internal
->GetSortOrder() == GTK_SORT_ASCENDING
) );
3228 WX_DEFINE_ARRAY_PTR( void**, wxGtkTreeModelChildrenPtr
);
3231 void wxGtkTreeModelNode::Resort()
3233 size_t child_count
= GetChildCount();
3234 if (child_count
== 0)
3237 size_t node_count
= GetNodesCount();
3239 if (child_count
== 1)
3241 if (node_count
== 1)
3243 wxGtkTreeModelNode
*node
= m_nodes
.Item( 0 );
3249 gint
*new_order
= new gint
[child_count
];
3252 // m_children has the original *void
3253 // ptrs points to these
3254 wxGtkTreeModelChildrenPtr ptrs
;
3256 for (i
= 0; i
< child_count
; i
++)
3257 ptrs
.Add( &(m_children
[i
]) );
3259 gs_internal
= m_internal
;
3260 ptrs
.Sort( &wxGtkTreeModelChildPtrCmp
);
3262 wxGtkTreeModelChildren temp
;
3263 void** base_ptr
= &(m_children
[0]);
3264 // Transfer positions to new_order array and
3266 for (i
= 0; i
< child_count
; i
++)
3268 new_order
[i
] = ptrs
[i
] - base_ptr
;
3269 temp
.Add( *ptrs
[i
] );
3272 // Transfer IDs back to m_children
3274 WX_APPEND_ARRAY( temp
, m_children
);
3279 // Build up array with IDs and original positions
3280 wxGtkTreeModelChildWithPos
* temp
= new wxGtkTreeModelChildWithPos
[child_count
];
3282 for (i
= 0; i
< child_count
; i
++)
3285 temp
[i
].id
= m_children
[i
];
3287 // Sort array keeping original positions
3288 wxQsort( temp
, child_count
, sizeof(wxGtkTreeModelChildWithPos
),
3289 &wxGtkTreeModelChildWithPosCmp
, m_internal
);
3290 // Transfer positions to new_order array and
3291 // IDs to m_children
3293 for (i
= 0; i
< child_count
; i
++)
3295 new_order
[i
] = temp
[i
].pos
;
3296 m_children
.Add( temp
[i
].id
);
3305 wxGtkTreeModelChildren temp
;
3306 WX_APPEND_ARRAY( temp
, m_children
);
3308 gs_internal
= m_internal
;
3309 m_children
.Sort( &wxGtkTreeModelChildCmp
);
3312 for (pos
= 0; pos
< child_count
; pos
++)
3314 void *id
= m_children
.Item( pos
);
3315 int old_pos
= temp
.Index( id
);
3316 new_order
[pos
] = old_pos
;
3320 GtkTreeModel
*gtk_tree_model
= GTK_TREE_MODEL( m_internal
->GetGtkModel() );
3323 iter
.user_data
= GetItem().GetID();
3324 iter
.stamp
= m_internal
->GetGtkModel()->stamp
;
3326 gtk_tree_model_rows_reordered( gtk_tree_model
,
3327 wxGtkTreePath(m_internal
->get_path(&iter
)), &iter
, new_order
);
3329 delete [] new_order
;
3332 for (pos
= 0; pos
< node_count
; pos
++)
3334 wxGtkTreeModelNode
*node
= m_nodes
.Item( pos
);
3339 //-----------------------------------------------------------------------------
3340 // wxDataViewCtrlInternal
3341 //-----------------------------------------------------------------------------
3343 wxDataViewCtrlInternal::wxDataViewCtrlInternal( wxDataViewCtrl
*owner
, wxDataViewModel
*wx_model
)
3346 m_wx_model
= wx_model
;
3350 m_sort_order
= GTK_SORT_ASCENDING
;
3352 m_dataview_sort_column
= NULL
;
3354 m_dragDataObject
= NULL
;
3355 m_dropDataObject
= NULL
;
3359 m_gtk_model
= wxgtk_tree_model_new();
3360 m_gtk_model
->internal
= this;
3362 m_notifier
= new wxGtkDataViewModelNotifier( wx_model
, this );
3364 wx_model
->AddNotifier( m_notifier
);
3366 // g_object_unref( gtk_model ); ???
3368 if (!m_wx_model
->IsVirtualListModel())
3371 gtk_tree_view_set_model( GTK_TREE_VIEW(m_owner
->GtkGetTreeView()), GTK_TREE_MODEL(m_gtk_model
) );
3374 wxDataViewCtrlInternal::~wxDataViewCtrlInternal()
3376 m_wx_model
->RemoveNotifier( m_notifier
);
3378 // remove the model from the GtkTreeView before it gets destroyed
3379 gtk_tree_view_set_model( GTK_TREE_VIEW( m_owner
->GtkGetTreeView() ), NULL
);
3381 g_object_unref( m_gtk_model
);
3383 delete m_dragDataObject
;
3384 delete m_dropDataObject
;
3387 void wxDataViewCtrlInternal::ScheduleRefresh()
3392 void wxDataViewCtrlInternal::OnInternalIdle()
3396 GtkWidget
*widget
= m_owner
->GtkGetTreeView();
3397 gtk_widget_queue_draw( widget
);
3402 void wxDataViewCtrlInternal::InitTree()
3404 wxDataViewItem item
;
3405 m_root
= new wxGtkTreeModelNode( NULL
, item
, this );
3407 BuildBranch( m_root
);
3410 void wxDataViewCtrlInternal::BuildBranch( wxGtkTreeModelNode
*node
)
3412 if (node
->GetChildCount() == 0)
3414 wxDataViewItemArray children
;
3415 unsigned int count
= m_wx_model
->GetChildren( node
->GetItem(), children
);
3418 for (pos
= 0; pos
< count
; pos
++)
3420 wxDataViewItem child
= children
[pos
];
3422 if (m_wx_model
->IsContainer( child
))
3423 node
->AddNode( new wxGtkTreeModelNode( node
, child
, this ) );
3425 node
->AddLeaf( child
.GetID() );
3427 // Don't send any events here
3434 bool wxDataViewCtrlInternal::EnableDragSource( const wxDataFormat
&format
)
3436 wxGtkString
atom_str( gdk_atom_name( format
) );
3437 m_dragSourceTargetEntryTarget
= wxCharBuffer( atom_str
);
3439 m_dragSourceTargetEntry
.target
= m_dragSourceTargetEntryTarget
.data();
3440 m_dragSourceTargetEntry
.flags
= 0;
3441 m_dragSourceTargetEntry
.info
= static_cast<guint
>(-1);
3443 gtk_tree_view_enable_model_drag_source( GTK_TREE_VIEW(m_owner
->GtkGetTreeView() ),
3444 GDK_BUTTON1_MASK
, &m_dragSourceTargetEntry
, 1, (GdkDragAction
) GDK_ACTION_COPY
);
3449 bool wxDataViewCtrlInternal::EnableDropTarget( const wxDataFormat
&format
)
3451 wxGtkString
atom_str( gdk_atom_name( format
) );
3452 m_dropTargetTargetEntryTarget
= wxCharBuffer( atom_str
);
3454 m_dropTargetTargetEntry
.target
= m_dropTargetTargetEntryTarget
.data();
3455 m_dropTargetTargetEntry
.flags
= 0;
3456 m_dropTargetTargetEntry
.info
= static_cast<guint
>(-1);
3458 gtk_tree_view_enable_model_drag_dest( GTK_TREE_VIEW(m_owner
->GtkGetTreeView() ),
3459 &m_dropTargetTargetEntry
, 1, (GdkDragAction
) GDK_ACTION_COPY
);
3464 gboolean
wxDataViewCtrlInternal::row_draggable( GtkTreeDragSource
*WXUNUSED(drag_source
),
3467 delete m_dragDataObject
;
3468 m_dragDataObject
= NULL
;
3470 wxDataViewItem
item(GetOwner()->GTKPathToItem(path
));
3474 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_BEGIN_DRAG
, m_owner
->GetId() );
3475 event
.SetEventObject( m_owner
);
3476 event
.SetItem( item
);
3477 event
.SetModel( m_wx_model
);
3479 gtk_widget_get_pointer(m_owner
->GtkGetTreeView(), &x
, &y
);
3480 event
.SetPosition(x
, y
);
3481 if (!m_owner
->HandleWindowEvent( event
))
3484 if (!event
.IsAllowed())
3487 wxDataObject
*obj
= event
.GetDataObject();
3491 m_dragDataObject
= obj
;
3497 wxDataViewCtrlInternal::drag_data_delete(GtkTreeDragSource
*WXUNUSED(drag_source
),
3498 GtkTreePath
*WXUNUSED(path
))
3503 gboolean
wxDataViewCtrlInternal::drag_data_get( GtkTreeDragSource
*WXUNUSED(drag_source
),
3504 GtkTreePath
*path
, GtkSelectionData
*selection_data
)
3506 wxDataViewItem
item(GetOwner()->GTKPathToItem(path
));
3510 GdkAtom target
= gtk_selection_data_get_target(selection_data
);
3511 if (!m_dragDataObject
->IsSupported(target
))
3514 size_t size
= m_dragDataObject
->GetDataSize(target
);
3518 void *buf
= malloc( size
);
3520 gboolean res
= FALSE
;
3521 if (m_dragDataObject
->GetDataHere(target
, buf
))
3525 gtk_selection_data_set(selection_data
, target
,
3526 8, (const guchar
*) buf
, size
);
3535 wxDataViewCtrlInternal::drag_data_received(GtkTreeDragDest
*WXUNUSED(drag_dest
),
3537 GtkSelectionData
*selection_data
)
3539 wxDataViewItem
item(GetOwner()->GTKPathToItem(path
));
3543 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_DROP
, m_owner
->GetId() );
3544 event
.SetEventObject( m_owner
);
3545 event
.SetItem( item
);
3546 event
.SetModel( m_wx_model
);
3547 event
.SetDataFormat(gtk_selection_data_get_target(selection_data
));
3548 event
.SetDataSize(gtk_selection_data_get_length(selection_data
));
3549 event
.SetDataBuffer(const_cast<guchar
*>(gtk_selection_data_get_data(selection_data
)));
3550 if (!m_owner
->HandleWindowEvent( event
))
3553 if (!event
.IsAllowed())
3560 wxDataViewCtrlInternal::row_drop_possible(GtkTreeDragDest
*WXUNUSED(drag_dest
),
3562 GtkSelectionData
*selection_data
)
3564 wxDataViewItem
item(GetOwner()->GTKPathToItem(path
));
3568 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_DROP_POSSIBLE
, m_owner
->GetId() );
3569 event
.SetEventObject( m_owner
);
3570 event
.SetItem( item
);
3571 event
.SetModel( m_wx_model
);
3572 event
.SetDataFormat(gtk_selection_data_get_target(selection_data
));
3573 if (!m_owner
->HandleWindowEvent( event
))
3576 if (!event
.IsAllowed())
3582 // notifications from wxDataViewModel
3584 bool wxDataViewCtrlInternal::Cleared()
3599 void wxDataViewCtrlInternal::Resort()
3601 if (!m_wx_model
->IsVirtualListModel())
3607 bool wxDataViewCtrlInternal::ItemAdded( const wxDataViewItem
&parent
, const wxDataViewItem
&item
)
3609 if (!m_wx_model
->IsVirtualListModel())
3611 wxGtkTreeModelNode
*parent_node
= FindNode( parent
);
3612 wxCHECK_MSG(parent_node
, false,
3613 "Did you forget a call to ItemAdded()? The parent node is unknown to the wxGtkTreeModel");
3615 wxDataViewItemArray modelSiblings
;
3616 m_wx_model
->GetChildren(parent
, modelSiblings
);
3617 const int modelSiblingsSize
= modelSiblings
.size();
3619 int posInModel
= modelSiblings
.Index(item
, /*fromEnd=*/true);
3620 wxCHECK_MSG( posInModel
!= wxNOT_FOUND
, false, "adding non-existent item?" );
3622 const wxGtkTreeModelChildren
& nodeSiblings
= parent_node
->GetChildren();
3623 const int nodeSiblingsSize
= nodeSiblings
.size();
3627 if ( posInModel
== modelSiblingsSize
- 1 )
3629 nodePos
= nodeSiblingsSize
;
3631 else if ( modelSiblingsSize
== nodeSiblingsSize
+ 1 )
3633 // This is the simple case when our node tree already matches the
3634 // model and only this one item is missing.
3635 nodePos
= posInModel
;
3639 // It's possible that a larger discrepancy between the model and
3640 // our realization exists. This can happen e.g. when adding a bunch
3641 // of items to the model and then calling ItemsAdded() just once
3642 // afterwards. In this case, we must find the right position by
3643 // looking at sibling items.
3645 // append to the end if we won't find a better position:
3646 nodePos
= nodeSiblingsSize
;
3648 for ( int nextItemPos
= posInModel
+ 1;
3649 nextItemPos
< modelSiblingsSize
;
3652 int nextNodePos
= parent_node
->FindChildByItem(modelSiblings
[nextItemPos
]);
3653 if ( nextNodePos
!= wxNOT_FOUND
)
3655 nodePos
= nextNodePos
;
3661 if (m_wx_model
->IsContainer( item
))
3662 parent_node
->InsertNode( new wxGtkTreeModelNode( parent_node
, item
, this ), nodePos
);
3664 parent_node
->InsertLeaf( item
.GetID(), nodePos
);
3672 bool wxDataViewCtrlInternal::ItemDeleted( const wxDataViewItem
&parent
, const wxDataViewItem
&item
)
3674 if (!m_wx_model
->IsVirtualListModel())
3676 wxGtkTreeModelNode
*parent_node
= FindNode( parent
);
3677 wxASSERT_MSG(parent_node
,
3678 "Did you forget a call to ItemAdded()? The parent node is unknown to the wxGtkTreeModel");
3680 parent_node
->DeleteChild( item
.GetID() );
3688 bool wxDataViewCtrlInternal::ItemChanged( const wxDataViewItem
&item
)
3690 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_VALUE_CHANGED
, m_owner
->GetId() );
3691 event
.SetEventObject( m_owner
);
3692 event
.SetModel( m_owner
->GetModel() );
3693 event
.SetItem( item
);
3694 m_owner
->HandleWindowEvent( event
);
3699 bool wxDataViewCtrlInternal::ValueChanged( const wxDataViewItem
&item
, unsigned int view_column
)
3701 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_VALUE_CHANGED
, m_owner
->GetId() );
3702 event
.SetEventObject( m_owner
);
3703 event
.SetModel( m_owner
->GetModel() );
3704 event
.SetColumn( view_column
);
3705 event
.SetDataViewColumn( GetOwner()->GetColumn(view_column
) );
3706 event
.SetItem( item
);
3707 m_owner
->HandleWindowEvent( event
);
3714 GtkTreeModelFlags
wxDataViewCtrlInternal::get_flags()
3718 if ( m_wx_model
->IsListModel() )
3719 flags
|= GTK_TREE_MODEL_LIST_ONLY
;
3721 if ( !m_wx_model
->IsVirtualListModel() )
3722 flags
|= GTK_TREE_MODEL_ITERS_PERSIST
;
3724 return GtkTreeModelFlags(flags
);
3727 gboolean
wxDataViewCtrlInternal::get_iter( GtkTreeIter
*iter
, GtkTreePath
*path
)
3730 if (m_wx_model
->IsVirtualListModel())
3732 wxDataViewVirtualListModel
*wx_model
= (wxDataViewVirtualListModel
*) m_wx_model
;
3734 unsigned int i
= (unsigned int)gtk_tree_path_get_indices (path
)[0];
3736 if (i
>= wx_model
->GetCount())
3739 iter
->stamp
= m_gtk_model
->stamp
;
3740 // user_data is just the index +1
3741 iter
->user_data
= wxUIntToPtr(i
+1);
3747 int depth
= gtk_tree_path_get_depth( path
);
3749 wxGtkTreeModelNode
*node
= m_root
;
3752 for (i
= 0; i
< depth
; i
++)
3754 BuildBranch( node
);
3756 gint pos
= gtk_tree_path_get_indices (path
)[i
];
3757 if (pos
< 0) return FALSE
;
3758 if ((size_t)pos
>= node
->GetChildCount()) return FALSE
;
3760 void* id
= node
->GetChildren().Item( (size_t) pos
);
3764 iter
->stamp
= m_gtk_model
->stamp
;
3765 iter
->user_data
= id
;
3769 size_t count
= node
->GetNodes().GetCount();
3771 for (pos2
= 0; pos2
< count
; pos2
++)
3773 wxGtkTreeModelNode
*child_node
= node
->GetNodes().Item( pos2
);
3774 if (child_node
->GetItem().GetID() == id
)
3786 GtkTreePath
*wxDataViewCtrlInternal::get_path( GtkTreeIter
*iter
)
3788 // When this is called from ItemDeleted(), the item is already
3789 // deleted in the model.
3791 GtkTreePath
*retval
= gtk_tree_path_new ();
3793 if (m_wx_model
->IsVirtualListModel())
3795 // iter is root, add nothing
3796 if (!iter
->user_data
)
3799 // user_data is just the index +1
3800 int i
= ( (wxUIntPtr
) iter
->user_data
) -1;
3801 gtk_tree_path_append_index (retval
, i
);
3805 void *id
= iter
->user_data
;
3807 wxGtkTreeModelNode
*node
= FindParentNode( iter
);
3810 int pos
= node
->GetChildren().Index( id
);
3812 gtk_tree_path_prepend_index( retval
, pos
);
3814 id
= node
->GetItem().GetID();
3815 node
= node
->GetParent();
3822 gboolean
wxDataViewCtrlInternal::iter_next( GtkTreeIter
*iter
)
3824 if (m_wx_model
->IsVirtualListModel())
3826 wxDataViewVirtualListModel
*wx_model
= (wxDataViewVirtualListModel
*) m_wx_model
;
3828 // user_data is just the index +1
3829 int n
= ( (wxUIntPtr
) iter
->user_data
) -1;
3833 iter
->user_data
= NULL
;
3837 if (n
>= (int) wx_model
->GetCount()-1)
3839 iter
->user_data
= NULL
;
3843 // user_data is just the index +1 (+2 because we need the next)
3844 iter
->user_data
= wxUIntToPtr(n
+2);
3848 wxGtkTreeModelNode
*parent
= FindParentNode( iter
);
3849 if( parent
== NULL
)
3851 iter
->user_data
= NULL
;
3855 int pos
= parent
->GetChildren().Index( iter
->user_data
);
3857 if (pos
== (int) parent
->GetChildCount()-1)
3859 iter
->user_data
= NULL
;
3863 iter
->user_data
= parent
->GetChildren().Item( pos
+1 );
3869 gboolean
wxDataViewCtrlInternal::iter_children( GtkTreeIter
*iter
, GtkTreeIter
*parent
)
3871 if (m_wx_model
->IsVirtualListModel())
3873 // this is a list, nodes have no children
3877 iter
->stamp
= m_gtk_model
->stamp
;
3878 iter
->user_data
= (gpointer
) 1;
3886 if (m_root
->GetChildCount() == 0) return FALSE
;
3887 iter
->stamp
= m_gtk_model
->stamp
;
3888 iter
->user_data
= (gpointer
) m_root
->GetChildren().Item( 0 );
3893 wxDataViewItem item
;
3895 item
= wxDataViewItem( (void*) parent
->user_data
);
3897 if (!m_wx_model
->IsContainer( item
))
3900 wxGtkTreeModelNode
*parent_node
= FindNode( parent
);
3901 wxASSERT_MSG(parent_node
,
3902 "Did you forget a call to ItemAdded()? The parent node is unknown to the wxGtkTreeModel");
3904 BuildBranch( parent_node
);
3906 if (parent_node
->GetChildCount() == 0)
3909 iter
->stamp
= m_gtk_model
->stamp
;
3910 iter
->user_data
= (gpointer
) parent_node
->GetChildren().Item( 0 );
3916 gboolean
wxDataViewCtrlInternal::iter_has_child( GtkTreeIter
*iter
)
3918 if (m_wx_model
->IsVirtualListModel())
3920 wxDataViewVirtualListModel
*wx_model
= (wxDataViewVirtualListModel
*) m_wx_model
;
3923 return (wx_model
->GetCount() > 0);
3925 // this is a list, nodes have no children
3931 return (m_root
->GetChildCount() > 0);
3933 wxDataViewItem
item( (void*) iter
->user_data
);
3935 bool is_container
= m_wx_model
->IsContainer( item
);
3940 wxGtkTreeModelNode
*node
= FindNode( iter
);
3942 "Did you forget a call to ItemAdded()? The iterator is unknown to the wxGtkTreeModel");
3944 BuildBranch( node
);
3946 return (node
->GetChildCount() > 0);
3950 gint
wxDataViewCtrlInternal::iter_n_children( GtkTreeIter
*iter
)
3952 if (m_wx_model
->IsVirtualListModel())
3954 wxDataViewVirtualListModel
*wx_model
= (wxDataViewVirtualListModel
*) m_wx_model
;
3957 return (gint
) wx_model
->GetCount();
3964 return m_root
->GetChildCount();
3966 wxDataViewItem
item( (void*) iter
->user_data
);
3968 if (!m_wx_model
->IsContainer( item
))
3971 wxGtkTreeModelNode
*parent_node
= FindNode( iter
);
3972 wxASSERT_MSG(parent_node
,
3973 "Did you forget a call to ItemAdded()? The parent node is unknown to the wxGtkTreeModel");
3975 BuildBranch( parent_node
);
3977 return parent_node
->GetChildCount();
3981 gboolean
wxDataViewCtrlInternal::iter_nth_child( GtkTreeIter
*iter
, GtkTreeIter
*parent
, gint n
)
3983 if (m_wx_model
->IsVirtualListModel())
3985 wxDataViewVirtualListModel
*wx_model
= (wxDataViewVirtualListModel
*) m_wx_model
;
3993 if (n
>= (gint
) wx_model
->GetCount())
3996 iter
->stamp
= m_gtk_model
->stamp
;
3997 // user_data is just the index +1
3998 iter
->user_data
= wxUIntToPtr(n
+1);
4005 if (parent
) id
= (void*) parent
->user_data
;
4006 wxDataViewItem
item( id
);
4008 if (!m_wx_model
->IsContainer( item
))
4011 wxGtkTreeModelNode
*parent_node
= FindNode( parent
);
4012 wxASSERT_MSG(parent_node
,
4013 "Did you forget a call to ItemAdded()? The parent node is unknown to the wxGtkTreeModel");
4015 BuildBranch( parent_node
);
4017 iter
->stamp
= m_gtk_model
->stamp
;
4018 iter
->user_data
= parent_node
->GetChildren().Item( n
);
4024 gboolean
wxDataViewCtrlInternal::iter_parent( GtkTreeIter
*iter
, GtkTreeIter
*child
)
4026 if (m_wx_model
->IsVirtualListModel())
4032 wxGtkTreeModelNode
*node
= FindParentNode( child
);
4036 iter
->stamp
= m_gtk_model
->stamp
;
4037 iter
->user_data
= (gpointer
) node
->GetItem().GetID();
4043 // item can be deleted already in the model
4044 int wxDataViewCtrlInternal::GetIndexOf( const wxDataViewItem
&parent
, const wxDataViewItem
&item
)
4046 if (m_wx_model
->IsVirtualListModel())
4048 return wxPtrToUInt(item
.GetID()) - 1;
4052 wxGtkTreeModelNode
*parent_node
= FindNode( parent
);
4053 wxGtkTreeModelChildren
&children
= parent_node
->GetChildren();
4055 for (j
= 0; j
< children
.GetCount(); j
++)
4057 if (children
[j
] == item
.GetID())
4065 static wxGtkTreeModelNode
*
4066 wxDataViewCtrlInternal_FindNode( wxDataViewModel
* model
, wxGtkTreeModelNode
*treeNode
, const wxDataViewItem
&item
)
4072 list
.DeleteContents( true );
4073 wxDataViewItem
it( item
);
4077 wxDataViewItem
* pItem
= new wxDataViewItem( it
);
4078 list
.Insert( pItem
);
4079 it
= model
->GetParent( it
);
4082 wxGtkTreeModelNode
* node
= treeNode
;
4083 for( ItemList::compatibility_iterator n
= list
.GetFirst(); n
; n
= n
->GetNext() )
4085 if( node
&& node
->GetNodes().GetCount() != 0 )
4087 int len
= node
->GetNodes().GetCount();
4088 wxGtkTreeModelNodes
&nodes
= node
->GetNodes();
4090 for( ; j
< len
; j
++)
4092 if( nodes
[j
]->GetItem() == *(n
->GetData()))
4111 wxGtkTreeModelNode
*wxDataViewCtrlInternal::FindNode( GtkTreeIter
*iter
)
4116 wxDataViewItem
item( (void*) iter
->user_data
);
4120 wxGtkTreeModelNode
*result
= wxDataViewCtrlInternal_FindNode( m_wx_model
, m_root
, item
);
4125 wxLogDebug( "Not found %p", iter->user_data );
4129 // TODO: remove this code
4135 wxGtkTreeModelNode
*wxDataViewCtrlInternal::FindNode( const wxDataViewItem
&item
)
4140 wxGtkTreeModelNode
*result
= wxDataViewCtrlInternal_FindNode( m_wx_model
, m_root
, item
);
4145 wxLogDebug( "Not found %p", item.GetID() );
4149 // TODO: remove this code
4155 static wxGtkTreeModelNode
*
4156 wxDataViewCtrlInternal_FindParentNode( wxDataViewModel
* model
, wxGtkTreeModelNode
*treeNode
, const wxDataViewItem
&item
)
4162 list
.DeleteContents( true );
4166 wxDataViewItem
it( model
->GetParent( item
) );
4169 wxDataViewItem
* pItem
= new wxDataViewItem( it
);
4170 list
.Insert( pItem
);
4171 it
= model
->GetParent( it
);
4174 wxGtkTreeModelNode
* node
= treeNode
;
4175 for( ItemList::compatibility_iterator n
= list
.GetFirst(); n
; n
= n
->GetNext() )
4177 if( node
&& node
->GetNodes().GetCount() != 0 )
4179 int len
= node
->GetNodes().GetCount();
4180 wxGtkTreeModelNodes nodes
= node
->GetNodes();
4182 for( ; j
< len
; j
++)
4184 if( nodes
[j
]->GetItem() == *(n
->GetData()))
4199 //Examine whether the node is item's parent node
4200 int len
= node
->GetChildCount();
4201 for( int i
= 0; i
< len
; i
++ )
4203 if( node
->GetChildren().Item( i
) == item
.GetID() )
4209 wxGtkTreeModelNode
*wxDataViewCtrlInternal::FindParentNode( GtkTreeIter
*iter
)
4214 wxDataViewItem
item( (void*) iter
->user_data
);
4218 return wxDataViewCtrlInternal_FindParentNode( m_wx_model
, m_root
, item
);
4221 wxGtkTreeModelNode
*wxDataViewCtrlInternal::FindParentNode( const wxDataViewItem
&item
)
4226 return wxDataViewCtrlInternal_FindParentNode( m_wx_model
, m_root
, item
);
4229 //-----------------------------------------------------------------------------
4230 // wxDataViewCtrl signal callbacks
4231 //-----------------------------------------------------------------------------
4234 wxdataview_selection_changed_callback( GtkTreeSelection
* WXUNUSED(selection
), wxDataViewCtrl
*dv
)
4236 if (!gtk_widget_get_realized(dv
->m_widget
))
4239 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_SELECTION_CHANGED
, dv
->GetId() );
4240 event
.SetItem( dv
->GetSelection() );
4241 event
.SetModel( dv
->GetModel() );
4242 dv
->HandleWindowEvent( event
);
4246 wxdataview_row_activated_callback( GtkTreeView
* WXUNUSED(treeview
), GtkTreePath
*path
,
4247 GtkTreeViewColumn
*WXUNUSED(column
), wxDataViewCtrl
*dv
)
4249 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_ACTIVATED
, dv
->GetId() );
4251 wxDataViewItem
item(dv
->GTKPathToItem(path
));
4252 event
.SetItem( item
);
4253 event
.SetModel( dv
->GetModel() );
4254 dv
->HandleWindowEvent( event
);
4258 wxdataview_test_expand_row_callback( GtkTreeView
* WXUNUSED(treeview
), GtkTreeIter
* iter
,
4259 GtkTreePath
*WXUNUSED(path
), wxDataViewCtrl
*dv
)
4261 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_EXPANDING
, dv
->GetId() );
4263 wxDataViewItem
item( (void*) iter
->user_data
);;
4264 event
.SetItem( item
);
4265 event
.SetModel( dv
->GetModel() );
4266 dv
->HandleWindowEvent( event
);
4268 return !event
.IsAllowed();
4272 wxdataview_row_expanded_callback( GtkTreeView
* WXUNUSED(treeview
), GtkTreeIter
* iter
,
4273 GtkTreePath
*WXUNUSED(path
), wxDataViewCtrl
*dv
)
4275 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_EXPANDED
, dv
->GetId() );
4277 wxDataViewItem
item( (void*) iter
->user_data
);;
4278 event
.SetItem( item
);
4279 event
.SetModel( dv
->GetModel() );
4280 dv
->HandleWindowEvent( event
);
4284 wxdataview_test_collapse_row_callback( GtkTreeView
* WXUNUSED(treeview
), GtkTreeIter
* iter
,
4285 GtkTreePath
*WXUNUSED(path
), wxDataViewCtrl
*dv
)
4287 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_COLLAPSING
, dv
->GetId() );
4289 wxDataViewItem
item( (void*) iter
->user_data
);;
4290 event
.SetItem( item
);
4291 event
.SetModel( dv
->GetModel() );
4292 dv
->HandleWindowEvent( event
);
4294 return !event
.IsAllowed();
4298 wxdataview_row_collapsed_callback( GtkTreeView
* WXUNUSED(treeview
), GtkTreeIter
* iter
,
4299 GtkTreePath
*WXUNUSED(path
), wxDataViewCtrl
*dv
)
4301 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_COLLAPSED
, dv
->GetId() );
4303 wxDataViewItem
item( (void*) iter
->user_data
);;
4304 event
.SetItem( item
);
4305 event
.SetModel( dv
->GetModel() );
4306 dv
->HandleWindowEvent( event
);
4309 //-----------------------------------------------------------------------------
4311 //-----------------------------------------------------------------------------
4313 void wxDataViewCtrl::AddChildGTK(wxWindowGTK
* child
)
4315 GtkWidget
* treeview
= GtkGetTreeView();
4317 // Insert widget in GtkTreeView
4318 if (gtk_widget_get_realized(treeview
))
4319 gtk_widget_set_parent_window( child
->m_widget
,
4320 gtk_tree_view_get_bin_window( GTK_TREE_VIEW(treeview
) ) );
4321 gtk_widget_set_parent( child
->m_widget
, treeview
);
4325 void gtk_dataviewctrl_size_callback( GtkWidget
*WXUNUSED(widget
),
4326 GtkAllocation
*WXUNUSED(gtk_alloc
),
4327 wxDataViewCtrl
*win
)
4329 wxWindowList::compatibility_iterator node
= win
->GetChildren().GetFirst();
4332 wxWindow
*child
= node
->GetData();
4335 gtk_widget_size_request( child
->m_widget
, &req
);
4337 GtkAllocation alloc
;
4338 alloc
.x
= child
->m_x
;
4339 alloc
.y
= child
->m_y
;
4340 alloc
.width
= child
->m_width
;
4341 alloc
.height
= child
->m_height
;
4342 gtk_widget_size_allocate( child
->m_widget
, &alloc
);
4344 node
= node
->GetNext();
4349 //-----------------------------------------------------------------------------
4350 // "motion_notify_event"
4351 //-----------------------------------------------------------------------------
4354 gtk_dataview_motion_notify_callback( GtkWidget
*WXUNUSED(widget
),
4355 GdkEventMotion
*gdk_event
,
4356 wxDataViewCtrl
*dv
)
4358 if (gdk_event
->is_hint
)
4362 GdkModifierType state
;
4363 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
4369 GtkTreeViewColumn
*column
= NULL
;
4372 if (gtk_tree_view_get_path_at_pos(
4373 GTK_TREE_VIEW(dv
->GtkGetTreeView()),
4374 (int) gdk_event
->x
, (int) gdk_event
->y
,
4383 dv
->GtkGetInternal()->get_iter( &iter
, path
);
4391 //-----------------------------------------------------------------------------
4392 // "button_press_event"
4393 //-----------------------------------------------------------------------------
4396 gtk_dataview_button_press_callback( GtkWidget
*WXUNUSED(widget
),
4397 GdkEventButton
*gdk_event
,
4398 wxDataViewCtrl
*dv
)
4400 if ((gdk_event
->button
== 3) && (gdk_event
->type
== GDK_BUTTON_PRESS
))
4403 GtkTreeViewColumn
*column
= NULL
;
4406 gtk_tree_view_get_path_at_pos
4408 GTK_TREE_VIEW(dv
->GtkGetTreeView()),
4409 (int) gdk_event
->x
, (int) gdk_event
->y
,
4416 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ITEM_CONTEXT_MENU
, dv
->GetId() );
4418 event
.SetItem(dv
->GTKPathToItem(path
));
4419 event
.SetModel( dv
->GetModel() );
4420 return dv
->HandleWindowEvent( event
);
4426 IMPLEMENT_DYNAMIC_CLASS(wxDataViewCtrl
, wxDataViewCtrlBase
)
4428 wxDataViewCtrl::~wxDataViewCtrl()
4430 // Stop editing before destroying the control to remove any event handlers
4431 // which are added when editing started: if we didn't do this, the base
4432 // class dtor would assert as it checks for any leftover handlers.
4435 GtkTreeViewColumn
*col
;
4436 gtk_tree_view_get_cursor(GTK_TREE_VIEW(m_treeview
), NULL
, &col
);
4438 wxDataViewColumn
* const wxcol
= FromGTKColumn(col
);
4441 // This won't do anything if we're not editing it
4442 wxcol
->GetRenderer()->CancelEditing();
4451 void wxDataViewCtrl::Init()
4455 m_cols
.DeleteContents( true );
4457 m_uniformRowHeight
= -1;
4460 bool wxDataViewCtrl::Create(wxWindow
*parent
,
4465 const wxValidator
& validator
,
4466 const wxString
& name
)
4468 if (!PreCreation( parent
, pos
, size
) ||
4469 !CreateBase( parent
, id
, pos
, size
, style
, validator
, name
))
4471 wxFAIL_MSG( wxT("wxDataViewCtrl creation failed") );
4475 m_widget
= gtk_scrolled_window_new (NULL
, NULL
);
4476 g_object_ref(m_widget
);
4478 GTKScrolledWindowSetBorder(m_widget
, style
);
4480 m_treeview
= gtk_tree_view_new();
4481 gtk_container_add (GTK_CONTAINER (m_widget
), m_treeview
);
4483 m_focusWidget
= GTK_WIDGET(m_treeview
);
4485 g_signal_connect (m_treeview
, "size_allocate",
4486 G_CALLBACK (gtk_dataviewctrl_size_callback
), this);
4489 if (!gtk_check_version(2,6,0))
4491 bool fixed
= (style
& wxDV_VARIABLE_LINE_HEIGHT
) == 0;
4492 gtk_tree_view_set_fixed_height_mode( GTK_TREE_VIEW(m_treeview
), fixed
);
4496 if (style
& wxDV_MULTIPLE
)
4498 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
4499 gtk_tree_selection_set_mode( selection
, GTK_SELECTION_MULTIPLE
);
4502 gtk_tree_view_set_headers_visible( GTK_TREE_VIEW(m_treeview
), (style
& wxDV_NO_HEADER
) == 0 );
4505 if (!gtk_check_version(2,10,0))
4507 GtkTreeViewGridLines grid
= GTK_TREE_VIEW_GRID_LINES_NONE
;
4509 if ((style
& wxDV_HORIZ_RULES
) != 0 &&
4510 (style
& wxDV_VERT_RULES
) != 0)
4511 grid
= GTK_TREE_VIEW_GRID_LINES_BOTH
;
4512 else if (style
& wxDV_VERT_RULES
)
4513 grid
= GTK_TREE_VIEW_GRID_LINES_VERTICAL
;
4514 else if (style
& wxDV_HORIZ_RULES
)
4515 grid
= GTK_TREE_VIEW_GRID_LINES_HORIZONTAL
;
4517 if (grid
!= GTK_TREE_VIEW_GRID_LINES_NONE
)
4518 gtk_tree_view_set_grid_lines( GTK_TREE_VIEW(m_treeview
), grid
);
4522 gtk_tree_view_set_rules_hint( GTK_TREE_VIEW(m_treeview
), (style
& wxDV_ROW_LINES
) != 0 );
4524 gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (m_widget
),
4525 GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC
);
4526 gtk_widget_show (m_treeview
);
4528 m_parent
->DoAddChild( this );
4532 GtkEnableSelectionEvents();
4534 g_signal_connect_after (m_treeview
, "row-activated",
4535 G_CALLBACK (wxdataview_row_activated_callback
), this);
4537 g_signal_connect (m_treeview
, "test-collapse-row",
4538 G_CALLBACK (wxdataview_test_collapse_row_callback
), this);
4540 g_signal_connect_after (m_treeview
, "row-collapsed",
4541 G_CALLBACK (wxdataview_row_collapsed_callback
), this);
4543 g_signal_connect (m_treeview
, "test-expand-row",
4544 G_CALLBACK (wxdataview_test_expand_row_callback
), this);
4546 g_signal_connect_after (m_treeview
, "row-expanded",
4547 G_CALLBACK (wxdataview_row_expanded_callback
), this);
4549 g_signal_connect (m_treeview
, "motion_notify_event",
4550 G_CALLBACK (gtk_dataview_motion_notify_callback
), this);
4552 g_signal_connect (m_treeview
, "button_press_event",
4553 G_CALLBACK (gtk_dataview_button_press_callback
), this);
4558 wxDataViewItem
wxDataViewCtrl::GTKPathToItem(GtkTreePath
*path
) const
4561 return wxDataViewItem(path
&& m_internal
->get_iter(&iter
, path
)
4566 void wxDataViewCtrl::OnInternalIdle()
4568 wxWindow::OnInternalIdle();
4570 m_internal
->OnInternalIdle();
4572 unsigned int cols
= GetColumnCount();
4574 for (i
= 0; i
< cols
; i
++)
4576 wxDataViewColumn
*col
= GetColumn( i
);
4577 col
->OnInternalIdle();
4580 if (m_ensureVisibleDefered
.IsOk())
4582 ExpandAncestors(m_ensureVisibleDefered
);
4584 iter
.user_data
= (gpointer
) m_ensureVisibleDefered
.GetID();
4585 wxGtkTreePath
path(m_internal
->get_path( &iter
));
4586 gtk_tree_view_scroll_to_cell( GTK_TREE_VIEW(m_treeview
), path
, NULL
, false, 0.0, 0.0 );
4587 m_ensureVisibleDefered
= wxDataViewItem(0);
4591 bool wxDataViewCtrl::AssociateModel( wxDataViewModel
*model
)
4593 wxDELETE(m_internal
);
4595 if (!wxDataViewCtrlBase::AssociateModel( model
))
4599 if (!gtk_check_version(2,6,0))
4601 bool fixed
= (((GetWindowStyle() & wxDV_VARIABLE_LINE_HEIGHT
) == 0) || (model
->IsVirtualListModel()));
4602 gtk_tree_view_set_fixed_height_mode( GTK_TREE_VIEW(m_treeview
), fixed
);
4606 m_internal
= new wxDataViewCtrlInternal( this, model
);
4611 bool wxDataViewCtrl::EnableDragSource( const wxDataFormat
&format
)
4613 return m_internal
->EnableDragSource( format
);
4616 bool wxDataViewCtrl::EnableDropTarget( const wxDataFormat
&format
)
4618 return m_internal
->EnableDropTarget( format
);
4621 bool wxDataViewCtrl::AppendColumn( wxDataViewColumn
*col
)
4623 if (!wxDataViewCtrlBase::AppendColumn(col
))
4626 m_cols
.Append( col
);
4629 if (!gtk_check_version(2,6,0))
4631 if (gtk_tree_view_column_get_sizing( GTK_TREE_VIEW_COLUMN(col
->GetGtkHandle()) ) !=
4632 GTK_TREE_VIEW_COLUMN_FIXED
)
4633 gtk_tree_view_set_fixed_height_mode( GTK_TREE_VIEW(m_treeview
), FALSE
);
4637 gtk_tree_view_append_column( GTK_TREE_VIEW(m_treeview
),
4638 GTK_TREE_VIEW_COLUMN(col
->GetGtkHandle()) );
4643 bool wxDataViewCtrl::PrependColumn( wxDataViewColumn
*col
)
4645 if (!wxDataViewCtrlBase::PrependColumn(col
))
4648 m_cols
.Insert( col
);
4651 if (!gtk_check_version(2,6,0))
4653 if (gtk_tree_view_column_get_sizing( GTK_TREE_VIEW_COLUMN(col
->GetGtkHandle()) ) !=
4654 GTK_TREE_VIEW_COLUMN_FIXED
)
4655 gtk_tree_view_set_fixed_height_mode( GTK_TREE_VIEW(m_treeview
), FALSE
);
4659 gtk_tree_view_insert_column( GTK_TREE_VIEW(m_treeview
),
4660 GTK_TREE_VIEW_COLUMN(col
->GetGtkHandle()), 0 );
4665 bool wxDataViewCtrl::InsertColumn( unsigned int pos
, wxDataViewColumn
*col
)
4667 if (!wxDataViewCtrlBase::InsertColumn(pos
,col
))
4670 m_cols
.Insert( pos
, col
);
4673 if (!gtk_check_version(2,6,0))
4675 if (gtk_tree_view_column_get_sizing( GTK_TREE_VIEW_COLUMN(col
->GetGtkHandle()) ) !=
4676 GTK_TREE_VIEW_COLUMN_FIXED
)
4677 gtk_tree_view_set_fixed_height_mode( GTK_TREE_VIEW(m_treeview
), FALSE
);
4681 gtk_tree_view_insert_column( GTK_TREE_VIEW(m_treeview
),
4682 GTK_TREE_VIEW_COLUMN(col
->GetGtkHandle()), pos
);
4687 unsigned int wxDataViewCtrl::GetColumnCount() const
4689 return m_cols
.GetCount();
4692 wxDataViewColumn
* wxDataViewCtrl::FromGTKColumn(GtkTreeViewColumn
*gtk_col
) const
4697 wxDataViewColumnList::const_iterator iter
;
4698 for (iter
= m_cols
.begin(); iter
!= m_cols
.end(); ++iter
)
4700 wxDataViewColumn
*col
= *iter
;
4701 if (GTK_TREE_VIEW_COLUMN(col
->GetGtkHandle()) == gtk_col
)
4707 wxFAIL_MSG( "No matching column?" );
4712 wxDataViewColumn
* wxDataViewCtrl::GetColumn( unsigned int pos
) const
4714 GtkTreeViewColumn
*gtk_col
= gtk_tree_view_get_column( GTK_TREE_VIEW(m_treeview
), pos
);
4716 return FromGTKColumn(gtk_col
);
4719 bool wxDataViewCtrl::DeleteColumn( wxDataViewColumn
*column
)
4721 gtk_tree_view_remove_column( GTK_TREE_VIEW(m_treeview
),
4722 GTK_TREE_VIEW_COLUMN(column
->GetGtkHandle()) );
4724 m_cols
.DeleteObject( column
);
4729 bool wxDataViewCtrl::ClearColumns()
4731 wxDataViewColumnList::iterator iter
;
4732 for (iter
= m_cols
.begin(); iter
!= m_cols
.end(); ++iter
)
4734 wxDataViewColumn
*col
= *iter
;
4735 gtk_tree_view_remove_column( GTK_TREE_VIEW(m_treeview
),
4736 GTK_TREE_VIEW_COLUMN(col
->GetGtkHandle()) );
4744 int wxDataViewCtrl::GetColumnPosition( const wxDataViewColumn
*column
) const
4746 GtkTreeViewColumn
*gtk_column
= GTK_TREE_VIEW_COLUMN(column
->GetGtkHandle());
4748 wxGtkList
list(gtk_tree_view_get_columns(GTK_TREE_VIEW(m_treeview
)));
4750 return g_list_index( list
, (gconstpointer
) gtk_column
);
4753 wxDataViewColumn
*wxDataViewCtrl::GetSortingColumn() const
4755 return m_internal
->GetDataViewSortColumn();
4758 void wxDataViewCtrl::Expand( const wxDataViewItem
& item
)
4761 iter
.user_data
= item
.GetID();
4762 wxGtkTreePath
path(m_internal
->get_path( &iter
));
4763 gtk_tree_view_expand_row( GTK_TREE_VIEW(m_treeview
), path
, false );
4766 void wxDataViewCtrl::Collapse( const wxDataViewItem
& item
)
4769 iter
.user_data
= item
.GetID();
4770 wxGtkTreePath
path(m_internal
->get_path( &iter
));
4771 gtk_tree_view_collapse_row( GTK_TREE_VIEW(m_treeview
), path
);
4774 bool wxDataViewCtrl::IsExpanded( const wxDataViewItem
& item
) const
4777 iter
.user_data
= item
.GetID();
4778 wxGtkTreePath
path(m_internal
->get_path( &iter
));
4779 return gtk_tree_view_row_expanded( GTK_TREE_VIEW(m_treeview
), path
);
4782 wxDataViewItem
wxDataViewCtrl::DoGetCurrentItem() const
4784 // The tree doesn't have any current item if it hadn't been created yet but
4785 // it's arguably not an error to call this function in this case so just
4786 // return an invalid item without asserting.
4788 return wxDataViewItem();
4791 gtk_tree_view_get_cursor(GTK_TREE_VIEW(m_treeview
), path
.ByRef(), NULL
);
4793 return GTKPathToItem(path
);
4796 void wxDataViewCtrl::DoSetCurrentItem(const wxDataViewItem
& item
)
4798 wxCHECK_RET( m_treeview
,
4799 "Current item can't be set before creating the control." );
4801 // We need to make sure the model knows about this item or the path would
4802 // be invalid and gtk_tree_view_set_cursor() would silently do nothing.
4803 ExpandAncestors(item
);
4805 // We also need to preserve the existing selection from changing.
4806 // Unfortunately the only way to do it seems to use our own selection
4807 // function and forbid any selection changes during set cursor call.
4808 wxGtkTreeSelectionLock
4809 lock(gtk_tree_view_get_selection(GTK_TREE_VIEW(m_treeview
)));
4811 // Do move the cursor now.
4813 iter
.user_data
= item
.GetID();
4814 wxGtkTreePath
path(m_internal
->get_path( &iter
));
4816 gtk_tree_view_set_cursor(GTK_TREE_VIEW(m_treeview
), path
, NULL
, FALSE
);
4819 void wxDataViewCtrl::StartEditor(const wxDataViewItem
& item
, unsigned int column
)
4821 wxCHECK_RET( m_treeview
,
4822 "Current item can't be set before creating the control." );
4824 // We need to make sure the model knows about this item or the path would
4825 // be invalid and gtk_tree_view_set_cursor() would silently do nothing.
4826 ExpandAncestors(item
);
4828 wxDataViewColumn
*dvcolumn
= GetColumn(column
);
4829 wxASSERT_MSG(dvcolumn
, "Could not retrieve column");
4830 GtkTreeViewColumn
*gcolumn
= GTK_TREE_VIEW_COLUMN(dvcolumn
->GetGtkHandle());
4832 // We also need to preserve the existing selection from changing.
4833 // Unfortunately the only way to do it seems to use our own selection
4834 // function and forbid any selection changes during set cursor call.
4835 wxGtkTreeSelectionLock
4836 lock(gtk_tree_view_get_selection(GTK_TREE_VIEW(m_treeview
)));
4838 // Do move the cursor now.
4840 iter
.user_data
= item
.GetID();
4841 wxGtkTreePath
path(m_internal
->get_path( &iter
));
4843 gtk_tree_view_set_cursor(GTK_TREE_VIEW(m_treeview
), path
, gcolumn
, TRUE
);
4846 int wxDataViewCtrl::GetSelectedItemsCount() const
4848 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
4850 return gtk_tree_selection_count_selected_rows(selection
);
4853 int wxDataViewCtrl::GetSelections( wxDataViewItemArray
& sel
) const
4857 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
4858 if (HasFlag(wxDV_MULTIPLE
))
4860 GtkTreeModel
*model
;
4861 wxGtkTreePathList
list(gtk_tree_selection_get_selected_rows(selection
, &model
));
4863 for ( GList
* current
= list
; current
; current
= g_list_next(current
) )
4865 GtkTreePath
*path
= (GtkTreePath
*) current
->data
;
4867 sel
.Add(GTKPathToItem(path
));
4873 if (gtk_tree_selection_get_selected( selection
, NULL
, &iter
))
4875 sel
.Add( wxDataViewItem(iter
.user_data
) );
4882 void wxDataViewCtrl::SetSelections( const wxDataViewItemArray
& sel
)
4884 GtkDisableSelectionEvents();
4886 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
4888 gtk_tree_selection_unselect_all( selection
);
4890 wxDataViewItem last_parent
;
4893 for (i
= 0; i
< sel
.GetCount(); i
++)
4895 wxDataViewItem item
= sel
[i
];
4896 wxDataViewItem parent
= GetModel()->GetParent( item
);
4899 if (parent
!= last_parent
)
4900 ExpandAncestors(item
);
4902 last_parent
= parent
;
4905 iter
.stamp
= m_internal
->GetGtkModel()->stamp
;
4906 iter
.user_data
= (gpointer
) item
.GetID();
4907 gtk_tree_selection_select_iter( selection
, &iter
);
4910 GtkEnableSelectionEvents();
4913 void wxDataViewCtrl::Select( const wxDataViewItem
& item
)
4915 ExpandAncestors(item
);
4917 GtkDisableSelectionEvents();
4919 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
4922 iter
.stamp
= m_internal
->GetGtkModel()->stamp
;
4923 iter
.user_data
= (gpointer
) item
.GetID();
4924 gtk_tree_selection_select_iter( selection
, &iter
);
4926 GtkEnableSelectionEvents();
4929 void wxDataViewCtrl::Unselect( const wxDataViewItem
& item
)
4931 GtkDisableSelectionEvents();
4933 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
4936 iter
.stamp
= m_internal
->GetGtkModel()->stamp
;
4937 iter
.user_data
= (gpointer
) item
.GetID();
4938 gtk_tree_selection_unselect_iter( selection
, &iter
);
4940 GtkEnableSelectionEvents();
4943 bool wxDataViewCtrl::IsSelected( const wxDataViewItem
& item
) const
4945 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
4948 iter
.stamp
= m_internal
->GetGtkModel()->stamp
;
4949 iter
.user_data
= (gpointer
) item
.GetID();
4951 return gtk_tree_selection_iter_is_selected( selection
, &iter
);
4954 void wxDataViewCtrl::SelectAll()
4956 GtkDisableSelectionEvents();
4958 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
4960 gtk_tree_selection_select_all( selection
);
4962 GtkEnableSelectionEvents();
4965 void wxDataViewCtrl::UnselectAll()
4967 GtkDisableSelectionEvents();
4969 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
4971 gtk_tree_selection_unselect_all( selection
);
4973 GtkEnableSelectionEvents();
4976 void wxDataViewCtrl::EnsureVisible(const wxDataViewItem
& item
,
4977 const wxDataViewColumn
*WXUNUSED(column
))
4979 m_ensureVisibleDefered
= item
;
4980 ExpandAncestors(item
);
4983 iter
.user_data
= (gpointer
) item
.GetID();
4984 wxGtkTreePath
path(m_internal
->get_path( &iter
));
4985 gtk_tree_view_scroll_to_cell( GTK_TREE_VIEW(m_treeview
), path
, NULL
, false, 0.0, 0.0 );
4988 void wxDataViewCtrl::HitTest(const wxPoint
& point
,
4989 wxDataViewItem
& item
,
4990 wxDataViewColumn
*& column
) const
4992 // gtk_tree_view_get_dest_row_at_pos() is the right one. But it does not tell the column.
4993 // gtk_tree_view_get_path_at_pos() is the wrong function. It doesn't mind the header but returns column.
4994 // See http://mail.gnome.org/archives/gtkmm-list/2005-January/msg00080.html
4995 // So we have to use both of them.
4996 // Friedrich Haase 2010-9-20
4997 wxGtkTreePath path
, pathScratch
;
4998 GtkTreeViewColumn
* GtkColumn
= NULL
;
4999 GtkTreeViewDropPosition pos
= GTK_TREE_VIEW_DROP_INTO_OR_AFTER
;
5003 // cannot directly call GtkGetTreeView(), HitTest is const and so is this pointer
5004 wxDataViewCtrl
* ctrl
= (wxDataViewCtrl
*)this; // ugly workaround, ctrl is NOT const
5005 GtkTreeView
* treeView
= GTK_TREE_VIEW(ctrl
->GtkGetTreeView());
5007 // is there possibly a better suited function to get the column?
5008 gtk_tree_view_get_path_at_pos( // and this is the wrong call but it delivers the column
5010 (int) point
.x
, (int) point
.y
,
5011 pathScratch
.ByRef(),
5012 &GtkColumn
, // here we get the GtkColumn
5016 if ( GtkColumn
!= NULL
)
5018 // we got GTK column
5019 // the right call now which takes the header into account
5020 gtk_tree_view_get_dest_row_at_pos( treeView
, (int) point
.x
, (int) point
.y
, path
.ByRef(), &pos
);
5023 item
= wxDataViewItem(GTKPathToItem(path
));
5024 // else we got a GTK column but the position is not over an item, e.g. below last item
5025 for ( unsigned int i
=0, cols
=GetColumnCount(); i
<cols
; ++i
) // search the wx column
5027 wxDataViewColumn
* col
= GetColumn(i
);
5028 if ( GTK_TREE_VIEW_COLUMN(col
->GetGtkHandle()) == GtkColumn
)
5030 column
= col
; // here we get the wx column
5035 // else no column and thus no item, both null
5039 wxDataViewCtrl::GetItemRect(const wxDataViewItem
& WXUNUSED(item
),
5040 const wxDataViewColumn
*WXUNUSED(column
)) const
5045 bool wxDataViewCtrl::SetRowHeight(int rowHeight
)
5047 m_uniformRowHeight
= rowHeight
;
5051 void wxDataViewCtrl::DoSetExpanderColumn()
5053 gtk_tree_view_set_expander_column( GTK_TREE_VIEW(m_treeview
),
5054 GTK_TREE_VIEW_COLUMN( GetExpanderColumn()->GetGtkHandle() ) );
5057 void wxDataViewCtrl::DoSetIndent()
5061 void wxDataViewCtrl::GtkDisableSelectionEvents()
5063 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
5064 g_signal_handlers_disconnect_by_func( selection
,
5065 (gpointer
) (wxdataview_selection_changed_callback
), this);
5068 void wxDataViewCtrl::GtkEnableSelectionEvents()
5070 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
5071 g_signal_connect_after (selection
, "changed",
5072 G_CALLBACK (wxdataview_selection_changed_callback
), this);
5075 // ----------------------------------------------------------------------------
5076 // visual attributes stuff
5077 // ----------------------------------------------------------------------------
5081 wxDataViewCtrl::GetClassDefaultAttributes(wxWindowVariant
WXUNUSED(variant
))
5083 return GetDefaultAttributesFromGTKWidget(gtk_tree_view_new
);
5086 void wxDataViewCtrl::DoApplyWidgetStyle(GtkRcStyle
*style
)
5088 wxDataViewCtrlBase::DoApplyWidgetStyle(style
);
5089 gtk_widget_modify_style(m_treeview
, style
);
5092 #endif // !wxUSE_GENERICDATAVIEWCTRL
5094 #endif // wxUSE_DATAVIEWCTRL