wxTreeCtrl::Sort() changed, adapted the sample to show it.

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@1237 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Vadim Zeitlin
1998-12-18 15:37:19 +00:00
parent 5fbc0fa65a
commit e1ee62bd79
4 changed files with 143 additions and 48 deletions

View File

@@ -46,7 +46,6 @@ class wxImageList;
class wxGenericTreeItem; class wxGenericTreeItem;
class wxTreeItemData; class wxTreeItemData;
typedef int (*wxTreeItemCmpFunc)(wxTreeItemData *item1, wxTreeItemData *item2);
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
// wxTreeItemId - unique identifier of a tree element // wxTreeItemId - unique identifier of a tree element
@@ -415,13 +414,17 @@ public:
// end editing and accept or discard the changes to item label // end editing and accept or discard the changes to item label
void EndEditLabel(const wxTreeItemId& item, bool discardChanges = FALSE); void EndEditLabel(const wxTreeItemId& item, bool discardChanges = FALSE);
// sort the children of this item using the specified callback function // sorting
// (it should return -1, 0 or +1 as usual), if it's not specified // this function is called to compare 2 items and should return -1, 0
// alphabetical comparaison is performed. // or +1 if the first item is less than, equal to or greater than the
// second one. The base class version performs alphabetic comparaison
// of item labels (GetText)
virtual int OnCompareItems(const wxTreeItemId& item1,
const wxTreeItemId& item2);
// sort the children of this item using OnCompareItems
// //
// NB: this function is not reentrant! // NB: this function is not reentrant and not MT-safe (FIXME)!
void SortChildren(const wxTreeItemId& item, void SortChildren(const wxTreeItemId& item);
wxTreeItemCmpFunc *cmpFunction = NULL);
// callbacks // callbacks
void OnPaint( wxPaintEvent &event ); void OnPaint( wxPaintEvent &event );

View File

