treetest.cpp 51 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698
  1. /////////////////////////////////////////////////////////////////////////////
  2. // Name: treetest.cpp
  3. // Purpose: wxTreeCtrl sample
  4. // Author: Julian Smart
  5. // Modified by:
  6. // Created: 04/01/98
  7. // Copyright: (c) Julian Smart
  8. // Licence: wxWindows licence
  9. /////////////////////////////////////////////////////////////////////////////
  10. // For compilers that support precompilation, includes "wx/wx.h".
  11. #include "wx/wxprec.h"
  12. #ifdef __BORLANDC__
  13. #pragma hdrstop
  14. #endif
  15. #ifndef WX_PRECOMP
  16. #include "wx/wx.h"
  17. #include "wx/log.h"
  18. #endif
  19. #include "wx/colordlg.h"
  20. #include "wx/numdlg.h"
  21. #include "wx/artprov.h"
  22. #include "wx/image.h"
  23. #include "wx/imaglist.h"
  24. #include "wx/treectrl.h"
  25. #include "wx/math.h"
  26. #include "wx/renderer.h"
  27. #include "wx/wupdlock.h"
  28. #ifdef __WIN32__
  29. // this is not supported by native control
  30. #define NO_VARIABLE_HEIGHT
  31. #endif
  32. #include "treetest.h"
  33. #include "icon1.xpm"
  34. #include "icon2.xpm"
  35. #include "icon3.xpm"
  36. #include "icon4.xpm"
  37. #include "icon5.xpm"
  38. #include "state1.xpm"
  39. #include "state2.xpm"
  40. #include "state3.xpm"
  41. #include "state4.xpm"
  42. #include "state5.xpm"
  43. #include "unchecked.xpm"
  44. #include "checked.xpm"
  45. #ifndef wxHAS_IMAGES_IN_RESOURCES
  46. #include "../sample.xpm"
  47. #endif
  48. static const int NUM_CHILDREN_PER_LEVEL = 5;
  49. static const int NUM_LEVELS = 2;
  50. // verify that the item is ok and insult the user if it is not
  51. #define CHECK_ITEM( item ) if ( !item.IsOk() ) { \
  52. wxMessageBox(wxT("Please select some item first!"), \
  53. wxT("Tree sample error"), \
  54. wxOK | wxICON_EXCLAMATION, \
  55. this); \
  56. return; \
  57. }
  58. #define MENU_LINK(name) EVT_MENU(TreeTest_##name, MyFrame::On##name)
  59. wxBEGIN_EVENT_TABLE(MyFrame, wxFrame)
  60. EVT_IDLE(MyFrame::OnIdle)
  61. EVT_SIZE(MyFrame::OnSize)
  62. MENU_LINK(Quit)
  63. MENU_LINK(About)
  64. MENU_LINK(ClearLog)
  65. MENU_LINK(TogButtons)
  66. MENU_LINK(TogTwist)
  67. MENU_LINK(TogLines)
  68. MENU_LINK(TogEdit)
  69. MENU_LINK(TogHideRoot)
  70. MENU_LINK(TogRootLines)
  71. MENU_LINK(TogBorder)
  72. MENU_LINK(TogFullHighlight)
  73. MENU_LINK(SetFgColour)
  74. MENU_LINK(SetBgColour)
  75. MENU_LINK(ResetStyle)
  76. MENU_LINK(Highlight)
  77. MENU_LINK(Dump)
  78. #ifndef NO_MULTIPLE_SELECTION
  79. MENU_LINK(DumpSelected)
  80. MENU_LINK(Select)
  81. MENU_LINK(Unselect)
  82. MENU_LINK(ToggleSel)
  83. MENU_LINK(SelectChildren)
  84. #endif // NO_MULTIPLE_SELECTION
  85. MENU_LINK(Rename)
  86. MENU_LINK(Count)
  87. MENU_LINK(CountRec)
  88. MENU_LINK(Sort)
  89. MENU_LINK(SortRev)
  90. MENU_LINK(SetBold)
  91. MENU_LINK(ClearBold)
  92. MENU_LINK(Delete)
  93. MENU_LINK(DeleteChildren)
  94. MENU_LINK(DeleteAll)
  95. MENU_LINK(Recreate)
  96. MENU_LINK(ToggleImages)
  97. MENU_LINK(ToggleStates)
  98. MENU_LINK(ToggleBell)
  99. MENU_LINK(ToggleAlternateImages)
  100. MENU_LINK(ToggleAlternateStates)
  101. MENU_LINK(ToggleButtons)
  102. MENU_LINK(SetImageSize)
  103. MENU_LINK(CollapseAndReset)
  104. MENU_LINK(EnsureVisible)
  105. MENU_LINK(SetFocus)
  106. MENU_LINK(AddItem)
  107. MENU_LINK(AddManyItems)
  108. MENU_LINK(InsertItem)
  109. MENU_LINK(IncIndent)
  110. MENU_LINK(DecIndent)
  111. MENU_LINK(IncSpacing)
  112. MENU_LINK(DecSpacing)
  113. MENU_LINK(ToggleIcon)
  114. MENU_LINK(ToggleState)
  115. MENU_LINK(SelectRoot)
  116. MENU_LINK(SetFocusedRoot)
  117. MENU_LINK(ClearFocused)
  118. MENU_LINK(ShowFirstVisible)
  119. #ifdef wxHAS_LAST_VISIBLE
  120. MENU_LINK(ShowLastVisible)
  121. #endif // wxHAS_LAST_VISIBLE
  122. MENU_LINK(ShowNextVisible)
  123. MENU_LINK(ShowPrevVisible)
  124. MENU_LINK(ShowParent)
  125. MENU_LINK(ShowPrevSibling)
  126. MENU_LINK(ShowNextSibling)
  127. MENU_LINK(ScrollTo)
  128. MENU_LINK(SelectLast)
  129. #undef MENU_LINK
  130. wxEND_EVENT_TABLE()
  131. #if USE_GENERIC_TREECTRL
  132. wxBEGIN_EVENT_TABLE(MyTreeCtrl, wxGenericTreeCtrl)
  133. #else
  134. wxBEGIN_EVENT_TABLE(MyTreeCtrl, wxTreeCtrl)
  135. #endif
  136. EVT_TREE_BEGIN_DRAG(TreeTest_Ctrl, MyTreeCtrl::OnBeginDrag)
  137. EVT_TREE_BEGIN_RDRAG(TreeTest_Ctrl, MyTreeCtrl::OnBeginRDrag)
  138. EVT_TREE_END_DRAG(TreeTest_Ctrl, MyTreeCtrl::OnEndDrag)
  139. EVT_TREE_BEGIN_LABEL_EDIT(TreeTest_Ctrl, MyTreeCtrl::OnBeginLabelEdit)
  140. EVT_TREE_END_LABEL_EDIT(TreeTest_Ctrl, MyTreeCtrl::OnEndLabelEdit)
  141. EVT_TREE_DELETE_ITEM(TreeTest_Ctrl, MyTreeCtrl::OnDeleteItem)
  142. #if 0 // there are so many of those that logging them causes flicker
  143. EVT_TREE_GET_INFO(TreeTest_Ctrl, MyTreeCtrl::OnGetInfo)
  144. #endif
  145. EVT_TREE_SET_INFO(TreeTest_Ctrl, MyTreeCtrl::OnSetInfo)
  146. EVT_TREE_ITEM_EXPANDED(TreeTest_Ctrl, MyTreeCtrl::OnItemExpanded)
  147. EVT_TREE_ITEM_EXPANDING(TreeTest_Ctrl, MyTreeCtrl::OnItemExpanding)
  148. EVT_TREE_ITEM_COLLAPSED(TreeTest_Ctrl, MyTreeCtrl::OnItemCollapsed)
  149. EVT_TREE_ITEM_COLLAPSING(TreeTest_Ctrl, MyTreeCtrl::OnItemCollapsing)
  150. EVT_TREE_SEL_CHANGED(TreeTest_Ctrl, MyTreeCtrl::OnSelChanged)
  151. EVT_TREE_SEL_CHANGING(TreeTest_Ctrl, MyTreeCtrl::OnSelChanging)
  152. EVT_TREE_KEY_DOWN(TreeTest_Ctrl, MyTreeCtrl::OnTreeKeyDown)
  153. EVT_TREE_ITEM_ACTIVATED(TreeTest_Ctrl, MyTreeCtrl::OnItemActivated)
  154. EVT_TREE_STATE_IMAGE_CLICK(TreeTest_Ctrl, MyTreeCtrl::OnItemStateClick)
  155. // so many different ways to handle right mouse button clicks...
  156. EVT_CONTEXT_MENU(MyTreeCtrl::OnContextMenu)
  157. // EVT_TREE_ITEM_MENU is the preferred event for creating context menus
  158. // on a tree control, because it includes the point of the click or item,
  159. // meaning that no additional placement calculations are required.
  160. EVT_TREE_ITEM_MENU(TreeTest_Ctrl, MyTreeCtrl::OnItemMenu)
  161. EVT_TREE_ITEM_RIGHT_CLICK(TreeTest_Ctrl, MyTreeCtrl::OnItemRClick)
  162. EVT_RIGHT_DOWN(MyTreeCtrl::OnRMouseDown)
  163. EVT_RIGHT_UP(MyTreeCtrl::OnRMouseUp)
  164. EVT_RIGHT_DCLICK(MyTreeCtrl::OnRMouseDClick)
  165. wxEND_EVENT_TABLE()
  166. IMPLEMENT_APP(MyApp)
  167. bool MyApp::OnInit()
  168. {
  169. if ( !wxApp::OnInit() )
  170. return false;
  171. // Create the main frame window
  172. MyFrame *frame = new MyFrame(wxT("wxTreeCtrl Test"), 50, 50, 450, 600);
  173. // Show the frame
  174. frame->Show(true);
  175. return true;
  176. }
  177. // My frame constructor
  178. MyFrame::MyFrame(const wxString& title, int x, int y, int w, int h)
  179. : wxFrame((wxFrame *)NULL, wxID_ANY, title, wxPoint(x, y), wxSize(w, h)),
  180. m_treeCtrl(NULL)
  181. #if wxUSE_LOG
  182. , m_textCtrl(NULL)
  183. #endif // wxUSE_LOG
  184. {
  185. // This reduces flicker effects - even better would be to define
  186. // OnEraseBackground to do nothing. When the tree control's scrollbars are
  187. // show or hidden, the frame is sent a background erase event.
  188. SetBackgroundColour(*wxWHITE);
  189. // Give it an icon
  190. SetIcon(wxICON(sample));
  191. #if wxUSE_MENUS
  192. // Make a menubar
  193. wxMenu *file_menu = new wxMenu,
  194. *style_menu = new wxMenu,
  195. *tree_menu = new wxMenu,
  196. *item_menu = new wxMenu;
  197. #if wxUSE_LOG
  198. file_menu->Append(TreeTest_ClearLog, wxT("&Clear log\tCtrl-L"));
  199. file_menu->AppendSeparator();
  200. #endif // wxUSE_LOG
  201. file_menu->Append(TreeTest_About, wxT("&About"));
  202. file_menu->AppendSeparator();
  203. file_menu->Append(TreeTest_Quit, wxT("E&xit\tAlt-X"));
  204. style_menu->AppendCheckItem(TreeTest_TogButtons, wxT("Toggle &normal buttons"));
  205. style_menu->AppendCheckItem(TreeTest_TogTwist, wxT("Toggle &twister buttons"));
  206. style_menu->AppendCheckItem(TreeTest_ToggleButtons, wxT("Toggle image &buttons"));
  207. style_menu->AppendSeparator();
  208. style_menu->AppendCheckItem(TreeTest_TogLines, wxT("Toggle &connecting lines"));
  209. style_menu->AppendCheckItem(TreeTest_TogRootLines, wxT("Toggle &lines at root"));
  210. style_menu->AppendCheckItem(TreeTest_TogHideRoot, wxT("Toggle &hidden root"));
  211. style_menu->AppendCheckItem(TreeTest_TogBorder, wxT("Toggle &item border"));
  212. style_menu->AppendCheckItem(TreeTest_TogFullHighlight, wxT("Toggle &full row highlight"));
  213. style_menu->AppendCheckItem(TreeTest_TogEdit, wxT("Toggle &edit mode"));
  214. #ifndef NO_MULTIPLE_SELECTION
  215. style_menu->AppendCheckItem(TreeTest_ToggleSel, wxT("Toggle &selection mode\tCtrl-S"));
  216. #endif // NO_MULTIPLE_SELECTION
  217. style_menu->AppendCheckItem(TreeTest_ToggleImages, wxT("Toggle show ima&ges"));
  218. style_menu->AppendCheckItem(TreeTest_ToggleStates, wxT("Toggle show st&ates"));
  219. style_menu->AppendCheckItem(TreeTest_ToggleBell, wxT("Toggle &bell on no match"));
  220. style_menu->AppendCheckItem(TreeTest_ToggleAlternateImages, wxT("Toggle alternate images"));
  221. style_menu->AppendCheckItem(TreeTest_ToggleAlternateStates, wxT("Toggle alternate state images"));
  222. style_menu->Append(TreeTest_SetImageSize, wxT("Set image si&ze..."));
  223. style_menu->AppendSeparator();
  224. style_menu->Append(TreeTest_SetFgColour, wxT("Set &foreground colour..."));
  225. style_menu->Append(TreeTest_SetBgColour, wxT("Set &background colour..."));
  226. style_menu->AppendSeparator();
  227. style_menu->Append(TreeTest_ResetStyle, wxT("&Reset to default\tF10"));
  228. tree_menu->Append(TreeTest_Recreate, wxT("&Recreate the tree"));
  229. tree_menu->Append(TreeTest_CollapseAndReset, wxT("C&ollapse and reset"));
  230. tree_menu->AppendSeparator();
  231. tree_menu->Append(TreeTest_AddItem, wxT("Append a &new item"));
  232. tree_menu->Append(TreeTest_AddManyItems, wxT("Appends &many items"));
  233. tree_menu->Append(TreeTest_InsertItem, wxT("&Insert a new item"));
  234. tree_menu->Append(TreeTest_Delete, wxT("&Delete this item"));
  235. tree_menu->Append(TreeTest_DeleteChildren, wxT("Delete &children"));
  236. tree_menu->Append(TreeTest_DeleteAll, wxT("Delete &all items"));
  237. tree_menu->Append(TreeTest_SelectRoot, wxT("Select root item"));
  238. tree_menu->AppendSeparator();
  239. tree_menu->Append(TreeTest_SetFocusedRoot, wxT("Set focus to root item"));
  240. tree_menu->Append(TreeTest_ClearFocused, wxT("Reset focus"));
  241. tree_menu->AppendSeparator();
  242. tree_menu->Append(TreeTest_Count, wxT("Count children of current item"));
  243. tree_menu->Append(TreeTest_CountRec, wxT("Recursively count children of current item"));
  244. tree_menu->AppendSeparator();
  245. tree_menu->Append(TreeTest_Sort, wxT("Sort children of current item"));
  246. tree_menu->Append(TreeTest_SortRev, wxT("Sort in reversed order"));
  247. tree_menu->AppendSeparator();
  248. tree_menu->Append(TreeTest_EnsureVisible, wxT("Make the last item &visible"));
  249. tree_menu->Append(TreeTest_SetFocus, wxT("Set &focus to the tree"));
  250. tree_menu->AppendSeparator();
  251. tree_menu->Append(TreeTest_IncIndent, wxT("Add 5 points to indentation\tAlt-I"));
  252. tree_menu->Append(TreeTest_DecIndent, wxT("Reduce indentation by 5 points\tAlt-R"));
  253. tree_menu->AppendSeparator();
  254. tree_menu->Append(TreeTest_IncSpacing, wxT("Add 5 points to spacing\tCtrl-I"));
  255. tree_menu->Append(TreeTest_DecSpacing, wxT("Reduce spacing by 5 points\tCtrl-R"));
  256. item_menu->Append(TreeTest_Dump, wxT("&Dump item children"));
  257. item_menu->Append(TreeTest_Rename, wxT("&Rename item..."));
  258. item_menu->AppendSeparator();
  259. item_menu->Append(TreeTest_SetBold, wxT("Make item &bold"));
  260. item_menu->Append(TreeTest_ClearBold, wxT("Make item &not bold"));
  261. item_menu->AppendSeparator();
  262. item_menu->Append(TreeTest_ToggleIcon, wxT("Toggle the item's &icon"));
  263. item_menu->Append(TreeTest_ToggleState, wxT("Toggle the item's &state"));
  264. item_menu->AppendSeparator();
  265. item_menu->Append(TreeTest_ShowFirstVisible, wxT("Show &first visible"));
  266. #ifdef wxHAS_LAST_VISIBLE
  267. item_menu->Append(TreeTest_ShowLastVisible, wxT("Show &last visible"));
  268. #endif // wxHAS_LAST_VISIBLE
  269. item_menu->Append(TreeTest_ShowNextVisible, wxT("Show &next visible"));
  270. item_menu->Append(TreeTest_ShowPrevVisible, wxT("Show &previous visible"));
  271. item_menu->AppendSeparator();
  272. item_menu->Append(TreeTest_ShowParent, "Show pa&rent");
  273. item_menu->Append(TreeTest_ShowPrevSibling, "Show &previous sibling");
  274. item_menu->Append(TreeTest_ShowNextSibling, "Show &next sibling");
  275. item_menu->AppendSeparator();
  276. item_menu->Append(TreeTest_ScrollTo, "Scroll &to item",
  277. "Scroll to the last by one top level child");
  278. item_menu->Append(TreeTest_SelectLast, "Select &last item",
  279. "Select the last item");
  280. #ifndef NO_MULTIPLE_SELECTION
  281. item_menu->AppendSeparator();
  282. item_menu->Append(TreeTest_DumpSelected, wxT("Dump selected items\tAlt-D"));
  283. item_menu->Append(TreeTest_Select, wxT("Select current item\tAlt-S"));
  284. item_menu->Append(TreeTest_Unselect, wxT("Unselect everything\tAlt-U"));
  285. item_menu->Append(TreeTest_SelectChildren, wxT("Select all children\tCtrl-A"));
  286. #endif // NO_MULTIPLE_SELECTION
  287. wxMenuBar *menu_bar = new wxMenuBar;
  288. menu_bar->Append(file_menu, wxT("&File"));
  289. menu_bar->Append(style_menu, wxT("&Style"));
  290. menu_bar->Append(tree_menu, wxT("&Tree"));
  291. menu_bar->Append(item_menu, wxT("&Item"));
  292. SetMenuBar(menu_bar);
  293. #endif // wxUSE_MENUS
  294. m_panel = new wxPanel(this);
  295. #if wxUSE_LOG
  296. // create the controls
  297. m_textCtrl = new wxTextCtrl(m_panel, wxID_ANY, wxT(""),
  298. wxDefaultPosition, wxDefaultSize,
  299. wxTE_MULTILINE | wxSUNKEN_BORDER);
  300. #ifdef __WXMOTIF__
  301. // For some reason, we get a memcpy crash in wxLogStream::DoLogStream
  302. // on gcc/wxMotif, if we use wxLogTextCtl. Maybe it's just gcc?
  303. delete wxLog::SetActiveTarget(new wxLogStderr);
  304. #else
  305. // set our text control as the log target
  306. wxLogTextCtrl *logWindow = new wxLogTextCtrl(m_textCtrl);
  307. delete wxLog::SetActiveTarget(logWindow);
  308. #endif
  309. #endif // wxUSE_LOG
  310. CreateTreeWithDefStyle();
  311. menu_bar->Check(TreeTest_ToggleImages, true);
  312. menu_bar->Check(TreeTest_ToggleStates, true);
  313. menu_bar->Check(TreeTest_ToggleAlternateImages, false);
  314. menu_bar->Check(TreeTest_ToggleAlternateStates, false);
  315. #if wxUSE_STATUSBAR
  316. // create a status bar
  317. CreateStatusBar(2);
  318. #endif // wxUSE_STATUSBAR
  319. }
  320. MyFrame::~MyFrame()
  321. {
  322. #if wxUSE_LOG
  323. delete wxLog::SetActiveTarget(NULL);
  324. #endif // wxUSE_LOG
  325. }
  326. void MyFrame::CreateTreeWithDefStyle()
  327. {
  328. long style = wxTR_DEFAULT_STYLE |
  329. #ifndef NO_VARIABLE_HEIGHT
  330. wxTR_HAS_VARIABLE_ROW_HEIGHT |
  331. #endif
  332. wxTR_EDIT_LABELS;
  333. CreateTree(style | wxSUNKEN_BORDER);
  334. // as we don't know what wxTR_DEFAULT_STYLE could contain, test for
  335. // everything
  336. wxMenuBar *mbar = GetMenuBar();
  337. mbar->Check(TreeTest_TogButtons, (style & wxTR_HAS_BUTTONS) != 0);
  338. mbar->Check(TreeTest_TogButtons, (style & wxTR_TWIST_BUTTONS) != 0);
  339. mbar->Check(TreeTest_TogLines, (style & wxTR_NO_LINES) == 0);
  340. mbar->Check(TreeTest_TogRootLines, (style & wxTR_LINES_AT_ROOT) != 0);
  341. mbar->Check(TreeTest_TogHideRoot, (style & wxTR_HIDE_ROOT) != 0);
  342. mbar->Check(TreeTest_TogEdit, (style & wxTR_EDIT_LABELS) != 0);
  343. mbar->Check(TreeTest_TogBorder, (style & wxTR_ROW_LINES) != 0);
  344. mbar->Check(TreeTest_TogFullHighlight, (style & wxTR_FULL_ROW_HIGHLIGHT) != 0);
  345. }
  346. void MyFrame::CreateTree(long style)
  347. {
  348. m_treeCtrl = new MyTreeCtrl(m_panel, TreeTest_Ctrl,
  349. wxDefaultPosition, wxDefaultSize,
  350. style);
  351. GetMenuBar()->Enable(TreeTest_SelectRoot, !(style & wxTR_HIDE_ROOT));
  352. Resize();
  353. }
  354. void MyFrame::TogStyle(int id, long flag)
  355. {
  356. long style = m_treeCtrl->GetWindowStyle() ^ flag;
  357. // most treectrl styles can't be changed on the fly using the native
  358. // control and the tree must be recreated
  359. #ifndef __WXMSW__
  360. m_treeCtrl->SetWindowStyle(style);
  361. #else // MSW
  362. delete m_treeCtrl;
  363. CreateTree(style);
  364. #endif // !MSW/MSW
  365. GetMenuBar()->Check(id, (style & flag) != 0);
  366. }
  367. void MyFrame::OnIdle(wxIdleEvent& event)
  368. {
  369. #if wxUSE_STATUSBAR
  370. if ( m_treeCtrl )
  371. {
  372. wxTreeItemId idRoot = m_treeCtrl->GetRootItem();
  373. wxString status;
  374. if (idRoot.IsOk())
  375. {
  376. wxTreeItemId idLast = m_treeCtrl->GetLastChild(idRoot);
  377. status = wxString::Format(
  378. wxT("Root/last item is %svisible/%svisible"),
  379. m_treeCtrl->IsVisible(idRoot) ? wxT("") : wxT("not "),
  380. idLast.IsOk() && m_treeCtrl->IsVisible(idLast)
  381. ? wxT("") : wxT("not "));
  382. }
  383. else
  384. status = wxT("No root item");
  385. SetStatusText(status, 1);
  386. }
  387. #endif // wxUSE_STATUSBAR
  388. event.Skip();
  389. }
  390. void MyFrame::OnSize(wxSizeEvent& event)
  391. {
  392. if ( m_treeCtrl
  393. #if wxUSE_LOG
  394. && m_textCtrl
  395. #endif // wxUSE_LOG
  396. )
  397. {
  398. Resize();
  399. }
  400. event.Skip();
  401. }
  402. void MyFrame::Resize()
  403. {
  404. wxSize size = GetClientSize();
  405. m_treeCtrl->SetSize(0, 0, size.x, size.y
  406. #if wxUSE_LOG
  407. *2/3);
  408. m_textCtrl->SetSize(0, 2*size.y/3, size.x, size.y/3
  409. #endif
  410. );
  411. }
  412. void MyFrame::OnQuit(wxCommandEvent& WXUNUSED(event))
  413. {
  414. Close(true);
  415. }
  416. void MyFrame::OnAbout(wxCommandEvent& WXUNUSED(event))
  417. {
  418. wxMessageBox(wxT("Tree test sample\n")
  419. wxT("(c) Julian Smart 1997, Vadim Zeitlin 1998"),
  420. wxT("About tree test"),
  421. wxOK | wxICON_INFORMATION, this);
  422. }
  423. void MyFrame::OnClearLog(wxCommandEvent& WXUNUSED(event))
  424. {
  425. m_textCtrl->Clear();
  426. }
  427. void MyFrame::OnRename(wxCommandEvent& WXUNUSED(event))
  428. {
  429. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  430. CHECK_ITEM( item );
  431. // old code - now we edit in place
  432. #if 0
  433. static wxString s_text;
  434. s_text = wxGetTextFromUser(wxT("New name: "), wxT("Tree sample question"),
  435. s_text, this);
  436. if ( !s_text.empty() )
  437. {
  438. m_treeCtrl->SetItemText(item, s_text);
  439. }
  440. #endif // 0
  441. // TODO demonstrate creating a custom edit control...
  442. (void)m_treeCtrl->EditLabel(item);
  443. }
  444. void MyFrame::OnCount(wxCommandEvent& WXUNUSED(event))
  445. {
  446. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  447. CHECK_ITEM( item );
  448. int i = m_treeCtrl->GetChildrenCount( item, false );
  449. wxLogMessage(wxT("%d children"), i);
  450. }
  451. void MyFrame::OnCountRec(wxCommandEvent& WXUNUSED(event))
  452. {
  453. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  454. CHECK_ITEM( item );
  455. int i = m_treeCtrl->GetChildrenCount( item );
  456. wxLogMessage(wxT("%d children"), i);
  457. }
  458. void MyFrame::DoSort(bool reverse)
  459. {
  460. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  461. CHECK_ITEM( item );
  462. m_treeCtrl->DoSortChildren(item, reverse);
  463. }
  464. void MyFrame::OnHighlight(wxCommandEvent& WXUNUSED(event))
  465. {
  466. wxTreeItemId id = m_treeCtrl->GetFocusedItem();
  467. CHECK_ITEM( id );
  468. wxRect r;
  469. if ( !m_treeCtrl->GetBoundingRect(id, r, true /* text, not full row */) )
  470. {
  471. wxLogMessage(wxT("Failed to get bounding item rect"));
  472. return;
  473. }
  474. wxClientDC dc(m_treeCtrl);
  475. dc.SetBrush(*wxRED);
  476. dc.SetPen(*wxTRANSPARENT_PEN);
  477. dc.DrawRectangle(r);
  478. m_treeCtrl->Update();
  479. }
  480. void MyFrame::OnDump(wxCommandEvent& WXUNUSED(event))
  481. {
  482. wxTreeItemId root = m_treeCtrl->GetFocusedItem();
  483. CHECK_ITEM( root );
  484. m_treeCtrl->GetItemsRecursively(root);
  485. }
  486. #ifndef NO_MULTIPLE_SELECTION
  487. void MyFrame::OnToggleSel(wxCommandEvent& event)
  488. {
  489. TogStyle(event.GetId(), wxTR_MULTIPLE);
  490. }
  491. void MyFrame::OnDumpSelected(wxCommandEvent& WXUNUSED(event))
  492. {
  493. wxArrayTreeItemIds array;
  494. size_t count = m_treeCtrl->GetSelections(array);
  495. wxLogMessage(wxT("%u items selected"), unsigned(count));
  496. for ( size_t n = 0; n < count; n++ )
  497. {
  498. wxLogMessage(wxT("\t%s"), m_treeCtrl->GetItemText(array.Item(n)).c_str());
  499. }
  500. }
  501. void MyFrame::OnSelect(wxCommandEvent& WXUNUSED(event))
  502. {
  503. m_treeCtrl->SelectItem(m_treeCtrl->GetFocusedItem());
  504. }
  505. void MyFrame::OnSelectRoot(wxCommandEvent& WXUNUSED(event))
  506. {
  507. if ( !m_treeCtrl->HasFlag(wxTR_HIDE_ROOT) )
  508. m_treeCtrl->SelectItem(m_treeCtrl->GetRootItem());
  509. }
  510. void MyFrame::OnSetFocusedRoot(wxCommandEvent& WXUNUSED(event))
  511. {
  512. if ( !m_treeCtrl->HasFlag(wxTR_HIDE_ROOT) )
  513. m_treeCtrl->SetFocusedItem(m_treeCtrl->GetRootItem());
  514. }
  515. void MyFrame::OnClearFocused(wxCommandEvent& WXUNUSED(event))
  516. {
  517. m_treeCtrl->ClearFocusedItem();
  518. }
  519. void MyFrame::OnUnselect(wxCommandEvent& WXUNUSED(event))
  520. {
  521. m_treeCtrl->UnselectAll();
  522. }
  523. void MyFrame::OnSelectChildren(wxCommandEvent& WXUNUSED(event))
  524. {
  525. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  526. if ( !item.IsOk() )
  527. item = m_treeCtrl->GetRootItem();
  528. m_treeCtrl->SelectChildren(item);
  529. }
  530. #endif // NO_MULTIPLE_SELECTION
  531. void MyFrame::DoSetBold(bool bold)
  532. {
  533. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  534. CHECK_ITEM( item );
  535. m_treeCtrl->SetItemBold(item, bold);
  536. }
  537. void MyFrame::OnDelete(wxCommandEvent& WXUNUSED(event))
  538. {
  539. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  540. CHECK_ITEM( item );
  541. m_treeCtrl->Delete(item);
  542. }
  543. void MyFrame::OnDeleteChildren(wxCommandEvent& WXUNUSED(event))
  544. {
  545. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  546. CHECK_ITEM( item );
  547. m_treeCtrl->DeleteChildren(item);
  548. }
  549. void MyFrame::OnDeleteAll(wxCommandEvent& WXUNUSED(event))
  550. {
  551. m_treeCtrl->DeleteAllItems();
  552. }
  553. void MyFrame::OnRecreate(wxCommandEvent& event)
  554. {
  555. OnDeleteAll(event);
  556. m_treeCtrl->AddTestItemsToTree(NUM_CHILDREN_PER_LEVEL, NUM_LEVELS);
  557. }
  558. void MyFrame::OnSetImageSize(wxCommandEvent& WXUNUSED(event))
  559. {
  560. int size = wxGetNumberFromUser(wxT("Enter the size for the images to use"),
  561. wxT("Size: "),
  562. wxT("TreeCtrl sample"),
  563. m_treeCtrl->ImageSize());
  564. if ( size == -1 )
  565. return;
  566. m_treeCtrl->CreateImageList(size);
  567. wxGetApp().SetShowImages(true);
  568. }
  569. void MyFrame::OnToggleImages(wxCommandEvent& WXUNUSED(event))
  570. {
  571. if ( wxGetApp().ShowImages() )
  572. {
  573. m_treeCtrl->CreateImageList(-1);
  574. wxGetApp().SetShowImages(false);
  575. }
  576. else
  577. {
  578. m_treeCtrl->CreateImageList(0);
  579. wxGetApp().SetShowImages(true);
  580. }
  581. }
  582. void MyFrame::OnToggleStates(wxCommandEvent& WXUNUSED(event))
  583. {
  584. if ( wxGetApp().ShowStates() )
  585. {
  586. m_treeCtrl->CreateStateImageList(true);
  587. wxGetApp().SetShowStates(false);
  588. }
  589. else
  590. {
  591. m_treeCtrl->CreateStateImageList(false);
  592. wxGetApp().SetShowStates(true);
  593. }
  594. }
  595. void MyFrame::OnToggleBell(wxCommandEvent& event)
  596. {
  597. m_treeCtrl->EnableBellOnNoMatch(event.IsChecked());
  598. }
  599. void MyFrame::OnToggleAlternateImages(wxCommandEvent& WXUNUSED(event))
  600. {
  601. bool alternateImages = m_treeCtrl->AlternateImages();
  602. m_treeCtrl->SetAlternateImages(!alternateImages);
  603. m_treeCtrl->CreateImageList(0);
  604. }
  605. void MyFrame::OnToggleAlternateStates(wxCommandEvent& WXUNUSED(event))
  606. {
  607. bool alternateStates = m_treeCtrl->AlternateStates();
  608. m_treeCtrl->SetAlternateStates(!alternateStates);
  609. m_treeCtrl->CreateStateImageList();
  610. // normal states < alternate states
  611. // so we must reset broken states
  612. if ( alternateStates )
  613. m_treeCtrl->ResetBrokenStateImages();
  614. }
  615. void MyFrame::OnToggleButtons(wxCommandEvent& WXUNUSED(event))
  616. {
  617. #if USE_GENERIC_TREECTRL || !defined(__WXMSW__)
  618. if ( wxGetApp().ShowButtons() )
  619. {
  620. m_treeCtrl->CreateButtonsImageList(-1);
  621. wxGetApp().SetShowButtons(false);
  622. }
  623. else
  624. {
  625. m_treeCtrl->CreateButtonsImageList(15);
  626. wxGetApp().SetShowButtons(true);
  627. }
  628. #endif
  629. }
  630. void MyFrame::OnCollapseAndReset(wxCommandEvent& WXUNUSED(event))
  631. {
  632. m_treeCtrl->CollapseAndReset(m_treeCtrl->GetRootItem());
  633. }
  634. void MyFrame::OnEnsureVisible(wxCommandEvent& WXUNUSED(event))
  635. {
  636. m_treeCtrl->DoEnsureVisible();
  637. }
  638. void MyFrame::OnSetFocus(wxCommandEvent& WXUNUSED(event))
  639. {
  640. m_treeCtrl->SetFocus();
  641. }
  642. void MyFrame::OnInsertItem(wxCommandEvent& WXUNUSED(event))
  643. {
  644. int image = wxGetApp().ShowImages() ? MyTreeCtrl::TreeCtrlIcon_File : -1;
  645. m_treeCtrl->InsertItem(m_treeCtrl->GetRootItem(), image, wxT("2nd item"));
  646. }
  647. void MyFrame::OnAddItem(wxCommandEvent& WXUNUSED(event))
  648. {
  649. static int s_num = 0;
  650. wxString text;
  651. text.Printf(wxT("Item #%d"), ++s_num);
  652. m_treeCtrl->AppendItem(m_treeCtrl->GetRootItem(),
  653. text /*,
  654. MyTreeCtrl::TreeCtrlIcon_File */ );
  655. }
  656. void MyFrame::OnAddManyItems(wxCommandEvent& WXUNUSED(event))
  657. {
  658. wxWindowUpdateLocker lockUpdates(this);
  659. const wxTreeItemId root = m_treeCtrl->GetRootItem();
  660. for ( int n = 0; n < 1000; n++ )
  661. {
  662. m_treeCtrl->AppendItem(root, wxString::Format("Item #%03d", n));
  663. }
  664. }
  665. void MyFrame::OnIncIndent(wxCommandEvent& WXUNUSED(event))
  666. {
  667. unsigned int indent = m_treeCtrl->GetIndent();
  668. if (indent < 100)
  669. m_treeCtrl->SetIndent( indent+5 );
  670. }
  671. void MyFrame::OnDecIndent(wxCommandEvent& WXUNUSED(event))
  672. {
  673. unsigned int indent = m_treeCtrl->GetIndent();
  674. if (indent > 10)
  675. m_treeCtrl->SetIndent( indent-5 );
  676. }
  677. void MyFrame::OnIncSpacing(wxCommandEvent& WXUNUSED(event))
  678. {
  679. unsigned int indent = m_treeCtrl->GetSpacing();
  680. if (indent < 100)
  681. {
  682. m_treeCtrl->SetSpacing( indent+5 );
  683. m_treeCtrl->Refresh();
  684. }
  685. }
  686. void MyFrame::OnDecSpacing(wxCommandEvent& WXUNUSED(event))
  687. {
  688. unsigned int indent = m_treeCtrl->GetSpacing();
  689. if (indent > 10)
  690. {
  691. m_treeCtrl->SetSpacing( indent-5 );
  692. m_treeCtrl->Refresh();
  693. }
  694. }
  695. void MyFrame::OnToggleIcon(wxCommandEvent& WXUNUSED(event))
  696. {
  697. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  698. CHECK_ITEM( item );
  699. m_treeCtrl->DoToggleIcon(item);
  700. }
  701. void MyFrame::OnToggleState(wxCommandEvent& WXUNUSED(event))
  702. {
  703. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  704. CHECK_ITEM( item );
  705. m_treeCtrl->DoToggleState(item);
  706. }
  707. void MyFrame::DoShowFirstOrLast(TreeFunc0_t pfn, const wxString& label)
  708. {
  709. const wxTreeItemId item = (m_treeCtrl->*pfn)();
  710. if ( !item.IsOk() )
  711. {
  712. wxLogMessage("There is no %s item", label);
  713. }
  714. else
  715. {
  716. wxLogMessage("The %s item is \"%s\"",
  717. label, m_treeCtrl->GetItemText(item));
  718. }
  719. }
  720. void MyFrame::DoShowRelativeItem(TreeFunc1_t pfn, const wxString& label)
  721. {
  722. wxTreeItemId item = m_treeCtrl->GetFocusedItem();
  723. CHECK_ITEM( item );
  724. if ((pfn == (TreeFunc1_t) &wxTreeCtrl::GetPrevVisible
  725. || pfn == (TreeFunc1_t) &wxTreeCtrl::GetNextVisible)
  726. && !m_treeCtrl->IsVisible(item))
  727. {
  728. wxLogMessage("The selected item must be visible.");
  729. return;
  730. }
  731. wxTreeItemId new_item = (m_treeCtrl->*pfn)(item);
  732. if ( !new_item.IsOk() )
  733. {
  734. wxLogMessage("There is no %s item", label);
  735. }
  736. else
  737. {
  738. wxLogMessage("The %s item is \"%s\"",
  739. label, m_treeCtrl->GetItemText(new_item));
  740. }
  741. }
  742. void MyFrame::OnScrollTo(wxCommandEvent& WXUNUSED(event))
  743. {
  744. // scroll to the last but one top level child
  745. wxTreeItemId item = m_treeCtrl->GetPrevSibling(
  746. m_treeCtrl->GetLastChild(
  747. m_treeCtrl->GetRootItem()));
  748. CHECK_ITEM( item );
  749. m_treeCtrl->ScrollTo(item);
  750. }
  751. void MyFrame::OnSelectLast(wxCommandEvent& WXUNUSED(event))
  752. {
  753. // select the very last item of the tree
  754. wxTreeItemId item = m_treeCtrl->GetRootItem();
  755. for ( ;; )
  756. {
  757. wxTreeItemId itemChild = m_treeCtrl->GetLastChild(item);
  758. if ( !itemChild.IsOk() )
  759. break;
  760. item = itemChild;
  761. }
  762. CHECK_ITEM( item );
  763. m_treeCtrl->SelectItem(item);
  764. }
  765. void MyFrame::OnSetFgColour(wxCommandEvent& WXUNUSED(event))
  766. {
  767. wxColour col = wxGetColourFromUser(this, m_treeCtrl->GetForegroundColour());
  768. if ( col.IsOk() )
  769. m_treeCtrl->SetForegroundColour(col);
  770. }
  771. void MyFrame::OnSetBgColour(wxCommandEvent& WXUNUSED(event))
  772. {
  773. wxColour col = wxGetColourFromUser(this, m_treeCtrl->GetBackgroundColour());
  774. if ( col.IsOk() )
  775. m_treeCtrl->SetBackgroundColour(col);
  776. }
  777. // MyTreeCtrl implementation
  778. #if USE_GENERIC_TREECTRL
  779. IMPLEMENT_DYNAMIC_CLASS(MyTreeCtrl, wxGenericTreeCtrl)
  780. #else
  781. IMPLEMENT_DYNAMIC_CLASS(MyTreeCtrl, wxTreeCtrl)
  782. #endif
  783. MyTreeCtrl::MyTreeCtrl(wxWindow *parent, const wxWindowID id,
  784. const wxPoint& pos, const wxSize& size,
  785. long style)
  786. : wxTreeCtrl(parent, id, pos, size, style),
  787. m_alternateImages(false),
  788. m_alternateStates(false)
  789. {
  790. m_reverseSort = false;
  791. CreateImageList();
  792. CreateStateImageList();
  793. // Add some items to the tree
  794. AddTestItemsToTree(NUM_CHILDREN_PER_LEVEL, NUM_LEVELS);
  795. }
  796. void MyTreeCtrl::CreateImageList(int size)
  797. {
  798. if ( size == -1 )
  799. {
  800. SetImageList(NULL);
  801. return;
  802. }
  803. if ( size == 0 )
  804. size = m_imageSize;
  805. else
  806. m_imageSize = size;
  807. // Make an image list containing small icons
  808. wxImageList *images = new wxImageList(size, size, true);
  809. // should correspond to TreeCtrlIcon_xxx enum
  810. wxBusyCursor wait;
  811. wxIcon icons[5];
  812. if (m_alternateImages)
  813. {
  814. icons[TreeCtrlIcon_File] = wxIcon(icon1_xpm);
  815. icons[TreeCtrlIcon_FileSelected] = wxIcon(icon2_xpm);
  816. icons[TreeCtrlIcon_Folder] = wxIcon(icon3_xpm);
  817. icons[TreeCtrlIcon_FolderSelected] = wxIcon(icon4_xpm);
  818. icons[TreeCtrlIcon_FolderOpened] = wxIcon(icon5_xpm);
  819. }
  820. else
  821. {
  822. wxSize iconSize(size, size);
  823. icons[TreeCtrlIcon_File] =
  824. icons[TreeCtrlIcon_FileSelected] = wxArtProvider::GetIcon(wxART_NORMAL_FILE, wxART_LIST, iconSize);
  825. icons[TreeCtrlIcon_Folder] =
  826. icons[TreeCtrlIcon_FolderSelected] =
  827. icons[TreeCtrlIcon_FolderOpened] = wxArtProvider::GetIcon(wxART_FOLDER, wxART_LIST, iconSize);
  828. }
  829. for ( size_t i = 0; i < WXSIZEOF(icons); i++ )
  830. {
  831. int sizeOrig = icons[0].GetWidth();
  832. if ( size == sizeOrig )
  833. {
  834. images->Add(icons[i]);
  835. }
  836. else
  837. {
  838. images->Add(wxBitmap(wxBitmap(icons[i]).ConvertToImage().Rescale(size, size)));
  839. }
  840. }
  841. AssignImageList(images);
  842. }
  843. void MyTreeCtrl::CreateStateImageList(bool del)
  844. {
  845. if ( del )
  846. {
  847. SetStateImageList(NULL);
  848. return;
  849. }
  850. wxImageList *states;
  851. wxBusyCursor wait;
  852. if (m_alternateStates)
  853. {
  854. wxIcon icons[5];
  855. icons[0] = wxIcon(state1_xpm); // yellow
  856. icons[1] = wxIcon(state2_xpm); // green
  857. icons[2] = wxIcon(state3_xpm); // red
  858. icons[3] = wxIcon(state4_xpm); // blue
  859. icons[4] = wxIcon(state5_xpm); // black
  860. int width = icons[0].GetWidth(),
  861. height = icons[0].GetHeight();
  862. // Make an state image list containing small icons
  863. states = new wxImageList(width, height, true);
  864. for ( size_t i = 0; i < WXSIZEOF(icons); i++ )
  865. states->Add(icons[i]);
  866. }
  867. else
  868. {
  869. wxIcon icons[2];
  870. icons[0] = wxIcon(unchecked_xpm);
  871. icons[1] = wxIcon(checked_xpm);
  872. int width = icons[0].GetWidth(),
  873. height = icons[0].GetHeight();
  874. // Make an state image list containing small icons
  875. states = new wxImageList(width, height, true);
  876. for ( size_t i = 0; i < WXSIZEOF(icons); i++ )
  877. states->Add(icons[i]);
  878. }
  879. AssignStateImageList(states);
  880. }
  881. #if USE_GENERIC_TREECTRL || !defined(__WXMSW__)
  882. void MyTreeCtrl::CreateButtonsImageList(int size)
  883. {
  884. if ( size == -1 )
  885. {
  886. SetButtonsImageList(NULL);
  887. return;
  888. }
  889. // Make an image list containing small icons
  890. wxImageList *images = new wxImageList(size, size, true);
  891. // should correspond to TreeCtrlIcon_xxx enum
  892. wxBusyCursor wait;
  893. wxIcon icons[4];
  894. if (m_alternateImages)
  895. {
  896. icons[0] = wxIcon(icon3_xpm); // closed
  897. icons[1] = wxIcon(icon3_xpm); // closed, selected
  898. icons[2] = wxIcon(icon5_xpm); // open
  899. icons[3] = wxIcon(icon5_xpm); // open, selected
  900. }
  901. else
  902. {
  903. wxSize iconSize(size, size);
  904. icons[0] = // closed
  905. icons[1] = wxArtProvider::GetIcon(wxART_FOLDER, wxART_LIST, iconSize); // closed, selected
  906. icons[2] = // open
  907. icons[3] = wxArtProvider::GetIcon(wxART_FOLDER_OPEN, wxART_LIST, iconSize);// open, selected
  908. }
  909. for ( size_t i = 0; i < WXSIZEOF(icons); i++ )
  910. {
  911. int sizeOrig = icons[i].GetWidth();
  912. if ( size == sizeOrig )
  913. {
  914. images->Add(icons[i]);
  915. }
  916. else
  917. {
  918. images->Add(wxBitmap(wxBitmap(icons[i]).ConvertToImage().Rescale(size, size)));
  919. }
  920. }
  921. AssignButtonsImageList(images);
  922. #else
  923. void MyTreeCtrl::CreateButtonsImageList(int WXUNUSED(size))
  924. {
  925. #endif
  926. }
  927. int MyTreeCtrl::OnCompareItems(const wxTreeItemId& item1,
  928. const wxTreeItemId& item2)
  929. {
  930. if ( m_reverseSort )
  931. {
  932. // just exchange 1st and 2nd items
  933. return wxTreeCtrl::OnCompareItems(item2, item1);
  934. }
  935. else
  936. {
  937. return wxTreeCtrl::OnCompareItems(item1, item2);
  938. }
  939. }
  940. void MyTreeCtrl::AddItemsRecursively(const wxTreeItemId& idParent,
  941. size_t numChildren,
  942. size_t depth,
  943. size_t folder)
  944. {
  945. if ( depth > 0 )
  946. {
  947. bool hasChildren = depth > 1;
  948. wxString str;
  949. for ( size_t n = 0; n < numChildren; n++ )
  950. {
  951. // at depth 1 elements won't have any more children
  952. if ( hasChildren )
  953. str.Printf(wxT("%s child %u"), wxT("Folder"), unsigned(n + 1));
  954. else
  955. str.Printf(wxT("%s child %u.%u"), wxT("File"), unsigned(folder), unsigned(n + 1));
  956. // here we pass to AppendItem() normal and selected item images (we
  957. // suppose that selected image follows the normal one in the enum)
  958. int image, imageSel;
  959. if ( wxGetApp().ShowImages() )
  960. {
  961. image = depth == 1 ? TreeCtrlIcon_File : TreeCtrlIcon_Folder;
  962. imageSel = image + 1;
  963. }
  964. else
  965. {
  966. image = imageSel = -1;
  967. }
  968. wxTreeItemId id = AppendItem(idParent, str, image, imageSel,
  969. new MyTreeItemData(str));
  970. if ( wxGetApp().ShowStates() )
  971. SetItemState(id, 0);
  972. // and now we also set the expanded one (only for the folders)
  973. if ( hasChildren && wxGetApp().ShowImages() )
  974. {
  975. SetItemImage(id, TreeCtrlIcon_FolderOpened,
  976. wxTreeItemIcon_Expanded);
  977. }
  978. // remember the last child for OnEnsureVisible()
  979. if ( !hasChildren && n == numChildren - 1 )
  980. {
  981. m_lastItem = id;
  982. }
  983. AddItemsRecursively(id, numChildren, depth - 1, n + 1);
  984. }
  985. }
  986. //else: done!
  987. }
  988. void MyTreeCtrl::AddTestItemsToTree(size_t numChildren,
  989. size_t depth)
  990. {
  991. int image = wxGetApp().ShowImages() ? MyTreeCtrl::TreeCtrlIcon_Folder : -1;
  992. wxTreeItemId rootId = AddRoot(wxT("Root"),
  993. image, image,
  994. new MyTreeItemData(wxT("Root item")));
  995. if ( !HasFlag(wxTR_HIDE_ROOT) && image != -1 )
  996. {
  997. SetItemImage(rootId, TreeCtrlIcon_FolderOpened, wxTreeItemIcon_Expanded);
  998. }
  999. AddItemsRecursively(rootId, numChildren, depth, 0);
  1000. // set some colours/fonts for testing
  1001. if ( !HasFlag(wxTR_HIDE_ROOT) )
  1002. SetItemFont(rootId, *wxITALIC_FONT);
  1003. wxTreeItemIdValue cookie;
  1004. wxTreeItemId id = GetFirstChild(rootId, cookie);
  1005. SetItemTextColour(id, *wxBLUE);
  1006. id = GetNextChild(rootId, cookie);
  1007. if ( id )
  1008. id = GetNextChild(rootId, cookie);
  1009. if ( id )
  1010. {
  1011. SetItemTextColour(id, *wxRED);
  1012. SetItemBackgroundColour(id, *wxLIGHT_GREY);
  1013. }
  1014. }
  1015. void MyTreeCtrl::GetItemsRecursively(const wxTreeItemId& idParent,
  1016. wxTreeItemIdValue cookie)
  1017. {
  1018. wxTreeItemId id;
  1019. if ( !cookie )
  1020. id = GetFirstChild(idParent, cookie);
  1021. else
  1022. id = GetNextChild(idParent, cookie);
  1023. if ( !id.IsOk() )
  1024. return;
  1025. wxString text = GetItemText(id);
  1026. wxLogMessage(text);
  1027. if (ItemHasChildren(id))
  1028. GetItemsRecursively(id);
  1029. GetItemsRecursively(idParent, cookie);
  1030. }
  1031. void MyTreeCtrl::DoToggleIcon(const wxTreeItemId& item)
  1032. {
  1033. int image = GetItemImage(item) == TreeCtrlIcon_Folder
  1034. ? TreeCtrlIcon_File
  1035. : TreeCtrlIcon_Folder;
  1036. SetItemImage(item, image, wxTreeItemIcon_Normal);
  1037. image = GetItemImage(item, wxTreeItemIcon_Selected) == TreeCtrlIcon_FolderSelected
  1038. ? TreeCtrlIcon_FileSelected
  1039. : TreeCtrlIcon_FolderSelected;
  1040. SetItemImage(item, image, wxTreeItemIcon_Selected);
  1041. }
  1042. void MyTreeCtrl::DoToggleState(const wxTreeItemId& item)
  1043. {
  1044. if ( m_alternateStates )
  1045. {
  1046. // sets random state unlike current
  1047. int state = GetItemState(item);
  1048. int nState;
  1049. srand (time(NULL));
  1050. do {
  1051. nState = rand() % GetStateImageList()->GetImageCount();
  1052. } while (nState == state);
  1053. SetItemState(item, nState);
  1054. }
  1055. else
  1056. {
  1057. // we have only 2 checkbox states, so next state will be reversed
  1058. SetItemState(item, wxTREE_ITEMSTATE_NEXT);
  1059. }
  1060. }
  1061. void MyTreeCtrl::DoResetBrokenStateImages(const wxTreeItemId& idParent,
  1062. wxTreeItemIdValue cookie, int state)
  1063. {
  1064. wxTreeItemId id;
  1065. if ( !cookie )
  1066. id = GetFirstChild(idParent, cookie);
  1067. else
  1068. id = GetNextChild(idParent, cookie);
  1069. if ( !id.IsOk() )
  1070. return;
  1071. int curState = GetItemState(id);
  1072. if ( curState != wxTREE_ITEMSTATE_NONE && curState > state )
  1073. SetItemState(id, state);
  1074. if (ItemHasChildren(id))
  1075. DoResetBrokenStateImages(id, 0, state);
  1076. DoResetBrokenStateImages(idParent, cookie, state);
  1077. }
  1078. void MyTreeCtrl::LogEvent(const wxChar *name, const wxTreeEvent& event)
  1079. {
  1080. wxTreeItemId item = event.GetItem();
  1081. wxString text;
  1082. if ( item.IsOk() )
  1083. text << wxT('"') << GetItemText(item).c_str() << wxT('"');
  1084. else
  1085. text = wxT("invalid item");
  1086. wxLogMessage(wxT("%s(%s)"), name, text.c_str());
  1087. }
  1088. // avoid repetition
  1089. #define TREE_EVENT_HANDLER(name) \
  1090. void MyTreeCtrl::name(wxTreeEvent& event) \
  1091. { \
  1092. LogEvent(wxT(#name), event); \
  1093. SetLastItem(wxTreeItemId()); \
  1094. event.Skip(); \
  1095. }
  1096. TREE_EVENT_HANDLER(OnBeginRDrag)
  1097. TREE_EVENT_HANDLER(OnDeleteItem)
  1098. TREE_EVENT_HANDLER(OnGetInfo)
  1099. TREE_EVENT_HANDLER(OnSetInfo)
  1100. TREE_EVENT_HANDLER(OnItemExpanded)
  1101. TREE_EVENT_HANDLER(OnItemExpanding)
  1102. TREE_EVENT_HANDLER(OnItemCollapsed)
  1103. TREE_EVENT_HANDLER(OnSelChanged)
  1104. TREE_EVENT_HANDLER(OnSelChanging)
  1105. #undef TREE_EVENT_HANDLER
  1106. void LogKeyEvent(const wxChar *name, const wxKeyEvent& event)
  1107. {
  1108. wxString key;
  1109. long keycode = event.GetKeyCode();
  1110. {
  1111. switch ( keycode )
  1112. {
  1113. case WXK_BACK: key = wxT("BACK"); break;
  1114. case WXK_TAB: key = wxT("TAB"); break;
  1115. case WXK_RETURN: key = wxT("RETURN"); break;
  1116. case WXK_ESCAPE: key = wxT("ESCAPE"); break;
  1117. case WXK_SPACE: key = wxT("SPACE"); break;
  1118. case WXK_DELETE: key = wxT("DELETE"); break;
  1119. case WXK_START: key = wxT("START"); break;
  1120. case WXK_LBUTTON: key = wxT("LBUTTON"); break;
  1121. case WXK_RBUTTON: key = wxT("RBUTTON"); break;
  1122. case WXK_CANCEL: key = wxT("CANCEL"); break;
  1123. case WXK_MBUTTON: key = wxT("MBUTTON"); break;
  1124. case WXK_CLEAR: key = wxT("CLEAR"); break;
  1125. case WXK_SHIFT: key = wxT("SHIFT"); break;
  1126. case WXK_ALT: key = wxT("ALT"); break;
  1127. case WXK_CONTROL: key = wxT("CONTROL"); break;
  1128. case WXK_MENU: key = wxT("MENU"); break;
  1129. case WXK_PAUSE: key = wxT("PAUSE"); break;
  1130. case WXK_CAPITAL: key = wxT("CAPITAL"); break;
  1131. case WXK_END: key = wxT("END"); break;
  1132. case WXK_HOME: key = wxT("HOME"); break;
  1133. case WXK_LEFT: key = wxT("LEFT"); break;
  1134. case WXK_UP: key = wxT("UP"); break;
  1135. case WXK_RIGHT: key = wxT("RIGHT"); break;
  1136. case WXK_DOWN: key = wxT("DOWN"); break;
  1137. case WXK_SELECT: key = wxT("SELECT"); break;
  1138. case WXK_PRINT: key = wxT("PRINT"); break;
  1139. case WXK_EXECUTE: key = wxT("EXECUTE"); break;
  1140. case WXK_SNAPSHOT: key = wxT("SNAPSHOT"); break;
  1141. case WXK_INSERT: key = wxT("INSERT"); break;
  1142. case WXK_HELP: key = wxT("HELP"); break;
  1143. case WXK_NUMPAD0: key = wxT("NUMPAD0"); break;
  1144. case WXK_NUMPAD1: key = wxT("NUMPAD1"); break;
  1145. case WXK_NUMPAD2: key = wxT("NUMPAD2"); break;
  1146. case WXK_NUMPAD3: key = wxT("NUMPAD3"); break;
  1147. case WXK_NUMPAD4: key = wxT("NUMPAD4"); break;
  1148. case WXK_NUMPAD5: key = wxT("NUMPAD5"); break;
  1149. case WXK_NUMPAD6: key = wxT("NUMPAD6"); break;
  1150. case WXK_NUMPAD7: key = wxT("NUMPAD7"); break;
  1151. case WXK_NUMPAD8: key = wxT("NUMPAD8"); break;
  1152. case WXK_NUMPAD9: key = wxT("NUMPAD9"); break;
  1153. case WXK_MULTIPLY: key = wxT("MULTIPLY"); break;
  1154. case WXK_ADD: key = wxT("ADD"); break;
  1155. case WXK_SEPARATOR: key = wxT("SEPARATOR"); break;
  1156. case WXK_SUBTRACT: key = wxT("SUBTRACT"); break;
  1157. case WXK_DECIMAL: key = wxT("DECIMAL"); break;
  1158. case WXK_DIVIDE: key = wxT("DIVIDE"); break;
  1159. case WXK_F1: key = wxT("F1"); break;
  1160. case WXK_F2: key = wxT("F2"); break;
  1161. case WXK_F3: key = wxT("F3"); break;
  1162. case WXK_F4: key = wxT("F4"); break;
  1163. case WXK_F5: key = wxT("F5"); break;
  1164. case WXK_F6: key = wxT("F6"); break;
  1165. case WXK_F7: key = wxT("F7"); break;
  1166. case WXK_F8: key = wxT("F8"); break;
  1167. case WXK_F9: key = wxT("F9"); break;
  1168. case WXK_F10: key = wxT("F10"); break;
  1169. case WXK_F11: key = wxT("F11"); break;
  1170. case WXK_F12: key = wxT("F12"); break;
  1171. case WXK_F13: key = wxT("F13"); break;
  1172. case WXK_F14: key = wxT("F14"); break;
  1173. case WXK_F15: key = wxT("F15"); break;
  1174. case WXK_F16: key = wxT("F16"); break;
  1175. case WXK_F17: key = wxT("F17"); break;
  1176. case WXK_F18: key = wxT("F18"); break;
  1177. case WXK_F19: key = wxT("F19"); break;
  1178. case WXK_F20: key = wxT("F20"); break;
  1179. case WXK_F21: key = wxT("F21"); break;
  1180. case WXK_F22: key = wxT("F22"); break;
  1181. case WXK_F23: key = wxT("F23"); break;
  1182. case WXK_F24: key = wxT("F24"); break;
  1183. case WXK_NUMLOCK: key = wxT("NUMLOCK"); break;
  1184. case WXK_SCROLL: key = wxT("SCROLL"); break;
  1185. case WXK_PAGEUP: key = wxT("PAGEUP"); break;
  1186. case WXK_PAGEDOWN: key = wxT("PAGEDOWN"); break;
  1187. case WXK_NUMPAD_SPACE: key = wxT("NUMPAD_SPACE"); break;
  1188. case WXK_NUMPAD_TAB: key = wxT("NUMPAD_TAB"); break;
  1189. case WXK_NUMPAD_ENTER: key = wxT("NUMPAD_ENTER"); break;
  1190. case WXK_NUMPAD_F1: key = wxT("NUMPAD_F1"); break;
  1191. case WXK_NUMPAD_F2: key = wxT("NUMPAD_F2"); break;
  1192. case WXK_NUMPAD_F3: key = wxT("NUMPAD_F3"); break;
  1193. case WXK_NUMPAD_F4: key = wxT("NUMPAD_F4"); break;
  1194. case WXK_NUMPAD_HOME: key = wxT("NUMPAD_HOME"); break;
  1195. case WXK_NUMPAD_LEFT: key = wxT("NUMPAD_LEFT"); break;
  1196. case WXK_NUMPAD_UP: key = wxT("NUMPAD_UP"); break;
  1197. case WXK_NUMPAD_RIGHT: key = wxT("NUMPAD_RIGHT"); break;
  1198. case WXK_NUMPAD_DOWN: key = wxT("NUMPAD_DOWN"); break;
  1199. case WXK_NUMPAD_PAGEUP: key = wxT("NUMPAD_PAGEUP"); break;
  1200. case WXK_NUMPAD_PAGEDOWN: key = wxT("NUMPAD_PAGEDOWN"); break;
  1201. case WXK_NUMPAD_END: key = wxT("NUMPAD_END"); break;
  1202. case WXK_NUMPAD_BEGIN: key = wxT("NUMPAD_BEGIN"); break;
  1203. case WXK_NUMPAD_INSERT: key = wxT("NUMPAD_INSERT"); break;
  1204. case WXK_NUMPAD_DELETE: key = wxT("NUMPAD_DELETE"); break;
  1205. case WXK_NUMPAD_EQUAL: key = wxT("NUMPAD_EQUAL"); break;
  1206. case WXK_NUMPAD_MULTIPLY: key = wxT("NUMPAD_MULTIPLY"); break;
  1207. case WXK_NUMPAD_ADD: key = wxT("NUMPAD_ADD"); break;
  1208. case WXK_NUMPAD_SEPARATOR: key = wxT("NUMPAD_SEPARATOR"); break;
  1209. case WXK_NUMPAD_SUBTRACT: key = wxT("NUMPAD_SUBTRACT"); break;
  1210. case WXK_NUMPAD_DECIMAL: key = wxT("NUMPAD_DECIMAL"); break;
  1211. default:
  1212. {
  1213. if ( keycode < 128 && wxIsprint((int)keycode) )
  1214. key.Printf(wxT("'%c'"), (char)keycode);
  1215. else if ( keycode > 0 && keycode < 27 )
  1216. key.Printf(_("Ctrl-%c"), wxT('A') + keycode - 1);
  1217. else
  1218. key.Printf(wxT("unknown (%ld)"), keycode);
  1219. }
  1220. }
  1221. }
  1222. wxLogMessage( wxT("%s event: %s (flags = %c%c%c%c)"),
  1223. name,
  1224. key.c_str(),
  1225. event.ControlDown() ? wxT('C') : wxT('-'),
  1226. event.AltDown() ? wxT('A') : wxT('-'),
  1227. event.ShiftDown() ? wxT('S') : wxT('-'),
  1228. event.MetaDown() ? wxT('M') : wxT('-'));
  1229. }
  1230. void MyTreeCtrl::OnTreeKeyDown(wxTreeEvent& event)
  1231. {
  1232. LogKeyEvent(wxT("Tree key down "), event.GetKeyEvent());
  1233. event.Skip();
  1234. }
  1235. void MyTreeCtrl::OnBeginDrag(wxTreeEvent& event)
  1236. {
  1237. // need to explicitly allow drag
  1238. if ( event.GetItem() != GetRootItem() )
  1239. {
  1240. m_draggedItem = event.GetItem();
  1241. wxPoint clientpt = event.GetPoint();
  1242. wxPoint screenpt = ClientToScreen(clientpt);
  1243. wxLogMessage(wxT("OnBeginDrag: started dragging %s at screen coords (%i,%i)"),
  1244. GetItemText(m_draggedItem).c_str(),
  1245. screenpt.x, screenpt.y);
  1246. event.Allow();
  1247. }
  1248. else
  1249. {
  1250. wxLogMessage(wxT("OnBeginDrag: this item can't be dragged."));
  1251. }
  1252. }
  1253. void MyTreeCtrl::OnEndDrag(wxTreeEvent& event)
  1254. {
  1255. wxTreeItemId itemSrc = m_draggedItem,
  1256. itemDst = event.GetItem();
  1257. m_draggedItem = (wxTreeItemId)0l;
  1258. // where to copy the item?
  1259. if ( itemDst.IsOk() && !ItemHasChildren(itemDst) )
  1260. {
  1261. // copy to the parent then
  1262. itemDst = GetItemParent(itemDst);
  1263. }
  1264. if ( !itemDst.IsOk() )
  1265. {
  1266. wxLogMessage(wxT("OnEndDrag: can't drop here."));
  1267. return;
  1268. }
  1269. wxString text = GetItemText(itemSrc);
  1270. wxLogMessage(wxT("OnEndDrag: '%s' copied to '%s'."),
  1271. text.c_str(), GetItemText(itemDst).c_str());
  1272. // just do append here - we could also insert it just before/after the item
  1273. // on which it was dropped, but this requires slightly more work... we also
  1274. // completely ignore the client data and icon of the old item but could
  1275. // copy them as well.
  1276. //
  1277. // Finally, we only copy one item here but we might copy the entire tree if
  1278. // we were dragging a folder.
  1279. int image = wxGetApp().ShowImages() ? TreeCtrlIcon_File : -1;
  1280. wxTreeItemId id = AppendItem(itemDst, text, image);
  1281. if ( wxGetApp().ShowStates() )
  1282. SetItemState(id, GetItemState(itemSrc));
  1283. }
  1284. void MyTreeCtrl::OnBeginLabelEdit(wxTreeEvent& event)
  1285. {
  1286. wxLogMessage(wxT("OnBeginLabelEdit"));
  1287. // for testing, prevent this item's label editing
  1288. wxTreeItemId itemId = event.GetItem();
  1289. if ( IsTestItem(itemId) )
  1290. {
  1291. wxMessageBox(wxT("You can't edit this item."));
  1292. event.Veto();
  1293. }
  1294. else if ( itemId == GetRootItem() )
  1295. {
  1296. // test that it is possible to change the text of the item being edited
  1297. SetItemText(itemId, wxT("Editing root item"));
  1298. }
  1299. }
  1300. void MyTreeCtrl::OnEndLabelEdit(wxTreeEvent& event)
  1301. {
  1302. wxLogMessage(wxT("OnEndLabelEdit"));
  1303. // don't allow anything except letters in the labels
  1304. if ( !event.GetLabel().IsWord() )
  1305. {
  1306. wxMessageBox(wxT("The new label should be a single word."));
  1307. event.Veto();
  1308. }
  1309. }
  1310. void MyTreeCtrl::OnItemCollapsing(wxTreeEvent& event)
  1311. {
  1312. wxLogMessage(wxT("OnItemCollapsing"));
  1313. // for testing, prevent the user from collapsing the first child folder
  1314. wxTreeItemId itemId = event.GetItem();
  1315. if ( IsTestItem(itemId) )
  1316. {
  1317. wxMessageBox(wxT("You can't collapse this item."));
  1318. event.Veto();
  1319. }
  1320. }
  1321. void MyTreeCtrl::OnItemActivated(wxTreeEvent& event)
  1322. {
  1323. // show some info about this item
  1324. wxTreeItemId itemId = event.GetItem();
  1325. MyTreeItemData *item = (MyTreeItemData *)GetItemData(itemId);
  1326. if ( item != NULL )
  1327. {
  1328. item->ShowInfo(this);
  1329. }
  1330. wxLogMessage(wxT("OnItemActivated"));
  1331. }
  1332. void MyTreeCtrl::OnItemStateClick(wxTreeEvent& event)
  1333. {
  1334. // toggle item state
  1335. wxTreeItemId itemId = event.GetItem();
  1336. DoToggleState(itemId);
  1337. wxLogMessage(wxT("Item \"%s\" state changed to %d"),
  1338. GetItemText(itemId), GetItemState(itemId));
  1339. }
  1340. void MyTreeCtrl::OnItemMenu(wxTreeEvent& event)
  1341. {
  1342. wxTreeItemId itemId = event.GetItem();
  1343. wxCHECK_RET( itemId.IsOk(), "should have a valid item" );
  1344. MyTreeItemData *item = (MyTreeItemData *)GetItemData(itemId);
  1345. wxPoint clientpt = event.GetPoint();
  1346. wxPoint screenpt = ClientToScreen(clientpt);
  1347. wxLogMessage(wxT("OnItemMenu for item \"%s\" at screen coords (%i, %i)"),
  1348. item->GetDesc(), screenpt.x, screenpt.y);
  1349. ShowMenu(itemId, clientpt);
  1350. event.Skip();
  1351. }
  1352. void MyTreeCtrl::OnContextMenu(wxContextMenuEvent& event)
  1353. {
  1354. wxPoint pt = event.GetPosition();
  1355. wxLogMessage(wxT("OnContextMenu at screen coords (%i, %i)"), pt.x, pt.y);
  1356. event.Skip();
  1357. }
  1358. void MyTreeCtrl::ShowMenu(wxTreeItemId id, const wxPoint& pt)
  1359. {
  1360. wxString title;
  1361. if ( id.IsOk() )
  1362. {
  1363. title << wxT("Menu for ") << GetItemText(id);
  1364. }
  1365. else
  1366. {
  1367. title = wxT("Menu for no particular item");
  1368. }
  1369. #if wxUSE_MENUS
  1370. wxMenu menu(title);
  1371. menu.Append(TreeTest_About, wxT("&About"));
  1372. menu.AppendSeparator();
  1373. menu.Append(TreeTest_Highlight, wxT("&Highlight item"));
  1374. menu.Append(TreeTest_Dump, wxT("&Dump"));
  1375. PopupMenu(&menu, pt);
  1376. #endif // wxUSE_MENUS
  1377. }
  1378. void MyTreeCtrl::OnItemRClick(wxTreeEvent& event)
  1379. {
  1380. wxTreeItemId itemId = event.GetItem();
  1381. wxCHECK_RET( itemId.IsOk(), "should have a valid item" );
  1382. MyTreeItemData *item = (MyTreeItemData *)GetItemData(itemId);
  1383. wxLogMessage(wxT("Item \"%s\" right clicked"), item->GetDesc());
  1384. event.Skip();
  1385. }
  1386. void MyTreeCtrl::OnRMouseDown(wxMouseEvent& event)
  1387. {
  1388. wxLogMessage(wxT("Right mouse button down"));
  1389. event.Skip();
  1390. }
  1391. void MyTreeCtrl::OnRMouseUp(wxMouseEvent& event)
  1392. {
  1393. wxLogMessage(wxT("Right mouse button up"));
  1394. event.Skip();
  1395. }
  1396. void MyTreeCtrl::OnRMouseDClick(wxMouseEvent& event)
  1397. {
  1398. wxTreeItemId id = HitTest(event.GetPosition());
  1399. if ( !id )
  1400. {
  1401. wxLogMessage(wxT("No item under mouse"));
  1402. }
  1403. else
  1404. {
  1405. MyTreeItemData *item = (MyTreeItemData *)GetItemData(id);
  1406. if ( item )
  1407. {
  1408. wxLogMessage(wxT("Item '%s' under mouse"), item->GetDesc());
  1409. }
  1410. }
  1411. event.Skip();
  1412. }
  1413. static inline const wxChar *Bool2String(bool b)
  1414. {
  1415. return b ? wxT("") : wxT("not ");
  1416. }
  1417. void MyTreeItemData::ShowInfo(wxTreeCtrl *tree)
  1418. {
  1419. wxLogMessage(wxT("Item '%s': %sselected, %sexpanded, %sbold,\n")
  1420. wxT("%u children (%u immediately under this item)."),
  1421. m_desc.c_str(),
  1422. Bool2String(tree->IsSelected(GetId())),
  1423. Bool2String(tree->IsExpanded(GetId())),
  1424. Bool2String(tree->IsBold(GetId())),
  1425. unsigned(tree->GetChildrenCount(GetId())),
  1426. unsigned(tree->GetChildrenCount(GetId(), false)));
  1427. }