1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/dataview.cpp
3 // Purpose: wxDataViewCtrl GTK+2 implementation
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
10 // For compilers that support precompilation, includes "wx.h".
11 #include "wx/wxprec.h"
13 #if wxUSE_DATAVIEWCTRL
15 #include "wx/dataview.h"
17 #ifndef wxUSE_GENERICDATAVIEWCTRL
21 #include "wx/dcclient.h"
25 #include "wx/stockitem.h"
26 #include "wx/calctrl.h"
27 #include "wx/popupwin.h"
31 #include "wx/gtk/private.h"
32 #include "wx/gtk/win_gtk.h"
34 #include <gobject/gvaluecollector.h>
35 #include <gtk/gtktreemodel.h>
36 #include <gtk/gtktreednd.h>
38 #include <gdk/gdkkeysyms.h>
40 //-----------------------------------------------------------------------------
42 //-----------------------------------------------------------------------------
46 //-----------------------------------------------------------------------------
48 //-----------------------------------------------------------------------------
50 extern bool g_blockEventsOnDrag
;
52 //-----------------------------------------------------------------------------
53 // define new GTK+ class wxGtkListStore
54 //-----------------------------------------------------------------------------
58 #define GTK_TYPE_WX_LIST_STORE (gtk_wx_list_store_get_type ())
59 #define GTK_WX_LIST_STORE(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_LIST_STORE, GtkWxListStore))
60 #define GTK_WX_LIST_STORE_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GTK_TYPE_WX_LIST_STORE, GtkWxListStoreClass))
61 #define GTK_IS_WX_LIST_STORE(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_LIST_STORE))
62 #define GTK_IS_WX_LIST_STORE_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_LIST_STORE))
63 #define GTK_WX_LIST_STORE_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_WX_LIST_STORE, GtkWxListStoreClass))
65 GType
gtk_wx_list_store_get_type (void);
67 typedef struct _GtkWxListStore GtkWxListStore
;
68 typedef struct _GtkWxListStoreClass GtkWxListStoreClass
;
70 struct _GtkWxListStore
76 wxDataViewListModel
*model
;
79 struct _GtkWxListStoreClass
81 GObjectClass list_parent_class
;
84 static GtkWxListStore
*wxgtk_list_store_new (void);
85 static void wxgtk_list_store_init (GtkWxListStore
*list_store
);
86 static void wxgtk_list_store_class_init (GtkWxListStoreClass
*klass
);
87 static void wxgtk_list_store_tree_model_init (GtkTreeModelIface
*iface
);
88 static void wxgtk_list_store_finalize (GObject
*object
);
89 static GtkTreeModelFlags
wxgtk_list_store_get_flags (GtkTreeModel
*tree_model
);
90 static gint
wxgtk_list_store_get_n_columns (GtkTreeModel
*tree_model
);
91 static GType
wxgtk_list_store_get_column_type (GtkTreeModel
*tree_model
,
93 static gboolean
wxgtk_list_store_get_iter (GtkTreeModel
*tree_model
,
96 static GtkTreePath
*wxgtk_list_store_get_path (GtkTreeModel
*tree_model
,
98 static void wxgtk_list_store_get_value (GtkTreeModel
*tree_model
,
102 static gboolean
wxgtk_list_store_iter_next (GtkTreeModel
*tree_model
,
104 static gboolean
wxgtk_list_store_iter_children (GtkTreeModel
*tree_model
,
106 GtkTreeIter
*parent
);
107 static gboolean
wxgtk_list_store_iter_has_child (GtkTreeModel
*tree_model
,
109 static gint
wxgtk_list_store_iter_n_children (GtkTreeModel
*tree_model
,
111 static gboolean
wxgtk_list_store_iter_nth_child (GtkTreeModel
*tree_model
,
115 static gboolean
wxgtk_list_store_iter_parent (GtkTreeModel
*tree_model
,
119 static GObjectClass
*list_parent_class
= NULL
;
122 gtk_wx_list_store_get_type (void)
124 static GType list_store_type
= 0;
126 if (!list_store_type
)
128 const GTypeInfo list_store_info
=
130 sizeof (GtkWxListStoreClass
),
131 NULL
, /* base_init */
132 NULL
, /* base_finalize */
133 (GClassInitFunc
) wxgtk_list_store_class_init
,
134 NULL
, /* class_finalize */
135 NULL
, /* class_data */
136 sizeof (GtkWxListStore
),
138 (GInstanceInitFunc
) wxgtk_list_store_init
,
141 static const GInterfaceInfo tree_model_info
=
143 (GInterfaceInitFunc
) wxgtk_list_store_tree_model_init
,
148 list_store_type
= g_type_register_static (G_TYPE_OBJECT
, "GtkWxListStore",
149 &list_store_info
, (GTypeFlags
)0 );
151 g_type_add_interface_static (list_store_type
,
156 return list_store_type
;
159 static GtkWxListStore
*
160 wxgtk_list_store_new(void)
162 GtkWxListStore
*retval
= (GtkWxListStore
*) g_object_new (GTK_TYPE_WX_LIST_STORE
, NULL
);
167 wxgtk_list_store_class_init (GtkWxListStoreClass
*klass
)
169 list_parent_class
= (GObjectClass
*) g_type_class_peek_parent (klass
);
170 GObjectClass
*object_class
= (GObjectClass
*) klass
;
171 object_class
->finalize
= wxgtk_list_store_finalize
;
175 wxgtk_list_store_tree_model_init (GtkTreeModelIface
*iface
)
177 iface
->get_flags
= wxgtk_list_store_get_flags
;
178 iface
->get_n_columns
= wxgtk_list_store_get_n_columns
;
179 iface
->get_column_type
= wxgtk_list_store_get_column_type
;
180 iface
->get_iter
= wxgtk_list_store_get_iter
;
181 iface
->get_path
= wxgtk_list_store_get_path
;
182 iface
->get_value
= wxgtk_list_store_get_value
;
183 iface
->iter_next
= wxgtk_list_store_iter_next
;
184 iface
->iter_children
= wxgtk_list_store_iter_children
;
185 iface
->iter_has_child
= wxgtk_list_store_iter_has_child
;
186 iface
->iter_n_children
= wxgtk_list_store_iter_n_children
;
187 iface
->iter_nth_child
= wxgtk_list_store_iter_nth_child
;
188 iface
->iter_parent
= wxgtk_list_store_iter_parent
;
192 wxgtk_list_store_init (GtkWxListStore
*list_store
)
194 list_store
->model
= NULL
;
195 list_store
->stamp
= g_random_int();
199 wxgtk_list_store_finalize (GObject
*object
)
201 /* GtkWxListStore *list_store = GTK_WX_LIST_STORE (object); */
203 /* we need to sort out, which class deletes what */
204 /* delete list_store->model; */
207 (* list_parent_class
->finalize
) (object
);
212 //-----------------------------------------------------------------------------
213 // implement callbacks from wxGtkListStore class by letting
214 // them call the methods of wxWidgets' wxDataViewListModel
215 //-----------------------------------------------------------------------------
217 static GtkTreeModelFlags
218 wxgtk_list_store_get_flags (GtkTreeModel
*tree_model
)
220 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), (GtkTreeModelFlags
)0 );
222 // GTK+ list store uses a linked list for storing the
223 // items and a pointer to a child is used as the member
224 // field of a GtkTreeIter. This means that the iter is
225 // valid in the GtkListStore as long as the child exists.
226 // We use the index of the row and since the index of a
227 // specific row will change if a row above is deleted,
228 // the iter does not persist
229 return /* GTK_TREE_MODEL_ITERS_PERSIST | */ GTK_TREE_MODEL_LIST_ONLY
;
233 wxgtk_list_store_get_n_columns (GtkTreeModel
*tree_model
)
235 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
236 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), 0);
238 return list_store
->model
->GetColumnCount();
242 wxgtk_list_store_get_column_type (GtkTreeModel
*tree_model
,
245 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
246 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), G_TYPE_INVALID
);
248 GType gtype
= G_TYPE_INVALID
;
250 wxString wxtype
= list_store
->model
->GetColumnType( (unsigned int) index
);
252 if (wxtype
== wxT("string"))
253 gtype
= G_TYPE_STRING
;
256 wxFAIL_MSG( _T("non-string columns not supported yet") );
263 wxgtk_list_store_get_iter (GtkTreeModel
*tree_model
,
267 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
268 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), FALSE
);
269 g_return_val_if_fail (gtk_tree_path_get_depth (path
) > 0, FALSE
);
271 unsigned int i
= (unsigned int)gtk_tree_path_get_indices (path
)[0];
273 if (i
>= list_store
->model
->GetRowCount())
276 iter
->stamp
= list_store
->stamp
;
277 // user_data is just the index
278 iter
->user_data
= (gpointer
) i
;
284 wxgtk_list_store_get_path (GtkTreeModel
*tree_model
,
287 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), NULL
);
288 g_return_val_if_fail (iter
->stamp
== GTK_WX_LIST_STORE (tree_model
)->stamp
, NULL
);
290 GtkTreePath
*retval
= gtk_tree_path_new ();
291 // user_data is just the index
292 int i
= (wxUIntPtr
) iter
->user_data
;
293 gtk_tree_path_append_index (retval
, i
);
298 wxgtk_list_store_get_value (GtkTreeModel
*tree_model
,
303 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
304 g_return_if_fail (GTK_IS_WX_LIST_STORE (tree_model
) );
306 wxDataViewListModel
*model
= list_store
->model
;
307 wxString mtype
= model
->GetColumnType( (unsigned int) column
);
308 if (mtype
== wxT("string"))
311 g_value_init( value
, G_TYPE_STRING
);
312 model
->GetValue( variant
,
313 (unsigned int) column
,
314 (unsigned int) iter
->user_data
);
316 // FIXME: we should support different encodings here
317 g_value_set_string( value
, wxGTK_CONV_SYS(variant
.GetString()) );
321 wxFAIL_MSG( _T("non-string columns not supported yet") );
326 wxgtk_list_store_iter_next (GtkTreeModel
*tree_model
,
329 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), FALSE
);
330 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
332 g_return_val_if_fail (list_store
->stamp
== iter
->stamp
, FALSE
);
334 int n
= (wxUIntPtr
) iter
->user_data
;
339 if (n
>= (int) list_store
->model
->GetRowCount()-1)
342 iter
->user_data
= (gpointer
) ++n
;
348 wxgtk_list_store_iter_children (GtkTreeModel
*tree_model
,
352 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), FALSE
);
353 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
355 // this is a list, nodes have no children
359 iter
->stamp
= list_store
->stamp
;
360 iter
->user_data
= (gpointer
) -1;
366 wxgtk_list_store_iter_has_child (GtkTreeModel
*tree_model
,
373 wxgtk_list_store_iter_n_children (GtkTreeModel
*tree_model
,
376 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), -1);
377 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
380 return (gint
) list_store
->model
->GetRowCount();
382 g_return_val_if_fail (list_store
->stamp
== iter
->stamp
, -1);
388 wxgtk_list_store_iter_nth_child (GtkTreeModel
*tree_model
,
393 g_return_val_if_fail (GTK_IS_WX_LIST_STORE (tree_model
), FALSE
);
394 GtkWxListStore
*list_store
= (GtkWxListStore
*) tree_model
;
402 if (n
>= (gint
) list_store
->model
->GetRowCount())
405 iter
->stamp
= list_store
->stamp
;
406 iter
->user_data
= (gpointer
) n
;
412 wxgtk_list_store_iter_parent (GtkTreeModel
*tree_model
,
419 //-----------------------------------------------------------------------------
420 // define new GTK+ class wxGtkRendererRenderer
421 //-----------------------------------------------------------------------------
425 #define GTK_TYPE_WX_CELL_RENDERER (gtk_wx_cell_renderer_get_type ())
426 #define GTK_WX_CELL_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRenderer))
427 #define GTK_WX_CELL_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRendererClass))
428 #define GTK_IS_WX_CELL_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_WX_CELL_RENDERER))
429 #define GTK_IS_WX_CELL_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_WX_CELL_RENDERER))
430 #define GTK_WX_CELL_RENDERER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_WX_CELL_RENDERER, GtkWxCellRendererClass))
432 GType
gtk_wx_cell_renderer_get_type (void);
434 typedef struct _GtkWxCellRenderer GtkWxCellRenderer
;
435 typedef struct _GtkWxCellRendererClass GtkWxCellRendererClass
;
437 struct _GtkWxCellRenderer
439 GtkCellRenderer parent
;
442 wxDataViewCustomRenderer
*cell
;
446 struct _GtkWxCellRendererClass
448 GtkCellRendererClass cell_parent_class
;
452 static GtkCellRenderer
*gtk_wx_cell_renderer_new (void);
453 static void gtk_wx_cell_renderer_init (
454 GtkWxCellRenderer
*cell
);
455 static void gtk_wx_cell_renderer_class_init(
456 GtkWxCellRendererClass
*klass
);
457 static void gtk_wx_cell_renderer_finalize (
459 static void gtk_wx_cell_renderer_get_size (
460 GtkCellRenderer
*cell
,
462 GdkRectangle
*rectangle
,
467 static void gtk_wx_cell_renderer_render (
468 GtkCellRenderer
*cell
,
471 GdkRectangle
*background_area
,
472 GdkRectangle
*cell_area
,
473 GdkRectangle
*expose_area
,
474 GtkCellRendererState flags
);
475 static gboolean
gtk_wx_cell_renderer_activate(
476 GtkCellRenderer
*cell
,
480 GdkRectangle
*background_area
,
481 GdkRectangle
*cell_area
,
482 GtkCellRendererState flags
);
484 static GObjectClass
*cell_parent_class
= NULL
;
489 gtk_wx_cell_renderer_get_type (void)
491 static GType cell_wx_type
= 0;
495 const GTypeInfo cell_wx_info
=
497 sizeof (GtkWxCellRendererClass
),
498 NULL
, /* base_init */
499 NULL
, /* base_finalize */
500 (GClassInitFunc
) gtk_wx_cell_renderer_class_init
,
501 NULL
, /* class_finalize */
502 NULL
, /* class_data */
503 sizeof (GtkWxCellRenderer
),
505 (GInstanceInitFunc
) gtk_wx_cell_renderer_init
,
508 cell_wx_type
= g_type_register_static( GTK_TYPE_CELL_RENDERER
,
509 "GtkWxCellRenderer", &cell_wx_info
, (GTypeFlags
)0 );
516 gtk_wx_cell_renderer_init (GtkWxCellRenderer
*cell
)
519 cell
->last_click
= 0;
523 gtk_wx_cell_renderer_class_init (GtkWxCellRendererClass
*klass
)
525 GObjectClass
*object_class
= G_OBJECT_CLASS (klass
);
526 GtkCellRendererClass
*cell_class
= GTK_CELL_RENDERER_CLASS (klass
);
528 cell_parent_class
= (GObjectClass
*) g_type_class_peek_parent (klass
);
530 object_class
->finalize
= gtk_wx_cell_renderer_finalize
;
532 cell_class
->get_size
= gtk_wx_cell_renderer_get_size
;
533 cell_class
->render
= gtk_wx_cell_renderer_render
;
534 cell_class
->activate
= gtk_wx_cell_renderer_activate
;
538 gtk_wx_cell_renderer_finalize (GObject
*object
)
541 (* G_OBJECT_CLASS (cell_parent_class
)->finalize
) (object
);
545 gtk_wx_cell_renderer_new (void)
547 return (GtkCellRenderer
*) g_object_new (GTK_TYPE_WX_CELL_RENDERER
, NULL
);
551 gtk_wx_cell_renderer_get_size (GtkCellRenderer
*renderer
,
553 GdkRectangle
*cell_area
,
559 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
560 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
562 wxSize size
= cell
->GetSize();
564 gint calc_width
= (gint
) renderer
->xpad
* 2 + size
.x
;
565 gint calc_height
= (gint
) renderer
->ypad
* 2 + size
.y
;
572 if (cell_area
&& size
.x
> 0 && size
.y
> 0)
576 *x_offset
= (gint
)((renderer
->xalign
*
577 (cell_area
->width
- calc_width
- 2 * renderer
->xpad
)));
578 *x_offset
= MAX (*x_offset
, 0) + renderer
->xpad
;
582 *y_offset
= (gint
)((renderer
->yalign
*
583 (cell_area
->height
- calc_height
- 2 * renderer
->ypad
)));
584 *y_offset
= MAX (*y_offset
, 0) + renderer
->ypad
;
592 *height
= calc_height
;
596 gtk_wx_cell_renderer_render (GtkCellRenderer
*renderer
,
599 GdkRectangle
*background_area
,
600 GdkRectangle
*cell_area
,
601 GdkRectangle
*expose_area
,
602 GtkCellRendererState flags
)
605 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
606 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
609 gtk_wx_cell_renderer_get_size (renderer
, widget
, cell_area
,
615 rect
.x
+= cell_area
->x
;
616 rect
.y
+= cell_area
->y
;
617 rect
.width
-= renderer
->xpad
* 2;
618 rect
.height
-= renderer
->ypad
* 2;
621 if (gdk_rectangle_intersect (expose_area
, &rect
, &dummy
))
623 wxRect
renderrect( rect
.x
, rect
.y
, rect
.width
, rect
.height
);
624 wxWindowDC
* dc
= (wxWindowDC
*) cell
->GetDC();
625 if (dc
->m_window
== NULL
)
627 dc
->m_window
= window
;
632 if (flags
& GTK_CELL_RENDERER_SELECTED
)
633 state
|= wxDATAVIEW_CELL_SELECTED
;
634 if (flags
& GTK_CELL_RENDERER_PRELIT
)
635 state
|= wxDATAVIEW_CELL_PRELIT
;
636 if (flags
& GTK_CELL_RENDERER_INSENSITIVE
)
637 state
|= wxDATAVIEW_CELL_INSENSITIVE
;
638 if (flags
& GTK_CELL_RENDERER_INSENSITIVE
)
639 state
|= wxDATAVIEW_CELL_INSENSITIVE
;
640 if (flags
& GTK_CELL_RENDERER_FOCUSED
)
641 state
|= wxDATAVIEW_CELL_FOCUSED
;
642 cell
->Render( renderrect
, dc
, state
);
647 gtk_wx_cell_renderer_activate(
648 GtkCellRenderer
*renderer
,
652 GdkRectangle
*background_area
,
653 GdkRectangle
*cell_area
,
654 GtkCellRendererState flags
)
656 GtkWxCellRenderer
*wxrenderer
= (GtkWxCellRenderer
*) renderer
;
657 wxDataViewCustomRenderer
*cell
= wxrenderer
->cell
;
660 gtk_wx_cell_renderer_get_size (renderer
, widget
, cell_area
,
666 rect
.x
+= cell_area
->x
;
667 rect
.y
+= cell_area
->y
;
668 rect
.width
-= renderer
->xpad
* 2;
669 rect
.height
-= renderer
->ypad
* 2;
671 wxRect
renderrect( rect
.x
, rect
.y
, rect
.width
, rect
.height
);
673 wxDataViewListModel
*model
= cell
->GetOwner()->GetOwner()->GetModel();
675 GtkTreePath
*treepath
= gtk_tree_path_new_from_string( path
);
676 unsigned int model_row
= (unsigned int)gtk_tree_path_get_indices (treepath
)[0];
677 gtk_tree_path_free( treepath
);
679 unsigned int model_col
= cell
->GetOwner()->GetModelColumn();
685 // activated by <ENTER>
686 if (cell
->Activate( renderrect
, model
, model_col
, model_row
))
691 else if (event
->type
== GDK_BUTTON_PRESS
)
693 GdkEventButton
*button_event
= (GdkEventButton
*) event
;
694 wxPoint
pt( ((int) button_event
->x
) - renderrect
.x
,
695 ((int) button_event
->y
) - renderrect
.y
);
698 if (button_event
->button
== 1)
700 if (cell
->LeftClick( pt
, renderrect
, model
, model_col
, model_row
))
702 // TODO: query system double-click time
703 if (button_event
->time
- wxrenderer
->last_click
< 400)
704 if (cell
->Activate( renderrect
, model
, model_col
, model_row
))
707 if (button_event
->button
== 3)
709 if (cell
->RightClick( pt
, renderrect
, model
, model_col
, model_row
))
713 wxrenderer
->last_click
= button_event
->time
;
721 // ---------------------------------------------------------
722 // wxGtkDataViewListModelNotifier
723 // ---------------------------------------------------------
725 class wxGtkDataViewListModelNotifier
: public wxDataViewListModelNotifier
728 wxGtkDataViewListModelNotifier( GtkWxListStore
* gtk_store
, wxDataViewListModel
*wx_model
);
730 virtual bool RowAppended();
731 virtual bool RowPrepended();
732 virtual bool RowInserted( unsigned int before
);
733 virtual bool RowDeleted( unsigned int row
);
734 virtual bool RowChanged( unsigned int row
);
735 virtual bool ValueChanged( unsigned int col
, unsigned int row
);
736 virtual bool RowsReordered( unsigned int *new_order
);
737 virtual bool Cleared();
743 return wxDataViewListModelNotifier::Freed();
746 GtkWxListStore
*m_gtk_store
;
747 wxDataViewListModel
*m_wx_model
;
750 // ---------------------------------------------------------
751 // wxGtkDataViewListModelNotifier
752 // ---------------------------------------------------------
754 wxGtkDataViewListModelNotifier::wxGtkDataViewListModelNotifier(
755 GtkWxListStore
* gtk_store
, wxDataViewListModel
*wx_model
)
757 m_gtk_store
= gtk_store
;
758 m_wx_model
= wx_model
;
761 bool wxGtkDataViewListModelNotifier::RowAppended()
763 unsigned int pos
= m_wx_model
->GetRowCount()-1;
766 iter
.stamp
= m_gtk_store
->stamp
;
767 iter
.user_data
= (gpointer
) pos
;
769 GtkTreePath
*path
= gtk_tree_path_new ();
770 gtk_tree_path_append_index (path
, (gint
) pos
);
771 gtk_tree_model_row_inserted (GTK_TREE_MODEL (m_gtk_store
), path
, &iter
);
772 gtk_tree_path_free (path
);
777 bool wxGtkDataViewListModelNotifier::RowPrepended()
780 iter
.stamp
= m_gtk_store
->stamp
;
781 iter
.user_data
= (gpointer
) 0;
783 GtkTreePath
*path
= gtk_tree_path_new ();
784 gtk_tree_path_append_index (path
, (gint
) 0);
785 gtk_tree_model_row_inserted (GTK_TREE_MODEL (m_gtk_store
), path
, &iter
);
786 gtk_tree_path_free (path
);
791 bool wxGtkDataViewListModelNotifier::RowInserted( unsigned int before
)
794 iter
.stamp
= m_gtk_store
->stamp
;
795 iter
.user_data
= (gpointer
) before
;
797 GtkTreePath
*path
= gtk_tree_path_new ();
798 gtk_tree_path_append_index (path
, (gint
) before
);
799 gtk_tree_model_row_inserted (GTK_TREE_MODEL (m_gtk_store
), path
, &iter
);
800 gtk_tree_path_free (path
);
805 bool wxGtkDataViewListModelNotifier::RowDeleted( unsigned int row
)
807 GtkTreePath
*path
= gtk_tree_path_new ();
808 gtk_tree_path_append_index (path
, (gint
) row
);
809 gtk_tree_model_row_deleted (GTK_TREE_MODEL (m_gtk_store
), path
);
810 gtk_tree_path_free (path
);
815 bool wxGtkDataViewListModelNotifier::RowChanged( unsigned int row
)
818 iter
.stamp
= m_gtk_store
->stamp
;
819 iter
.user_data
= (gpointer
) row
;
820 GtkTreePath
*path
= gtk_tree_model_get_path (GTK_TREE_MODEL (m_gtk_store
), &iter
);
821 gtk_tree_model_row_changed (GTK_TREE_MODEL (m_gtk_store
), path
, &iter
);
822 gtk_tree_path_free (path
);
827 bool wxGtkDataViewListModelNotifier::ValueChanged( unsigned int model_col
, unsigned int model_row
)
829 // This adds GTK+'s missing MVC logic for ValueChanged
830 wxObjectList::compatibility_iterator
831 node
= GetOwner()->m_viewingColumns
.GetFirst();
834 wxDataViewViewingColumn
* viewing_column
= (wxDataViewViewingColumn
*) node
->GetData();
835 if (viewing_column
->m_modelColumn
== model_col
)
837 GtkTreeView
*widget
= GTK_TREE_VIEW(viewing_column
->m_viewColumn
->GetOwner()->m_treeview
);
838 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(viewing_column
->m_viewColumn
->GetGtkHandle());
841 GtkTreePath
*path
= gtk_tree_path_new();
842 gtk_tree_path_append_index( path
, model_row
);
843 GdkRectangle cell_area
;
844 gtk_tree_view_get_cell_area( widget
, path
, column
, &cell_area
);
845 gtk_tree_path_free( path
);
847 int ydiff
= column
->button
->allocation
.height
;
849 gtk_widget_queue_draw_area( GTK_WIDGET(widget
),
850 cell_area
.x
, ydiff
+ cell_area
.y
, cell_area
.width
, cell_area
.height
);
853 node
= node
->GetNext();
859 bool wxGtkDataViewListModelNotifier::RowsReordered( unsigned int *new_order
)
861 // Assume sizeof(unsigned int)= == sizeof(gint)
863 GtkTreePath
*path
= gtk_tree_path_new ();
864 gtk_tree_model_rows_reordered (GTK_TREE_MODEL (m_gtk_store
), path
, NULL
, (gint
*)new_order
);
865 gtk_tree_path_free (path
);
867 // This adds GTK+'s missing MVC logic for RowsReordered
868 wxObjectList::compatibility_iterator
869 node
= GetOwner()->m_viewingColumns
.GetFirst();
872 wxDataViewViewingColumn
* viewing_column
= (wxDataViewViewingColumn
*) node
->GetData();
873 GtkTreeView
*widget
= GTK_TREE_VIEW(viewing_column
->m_viewColumn
->GetOwner()->m_treeview
);
874 // Doesn't work yet...
875 gtk_widget_queue_draw( GTK_WIDGET(widget
) );
877 node
= node
->GetNext();
883 bool wxGtkDataViewListModelNotifier::Cleared()
888 // ---------------------------------------------------------
889 // wxDataViewRenderer
890 // ---------------------------------------------------------
892 IMPLEMENT_ABSTRACT_CLASS(wxDataViewRenderer
, wxDataViewRendererBase
)
894 wxDataViewRenderer::wxDataViewRenderer( const wxString
&varianttype
, wxDataViewCellMode mode
,
896 wxDataViewRendererBase( varianttype
, mode
, align
)
900 // NOTE: SetMode() and SetAlignment() needs to be called in the renderer's ctor,
901 // after the m_renderer pointer has been initialized
904 void wxDataViewRenderer::SetMode( wxDataViewCellMode mode
)
906 GtkCellRendererMode gtkMode
;
909 case wxDATAVIEW_CELL_INERT
:
910 gtkMode
= GTK_CELL_RENDERER_MODE_INERT
;
912 case wxDATAVIEW_CELL_ACTIVATABLE
:
913 gtkMode
= GTK_CELL_RENDERER_MODE_ACTIVATABLE
;
915 case wxDATAVIEW_CELL_EDITABLE
:
916 gtkMode
= GTK_CELL_RENDERER_MODE_EDITABLE
;
920 GValue gvalue
= { 0, };
921 g_value_init( &gvalue
, gtk_cell_renderer_mode_get_type() );
922 g_value_set_enum( &gvalue
, gtkMode
);
923 g_object_set_property( G_OBJECT(m_renderer
), "mode", &gvalue
);
924 g_value_unset( &gvalue
);
927 wxDataViewCellMode
wxDataViewRenderer::GetMode() const
929 wxDataViewCellMode ret
;
932 g_object_get( G_OBJECT(m_renderer
), "mode", &gvalue
, NULL
);
934 switch (g_value_get_enum(&gvalue
))
936 case GTK_CELL_RENDERER_MODE_INERT
:
937 ret
= wxDATAVIEW_CELL_INERT
;
939 case GTK_CELL_RENDERER_MODE_ACTIVATABLE
:
940 ret
= wxDATAVIEW_CELL_ACTIVATABLE
;
942 case GTK_CELL_RENDERER_MODE_EDITABLE
:
943 ret
= wxDATAVIEW_CELL_EDITABLE
;
947 g_value_unset( &gvalue
);
952 void wxDataViewRenderer::SetAlignment( int align
)
954 // horizontal alignment:
957 if (align
& wxALIGN_RIGHT
)
959 else if (align
& wxALIGN_CENTER_HORIZONTAL
)
962 GValue gvalue
= { 0, };
963 g_value_init( &gvalue
, G_TYPE_FLOAT
);
964 g_value_set_float( &gvalue
, xalign
);
965 g_object_set_property( G_OBJECT(m_renderer
), "xalign", &gvalue
);
966 g_value_unset( &gvalue
);
968 // vertical alignment:
971 if (align
& wxALIGN_BOTTOM
)
973 else if (align
& wxALIGN_CENTER_VERTICAL
)
976 GValue gvalue2
= { 0, };
977 g_value_init( &gvalue2
, G_TYPE_FLOAT
);
978 g_value_set_float( &gvalue2
, yalign
);
979 g_object_set_property( G_OBJECT(m_renderer
), "yalign", &gvalue2
);
980 g_value_unset( &gvalue2
);
983 int wxDataViewRenderer::GetAlignment() const
988 // horizontal alignment:
990 g_object_get( G_OBJECT(m_renderer
), "xalign", &gvalue
, NULL
);
991 float xalign
= g_value_get_float( &gvalue
);
994 else if (xalign
== 0.5)
995 ret
|= wxALIGN_CENTER_HORIZONTAL
;
997 ret
|= wxALIGN_RIGHT
;
998 g_value_unset( &gvalue
);
1001 // vertical alignment:
1003 g_object_get( G_OBJECT(m_renderer
), "yalign", &gvalue
, NULL
);
1004 float yalign
= g_value_get_float( &gvalue
);
1007 else if (yalign
== 0.5)
1008 ret
|= wxALIGN_CENTER_VERTICAL
;
1010 ret
|= wxALIGN_BOTTOM
;
1011 g_value_unset( &gvalue
);
1018 // ---------------------------------------------------------
1019 // wxDataViewTextRenderer
1020 // ---------------------------------------------------------
1023 static void wxGtkTextRendererEditedCallback( GtkCellRendererText
*renderer
,
1024 gchar
*arg1
, gchar
*arg2
, gpointer user_data
);
1027 static void wxGtkTextRendererEditedCallback( GtkCellRendererText
*renderer
,
1028 gchar
*arg1
, gchar
*arg2
, gpointer user_data
)
1030 wxDataViewTextRenderer
*cell
= (wxDataViewTextRenderer
*) user_data
;
1032 wxString tmp
= wxGTK_CONV_BACK_FONT(arg2
, cell
->GetOwner()->GetOwner()->GetFont());
1033 wxVariant value
= tmp
;
1034 if (!cell
->Validate( value
))
1037 wxDataViewListModel
*model
= cell
->GetOwner()->GetOwner()->GetModel();
1039 GtkTreePath
*path
= gtk_tree_path_new_from_string( arg1
);
1040 unsigned int model_row
= (unsigned int)gtk_tree_path_get_indices (path
)[0];
1041 gtk_tree_path_free( path
);
1043 unsigned int model_col
= cell
->GetOwner()->GetModelColumn();
1045 model
->SetValue( value
, model_col
, model_row
);
1046 model
->ValueChanged( model_col
, model_row
);
1049 IMPLEMENT_CLASS(wxDataViewTextRenderer
, wxDataViewRenderer
)
1051 wxDataViewTextRenderer::wxDataViewTextRenderer( const wxString
&varianttype
, wxDataViewCellMode mode
,
1053 wxDataViewRenderer( varianttype
, mode
, align
)
1055 m_renderer
= (GtkWidget
*) gtk_cell_renderer_text_new();
1057 if (mode
& wxDATAVIEW_CELL_EDITABLE
)
1059 GValue gvalue
= { 0, };
1060 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1061 g_value_set_boolean( &gvalue
, true );
1062 g_object_set_property( G_OBJECT(m_renderer
), "editable", &gvalue
);
1063 g_value_unset( &gvalue
);
1065 g_signal_connect_after( m_renderer
, "edited", G_CALLBACK(wxGtkTextRendererEditedCallback
), this );
1069 SetAlignment(align
);
1072 bool wxDataViewTextRenderer::SetValue( const wxVariant
&value
)
1074 wxString tmp
= value
;
1076 GValue gvalue
= { 0, };
1077 g_value_init( &gvalue
, G_TYPE_STRING
);
1078 g_value_set_string( &gvalue
, wxGTK_CONV_FONT( tmp
, GetOwner()->GetOwner()->GetFont() ) );
1079 g_object_set_property( G_OBJECT(m_renderer
), "text", &gvalue
);
1080 g_value_unset( &gvalue
);
1085 bool wxDataViewTextRenderer::GetValue( wxVariant
&value
) const
1087 GValue gvalue
= { 0, };
1088 g_value_init( &gvalue
, G_TYPE_STRING
);
1089 g_object_get_property( G_OBJECT(m_renderer
), "text", &gvalue
);
1090 wxString tmp
= wxGTK_CONV_BACK_FONT( g_value_get_string( &gvalue
),
1091 wx_const_cast(wxDataViewTextRenderer
*, this)->GetOwner()->GetOwner()->GetFont() );
1092 g_value_unset( &gvalue
);
1099 void wxDataViewTextRenderer::SetAlignment( int align
)
1101 wxDataViewRenderer::SetAlignment(align
);
1103 if (gtk_check_version(2,10,0))
1106 // horizontal alignment:
1107 PangoAlignment pangoAlign
= PANGO_ALIGN_LEFT
;
1108 if (align
& wxALIGN_RIGHT
)
1109 pangoAlign
= PANGO_ALIGN_RIGHT
;
1110 else if (align
& wxALIGN_CENTER_HORIZONTAL
)
1111 pangoAlign
= PANGO_ALIGN_CENTER
;
1113 GValue gvalue
= { 0, };
1114 g_value_init( &gvalue
, gtk_cell_renderer_mode_get_type() );
1115 g_value_set_enum( &gvalue
, pangoAlign
);
1116 g_object_set_property( G_OBJECT(m_renderer
), "alignment", &gvalue
);
1117 g_value_unset( &gvalue
);
1120 // ---------------------------------------------------------
1121 // wxDataViewBitmapRenderer
1122 // ---------------------------------------------------------
1124 IMPLEMENT_CLASS(wxDataViewBitmapRenderer
, wxDataViewRenderer
)
1126 wxDataViewBitmapRenderer::wxDataViewBitmapRenderer( const wxString
&varianttype
, wxDataViewCellMode mode
,
1128 wxDataViewRenderer( varianttype
, mode
, align
)
1130 m_renderer
= (GtkWidget
*) gtk_cell_renderer_pixbuf_new();
1133 SetAlignment(align
);
1136 bool wxDataViewBitmapRenderer::SetValue( const wxVariant
&value
)
1138 if (value
.GetType() == wxT("wxBitmap"))
1143 // This may create a Pixbuf representation in the
1144 // wxBitmap object (and it will stay there)
1145 GdkPixbuf
*pixbuf
= bitmap
.GetPixbuf();
1147 GValue gvalue
= { 0, };
1148 g_value_init( &gvalue
, G_TYPE_OBJECT
);
1149 g_value_set_object( &gvalue
, pixbuf
);
1150 g_object_set_property( G_OBJECT(m_renderer
), "pixbuf", &gvalue
);
1151 g_value_unset( &gvalue
);
1156 if (value
.GetType() == wxT("wxIcon"))
1161 // This may create a Pixbuf representation in the
1162 // wxBitmap object (and it will stay there)
1163 GdkPixbuf
*pixbuf
= bitmap
.GetPixbuf();
1165 GValue gvalue
= { 0, };
1166 g_value_init( &gvalue
, G_TYPE_OBJECT
);
1167 g_value_set_object( &gvalue
, pixbuf
);
1168 g_object_set_property( G_OBJECT(m_renderer
), "pixbuf", &gvalue
);
1169 g_value_unset( &gvalue
);
1177 bool wxDataViewBitmapRenderer::GetValue( wxVariant
&value
) const
1182 // ---------------------------------------------------------
1183 // wxDataViewToggleRenderer
1184 // ---------------------------------------------------------
1187 static void wxGtkToggleRendererToggledCallback( GtkCellRendererToggle
*renderer
,
1188 gchar
*path
, gpointer user_data
);
1191 static void wxGtkToggleRendererToggledCallback( GtkCellRendererToggle
*renderer
,
1192 gchar
*path
, gpointer user_data
)
1194 wxDataViewToggleRenderer
*cell
= (wxDataViewToggleRenderer
*) user_data
;
1197 GValue gvalue
= { 0, };
1198 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1199 g_object_get_property( G_OBJECT(renderer
), "active", &gvalue
);
1200 bool tmp
= g_value_get_boolean( &gvalue
);
1201 g_value_unset( &gvalue
);
1205 wxVariant value
= tmp
;
1206 if (!cell
->Validate( value
))
1209 wxDataViewListModel
*model
= cell
->GetOwner()->GetOwner()->GetModel();
1211 GtkTreePath
*gtk_path
= gtk_tree_path_new_from_string( path
);
1212 unsigned int model_row
= (unsigned int)gtk_tree_path_get_indices (gtk_path
)[0];
1213 gtk_tree_path_free( gtk_path
);
1215 unsigned int model_col
= cell
->GetOwner()->GetModelColumn();
1217 model
->SetValue( value
, model_col
, model_row
);
1218 model
->ValueChanged( model_col
, model_row
);
1221 IMPLEMENT_CLASS(wxDataViewToggleRenderer
, wxDataViewRenderer
)
1223 wxDataViewToggleRenderer::wxDataViewToggleRenderer( const wxString
&varianttype
,
1224 wxDataViewCellMode mode
, int align
) :
1225 wxDataViewRenderer( varianttype
, mode
, align
)
1227 m_renderer
= (GtkWidget
*) gtk_cell_renderer_toggle_new();
1229 if (mode
& wxDATAVIEW_CELL_ACTIVATABLE
)
1231 g_signal_connect_after( m_renderer
, "toggled",
1232 G_CALLBACK(wxGtkToggleRendererToggledCallback
), this );
1236 GValue gvalue
= { 0, };
1237 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1238 g_value_set_boolean( &gvalue
, false );
1239 g_object_set_property( G_OBJECT(m_renderer
), "activatable", &gvalue
);
1240 g_value_unset( &gvalue
);
1244 SetAlignment(align
);
1247 bool wxDataViewToggleRenderer::SetValue( const wxVariant
&value
)
1251 GValue gvalue
= { 0, };
1252 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1253 g_value_set_boolean( &gvalue
, tmp
);
1254 g_object_set_property( G_OBJECT(m_renderer
), "active", &gvalue
);
1255 g_value_unset( &gvalue
);
1260 bool wxDataViewToggleRenderer::GetValue( wxVariant
&value
) const
1262 GValue gvalue
= { 0, };
1263 g_value_init( &gvalue
, G_TYPE_BOOLEAN
);
1264 g_object_get_property( G_OBJECT(m_renderer
), "active", &gvalue
);
1265 bool tmp
= g_value_get_boolean( &gvalue
);
1266 g_value_unset( &gvalue
);
1273 // ---------------------------------------------------------
1274 // wxDataViewCustomRenderer
1275 // ---------------------------------------------------------
1277 class wxDataViewCtrlDC
: public wxWindowDC
1280 wxDataViewCtrlDC( wxDataViewCtrl
*window
)
1282 GtkWidget
*widget
= window
->m_treeview
;
1286 m_context
= window
->GtkGetPangoDefaultContext();
1287 m_layout
= pango_layout_new( m_context
);
1288 m_fontdesc
= pango_font_description_copy( widget
->style
->font_desc
);
1290 m_cmap
= gtk_widget_get_colormap( widget
? widget
: window
->m_widget
);
1292 // Set m_window later
1294 // m_owner = window;
1298 // ---------------------------------------------------------
1299 // wxDataViewCustomRenderer
1300 // ---------------------------------------------------------
1302 IMPLEMENT_CLASS(wxDataViewCustomRenderer
, wxDataViewRenderer
)
1304 wxDataViewCustomRenderer::wxDataViewCustomRenderer( const wxString
&varianttype
,
1305 wxDataViewCellMode mode
, int align
,
1307 wxDataViewRenderer( varianttype
, mode
, align
)
1317 bool wxDataViewCustomRenderer::Init(wxDataViewCellMode mode
, int align
)
1319 GtkWxCellRenderer
*renderer
= (GtkWxCellRenderer
*) gtk_wx_cell_renderer_new();
1320 renderer
->cell
= this;
1322 m_renderer
= (GtkWidget
*) renderer
;
1325 SetAlignment(align
);
1330 wxDataViewCustomRenderer::~wxDataViewCustomRenderer()
1336 wxDC
*wxDataViewCustomRenderer::GetDC()
1340 if (GetOwner() == NULL
)
1342 if (GetOwner()->GetOwner() == NULL
)
1344 m_dc
= new wxDataViewCtrlDC( GetOwner()->GetOwner() );
1350 // ---------------------------------------------------------
1351 // wxDataViewProgressRenderer
1352 // ---------------------------------------------------------
1354 IMPLEMENT_CLASS(wxDataViewProgressRenderer
, wxDataViewCustomRenderer
)
1356 wxDataViewProgressRenderer::wxDataViewProgressRenderer( const wxString
&label
,
1357 const wxString
&varianttype
, wxDataViewCellMode mode
, int align
) :
1358 wxDataViewCustomRenderer( varianttype
, mode
, align
, true )
1364 if (!gtk_check_version(2,6,0))
1366 m_renderer
= (GtkWidget
*) gtk_cell_renderer_progress_new();
1368 GValue gvalue
= { 0, };
1369 g_value_init( &gvalue
, G_TYPE_STRING
);
1371 // FIXME: font encoding support
1372 g_value_set_string( &gvalue
, wxGTK_CONV_SYS(m_label
) );
1373 g_object_set_property( G_OBJECT(m_renderer
), "text", &gvalue
);
1374 g_value_unset( &gvalue
);
1377 SetAlignment(align
);
1382 // Use custom cell code
1383 wxDataViewCustomRenderer::Init(mode
, align
);
1387 wxDataViewProgressRenderer::~wxDataViewProgressRenderer()
1391 bool wxDataViewProgressRenderer::SetValue( const wxVariant
&value
)
1394 if (!gtk_check_version(2,6,0))
1396 gint tmp
= (long) value
;
1397 GValue gvalue
= { 0, };
1398 g_value_init( &gvalue
, G_TYPE_INT
);
1399 g_value_set_int( &gvalue
, tmp
);
1400 g_object_set_property( G_OBJECT(m_renderer
), "value", &gvalue
);
1401 g_value_unset( &gvalue
);
1406 m_value
= (long) value
;
1408 if (m_value
< 0) m_value
= 0;
1409 if (m_value
> 100) m_value
= 100;
1415 bool wxDataViewProgressRenderer::GetValue( wxVariant
&value
) const
1420 bool wxDataViewProgressRenderer::Render( wxRect cell
, wxDC
*dc
, int state
)
1422 double pct
= (double)m_value
/ 100.0;
1424 bar
.width
= (int)(cell
.width
* pct
);
1425 dc
->SetPen( *wxTRANSPARENT_PEN
);
1426 dc
->SetBrush( *wxBLUE_BRUSH
);
1427 dc
->DrawRectangle( bar
);
1429 dc
->SetBrush( *wxTRANSPARENT_BRUSH
);
1430 dc
->SetPen( *wxBLACK_PEN
);
1431 dc
->DrawRectangle( cell
);
1436 wxSize
wxDataViewProgressRenderer::GetSize() const
1438 return wxSize(40,12);
1441 // ---------------------------------------------------------
1442 // wxDataViewDateRenderer
1443 // ---------------------------------------------------------
1445 class wxDataViewDateRendererPopupTransient
: public wxPopupTransientWindow
1448 wxDataViewDateRendererPopupTransient( wxWindow
* parent
, wxDateTime
*value
,
1449 wxDataViewListModel
*model
, unsigned int col
, unsigned int row
) :
1450 wxPopupTransientWindow( parent
, wxBORDER_SIMPLE
)
1455 m_cal
= new wxCalendarCtrl( this, -1, *value
);
1456 wxBoxSizer
*sizer
= new wxBoxSizer( wxHORIZONTAL
);
1457 sizer
->Add( m_cal
, 1, wxGROW
);
1462 virtual void OnDismiss()
1466 void OnCalendar( wxCalendarEvent
&event
);
1468 wxCalendarCtrl
*m_cal
;
1469 wxDataViewListModel
*m_model
;
1474 DECLARE_EVENT_TABLE()
1477 BEGIN_EVENT_TABLE(wxDataViewDateRendererPopupTransient
,wxPopupTransientWindow
)
1478 EVT_CALENDAR( -1, wxDataViewDateRendererPopupTransient::OnCalendar
)
1481 void wxDataViewDateRendererPopupTransient::OnCalendar( wxCalendarEvent
&event
)
1483 wxDateTime date
= event
.GetDate();
1484 wxVariant value
= date
;
1485 m_model
->SetValue( value
, m_col
, m_row
);
1486 m_model
->ValueChanged( m_col
, m_row
);
1490 IMPLEMENT_CLASS(wxDataViewDateRenderer
, wxDataViewCustomRenderer
)
1492 wxDataViewDateRenderer::wxDataViewDateRenderer( const wxString
&varianttype
,
1493 wxDataViewCellMode mode
, int align
) :
1494 wxDataViewCustomRenderer( varianttype
, mode
, align
)
1497 SetAlignment(align
);
1500 bool wxDataViewDateRenderer::SetValue( const wxVariant
&value
)
1502 m_date
= value
.GetDateTime();
1507 bool wxDataViewDateRenderer::GetValue( wxVariant
&value
) const
1512 bool wxDataViewDateRenderer::Render( wxRect cell
, wxDC
*dc
, int state
)
1514 dc
->SetFont( GetOwner()->GetOwner()->GetFont() );
1515 wxString tmp
= m_date
.FormatDate();
1516 dc
->DrawText( tmp
, cell
.x
, cell
.y
);
1521 wxSize
wxDataViewDateRenderer::GetSize() const
1523 wxString tmp
= m_date
.FormatDate();
1525 GetView()->GetTextExtent( tmp
, &x
, &y
, &d
);
1526 return wxSize(x
,y
+d
);
1529 bool wxDataViewDateRenderer::Activate( wxRect cell
, wxDataViewListModel
*model
,
1530 unsigned int col
, unsigned int row
)
1533 model
->GetValue( variant
, col
, row
);
1534 wxDateTime value
= variant
.GetDateTime();
1536 wxDataViewDateRendererPopupTransient
*popup
= new wxDataViewDateRendererPopupTransient(
1537 GetOwner()->GetOwner()->GetParent(), &value
, model
, col
, row
);
1538 wxPoint pos
= wxGetMousePosition();
1541 popup
->Popup( popup
->m_cal
);
1546 // ---------------------------------------------------------
1548 // ---------------------------------------------------------
1552 gtk_dataview_header_button_press_callback( GtkWidget
*widget
,
1553 GdkEventButton
*gdk_event
,
1554 wxDataViewColumn
*column
)
1556 if (gdk_event
->type
!= GDK_BUTTON_PRESS
)
1559 if (gdk_event
->button
== 1)
1561 wxDataViewCtrl
*dv
= column
->GetOwner();
1562 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_COLUMN_HEADER_CLICK
, dv
->GetId() );
1563 event
.SetDataViewColumn( column
);
1564 event
.SetModel( dv
->GetModel() );
1565 dv
->GetEventHandler()->ProcessEvent( event
);
1572 static void wxGtkTreeCellDataFunc( GtkTreeViewColumn
*column
,
1573 GtkCellRenderer
*cell
,
1574 GtkTreeModel
*model
,
1580 static void wxGtkTreeCellDataFunc( GtkTreeViewColumn
*column
,
1581 GtkCellRenderer
*renderer
,
1582 GtkTreeModel
*model
,
1586 g_return_if_fail (GTK_IS_WX_LIST_STORE (model
));
1587 GtkWxListStore
*list_store
= (GtkWxListStore
*) model
;
1589 wxDataViewRenderer
*cell
= (wxDataViewRenderer
*) data
;
1591 unsigned int model_row
= (unsigned int) iter
->user_data
;
1594 list_store
->model
->GetValue( value
, cell
->GetOwner()->GetModelColumn(), model_row
);
1596 if (value
.GetType() != cell
->GetVariantType())
1597 wxLogError( wxT("Wrong type, required: %s but: %s"),
1598 value
.GetType().c_str(),
1599 cell
->GetVariantType().c_str() );
1601 cell
->SetValue( value
);
1604 IMPLEMENT_CLASS(wxDataViewColumn
, wxDataViewColumnBase
)
1606 wxDataViewColumn::wxDataViewColumn( const wxString
&title
, wxDataViewRenderer
*cell
,
1607 unsigned int model_column
, int width
,
1608 wxAlignment align
, int flags
) :
1609 wxDataViewColumnBase( title
, cell
, model_column
, width
, align
, flags
)
1611 Init( align
, flags
, width
);
1613 gtk_tree_view_column_set_clickable( GTK_TREE_VIEW_COLUMN(m_column
), TRUE
);
1617 wxDataViewColumn::wxDataViewColumn( const wxBitmap
&bitmap
, wxDataViewRenderer
*cell
,
1618 unsigned int model_column
, int width
,
1619 wxAlignment align
, int flags
) :
1620 wxDataViewColumnBase( bitmap
, cell
, model_column
, width
, align
, flags
)
1622 Init( align
, flags
, width
);
1624 SetBitmap( bitmap
);
1627 void wxDataViewColumn::Init(wxAlignment align
, int flags
, int width
)
1629 m_isConnected
= false;
1631 GtkCellRenderer
*renderer
= (GtkCellRenderer
*) GetRenderer()->GetGtkHandle();
1632 GtkTreeViewColumn
*column
= gtk_tree_view_column_new();
1633 m_column
= (GtkWidget
*) column
;
1636 SetAlignment( align
);
1638 // NOTE: we prefer not to call SetMinWidth(wxDVC_DEFAULT_MINWIDTH);
1639 // as GTK+ is smart and unless explicitely told, will set the minimal
1640 // width to the title's lenght, which is a better default
1642 // the GTK_TREE_VIEW_COLUMN_FIXED is required by the "fixed height" mode
1643 // that we use for the wxDataViewCtrl
1644 gtk_tree_view_column_set_fixed_width( column
, width
< 0 ? wxDVC_DEFAULT_WIDTH
: width
);
1645 gtk_tree_view_column_set_sizing( column
, GTK_TREE_VIEW_COLUMN_FIXED
);
1647 gtk_tree_view_column_pack_end( column
, renderer
, TRUE
);
1649 gtk_tree_view_column_set_cell_data_func( column
, renderer
,
1650 wxGtkTreeCellDataFunc
, (gpointer
) GetRenderer(), NULL
);
1653 wxDataViewColumn::~wxDataViewColumn()
1657 void wxDataViewColumn::OnInternalIdle()
1662 if (GTK_WIDGET_REALIZED(GetOwner()->m_treeview
))
1664 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1667 g_signal_connect(column
->button
, "button_press_event",
1668 G_CALLBACK (gtk_dataview_header_button_press_callback
), this);
1670 m_isConnected
= true;
1675 void wxDataViewColumn::SetOwner( wxDataViewCtrl
*owner
)
1677 wxDataViewColumnBase::SetOwner( owner
);
1679 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1681 gtk_tree_view_column_set_title( column
, wxGTK_CONV_FONT(GetTitle(), GetOwner()->GetFont() ) );
1684 void wxDataViewColumn::SetTitle( const wxString
&title
)
1686 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1690 // disconnect before column->button gets recreated
1691 g_signal_handlers_disconnect_by_func( column
->button
,
1692 (GtkWidget
*) gtk_dataview_header_button_press_callback
, this);
1694 m_isConnected
= false;
1697 // FIXME: can it really happen that we don't have the owner here??
1698 wxDataViewCtrl
*ctrl
= GetOwner();
1699 gtk_tree_view_column_set_title( column
, ctrl
? wxGTK_CONV_FONT(title
, ctrl
->GetFont())
1700 : wxGTK_CONV_SYS(title
) );
1702 gtk_tree_view_column_set_widget( column
, NULL
);
1705 wxString
wxDataViewColumn::GetTitle() const
1707 const gchar
*str
= gtk_tree_view_column_get_title( GTK_TREE_VIEW_COLUMN(m_column
) );
1708 return wxConvFileName
->cMB2WX(str
);
1711 void wxDataViewColumn::SetBitmap( const wxBitmap
&bitmap
)
1713 wxDataViewColumnBase::SetBitmap( bitmap
);
1715 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1718 GtkImage
*gtk_image
= GTK_IMAGE( gtk_image_new() );
1720 GdkBitmap
*mask
= (GdkBitmap
*) NULL
;
1721 if (bitmap
.GetMask())
1722 mask
= bitmap
.GetMask()->GetBitmap();
1724 if (bitmap
.HasPixbuf())
1726 gtk_image_set_from_pixbuf(GTK_IMAGE(gtk_image
),
1727 bitmap
.GetPixbuf());
1731 gtk_image_set_from_pixmap(GTK_IMAGE(gtk_image
),
1732 bitmap
.GetPixmap(), mask
);
1734 gtk_widget_show( GTK_WIDGET(gtk_image
) );
1736 gtk_tree_view_column_set_widget( column
, GTK_WIDGET(gtk_image
) );
1740 gtk_tree_view_column_set_widget( column
, NULL
);
1744 void wxDataViewColumn::SetHidden( bool hidden
)
1746 gtk_tree_view_column_set_visible( GTK_TREE_VIEW_COLUMN(m_column
), !hidden
);
1749 void wxDataViewColumn::SetResizeable( bool resizeable
)
1751 gtk_tree_view_column_set_resizable( GTK_TREE_VIEW_COLUMN(m_column
), resizeable
);
1754 void wxDataViewColumn::SetAlignment( wxAlignment align
)
1756 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1758 gfloat xalign
= 0.0;
1759 if (align
== wxALIGN_RIGHT
)
1761 if (align
== wxALIGN_CENTER_HORIZONTAL
||
1762 align
== wxALIGN_CENTER
)
1765 gtk_tree_view_column_set_alignment( column
, xalign
);
1768 wxAlignment
wxDataViewColumn::GetAlignment() const
1770 gfloat xalign
= gtk_tree_view_column_get_alignment( GTK_TREE_VIEW_COLUMN(m_column
) );
1773 return wxALIGN_RIGHT
;
1775 return wxALIGN_CENTER_HORIZONTAL
;
1777 return wxALIGN_LEFT
;
1780 void wxDataViewColumn::SetSortable( bool sortable
)
1782 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1783 gtk_tree_view_column_set_sort_indicator( column
, sortable
);
1786 bool wxDataViewColumn::IsSortable() const
1788 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1789 return gtk_tree_view_column_get_sort_indicator( column
);
1792 bool wxDataViewColumn::IsResizeable() const
1794 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1795 return gtk_tree_view_column_get_resizable( column
);
1798 bool wxDataViewColumn::IsHidden() const
1800 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1801 return !gtk_tree_view_column_get_visible( column
);
1804 void wxDataViewColumn::SetSortOrder( bool ascending
)
1806 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1809 gtk_tree_view_column_set_sort_order( column
, GTK_SORT_ASCENDING
);
1811 gtk_tree_view_column_set_sort_order( column
, GTK_SORT_DESCENDING
);
1814 bool wxDataViewColumn::IsSortOrderAscending() const
1816 GtkTreeViewColumn
*column
= GTK_TREE_VIEW_COLUMN(m_column
);
1818 return (gtk_tree_view_column_get_sort_order( column
) != GTK_SORT_DESCENDING
);
1821 void wxDataViewColumn::SetMinWidth( int width
)
1823 gtk_tree_view_column_set_min_width( GTK_TREE_VIEW_COLUMN(m_column
), width
);
1826 int wxDataViewColumn::GetMinWidth() const
1828 return gtk_tree_view_column_get_min_width( GTK_TREE_VIEW_COLUMN(m_column
) );
1831 int wxDataViewColumn::GetWidth() const
1833 return gtk_tree_view_column_get_width( GTK_TREE_VIEW_COLUMN(m_column
) );
1836 void wxDataViewColumn::SetWidth( int width
)
1838 gtk_tree_view_column_set_fixed_width( GTK_TREE_VIEW_COLUMN(m_column
), width
);
1842 //-----------------------------------------------------------------------------
1843 // wxDataViewCtrl signal callbacks
1844 //-----------------------------------------------------------------------------
1847 wxdataview_selection_changed_callback( GtkTreeSelection
* selection
, wxDataViewCtrl
*dv
)
1849 if (!GTK_WIDGET_REALIZED(dv
->m_widget
))
1852 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ROW_SELECTED
, dv
->GetId() );
1853 event
.SetRow( dv
->GetSelection() );
1854 event
.SetModel( dv
->GetModel() );
1855 dv
->GetEventHandler()->ProcessEvent( event
);
1859 wxdataview_row_activated_callback( GtkTreeView
* treeview
, GtkTreePath
*path
,
1860 GtkTreeViewColumn
*column
, wxDataViewCtrl
*dv
)
1862 wxDataViewEvent
event( wxEVT_COMMAND_DATAVIEW_ROW_ACTIVATED
, dv
->GetId() );
1863 unsigned int row
= (unsigned int)gtk_tree_path_get_indices (path
)[0];
1864 event
.SetRow( row
);
1865 event
.SetModel( dv
->GetModel() );
1866 dv
->GetEventHandler()->ProcessEvent( event
);
1869 //-----------------------------------------------------------------------------
1871 //-----------------------------------------------------------------------------
1873 IMPLEMENT_DYNAMIC_CLASS(wxDataViewCtrl
, wxDataViewCtrlBase
)
1875 wxDataViewCtrl::~wxDataViewCtrl()
1878 GetModel()->RemoveNotifier( m_notifier
);
1880 // remove the model from the GtkTreeView before it gets destroyed by the
1881 // wxDataViewCtrlBase's dtor
1882 gtk_tree_view_set_model( GTK_TREE_VIEW(m_treeview
), NULL
);
1885 void wxDataViewCtrl::Init()
1890 bool wxDataViewCtrl::Create(wxWindow
*parent
, wxWindowID id
,
1891 const wxPoint
& pos
, const wxSize
& size
,
1892 long style
, const wxValidator
& validator
)
1896 m_needParent
= true;
1898 if (!PreCreation( parent
, pos
, size
) ||
1899 !CreateBase( parent
, id
, pos
, size
, style
, validator
))
1901 wxFAIL_MSG( wxT("wxDataViewCtrl creation failed") );
1905 m_widget
= gtk_scrolled_window_new (NULL
, NULL
);
1907 GtkScrolledWindowSetBorder(m_widget
, style
);
1909 m_treeview
= gtk_tree_view_new();
1910 gtk_container_add (GTK_CONTAINER (m_widget
), m_treeview
);
1913 if (!gtk_check_version(2,6,0))
1914 gtk_tree_view_set_fixed_height_mode( GTK_TREE_VIEW(m_treeview
), TRUE
);
1917 if (style
& wxDV_MULTIPLE
)
1919 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
1920 gtk_tree_selection_set_mode( selection
, GTK_SELECTION_MULTIPLE
);
1923 gtk_tree_view_set_headers_visible( GTK_TREE_VIEW(m_treeview
), (style
& wxDV_NO_HEADER
) == 0 );
1926 if (!gtk_check_version(2,10,0))
1928 GtkTreeViewGridLines grid
= GTK_TREE_VIEW_GRID_LINES_NONE
;
1930 if ((style
& wxDV_HORIZ_RULES
) != 0 &&
1931 (style
& wxDV_VERT_RULES
) != 0)
1932 grid
= GTK_TREE_VIEW_GRID_LINES_BOTH
;
1933 else if (style
& wxDV_VERT_RULES
)
1934 grid
= GTK_TREE_VIEW_GRID_LINES_VERTICAL
;
1935 else if (style
& wxDV_HORIZ_RULES
)
1936 grid
= GTK_TREE_VIEW_GRID_LINES_HORIZONTAL
;
1938 gtk_tree_view_set_grid_lines( GTK_TREE_VIEW(m_treeview
), grid
);
1942 gtk_tree_view_set_rules_hint( GTK_TREE_VIEW(m_treeview
), (style
& wxDV_HORIZ_RULES
) != 0 );
1944 gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (m_widget
),
1945 GTK_POLICY_AUTOMATIC
, GTK_POLICY_ALWAYS
);
1946 gtk_widget_show (m_treeview
);
1948 m_parent
->DoAddChild( this );
1952 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
1953 g_signal_connect_after (selection
, "changed",
1954 G_CALLBACK (wxdataview_selection_changed_callback
), this);
1955 g_signal_connect_after (m_treeview
, "row_activated",
1956 G_CALLBACK (wxdataview_row_activated_callback
), this);
1961 void wxDataViewCtrl::OnInternalIdle()
1963 wxWindow::OnInternalIdle();
1965 unsigned int cols
= GetColumnCount();
1967 for (i
= 0; i
< cols
; i
++)
1969 wxDataViewColumn
*col
= GetColumn( i
);
1970 col
->OnInternalIdle();
1974 bool wxDataViewCtrl::AssociateModel( wxDataViewListModel
*model
)
1976 if (!wxDataViewCtrlBase::AssociateModel( model
))
1979 GtkWxListStore
*gtk_store
= wxgtk_list_store_new();
1980 gtk_store
->model
= model
;
1982 m_notifier
= new wxGtkDataViewListModelNotifier( gtk_store
, model
);
1984 model
->AddNotifier( m_notifier
);
1986 gtk_tree_view_set_model( GTK_TREE_VIEW(m_treeview
), GTK_TREE_MODEL(gtk_store
) );
1987 g_object_unref( gtk_store
);
1992 bool wxDataViewCtrl::AppendColumn( wxDataViewColumn
*col
)
1994 if (!wxDataViewCtrlBase::AppendColumn(col
))
1997 GtkTreeViewColumn
*column
= (GtkTreeViewColumn
*)col
->GetGtkHandle();
1999 gtk_tree_view_append_column( GTK_TREE_VIEW(m_treeview
), column
);
2004 void wxDataViewCtrl::SetSelection( int row
)
2006 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
2010 gtk_tree_selection_unselect_all( selection
);
2014 GtkTreePath
*path
= gtk_tree_path_new ();
2015 gtk_tree_path_append_index( path
, row
);
2017 gtk_tree_selection_select_path( selection
, path
);
2019 gtk_tree_path_free( path
);
2023 void wxDataViewCtrl::Unselect( unsigned int row
)
2025 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
2027 GtkTreePath
*path
= gtk_tree_path_new ();
2028 gtk_tree_path_append_index( path
, row
);
2030 gtk_tree_selection_unselect_path( selection
, path
);
2032 gtk_tree_path_free( path
);
2035 void wxDataViewCtrl::SetSelectionRange( unsigned int from
, unsigned int to
)
2039 void wxDataViewCtrl::SetSelections( const wxArrayInt
& aSelections
)
2043 bool wxDataViewCtrl::IsSelected( unsigned int row
) const
2045 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
2047 GtkTreePath
*path
= gtk_tree_path_new ();
2048 gtk_tree_path_append_index( path
, row
);
2050 gboolean ret
= gtk_tree_selection_path_is_selected( selection
, path
);
2052 gtk_tree_path_free( path
);
2057 int wxDataViewCtrl::GetSelection() const
2059 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
2060 if (HasFlag(wxDV_MULTIPLE
))
2062 GtkTreeModel
*model
;
2063 GList
*list
= gtk_tree_selection_get_selected_rows( selection
, &model
);
2068 // list = g_list_nth( list, 0 ); should be a noop
2069 GtkTreePath
*path
= (GtkTreePath
*) list
->data
;
2071 unsigned int row
= (unsigned int)gtk_tree_path_get_indices (path
)[0];
2074 g_list_foreach( list
, (GFunc
) gtk_tree_path_free
, NULL
);
2075 g_list_free( list
);
2083 GtkTreeModel
*model
;
2085 gboolean has_selection
= gtk_tree_selection_get_selected( selection
, &model
, &iter
);
2088 unsigned int row
= (wxUIntPtr
) iter
.user_data
;
2096 int wxDataViewCtrl::GetSelections(wxArrayInt
& aSelections
) const
2098 aSelections
.Clear();
2100 GtkTreeSelection
*selection
= gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview
) );
2101 if (HasFlag(wxDV_MULTIPLE
))
2103 GtkTreeModel
*model
;
2104 GList
*list
= gtk_tree_selection_get_selected_rows( selection
, &model
);
2110 // list = g_list_nth( list, 0 ); should be a noop
2111 GtkTreePath
*path
= (GtkTreePath
*) list
->data
;
2113 unsigned int row
= (unsigned int)gtk_tree_path_get_indices (path
)[0];
2115 aSelections
.Add( (int) row
);
2117 list
= g_list_next( list
);
2121 g_list_foreach( list
, (GFunc
) gtk_tree_path_free
, NULL
);
2122 g_list_free( list
);
2128 GtkTreeModel
*model
;
2130 gboolean has_selection
= gtk_tree_selection_get_selected( selection
, &model
, &iter
);
2133 unsigned int row
= (wxUIntPtr
) iter
.user_data
;
2134 aSelections
.Add( (int) row
);
2144 wxDataViewCtrl::GetClassDefaultAttributes(wxWindowVariant
WXUNUSED(variant
))
2146 return GetDefaultAttributesFromGTKWidget(gtk_tree_view_new
);
2151 // !wxUSE_GENERICDATAVIEWCTRL
2154 // wxUSE_DATAVIEWCTRL