@@ -39,10 +39,22 @@
#include "treetest.h" #include "treetest.h"
// verify that the item is ok and insult the user if it is not
#define CHECK_ITEM( item ) if ( !item.IsOk() ) { \
wxMessageBox("Please select some item first!", \
"Tree sample error", \
wxOK | wxICON_EXCLAMATION, \
this); \
return; \
}
BEGIN_EVENT_TABLE(MyFrame, wxFrame) BEGIN_EVENT_TABLE(MyFrame, wxFrame)
EVT_MENU(TreeTest_Quit, MyFrame::OnQuit) EVT_MENU(TreeTest_Quit, MyFrame::OnQuit)
EVT_MENU(TreeTest_About, MyFrame::OnAbout) EVT_MENU(TreeTest_About, MyFrame::OnAbout)
EVT_MENU(TreeTest_Dump, MyFrame::OnDump) EVT_MENU(TreeTest_Dump, MyFrame::OnDump)
EVT_MENU(TreeTest_Rename, MyFrame::OnRename)
EVT_MENU(TreeTest_Sort, MyFrame::OnSort)
EVT_MENU(TreeTest_SortRev, MyFrame::OnSortRev)
EVT_MENU(TreeTest_Bold, MyFrame::OnSetBold) EVT_MENU(TreeTest_Bold, MyFrame::OnSetBold)
EVT_MENU(TreeTest_UnBold, MyFrame::OnClearBold) EVT_MENU(TreeTest_UnBold, MyFrame::OnClearBold)
EVT_MENU(TreeTest_Delete, MyFrame::OnDelete) EVT_MENU(TreeTest_Delete, MyFrame::OnDelete)
@@ -98,24 +110,30 @@ MyFrame::MyFrame(const wxString& title, int x, int y, int w, int h)
SetIcon(wxICON(mondrian)); SetIcon(wxICON(mondrian));
// Make a menubar // Make a menubar
wxMenu *file_menu = new wxMenu; wxMenu *file_menu = new wxMenu,
*tree_menu = new wxMenu;
file_menu->Append(TreeTest_Dump, "D&ump tree items");
file_menu->Append(TreeTest_Recreate, "&Recreate the tree");
file_menu->AppendSeparator();
file_menu->Append(TreeTest_Delete, "&Delete this item");
file_menu->Append(TreeTest_DeleteChildren, "Delete &children");
file_menu->Append(TreeTest_DeleteAll, "Delete &all items");
file_menu->AppendSeparator();
file_menu->Append(TreeTest_Bold, "Make item &bold");
file_menu->Append(TreeTest_UnBold, "Make item &not bold");
file_menu->AppendSeparator();
file_menu->Append(TreeTest_About, "&About..."); file_menu->Append(TreeTest_About, "&About...");
file_menu->AppendSeparator(); file_menu->AppendSeparator();
file_menu->Append(TreeTest_Quit, "E&xit"); file_menu->Append(TreeTest_Quit, "E&xit");
tree_menu->Append(TreeTest_Dump, "D&ump tree items");
tree_menu->Append(TreeTest_Recreate, "&Recreate the tree");
tree_menu->AppendSeparator();
tree_menu->Append(TreeTest_Delete, "&Delete this item");
tree_menu->Append(TreeTest_DeleteChildren, "Delete &children");
tree_menu->Append(TreeTest_DeleteAll, "Delete &all items");
tree_menu->AppendSeparator();
tree_menu->Append(TreeTest_Sort, "Sort children of current item");
tree_menu->Append(TreeTest_SortRev, "Sort in reversed order");
tree_menu->Append(TreeTest_Rename, "Rename item...");
tree_menu->AppendSeparator();
tree_menu->Append(TreeTest_Bold, "Make item &bold");
tree_menu->Append(TreeTest_UnBold, "Make item &not bold");
wxMenuBar *menu_bar = new wxMenuBar; wxMenuBar *menu_bar = new wxMenuBar;
menu_bar->Append(file_menu, "&File"); menu_bar->Append(file_menu, "&File");
menu_bar->Append(tree_menu, "&Tree");
SetMenuBar(menu_bar); SetMenuBar(menu_bar);
// Make a panel with a message // Make a panel with a message
@@ -168,34 +186,70 @@ void MyFrame::OnQuit(wxCommandEvent& WXUNUSED(event))
void MyFrame::OnAbout(wxCommandEvent& WXUNUSED(event)) void MyFrame::OnAbout(wxCommandEvent& WXUNUSED(event))
{ {
wxMessageDialog dialog(this, wxMessageBox("Tree test sample\n"
"Tree test sample\n" "Julian Smart (c) 1997,\n"
"Julian Smart (c) 1997", "Vadim Zeitlin (c) 1998",
"About tree test", wxOK); "About tree test",
wxOK | wxICON_INFORMATION, this);
}
dialog.ShowModal(); void MyFrame::OnRename(wxCommandEvent& WXUNUSED(event))
{
wxTreeItemId item = m_treeCtrl->GetSelection();
CHECK_ITEM( item );
static wxString s_text;
s_text = wxGetTextFromUser("New name: ", "Tree sample question",
s_text, this);
if ( !s_text.IsEmpty() )
{
m_treeCtrl->SetItemText(item, s_text);
}
}
void MyFrame::DoSort(bool reverse)
{
wxTreeItemId item = m_treeCtrl->GetSelection();
CHECK_ITEM( item );
m_treeCtrl->DoSortChildren(item, reverse);
} }
void MyFrame::OnDump(wxCommandEvent& WXUNUSED(event)) void MyFrame::OnDump(wxCommandEvent& WXUNUSED(event))
{ {
wxTreeItemId root = m_treeCtrl->GetSelection(); wxTreeItemId root = m_treeCtrl->GetSelection();
CHECK_ITEM( root );
m_treeCtrl->GetItemsRecursively(root, -1); m_treeCtrl->GetItemsRecursively(root, -1);
} }
void MyFrame::DoSetBold(bool bold) void MyFrame::DoSetBold(bool bold)
{ {
m_treeCtrl->SetItemBold(m_treeCtrl->GetSelection(), bold); wxTreeItemId item = m_treeCtrl->GetSelection();
CHECK_ITEM( item );
m_treeCtrl->SetItemBold(item, bold);
} }
void MyFrame::OnDelete(wxCommandEvent& WXUNUSED(event)) void MyFrame::OnDelete(wxCommandEvent& WXUNUSED(event))
{ {
wxTreeItemId item = m_treeCtrl->GetSelection(); wxTreeItemId item = m_treeCtrl->GetSelection();
CHECK_ITEM( item );
m_treeCtrl->Delete(item); m_treeCtrl->Delete(item);
} }
void MyFrame::OnDeleteChildren(wxCommandEvent& WXUNUSED(event)) void MyFrame::OnDeleteChildren(wxCommandEvent& WXUNUSED(event))
{ {
wxTreeItemId item = m_treeCtrl->GetSelection(); wxTreeItemId item = m_treeCtrl->GetSelection();
CHECK_ITEM( item );
m_treeCtrl->DeleteChildren(item); m_treeCtrl->DeleteChildren(item);
} }
@@ -216,6 +270,8 @@ MyTreeCtrl::MyTreeCtrl(wxWindow *parent, const wxWindowID id,
long style) long style)
: wxTreeCtrl(parent, id, pos, size, style) : wxTreeCtrl(parent, id, pos, size, style)
{ {
m_reverseSort = FALSE;
// Make an image list containing small icons // Make an image list containing small icons
m_imageListNormal = new wxImageList(16, 16, TRUE); m_imageListNormal = new wxImageList(16, 16, TRUE);
@@ -234,6 +290,20 @@ MyTreeCtrl::~MyTreeCtrl()
delete m_imageListNormal; delete m_imageListNormal;
} }
int MyTreeCtrl::OnCompareItems(const wxTreeItemId& item1,
const wxTreeItemId& item2)
{
if ( m_reverseSort )
{
// just exchange 1st and 2nd items
return wxTreeCtrl::OnCompareItems(item2, item1);
}
else
{
return wxTreeCtrl::OnCompareItems(item1, item2);
}
}
void MyTreeCtrl::AddItemsRecursively(const wxTreeItemId& idParent, void MyTreeCtrl::AddItemsRecursively(const wxTreeItemId& idParent,
size_t numChildren, size_t numChildren,
size_t depth, size_t depth,

View File

@@ -62,6 +62,13 @@ public:
void AddTestItemsToTree(size_t numChildren, void AddTestItemsToTree(size_t numChildren,
size_t depth); size_t depth);
void DoSortChildren(const wxTreeItemId& item, bool reverse = FALSE)
{ m_reverseSort = reverse; wxTreeCtrl::SortChildren(item); }
protected:
virtual int OnCompareItems(const wxTreeItemId& item1,
const wxTreeItemId& item2);
private: private:
void AddItemsRecursively(const wxTreeItemId& idParent, void AddItemsRecursively(const wxTreeItemId& idParent,
size_t nChildren, size_t nChildren,
@@ -69,6 +76,7 @@ private:
size_t folder); size_t folder);
wxImageList *m_imageListNormal; wxImageList *m_imageListNormal;
bool m_reverseSort; // flag for OnCompareItems
DECLARE_EVENT_TABLE() DECLARE_EVENT_TABLE()
}; };
@@ -93,7 +101,13 @@ public:
void OnSetBold(wxCommandEvent& WXUNUSED(event)) { DoSetBold(TRUE); } void OnSetBold(wxCommandEvent& WXUNUSED(event)) { DoSetBold(TRUE); }
void OnClearBold(wxCommandEvent& WXUNUSED(event)) { DoSetBold(FALSE); } void OnClearBold(wxCommandEvent& WXUNUSED(event)) { DoSetBold(FALSE); }
void OnRename(wxCommandEvent& event);
void OnSort(wxCommandEvent& event) { DoSort(); }
void OnSortRev(wxCommandEvent& event) { DoSort(TRUE); }
private: private:
void DoSort(bool reverse = FALSE);
MyTreeCtrl *m_treeCtrl; MyTreeCtrl *m_treeCtrl;
void DoSetBold(bool bold = TRUE); void DoSetBold(bool bold = TRUE);
@@ -107,8 +121,11 @@ enum
TreeTest_Quit, TreeTest_Quit,
TreeTest_About, TreeTest_About,
TreeTest_Dump, TreeTest_Dump,
TreeTest_Sort,
TreeTest_SortRev,
TreeTest_Bold, TreeTest_Bold,
TreeTest_UnBold, TreeTest_UnBold,
TreeTest_Rename,
TreeTest_Delete, TreeTest_Delete,
TreeTest_DeleteChildren, TreeTest_DeleteChildren,
TreeTest_DeleteAll, TreeTest_DeleteAll,

