]>
Commit | Line | Data |
---|---|---|
bbf1f0e5 | 1 | /////////////////////////////////////////////////////////////////////////////// |
23f681ec | 2 | // Name: regtest.cpp |
bbf1f0e5 KB |
3 | // Purpose: wxRegKey class demo |
4 | // Author: Vadim Zeitlin | |
23f681ec | 5 | // Modified by: |
bbf1f0e5 KB |
6 | // Created: 03.04.98 |
7 | // RCS-ID: $Id$ | |
8 | // Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> | |
9 | // Licence: wxWindows license | |
10 | /////////////////////////////////////////////////////////////////////////////// | |
11 | ||
12 | // ============================================================================ | |
13 | // declarations | |
14 | // ============================================================================ | |
15 | ||
16 | // ---------------------------------------------------------------------------- | |
17 | // headers | |
18 | // ---------------------------------------------------------------------------- | |
19 | #include "wx/wxprec.h" | |
20 | ||
21 | #ifdef __BORLANDC__ | |
721b8397 | 22 | # pragma hdrstop |
bbf1f0e5 KB |
23 | #endif |
24 | ||
25 | #ifndef WX_PRECOMP | |
721b8397 | 26 | # include "wx/wx.h" |
bbf1f0e5 KB |
27 | #endif |
28 | ||
bbf1f0e5 | 29 | #include "wx/treectrl.h" |
721b8397 DS |
30 | #include "wx/config.h" |
31 | #include "wx/imaglist.h" | |
c41ea66a VZ |
32 | #include "wx/tokenzr.h" |
33 | ||
721b8397 DS |
34 | #if wxUSE_CONFIG_NATIVE && defined( __WXMSW__ ) |
35 | # define DO_REGTEST 1 | |
36 | #else | |
37 | # define DO_REGTEST 0 | |
38 | #endif | |
39 | ||
bbf1f0e5 KB |
40 | // ---------------------------------------------------------------------------- |
41 | // application type | |
42 | // ---------------------------------------------------------------------------- | |
43 | class RegApp : public wxApp | |
23f681ec | 44 | { |
bbf1f0e5 | 45 | public: |
721b8397 | 46 | bool OnInit(); |
bbf1f0e5 KB |
47 | }; |
48 | ||
49 | // ---------------------------------------------------------------------------- | |
50 | // image list with registry icons | |
51 | // ---------------------------------------------------------------------------- | |
52 | class RegImageList : public wxImageList | |
53 | { | |
54 | public: | |
721b8397 DS |
55 | enum Icon |
56 | { | |
57 | Root, | |
58 | ClosedKey, | |
59 | OpenedKey, | |
60 | TextValue, | |
61 | BinaryValue | |
62 | }; | |
63 | ||
64 | RegImageList(); | |
bbf1f0e5 KB |
65 | }; |
66 | ||
721b8397 DS |
67 | #if DO_REGTEST |
68 | ||
bbf1f0e5 KB |
69 | // ---------------------------------------------------------------------------- |
70 | // our control | |
71 | // ---------------------------------------------------------------------------- | |
72 | class RegTreeCtrl : public wxTreeCtrl | |
73 | { | |
74 | public: | |
721b8397 DS |
75 | // ctor & dtor |
76 | RegTreeCtrl(wxWindow *parent, wxWindowID id); | |
77 | virtual ~RegTreeCtrl(); | |
bbf1f0e5 | 78 | |
721b8397 DS |
79 | // notifications |
80 | void OnDeleteItem (wxTreeEvent& event); | |
81 | void OnItemExpanding (wxTreeEvent& event); | |
82 | void OnSelChanged (wxTreeEvent& event); | |
bbf1f0e5 | 83 | |
721b8397 DS |
84 | void OnBeginEdit (wxTreeEvent& event); |
85 | void OnEndEdit (wxTreeEvent& event); | |
c41ea66a | 86 | |
721b8397 DS |
87 | void OnBeginDrag (wxTreeEvent& event); |
88 | void OnEndDrag (wxTreeEvent& event); | |
23f681ec | 89 | |
721b8397 DS |
90 | void OnRightClick (wxMouseEvent& event); |
91 | void OnChar (wxKeyEvent& event); | |
92 | void OnIdle (wxIdleEvent& event); | |
bbf1f0e5 | 93 | |
721b8397 DS |
94 | // forwarded notifications (by the frame) |
95 | void OnMenuTest(); | |
bbf1f0e5 | 96 | |
721b8397 DS |
97 | // operations |
98 | void GoTo(const wxString& location); | |
99 | void DoRefresh(); | |
100 | void DeleteSelected(); | |
101 | void ShowProperties(); | |
102 | void CreateNewKey(const wxString& strName); | |
103 | void CreateNewTextValue(const wxString& strName); | |
104 | void CreateNewBinaryValue(const wxString& strName); | |
bbf1f0e5 | 105 | |
721b8397 DS |
106 | // information |
107 | bool IsKeySelected() const; | |
bbf1f0e5 | 108 | |
bbf1f0e5 | 109 | private: |
721b8397 DS |
110 | // structure describing a registry key/value |
111 | class TreeNode : public wxTreeItemData | |
112 | { | |
113 | WX_DEFINE_ARRAY_PTR(TreeNode *, TreeChildren); | |
114 | public: | |
115 | RegTreeCtrl *m_pTree; // must be !NULL | |
116 | TreeNode *m_pParent; // NULL only for the root node | |
945d96f6 | 117 | wxTreeItemId m_id; // the id of the tree control item |
721b8397 DS |
118 | wxString m_strName; // name of the key/value |
119 | TreeChildren m_aChildren; // array of subkeys/values | |
120 | bool m_bKey; // key or value? | |
121 | wxRegKey *m_pKey; // only may be !NULL if m_bKey == true | |
23f681ec | 122 | |
721b8397 | 123 | // trivial accessors |
945d96f6 WS |
124 | wxTreeItemId Id() const { return m_id; } |
125 | bool IsRoot() const { return m_pParent == NULL; } | |
126 | bool IsKey() const { return m_bKey; } | |
127 | TreeNode *Parent() const { return m_pParent; } | |
23f681ec | 128 | |
721b8397 DS |
129 | // notifications |
130 | bool OnExpand(); | |
131 | void OnCollapse(); | |
23f681ec | 132 | |
721b8397 DS |
133 | // operations |
134 | void Refresh(); | |
135 | bool DeleteChild(TreeNode *child); | |
136 | void DestroyChildren(); | |
137 | const wxChar *FullName() const; | |
23f681ec | 138 | |
721b8397 DS |
139 | // get the associated key: make sure the pointer is !NULL |
140 | wxRegKey& Key() { if ( !m_pKey ) OnExpand(); return *m_pKey; } | |
23f681ec | 141 | |
721b8397 DS |
142 | // dtor deletes all children |
143 | ~TreeNode(); | |
144 | }; | |
bbf1f0e5 | 145 | |
721b8397 DS |
146 | wxImageList *m_imageList; |
147 | wxMenu *m_pMenuPopup; | |
23f681ec | 148 | |
721b8397 | 149 | TreeNode *m_pRoot; |
23f681ec | 150 | |
721b8397 DS |
151 | TreeNode *m_draggedItem; // the item being dragged |
152 | bool m_copyOnDrop; // if false, then move | |
23f681ec | 153 | |
721b8397 | 154 | bool m_restoreStatus; // after OnItemExpanding() |
bbf1f0e5 | 155 | |
721b8397 | 156 | wxString m_nameOld; // the initial value of item being renamed |
c41ea66a | 157 | |
721b8397 | 158 | TreeNode *GetNode(const wxTreeEvent& event) |
2e3b8fa7 | 159 | { return (TreeNode *)GetItemData(event.GetItem()); } |
bbf1f0e5 KB |
160 | |
161 | public: | |
721b8397 DS |
162 | // create a new node and insert it to the tree |
163 | TreeNode *InsertNewTreeNode(TreeNode *pParent, | |
164 | const wxString& strName, | |
165 | int idImage = RegImageList::ClosedKey, | |
166 | const wxString *pstrValue = NULL); | |
167 | ||
168 | // add standard registry keys | |
169 | void AddStdKeys(); | |
23f681ec VZ |
170 | |
171 | private: | |
721b8397 | 172 | DECLARE_EVENT_TABLE() |
bbf1f0e5 KB |
173 | }; |
174 | ||
721b8397 DS |
175 | #endif // #if DO_REGTEST |
176 | ||
bbf1f0e5 KB |
177 | // ---------------------------------------------------------------------------- |
178 | // the main window of our application | |
179 | // ---------------------------------------------------------------------------- | |
180 | class RegFrame : public wxFrame | |
23f681ec | 181 | { |
bbf1f0e5 | 182 | public: |
721b8397 | 183 | // ctor & dtor |
fbfb8bcc | 184 | RegFrame(wxFrame *parent, const wxChar *title, int x, int y, int w, int h); |
721b8397 | 185 | virtual ~RegFrame(); |
23f681ec | 186 | |
721b8397 DS |
187 | // callbacks |
188 | void OnQuit (wxCommandEvent& event); | |
189 | void OnAbout(wxCommandEvent& event); | |
190 | void OnTest (wxCommandEvent& event); | |
bbf1f0e5 | 191 | |
721b8397 | 192 | void OnGoTo (wxCommandEvent& event); |
c41ea66a | 193 | |
721b8397 DS |
194 | void OnExpand (wxCommandEvent& event); |
195 | void OnCollapse(wxCommandEvent& event); | |
196 | void OnToggle (wxCommandEvent& event); | |
197 | void OnRefresh (wxCommandEvent& event); | |
bbf1f0e5 | 198 | |
721b8397 DS |
199 | void OnDelete (wxCommandEvent& event); |
200 | void OnNewKey (wxCommandEvent& event); | |
201 | void OnNewText (wxCommandEvent& event); | |
202 | void OnNewBinary(wxCommandEvent& event); | |
bbf1f0e5 | 203 | |
721b8397 | 204 | void OnInfo (wxCommandEvent& event); |
bbf1f0e5 | 205 | |
721b8397 | 206 | DECLARE_EVENT_TABLE() |
bbf1f0e5 KB |
207 | |
208 | private: | |
721b8397 DS |
209 | |
210 | #if DO_REGTEST | |
211 | RegTreeCtrl *m_treeCtrl; | |
212 | #endif | |
bbf1f0e5 KB |
213 | }; |
214 | ||
215 | // ---------------------------------------------------------------------------- | |
216 | // various ids | |
217 | // ---------------------------------------------------------------------------- | |
218 | ||
219 | enum | |
220 | { | |
721b8397 DS |
221 | Menu_Quit = 100, |
222 | Menu_About, | |
223 | Menu_Test, | |
224 | Menu_GoTo, | |
225 | Menu_Expand, | |
226 | Menu_Collapse, | |
227 | Menu_Toggle, | |
228 | Menu_Refresh, | |
229 | Menu_New, | |
230 | Menu_NewKey, | |
231 | Menu_NewText, | |
232 | Menu_NewBinary, | |
233 | Menu_Delete, | |
234 | Menu_Info, | |
235 | ||
236 | Ctrl_RegTree = 200, | |
bbf1f0e5 KB |
237 | }; |
238 | ||
239 | // ---------------------------------------------------------------------------- | |
240 | // event tables | |
241 | // ---------------------------------------------------------------------------- | |
242 | ||
243 | BEGIN_EVENT_TABLE(RegFrame, wxFrame) | |
721b8397 DS |
244 | EVT_MENU(Menu_Test, RegFrame::OnTest) |
245 | EVT_MENU(Menu_About, RegFrame::OnAbout) | |
246 | EVT_MENU(Menu_Quit, RegFrame::OnQuit) | |
247 | EVT_MENU(Menu_GoTo, RegFrame::OnGoTo) | |
248 | EVT_MENU(Menu_Expand, RegFrame::OnExpand) | |
249 | EVT_MENU(Menu_Collapse, RegFrame::OnCollapse) | |
250 | EVT_MENU(Menu_Toggle, RegFrame::OnToggle) | |
251 | EVT_MENU(Menu_Refresh, RegFrame::OnRefresh) | |
252 | EVT_MENU(Menu_Delete, RegFrame::OnDelete) | |
253 | EVT_MENU(Menu_NewKey, RegFrame::OnNewKey) | |
254 | EVT_MENU(Menu_NewText, RegFrame::OnNewText) | |
255 | EVT_MENU(Menu_NewBinary,RegFrame::OnNewBinary) | |
256 | EVT_MENU(Menu_Info, RegFrame::OnInfo) | |
bbf1f0e5 KB |
257 | END_EVENT_TABLE() |
258 | ||
721b8397 DS |
259 | #if DO_REGTEST |
260 | ||
bbf1f0e5 | 261 | BEGIN_EVENT_TABLE(RegTreeCtrl, wxTreeCtrl) |
721b8397 DS |
262 | EVT_TREE_DELETE_ITEM (Ctrl_RegTree, RegTreeCtrl::OnDeleteItem) |
263 | EVT_TREE_ITEM_EXPANDING(Ctrl_RegTree, RegTreeCtrl::OnItemExpanding) | |
264 | EVT_TREE_SEL_CHANGED (Ctrl_RegTree, RegTreeCtrl::OnSelChanged) | |
c41ea66a | 265 | |
721b8397 DS |
266 | EVT_TREE_BEGIN_LABEL_EDIT(Ctrl_RegTree, RegTreeCtrl::OnBeginEdit) |
267 | EVT_TREE_END_LABEL_EDIT (Ctrl_RegTree, RegTreeCtrl::OnEndEdit) | |
c41ea66a | 268 | |
721b8397 DS |
269 | EVT_TREE_BEGIN_DRAG (Ctrl_RegTree, RegTreeCtrl::OnBeginDrag) |
270 | EVT_TREE_BEGIN_RDRAG (Ctrl_RegTree, RegTreeCtrl::OnBeginDrag) | |
271 | EVT_TREE_END_DRAG (Ctrl_RegTree, RegTreeCtrl::OnEndDrag) | |
bbf1f0e5 | 272 | |
721b8397 DS |
273 | EVT_CHAR (RegTreeCtrl::OnChar) |
274 | EVT_RIGHT_DOWN(RegTreeCtrl::OnRightClick) | |
275 | EVT_IDLE (RegTreeCtrl::OnIdle) | |
bbf1f0e5 KB |
276 | END_EVENT_TABLE() |
277 | ||
721b8397 DS |
278 | #endif |
279 | ||
bbf1f0e5 KB |
280 | // ============================================================================ |
281 | // implementation | |
282 | // ============================================================================ | |
283 | ||
284 | // ---------------------------------------------------------------------------- | |
285 | // global functions | |
286 | // ---------------------------------------------------------------------------- | |
287 | ||
288 | // create the "registry operations" menu | |
289 | wxMenu *CreateRegistryMenu() | |
290 | { | |
721b8397 DS |
291 | wxMenu *pMenuNew = new wxMenu; |
292 | pMenuNew->Append(Menu_NewKey, _T("&Key"), _T("Create a new key")); | |
293 | pMenuNew->AppendSeparator(); | |
294 | pMenuNew->Append(Menu_NewText, _T("&Text value"), _T("Create a new text value")); | |
295 | pMenuNew->Append(Menu_NewBinary, _T("&Binary value"), _T("Create a new binary value")); | |
296 | ||
297 | wxMenu *pMenuReg = new wxMenu; | |
298 | pMenuReg->Append(Menu_New, _T("&New"), pMenuNew); | |
299 | pMenuReg->Append(Menu_Delete, _T("&Delete..."), _T("Delete selected key/value")); | |
300 | pMenuReg->AppendSeparator(); | |
301 | pMenuReg->Append(Menu_GoTo, _T("&Go to...\tCtrl-G"), _T("Go to registry key")); | |
302 | pMenuReg->Append(Menu_Expand, _T("&Expand"), _T("Expand current key")); | |
303 | pMenuReg->Append(Menu_Collapse, _T("&Collapse"), _T("Collapse current key")); | |
304 | pMenuReg->Append(Menu_Toggle, _T("&Toggle"), _T("Toggle current key")); | |
305 | pMenuReg->AppendSeparator(); | |
306 | pMenuReg->Append(Menu_Refresh, _T("&Refresh"), _T("Refresh the subtree")); | |
307 | pMenuReg->AppendSeparator(); | |
308 | pMenuReg->Append(Menu_Info, _T("&Properties"),_T("Information about current selection")); | |
309 | ||
310 | return pMenuReg; | |
bbf1f0e5 KB |
311 | } |
312 | ||
313 | // ---------------------------------------------------------------------------- | |
314 | // application class | |
315 | // ---------------------------------------------------------------------------- | |
316 | IMPLEMENT_APP(RegApp) | |
317 | ||
318 | // `Main program' equivalent, creating windows and returning main app frame | |
319 | bool RegApp::OnInit() | |
320 | { | |
721b8397 DS |
321 | // create the main frame window and show it |
322 | RegFrame *frame = new RegFrame(NULL, _T("wxRegTest"), 50, 50, 600, 350); | |
323 | frame->Show(true); | |
23f681ec | 324 | |
721b8397 | 325 | SetTopWindow(frame); |
bbf1f0e5 | 326 | |
721b8397 | 327 | return true; |
bbf1f0e5 KB |
328 | } |
329 | ||
330 | // ---------------------------------------------------------------------------- | |
331 | // RegFrame | |
332 | // ---------------------------------------------------------------------------- | |
333 | ||
fbfb8bcc | 334 | RegFrame::RegFrame(wxFrame *parent, const wxChar *title, int x, int y, int w, int h) |
945d96f6 | 335 | : wxFrame(parent, wxID_ANY, title, wxPoint(x, y), wxSize(w, h)) |
bbf1f0e5 | 336 | { |
721b8397 DS |
337 | // this reduces flicker effects |
338 | SetBackgroundColour(wxColour(255, 255, 255)); | |
339 | ||
340 | // set the icon | |
341 | // ------------ | |
342 | SetIcon(wxIcon(_T("app_icon"))); | |
343 | ||
344 | // create menu | |
345 | // ----------- | |
346 | wxMenu *pMenuFile = new wxMenu; | |
347 | pMenuFile->Append(Menu_Test, _T("Te&st"), _T("Test key creation")); | |
348 | pMenuFile->AppendSeparator(); | |
349 | pMenuFile->Append(Menu_About, _T("&About..."), _T("Show an extraordinarly beautiful dialog")); | |
350 | pMenuFile->AppendSeparator(); | |
351 | pMenuFile->Append(Menu_Quit, _T("E&xit"), _T("Quit this program")); | |
352 | ||
353 | wxMenuBar *pMenu = new wxMenuBar; | |
354 | pMenu->Append(pMenuFile, _T("&File")); | |
355 | pMenu->Append(CreateRegistryMenu(), _T("&Registry")); | |
356 | SetMenuBar(pMenu); | |
357 | ||
358 | #if DO_REGTEST | |
359 | // create child controls | |
360 | // --------------------- | |
361 | m_treeCtrl = new RegTreeCtrl(this, Ctrl_RegTree); | |
362 | #endif | |
363 | ||
8520f137 | 364 | #if wxUSE_STATUSBAR |
721b8397 DS |
365 | // create the status line |
366 | // ---------------------- | |
367 | CreateStatusBar(2); | |
8520f137 | 368 | #endif // wxUSE_STATUSBAR |
bbf1f0e5 KB |
369 | } |
370 | ||
23f681ec | 371 | RegFrame::~RegFrame() |
bbf1f0e5 | 372 | { |
721b8397 DS |
373 | #if DO_REGTEST |
374 | // this makes deletion of it *much* quicker | |
375 | m_treeCtrl->Hide(); | |
376 | #endif | |
bbf1f0e5 KB |
377 | } |
378 | ||
721b8397 | 379 | void RegFrame::OnQuit(wxCommandEvent& WXUNUSED(event)) |
bbf1f0e5 | 380 | { |
721b8397 | 381 | Close(true); |
bbf1f0e5 KB |
382 | } |
383 | ||
721b8397 | 384 | void RegFrame::OnAbout(wxCommandEvent& WXUNUSED(event)) |
bbf1f0e5 | 385 | { |
721b8397 DS |
386 | wxMessageDialog dialog(this, |
387 | _T("wxRegistry sample\n") | |
749bfe9a | 388 | _T("(c) 1998, 2000 Vadim Zeitlin"), |
721b8397 | 389 | _T("About wxRegTest"), wxOK); |
bbf1f0e5 | 390 | |
721b8397 | 391 | dialog.ShowModal(); |
bbf1f0e5 KB |
392 | } |
393 | ||
c41ea66a | 394 | void RegFrame::OnTest(wxCommandEvent& WXUNUSED(event)) |
bbf1f0e5 | 395 | { |
721b8397 DS |
396 | #if DO_REGTEST |
397 | m_treeCtrl->OnMenuTest(); | |
398 | #endif | |
bbf1f0e5 KB |
399 | } |
400 | ||
c41ea66a VZ |
401 | void RegFrame::OnGoTo(wxCommandEvent& WXUNUSED(event)) |
402 | { | |
be5a51fb | 403 | static wxString s_location = _T("HKEY_CURRENT_USER\\Software\\wxWidgets"); |
c41ea66a | 404 | |
721b8397 DS |
405 | wxString location = wxGetTextFromUser( |
406 | _T("Enter the location to go to:"), | |
407 | _T("wxRegTest question"), | |
408 | s_location, | |
409 | this); | |
410 | ||
c41ea66a VZ |
411 | if ( !location ) |
412 | return; | |
413 | ||
414 | s_location = location; | |
721b8397 | 415 | #if DO_REGTEST |
c41ea66a | 416 | m_treeCtrl->GoTo(location); |
721b8397 | 417 | #endif |
c41ea66a VZ |
418 | } |
419 | ||
420 | void RegFrame::OnExpand(wxCommandEvent& WXUNUSED(event)) | |
bbf1f0e5 | 421 | { |
721b8397 DS |
422 | #if DO_REGTEST |
423 | m_treeCtrl->Expand(m_treeCtrl->GetSelection()); | |
424 | #endif | |
bbf1f0e5 KB |
425 | } |
426 | ||
c41ea66a | 427 | void RegFrame::OnCollapse(wxCommandEvent& WXUNUSED(event)) |
bbf1f0e5 | 428 | { |
721b8397 DS |
429 | #if DO_REGTEST |
430 | m_treeCtrl->Collapse(m_treeCtrl->GetSelection()); | |
431 | #endif | |
bbf1f0e5 KB |
432 | } |
433 | ||
c41ea66a | 434 | void RegFrame::OnToggle(wxCommandEvent& WXUNUSED(event)) |
bbf1f0e5 | 435 | { |
721b8397 DS |
436 | #if DO_REGTEST |
437 | m_treeCtrl->Toggle(m_treeCtrl->GetSelection()); | |
438 | #endif | |
bbf1f0e5 KB |
439 | } |
440 | ||
c41ea66a | 441 | void RegFrame::OnRefresh(wxCommandEvent& WXUNUSED(event)) |
5888ef1e | 442 | { |
721b8397 DS |
443 | #if DO_REGTEST |
444 | m_treeCtrl->DoRefresh(); | |
445 | #endif | |
5888ef1e VZ |
446 | } |
447 | ||
c41ea66a | 448 | void RegFrame::OnDelete(wxCommandEvent& WXUNUSED(event)) |
bbf1f0e5 | 449 | { |
721b8397 DS |
450 | #if DO_REGTEST |
451 | m_treeCtrl->DeleteSelected(); | |
452 | #endif | |
bbf1f0e5 KB |
453 | } |
454 | ||
c41ea66a | 455 | void RegFrame::OnNewKey(wxCommandEvent& WXUNUSED(event)) |
bbf1f0e5 | 456 | { |
721b8397 DS |
457 | #if DO_REGTEST |
458 | if ( m_treeCtrl->IsKeySelected() ) | |
459 | { | |
460 | m_treeCtrl->CreateNewKey( | |
461 | wxGetTextFromUser(_T("Enter the name of the new key"))); | |
462 | } | |
463 | #endif | |
bbf1f0e5 KB |
464 | } |
465 | ||
c41ea66a | 466 | void RegFrame::OnNewText(wxCommandEvent& WXUNUSED(event)) |
bbf1f0e5 | 467 | { |
721b8397 DS |
468 | #if DO_REGTEST |
469 | if ( m_treeCtrl->IsKeySelected() ) | |
470 | { | |
471 | m_treeCtrl->CreateNewTextValue( | |
472 | wxGetTextFromUser(_T("Enter the name for the new text value"))); | |
473 | } | |
474 | #endif | |
bbf1f0e5 KB |
475 | } |
476 | ||
c41ea66a | 477 | void RegFrame::OnNewBinary(wxCommandEvent& WXUNUSED(event)) |
bbf1f0e5 | 478 | { |
721b8397 DS |
479 | #if DO_REGTEST |
480 | if ( m_treeCtrl->IsKeySelected() ) | |
481 | { | |
482 | m_treeCtrl->CreateNewBinaryValue( | |
483 | wxGetTextFromUser(_T("Enter the name for the new binary value"))); | |
484 | } | |
485 | #endif | |
bbf1f0e5 KB |
486 | } |
487 | ||
c41ea66a | 488 | void RegFrame::OnInfo(wxCommandEvent& WXUNUSED(event)) |
23f681ec | 489 | { |
721b8397 | 490 | #if DO_REGTEST |
23f681ec | 491 | m_treeCtrl->ShowProperties(); |
721b8397 | 492 | #endif |
23f681ec VZ |
493 | } |
494 | ||
bbf1f0e5 KB |
495 | // ---------------------------------------------------------------------------- |
496 | // RegImageList | |
497 | // ---------------------------------------------------------------------------- | |
721b8397 | 498 | RegImageList::RegImageList() : wxImageList(16, 16, true) |
bbf1f0e5 | 499 | { |
721b8397 DS |
500 | // should be in sync with enum RegImageList::RegIcon |
501 | static const wxChar *aszIcons[] = { _T("key1"),_T("key2"),_T("key3"),_T("value1"),_T("value2") }; | |
502 | wxString str = _T("icon_"); | |
503 | for ( unsigned int n = 0; n < WXSIZEOF(aszIcons); n++ ) | |
504 | { | |
505 | Add(wxIcon(str + aszIcons[n], wxBITMAP_TYPE_ICO_RESOURCE)); | |
506 | } | |
bbf1f0e5 KB |
507 | } |
508 | ||
721b8397 DS |
509 | #if DO_REGTEST |
510 | ||
bbf1f0e5 KB |
511 | // ---------------------------------------------------------------------------- |
512 | // RegTreeCtrl | |
513 | // ---------------------------------------------------------------------------- | |
514 | ||
515 | // create a new tree item and insert it into the tree | |
23f681ec | 516 | RegTreeCtrl::TreeNode *RegTreeCtrl::InsertNewTreeNode(TreeNode *pParent, |
bbf1f0e5 KB |
517 | const wxString& strName, |
518 | int idImage, | |
519 | const wxString *pstrValue) | |
520 | { | |
721b8397 DS |
521 | // create new item & insert it |
522 | TreeNode *pNewNode = new TreeNode; | |
523 | pNewNode->m_pTree = this; | |
524 | pNewNode->m_pParent = pParent; | |
525 | pNewNode->m_strName = strName; | |
526 | pNewNode->m_bKey = pstrValue == NULL; | |
527 | pNewNode->m_pKey = NULL; | |
528 | if (pParent) | |
529 | { | |
530 | pNewNode->m_id = AppendItem(pParent->Id(), | |
531 | pNewNode->IsKey() ? strName : *pstrValue, | |
532 | idImage); | |
533 | } | |
534 | else | |
535 | { | |
536 | pNewNode->m_id = AddRoot(strName); | |
537 | } | |
538 | ||
539 | wxASSERT_MSG( pNewNode->m_id, wxT("can't create tree control item!")); | |
540 | ||
541 | // save the pointer in the item | |
542 | SetItemData(pNewNode->m_id, pNewNode); | |
543 | ||
544 | // add it to the list of parent's children | |
545 | if ( pParent != NULL ) | |
546 | { | |
547 | pParent->m_aChildren.Add(pNewNode); | |
548 | } | |
549 | ||
550 | if ( pNewNode->IsKey() ) | |
551 | { | |
552 | SetItemHasChildren(pNewNode->Id()); | |
553 | ||
554 | if ( !pNewNode->IsRoot() ) | |
555 | { | |
556 | // set the expanded icon as well | |
557 | SetItemImage(pNewNode->Id(), | |
558 | RegImageList::OpenedKey, | |
559 | wxTreeItemIcon_Expanded); | |
560 | } | |
23f681ec | 561 | } |
bbf1f0e5 | 562 | |
721b8397 | 563 | return pNewNode; |
bbf1f0e5 KB |
564 | } |
565 | ||
566 | RegTreeCtrl::RegTreeCtrl(wxWindow *parent, wxWindowID id) | |
721b8397 DS |
567 | : wxTreeCtrl(parent, id, wxDefaultPosition, wxDefaultSize, |
568 | wxTR_HAS_BUTTONS | wxTR_EDIT_LABELS | wxSUNKEN_BORDER) | |
bbf1f0e5 | 569 | { |
721b8397 DS |
570 | // init members |
571 | m_draggedItem = NULL; | |
572 | m_restoreStatus = false; | |
573 | ||
574 | // create the image list | |
575 | // --------------------- | |
576 | m_imageList = new RegImageList; | |
577 | SetImageList(m_imageList); | |
578 | ||
579 | // create root keys | |
580 | // ---------------- | |
581 | m_pRoot = InsertNewTreeNode(NULL, _T("Registry Root"), RegImageList::Root); | |
582 | ||
583 | // create popup menu | |
584 | // ----------------- | |
585 | m_pMenuPopup = CreateRegistryMenu(); | |
bbf1f0e5 KB |
586 | } |
587 | ||
588 | RegTreeCtrl::~RegTreeCtrl() | |
589 | { | |
721b8397 DS |
590 | delete m_pMenuPopup; |
591 | // delete m_pRoot; -- this is done by the tree now | |
592 | delete m_imageList; | |
bbf1f0e5 KB |
593 | } |
594 | ||
595 | void RegTreeCtrl::AddStdKeys() | |
596 | { | |
721b8397 DS |
597 | for ( unsigned int ui = 0; ui < wxRegKey::nStdKeys; ui++ ) |
598 | { | |
599 | InsertNewTreeNode(m_pRoot, wxRegKey::GetStdKeyName(ui)); | |
600 | } | |
bbf1f0e5 KB |
601 | } |
602 | ||
603 | // ---------------------------------------------------------------------------- | |
604 | // notifications | |
605 | // ---------------------------------------------------------------------------- | |
606 | ||
23f681ec VZ |
607 | void RegTreeCtrl::OnIdle(wxIdleEvent& WXUNUSED(event)) |
608 | { | |
721b8397 DS |
609 | if ( m_restoreStatus ) |
610 | { | |
23f681ec | 611 | // restore it after OnItemExpanding() |
a60b1f5d | 612 | wxLogStatus(wxT("Ok")); |
23f681ec VZ |
613 | wxSetCursor(*wxSTANDARD_CURSOR); |
614 | ||
721b8397 | 615 | m_restoreStatus = false; |
23f681ec VZ |
616 | } |
617 | } | |
618 | ||
bbf1f0e5 KB |
619 | void RegTreeCtrl::OnRightClick(wxMouseEvent& event) |
620 | { | |
721b8397 | 621 | int iFlags; |
945d96f6 | 622 | wxTreeItemId lId = HitTest(wxPoint(event.GetX(), event.GetY()), iFlags); |
721b8397 DS |
623 | if ( iFlags & wxTREE_HITTEST_ONITEMLABEL ) |
624 | { | |
625 | // select the item first | |
626 | SelectItem(lId); | |
627 | } | |
628 | //else: take the currently selected item if click not on item | |
629 | ||
630 | PopupMenu(m_pMenuPopup, event.GetX(), event.GetY()); | |
bbf1f0e5 KB |
631 | } |
632 | ||
633 | ||
721b8397 | 634 | void RegTreeCtrl::OnDeleteItem(wxTreeEvent& WXUNUSED(event)) |
bbf1f0e5 KB |
635 | { |
636 | } | |
637 | ||
638 | // test the key creation functions | |
639 | void RegTreeCtrl::OnMenuTest() | |
640 | { | |
945d96f6 | 641 | wxTreeItemId lId = GetSelection(); |
721b8397 DS |
642 | TreeNode *pNode = (TreeNode *)GetItemData(lId); |
643 | ||
644 | wxCHECK_RET( pNode != NULL, wxT("tree item without data?") ); | |
645 | ||
646 | if ( pNode->IsRoot() ) | |
647 | { | |
648 | wxLogError(wxT("Can't create a subkey under the root key.")); | |
649 | return; | |
650 | } | |
651 | ||
652 | if ( !pNode->IsKey() ) | |
653 | { | |
654 | wxLogError(wxT("Can't create a subkey under a value!")); | |
655 | return; | |
656 | } | |
657 | ||
658 | wxRegKey key1(pNode->Key(), _T("key1")); | |
659 | if ( key1.Create() ) | |
660 | { | |
661 | wxRegKey key2a(key1, _T("key2a")), key2b(key1, _T("key2b")); | |
662 | if ( key2a.Create() && key2b.Create() ) | |
663 | { | |
664 | // put some values under the newly created keys | |
665 | key1.SetValue(wxT("first_term"), _T("10")); | |
666 | key1.SetValue(wxT("second_term"), _T("7")); | |
667 | key2a = _T("this is the unnamed value"); | |
668 | key2b.SetValue(wxT("sum"), 17); | |
669 | ||
670 | // refresh tree | |
671 | pNode->Refresh(); | |
672 | wxLogStatus(wxT("Test keys successfully added.")); | |
673 | return; | |
674 | } | |
bbf1f0e5 | 675 | } |
23f681ec | 676 | |
721b8397 | 677 | wxLogError(wxT("Creation of test keys failed.")); |
bbf1f0e5 KB |
678 | } |
679 | ||
680 | void RegTreeCtrl::OnChar(wxKeyEvent& event) | |
681 | { | |
721b8397 DS |
682 | switch ( event.GetKeyCode() ) |
683 | { | |
684 | case WXK_DELETE: | |
685 | DeleteSelected(); | |
686 | return; | |
687 | ||
688 | case WXK_RETURN: | |
689 | if ( event.AltDown() ) | |
690 | { | |
691 | ShowProperties(); | |
692 | ||
693 | return; | |
694 | } | |
695 | } | |
696 | ||
697 | event.Skip(); | |
bbf1f0e5 KB |
698 | } |
699 | ||
700 | void RegTreeCtrl::OnSelChanged(wxTreeEvent& event) | |
701 | { | |
8520f137 | 702 | #if wxUSE_STATUSBAR |
721b8397 DS |
703 | wxFrame *pFrame = (wxFrame *) wxWindow::GetParent(); |
704 | pFrame->SetStatusText(GetNode(event)->FullName(), 1); | |
8520f137 WS |
705 | #else |
706 | wxUnusedVar(event); | |
707 | #endif // wxUSE_STATUSBAR | |
bbf1f0e5 KB |
708 | } |
709 | ||
710 | void RegTreeCtrl::OnItemExpanding(wxTreeEvent& event) | |
711 | { | |
721b8397 DS |
712 | TreeNode *pNode = GetNode(event); |
713 | bool bExpanding = event.GetKeyCode() == wxTREE_EXPAND_EXPAND; | |
714 | ||
715 | // expansion might take some time | |
716 | wxSetCursor(*wxHOURGLASS_CURSOR); | |
717 | wxLogStatus(wxT("Working...")); | |
718 | wxYield(); // to give the status line a chance to refresh itself | |
719 | m_restoreStatus = true; // some time later... | |
720 | ||
721 | if ( pNode->IsKey() ) | |
722 | { | |
723 | if ( bExpanding ) | |
724 | { | |
725 | // expanding: add subkeys/values | |
726 | if ( !pNode->OnExpand() ) | |
727 | return; | |
728 | } | |
729 | else | |
730 | { | |
731 | // collapsing: clean up | |
732 | pNode->OnCollapse(); | |
733 | } | |
bbf1f0e5 | 734 | } |
23f681ec VZ |
735 | } |
736 | ||
c41ea66a VZ |
737 | void RegTreeCtrl::OnBeginEdit(wxTreeEvent& event) |
738 | { | |
739 | TreeNode *pNode = GetNode(event); | |
721b8397 DS |
740 | if ( pNode->IsRoot() || pNode->Parent()->IsRoot() ) |
741 | { | |
c41ea66a VZ |
742 | wxLogStatus(_T("This registry key can't be renamed.")); |
743 | ||
744 | event.Veto(); | |
745 | } | |
721b8397 DS |
746 | else |
747 | { | |
c41ea66a VZ |
748 | m_nameOld = pNode->m_strName; |
749 | } | |
750 | } | |
751 | ||
752 | void RegTreeCtrl::OnEndEdit(wxTreeEvent& event) | |
753 | { | |
754 | bool ok; | |
755 | ||
756 | wxString name = event.GetLabel(); | |
757 | ||
758 | TreeNode *pNode = GetNode(event); | |
759 | if ( pNode->IsKey() ) | |
760 | { | |
761 | wxRegKey& key = pNode->Key(); | |
762 | ok = key.Rename(name); | |
763 | } | |
764 | else | |
765 | { | |
766 | pNode = pNode->Parent(); | |
767 | wxRegKey& key = pNode->Key(); | |
768 | ||
769 | ok = key.RenameValue(m_nameOld, name); | |
770 | } | |
771 | ||
721b8397 DS |
772 | if ( !ok ) |
773 | { | |
c41ea66a | 774 | wxLogError(_T("Failed to rename '%s' to '%s'."), |
721b8397 | 775 | m_nameOld.c_str(), name.c_str()); |
c41ea66a VZ |
776 | } |
777 | #if 0 // MSW tree ctrl doesn't like this at all, it hangs | |
721b8397 DS |
778 | else |
779 | { | |
c41ea66a VZ |
780 | pNode->Refresh(); |
781 | } | |
782 | #endif // 0 | |
783 | } | |
784 | ||
23f681ec VZ |
785 | void RegTreeCtrl::OnBeginDrag(wxTreeEvent& event) |
786 | { | |
787 | m_copyOnDrop = event.GetEventType() == wxEVT_COMMAND_TREE_BEGIN_DRAG; | |
788 | ||
789 | TreeNode *pNode = GetNode(event); | |
790 | if ( pNode->IsRoot() || pNode->Parent()->IsRoot() ) | |
791 | { | |
a60b1f5d | 792 | wxLogStatus(wxT("This registry key can't be %s."), |
721b8397 | 793 | m_copyOnDrop ? wxT("copied") : wxT("moved")); |
23f681ec VZ |
794 | } |
795 | else | |
796 | { | |
a60b1f5d | 797 | wxLogStatus(wxT("%s item %s..."), |
721b8397 DS |
798 | m_copyOnDrop ? wxT("Copying") : wxT("Moving"), |
799 | pNode->FullName()); | |
bbf1f0e5 | 800 | |
23f681ec VZ |
801 | m_draggedItem = pNode; |
802 | ||
803 | event.Allow(); | |
804 | } | |
805 | } | |
806 | ||
807 | void RegTreeCtrl::OnEndDrag(wxTreeEvent& event) | |
808 | { | |
a60b1f5d | 809 | wxCHECK_RET( m_draggedItem, wxT("end drag without begin drag?") ); |
23f681ec VZ |
810 | |
811 | // clear the pointer anyhow | |
812 | TreeNode *src = m_draggedItem; | |
813 | m_draggedItem = NULL; | |
814 | ||
815 | // where are we going to drop it? | |
816 | TreeNode *dst = GetNode(event); | |
721b8397 DS |
817 | if ( dst && !dst->IsKey() ) |
818 | { | |
23f681ec VZ |
819 | // we need a parent key |
820 | dst = dst->Parent(); | |
821 | } | |
721b8397 DS |
822 | |
823 | if ( !dst || dst->IsRoot() ) | |
824 | { | |
a60b1f5d | 825 | wxLogError(wxT("Can't create a key here.")); |
23f681ec VZ |
826 | |
827 | return; | |
828 | } | |
829 | ||
830 | bool isKey = src->IsKey(); | |
831 | if ( (isKey && (src == dst)) || | |
5888ef1e | 832 | (!isKey && (dst->Parent() == src)) ) { |
a60b1f5d | 833 | wxLogStatus(wxT("Can't copy something on itself")); |
23f681ec VZ |
834 | |
835 | return; | |
836 | } | |
837 | ||
838 | // remove the "Registry Root\\" from the full name | |
839 | wxString nameSrc, nameDst; | |
840 | nameSrc << wxString(src->FullName()).AfterFirst('\\'); | |
841 | nameDst << wxString(dst->FullName()).AfterFirst('\\') << '\\' | |
842 | << wxString(src->FullName()).AfterLast('\\'); | |
843 | ||
600683ca MB |
844 | wxString verb = m_copyOnDrop ? _T("copy") : _T("move"); |
845 | wxString what = isKey ? _T("key") : _T("value"); | |
23f681ec VZ |
846 | |
847 | if ( wxMessageBox(wxString::Format | |
848 | ( | |
a60b1f5d | 849 | wxT("Do you really want to %s the %s %s to %s?"), |
23f681ec VZ |
850 | verb.c_str(), |
851 | what.c_str(), | |
852 | nameSrc.c_str(), | |
853 | nameDst.c_str() | |
854 | ), | |
600683ca | 855 | _T("RegTest Confirm"), |
23f681ec VZ |
856 | wxICON_QUESTION | wxYES_NO | wxCANCEL, this) != wxYES ) { |
857 | return; | |
858 | } | |
859 | ||
23f681ec | 860 | bool ok; |
721b8397 DS |
861 | if ( isKey ) |
862 | { | |
23f681ec | 863 | wxRegKey& key = src->Key(); |
5888ef1e | 864 | wxRegKey keyDst(dst->Key(), src->m_strName); |
721b8397 DS |
865 | ok = keyDst.Create(false); |
866 | if ( !ok ) | |
867 | { | |
2b5f62a0 | 868 | wxLogError(wxT("Key '%s' already exists"), keyDst.GetName().c_str()); |
5888ef1e | 869 | } |
721b8397 DS |
870 | else |
871 | { | |
5888ef1e VZ |
872 | ok = key.Copy(keyDst); |
873 | } | |
23f681ec | 874 | |
721b8397 DS |
875 | if ( ok && !m_copyOnDrop ) |
876 | { | |
23f681ec VZ |
877 | // delete the old key |
878 | ok = key.DeleteSelf(); | |
721b8397 DS |
879 | if ( ok ) |
880 | { | |
23f681ec VZ |
881 | src->Parent()->Refresh(); |
882 | } | |
883 | } | |
884 | } | |
721b8397 DS |
885 | else // value |
886 | { | |
23f681ec VZ |
887 | wxRegKey& key = src->Parent()->Key(); |
888 | ok = key.CopyValue(src->m_strName, dst->Key()); | |
721b8397 DS |
889 | if ( ok && !m_copyOnDrop ) |
890 | { | |
5888ef1e | 891 | // we moved it, so delete the old one |
23f681ec | 892 | ok = key.DeleteValue(src->m_strName); |
23f681ec | 893 | } |
bbf1f0e5 | 894 | } |
bbf1f0e5 | 895 | |
721b8397 DS |
896 | if ( !ok ) |
897 | { | |
a60b1f5d MB |
898 | wxLogError(wxT("Failed to %s registry %s."), |
899 | verb.c_str(), what.c_str()); | |
23f681ec | 900 | } |
721b8397 DS |
901 | else |
902 | { | |
c41ea66a VZ |
903 | dst->Refresh(); |
904 | } | |
bbf1f0e5 KB |
905 | } |
906 | ||
907 | // ---------------------------------------------------------------------------- | |
908 | // TreeNode implementation | |
909 | // ---------------------------------------------------------------------------- | |
910 | bool RegTreeCtrl::TreeNode::OnExpand() | |
911 | { | |
721b8397 DS |
912 | // we add children only once |
913 | if ( !m_aChildren.IsEmpty() ) | |
914 | { | |
915 | // we've been already expanded | |
916 | return true; | |
917 | } | |
918 | ||
919 | if ( IsRoot() ) | |
920 | { | |
921 | // we're the root key | |
922 | m_pTree->AddStdKeys(); | |
923 | return true; | |
924 | } | |
925 | ||
926 | if ( Parent()->IsRoot() ) | |
927 | { | |
928 | // we're a standard key | |
929 | m_pKey = new wxRegKey(m_strName); | |
930 | } | |
bbf1f0e5 | 931 | else |
721b8397 DS |
932 | { |
933 | // we're a normal key | |
934 | m_pKey = new wxRegKey(*(Parent()->m_pKey), m_strName); | |
935 | } | |
936 | ||
937 | if ( !m_pKey->Open() ) | |
938 | { | |
939 | wxLogError(wxT("The key '%s' can't be opened."), FullName()); | |
940 | return false; | |
941 | } | |
942 | ||
943 | // if we're empty, we shouldn't be expandable at all | |
944 | bool isEmpty = true; | |
945 | ||
946 | // enumeration variables | |
947 | long l; | |
948 | wxString str; | |
949 | bool bCont; | |
950 | ||
951 | // enumerate all subkeys | |
952 | bCont = m_pKey->GetFirstKey(str, l); | |
953 | while ( bCont ) | |
954 | { | |
955 | m_pTree->InsertNewTreeNode(this, str, RegImageList::ClosedKey); | |
956 | bCont = m_pKey->GetNextKey(str, l); | |
957 | ||
958 | // we have at least this key... | |
959 | isEmpty = false; | |
960 | } | |
961 | ||
962 | // enumerate all values | |
963 | bCont = m_pKey->GetFirstValue(str, l); | |
964 | while ( bCont ) | |
965 | { | |
966 | wxString strItem; | |
8afd90d5 | 967 | if (str.empty()) |
721b8397 DS |
968 | strItem = _T("<default>"); |
969 | else | |
970 | strItem = str; | |
971 | strItem += _T(" = "); | |
972 | ||
973 | // determine the appropriate icon | |
974 | RegImageList::Icon icon; | |
975 | switch ( m_pKey->GetValueType(str) ) | |
bbf1f0e5 | 976 | { |
721b8397 DS |
977 | case wxRegKey::Type_String: |
978 | case wxRegKey::Type_Expand_String: | |
979 | case wxRegKey::Type_Multi_String: | |
980 | { | |
981 | wxString strValue; | |
982 | icon = RegImageList::TextValue; | |
983 | m_pKey->QueryValue(str, strValue); | |
984 | strItem += strValue; | |
bbf1f0e5 KB |
985 | } |
986 | break; | |
987 | ||
721b8397 DS |
988 | case wxRegKey::Type_None: |
989 | // @@ handle the error... | |
990 | icon = RegImageList::BinaryValue; | |
991 | break; | |
bbf1f0e5 | 992 | |
721b8397 | 993 | case wxRegKey::Type_Dword: |
bbf1f0e5 | 994 | { |
721b8397 DS |
995 | long l; |
996 | m_pKey->QueryValue(str, &l); | |
997 | strItem << l; | |
bbf1f0e5 KB |
998 | } |
999 | ||
1000 | // fall through | |
1001 | ||
721b8397 DS |
1002 | default: |
1003 | icon = RegImageList::BinaryValue; | |
1004 | } | |
bbf1f0e5 | 1005 | |
721b8397 DS |
1006 | m_pTree->InsertNewTreeNode(this, str, icon, &strItem); |
1007 | bCont = m_pKey->GetNextValue(str, l); | |
23f681ec | 1008 | |
721b8397 DS |
1009 | // we have at least this value... |
1010 | isEmpty = false; | |
1011 | } | |
23f681ec | 1012 | |
721b8397 DS |
1013 | if ( isEmpty ) |
1014 | { | |
1015 | // this is for the case when our last child was just deleted | |
1016 | wxTreeItemId theId(Id()); // Temp variable seems necessary for BC++ | |
1017 | m_pTree->Collapse(theId); | |
23f681ec | 1018 | |
721b8397 DS |
1019 | // we won't be expanded any more |
1020 | m_pTree->SetItemHasChildren(theId, false); | |
1021 | } | |
bbf1f0e5 | 1022 | |
721b8397 | 1023 | return true; |
bbf1f0e5 KB |
1024 | } |
1025 | ||
1026 | void RegTreeCtrl::TreeNode::OnCollapse() | |
1027 | { | |
721b8397 | 1028 | DestroyChildren(); |
bbf1f0e5 | 1029 | |
721b8397 DS |
1030 | delete m_pKey; |
1031 | m_pKey = NULL; | |
bbf1f0e5 KB |
1032 | } |
1033 | ||
23f681ec VZ |
1034 | void RegTreeCtrl::TreeNode::Refresh() |
1035 | { | |
5888ef1e VZ |
1036 | if ( !IsKey() ) |
1037 | return; | |
1038 | ||
f6bcfd97 BP |
1039 | wxTreeItemId theId(Id()); // Temp variable seems necessary for BC++ |
1040 | bool wasExpanded = m_pTree->IsExpanded(theId); | |
c41ea66a | 1041 | if ( wasExpanded ) |
f6bcfd97 | 1042 | m_pTree->Collapse(theId); |
c41ea66a VZ |
1043 | |
1044 | OnCollapse(); | |
f6bcfd97 | 1045 | m_pTree->SetItemHasChildren(theId); |
721b8397 DS |
1046 | if ( wasExpanded ) |
1047 | { | |
f6bcfd97 | 1048 | m_pTree->Expand(theId); |
5888ef1e VZ |
1049 | OnExpand(); |
1050 | } | |
23f681ec VZ |
1051 | } |
1052 | ||
1053 | bool RegTreeCtrl::TreeNode::DeleteChild(TreeNode *child) | |
bbf1f0e5 | 1054 | { |
23f681ec | 1055 | int index = m_aChildren.Index(child); |
721b8397 | 1056 | wxCHECK_MSG( index != wxNOT_FOUND, false, |
a60b1f5d | 1057 | wxT("our child in tree should be in m_aChildren") ); |
23f681ec VZ |
1058 | |
1059 | m_aChildren.RemoveAt((size_t)index); | |
1060 | ||
1061 | bool ok; | |
721b8397 DS |
1062 | if ( child->IsKey() ) |
1063 | { | |
23f681ec VZ |
1064 | // must close key before deleting it |
1065 | child->OnCollapse(); | |
1066 | ||
1067 | ok = Key().DeleteKey(child->m_strName); | |
1068 | } | |
721b8397 DS |
1069 | else |
1070 | { | |
23f681ec VZ |
1071 | ok = Key().DeleteValue(child->m_strName); |
1072 | } | |
1073 | ||
721b8397 DS |
1074 | if ( ok ) |
1075 | { | |
f6bcfd97 BP |
1076 | wxTreeItemId theId(child->Id()); // Temp variable seems necessary for BC++ |
1077 | m_pTree->Delete(theId); | |
23f681ec VZ |
1078 | |
1079 | Refresh(); | |
1080 | } | |
1081 | ||
1082 | return ok; | |
bbf1f0e5 KB |
1083 | } |
1084 | ||
1085 | void RegTreeCtrl::TreeNode::DestroyChildren() | |
1086 | { | |
721b8397 DS |
1087 | // destroy all children |
1088 | size_t nCount = m_aChildren.GetCount(); | |
1089 | for ( size_t n = 0; n < nCount; n++ ) | |
1090 | { | |
945d96f6 WS |
1091 | wxTreeItemId lId = m_aChildren[n]->Id(); |
1092 | m_pTree->Delete(lId); | |
721b8397 DS |
1093 | } |
1094 | ||
1095 | m_aChildren.Empty(); | |
bbf1f0e5 KB |
1096 | } |
1097 | ||
1098 | RegTreeCtrl::TreeNode::~TreeNode() | |
1099 | { | |
721b8397 | 1100 | delete m_pKey; |
bbf1f0e5 KB |
1101 | } |
1102 | ||
a60b1f5d | 1103 | const wxChar *RegTreeCtrl::TreeNode::FullName() const |
bbf1f0e5 | 1104 | { |
721b8397 DS |
1105 | static wxString s_strName; |
1106 | ||
1107 | if ( IsRoot() ) | |
1108 | { | |
1109 | return wxT("Registry Root"); | |
1110 | } | |
1111 | else | |
1112 | { | |
1113 | // our own registry key might not (yet) exist or we might be a value, | |
1114 | // so just use the parent's and concatenate | |
1115 | s_strName = Parent()->FullName(); | |
1116 | s_strName << wxT('\\') << m_strName; | |
1117 | ||
1118 | return s_strName; | |
1119 | } | |
bbf1f0e5 KB |
1120 | } |
1121 | ||
1122 | // ---------------------------------------------------------------------------- | |
1123 | // operations on RegTreeCtrl | |
1124 | // ---------------------------------------------------------------------------- | |
1125 | ||
c41ea66a VZ |
1126 | void RegTreeCtrl::GoTo(const wxString& location) |
1127 | { | |
1128 | wxStringTokenizer tk(location, _T("\\")); | |
1129 | ||
1130 | wxTreeItemId id = GetRootItem(); | |
1131 | ||
721b8397 DS |
1132 | while ( tk.HasMoreTokens() ) |
1133 | { | |
c41ea66a VZ |
1134 | wxString subkey = tk.GetNextToken(); |
1135 | ||
1136 | wxTreeItemId idCurrent = id; | |
1137 | if ( !IsExpanded(idCurrent) ) | |
1138 | Expand(idCurrent); | |
1139 | ||
721b8397 | 1140 | wxTreeItemIdValue dummy; |
c41ea66a VZ |
1141 | id = GetFirstChild(idCurrent, dummy); |
1142 | ||
721b8397 DS |
1143 | if ( idCurrent == GetRootItem() ) |
1144 | { | |
c41ea66a | 1145 | // special case: we understand both HKCU and HKEY_CURRENT_USER here |
721b8397 DS |
1146 | for ( size_t key = 0; key < wxRegKey::nStdKeys; key++ ) |
1147 | { | |
1148 | if ( subkey == wxRegKey::GetStdKeyName(key) | |
1149 | || subkey == wxRegKey::GetStdKeyShortName(key) ) | |
1150 | { | |
c41ea66a VZ |
1151 | break; |
1152 | } | |
1153 | ||
1154 | id = GetNextChild(idCurrent, dummy); | |
1155 | } | |
1156 | } | |
721b8397 DS |
1157 | else |
1158 | { | |
c41ea66a | 1159 | // enum all children |
721b8397 DS |
1160 | while ( id.IsOk() ) |
1161 | { | |
c41ea66a VZ |
1162 | if ( subkey == ((TreeNode *)GetItemData(id))->m_strName ) |
1163 | break; | |
1164 | ||
1165 | id = GetNextChild(idCurrent, dummy); | |
1166 | } | |
1167 | } | |
1168 | ||
721b8397 DS |
1169 | if ( !id.IsOk() ) |
1170 | { | |
c41ea66a VZ |
1171 | wxLogError(_T("No such key '%s'."), location.c_str()); |
1172 | ||
1173 | return; | |
1174 | } | |
1175 | } | |
1176 | ||
1177 | if ( id.IsOk() ) | |
1178 | SelectItem(id); | |
1179 | } | |
1180 | ||
bbf1f0e5 KB |
1181 | void RegTreeCtrl::DeleteSelected() |
1182 | { | |
945d96f6 WS |
1183 | wxTreeItemId lCurrent = GetSelection(), |
1184 | lParent = GetItemParent(lCurrent); | |
721b8397 | 1185 | |
945d96f6 | 1186 | if ( lParent == GetRootItem() ) |
721b8397 DS |
1187 | { |
1188 | wxLogError(wxT("Can't delete root key.")); | |
1189 | return; | |
1190 | } | |
1191 | ||
1192 | TreeNode *pCurrent = (TreeNode *)GetItemData(lCurrent), | |
945d96f6 | 1193 | *pParent = (TreeNode *)GetItemData(lParent); |
721b8397 DS |
1194 | |
1195 | wxCHECK_RET(pCurrent && pParent, wxT("either node or parent without data?")); | |
1196 | ||
1197 | if ( pParent->IsRoot() ) | |
1198 | { | |
1199 | wxLogError(wxT("Can't delete standard key.")); | |
1200 | return; | |
1201 | } | |
1202 | ||
1203 | wxString what = pCurrent->IsKey() ? _T("key") : _T("value"); | |
1204 | if ( wxMessageBox(wxString::Format | |
1205 | ( | |
1206 | wxT("Do you really want to delete this %s?"), | |
1207 | what.c_str() | |
1208 | ), | |
1209 | _T("Confirmation"), | |
1210 | wxICON_QUESTION | wxYES_NO | wxCANCEL, this) != wxYES ) | |
1211 | { | |
1212 | return; | |
1213 | } | |
1214 | ||
1215 | pParent->DeleteChild(pCurrent); | |
bbf1f0e5 KB |
1216 | } |
1217 | ||
1218 | void RegTreeCtrl::CreateNewKey(const wxString& strName) | |
1219 | { | |
945d96f6 | 1220 | wxTreeItemId lCurrent = GetSelection(); |
721b8397 | 1221 | TreeNode *pCurrent = (TreeNode *)GetItemData(lCurrent); |
bbf1f0e5 | 1222 | |
721b8397 | 1223 | wxCHECK_RET( pCurrent != NULL, wxT("node without data?") ); |
23f681ec | 1224 | |
721b8397 | 1225 | wxASSERT( pCurrent->IsKey() ); // check must have been done before |
bbf1f0e5 | 1226 | |
721b8397 DS |
1227 | if ( pCurrent->IsRoot() ) |
1228 | { | |
1229 | wxLogError(wxT("Can't create a new key under the root key.")); | |
1230 | return; | |
1231 | } | |
bbf1f0e5 | 1232 | |
721b8397 DS |
1233 | wxRegKey key(pCurrent->Key(), strName); |
1234 | if ( key.Create() ) | |
1235 | pCurrent->Refresh(); | |
bbf1f0e5 KB |
1236 | } |
1237 | ||
1238 | void RegTreeCtrl::CreateNewTextValue(const wxString& strName) | |
1239 | { | |
945d96f6 | 1240 | wxTreeItemId lCurrent = GetSelection(); |
721b8397 | 1241 | TreeNode *pCurrent = (TreeNode *)GetItemData(lCurrent); |
bbf1f0e5 | 1242 | |
721b8397 | 1243 | wxCHECK_RET( pCurrent != NULL, wxT("node without data?") ); |
23f681ec | 1244 | |
721b8397 | 1245 | wxASSERT( pCurrent->IsKey() ); // check must have been done before |
bbf1f0e5 | 1246 | |
721b8397 DS |
1247 | if ( pCurrent->IsRoot() ) |
1248 | { | |
1249 | wxLogError(wxT("Can't create a new value under the root key.")); | |
1250 | return; | |
1251 | } | |
bbf1f0e5 | 1252 | |
721b8397 DS |
1253 | if ( pCurrent->Key().SetValue(strName, wxEmptyString) ) |
1254 | pCurrent->Refresh(); | |
bbf1f0e5 KB |
1255 | } |
1256 | ||
1257 | void RegTreeCtrl::CreateNewBinaryValue(const wxString& strName) | |
1258 | { | |
945d96f6 | 1259 | wxTreeItemId lCurrent = GetSelection(); |
721b8397 | 1260 | TreeNode *pCurrent = (TreeNode *)GetItemData(lCurrent); |
bbf1f0e5 | 1261 | |
721b8397 | 1262 | wxCHECK_RET( pCurrent != NULL, wxT("node without data?") ); |
23f681ec | 1263 | |
721b8397 | 1264 | wxASSERT( pCurrent->IsKey() ); // check must have been done before |
bbf1f0e5 | 1265 | |
721b8397 DS |
1266 | if ( pCurrent->IsRoot() ) |
1267 | { | |
1268 | wxLogError(wxT("Can't create a new value under the root key.")); | |
1269 | return; | |
1270 | } | |
bbf1f0e5 | 1271 | |
721b8397 DS |
1272 | if ( pCurrent->Key().SetValue(strName, 0) ) |
1273 | pCurrent->Refresh(); | |
bbf1f0e5 KB |
1274 | } |
1275 | ||
23f681ec VZ |
1276 | void RegTreeCtrl::ShowProperties() |
1277 | { | |
945d96f6 | 1278 | wxTreeItemId lCurrent = GetSelection(); |
23f681ec VZ |
1279 | TreeNode *pCurrent = (TreeNode *)GetItemData(lCurrent); |
1280 | ||
1281 | if ( !pCurrent || pCurrent->IsRoot() ) | |
1282 | { | |
a60b1f5d | 1283 | wxLogStatus(wxT("No properties")); |
23f681ec VZ |
1284 | |
1285 | return; | |
1286 | } | |
1287 | ||
1288 | if ( pCurrent->IsKey() ) | |
1289 | { | |
1290 | const wxRegKey& key = pCurrent->Key(); | |
1291 | size_t nSubKeys, nValues; | |
1292 | if ( !key.GetKeyInfo(&nSubKeys, NULL, &nValues, NULL) ) | |
1293 | { | |
a60b1f5d | 1294 | wxLogError(wxT("Couldn't get key info")); |
23f681ec VZ |
1295 | } |
1296 | else | |
1297 | { | |
a60b1f5d | 1298 | wxLogMessage(wxT("Key '%s' has %u subkeys and %u values."), |
23f681ec VZ |
1299 | key.GetName().c_str(), nSubKeys, nValues); |
1300 | } | |
1301 | } | |
1302 | else // it's a value | |
1303 | { | |
1304 | TreeNode *parent = pCurrent->Parent(); | |
a60b1f5d | 1305 | wxCHECK_RET( parent, wxT("reg value without key?") ); |
23f681ec VZ |
1306 | |
1307 | const wxRegKey& key = parent->Key(); | |
a60b1f5d MB |
1308 | const wxChar *value = pCurrent->m_strName.c_str(); |
1309 | wxLogMessage(wxT("Value '%s' under the key '%s' is of type ") | |
721b8397 DS |
1310 | wxT("%d (%s)."), |
1311 | value, | |
1312 | parent->m_strName.c_str(), | |
1313 | key.GetValueType(value), | |
1314 | key.IsNumericValue(value) ? wxT("numeric") : wxT("string")); | |
23f681ec VZ |
1315 | |
1316 | } | |
1317 | } | |
1318 | ||
bbf1f0e5 KB |
1319 | bool RegTreeCtrl::IsKeySelected() const |
1320 | { | |
945d96f6 | 1321 | wxTreeItemId lCurrent = GetSelection(); |
721b8397 | 1322 | TreeNode *pCurrent = (TreeNode *) GetItemData(lCurrent); |
bbf1f0e5 | 1323 | |
721b8397 | 1324 | wxCHECK( pCurrent != NULL, false ); |
bbf1f0e5 | 1325 | |
721b8397 | 1326 | return pCurrent->IsKey(); |
23f681ec | 1327 | } |
5888ef1e | 1328 | |
721b8397 | 1329 | void RegTreeCtrl::DoRefresh() |
5888ef1e | 1330 | { |
945d96f6 | 1331 | wxTreeItemId lId = GetSelection(); |
5888ef1e VZ |
1332 | if ( !lId ) |
1333 | return; | |
1334 | ||
721b8397 | 1335 | TreeNode *pNode = (TreeNode *) GetItemData(lId); |
5888ef1e | 1336 | |
a60b1f5d | 1337 | wxCHECK_RET( pNode != NULL, wxT("tree item without data?") ); |
5888ef1e VZ |
1338 | |
1339 | pNode->Refresh(); | |
1340 | } | |
1341 | ||
721b8397 | 1342 | #endif |