// wxDataViewCtrl globals
// ----------------------------------------------------------------------------
+class WXDLLIMPEXP_CORE wxDataViewModel;
+class WXDLLIMPEXP_CORE wxDataViewListModel;
class WXDLLIMPEXP_CORE wxDataViewCtrl;
class WXDLLIMPEXP_CORE wxDataViewColumn;
class WXDLLIMPEXP_CORE wxDataViewCell;
// wxDataViewListModelNotifier
// ---------------------------------------------------------
-class wxDataViewListModelNotifier
+
+class wxDataViewListModelNotifier: public wxObject
{
public:
wxDataViewListModelNotifier() { }
virtual bool RowChanged( size_t row ) = 0;
virtual bool ValueChanged( size_t col, size_t row ) = 0;
virtual bool Cleared() = 0;
- virtual bool ValueChanged( wxDataViewColumn *view_column, size_t model_column, size_t row ) = 0;
+
+ void SetOwner( wxDataViewListModel *owner ) { m_owner = owner; }
+ wxDataViewListModel *GetOwner() { return m_owner; }
+
+private:
+ wxDataViewListModel *m_owner;
};
// ---------------------------------------------------------
// wxDataViewListModel
// ---------------------------------------------------------
+class wxDataViewViewingColumn: public wxObject
+{
+public:
+ wxDataViewViewingColumn( wxDataViewColumn *view_column, size_t model_column )
+ {
+ m_viewColumn = view_column;
+ m_modelColumn = model_column;
+ }
+
+ wxDataViewColumn *m_viewColumn;
+ size_t m_modelColumn;
+};
+
class wxDataViewListModel: public wxDataViewModel
{
public:
virtual void AddViewingColumn( wxDataViewColumn *view_column, size_t model_column );
virtual void RemoveViewingColumn( wxDataViewColumn *column );
- // Used internally
- virtual void SetNotifier( wxDataViewListModelNotifier *notifier );
- virtual wxDataViewListModelNotifier* GetNotifier();
+ virtual void AddNotifier( wxDataViewListModelNotifier *notifier );
+ virtual void RemoveNotifier( wxDataViewListModelNotifier *notifier );
-private:
- wxDataViewListModelNotifier *m_notifier;
- wxList m_viewingColumns;
+ wxList m_notifiers;
+ wxList m_viewingColumns;
protected:
DECLARE_DYNAMIC_CLASS_NO_COPY(wxDataViewListModel)
virtual bool ValueChanged( size_t col, size_t row );
virtual bool Cleared();
- // Used internally
- void SetNotifier( wxDataViewListModelNotifier *notifier );
- wxDataViewListModelNotifier* GetNotifier();
-
+ virtual void Resort();
+
private:
wxDataViewListModel *m_child;
wxDataViewSortedIndexArray m_array;
private:
friend class wxDataViewCtrlDC;
friend class wxGtkDataViewListModelNotifier;
- GtkWidget *m_treeview;
+ GtkWidget *m_treeview;
+ wxDataViewListModelNotifier *m_notifier;
private:
DECLARE_DYNAMIC_CLASS(wxDataViewCtrl)
// wxDataViewListModel
// ---------------------------------------------------------
-class wxDataViewViewingColumn: public wxObject
-{
-public:
- wxDataViewViewingColumn( wxDataViewColumn *view_column, size_t model_column )
- {
- m_viewColumn = view_column;
- m_modelColumn = model_column;
- }
-
- wxDataViewColumn *m_viewColumn;
- size_t m_modelColumn;
-};
-
-
IMPLEMENT_ABSTRACT_CLASS(wxDataViewListModel, wxDataViewModel)
wxDataViewListModel::wxDataViewListModel()
{
- m_notifier = NULL;
m_viewingColumns.DeleteContents( true );
+ m_notifiers.DeleteContents( true );
}
wxDataViewListModel::~wxDataViewListModel()
{
- if (m_notifier)
- delete m_notifier;
}
bool wxDataViewListModel::RowAppended()
{
- if (m_notifier)
- return m_notifier->RowAppended();
+ bool ret = true;
+
+ wxNode *node = m_notifiers.GetFirst();
+ while (node)
+ {
+ wxDataViewListModelNotifier* notifier = (wxDataViewListModelNotifier*) node->GetData();
+ if (!notifier->RowAppended())
+ ret = false;
+ node = node->GetNext();
+ }
- return false;
+ return ret;
}
bool wxDataViewListModel::RowPrepended()
{
- if (m_notifier)
- return m_notifier->RowPrepended();
+ bool ret = true;
+
+ wxNode *node = m_notifiers.GetFirst();
+ while (node)
+ {
+ wxDataViewListModelNotifier* notifier = (wxDataViewListModelNotifier*) node->GetData();
+ if (!notifier->RowPrepended())
+ ret = false;
+ node = node->GetNext();
+ }
- return false;
+ return ret;
}
bool wxDataViewListModel::RowInserted( size_t before )
{
- if (m_notifier)
- return m_notifier->RowInserted( before );
+ bool ret = true;
+
+ wxNode *node = m_notifiers.GetFirst();
+ while (node)
+ {
+ wxDataViewListModelNotifier* notifier = (wxDataViewListModelNotifier*) node->GetData();
+ if (!notifier->RowInserted(before))
+ ret = false;
+ node = node->GetNext();
+ }
- return false;
+ return ret;
}
bool wxDataViewListModel::RowDeleted( size_t row )
{
- if (m_notifier)
- return m_notifier->RowDeleted( row );
+ bool ret = true;
+
+ wxNode *node = m_notifiers.GetFirst();
+ while (node)
+ {
+ wxDataViewListModelNotifier* notifier = (wxDataViewListModelNotifier*) node->GetData();
+ if (!notifier->RowDeleted( row ))
+ ret = false;
+ node = node->GetNext();
+ }
- return false;
+ return ret;
}
bool wxDataViewListModel::RowChanged( size_t row )
{
- if (m_notifier)
- return m_notifier->RowChanged( row );
+ bool ret = true;
+
+ wxNode *node = m_notifiers.GetFirst();
+ while (node)
+ {
+ wxDataViewListModelNotifier* notifier = (wxDataViewListModelNotifier*) node->GetData();
+ if (!notifier->RowChanged( row ))
+ ret = false;
+ node = node->GetNext();
+ }
- return false;
+ return ret;
}
bool wxDataViewListModel::ValueChanged( size_t col, size_t row )
{
- bool ret = false;
+ bool ret = true;
- if (m_notifier)
- ret = m_notifier->ValueChanged( col, row );
-
- wxNode *node = m_viewingColumns.GetFirst();
+ wxNode *node = m_notifiers.GetFirst();
while (node)
{
- wxDataViewViewingColumn* tmp = (wxDataViewViewingColumn*) node->GetData();
-
- if (tmp->m_modelColumn == col)
- m_notifier->ValueChanged( tmp->m_viewColumn, col, row );
-
+ wxDataViewListModelNotifier* notifier = (wxDataViewListModelNotifier*) node->GetData();
+ if (!notifier->ValueChanged( col, row ))
+ ret = false;
node = node->GetNext();
}
bool wxDataViewListModel::Cleared()
{
- if (m_notifier)
- return m_notifier->Cleared();
+ bool ret = true;
+
+ wxNode *node = m_notifiers.GetFirst();
+ while (node)
+ {
+ wxDataViewListModelNotifier* notifier = (wxDataViewListModelNotifier*) node->GetData();
+ if (!notifier->Cleared())
+ ret = false;
+ node = node->GetNext();
+ }
- return false;
+ return ret;
}
void wxDataViewListModel::AddViewingColumn( wxDataViewColumn *view_column, size_t model_column )
}
}
-void wxDataViewListModel::SetNotifier( wxDataViewListModelNotifier *notifier )
+void wxDataViewListModel::AddNotifier( wxDataViewListModelNotifier *notifier )
{
- if (m_notifier)
- delete m_notifier;
-
- m_notifier = notifier;
+ m_notifiers.Append( notifier );
+ notifier->SetOwner( this );
}
-wxDataViewListModelNotifier* wxDataViewListModel::GetNotifier()
+void wxDataViewListModel::RemoveNotifier( wxDataViewListModelNotifier *notifier )
{
- return m_notifier;
+ m_notifiers.DeleteObject( notifier );
}
// ---------------------------------------------------------
s_CmpCol = 0;
s_CmpModel = child;
s_CmpFunc = wxDataViewListModelSortedDefaultCompare;
+
+ Resort();
}
wxDataViewSortedListModel::~wxDataViewSortedListModel()
{
}
+void wxDataViewSortedListModel::Resort()
+{
+ m_array.Clear();
+ size_t n = m_child->GetNumberOfRows();
+ size_t i;
+ for (i = 0; i < n; i++)
+ m_array.Add( i );
+}
+
size_t wxDataViewSortedListModel::GetNumberOfRows()
{
return m_child->GetNumberOfRows();
return ret;
}
-void wxDataViewSortedListModel::SetNotifier( wxDataViewListModelNotifier *notifier )
-{
- wxDataViewListModel::SetNotifier( notifier );
-}
-
-wxDataViewListModelNotifier* wxDataViewSortedListModel::GetNotifier()
-{
- return wxDataViewListModel::GetNotifier();
-}
-
// ---------------------------------------------------------
// wxDataViewCellBase
// ---------------------------------------------------------
virtual bool RowChanged( size_t row );
virtual bool ValueChanged( size_t col, size_t row );
virtual bool Cleared();
- virtual bool ValueChanged( wxDataViewColumn *view_column, size_t model_column, size_t row );
GtkWxListStore *m_gtk_store;
wxDataViewListModel *m_wx_model;
return true;
}
-bool wxGtkDataViewListModelNotifier::ValueChanged( size_t col, size_t row )
+bool wxGtkDataViewListModelNotifier::ValueChanged( size_t model_col, size_t model_row )
{
- return RowChanged( row );
+ // This adds GTK+'s missing MVC logic for SetValue
+ wxNode *node = GetOwner()->m_viewingColumns.GetFirst();
+ while (node)
+ {
+ wxDataViewViewingColumn* viewing_column = (wxDataViewViewingColumn*) node->GetData();
+ if (viewing_column->m_modelColumn == model_col)
+ {
+ GtkTreeView *widget = GTK_TREE_VIEW(viewing_column->m_viewColumn->GetOwner()->m_treeview);
+ GtkTreeViewColumn *column = GTK_TREE_VIEW_COLUMN(viewing_column->m_viewColumn->GetGtkHandle());
+
+ // Get cell area
+ GtkTreePath *path = gtk_tree_path_new();
+ gtk_tree_path_append_index( path, model_row );
+ GdkRectangle cell_area;
+ gtk_tree_view_get_cell_area( widget, path, column, &cell_area );
+ gtk_tree_path_free( path );
+
+ int ydiff = column->button->allocation.height;
+ // Redraw
+ gtk_widget_queue_draw_area( GTK_WIDGET(widget),
+ cell_area.x, ydiff + cell_area.y, cell_area.width, cell_area.height );
+ }
+
+ node = node->GetNext();
+ }
+
+ return true;
}
bool wxGtkDataViewListModelNotifier::Cleared()
return false;
}
-bool wxGtkDataViewListModelNotifier::ValueChanged( wxDataViewColumn *view_column, size_t model_column, size_t row )
-{
- GtkTreeView *widget = GTK_TREE_VIEW(view_column->GetOwner()->m_treeview);
- GtkTreeViewColumn *column = GTK_TREE_VIEW_COLUMN(view_column->GetGtkHandle());
-
- // Get cell area
- GtkTreePath *path = gtk_tree_path_new();
- gtk_tree_path_append_index( path, row );
- GdkRectangle cell_area;
- gtk_tree_view_get_cell_area( widget, path, column, &cell_area );
- gtk_tree_path_free( path );
-
- int ydiff = column->button->allocation.height;
- // Redraw
- gtk_widget_queue_draw_area( GTK_WIDGET(widget),
- cell_area.x, ydiff + cell_area.y, cell_area.width, cell_area.height );
-
- return true;
-}
-
// ---------------------------------------------------------
// wxDataViewCell
// ---------------------------------------------------------
wxDataViewCtrl::~wxDataViewCtrl()
{
+ if (m_notifier)
+ GetModel()->RemoveNotifier( m_notifier );
}
void wxDataViewCtrl::Init()
{
+ m_notifier = NULL;
}
bool wxDataViewCtrl::Create(wxWindow *parent, wxWindowID id,
GtkWxListStore *gtk_store = wxgtk_list_store_new();
gtk_store->model = model;
- wxGtkDataViewListModelNotifier *notifier =
- new wxGtkDataViewListModelNotifier( gtk_store, model );
+ m_notifier = new wxGtkDataViewListModelNotifier( gtk_store, model );
- model->SetNotifier( notifier );
+ model->AddNotifier( m_notifier );
gtk_tree_view_set_model( GTK_TREE_VIEW(m_treeview), GTK_TREE_MODEL(gtk_store) );
g_object_unref( gtk_store );