class wxGenericTreeItem;
class wxTreeItemData;
-typedef int (*wxTreeItemCmpFunc)(wxTreeItemData *item1, wxTreeItemData *item2);
// -----------------------------------------------------------------------------
// wxTreeItemId - unique identifier of a tree element
// end editing and accept or discard the changes to item label
void EndEditLabel(const wxTreeItemId& item, bool discardChanges = FALSE);
- // sort the children of this item using the specified callback function
- // (it should return -1, 0 or +1 as usual), if it's not specified
- // alphabetical comparaison is performed.
+ // sorting
+ // this function is called to compare 2 items and should return -1, 0
+ // 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!
- void SortChildren(const wxTreeItemId& item,
- wxTreeItemCmpFunc *cmpFunction = NULL);
+ // NB: this function is not reentrant and not MT-safe (FIXME)!
+ void SortChildren(const wxTreeItemId& item);
// callbacks
void OnPaint( wxPaintEvent &event );
#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)
EVT_MENU(TreeTest_Quit, MyFrame::OnQuit)
EVT_MENU(TreeTest_About, MyFrame::OnAbout)
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_UnBold, MyFrame::OnClearBold)
EVT_MENU(TreeTest_Delete, MyFrame::OnDelete)
SetIcon(wxICON(mondrian));
// 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 ¬ bold");
- file_menu->AppendSeparator();
file_menu->Append(TreeTest_About, "&About...");
file_menu->AppendSeparator();
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 ¬ bold");
+
wxMenuBar *menu_bar = new wxMenuBar;
menu_bar->Append(file_menu, "&File");
+ menu_bar->Append(tree_menu, "&Tree");
SetMenuBar(menu_bar);
// Make a panel with a message
void MyFrame::OnAbout(wxCommandEvent& WXUNUSED(event))
{
- wxMessageDialog dialog(this,
- "Tree test sample\n"
- "Julian Smart (c) 1997",
- "About tree test", wxOK);
+ wxMessageBox("Tree test sample\n"
+ "Julian Smart (c) 1997,\n"
+ "Vadim Zeitlin (c) 1998",
+ "About tree test",
+ wxOK | wxICON_INFORMATION, this);
+}
+
+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();
- dialog.ShowModal();
+ CHECK_ITEM( item );
+
+ m_treeCtrl->DoSortChildren(item, reverse);
}
void MyFrame::OnDump(wxCommandEvent& WXUNUSED(event))
{
wxTreeItemId root = m_treeCtrl->GetSelection();
+
+ CHECK_ITEM( root );
+
m_treeCtrl->GetItemsRecursively(root, -1);
}
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))
{
wxTreeItemId item = m_treeCtrl->GetSelection();
+
+ CHECK_ITEM( item );
+
m_treeCtrl->Delete(item);
}
void MyFrame::OnDeleteChildren(wxCommandEvent& WXUNUSED(event))
{
wxTreeItemId item = m_treeCtrl->GetSelection();
+
+ CHECK_ITEM( item );
+
m_treeCtrl->DeleteChildren(item);
}
long style)
: wxTreeCtrl(parent, id, pos, size, style)
{
+ m_reverseSort = FALSE;
+
// Make an image list containing small icons
m_imageListNormal = new wxImageList(16, 16, TRUE);
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,
size_t numChildren,
size_t depth,
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)
- {
- return strcmp( (*line1)->GetText(), (*line2)->GetText() );
- }
- else
- {
- wxTreeItemData *data1 = (*line1)->GetData();
- wxTreeItemData *data2 = (*line2)->GetData();
- return tree_ctrl_compare_func_2( data1, data2 );
- }
+ wxCHECK_MSG( s_treeBeingSorted, 0, "bug in wxTreeCtrl::SortChildren()" );
+
+ return s_treeBeingSorted->OnCompareItems(*item1, *item2);
}
-void wxTreeCtrl::SortChildren( const wxTreeItemId& item,
- wxTreeItemCmpFunc *cmpFunction)
+int wxTreeCtrl::OnCompareItems(const wxTreeItemId& item1,
+ const wxTreeItemId& item2)
{
- wxGenericTreeItem *gitem = item.m_pItem;
-
- if (!gitem) return;
+ return strcmp(GetItemText(item1), GetItemText(item2));
+}
+
+void wxTreeCtrl::SortChildren(const wxTreeItemId& itemId)
+{
+ wxCHECK_RET( itemId.IsOk(), "invalid tree item" );
+
+ wxGenericTreeItem *item = itemId.m_pItem;
- if (cmpFunction == NULL)
- 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