1 /////////////////////////////////////////////////////////////////////////////
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"
15 #if wxUSE_DATAVIEWCTRL
17 #include "wx/dataview.h"
18 #include "wx/stockitem.h"
19 #include "wx/dcclient.h"
21 #include "wx/gtk/private.h"
22 #include "wx/gtk/win_gtk.h"
24 #include <gobject/gvaluecollector.h>
25 #include <gtk/gtktreemodel.h>
26 #include <gtk/gtktreednd.h>
28 //-----------------------------------------------------------------------------
30 //-----------------------------------------------------------------------------
34 //-----------------------------------------------------------------------------
36 //-----------------------------------------------------------------------------
38 extern void wxapp_install_idle_handler();
41 //-----------------------------------------------------------------------------
43 //-----------------------------------------------------------------------------
45 extern bool g_blockEventsOnDrag
;
47 //-----------------------------------------------------------------------------
48 // define new GTK+ class wxGtkListStore
49 //-----------------------------------------------------------------------------
53 #define GTK_TYPE_WX_LIST_STORE (gtk_wx_list_store_get_type ())
54 #define GTK_WX_LIST_STORE(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_LIST_STORE, GtkWxListStore))
55 #define GTK_WX_LIST_STORE_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GTK_TYPE_WX_LIST_STORE, GtkWxListStoreClass))
56 #define GTK_IS_WX_LIST_STORE(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_LIST_STORE))
57 #define GTK_IS_WX_LIST_STORE_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_LIST_STORE))
58 #define GTK_WX_LIST_STORE_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_WX_LIST_STORE, GtkWxListStoreClass))
60 GType
gtk_wx_list_store_get_type (void);
62 typedef struct _GtkWxListStore GtkWxListStore
;
63 typedef struct _GtkWxListStoreClass GtkWxListStoreClass
;
65 struct _GtkWxListStore
71 wxDataViewListModel
*model
;
74 struct _GtkWxListStoreClass
76 GObjectClass list_parent_class
;
80 static GtkWxListStore
*wxgtk_list_store_new (void);
81 static void wxgtk_list_store_init (GtkWxListStore
*list_store
);
82 static void wxgtk_list_store_class_init (GtkWxListStoreClass
*klass
);
83 static void wxgtk_list_store_tree_model_init (GtkTreeModelIface
*iface
);
84 static void wxgtk_list_store_finalize (GObject
*object
);
85 static GtkTreeModelFlags
wxgtk_list_store_get_flags (GtkTreeModel
*tree_model
);
86 static gint
wxgtk_list_store_get_n_columns (GtkTreeModel
*tree_model
);
87 static GType
wxgtk_list_store_get_column_type (GtkTreeModel
*tree_model
,
89 static gboolean
wxgtk_list_store_get_iter (GtkTreeModel
*tree_model
,
92 static GtkTreePath
*wxgtk_list_store_get_path (GtkTreeModel
*tree_model
,
94 static void wxgtk_list_store_get_value (GtkTreeModel
*tree_model
,
98 static gboolean
wxgtk_list_store_iter_next (GtkTreeModel
*tree_model
,
100 static gboolean
wxgtk_list_store_iter_children (GtkTreeModel
*tree_model
,
102 GtkTreeIter
*parent
);
103 static gboolean
wxgtk_list_store_iter_has_child (GtkTreeModel
*tree_model
,
105 static gint
wxgtk_list_store_iter_n_children (GtkTreeModel
*tree_model
,
107 static gboolean
wxgtk_list_store_iter_nth_child (GtkTreeModel
*tree_model
,
111 static gboolean
wxgtk_list_store_iter_parent (GtkTreeModel
*tree_model
,
115 static GObjectClass
*list_parent_class
= NULL
;
118 gtk_wx_list_store_get_type (void)
120 static GType list_store_type
= 0;
122 if (!list_store_type
)
124 static const GTypeInfo list_store_info
=
126 sizeof (GtkWxListStoreClass
),
127 NULL
, /* base_init */
128 NULL
, /* base_finalize */
129 (GClassInitFunc
) wxgtk_list_store_class_init
,
130 NULL
, /* class_finalize */
131 NULL
, /* class_data */
132 sizeof (GtkWxListStore
),
134 (GInstanceInitFunc
) wxgtk_list_store_init
,
137 static const GInterfaceInfo tree_model_info
=
139 (GInterfaceInitFunc
) wxgtk_list_store_tree_model_init
,
144 list_store_type
= g_type_register_static (G_TYPE_OBJECT
, "GtkWxListStore",
145 &list_store_info
, (GTypeFlags
)0 );
147 g_type_add_interface_static (list_store_type
,
152 return list_store_type
;
155 static GtkWxListStore
*
156 wxgtk_list_store_new(void)
158 GtkWxListStore
*retval
= (GtkWxListStore
*) g_object_new (GTK_TYPE_WX_LIST_STORE
, NULL
);
163 wxgtk_list_store_class_init (GtkWxListStoreClass
*klass
)
165 list_parent_class
= (GObjectClass
*) g_type_class_peek_parent (klass
);
166 GObjectClass
*object_class
= (GObjectClass
*) klass
;
167 object_class
->finalize
= wxgtk_list_store_finalize
;
171 wxgtk_list_store_tree_model_init (GtkTreeModelIface
*iface
)
173 iface
->get_flags
= wxgtk_list_store_get_flags
;
174 iface
->get_n_columns
= wxgtk_list_store_get_n_columns
;
175 iface
->get_column_type
= wxgtk_list_store_get_column_type
;
176 iface
->get_iter
= wxgtk_list_store_get_iter
;
177 iface
->get_path
= wxgtk_list_store_get_path
;
178 iface
->get_value
= wxgtk_list_store_get_value
;
179 iface
->iter_next
= wxgtk_list_store_iter_next
;
180 iface
->iter_children
= wxgtk_list_store_iter_children
;
181 iface
->iter_has_child
= wxgtk_list_store_iter_has_child
;
182 iface
->iter_n_children
= wxgtk_list_store_iter_n_children
;
183 iface
->iter_nth_child
= wxgtk_list_store_iter_nth_child
;
184 iface
->iter_parent
= wxgtk_list_store_iter_parent
;
188 wxgtk_list_store_init (GtkWxListStore
*list_store
)
190 list_store
->model
= NULL
;
191 list_store
->stamp
= g_random_int();
195 wxgtk_list_store_finalize (GObject
*object
)
197 /* GtkWxListStore *list_store = GTK_WX_LIST_STORE (object); */
199 /* we need to sort out, which class deletes what */
200 /* delete list_store->model; */
203 (* list_parent_class
->finalize
) (object
);
208 //-----------------------------------------------------------------------------
209 // implement callbacks from wxGtkListStore class by letting
210 // them call the methods of wxWidgets' wxDataViewListModel
211 //-----------------------------------------------------------------------------
213 static GtkTreeModelFlags
214 wxgtk_list_store_get_flags (GtkTreeModel
*tree_model
)
216 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), (GtkTreeModelFlags
)0 );
218 // GTK+ list store uses a linked list for storing the
219 // items and a pointer to a child is used as the member
220 // field of a GtkTreeIter. This means that the iter is
221 // valid in the GtkListStore as long as the child exists.
222 // We use the index of the row and since the index of a
223 // specific row will change if a row above is deleted,
224 // the iter does not persist
225 return /* GTK_TREE_MODEL_ITERS_PERSIST | */ GTK_TREE_MODEL_LIST_ONLY
;
229 wxgtk_list_store_get_n_columns (GtkTreeModel
*tree_model
)
231 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
232 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), 0);
234 return list_store
->model
->GetNumberOfCols();
238 wxgtk_list_store_get_column_type (GtkTreeModel
*tree_model
,
241 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
242 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), G_TYPE_INVALID
);
244 GType gtype
= G_TYPE_INVALID
;
246 wxString wxtype
= list_store
->model
->GetColType( (size_t) index
);
248 if (wxtype
== wxT("string"))
249 gtype
= G_TYPE_STRING
;
255 wxgtk_list_store_get_iter (GtkTreeModel
*tree_model
,
259 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
260 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), FALSE
);
261 g_return_val_if_fail (gtk_tree_path_get_depth (path
) > 0, FALSE
);
263 size_t i
= (size_t)gtk_tree_path_get_indices (path
)[0];
265 if (i
>= list_store
->model
->GetNumberOfRows())
268 iter
->stamp
= list_store
->stamp
;
269 // user_data is just the index
270 iter
->user_data
= (gpointer
) i
;
276 wxgtk_list_store_get_path (GtkTreeModel
*tree_model
,
279 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), NULL
);
280 g_return_val_if_fail (iter
->stamp
== GTK_WX_LIST_STORE (tree_model
)->stamp
, NULL
);
282 GtkTreePath
*retval
= gtk_tree_path_new ();
283 // user_data is just the index
284 int i
= (int) iter
->user_data
;
285 gtk_tree_path_append_index (retval
, i
);
290 wxgtk_list_store_get_value (GtkTreeModel
*tree_model
,
295 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
296 g_return_if_fail (GTK_IS_WX_LIST_STORE (tree_model
) );
298 wxDataViewListModel
*model
= list_store
->model
;
299 wxString mtype
= model
->GetColType( (size_t) column
);
300 if (mtype
== wxT("string"))
302 g_value_init( value
, G_TYPE_STRING
);
303 wxVariant variant
= model
->GetValue( (size_t) column
, (size_t) iter
->user_data
);
304 g_value_set_string( value
, wxGTK_CONV(variant
.GetString()) );
311 GtkTreeDataList
*list
;
312 gint tmp_column
= column
;
314 g_return_if_fail (column
< GTK_LIST_STORE (tree_model
)->n_columns
);
315 g_return_if_fail (GTK_LIST_STORE (tree_model
)->stamp
== iter
->stamp
);
317 list
= G_SLIST (iter
->user_data
)->data
;
319 while (tmp_column
-- > 0 && list
)
323 g_value_init (value
, GTK_LIST_STORE (tree_model
)->column_headers
[column
]);
325 _gtk_tree_data_list_node_to_value (list
,
326 GTK_LIST_STORE (tree_model
)->column_headers
[column
],
333 wxgtk_list_store_iter_next (GtkTreeModel
*tree_model
,
336 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), FALSE
);
337 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
339 g_return_val_if_fail (list_store
->stamp
== iter
->stamp
, FALSE
);
341 int n
= (int) iter
->user_data
;
346 if (n
>= (int) list_store
->model
->GetNumberOfRows()-1)
349 iter
->user_data
= (gpointer
) ++n
;
355 wxgtk_list_store_iter_children (GtkTreeModel
*tree_model
,
359 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), FALSE
);
360 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
362 // this is a list, nodes have no children
366 iter
->stamp
= list_store
->stamp
;
367 iter
->user_data
= (gpointer
) -1;
373 wxgtk_list_store_iter_has_child (GtkTreeModel
*tree_model
,
380 wxgtk_list_store_iter_n_children (GtkTreeModel
*tree_model
,
383 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), -1);
384 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
387 return (gint
) list_store
->model
->GetNumberOfRows();
389 g_return_val_if_fail (list_store
->stamp
== iter
->stamp
, -1);
395 wxgtk_list_store_iter_nth_child (GtkTreeModel
*tree_model
,
400 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), FALSE
);
401 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
409 if (n
>= (gint
) list_store
->model
->GetNumberOfRows())
412 iter
->stamp
= list_store
->stamp
;
413 iter
->user_data
= (gpointer
) n
;
419 wxgtk_list_store_iter_parent (GtkTreeModel
*tree_model
,
426 //-----------------------------------------------------------------------------
427 // define new GTK+ class wxGtkCellRenderer
428 //-----------------------------------------------------------------------------
432 #define GTK_TYPE_WX_CELL_RENDERER (gtk_wx_cell_renderer_get_type ())
433 #define GTK_WX_CELL_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRenderer))
434 #define GTK_WX_CELL_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRendererClass))
435 #define GTK_IS_WX_CELL_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_CELL_RENDERER))
436 #define GTK_IS_WX_CELL_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_CELL_RENDERER))
437 #define GTK_WX_CELL_RENDERER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRendererClass))
439 GType
gtk_wx_cell_renderer_get_type (void);
441 typedef struct _GtkWxCellRenderer GtkWxCellRenderer
;
442 typedef struct _GtkWxCellRendererClass GtkWxCellRendererClass
;
444 struct _GtkWxCellRenderer
446 GtkCellRenderer parent
;
449 wxDataViewCustomCell
*cell
;
452 struct _GtkWxCellRendererClass
454 GtkCellRendererClass cell_parent_class
;
458 static GtkCellRenderer
*gtk_wx_cell_renderer_new (void);
459 static void gtk_wx_cell_renderer_init (
460 GtkWxCellRenderer
*cell
);
461 static void gtk_wx_cell_renderer_class_init(
462 GtkWxCellRendererClass
*klass
);
463 static void gtk_wx_cell_renderer_finalize (
465 static void gtk_wx_cell_renderer_get_size (
466 GtkCellRenderer
*cell
,
468 GdkRectangle
*rectangle
,
473 static void gtk_wx_cell_renderer_render (
474 GtkCellRenderer
*cell
,
477 GdkRectangle
*background_area
,
478 GdkRectangle
*cell_area
,
479 GdkRectangle
*expose_area
,
480 GtkCellRendererState flags
);
481 static gboolean
gtk_wx_cell_renderer_activate(
482 GtkCellRenderer
*cell
,
486 GdkRectangle
*background_area
,
487 GdkRectangle
*cell_area
,
488 GtkCellRendererState flags
);
490 static GObjectClass
*cell_parent_class
= NULL
;
495 gtk_wx_cell_renderer_get_type (void)
497 static GType cell_wx_type
= 0;
501 static const GTypeInfo cell_wx_info
=
503 sizeof (GtkWxCellRendererClass
),
504 NULL
, /* base_init */
505 NULL
, /* base_finalize */
506 (GClassInitFunc
) gtk_wx_cell_renderer_class_init
,
507 NULL
, /* class_finalize */
508 NULL
, /* class_data */
509 sizeof (GtkWxCellRenderer
),
511 (GInstanceInitFunc
) gtk_wx_cell_renderer_init
,
514 cell_wx_type
= g_type_register_static( GTK_TYPE_CELL_RENDERER
,
515 "GtkWxCellRenderer", &cell_wx_info
, (GTypeFlags
)0 );
522 gtk_wx_cell_renderer_init (GtkWxCellRenderer
*cell
)
528 gtk_wx_cell_renderer_class_init (GtkWxCellRendererClass
*klass
)
530 GObjectClass
*object_class
= G_OBJECT_CLASS (klass
);
531 GtkCellRendererClass
*cell_class
= GTK_CELL_RENDERER_CLASS (klass
);
533 cell_parent_class
= (GObjectClass
*) g_type_class_peek_parent (klass
);
535 object_class
->finalize
= gtk_wx_cell_renderer_finalize
;
537 cell_class
->get_size
= gtk_wx_cell_renderer_get_size
;
538 cell_class
->render
= gtk_wx_cell_renderer_render
;
539 cell_class
->activate
= gtk_wx_cell_renderer_activate
;
543 gtk_wx_cell_renderer_finalize (GObject
*object
)
546 (* G_OBJECT_CLASS (cell_parent_class
)->finalize
) (object
);
550 gtk_wx_cell_renderer_new (void)
552 return (GtkCellRenderer
*) g_object_new (GTK_TYPE_WX_CELL_RENDERER
, NULL
);
556 gtk_wx_cell_renderer_get_size (GtkCellRenderer
*renderer
,
558 GdkRectangle
*cell_area
,
564 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
565 wxDataViewCustomCell
*cell
= wxrenderer
->cell
;
567 wxSize size
= cell
->GetSize();
569 gint calc_width
= (gint
) renderer
->xpad
* 2 + size
.x
;
570 gint calc_height
= (gint
) renderer
->ypad
* 2 + size
.y
;
577 if (cell_area
&& size
.x
> 0 && size
.y
> 0)
581 *x_offset
= (gint
)((renderer
->xalign
*
582 (cell_area
->width
- calc_width
- 2 * renderer
->xpad
)));
583 *x_offset
= MAX (*x_offset
, 0) + renderer
->xpad
;
587 *y_offset
= (gint
)((renderer
->yalign
*
588 (cell_area
->height
- calc_height
- 2 * renderer
->ypad
)));
589 *y_offset
= MAX (*y_offset
, 0) + renderer
->ypad
;
597 *height
= calc_height
;
601 gtk_wx_cell_renderer_render (GtkCellRenderer
*renderer
,
604 GdkRectangle
*background_area
,
605 GdkRectangle
*cell_area
,
606 GdkRectangle
*expose_area
,
607 GtkCellRendererState flags
)
610 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
611 wxDataViewCustomCell
*cell
= wxrenderer
->cell
;
614 gtk_wx_cell_renderer_get_size (renderer
, widget
, cell_area
,
620 rect
.x
+= cell_area
->x
;
621 rect
.y
+= cell_area
->y
;
622 rect
.width
-= renderer
->xpad
* 2;
623 rect
.height
-= renderer
->ypad
* 2;
626 if (gdk_rectangle_intersect (expose_area
, &rect
, &dummy
))
628 wxRect
renderrect( rect
.x
, rect
.y
, rect
.width
, rect
.height
);
629 wxWindowDC
* dc
= (wxWindowDC
*) cell
->GetDC();
630 dc
->m_window
= window
;
633 if (flags
& GTK_CELL_RENDERER_SELECTED
)
634 state
|= wxDATAVIEW_CELL_SELECTED
;
635 if (flags
& GTK_CELL_RENDERER_PRELIT
)
636 state
|= wxDATAVIEW_CELL_PRELIT
;
637 if (flags
& GTK_CELL_RENDERER_INSENSITIVE
)
638 state
|= wxDATAVIEW_CELL_INSENSITIVE
;
639 if (flags
& GTK_CELL_RENDERER_INSENSITIVE
)
640 state
|= wxDATAVIEW_CELL_INSENSITIVE
;
641 if (flags
& GTK_CELL_RENDERER_FOCUSED
)
642 state
|= wxDATAVIEW_CELL_FOCUSED
;
643 cell
->Render( renderrect
, dc
, state
);
648 gtk_wx_cell_renderer_activate(
649 GtkCellRenderer
*renderer
,
653 GdkRectangle
*background_area
,
654 GdkRectangle
*cell_area
,
655 GtkCellRendererState flags
)
657 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
658 wxDataViewCustomCell
*cell
= wxrenderer
->cell
;
661 gtk_wx_cell_renderer_get_size (renderer
, widget
, cell_area
,
667 rect
.x
+= cell_area
->x
;
668 rect
.y
+= cell_area
->y
;
669 rect
.width
-= renderer
->xpad
* 2;
670 rect
.height
-= renderer
->ypad
* 2;
672 wxRect
renderrect( rect
.x
, rect
.y
, rect
.width
, rect
.height
);
674 return cell
->Activate( renderrect
);
677 // ---------------------------------------------------------
678 // wxGtkDataViewListModelNotifier
679 // ---------------------------------------------------------
681 class wxGtkDataViewListModelNotifier
: public wxDataViewListModelNotifier
684 wxGtkDataViewListModelNotifier( GtkWxListStore
* gtk_store
, wxDataViewListModel
*wx_model
);
686 virtual bool RowAppended();
687 virtual bool RowPrepended();
688 virtual bool RowInserted( size_t before
);
689 virtual bool RowDeleted( size_t row
);
690 virtual bool RowChanged( size_t row
);
691 virtual bool ValueChanged( size_t col
, size_t row
);
692 virtual bool Cleared();
694 GtkWxListStore
*m_gtk_store
;
695 wxDataViewListModel
*m_wx_model
;
698 // ---------------------------------------------------------
699 // wxGtkDataViewListModelNotifier
700 // ---------------------------------------------------------
702 wxGtkDataViewListModelNotifier::wxGtkDataViewListModelNotifier(
703 GtkWxListStore
* gtk_store
, wxDataViewListModel
*wx_model
)
705 m_gtk_store
= gtk_store
;
706 m_wx_model
= wx_model
;
709 bool wxGtkDataViewListModelNotifier::RowAppended()
711 size_t pos
= m_wx_model
->GetNumberOfRows()-1;
714 iter
.stamp
= m_gtk_store
->stamp
;
715 iter
.user_data
= (gpointer
) pos
;
717 GtkTreePath
*path
= gtk_tree_path_new ();
718 gtk_tree_path_append_index (path
, (gint
) pos
);
719 gtk_tree_model_row_inserted (GTK_TREE_MODEL (m_gtk_store
), path
, &iter
);
720 gtk_tree_path_free (path
);
725 bool wxGtkDataViewListModelNotifier::RowPrepended()
728 iter
.stamp
= m_gtk_store
->stamp
;
729 iter
.user_data
= (gpointer
) 0;
731 GtkTreePath
*path
= gtk_tree_path_new ();
732 gtk_tree_path_append_index (path
, (gint
) 0);
733 gtk_tree_model_row_inserted (GTK_TREE_MODEL (m_gtk_store
), path
, &iter
);
734 gtk_tree_path_free (path
);
739 bool wxGtkDataViewListModelNotifier::RowInserted( size_t before
)
744 bool wxGtkDataViewListModelNotifier::RowDeleted( size_t row
)
749 bool wxGtkDataViewListModelNotifier::RowChanged( size_t row
)
752 iter
.stamp
= m_gtk_store
->stamp
;
753 iter
.user_data
= (gpointer
) row
;
754 GtkTreePath
*path
= gtk_tree_model_get_path (GTK_TREE_MODEL (m_gtk_store
), &iter
);
755 gtk_tree_model_row_changed (GTK_TREE_MODEL (m_gtk_store
), path
, &iter
);
756 gtk_tree_path_free (path
);
761 bool wxGtkDataViewListModelNotifier::ValueChanged( size_t col
, size_t row
)
763 return RowChanged( row
);
766 bool wxGtkDataViewListModelNotifier::Cleared()
771 // ---------------------------------------------------------
773 // ---------------------------------------------------------
775 IMPLEMENT_ABSTRACT_CLASS(wxDataViewCell
, wxDataViewCellBase
)
777 wxDataViewCell::wxDataViewCell( const wxString
&varianttype
, wxDataViewCellMode mode
) :
778 wxDataViewCellBase( varianttype
, mode
)
783 // ---------------------------------------------------------
784 // wxDataViewTextCell
785 // ---------------------------------------------------------
788 static void wxGtkTextRendererEditedCallback( GtkCellRendererText
*renderer
,
789 gchar
*arg1
, gchar
*arg2
, gpointer user_data
);
792 static void wxGtkTextRendererEditedCallback( GtkCellRendererText
*renderer
,
793 gchar
*arg1
, gchar
*arg2
, gpointer user_data
)
795 wxDataViewTextCell
*cell
= (wxDataViewTextCell
*) user_data
;
797 wxString tmp
= wxGTK_CONV_BACK( arg2
);
798 wxVariant value
= tmp
;
799 if (!cell
->Validate( value
))
802 wxDataViewListModel
*model
= cell
->GetOwner()->GetOwner()->GetModel();
804 GtkTreePath
*path
= gtk_tree_path_new_from_string( arg1
);
805 size_t model_row
= (size_t)gtk_tree_path_get_indices (path
)[0];
806 gtk_tree_path_free( path
);
808 size_t model_col
= cell
->GetOwner()->GetModelColumn();
810 model
->SetValue( value
, model_col
, model_row
);
811 model
->ValueChanged( model_col
, model_row
);
814 IMPLEMENT_ABSTRACT_CLASS(wxDataViewTextCell
, wxDataViewCell
)
816 wxDataViewTextCell::wxDataViewTextCell( const wxString
&varianttype
, wxDataViewCellMode mode
) :
817 wxDataViewCell( varianttype
, mode
)
819 m_renderer
= (void*) gtk_cell_renderer_text_new();
821 if (m_mode
& wxDATAVIEW_CELL_EDITABLE
)
823 GValue gvalue
= { 0, };
824 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
825 g_value_set_boolean( &gvalue
, true );
826 g_object_set_property( G_OBJECT(m_renderer
), "editable", &gvalue
);
827 g_value_unset( &gvalue
);
829 g_signal_connect_after( m_renderer
, "edited", G_CALLBACK(wxGtkTextRendererEditedCallback
), this );
833 bool wxDataViewTextCell::SetValue( const wxVariant
&value
)
835 wxString tmp
= value
;
837 GValue gvalue
= { 0, };
838 g_value_init( &gvalue
, G_TYPE_STRING
);
839 g_value_set_string( &gvalue
, wxGTK_CONV( tmp
) );
840 g_object_set_property( G_OBJECT(m_renderer
), "text", &gvalue
);
841 g_value_unset( &gvalue
);
846 bool wxDataViewTextCell::GetValue( wxVariant
&value
)
848 GValue gvalue
= { 0, };
849 g_value_init( &gvalue
, G_TYPE_STRING
);
850 g_object_get_property( G_OBJECT(m_renderer
), "text", &gvalue
);
851 wxString tmp
= wxGTK_CONV_BACK( g_value_get_string( &gvalue
) );
852 g_value_unset( &gvalue
);
859 // ---------------------------------------------------------
860 // wxDataViewToggleCell
861 // ---------------------------------------------------------
864 static void wxGtkToggleRendererToggledCallback( GtkCellRendererToggle
*renderer
,
865 gchar
*path
, gpointer user_data
);
868 static void wxGtkToggleRendererToggledCallback( GtkCellRendererToggle
*renderer
,
869 gchar
*path
, gpointer user_data
)
871 wxDataViewToggleCell
*cell
= (wxDataViewToggleCell
*) user_data
;
874 GValue gvalue
= { 0, };
875 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
876 g_object_get_property( G_OBJECT(renderer
), "active", &gvalue
);
877 bool tmp
= g_value_get_boolean( &gvalue
);
878 g_value_unset( &gvalue
);
882 wxVariant value
= tmp
;
883 if (!cell
->Validate( value
))
886 wxDataViewListModel
*model
= cell
->GetOwner()->GetOwner()->GetModel();
888 GtkTreePath
*gtk_path
= gtk_tree_path_new_from_string( path
);
889 size_t model_row
= (size_t)gtk_tree_path_get_indices (gtk_path
)[0];
890 gtk_tree_path_free( gtk_path
);
892 size_t model_col
= cell
->GetOwner()->GetModelColumn();
894 model
->SetValue( value
, model_col
, model_row
);
895 model
->ValueChanged( model_col
, model_row
);
898 IMPLEMENT_ABSTRACT_CLASS(wxDataViewToggleCell
, wxDataViewCell
)
900 wxDataViewToggleCell::wxDataViewToggleCell( const wxString
&varianttype
,
901 wxDataViewCellMode mode
) :
902 wxDataViewCell( varianttype
, mode
)
904 m_renderer
= (void*) gtk_cell_renderer_toggle_new();
906 if (m_mode
& wxDATAVIEW_CELL_EDITABLE
)
908 g_signal_connect_after( m_renderer
, "toggled", G_CALLBACK(wxGtkToggleRendererToggledCallback
), this );
913 GValue gvalue
= { 0, };
914 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
915 g_value_set_boolean( &gvalue
, false );
916 g_object_set_property( G_OBJECT(m_renderer
), "activatable", &gvalue
);
917 g_value_unset( &gvalue
);
919 GValue gvalue2
= { 0, };
920 g_value_init( &gvalue2
, gtk_cell_renderer_mode_get_type() );
921 g_value_set_enum( &gvalue2
, GTK_CELL_RENDERER_MODE_INERT
);
922 g_object_set_property( G_OBJECT(m_renderer
), "mode", &gvalue2
);
923 g_value_unset( &gvalue2
);
928 bool wxDataViewToggleCell::SetValue( const wxVariant
&value
)
932 GValue gvalue
= { 0, };
933 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
934 g_value_set_boolean( &gvalue
, tmp
);
935 g_object_set_property( G_OBJECT(m_renderer
), "active", &gvalue
);
936 g_value_unset( &gvalue
);
941 bool wxDataViewToggleCell::GetValue( wxVariant
&value
)
943 GValue gvalue
= { 0, };
944 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
945 g_object_get_property( G_OBJECT(m_renderer
), "active", &gvalue
);
946 bool tmp
= g_value_get_boolean( &gvalue
);
947 g_value_unset( &gvalue
);
954 // ---------------------------------------------------------
955 // wxDataViewCustomCell
956 // ---------------------------------------------------------
958 class wxDataViewCtrlDC
: public wxWindowDC
961 wxDataViewCtrlDC( wxDataViewCtrl
*window
)
963 GtkWidget
*widget
= window
->m_treeview
;
967 m_context
= window
->GtkGetPangoDefaultContext();
968 m_layout
= pango_layout_new( m_context
);
969 m_fontdesc
= pango_font_description_copy( widget
->style
->font_desc
);
971 m_cmap
= gtk_widget_get_colormap( widget
? widget
: window
->m_widget
);
979 // ---------------------------------------------------------
980 // wxDataViewCustomCell
981 // ---------------------------------------------------------
983 IMPLEMENT_ABSTRACT_CLASS(wxDataViewCustomCell
, wxDataViewCell
)
985 wxDataViewCustomCell::wxDataViewCustomCell( const wxString
&varianttype
,
986 wxDataViewCellMode mode
, bool no_init
) :
987 wxDataViewCell( varianttype
, mode
)
997 bool wxDataViewCustomCell::Init()
999 GtkWxCellRenderer
*renderer
= (GtkWxCellRenderer
*) gtk_wx_cell_renderer_new();
1000 renderer
->cell
= this;
1002 m_renderer
= (void*) renderer
;
1004 if (m_mode
& wxDATAVIEW_CELL_ACTIVATABLE
)
1006 GValue gvalue
= { 0, };
1007 g_value_init( &gvalue
, gtk_cell_renderer_mode_get_type() );
1008 g_value_set_enum( &gvalue
, GTK_CELL_RENDERER_MODE_ACTIVATABLE
);
1009 g_object_set_property( G_OBJECT(m_renderer
), "mode", &gvalue
);
1010 g_value_unset( &gvalue
);
1016 wxDataViewCustomCell::~wxDataViewCustomCell()
1022 wxDC
*wxDataViewCustomCell::GetDC()
1025 m_dc
= new wxDataViewCtrlDC( GetOwner()->GetOwner() );
1030 // ---------------------------------------------------------
1031 // wxDataViewProgressCell
1032 // ---------------------------------------------------------
1034 IMPLEMENT_ABSTRACT_CLASS(wxDataViewProgressCell
, wxDataViewCustomCell
)
1036 wxDataViewProgressCell::wxDataViewProgressCell( const wxString
&label
,
1037 const wxString
&varianttype
, wxDataViewCellMode mode
) :
1038 wxDataViewCustomCell( varianttype
, mode
, true )
1044 if (!gtk_check_version(2,6,0))
1046 m_renderer
= (void*) gtk_cell_renderer_progress_new();
1048 GValue gvalue
= { 0, };
1049 g_value_init( &gvalue
, G_TYPE_STRING
);
1050 g_value_set_boolean( &gvalue
, wxGTK_CONV(m_label
) );
1051 g_object_set_property( G_OBJECT(m_renderer
), "text", &gvalue
);
1052 g_value_unset( &gvalue
);
1057 // Use custom cell code
1058 wxDataViewCustomCell::Init();
1062 wxDataViewProgressCell::~wxDataViewProgressCell()
1066 bool wxDataViewProgressCell::SetValue( const wxVariant
&value
)
1069 if (!gtk_check_version(2,6,0))
1071 gint tmp
= (int) value
;
1072 GValue gvalue
= { 0, };
1073 g_value_init( &gvalue
, G_TYPE_INT
);
1074 g_value_set_boolean( &gvalue
, tmp
);
1075 g_object_set_property( G_OBJECT(m_renderer
), "value", &gvalue
);
1076 g_value_unset( &gvalue
);
1081 m_value
= (long) value
;
1083 if (m_value
< 0) m_value
= 0;
1084 if (m_value
> 100) m_value
= 100;
1090 bool wxDataViewProgressCell::Render( wxRect cell
, wxDC
*dc
, int state
)
1092 double pct
= (double)m_value
/ 100.0;
1094 bar
.width
= (int)(cell
.width
* pct
);
1095 dc
->SetPen( *wxTRANSPARENT_PEN
);
1096 dc
->SetBrush( *wxBLUE_BRUSH
);
1097 dc
->DrawRectangle( bar
);
1099 dc
->SetBrush( *wxTRANSPARENT_BRUSH
);
1100 dc
->SetPen( *wxBLACK_PEN
);
1101 dc
->DrawRectangle( cell
);
1106 wxSize
wxDataViewProgressCell::GetSize()
1108 return wxSize(40,12);
1111 // ---------------------------------------------------------
1113 // ---------------------------------------------------------
1116 static void wxGtkTreeCellDataFunc( GtkTreeViewColumn
*column
,
1117 GtkCellRenderer
*cell
,
1118 GtkTreeModel
*model
,
1124 static void wxGtkTreeCellDataFunc( GtkTreeViewColumn
*column
,
1125 GtkCellRenderer
*renderer
,
1126 GtkTreeModel
*model
,
1130 g_return_if_fail (GTK_IS_WX_LIST_STORE (model
));
1131 GtkWxListStore
*list_store
= (GtkWxListStore
*) model
;
1133 wxDataViewCell
*cell
= (wxDataViewCell
*) data
;
1135 size_t model_row
= (size_t) iter
->user_data
;
1137 wxVariant value
= list_store
->model
->GetValue(
1138 cell
->GetOwner()->GetModelColumn(), model_row
);
1140 if (value
.GetType() != cell
->GetVariantType())
1141 wxPrintf( wxT("Wrong type\n") );
1143 cell
->SetValue( value
);
1146 IMPLEMENT_ABSTRACT_CLASS(wxDataViewColumn
, wxDataViewColumnBase
)
1148 wxDataViewColumn::wxDataViewColumn( const wxString
&title
, wxDataViewCell
*cell
,
1149 size_t model_column
, int flags
) :
1150 wxDataViewColumnBase( title
, cell
, model_column
, flags
)
1152 GtkCellRenderer
*renderer
= (GtkCellRenderer
*) cell
->GetGtkHandle();
1154 GtkTreeViewColumn
*column
= gtk_tree_view_column_new();
1156 gtk_tree_view_column_set_title( column
, wxGTK_CONV(title
) );
1158 gtk_tree_view_column_pack_start( column
, renderer
, TRUE
);
1160 gtk_tree_view_column_set_cell_data_func( column
, renderer
,
1161 wxGtkTreeCellDataFunc
, (gpointer
) cell
, NULL
);
1163 m_column
= (void*) column
;
1166 wxDataViewColumn::~wxDataViewColumn()
1170 void wxDataViewColumn::SetTitle( const wxString
&title
)
1172 wxDataViewColumnBase::SetTitle( title
);
1174 GtkTreeViewColumn
*column
= (GtkTreeViewColumn
*)m_column
;
1175 gtk_tree_view_column_set_title( column
, wxGTK_CONV(title
) );
1178 //-----------------------------------------------------------------------------
1180 //-----------------------------------------------------------------------------
1182 IMPLEMENT_DYNAMIC_CLASS(wxDataViewCtrl
, wxDataViewCtrlBase
)
1184 wxDataViewCtrl::~wxDataViewCtrl()
1188 void wxDataViewCtrl::Init()
1192 bool wxDataViewCtrl::Create(wxWindow
*parent
, wxWindowID id
,
1193 const wxPoint
& pos
, const wxSize
& size
,
1194 long style
, const wxValidator
& validator
)
1198 m_needParent
= TRUE
;
1199 m_acceptsFocus
= TRUE
;
1201 if (!PreCreation( parent
, pos
, size
) ||
1202 !CreateBase( parent
, id
, pos
, size
, style
, validator
))
1204 wxFAIL_MSG( wxT("wxDataViewCtrl creation failed") );
1208 m_widget
= gtk_scrolled_window_new (NULL
, NULL
);
1209 gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (m_widget
), GTK_SHADOW_IN
);
1211 m_treeview
= gtk_tree_view_new();
1212 gtk_container_add (GTK_CONTAINER (m_widget
), m_treeview
);
1214 gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (m_widget
),
1215 GTK_POLICY_AUTOMATIC
, GTK_POLICY_ALWAYS
);
1216 gtk_widget_show (m_treeview
);
1218 m_parent
->DoAddChild( this );
1225 bool wxDataViewCtrl::AssociateModel( wxDataViewListModel
*model
)
1227 if (!wxDataViewCtrlBase::AssociateModel( model
))
1230 GtkWxListStore
*gtk_store
= wxgtk_list_store_new();
1231 gtk_store
->model
= model
;
1233 wxGtkDataViewListModelNotifier
*notifier
=
1234 new wxGtkDataViewListModelNotifier( gtk_store
, model
);
1236 model
->SetNotifier( notifier
);
1238 gtk_tree_view_set_model( GTK_TREE_VIEW(m_treeview
), GTK_TREE_MODEL(gtk_store
) );
1239 g_object_unref( gtk_store
);
1244 bool wxDataViewCtrl::AppendColumn( wxDataViewColumn
*col
)
1246 if (!wxDataViewCtrlBase::AppendColumn(col
))
1249 GtkTreeViewColumn
*column
= (GtkTreeViewColumn
*)col
->GetGtkHandle();
1251 gtk_tree_view_append_column( GTK_TREE_VIEW(m_treeview
), column
);
1257 #endif // wxUSE_DATAVIEWCTRL