View File

@@ -935,38 +935,43 @@ void wxTreeCtrl::EndEditLabel(const wxTreeItemId& WXUNUSED(item), bool WXUNUSED(
wxFAIL_MSG("not implemented"); wxFAIL_MSG("not implemented");
} }
wxTreeItemCmpFunc tree_ctrl_compare_func_2; // FIXME: tree sorting functions are not reentrant and not MT-safe!
static wxTreeCtrl *s_treeBeingSorted = NULL;
int tree_ctrl_compare_func_1( wxGenericTreeItem **line1, wxGenericTreeItem **line2 ) static int tree_ctrl_compare_func(wxGenericTreeItem **item1,
wxGenericTreeItem **item2)
{ {
if (tree_ctrl_compare_func_2 == NULL) wxCHECK_MSG( s_treeBeingSorted, 0, "bug in wxTreeCtrl::SortChildren()" );
{
return strcmp( (*line1)->GetText(), (*line2)->GetText() ); return s_treeBeingSorted->OnCompareItems(*item1, *item2);
}
else
{
wxTreeItemData *data1 = (*line1)->GetData();
wxTreeItemData *data2 = (*line2)->GetData();
return tree_ctrl_compare_func_2( data1, data2 );
}
} }
void wxTreeCtrl::SortChildren( const wxTreeItemId& item, int wxTreeCtrl::OnCompareItems(const wxTreeItemId& item1,
wxTreeItemCmpFunc *cmpFunction) const wxTreeItemId& item2)
{ {
wxGenericTreeItem *gitem = item.m_pItem; return strcmp(GetItemText(item1), GetItemText(item2));
}
if (!gitem) return; void wxTreeCtrl::SortChildren(const wxTreeItemId& itemId)
{
wxCHECK_RET( itemId.IsOk(), "invalid tree item" );
if (cmpFunction == NULL) wxGenericTreeItem *item = itemId.m_pItem;
tree_ctrl_compare_func_2 = NULL;
else
tree_ctrl_compare_func_2 = *cmpFunction;
gitem->GetChildren().Sort( *tree_ctrl_compare_func_1 ); wxCHECK_RET( !s_treeBeingSorted,
"wxTreeCtrl::SortChildren is not reentrant" );
wxArrayTreeItems& children = item->GetChildren();
if ( children.Count() > 1 )
{
s_treeBeingSorted = this;
children.Sort(tree_ctrl_compare_func);
s_treeBeingSorted = NULL;
m_dirty = TRUE; m_dirty = TRUE;
} }
//else: don't make the tree dirty as nothing changed
}
wxImageList *wxTreeCtrl::GetImageList() const wxImageList *wxTreeCtrl::GetImageList() const
{ {