Made wxDataViewCtrl sample use a real, expandabl data tree

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@47459 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Robert Roebling
2007-07-14 13:54:14 +00:00
parent 94473fa8e4
commit 1e08ad102a
4 changed files with 267 additions and 80 deletions

View File

@@ -24,7 +24,7 @@
#if defined(__WXGTK20__) #if defined(__WXGTK20__)
// for testing // for testing
#define wxUSE_GENERICDATAVIEWCTRL 1 // #define wxUSE_GENERICDATAVIEWCTRL 1
#elif defined(__WXMAC__) #elif defined(__WXMAC__)
#else #else
#define wxUSE_GENERICDATAVIEWCTRL 1 #define wxUSE_GENERICDATAVIEWCTRL 1
@@ -418,6 +418,7 @@ public:
{ return m_indent; } { return m_indent; }
// TODO selection code // TODO selection code
virtual wxDataViewItem GetSelection() = 0;
protected: protected:
virtual void DoSetExpanderColumn() = 0 ; virtual void DoSetExpanderColumn() = 0 ;

View File

@@ -305,6 +305,9 @@ public:
virtual bool AssociateModel( wxDataViewModel *model ); virtual bool AssociateModel( wxDataViewModel *model );
virtual bool AppendColumn( wxDataViewColumn *col ); virtual bool AppendColumn( wxDataViewColumn *col );
virtual wxDataViewItem GetSelection();
// selection code // selection code
static wxVisualAttributes static wxVisualAttributes

View File

