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__)
// for testing
#define wxUSE_GENERICDATAVIEWCTRL 1
// #define wxUSE_GENERICDATAVIEWCTRL 1
#elif defined(__WXMAC__)
#else
#define wxUSE_GENERICDATAVIEWCTRL 1
@ -418,6 +418,7 @@ public:
{ return m_indent; }
// TODO selection code
virtual wxDataViewItem GetSelection() = 0;
protected:
virtual void DoSetExpanderColumn() = 0 ;

View File

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

View File

@ -57,11 +57,102 @@ Implement this data model
7: German Requiem Johannes Brahms 1868
*/
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
{
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
{
@ -76,100 +167,105 @@ public:
virtual void GetValue( wxVariant &variant,
const wxDataViewItem &item, unsigned int col ) const
{
variant = wxString("");
int ID = item.GetID();
switch (ID)
MyMusicModelNode *node = FindNode( item );
switch (col)
{
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)
{
case 0: variant = wxString("You are not alone"); break;
case 1: variant = wxString("Michael Jackson"); break;
case 2: variant = wxString("1995");
}
}
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;
case 0: variant = node->m_title; break;
case 1: variant = node->m_artist; break;
case 2: variant = node->m_year; break;
default: wxLogError( "MyMusicModel::GetValue: wrong column" );
}
}
virtual bool SetValue( const wxVariant &variant,
const wxDataViewItem &item, unsigned int col )
{
// readonly
return true;
MyMusicModelNode *node = FindNode( item );
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
{
int ID = item.GetID();
return ((ID == 1) || (ID == 2) || (ID == 5) || (ID == 0));
if (item.GetID() == 0)
return true;
MyMusicModelNode *node = FindNode( item );
return node->IsContainer();
}
virtual wxDataViewItem GetFirstChild( const wxDataViewItem &parent ) const
{
int ID = parent.GetID();
switch (ID)
{
case 0: return wxDataViewItem( 1 );
case 1: return wxDataViewItem( 2 );
case 2: return wxDataViewItem( 3 );
case 5: return wxDataViewItem( 6 );
}
if (parent.GetID() == 0)
return wxDataViewItem( m_root->GetID() );
return wxDataViewItem(0);
MyMusicModelNode *node = FindNode( parent );
if (node->GetChildCount() == 0)
return wxDataViewItem( 0 );
MyMusicModelNode *first_child = node->GetChildren().Item( 0 );
return wxDataViewItem( first_child->GetID() );
}
virtual wxDataViewItem GetNextSibling( const wxDataViewItem &item ) const
{
int ID = item.GetID();
switch (ID)
{
case 2: return wxDataViewItem( 5 );
case 3: return wxDataViewItem( 4 );
case 6: return wxDataViewItem( 7 );
}
return wxDataViewItem(0);
MyMusicModelNode *node = FindNode( item );
MyMusicModelNode *parent = node->GetParent();
if (!parent)
return wxDataViewItem(0);
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() );
}
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:
void OnQuit(wxCommandEvent& event);
void OnAbout(wxCommandEvent& event);
void OnAdd(wxCommandEvent& event);
private:
wxDataViewCtrl* m_dataview;
wxTextCtrl * m_log;
wxObjectDataPtr<MyMusicModel> m_model;
private:
DECLARE_EVENT_TABLE()
@ -216,7 +315,7 @@ bool MyApp::OnInit(void)
// build the first 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);
SetTopWindow(frame);
@ -238,11 +337,14 @@ enum
// file menu
ID_ABOUT = wxID_ABOUT,
ID_EXIT = wxID_EXIT,
ID_ADD = 100,
};
BEGIN_EVENT_TABLE(MyFrame, wxFrame)
EVT_MENU( ID_ABOUT, MyFrame::OnAbout )
EVT_MENU( ID_EXIT, MyFrame::OnQuit )
EVT_BUTTON( ID_ADD, MyFrame::OnAdd )
END_EVENT_TABLE()
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);
CreateStatusBar();
wxBoxSizer *main_sizer = new wxBoxSizer( wxVERTICAL );
m_dataview = new wxDataViewCtrl( this, wxID_ANY, wxDefaultPosition,
wxDefaultSize );
wxObjectDataPtr<MyMusicModel> model(new MyMusicModel);
m_dataview->AssociateModel( model.get() );
m_model = new MyMusicModel;
m_dataview->AssociateModel( m_model.get() );
m_dataview->AppendTextColumn( "Title", 0, wxDATAVIEW_CELL_INERT, 200,
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 );
m_dataview->AppendTextColumn( "Year", 2, wxDATAVIEW_CELL_INERT, 50,
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) )
@ -282,6 +400,25 @@ void MyFrame::OnQuit(wxCommandEvent& WXUNUSED(event) )
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) )
{
wxAboutDialogInfo info;

View File

@ -116,9 +116,12 @@ public:
wxDataViewModel* GetDataViewModel() { return m_wx_model; }
GtkWxTreeModel* GetOwner() { return m_owner; }
bool ItemAdded( const wxDataViewItem &parent, const wxDataViewItem &item );
protected:
void InitTree();
wxGtkTreeModelNode *FindNode( const wxDataViewItem &item );
wxGtkTreeModelNode *FindNode( GtkTreeIter *iter );
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 )
{
int depth = gtk_tree_path_get_depth( path );
@ -634,7 +644,7 @@ GtkTreePath *wxGtkTreeModel::get_path( GtkTreeIter *iter )
while (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 );
@ -779,6 +789,20 @@ wxGtkTreeModelNode *wxGtkTreeModel::FindNode( GtkTreeIter *iter )
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
//-----------------------------------------------------------------------------
@ -1177,6 +1201,8 @@ wxGtkDataViewModelNotifier::~wxGtkDataViewModelNotifier()
bool wxGtkDataViewModelNotifier::ItemAdded( const wxDataViewItem &parent, const wxDataViewItem &item )
{
m_wxgtk_model->model->ItemAdded( parent, item );
GtkTreeIter iter;
iter.stamp = m_wxgtk_model->stamp;
iter.user_data = (gpointer) item.GetID();
@ -2456,6 +2482,26 @@ bool wxDataViewCtrl::AppendColumn( wxDataViewColumn *col )
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()
{
}