@@ -58,10 +58,101 @@ Implement this data model
*/ */
class MyMusicModelNode;
WX_DEFINE_ARRAY_PTR( MyMusicModelNode*, MyMusicModelNodes );
class MyMusicModelNode
{
public:
MyMusicModelNode( MyMusicModelNode* parent, const wxUint32 id,
const wxString &title, const wxString &artist, const wxString &year )
{
m_parent = parent;
m_id = id;
m_title = title;
m_artist = artist;
m_year = year;
m_isContainer = false;
}
MyMusicModelNode( MyMusicModelNode* parent, const wxUint32 id,
const wxString &branch )
{
m_parent = parent;
m_id = id;
m_title = branch;
m_isContainer = true;
}
~MyMusicModelNode()
{
size_t count = m_children.GetCount();
size_t i;
for (i = 0; i < count; i++)
{
MyMusicModelNode *child = m_children[i];
delete child;
}
}
wxUint32 GetID() { return m_id; }
bool IsContainer() { return m_isContainer; }
MyMusicModelNode* GetParent() { return m_parent; }
MyMusicModelNodes &GetChildren() { return m_children; }
MyMusicModelNode* GetNthChild( unsigned int n ) { return m_children.Item( n ); }
void Insert( MyMusicModelNode* child, unsigned int n) { m_children.Insert( child, n); }
void Append( MyMusicModelNode* child ) { m_children.Add( child ); }
unsigned int GetChildCount() { return m_children.GetCount(); }
public:
wxString m_title;
wxString m_artist;
wxString m_year;
private:
MyMusicModelNode *m_parent;
MyMusicModelNodes m_children;
wxUint32 m_id;
bool m_isContainer;
};
class MyMusicModel: public wxDataViewModel class MyMusicModel: public wxDataViewModel
{ {
public: public:
MyMusicModel() {} MyMusicModel()
{
m_idCounter = 0;
m_root = new MyMusicModelNode( NULL, GetNewId(), "My Music" );
m_pop = new MyMusicModelNode( m_root, GetNewId(), "Pop music" );
m_root->Append( m_pop );
m_pop->Append( new MyMusicModelNode( m_pop, GetNewId(),
"You are not alone", "Michael Jackson", "1995" ) );
m_pop->Append( new MyMusicModelNode( m_pop, GetNewId(),
"Take a bow", "Madonna", "1994" ) );
m_classical = new MyMusicModelNode( m_root, GetNewId(), "Classical music" );
m_root->Append( m_classical );
m_classical->Append( new MyMusicModelNode( m_classical, GetNewId(),
"Ninth symphony", "Ludwig van Beethoven", "1824" ) );
m_classical->Append( new MyMusicModelNode( m_classical, GetNewId(),
"German Requiem", "Johannes Brahms", "1868" ) );
}
void AddToClassical( const wxString &title, const wxString &artist, const wxString &year )
{
// add to data
MyMusicModelNode *child_node =
new MyMusicModelNode( m_classical, GetNewId(), title, artist, year );
m_classical->Append( child_node );
// notify control
wxDataViewItem child( child_node->GetID() );
wxDataViewItem parent( m_classical->GetID() );
wxPrintf( "parent id %d\n", m_classical->GetID() );
ItemAdded( parent, child );
}
virtual unsigned int GetColumnCount() const virtual unsigned int GetColumnCount() const
{ {
@@ -76,100 +167,105 @@ public:
virtual void GetValue( wxVariant &variant, virtual void GetValue( wxVariant &variant,
const wxDataViewItem &item, unsigned int col ) const const wxDataViewItem &item, unsigned int col ) const
{ {
variant = wxString(""); MyMusicModelNode *node = FindNode( item );
int ID = item.GetID();
switch (ID)
{
case 1: if (col == 0) variant = wxString("My Music"); break;
case 2: if (col == 0) variant = wxString("Pop music"); break;
case 5: if (col == 0) variant = wxString("Classical music"); break;
case 3:
{
switch (col) switch (col)
{ {
case 0: variant = wxString("You are not alone"); break; case 0: variant = node->m_title; break;
case 1: variant = wxString("Michael Jackson"); break; case 1: variant = node->m_artist; break;
case 2: variant = wxString("1995"); case 2: variant = node->m_year; break;
default: wxLogError( "MyMusicModel::GetValue: wrong column" );
} }
} }
break;
case 4:
{
switch (col)
{
case 0: variant = wxString("Take a bow"); break;
case 1: variant = wxString("Madonna"); break;
case 2: variant = wxString("1994");
}
}
break;
case 6:
{
switch (col)
{
case 0: variant = wxString("Ninth symphony"); break;
case 1: variant = wxString("Ludwig v. Beethoven"); break;
case 2: variant = wxString("1824");
}
}
break;
case 7:
{
switch (col)
{
case 0: variant = wxString("German requiem"); break;
case 1: variant = wxString("Johannes Brahms"); break;
case 2: variant = wxString("1868");
}
}
break;
}
}
virtual bool SetValue( const wxVariant &variant, virtual bool SetValue( const wxVariant &variant,
const wxDataViewItem &item, unsigned int col ) const wxDataViewItem &item, unsigned int col )
{ {
// readonly MyMusicModelNode *node = FindNode( item );
return true; switch (col)
{
case 0: node->m_title = variant.GetString(); break;
case 1: node->m_artist = variant.GetString(); break;
case 2: node->m_year = variant.GetString(); break;
default: wxLogError( "MyMusicModel::SetValue: wrong column" );
}
} }
/*****************************************************************
If wxDataViewItem is not valid in the two methods I quote above
then it means "return the child item from the invisible root".
******************************************************************/
virtual bool HasChildren( const wxDataViewItem &item ) const virtual bool HasChildren( const wxDataViewItem &item ) const
{ {
int ID = item.GetID(); if (item.GetID() == 0)
return ((ID == 1) || (ID == 2) || (ID == 5) || (ID == 0)); return true;
MyMusicModelNode *node = FindNode( item );
return node->IsContainer();
} }
virtual wxDataViewItem GetFirstChild( const wxDataViewItem &parent ) const virtual wxDataViewItem GetFirstChild( const wxDataViewItem &parent ) const
{ {
int ID = parent.GetID(); if (parent.GetID() == 0)
switch (ID) return wxDataViewItem( m_root->GetID() );
{
case 0: return wxDataViewItem( 1 ); MyMusicModelNode *node = FindNode( parent );
case 1: return wxDataViewItem( 2 );
case 2: return wxDataViewItem( 3 ); if (node->GetChildCount() == 0)
case 5: return wxDataViewItem( 6 ); return wxDataViewItem( 0 );
MyMusicModelNode *first_child = node->GetChildren().Item( 0 );
return wxDataViewItem( first_child->GetID() );
} }
return wxDataViewItem(0);
}
virtual wxDataViewItem GetNextSibling( const wxDataViewItem &item ) const virtual wxDataViewItem GetNextSibling( const wxDataViewItem &item ) const
{ {
int ID = item.GetID(); MyMusicModelNode *node = FindNode( item );
switch (ID) MyMusicModelNode *parent = node->GetParent();
{ if (!parent)
case 2: return wxDataViewItem( 5 ); return wxDataViewItem(0);
case 3: return wxDataViewItem( 4 );
case 6: return wxDataViewItem( 7 ); int pos = parent->GetChildren().Index( node );
if (pos == wxNOT_FOUND)
return wxDataViewItem(0);
if (pos == parent->GetChildCount()-1)
return wxDataViewItem(0);
node = parent->GetChildren().Item( pos+1 );
return wxDataViewItem( node->GetID() );
} }
return wxDataViewItem(0); private:
wxUint32 GetNewId() { m_idCounter++; return m_idCounter; }
MyMusicModelNode *FindNodeRec( MyMusicModelNode *node, const wxDataViewItem &item ) const
{
if (node->GetID() == item.GetID())
return node;
size_t count = node->GetChildCount();
size_t i;
for (i = 0; i < count; i++)
{
MyMusicModelNode *child = node->GetChildren().Item( i );
MyMusicModelNode *node2 = FindNodeRec( child, item );
if (node2)
return node2;
} }
return NULL;
}
MyMusicModelNode *FindNode( const wxDataViewItem &item ) const
{
if (item.GetID() == 0)
return NULL;
if (!m_root)
return NULL;
return FindNodeRec( m_root, item );
}
MyMusicModelNode* m_root;
MyMusicModelNode* m_pop;
MyMusicModelNode* m_classical;
wxUint32 m_idCounter;
}; };
// ------------------------------------- // -------------------------------------
@@ -195,9 +291,12 @@ public:
public: public:
void OnQuit(wxCommandEvent& event); void OnQuit(wxCommandEvent& event);
void OnAbout(wxCommandEvent& event); void OnAbout(wxCommandEvent& event);
void OnAdd(wxCommandEvent& event);
private: private:
wxDataViewCtrl* m_dataview; wxDataViewCtrl* m_dataview;
wxTextCtrl * m_log;
wxObjectDataPtr<MyMusicModel> m_model;
private: private:
DECLARE_EVENT_TABLE() DECLARE_EVENT_TABLE()
@@ -216,7 +315,7 @@ bool MyApp::OnInit(void)
// build the first frame // build the first frame
MyFrame *frame = MyFrame *frame =
new MyFrame(NULL, wxT("wxDataViewCtrl feature test"), 10, 10, 800, 340); new MyFrame(NULL, wxT("wxDataViewCtrl feature test"), 10, 10, 700, 440);
frame->Show(true); frame->Show(true);
SetTopWindow(frame); SetTopWindow(frame);
@@ -238,11 +337,14 @@ enum
// file menu // file menu
ID_ABOUT = wxID_ABOUT, ID_ABOUT = wxID_ABOUT,
ID_EXIT = wxID_EXIT, ID_EXIT = wxID_EXIT,
ID_ADD = 100,
}; };
BEGIN_EVENT_TABLE(MyFrame, wxFrame) BEGIN_EVENT_TABLE(MyFrame, wxFrame)
EVT_MENU( ID_ABOUT, MyFrame::OnAbout ) EVT_MENU( ID_ABOUT, MyFrame::OnAbout )
EVT_MENU( ID_EXIT, MyFrame::OnQuit ) EVT_MENU( ID_EXIT, MyFrame::OnQuit )
EVT_BUTTON( ID_ADD, MyFrame::OnAdd )
END_EVENT_TABLE() END_EVENT_TABLE()
MyFrame::MyFrame(wxFrame *frame, wxChar *title, int x, int y, int w, int h): MyFrame::MyFrame(wxFrame *frame, wxChar *title, int x, int y, int w, int h):
@@ -263,11 +365,13 @@ MyFrame::MyFrame(wxFrame *frame, wxChar *title, int x, int y, int w, int h):
SetMenuBar(menu_bar); SetMenuBar(menu_bar);
CreateStatusBar(); CreateStatusBar();
wxBoxSizer *main_sizer = new wxBoxSizer( wxVERTICAL );
m_dataview = new wxDataViewCtrl( this, wxID_ANY, wxDefaultPosition, m_dataview = new wxDataViewCtrl( this, wxID_ANY, wxDefaultPosition,
wxDefaultSize ); wxDefaultSize );
wxObjectDataPtr<MyMusicModel> model(new MyMusicModel); m_model = new MyMusicModel;
m_dataview->AssociateModel( model.get() ); m_dataview->AssociateModel( m_model.get() );
m_dataview->AppendTextColumn( "Title", 0, wxDATAVIEW_CELL_INERT, 200, m_dataview->AppendTextColumn( "Title", 0, wxDATAVIEW_CELL_INERT, 200,
DEFAULT_ALIGN, wxDATAVIEW_COL_SORTABLE ); DEFAULT_ALIGN, wxDATAVIEW_COL_SORTABLE );
@@ -275,6 +379,20 @@ MyFrame::MyFrame(wxFrame *frame, wxChar *title, int x, int y, int w, int h):
DEFAULT_ALIGN, wxDATAVIEW_COL_SORTABLE ); DEFAULT_ALIGN, wxDATAVIEW_COL_SORTABLE );
m_dataview->AppendTextColumn( "Year", 2, wxDATAVIEW_CELL_INERT, 50, m_dataview->AppendTextColumn( "Year", 2, wxDATAVIEW_CELL_INERT, 50,
DEFAULT_ALIGN ); DEFAULT_ALIGN );
main_sizer->Add( m_dataview, 2, wxGROW );
wxBoxSizer *button_sizer = new wxBoxSizer( wxHORIZONTAL );
button_sizer->Add( new wxButton( this, ID_ADD, "Add Mozart"), 0, wxALL, 10 );
main_sizer->Add( button_sizer, 0, 0, 0 );
m_log = new wxTextCtrl( this, -1, "", wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE );
main_sizer->Add( m_log, 1, wxGROW );
SetSizer( main_sizer );
} }
void MyFrame::OnQuit(wxCommandEvent& WXUNUSED(event) ) void MyFrame::OnQuit(wxCommandEvent& WXUNUSED(event) )
@@ -282,6 +400,25 @@ void MyFrame::OnQuit(wxCommandEvent& WXUNUSED(event) )
Close(true); Close(true);
} }
void MyFrame::OnAdd(wxCommandEvent& WXUNUSED(event) )
{
#if 0
// ignore selection, do something better later
wxDataViewItem item = m_dataview->GetSelection();
if (item.IsOk())
{
if (m_model->HasChildren(item))
{
}
else
{
}
}
#endif
m_model->AddToClassical( "Kleine Nachtmusik", "Wolfgang Mozart", "1787" );
}
void MyFrame::OnAbout(wxCommandEvent& WXUNUSED(event) ) void MyFrame::OnAbout(wxCommandEvent& WXUNUSED(event) )
{ {
wxAboutDialogInfo info; wxAboutDialogInfo info;

View File

@@ -117,8 +117,11 @@ public:
wxDataViewModel* GetDataViewModel() { return m_wx_model; } wxDataViewModel* GetDataViewModel() { return m_wx_model; }
GtkWxTreeModel* GetOwner() { return m_owner; } GtkWxTreeModel* GetOwner() { return m_owner; }
bool ItemAdded( const wxDataViewItem &parent, const wxDataViewItem &item );
protected: protected:
void InitTree(); void InitTree();
wxGtkTreeModelNode *FindNode( const wxDataViewItem &item );
wxGtkTreeModelNode *FindNode( GtkTreeIter *iter ); wxGtkTreeModelNode *FindNode( GtkTreeIter *iter );
void BuildBranch( wxGtkTreeModelNode *branch ); void BuildBranch( wxGtkTreeModelNode *branch );
@@ -602,6 +605,13 @@ void wxGtkTreeModel::BuildBranch( wxGtkTreeModelNode *node )
} }
} }
bool wxGtkTreeModel::ItemAdded( const wxDataViewItem &parent, const wxDataViewItem &item )
{
wxGtkTreeModelNode *parent_node = FindNode( parent );
parent_node->Append( new wxGtkTreeModelNode( parent_node, item ) );
return true;
}
gboolean wxGtkTreeModel::get_iter( GtkTreeIter *iter, GtkTreePath *path ) gboolean wxGtkTreeModel::get_iter( GtkTreeIter *iter, GtkTreePath *path )
{ {
int depth = gtk_tree_path_get_depth( path ); int depth = gtk_tree_path_get_depth( path );
@@ -634,7 +644,7 @@ GtkTreePath *wxGtkTreeModel::get_path( GtkTreeIter *iter )
while (node->GetParent()) while (node->GetParent())
{ {
wxGtkTreeModelNode *parent = node->GetParent(); wxGtkTreeModelNode *parent = node->GetParent();
int pos = parent->GetChildren().Index( parent ); int pos = parent->GetChildren().Index( node );
gtk_tree_path_prepend_index( retval, pos ); gtk_tree_path_prepend_index( retval, pos );
@@ -779,6 +789,20 @@ wxGtkTreeModelNode *wxGtkTreeModel::FindNode( GtkTreeIter *iter )
return result; return result;
} }
wxGtkTreeModelNode *wxGtkTreeModel::FindNode( const wxDataViewItem &item )
{
wxGtkTreeModelNode *result = wxGtkTreeModel_FindNode( m_root, item );
if (!result)
{
wxPrintf( "Not found %d\n", (int) item.GetID() );
char *crash = NULL;
*crash = 0;
}
return result;
}
//----------------------------------------------------------------------------- //-----------------------------------------------------------------------------
// define new GTK+ class wxGtkRendererRenderer // define new GTK+ class wxGtkRendererRenderer
//----------------------------------------------------------------------------- //-----------------------------------------------------------------------------
@@ -1177,6 +1201,8 @@ wxGtkDataViewModelNotifier::~wxGtkDataViewModelNotifier()
bool wxGtkDataViewModelNotifier::ItemAdded( const wxDataViewItem &parent, const wxDataViewItem &item ) bool wxGtkDataViewModelNotifier::ItemAdded( const wxDataViewItem &parent, const wxDataViewItem &item )
{ {
m_wxgtk_model->model->ItemAdded( parent, item );
GtkTreeIter iter; GtkTreeIter iter;
iter.stamp = m_wxgtk_model->stamp; iter.stamp = m_wxgtk_model->stamp;
iter.user_data = (gpointer) item.GetID(); iter.user_data = (gpointer) item.GetID();
@@ -2456,6 +2482,26 @@ bool wxDataViewCtrl::AppendColumn( wxDataViewColumn *col )
return true; return true;
} }
wxDataViewItem wxDataViewCtrl::GetSelection()
{
GtkTreeSelection *selection = gtk_tree_view_get_selection( GTK_TREE_VIEW(m_treeview) );
if (m_windowStyle & wxDV_MULTIPLE)
{
}
else
{
GtkTreeIter iter;
if (gtk_tree_selection_get_selected( selection, NULL, &iter ))
{
wxDataViewItem item( (wxUint32) iter.user_data );
return item;
}
}
return wxDataViewItem(0);
}
void wxDataViewCtrl::DoSetExpanderColumn() void wxDataViewCtrl::DoSetExpanderColumn()
{ {
} }