]> git.sur5r.net Git - bacula/bacula/blob - bacula/src/wx-console/wxbrestorepanel.cpp
Fix file size truncated if >100mb.
[bacula/bacula] / bacula / src / wx-console / wxbrestorepanel.cpp
1 /*
2  *
3  *   wxbPanel for restoring files
4  *
5  *    Nicolas Boichat, April-May 2004
6  *
7  */
8 /*
9    Copyright (C) 2004 Kern Sibbald and John Walker
10
11    This program is free software; you can redistribute it and/or
12    modify it under the terms of the GNU General Public License
13    as published by the Free Software Foundation; either version 2
14    of the License, or (at your option) any later version.
15
16    This program is distributed in the hope that it will be useful,
17    but WITHOUT ANY WARRANTY; without even the implied warranty of
18    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19    GNU General Public License for more details.
20
21    You should have received a copy of the GNU General Public License
22    along with this program; if not, write to the Free Software
23    Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
24  */
25 /* Note concerning "done" output (modifiable marked with +)
26 Run Restore job
27 +JobName:    RestoreFiles
28 Bootstrap:  /var/lib/bacula/restore.bsr
29 +Where:      /tmp/bacula-restores
30 +Replace:    always
31 +FileSet:    Full Set
32 +Client:     tom-fd
33 +Storage:    File
34 +When:       2004-04-18 01:18:56
35 +Priority:   10
36 OK to run? (yes/mod/no):mod
37 Parameters to modify:
38      1: Level (not appropriate)
39      2: Storage (automatic ?)
40      3: Job (no)
41      4: FileSet (yes)
42      5: Client (yes : "The defined Client resources are:\n\t1: velours-fd\n\t2: tom-fd\nSelect Client (File daemon) resource (1-2):")
43      6: When (yes : "Please enter desired start time as YYYY-MM-DD HH:MM:SS (return for now):")
44      7: Priority (yes : "Enter new Priority: (positive integer)")
45      8: Bootstrap (?)
46      9: Where (yes : "Please enter path prefix for restore (/ for none):")
47     10: Replace (yes : "Replace:\n 1: always\n 2: ifnewer\n 3: ifolder\n 4: never\n 
48           Select replace option (1-4):")
49     11: JobId (no)
50 Select parameter to modify (1-11):
51        */
52
53 #include "wxbrestorepanel.h"
54
55 #include "wxbmainframe.h"
56
57 #include "csprint.h"
58
59 #include <wx/choice.h>
60 #include <wx/datetime.h>
61
62 #include <wx/timer.h>
63
64 #include "unmarked.xpm"
65 #include "marked.xpm"
66 #include "partmarked.xpm"
67
68 #include <wx/listimpl.cpp>
69
70 /* A macro named Yield is defined under MinGW */
71 #undef Yield
72
73 WX_DEFINE_LIST(wxbEventList);
74
75 /*
76  *  Class which is stored in the tree and in the list to keep informations
77  *  about the element.
78  */
79 class wxbTreeItemData : public wxTreeItemData {
80    public:
81       wxbTreeItemData(wxString path, wxString name, int marked, long listid = -1);
82       wxbTreeItemData(wxString path, wxString name, wxString marked, long listid = -1);
83       ~wxbTreeItemData();
84       wxString GetPath();
85       wxString GetName();
86       
87       int GetMarked();
88       void SetMarked(int marked);
89       void SetMarked(wxString marked);
90       
91       long GetListId();
92
93       static int GetMarkedStatus(wxString file);
94    private:
95       wxString* path; /* Full path */
96       wxString* name; /* File name */
97       int marked; /* 0 - Not Marked, 1 - Marked, 2 - Some file under is marked */
98       long listid; /* list ID : >-1 if this data is in the list (and/or on the tree) */
99 };
100
101 wxbTreeItemData::wxbTreeItemData(wxString path, wxString name, int marked, long listid): wxTreeItemData() {
102    this->path = new wxString(path);
103    this->name = new wxString(name);
104    this->marked = marked;
105    this->listid = listid;
106 }
107
108 wxbTreeItemData::wxbTreeItemData(wxString path, wxString name, wxString marked, long listid): wxTreeItemData() {
109    this->path = new wxString(path);
110    this->name = new wxString(name);
111    SetMarked(marked);
112    this->listid = listid;
113 }
114
115 wxbTreeItemData::~wxbTreeItemData() {
116    delete path;
117    delete name;
118 }
119
120 int wxbTreeItemData::GetMarked() {
121    return marked;
122 }
123
124 void wxbTreeItemData::SetMarked(wxString marked) {
125    if (marked == "*") {
126       this->marked = 1;
127    }
128    else if (marked == "+") {
129       this->marked = 2;
130    }
131    else {
132       this->marked = 0;
133    }
134 }
135
136 void wxbTreeItemData::SetMarked(int marked) {
137    this->marked = marked;
138 }
139
140 long wxbTreeItemData::GetListId() {
141    return listid;
142 }
143
144 wxString wxbTreeItemData::GetPath() {
145    return *path;
146 }
147
148 wxString wxbTreeItemData::GetName() {
149    return *name;
150 }
151
152 /*wxbTreeItemData* wxbTreeItemData::GetChild(wxString dirname) {
153    int marked = GetMarkedStatus(dirname);
154    return new wxbTreeItemData(path + (marked ? dirname.Mid(1) : dirname), marked);
155 }*/
156
157 int wxbTreeItemData::GetMarkedStatus(wxString file) {
158    if (file.Length() == 0)
159       return 0;
160    
161    switch (file.GetChar(0)) {
162        case '*':
163           return 1;
164        case '+':
165           return 2;
166        default:
167           return 0;
168     }
169 }
170
171 // ----------------------------------------------------------------------------
172 // event tables and other macros for wxWindows
173 // ----------------------------------------------------------------------------
174
175 enum
176 {
177    RestoreStart = 1,
178    RestoreCancel = 2,
179    TreeCtrl = 3,
180    ListCtrl = 4,
181    ConfigOk = 5,
182    ConfigApply = 6,
183    ConfigCancel = 7,
184    ConfigWhere = 8,
185    ConfigReplace = 9,
186    ConfigWhen = 10,
187    ConfigPriority = 11,
188    ConfigClient = 12,
189    ConfigFileset = 13,
190    ConfigStorage = 14,
191    ConfigJobName = 15,
192    ConfigPool = 16,
193    TreeAdd = 17,
194    TreeRemove = 18,
195    TreeRefresh = 19,
196    ListAdd = 20,
197    ListRemove = 21,
198    ListRefresh = 22
199 };
200
201 BEGIN_EVENT_TABLE(wxbRestorePanel, wxPanel)
202    EVT_BUTTON(RestoreStart, wxbRestorePanel::OnStart)
203    EVT_BUTTON(RestoreCancel, wxbRestorePanel::OnCancel)
204    
205    EVT_TREE_SEL_CHANGING(TreeCtrl, wxbRestorePanel::OnTreeChanging)
206    EVT_TREE_SEL_CHANGED(TreeCtrl, wxbRestorePanel::OnTreeChanged)
207    EVT_TREE_ITEM_EXPANDING(TreeCtrl, wxbRestorePanel::OnTreeExpanding)
208    EVT_TREE_MARKED_EVENT(TreeCtrl, wxbRestorePanel::OnTreeMarked)
209    EVT_BUTTON(TreeAdd, wxbRestorePanel::OnTreeAdd)
210    EVT_BUTTON(TreeRemove, wxbRestorePanel::OnTreeRemove)
211    EVT_BUTTON(TreeRefresh, wxbRestorePanel::OnTreeRefresh)
212
213    EVT_LIST_ITEM_ACTIVATED(ListCtrl, wxbRestorePanel::OnListActivated)
214    EVT_LIST_MARKED_EVENT(ListCtrl, wxbRestorePanel::OnListMarked)
215    EVT_LIST_ITEM_SELECTED(ListCtrl, wxbRestorePanel::OnListChanged)
216    EVT_LIST_ITEM_DESELECTED(ListCtrl, wxbRestorePanel::OnListChanged)
217    EVT_BUTTON(ListAdd, wxbRestorePanel::OnListAdd)
218    EVT_BUTTON(ListRemove, wxbRestorePanel::OnListRemove)
219    EVT_BUTTON(ListRefresh, wxbRestorePanel::OnListRefresh)
220   
221    EVT_TEXT(ConfigWhere, wxbRestorePanel::OnConfigUpdated)
222    EVT_TEXT(ConfigWhen, wxbRestorePanel::OnConfigUpdated)
223    EVT_TEXT(ConfigPriority, wxbRestorePanel::OnConfigUpdated)
224    EVT_CHOICE(ConfigWhen, wxbRestorePanel::OnConfigUpdated)
225    EVT_CHOICE(ConfigReplace, wxbRestorePanel::OnConfigUpdated)
226    EVT_CHOICE(ConfigClient, wxbRestorePanel::OnConfigUpdated)
227    EVT_CHOICE(ConfigFileset, wxbRestorePanel::OnConfigUpdated)
228    EVT_CHOICE(ConfigStorage, wxbRestorePanel::OnConfigUpdated)
229    EVT_CHOICE(ConfigJobName, wxbRestorePanel::OnConfigUpdated)
230    EVT_CHOICE(ConfigPool, wxbRestorePanel::OnConfigUpdated)
231    
232    EVT_BUTTON(ConfigOk, wxbRestorePanel::OnConfigOk)
233    EVT_BUTTON(ConfigApply, wxbRestorePanel::OnConfigApply)
234    EVT_BUTTON(ConfigCancel, wxbRestorePanel::OnConfigCancel)
235 END_EVENT_TABLE()
236
237 /*
238  *  wxbRestorePanel constructor
239  */
240 wxbRestorePanel::wxbRestorePanel(wxWindow* parent): wxbPanel(parent) {
241    //pendingEvents = new wxbEventList(); //EVTQUEUE
242    //processing = false; //EVTQUEUE
243    SetWorking(false);
244    
245    imagelist = new wxImageList(16, 16, TRUE, 3);
246    imagelist->Add(wxIcon(unmarked_xpm));
247    imagelist->Add(wxIcon(marked_xpm));
248    imagelist->Add(wxIcon(partmarked_xpm));
249
250    wxFlexGridSizer* mainSizer = new wxFlexGridSizer(3, 1, 10, 10);
251    mainSizer->AddGrowableCol(0);
252    mainSizer->AddGrowableRow(1);
253
254    wxFlexGridSizer *firstSizer = new wxFlexGridSizer(1, 2, 10, 10);
255
256    firstSizer->AddGrowableCol(0);
257    firstSizer->AddGrowableRow(0);
258
259    start = new wxButton(this, RestoreStart, "Enter restore mode", wxDefaultPosition, wxSize(150, 30));
260    firstSizer->Add(start, 0, wxALIGN_CENTER_HORIZONTAL | wxALIGN_CENTER_VERTICAL, 10);
261
262    cancel = new wxButton(this, RestoreCancel, "Cancel restore", wxDefaultPosition, wxSize(150, 30));
263    firstSizer->Add(cancel, 0, wxALIGN_CENTER_HORIZONTAL | wxALIGN_RIGHT, 10);
264
265    wxString elist[1];
266
267 /*   clientChoice = new wxChoice(this, ClientChoice, wxDefaultPosition, wxSize(150, 30), 0, elist);
268    firstSizer->Add(clientChoice, 1, wxALIGN_CENTER_HORIZONTAL | wxALIGN_CENTER_VERTICAL, 10);
269
270    jobChoice = new wxChoice(this, -1, wxDefaultPosition, wxSize(150, 30), 0, elist);
271    firstSizer->Add(jobChoice, 1, wxALIGN_CENTER_HORIZONTAL | wxALIGN_CENTER_VERTICAL, 10);*/
272
273    mainSizer->Add(firstSizer, 1, wxEXPAND, 10);
274
275    treelistPanel = new wxSplitterWindow(this);
276
277    wxPanel* treePanel = new wxPanel(treelistPanel);
278    wxFlexGridSizer *treeSizer = new wxFlexGridSizer(2, 1, 0, 0);
279    treeSizer->AddGrowableCol(0);
280    treeSizer->AddGrowableRow(0);
281
282    tree = new wxbTreeCtrl(treePanel, this, TreeCtrl, wxDefaultPosition, wxSize(200,50));  
283    tree->SetImageList(imagelist);
284    
285    treeSizer->Add(tree, 1, wxEXPAND, 0);
286    
287    wxBoxSizer *treeCtrlSizer = new wxBoxSizer(wxHORIZONTAL);
288    treeadd = new wxButton(treePanel, TreeAdd, "Add", wxDefaultPosition, wxSize(60, 25));
289    treeCtrlSizer->Add(treeadd, 0, wxLEFT | wxRIGHT, 3);
290    treeremove = new wxButton(treePanel, TreeRemove, "Remove", wxDefaultPosition, wxSize(60, 25));
291    treeCtrlSizer->Add(treeremove, 0, wxLEFT | wxRIGHT, 3);
292    treerefresh = new wxButton(treePanel, TreeRefresh, "Refresh", wxDefaultPosition, wxSize(60, 25));
293    treeCtrlSizer->Add(treerefresh, 0, wxLEFT | wxRIGHT, 3);
294    
295    treeSizer->Add(treeCtrlSizer, 1, wxALIGN_CENTER_HORIZONTAL, 0);
296    
297    treePanel->SetSizer(treeSizer);
298    
299    wxPanel* listPanel = new wxPanel(treelistPanel);
300    wxFlexGridSizer *listSizer = new wxFlexGridSizer(2, 1, 0, 0);
301    listSizer->AddGrowableCol(0);
302    listSizer->AddGrowableRow(0);
303    
304    list = new wxbListCtrl(listPanel, this, ListCtrl, wxDefaultPosition, wxSize(200,50));
305    //treelistSizer->Add(list, 1, wxEXPAND, 10);
306
307    list->SetImageList(imagelist, wxIMAGE_LIST_SMALL);
308
309    wxListItem info;
310    info.SetMask(wxLIST_MASK_TEXT | wxLIST_MASK_FORMAT);
311    info.SetText("M");
312    info.SetAlign(wxLIST_FORMAT_CENTER);
313    list->InsertColumn(0, info);
314    
315    info.SetText("Filename");
316    info.SetAlign(wxLIST_FORMAT_LEFT);
317    list->InsertColumn(1, info);
318
319    info.SetText("Size");
320    info.SetAlign(wxLIST_FORMAT_RIGHT);   
321    list->InsertColumn(2, info);
322
323    info.SetText("Date");
324    info.SetAlign(wxLIST_FORMAT_LEFT);
325    list->InsertColumn(3, info);
326
327    info.SetText("Perm.");
328    info.SetAlign(wxLIST_FORMAT_LEFT);
329    list->InsertColumn(4, info);
330    
331    info.SetText("User");
332    info.SetAlign(wxLIST_FORMAT_RIGHT);
333    list->InsertColumn(5, info);
334    
335    info.SetText("Group");
336    info.SetAlign(wxLIST_FORMAT_RIGHT);
337    list->InsertColumn(6, info);
338     
339    listSizer->Add(list, 1, wxEXPAND, 0);
340    
341    wxBoxSizer *listCtrlSizer = new wxBoxSizer(wxHORIZONTAL);
342    listadd = new wxButton(listPanel, ListAdd, "Add", wxDefaultPosition, wxSize(60, 25));
343    listCtrlSizer->Add(listadd, 0, wxLEFT | wxRIGHT, 5);
344    listremove = new wxButton(listPanel, ListRemove, "Remove", wxDefaultPosition, wxSize(60, 25));
345    listCtrlSizer->Add(listremove, 0, wxLEFT | wxRIGHT, 5);
346    listrefresh = new wxButton(listPanel, ListRefresh, "Refresh", wxDefaultPosition, wxSize(60, 25));
347    listCtrlSizer->Add(listrefresh, 0, wxLEFT | wxRIGHT, 5);
348    
349    listSizer->Add(listCtrlSizer, 1, wxALIGN_CENTER_HORIZONTAL, 0);
350    
351    listPanel->SetSizer(listSizer);
352    
353    treelistPanel->SplitVertically(treePanel, listPanel, 210);
354    
355    treelistPanel->SetMinimumPaneSize(210);
356      
357    treelistPanel->Show(false);
358    
359    wxbConfig* config = new wxbConfig();
360    config->Add(new wxbConfigParam("Job Name", ConfigJobName, choice, 0, elist));
361    config->Add(new wxbConfigParam("Client", ConfigClient, choice, 0, elist));
362    config->Add(new wxbConfigParam("Fileset", ConfigFileset, choice, 0, elist));
363    config->Add(new wxbConfigParam("Pool", ConfigPool, choice, 0, elist));
364    config->Add(new wxbConfigParam("Storage", ConfigStorage, choice, 0, elist));
365    config->Add(new wxbConfigParam("Before", ConfigWhen, choice, 0, elist));
366    
367    configPanel = new wxbConfigPanel(this, config, "Please configure parameters concerning files to restore :", RestoreStart, RestoreCancel, -1);
368    
369    configPanel->Show(true);
370    configPanel->Enable(false);
371    
372    config = new wxbConfig();
373    config->Add(new wxbConfigParam("Job Name", -1, text, ""));
374    config->Add(new wxbConfigParam("Bootstrap", -1, text, ""));
375    config->Add(new wxbConfigParam("Where", ConfigWhere, modifiableText, ""));
376    wxString erlist[] = {"always", "if newer", "if older", "never"};
377    config->Add(new wxbConfigParam("Replace", ConfigReplace, choice, 4, erlist));
378    config->Add(new wxbConfigParam("Fileset", ConfigFileset, choice, 0, erlist));
379    config->Add(new wxbConfigParam("Client", ConfigClient, choice, 0, erlist));
380    config->Add(new wxbConfigParam("Storage", ConfigStorage, choice, 0, erlist));
381    config->Add(new wxbConfigParam("When", ConfigWhen, modifiableText, ""));
382    config->Add(new wxbConfigParam("Priority", ConfigPriority, modifiableText, ""));
383    
384    restorePanel = new wxbConfigPanel(this, config, "Please configure parameters concerning files restoration :", ConfigOk, ConfigCancel, ConfigApply);
385     
386    restorePanel->Show(false);
387    
388    centerSizer = new wxBoxSizer(wxHORIZONTAL);
389    //centerSizer->Add(treelistPanel, 1, wxEXPAND | wxADJUST_MINSIZE);
390       
391    mainSizer->Add(centerSizer, 1, wxEXPAND, 10);
392
393    gauge = new wxGauge(this, -1, 1, wxDefaultPosition, wxSize(200,20));
394
395    mainSizer->Add(gauge, 1, wxEXPAND, 5);
396    gauge->SetValue(0);
397    gauge->Enable(false);
398
399    SetSizer(mainSizer);
400    mainSizer->SetSizeHints(this);
401
402    SetStatus(disabled);
403
404    for (int i = 0; i < 7; i++) {
405       list->SetColumnWidth(i, 70);
406    }
407
408    SetCursor(*wxSTANDARD_CURSOR);
409
410    markWhenListingDone = false;
411    
412    cancelled = 0;
413 }
414
415 /*
416  *  wxbRestorePanel destructor
417  */
418 wxbRestorePanel::~wxbRestorePanel() {
419    delete imagelist;
420 }
421
422 /*----------------------------------------------------------------------------
423    wxbPanel overloadings
424   ----------------------------------------------------------------------------*/
425
426 wxString wxbRestorePanel::GetTitle() {
427    return "Restore";
428 }
429
430 void wxbRestorePanel::EnablePanel(bool enable) {
431    if (enable) {
432       if (status == disabled) {
433          SetStatus(activable);
434       }
435    }
436    else {
437       SetStatus(disabled);
438    }
439 }
440
441 /*----------------------------------------------------------------------------
442    Commands called by events handler
443   ----------------------------------------------------------------------------*/
444
445 /* The main button has been clicked */
446 void wxbRestorePanel::CmdStart() {
447    unsigned int i;
448    if (status == activable) {
449       wxbMainFrame::GetInstance()->SetStatusText("Getting parameters list.");
450       wxbDataTokenizer* dt = wxbUtils::WaitForEnd(".clients\n", true, false);
451       wxString str;
452
453       configPanel->ClearRowChoices("Client");
454       restorePanel->ClearRowChoices("Client");
455       
456       if (dt->GetCount() == 0) {
457          wxbMainFrame::GetInstance()->SetStatusText("Error : no clients returned by the director.");
458          return;
459       }
460       
461       for (i = 0; i < dt->GetCount(); i++) {
462          str = (*dt)[i];
463          str.RemoveLast();
464          configPanel->AddRowChoice("Client", str);
465          restorePanel->AddRowChoice("Client", str);
466       }
467           
468       delete dt;
469       
470       if (cancelled) {
471          cancelled = 2;
472          return;
473       }
474       
475       dt = wxbUtils::WaitForEnd(".filesets\n", true, false);
476       
477       configPanel->ClearRowChoices("Fileset");
478       restorePanel->ClearRowChoices("Fileset");
479     
480       if (dt->GetCount() == 0) {
481          wxbMainFrame::GetInstance()->SetStatusText("Error : no filesets returned by the director.");
482          return;
483       }
484       
485       for (i = 0; i < dt->GetCount(); i++) {
486          str = (*dt)[i];
487          str.RemoveLast();
488          configPanel->AddRowChoice("Fileset", str);
489          restorePanel->AddRowChoice("Fileset", str);
490       }
491       
492       delete dt;
493       
494       if (cancelled) {
495          cancelled = 2;
496          return;
497       }
498       
499       dt = wxbUtils::WaitForEnd(".storage\n", true, false);
500     
501       configPanel->ClearRowChoices("Storage");
502       restorePanel->ClearRowChoices("Storage");
503     
504       if (dt->GetCount() == 0) {
505          wxbMainFrame::GetInstance()->SetStatusText("Error : no storage returned by the director.");
506          return;
507       }
508       
509       for (i = 0; i < dt->GetCount(); i++) {
510          str = (*dt)[i];
511          str.RemoveLast();
512          configPanel->AddRowChoice("Storage", str);
513          restorePanel->AddRowChoice("Storage", str);
514       }
515       
516       delete dt;
517       
518       if (cancelled) {
519          cancelled = 2;
520          return;
521       }
522       
523       dt = wxbUtils::WaitForEnd(".jobs\n", true, false);
524     
525       configPanel->ClearRowChoices("Job Name");
526     
527       if (dt->GetCount() == 0) {
528          wxbMainFrame::GetInstance()->SetStatusText("Error : no jobs returned by the director.");
529          return;
530       }
531       
532       for (i = 0; i < dt->GetCount(); i++) {
533          str = (*dt)[i];
534          str.RemoveLast();
535          configPanel->AddRowChoice("Job Name", str);
536       }
537       
538       configPanel->SetRowString("Job Name", "RestoreFiles");
539       
540       delete dt;
541       
542       if (cancelled) {
543          cancelled = 2;
544          return;
545       }
546       
547       dt = wxbUtils::WaitForEnd(".pools\n", true, false);
548     
549       configPanel->ClearRowChoices("Pool");
550     
551       if (dt->GetCount() == 0) {
552          wxbMainFrame::GetInstance()->SetStatusText("Error : no jobs returned by the director.");
553          return;
554       }
555       
556       for (i = 0; i < dt->GetCount(); i++) {
557          str = (*dt)[i];
558          str.RemoveLast();
559          configPanel->AddRowChoice("Pool", str);
560       }
561          
562       delete dt; 
563       
564       if (cancelled) {
565          cancelled = 2;
566          return;
567       }
568
569       SetStatus(entered);
570
571       UpdateFirstConfig();
572            
573       wxbMainFrame::GetInstance()->SetStatusText("Please configure your restore parameters.");
574    }
575    else if (status == entered) {
576 /*      if (clientChoice->GetStringSelection().Length() < 1) {
577          wxbMainFrame::GetInstance()->SetStatusText("Please select a client.");
578          return;
579       }
580       if (jobChoice->GetStringSelection().Length() < 1) {
581          wxbMainFrame::GetInstance()->SetStatusText("Please select a restore date.");
582          return;
583       }*/
584       wxbMainFrame::GetInstance()->SetStatusText("Building restore tree...");
585       
586       SetStatus(choosing);
587       
588       wxbUtils::WaitForPrompt(wxString("restore") <<
589          " client=\"" << configPanel->GetRowString("Client") <<
590          "\" fileset=\"" << configPanel->GetRowString("Fileset") <<
591          "\" pool=\"" << configPanel->GetRowString("Pool") <<
592          "\" storage=\"" << configPanel->GetRowString("Storage") << "\"\n");
593       wxbUtils::WaitForPrompt("6\n");
594       //WaitForEnd();
595       /*wxbPromptParser *pp = wxbUtils::WaitForPrompt(wxString() << configPanel->GetRowString("Before") << "\n", true);
596       int client = pp->getChoices()->Index(configPanel->GetRowString("Client"));
597       if (client == wxNOT_FOUND) {
598          wxbMainFrame::GetInstance()->SetStatusText("Failed to find the selected client.");
599          return;
600       }
601       delete pp;*/
602       
603       wxbTableParser* tableparser = new wxbTableParser();
604       wxbDataTokenizer* dt = new wxbDataTokenizer(true);
605       
606       wxbMainFrame::GetInstance()->Send(wxString() << configPanel->GetRowString("Before") << "\n");
607    
608       while (!tableparser->hasFinished() && !dt->hasFinished()) {
609          wxTheApp->Yield(true);
610          ::wxUsleep(100);
611       }
612       
613       wxString str;
614
615       if (dt->hasFinished() && !tableparser->hasFinished()) {
616          str = "";
617          if (dt->GetCount() > 1) {
618             str = (*dt)[dt->GetCount()-2];
619             str.RemoveLast();
620          }
621          wxbMainFrame::GetInstance()->SetStatusText(wxString("Error while starting restore: ") << str);
622          delete dt;
623          delete tableparser;
624          SetStatus(finished);
625          return;
626       }
627            
628       int tot = 0;
629       long l;
630       
631       for (i = 0; i < tableparser->GetCount(); i++) {
632          str = (*tableparser)[i][2];
633          str.Replace(",", "");
634          if (str.ToLong(&l)) {
635             tot += l;
636          }
637       }
638            
639       gauge->SetValue(0);
640       gauge->SetRange(tot);
641       
642       /*wxbMainFrame::GetInstance()->Print(
643                wxString("[") << tot << "]", CS_DEBUG);*/
644       
645       wxDateTime base = wxDateTime::Now();
646       wxDateTime newdate;
647       int done = 0;
648       int willdo = 0;
649       unsigned int lastindex = 0;
650       
651       int var = 0;
652       
653       while (!dt->hasFinished()) {
654          newdate = wxDateTime::Now();
655          if (newdate.Subtract(base).GetMilliseconds() > 10 ) {
656             base = newdate;
657             for (; lastindex < dt->GetCount(); lastindex++) {
658                if (((*dt)[lastindex].Find("Building directory tree for JobId ") == 0) && 
659                      ((i = (*dt)[lastindex].Find(" ...")) > 0)) {
660                   str = (*dt)[lastindex].Mid(34, i-34);
661                   for (i = 0; i < tableparser->GetCount(); i++) {
662                      if (str == (*tableparser)[i][0]) {
663                         str = (*tableparser)[i][2];
664                         str.Replace(",", "");
665                         if (str.ToLong(&l)) {
666                            done += willdo;
667                            willdo += l;
668                            var = (willdo-done)/3;
669                         }
670                         break;
671                      }
672                   }
673                }
674             }
675             
676             if (gauge->GetValue() <= done) {
677                gauge->SetValue(done);
678                if (var < 0)
679                   var = -var;
680             }
681             else if (gauge->GetValue() >= willdo) {
682                gauge->SetValue(willdo);
683                if (var > 0)
684                   var = -var;
685             }
686             
687             gauge->SetValue(gauge->GetValue()+var);
688             
689             /*wxbMainFrame::GetInstance()->Print(
690                wxString("[") << gauge->GetValue() << "/" << done
691                   << "-" << willdo << "]", CS_DEBUG);*/
692          }
693          wxTheApp->Yield(true);
694          ::wxUsleep(1);
695       }
696
697       gauge->SetValue(tot);
698       wxTheApp->Yield(true);
699       gauge->SetValue(0);
700       
701       delete dt;
702       delete tableparser;
703
704       if (cancelled) {
705          cancelled = 2;
706          return;
707       }
708
709       wxbUtils::WaitForEnd("unmark *\n");
710       wxTreeItemId root = tree->AddRoot(configPanel->GetRowString("Client"), -1, -1, new wxbTreeItemData("/", configPanel->GetRowString("Client"), 0));
711       currentTreeItem = root;
712       tree->Refresh();
713       tree->SelectItem(root);
714       CmdList(root);
715       wxbMainFrame::GetInstance()->SetStatusText("Right click on a file or on a directory, or double-click on its mark to add it to the restore list.");
716       tree->Expand(root);
717    }
718    else if (status == choosing) {
719       EnableConfig(false);
720       
721       totfilemessages = 0;
722       wxbDataTokenizer* dt;
723            
724       int j;
725       
726       dt = new wxbDataTokenizer(true);
727       wxbUtils::WaitForPrompt("done\n");
728
729       SetStatus(configuring);
730
731       for (i = 0; i < dt->GetCount(); i++) {
732          if ((j = (*dt)[i].Find(" files selected to be restored.")) > -1) {
733             (*dt)[i].Mid(0, j).ToLong(&totfilemessages);
734             break;
735          }
736
737          if ((j = (*dt)[i].Find(" file selected to be restored.")) > -1) {
738             (*dt)[i].Mid(0, j).ToLong(&totfilemessages);
739             break;
740          }
741       }
742       
743       wxbMainFrame::GetInstance()->SetStatusText(
744          wxString("Please configure your restore (") 
745             << totfilemessages <<  " files selected to be restored)...");
746       
747       UpdateSecondConfig(dt);
748       
749       delete dt;
750       
751       EnableConfig(true);
752       restorePanel->EnableApply(false);
753
754       if (totfilemessages == 0) {
755          wxbMainFrame::GetInstance()->Print("Restore failed : no file selected.\n", CS_DEBUG);
756          wxbMainFrame::GetInstance()->SetStatusText("Restore failed : no file selected.");
757          SetStatus(finished);
758          return;
759       }
760    }
761    else if (status == configuring) {
762       cancel->Enable(false);
763       jobid = "";
764       EnableConfig(false);
765     
766       wxbMainFrame::GetInstance()->SetStatusText("Restoring, please wait...");
767     
768       wxbDataTokenizer* dt;
769     
770       SetStatus(restoring);
771       wxbUtils::WaitForEnd("yes\n");
772
773       gauge->SetValue(0);
774       gauge->SetRange(totfilemessages);
775
776       wxDateTime currenttime;
777       
778       dt = wxbUtils::WaitForEnd("time\n", true);
779       wxStringTokenizer ttkz((*dt)[0], " ", wxTOKEN_STRTOK);
780       if ((currenttime.ParseDate(ttkz.GetNextToken()) == NULL) || // Date
781            (currenttime.ParseTime(ttkz.GetNextToken()) == NULL)) { // Time
782          currenttime.SetYear(1990); // If parsing fails, set currenttime to a dummy date
783       }
784       else {
785          currenttime -= wxTimeSpan::Seconds(30); //Adding a 30" tolerance
786       }
787       delete dt;
788     
789       wxDateTime scheduledtime;
790       wxStringTokenizer stkz(restorePanel->GetRowString("When"), " ", wxTOKEN_STRTOK);
791       
792       if ((scheduledtime.ParseDate(stkz.GetNextToken()) == NULL) || // Date
793            (scheduledtime.ParseTime(stkz.GetNextToken()) == NULL)) { // Time
794          scheduledtime.SetYear(2090); // If parsing fails, set scheduledtime to a dummy date
795       }
796
797       if (scheduledtime.Subtract(currenttime).IsLongerThan(wxTimeSpan::Seconds(150))) {
798          wxbMainFrame::GetInstance()->Print("Restore is scheduled in more than two minutes, wx-console will not wait for its completion.\n", CS_DEBUG);
799          wxbMainFrame::GetInstance()->SetStatusText("Restore is scheduled in more than two minutes, wx-console will not wait for its completion.");
800          SetStatus(finished);
801          return;
802       }
803
804       wxString cmd = "list jobid=";
805
806       wxString jobname = restorePanel->GetRowString("Job Name");
807
808       wxStopWatch sw;
809
810       wxbTableParser* tableparser;
811
812       while (true) {
813          tableparser = wxbUtils::CreateAndWaitForParser("list jobs\n");
814          
815          wxDateTime jobtime;
816          
817          if (jobname == (*tableparser)[tableparser->GetCount()-1][1]) {
818             wxStringTokenizer jtkz((*tableparser)[tableparser->GetCount()-1][2], " ", wxTOKEN_STRTOK);
819             if ((jobtime.ParseDate(jtkz.GetNextToken()) != NULL) && // Date
820                   (jobtime.ParseTime(jtkz.GetNextToken()) != NULL)) { // Time
821                if (jobtime.IsLaterThan(currenttime)) {
822                   jobid = (*tableparser)[tableparser->GetCount()-1][0];
823                   cmd << jobid << "\n";
824                   delete tableparser;
825                   cancel->Enable(true);
826                   break;
827                }
828             }
829          }
830    
831          delete tableparser;
832          
833          wxStopWatch sw2;
834          while (sw2.Time() < 2000) {
835             wxTheApp->Yield(true);
836             ::wxUsleep(100);
837          }
838          if (sw.Time() > 60000) {
839             wxbMainFrame::GetInstance()->Print("The restore job has not been created within one minute, wx-console will not wait for its completion anymore.\n", CS_DEBUG);
840             wxbMainFrame::GetInstance()->SetStatusText("The restore job has not been created within one minute, wx-console will not wait for its completion anymore.");
841             SetStatus(finished);
842             cancel->Enable(true);
843             return;
844          }
845       }
846       
847       long filemessages = 0;
848
849       while (true) {
850          tableparser = wxbUtils::CreateAndWaitForParser(cmd);
851          if ((*tableparser)[0][7] != "C") {
852             break;
853          }
854          delete tableparser;
855
856          dt = wxbUtils::WaitForEnd("messages\n", true);
857          
858          for (unsigned int i = 0; i < dt->GetCount(); i++) {
859             wxStringTokenizer tkz((*dt)[i], " ", wxTOKEN_STRTOK);
860    
861             wxDateTime datetime;
862    
863             //   Date    Time   name:   perm      ?   user   grp      size    date     time
864             //04-Apr-2004 17:19 Tom-fd: -rwx------   1 nicolas  None     514967 2004-03-20 20:03:42  filename
865    
866             if (datetime.ParseDate(tkz.GetNextToken()) != NULL) { // Date
867                if (datetime.ParseTime(tkz.GetNextToken()) != NULL) { // Time
868                   if (tkz.GetNextToken().Last() == ':') { // name:
869                   tkz.GetNextToken(); // perm
870                   tkz.GetNextToken(); // ?
871                   tkz.GetNextToken(); // user
872                   tkz.GetNextToken(); // grp
873                   tkz.GetNextToken(); // size
874                   if (datetime.ParseDate(tkz.GetNextToken()) != NULL) { //date
875                         if (datetime.ParseTime(tkz.GetNextToken()) != NULL) { //time
876                            filemessages++;
877                            //wxbMainFrame::GetInstance()->Print(wxString("(") << filemessages << ")", CS_DEBUG);
878                            gauge->SetValue(filemessages);
879                         }
880                      }
881                   }
882                }
883             }
884          }
885          
886          delete dt;
887
888          wxbMainFrame::GetInstance()->SetStatusText(wxString("Restoring, please wait (") << filemessages << " of " << totfilemessages << " files done)...");
889
890          wxStopWatch sw2;
891          while (sw2.Time() < 10000) {  
892             wxTheApp->Yield(true);
893             ::wxUsleep(100);
894          }
895       }
896
897       wxbUtils::WaitForEnd("messages\n");
898
899       gauge->SetValue(totfilemessages);
900
901       if ((*tableparser)[0][7] == "T") {
902          wxbMainFrame::GetInstance()->Print("Restore done successfully.\n", CS_DEBUG);
903          wxbMainFrame::GetInstance()->SetStatusText("Restore done successfully.");
904       }
905       else {
906          wxbMainFrame::GetInstance()->Print("Restore failed, please look at messages.\n", CS_DEBUG);
907          wxbMainFrame::GetInstance()->SetStatusText("Restore failed, please look at messages in console.");
908       }
909       delete tableparser;
910       SetStatus(finished);
911    }
912 }
913
914 /* The cancel button has been clicked */
915 void wxbRestorePanel::CmdCancel() {
916    cancelled = 1;
917    
918    if (status == restoring) {
919       if (jobid != "") {
920          wxbMainFrame::GetInstance()->Send(wxString("cancel job=") << jobid << "\n");
921       }
922       cancel->Enable(true);
923       return;
924    }
925    
926    wxStopWatch sw;
927    while ((IsWorking()) && (cancelled != 2)) {
928       wxTheApp->Yield(true);
929       ::wxUsleep(100);
930       if (sw.Time() > 30000) { /* 30 seconds timeout */
931          if (status == choosing) {
932             wxbMainFrame::GetInstance()->Send("quit\n");
933          }
934          else if (status == configuring) {
935             wxbMainFrame::GetInstance()->Send("no\n");
936          }
937          else if (status == restoring) {
938             
939          }
940          SetStatus(finished);
941          ::wxUsleep(1000);
942          return;
943       }
944    }
945    
946    switch (status) {
947    case choosing:
948       wxbMainFrame::GetInstance()->Send("quit\n");      
949       break;
950    case configuring:
951       wxbMainFrame::GetInstance()->Send("no\n");      
952       break;
953    default:
954       break;
955    }
956    ::wxUsleep(1000);
957    SetStatus(finished);
958 }
959
960 /* Apply configuration changes */
961
962 /*   1: Level (not appropriate)
963  *   2: Storage (yes)
964  *   3: Job (no)
965  *   4: FileSet (yes)
966  *   5: Client (yes)
967  *   6: When (yes : "Please enter desired start time as YYYY-MM-DD HH:MM:SS (return for now):")
968  *   7: Priority (yes : "Enter new Priority: (positive integer)")
969  *   8: Bootstrap (?)
970  *   9: Where (yes : "Please enter path prefix for restore (/ for none):")
971  *  10: Replace (yes : "Replace:\n 1: always\n 2: ifnewer\n 3: ifolder\n 4: never\n 
972  *         Select replace option (1-4):")
973  *  11: JobId (no)
974  */
975
976 void wxbRestorePanel::CmdConfigApply() {
977    if (cfgUpdated == 0) return;
978    
979    wxbMainFrame::GetInstance()->SetStatusText("Applying restore configuration changes...");
980    
981    EnableConfig(false);
982    
983    wxbDataTokenizer* dt = NULL;
984    
985    bool failed = false;
986    
987    while (cfgUpdated > 0) {
988       if (cancelled) {
989          cancelled = 2;
990          return;
991       }
992       wxString def; //String to send if can't use our data
993       if ((cfgUpdated >> ConfigWhere) & 1) {
994          wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
995          wxbUtils::WaitForPrompt("9\n");
996          dt = new wxbDataTokenizer(true);
997          wxbUtils::WaitForPrompt(restorePanel->GetRowString("Where") + "\n");
998          def = "/tmp";
999          cfgUpdated = cfgUpdated & (~(1 << ConfigWhere));
1000       }
1001       else if ((cfgUpdated >> ConfigReplace) & 1) {
1002          wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1003          wxbUtils::WaitForPrompt("10\n");
1004          dt = new wxbDataTokenizer(true);
1005          wxbUtils::WaitForPrompt(wxString() << (restorePanel->GetRowSelection("Replace")+1) << "\n");
1006          def = "1";
1007          cfgUpdated = cfgUpdated & (~(1 << ConfigReplace));
1008       }
1009       else if ((cfgUpdated >> ConfigWhen) & 1) {
1010          wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1011          wxbUtils::WaitForPrompt("6\n");
1012          dt = new wxbDataTokenizer(true);
1013          wxbUtils::WaitForPrompt(restorePanel->GetRowString("When") + "\n");
1014          def = "";
1015          cfgUpdated = cfgUpdated & (~(1 << ConfigWhen));
1016       }
1017       else if ((cfgUpdated >> ConfigPriority) & 1) {
1018          wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1019          wxbUtils::WaitForPrompt("7\n");
1020          dt = new wxbDataTokenizer(true);
1021          wxbUtils::WaitForPrompt(restorePanel->GetRowString("Priority") + "\n");
1022          def = "10";
1023          cfgUpdated = cfgUpdated & (~(1 << ConfigPriority));
1024       }
1025       else if ((cfgUpdated >> ConfigClient) & 1) {
1026          wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1027          wxbPromptParser *pp = wxbUtils::WaitForPrompt("5\n", true);
1028          int client = pp->getChoices()->Index(restorePanel->GetRowString("Client"));
1029          if (client == wxNOT_FOUND) {
1030             wxbMainFrame::GetInstance()->SetStatusText("Failed to find the selected client.");
1031             failed = true;
1032             client = 1;
1033          }
1034          delete pp;
1035          dt = new wxbDataTokenizer(true);
1036          wxbUtils::WaitForPrompt(wxString() << client << "\n");
1037          def = "1";
1038          cfgUpdated = cfgUpdated & (~(1 << ConfigClient));
1039       }
1040       else if ((cfgUpdated >> ConfigFileset) & 1) {
1041          wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1042          wxbPromptParser *pp = wxbUtils::WaitForPrompt("4\n", true);
1043          int fileset = pp->getChoices()->Index(restorePanel->GetRowString("Fileset"));
1044          if (fileset == wxNOT_FOUND) {
1045             wxbMainFrame::GetInstance()->SetStatusText("Failed to find the selected fileset.");
1046             failed = true;
1047             fileset = 1;
1048          }
1049          delete pp;
1050          dt = new wxbDataTokenizer(true);
1051          wxbUtils::WaitForPrompt(wxString() << fileset << "\n");
1052          def = "1";
1053          cfgUpdated = cfgUpdated & (~(1 << ConfigFileset));
1054       }
1055       else if ((cfgUpdated >> ConfigStorage) & 1) {
1056          wxbUtils::WaitForPrompt("mod\n"); /* TODO: check results */
1057          wxbPromptParser *pp = wxbUtils::WaitForPrompt("2\n", true);
1058          int fileset = pp->getChoices()->Index(restorePanel->GetRowString("Storage"));
1059          if (fileset == wxNOT_FOUND) {
1060             wxbMainFrame::GetInstance()->SetStatusText("Failed to find the selected storage.");
1061             failed = true;
1062             fileset = 1;
1063          }
1064          delete pp;
1065          dt = new wxbDataTokenizer(true);
1066          wxbUtils::WaitForPrompt(wxString() << fileset << "\n");
1067          def = "1";
1068          cfgUpdated = cfgUpdated & (~(1 << ConfigFileset));
1069       }
1070       else {
1071          cfgUpdated = 0;
1072          break;
1073       }
1074                  
1075       unsigned int i;
1076       for (i = 0; i < dt->GetCount(); i++) {
1077          if ((*dt)[i].Find("Run Restore job") == 0) {
1078             break;
1079          }
1080       }
1081       
1082       if (i == dt->GetCount()) {
1083          delete dt;   
1084          dt = wxbUtils::WaitForEnd(def + "\n", true);
1085          failed = true;
1086       }
1087    }
1088    UpdateSecondConfig(dt); /* TODO: Check result */
1089    
1090    EnableConfig(true);
1091
1092    if (!failed) {
1093       wxbMainFrame::GetInstance()->SetStatusText("Restore configuration changes were applied.");
1094    }
1095
1096    delete dt;
1097 }
1098
1099 /* Cancel restore */
1100 void wxbRestorePanel::CmdConfigCancel() {
1101    wxbUtils::WaitForEnd("no\n");
1102    wxbMainFrame::GetInstance()->Print("Restore cancelled.\n", CS_DEBUG);
1103    wxbMainFrame::GetInstance()->SetStatusText("Restore cancelled.");
1104    SetStatus(finished);
1105 }
1106
1107 /* List jobs for a specified client */
1108 void wxbRestorePanel::CmdListJobs() {
1109    if (status == entered) {
1110       configPanel->ClearRowChoices("Before");
1111       wxbUtils::WaitForPrompt("query\n");
1112       wxbUtils::WaitForPrompt("6\n");
1113       wxbTableParser* tableparser = wxbUtils::CreateAndWaitForParser(configPanel->GetRowString("Client") + "\n");
1114
1115       for (int i = tableparser->GetCount()-1; i > -1; i--) {
1116          wxString str = (*tableparser)[i][3];
1117          wxDateTime datetime;
1118          const char* chr;
1119          if ( ( (chr = datetime.ParseDate(str.GetData()) ) != NULL ) && ( datetime.ParseTime(++chr) != NULL ) ) {
1120             datetime += wxTimeSpan::Seconds(1);
1121             //wxbMainFrame::GetInstance()->Print(wxString("-") << datetime.Format("%Y-%m-%d %H:%M:%S"), CS_DEBUG);
1122             configPanel->AddRowChoice("Before", datetime.Format("%Y-%m-%d %H:%M:%S"));
1123          }
1124          /*else {
1125          jobChoice->Append("Invalid");
1126          }*/
1127       }
1128       
1129       delete tableparser;
1130
1131       configPanel->SetRowSelection("Before", 0);
1132    }
1133 }
1134
1135 /* List files and directories for a specified tree item */
1136 void wxbRestorePanel::CmdList(wxTreeItemId item) {
1137    if (status == choosing) {
1138       list->DeleteAllItems();
1139
1140       if (!item.IsOk()) {
1141          return;
1142       }
1143       UpdateTreeItem(item, true, false);
1144     
1145       if (list->GetItemCount() >= 1) {
1146          int firstwidth = list->GetSize().GetWidth(); 
1147          for (int i = 2; i < 7; i++) {
1148             list->SetColumnWidth(i, wxLIST_AUTOSIZE);
1149             firstwidth -= list->GetColumnWidth(i);
1150          }
1151        
1152          list->SetColumnWidth(0, 18);
1153          firstwidth -= 18;
1154          list->SetColumnWidth(1, wxLIST_AUTOSIZE);
1155          if (list->GetColumnWidth(1) < firstwidth) {
1156             list->SetColumnWidth(1, firstwidth-25);
1157          }
1158       }
1159    }
1160 }
1161
1162 /* Mark a treeitem (directory) or a listitem (file or directory) */
1163 void wxbRestorePanel::CmdMark(wxTreeItemId treeitem, long* listitems, int listsize, int state) {
1164    if (status == choosing) {
1165       wxbTreeItemData** itemdata;
1166       int itemdatasize = 0;
1167       if (listsize == 0) {
1168          itemdata = new wxbTreeItemData*[1];
1169          itemdatasize = 1;
1170       }
1171       else {
1172          itemdata = new wxbTreeItemData*[listsize];
1173          itemdatasize = listsize;
1174       }
1175       
1176       if (listitems != NULL) {
1177          for (int i = 0; i < listsize; i++) {
1178             itemdata[i] = (wxbTreeItemData*)list->GetItemData(listitems[i]);
1179          }
1180       }
1181       else if (treeitem.IsOk()) {
1182          itemdata[0] = (wxbTreeItemData*)tree->GetItemData(treeitem);
1183       }
1184       else {
1185          delete[] itemdata;
1186          return;
1187       }
1188
1189       if (itemdata[0] == NULL) { //Should never happen
1190          delete[] itemdata;
1191          return;
1192       }
1193
1194       wxString dir = itemdata[0]->GetPath();
1195       wxString file;
1196
1197       if (dir != "/") {
1198          if (dir.GetChar(dir.Length()-1) == '/') {
1199             dir.RemoveLast();
1200          }
1201
1202          int i = dir.Find('/', TRUE);
1203          if (i == -1) {
1204             file = dir;
1205             dir = "/";
1206          }
1207          else { /* first dir below root */
1208             file = dir.Mid(i+1);
1209             dir = dir.Mid(0, i+1);
1210          }
1211       }
1212       else {
1213          dir = "/";
1214          file = "*";
1215       }
1216
1217       if (state == -1) {
1218          bool marked = false;
1219          bool unmarked = false;
1220          state = 0;
1221          for (int i = 0; i < itemdatasize; i++) {
1222             switch(itemdata[i]->GetMarked()) {
1223             case 0:
1224                unmarked = true;
1225                break;
1226             case 1:
1227                marked = true;
1228                break;
1229             case 2:
1230                marked = true;
1231                unmarked = true;
1232                break;
1233             default:
1234                break;
1235             }
1236             if (marked && unmarked)
1237                break;
1238          }
1239          if (marked) {
1240             if (unmarked) {
1241                state = 1;
1242             }
1243             else {
1244                state = 0;
1245             }
1246          }
1247          else {
1248             state = 1;
1249          }
1250       }
1251
1252       wxbUtils::WaitForEnd(wxString("cd \"") << dir << "\"\n");
1253       wxbUtils::WaitForEnd(wxString((state==1) ? "mark" : "unmark") << " \"" << file << "\"\n");
1254
1255       /* TODO: Check commands results */
1256
1257       /*if ((dir == "/") && (file == "*")) {
1258             itemdata->SetMarked((itemdata->GetMarked() == 1) ? 0 : 1);
1259       }*/
1260
1261       if (listitems == NULL) { /* tree item state changed */
1262          SetTreeItemState(treeitem, state);
1263          /*treeitem = tree->GetSelection();
1264          UpdateTree(treeitem, true);
1265          treeitem = tree->GetItemParent(treeitem);*/
1266       }
1267       else {
1268          for (int i = 0; i < listsize; i++) {
1269             SetListItemState(listitems[i], state);
1270          }
1271          /*UpdateTree(treeitem, (tree->GetSelection() == treeitem));
1272          treeitem = tree->GetItemParent(treeitem);*/
1273       }
1274
1275       /*while (treeitem.IsOk()) {
1276          WaitForList(treeitem, false);
1277          treeitem = tree->GetItemParent(treeitem);
1278       }*/
1279       
1280       delete[] itemdata;
1281    }
1282 }
1283
1284 /*----------------------------------------------------------------------------
1285    General functions
1286   ----------------------------------------------------------------------------*/
1287
1288 /* Run a dir command, and waits until result is fully received. */
1289 void wxbRestorePanel::UpdateTreeItem(wxTreeItemId item, bool updatelist, bool recurse) {
1290 //   this->updatelist = updatelist;
1291    wxbDataTokenizer* dt;
1292
1293    dt = wxbUtils::WaitForEnd(wxString("cd \"") << 
1294       static_cast<wxbTreeItemData*>(tree->GetItemData(item))
1295          ->GetPath() << "\"\n", false);
1296
1297    /* TODO: check command result */
1298    
1299    //delete dt;
1300
1301    status = listing;
1302
1303    if (updatelist)
1304       list->DeleteAllItems();
1305    dt = wxbUtils::WaitForEnd("dir\n", true);
1306    
1307    wxString str;
1308    
1309    for (unsigned int i = 0; i < dt->GetCount(); i++) {
1310       str = (*dt)[i];
1311       
1312       if (str.Find("cwd is:") == 0) { // Sometimes cd command result "infiltrate" into listings.
1313          break;
1314       }
1315
1316       str.RemoveLast();
1317
1318       wxString* file = ParseList(str);
1319       
1320       if (file == NULL)
1321             break;
1322
1323       wxTreeItemId treeid;
1324
1325       if (file[8].GetChar(file[8].Length()-1) == '/') {
1326          wxString itemStr;
1327
1328          long cookie;
1329          treeid = tree->GetFirstChild(item, cookie);
1330
1331          bool updated = false;
1332
1333          while (treeid.IsOk()) {
1334             itemStr = tree->GetItemText(treeid);
1335             if (file[8] == itemStr) {
1336                int stat = wxbTreeItemData::GetMarkedStatus(file[6]);
1337                if (static_cast<wxbTreeItemData*>(tree->GetItemData(treeid))->GetMarked() != stat) {
1338                   tree->SetItemImage(treeid, stat, wxTreeItemIcon_Normal);
1339                   tree->SetItemImage(treeid, stat, wxTreeItemIcon_Selected);
1340                   static_cast<wxbTreeItemData*>(tree->GetItemData(treeid))->SetMarked(file[6]);
1341                }
1342                if ((recurse) && (tree->IsExpanded(treeid))) {
1343                   UpdateTreeItem(treeid, false, true);
1344                }
1345                updated = true;
1346                break;
1347             }
1348             treeid = tree->GetNextChild(item, cookie);
1349          }
1350
1351          if (!updated) {
1352             int img = wxbTreeItemData::GetMarkedStatus(file[6]);
1353             treeid = tree->AppendItem(item, file[8], img, img, new wxbTreeItemData(file[7], file[8], file[6]));
1354          }
1355       }
1356
1357       if (updatelist) {
1358          long ind = list->InsertItem(list->GetItemCount(), wxbTreeItemData::GetMarkedStatus(file[6]));
1359          wxbTreeItemData* data = new wxbTreeItemData(file[7], file[8], file[6], ind);
1360          data->SetId(treeid);
1361          list->SetItemData(ind, (long)data);
1362          list->SetItem(ind, 1, file[8]); // filename
1363          list->SetItem(ind, 2, file[4]); //Size
1364          list->SetItem(ind, 3, file[5]); //date
1365          list->SetItem(ind, 4, file[0]); //perm
1366          list->SetItem(ind, 5, file[2]); //user
1367          list->SetItem(ind, 6, file[3]); //grp
1368       }
1369
1370       delete[] file;
1371    }
1372    
1373    delete dt;
1374    
1375    tree->Refresh();
1376    status = choosing;
1377 }
1378
1379 /* Parse dir command results. */
1380 wxString* wxbRestorePanel::ParseList(wxString line) {
1381    /* See ls_output in dird/ua_tree.c */
1382   
1383    //drwxrwxrwx   1 root     root           0  2004-04-03 14:35:21  f:/tocd/NVSU 1.00.00/
1384    //+ 10     +  ++ +   8  + +   8  ++   8  +  +      19         + *+ ->
1385    //0           12 15       24      32        42                  62
1386
1387    if (line.Length() < 63)
1388       return NULL;
1389
1390    wxString* ret = new wxString[9];
1391
1392    ret[0] = line.Mid(0, 10).Trim();
1393    ret[1] = line.Mid(12, 2).Trim();
1394    ret[2] = line.Mid(15, 8).Trim();
1395    ret[3] = line.Mid(24, 8).Trim();
1396    ret[4] = line.Mid(32, 8).Trim();
1397    ret[5] = line.Mid(42, 19).Trim();
1398    ret[6] = line.Mid(62, 1);
1399    ret[7] = line.Mid(63).Trim();
1400
1401    if (ret[6] == " ") ret[6] = "";
1402
1403    if (ret[7].GetChar(ret[7].Length()-1) == '/') {
1404       ret[8] = ret[7];
1405       ret[8].RemoveLast();
1406       ret[8] = ret[7].Mid(ret[8].Find('/', true)+1);
1407    }
1408    else {
1409       ret[8] = ret[7].Mid(ret[7].Find('/', true)+1);
1410    }
1411
1412    return ret;
1413 }
1414
1415 /* Sets a list item state, and update its parents and children if it is a directory */
1416 void wxbRestorePanel::SetListItemState(long listitem, int newstate) {
1417    wxbTreeItemData* itemdata = (wxbTreeItemData*)list->GetItemData(listitem);
1418    
1419    wxTreeItemId treeitem;
1420    
1421    itemdata->SetMarked(newstate);
1422    list->SetItemImage(listitem, newstate, 0); /* TODO: Find what these ints are for */
1423    list->SetItemImage(listitem, newstate, 1);
1424       
1425    if ((treeitem = itemdata->GetId()).IsOk()) {
1426       SetTreeItemState(treeitem, newstate);
1427    }
1428    else {
1429       UpdateTreeItemState(tree->GetSelection());
1430    }
1431 }
1432
1433 /* Sets a tree item state, and update its children, parents and list (if necessary) */
1434 void wxbRestorePanel::SetTreeItemState(wxTreeItemId item, int newstate) {
1435    long cookie;
1436    wxTreeItemId currentChild = tree->GetFirstChild(item, cookie);
1437
1438    wxbTreeItemData* itemdata;
1439
1440    while (currentChild.IsOk()) {
1441       itemdata = (wxbTreeItemData*)tree->GetItemData(currentChild);
1442       int state = itemdata->GetMarked();
1443       
1444       if (state != newstate) {
1445          itemdata->SetMarked(newstate);
1446          tree->SetItemImage(currentChild, newstate, wxTreeItemIcon_Normal);
1447          tree->SetItemImage(currentChild, newstate, wxTreeItemIcon_Selected);
1448       }
1449       
1450       currentChild = tree->GetNextChild(item, cookie);
1451    }
1452      
1453    itemdata = (wxbTreeItemData*)tree->GetItemData(item);  
1454    itemdata->SetMarked(newstate);
1455    tree->SetItemImage(item, newstate, wxTreeItemIcon_Normal);
1456    tree->SetItemImage(item, newstate, wxTreeItemIcon_Selected);
1457    tree->Refresh();
1458    
1459    if (tree->GetSelection() == item) {
1460       for (long i = 0; i < list->GetItemCount(); i++) {
1461          list->SetItemImage(i, newstate, 0); /* TODO: Find what these ints are for */
1462          list->SetItemImage(i, newstate, 1);
1463       }
1464    }
1465    
1466    UpdateTreeItemState(tree->GetItemParent(item));
1467 }
1468
1469 /* Update a tree item state, and its parents' state */
1470 void wxbRestorePanel::UpdateTreeItemState(wxTreeItemId item) {  
1471    if (!item.IsOk()) {
1472       return;
1473    }
1474    
1475    int state = 0;
1476        
1477    long cookie;
1478    wxTreeItemId currentChild = tree->GetFirstChild(item, cookie);
1479
1480    bool onechildmarked = false;
1481    bool onechildunmarked = false;
1482
1483    while (currentChild.IsOk()) {
1484       state = ((wxbTreeItemData*)tree->GetItemData(currentChild))->GetMarked();
1485       switch (state) {
1486       case 0:
1487          onechildunmarked = true;
1488          break;
1489       case 1:
1490          onechildmarked = true;
1491          break;
1492       case 2:
1493          onechildmarked = true;
1494          onechildunmarked = true;
1495          break;
1496       }
1497       
1498       if (onechildmarked && onechildunmarked) {
1499          break;
1500       }
1501       
1502       currentChild = tree->GetNextChild(item, cookie);
1503    }
1504    
1505    if (tree->GetSelection() == item) {
1506       for (long i = 0; i < list->GetItemCount(); i++) {
1507          state = ((wxbTreeItemData*)list->GetItemData(i))->GetMarked();
1508          
1509          switch (state) {
1510          case 0:
1511             onechildunmarked = true;
1512             break;
1513          case 1:
1514             onechildmarked = true;
1515             break;
1516          case 2:
1517             onechildmarked = true;
1518             onechildunmarked = true;
1519             break;
1520          }
1521          
1522          if (onechildmarked && onechildunmarked) {
1523             break;
1524          }
1525       }
1526    }
1527    
1528    state = 0;
1529    
1530    if (onechildmarked && onechildunmarked) {
1531       state = 2;
1532    }
1533    else if (onechildmarked) {
1534       state = 1;
1535    }
1536    else if (onechildunmarked) {
1537       state = 0;
1538    }
1539    else { // no child, don't change anything
1540       UpdateTreeItemState(tree->GetItemParent(item));
1541       return;
1542    }
1543    
1544    wxbTreeItemData* itemdata = (wxbTreeItemData*)tree->GetItemData(item);
1545       
1546    itemdata->SetMarked(state);
1547    tree->SetItemImage(item, state, wxTreeItemIcon_Normal);
1548    tree->SetItemImage(item, state, wxTreeItemIcon_Selected);
1549    
1550    UpdateTreeItemState(tree->GetItemParent(item));
1551 }
1552
1553 /* Refresh the whole tree. */
1554 void wxbRestorePanel::RefreshTree() {
1555    /* Save current selection */
1556    wxArrayString current;
1557    
1558    wxTreeItemId item = currentTreeItem;
1559    
1560    while ((item.IsOk()) && (item != tree->GetRootItem())) {
1561       current.Add(tree->GetItemText(item));
1562       item = tree->GetItemParent(item);
1563    }
1564
1565    /* Update the tree */
1566    UpdateTreeItem(tree->GetRootItem(), false, true);
1567
1568    /* Reselect the former selected item */   
1569    item = tree->GetRootItem();
1570    
1571    if (current.Count() == 0) {
1572       tree->SelectItem(item);
1573       return;
1574    }
1575    
1576    bool match;
1577    
1578    for (int i = current.Count()-1; i >= 0; i--) {
1579       long cookie;
1580       wxTreeItemId currentChild = tree->GetFirstChild(item, cookie);
1581       
1582       match = false;
1583       
1584       while (currentChild.IsOk()) {
1585          if (tree->GetItemText(currentChild) == current[i]) {
1586             item = currentChild;
1587             match = true;
1588             break;
1589          }
1590    
1591          currentChild = tree->GetNextChild(item, cookie);
1592       }
1593       
1594       if (!match) break;
1595    }
1596    
1597    UpdateTreeItem(item, true, false); /* Update the list */
1598    
1599    tree->SelectItem(item);
1600 }
1601
1602 void wxbRestorePanel::RefreshList() {
1603    if (currentTreeItem.IsOk()) {
1604       UpdateTreeItem(currentTreeItem, true, false); /* Update the list */
1605    }
1606 }
1607
1608 /* Update first config, adapting settings to the job name selected */
1609 void wxbRestorePanel::UpdateFirstConfig() {
1610    configPanel->Enable(false);
1611    wxbDataTokenizer* dt = wxbUtils::WaitForEnd(wxString(".defaults job=") + configPanel->GetRowString("Job Name") + "\n", true, false);
1612    /* job=RestoreFiles
1613     * pool=Default
1614     * messages=Standard
1615     * client=***
1616     * storage=File
1617     * where=/tmp/bacula-restores
1618     * level=0
1619     * type=Restore
1620     * fileset=Full Set */
1621    
1622    wxString name, str;
1623    unsigned int i;
1624    int j;
1625    wxString client;
1626    bool dolistjobs = false;
1627    
1628    for (i = 0; i < dt->GetCount(); i++) {
1629       str = (*dt)[i];
1630       if ((j = str.Find('=')) > -1) {
1631          name = str.Mid(0, j);
1632          if (name == "pool") {
1633             configPanel->SetRowString("Pool", str.Mid(j+1));
1634          }
1635          else if (name == "client") {
1636             str = str.Mid(j+1);
1637             if ((str != configPanel->GetRowString("Client")) || (configPanel->GetRowString("Before") == "")) {
1638                configPanel->SetRowString("Client", str);
1639                dolistjobs = true;
1640             }
1641          }
1642          else if (name == "storage") {
1643             configPanel->SetRowString("Storage", str.Mid(j+1));
1644          }
1645          else if (name == "fileset") {
1646             configPanel->SetRowString("Fileset", str.Mid(j+1));
1647          }
1648       }
1649    }
1650       
1651    delete dt;
1652    
1653    if (dolistjobs) {
1654       //wxTheApp->Yield(false);
1655       CmdListJobs();
1656    }
1657    configPanel->Enable(true);
1658 }
1659
1660 /* 
1661  * Update second config.
1662  * 
1663  * Run Restore job
1664  * JobName:    RestoreFiles
1665  * Bootstrap:  /var/lib/bacula/restore.bsr
1666  * Where:      /tmp/bacula-restores
1667  * Replace:    always
1668  * FileSet:    Full Set
1669  * Client:     tom-fd
1670  * Storage:    File
1671  * When:       2004-04-18 01:18:56
1672  * Priority:   10
1673  * OK to run? (yes/mod/no):
1674  * 
1675  */
1676 bool wxbRestorePanel::UpdateSecondConfig(wxbDataTokenizer* dt) {
1677    unsigned int i;
1678    for (i = 0; i < dt->GetCount(); i++) {
1679       if ((*dt)[i].Find("Run Restore job") == 0)
1680          break;
1681    }
1682    
1683    if ((i + 10) > dt->GetCount()) {
1684       return false;
1685    }
1686    
1687    int k;
1688    
1689    if ((k = (*dt)[++i].Find("JobName:")) != 0) return false;
1690    restorePanel->SetRowString("Job Name", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1691    if ((k = (*dt)[++i].Find("Bootstrap:")) != 0) return false;
1692    restorePanel->SetRowString("Bootstrap", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1693    if ((k = (*dt)[++i].Find("Where:")) != 0) return false;
1694    restorePanel->SetRowString("Where", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1695    
1696    if ((k = (*dt)[++i].Find("Replace:")) != 0) return false;
1697    wxString str = (*dt)[i].Mid(10).Trim(false).RemoveLast();
1698    if (str == "always") restorePanel->SetRowSelection("Replace", 0);
1699    else if (str == "ifnewer") restorePanel->SetRowSelection("Replace", 1);
1700    else if (str == "ifolder") restorePanel->SetRowSelection("Replace", 2);
1701    else if (str == "never") restorePanel->SetRowSelection("Replace", 3);
1702    else restorePanel->SetRowSelection("Replace", 0);
1703
1704    if ((k = (*dt)[++i].Find("FileSet:")) != 0) return false;
1705    restorePanel->SetRowString("Fileset", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1706    if ((k = (*dt)[++i].Find("Client:")) != 0) return false;
1707    restorePanel->SetRowString("Client", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1708    if ((k = (*dt)[++i].Find("Storage:")) != 0) return false;
1709    restorePanel->SetRowString("Storage", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1710    if ((k = (*dt)[++i].Find("When:")) != 0) return false;
1711    restorePanel->SetRowString("When", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1712    if ((k = (*dt)[++i].Find("Priority:")) != 0) return false;
1713    restorePanel->SetRowString("Priority", (*dt)[i].Mid(10).Trim(false).RemoveLast());
1714    cfgUpdated = 0;
1715    
1716    restorePanel->Layout();
1717    
1718    return true;
1719 }
1720
1721 /*----------------------------------------------------------------------------
1722    Status function
1723   ----------------------------------------------------------------------------*/
1724
1725 /* Set current status by enabling/disabling components */
1726 void wxbRestorePanel::SetStatus(status_enum newstatus) {
1727    switch (newstatus) {
1728    case disabled:
1729       centerSizer->Remove(configPanel);
1730       centerSizer->Remove(restorePanel);
1731       centerSizer->Remove(treelistPanel);
1732       treelistPanel->Show(false);
1733       restorePanel->Show(false);
1734       centerSizer->Add(configPanel, 1, wxEXPAND);
1735       configPanel->Show(true);
1736       configPanel->Layout();
1737       centerSizer->Layout();
1738       this->Layout();
1739       start->SetLabel("Enter restore mode");
1740       start->Enable(false);
1741       configPanel->Enable(false);
1742       tree->Enable(false);
1743       list->Enable(false);
1744       gauge->Enable(false);
1745       cancel->Enable(false);
1746       cfgUpdated = 0;
1747       cancelled = 0;
1748       break;
1749    case finished:
1750       centerSizer->Remove(configPanel);
1751       centerSizer->Remove(restorePanel);
1752       centerSizer->Remove(treelistPanel);
1753       treelistPanel->Show(false);
1754       restorePanel->Show(false);
1755       centerSizer->Add(configPanel, 1, wxEXPAND);
1756       configPanel->Show(true);
1757       configPanel->Layout();
1758       centerSizer->Layout();
1759       this->Layout();
1760       tree->DeleteAllItems();
1761       list->DeleteAllItems();
1762       configPanel->ClearRowChoices("Client");
1763       configPanel->ClearRowChoices("Before");
1764       wxbMainFrame::GetInstance()->EnablePanels();
1765       newstatus = activable;
1766    case activable:
1767       cancelled = 0;
1768       start->SetLabel("Enter restore mode");
1769       start->Enable(true);
1770       configPanel->Enable(false);
1771       tree->Enable(false);
1772       list->Enable(false);
1773       gauge->Enable(false);
1774       cancel->Enable(false);
1775       cfgUpdated = 0;
1776       break;
1777    case entered:
1778       wxbMainFrame::GetInstance()->DisablePanels(this);
1779       gauge->SetValue(0);
1780       start->Enable(false);
1781       //start->SetLabel("Choose files to restore");
1782       configPanel->Enable(true);
1783       tree->Enable(false);
1784       list->Enable(false);
1785       cancel->Enable(true);
1786       cfgUpdated = 0;
1787       break;
1788    case listing:
1789       
1790       break;
1791    case choosing:
1792       start->Enable(true);
1793       start->SetLabel("Restore");
1794       centerSizer->Remove(configPanel);
1795       configPanel->Show(false);
1796       centerSizer->Add(treelistPanel, 1, wxEXPAND);
1797       treelistPanel->Show(true);
1798       treelistPanel->Layout();
1799       centerSizer->Layout();
1800       this->Layout();
1801       tree->Enable(true);
1802       list->Enable(true);
1803       SetWorking(false);
1804       break;
1805    case configuring:
1806       start->Enable(false);
1807       configPanel->Enable(false);
1808       tree->Enable(false);
1809       list->Enable(false);
1810       centerSizer->Remove(treelistPanel);
1811       treelistPanel->Show(false);
1812       centerSizer->Add(restorePanel, 1, wxEXPAND);
1813       restorePanel->Show(true);
1814       restorePanel->Layout();
1815       centerSizer->Layout();
1816       this->Layout();
1817       restorePanel->EnableApply(false);
1818       break;
1819    case restoring:
1820       start->SetLabel("Restoring...");
1821       gauge->Enable(true);
1822       gauge->SetValue(0);
1823       start->Enable(false);
1824       configPanel->Enable(false);
1825       tree->Enable(false);
1826       list->Enable(false);
1827       SetWorking(true);
1828       break;
1829    }
1830    status = newstatus;
1831 }
1832
1833 /*----------------------------------------------------------------------------
1834    UI related
1835   ----------------------------------------------------------------------------*/
1836
1837 void wxbRestorePanel::SetWorking(bool working) {
1838    this->working = working;
1839    if (working) {
1840       SetCursor(*wxHOURGLASS_CURSOR);
1841 //      SetEvtHandlerEnabled(false); //EVTQUEUE
1842    }
1843 //   else if (!processing) { /* Empty event queue if we aren't already doing this */ //EVTQUEUE
1844    else {
1845 //      processing = true; //EVTQUEUE
1846       SetCursor(*wxSTANDARD_CURSOR);
1847 //      SetEvtHandlerEnabled(true); //EVTQUEUE
1848 /*      wxNode *node = pendingEvents->First(); //EVTQUEUE
1849       while ( node ) {
1850          wxEvent *event = (wxEvent *)node->Data();
1851          delete node;
1852    
1853          wxEvtHandler::ProcessEvent(*event);
1854          delete event;
1855    
1856          node = pendingEvents->First();
1857       }
1858       processing = false;*/
1859    }
1860 }
1861
1862 bool wxbRestorePanel::IsWorking() {
1863    return this->working;
1864 }
1865
1866 void wxbRestorePanel::EnableConfig(bool enable) {
1867    restorePanel->Enable(enable);
1868 }
1869
1870 /*----------------------------------------------------------------------------
1871    Event handling
1872   ----------------------------------------------------------------------------*/
1873
1874
1875 //EVTQUEUE
1876 /*
1877 bool wxbRestorePanel::ProcessEvent(wxEvent& event) {
1878    if (IsWorking() || processing) {
1879       wxEvent *eventCopy = event.Clone();
1880       
1881       pendingEvents->Append(eventCopy);
1882       return TRUE;
1883    }
1884    else {
1885       return wxEvtHandler::ProcessEvent(event);
1886    }
1887 }
1888 */
1889
1890 void wxbRestorePanel::OnCancel(wxCommandEvent& event) {
1891    cancel->Enable(false);
1892    SetCursor(*wxHOURGLASS_CURSOR);
1893    CmdCancel();
1894    SetCursor(*wxSTANDARD_CURSOR);
1895 }
1896
1897 void wxbRestorePanel::OnStart(wxCommandEvent& event) {
1898    if (IsWorking()) {
1899       AddPendingEvent(event);
1900       return;
1901    }
1902    SetWorking(true);
1903    CmdStart();
1904    SetWorking(false);
1905 }
1906
1907 void wxbRestorePanel::OnTreeChanging(wxTreeEvent& event) {
1908    if (IsWorking()) {
1909       AddPendingEvent(event);
1910       event.Veto();
1911    }
1912 }
1913
1914 void wxbRestorePanel::OnTreeExpanding(wxTreeEvent& event) {
1915    if (IsWorking()) {
1916       AddPendingEvent(event);
1917       event.Veto();
1918       return;
1919    }
1920    //working = true;
1921    //CmdList(event.GetItem());
1922    if (tree->GetSelection() != event.GetItem()) {
1923       tree->SelectItem(event.GetItem());
1924    }
1925    //working = false;
1926 }
1927
1928 void wxbRestorePanel::OnTreeChanged(wxTreeEvent& event) {
1929    if (IsWorking()) {
1930       AddPendingEvent(event);
1931       return;
1932    }
1933    if (currentTreeItem == event.GetItem()) {
1934       return;
1935    }
1936    treeadd->Enable(false);
1937    treeremove->Enable(false);
1938    treerefresh->Enable(false);
1939    markWhenListingDone = false;
1940    SetWorking(true);
1941    currentTreeItem = event.GetItem();
1942    CmdList(event.GetItem());
1943    if (markWhenListingDone) {
1944       CmdMark(event.GetItem(), NULL, 0);
1945       tree->Refresh();
1946    }
1947    SetWorking(false);
1948    if (event.GetItem().IsOk()) {
1949       int status = ((wxbTreeItemData*)tree->GetItemData(event.GetItem()))->GetMarked();
1950       treeadd->Enable(status != 1);
1951       treeremove->Enable(status != 0);
1952    }
1953    treerefresh->Enable(true);
1954 }
1955
1956 void wxbRestorePanel::OnTreeMarked(wxbTreeMarkedEvent& event) {
1957    csprint("Tree marked", CS_DEBUG);
1958    if (IsWorking()) {
1959       if (tree->GetSelection() == event.GetItem()) {
1960          markWhenListingDone = !markWhenListingDone;
1961       }
1962       AddPendingEvent(event);
1963       return;
1964    }
1965    SetWorking(true);
1966    CmdMark(event.GetItem(), NULL, 0);
1967    //event.Skip();
1968    tree->Refresh();
1969    SetWorking(false);
1970    if (event.GetItem().IsOk()) {
1971       int status = ((wxbTreeItemData*)tree->GetItemData(event.GetItem()))->GetMarked();
1972       treeadd->Enable(status != 1);
1973       treeremove->Enable(status != 0);
1974    }
1975 }
1976
1977 void wxbRestorePanel::OnTreeAdd(wxCommandEvent& event) {
1978    if (IsWorking()) {
1979       AddPendingEvent(event);
1980       return;
1981    }
1982    
1983    if (currentTreeItem.IsOk()) {
1984       SetWorking(true);
1985       CmdMark(currentTreeItem, NULL, 0, 1);
1986       tree->Refresh();
1987       treeadd->Enable(0);
1988       treeremove->Enable(1);
1989       SetWorking(false);
1990    }
1991 }
1992
1993 void wxbRestorePanel::OnTreeRemove(wxCommandEvent& event) {
1994    if (IsWorking()) {
1995       AddPendingEvent(event);
1996       return;
1997    }
1998    
1999    if (currentTreeItem.IsOk()) {
2000       SetWorking(true);
2001       CmdMark(currentTreeItem, NULL, 0, 0);
2002       tree->Refresh();
2003       treeadd->Enable(1);
2004       treeremove->Enable(0);
2005       SetWorking(false);
2006    }
2007 }
2008
2009 void wxbRestorePanel::OnTreeRefresh(wxCommandEvent& event) {
2010    if (IsWorking()) {
2011       AddPendingEvent(event);
2012       return;
2013    }
2014    
2015    SetWorking(true);
2016    RefreshTree();
2017    SetWorking(false);
2018 }
2019
2020 void wxbRestorePanel::OnListMarked(wxbListMarkedEvent& event) {
2021    if (IsWorking()) {
2022       AddPendingEvent(event);
2023       //event.Skip();
2024       return;
2025    }
2026    
2027    if (list->GetSelectedItemCount() == 0) {
2028       return;
2029    }
2030    
2031    SetWorking(true);  
2032    
2033    long* items = new long[list->GetSelectedItemCount()];
2034    
2035    int num = 0;
2036    
2037    long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2038    while (item != -1) {
2039       items[num] = item;
2040       num++;
2041       item = list->GetNextItem(item, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2042    }
2043      
2044    CmdMark(wxTreeItemId(), items, num);
2045    
2046    delete[] items;
2047    
2048    wxListEvent listevt;
2049    
2050    OnListChanged(listevt);
2051    
2052    event.Skip();
2053    tree->Refresh();
2054    SetWorking(false);
2055 }
2056
2057 void wxbRestorePanel::OnListActivated(wxListEvent& event) {
2058    if (IsWorking()) {
2059       AddPendingEvent(event);
2060       //event.Skip();
2061       return;
2062    }
2063    SetWorking(true);
2064    long item = event.GetIndex();
2065 //   long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_FOCUSED);
2066    if (item > -1) {
2067       wxbTreeItemData* itemdata = (wxbTreeItemData*)list->GetItemData(item);
2068       wxString name = itemdata->GetName();
2069       event.Skip();
2070
2071       wxString itemStr;
2072
2073       long cookie;
2074
2075       if (name.GetChar(name.Length()-1) == '/') {
2076          wxTreeItemId currentChild = tree->GetFirstChild(currentTreeItem, cookie);
2077
2078          while (currentChild.IsOk()) {
2079             wxString name2 = tree->GetItemText(currentChild);
2080             if (name2 == name) {
2081                //tree->UnselectAll();
2082                SetWorking(false);
2083                tree->Expand(currentTreeItem);
2084                tree->SelectItem(currentChild);
2085                //tree->Refresh();
2086                return;
2087             }
2088             currentChild = tree->GetNextChild(currentTreeItem, cookie);
2089          }
2090       }
2091    }
2092    SetWorking(false);
2093 }
2094
2095 void wxbRestorePanel::OnListChanged(wxListEvent& event) {
2096    if (IsWorking()) {
2097       AddPendingEvent(event);
2098       return;
2099    }
2100  
2101    listadd->Enable(false);
2102    listremove->Enable(false);
2103    
2104    bool marked = false;
2105    bool unmarked = false;
2106    
2107    long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2108    while (item != -1) {
2109       switch (((wxbTreeItemData*)list->GetItemData(item))->GetMarked()) {
2110       case 0:
2111          unmarked = true;
2112          break;
2113       case 1:
2114          marked = true;
2115          break;
2116       case 2:
2117          marked = true;
2118          unmarked = true;
2119          break;
2120       default:
2121          break;
2122          // Should never happen
2123       }
2124       if (marked && unmarked) break;
2125       item = list->GetNextItem(item, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2126    }
2127    
2128    listadd->Enable(unmarked);
2129    listremove->Enable(marked);
2130 }
2131
2132 void wxbRestorePanel::OnListAdd(wxCommandEvent& event) {
2133    if (IsWorking()) {
2134       AddPendingEvent(event);
2135       return;
2136    }
2137    
2138    SetWorking(true);
2139    
2140    long* items = new long[list->GetSelectedItemCount()];
2141    
2142    int num = 0;
2143    
2144    long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2145    while (item != -1) {
2146       items[num] = item;
2147       num++;
2148       item = list->GetNextItem(item, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2149    }
2150      
2151    CmdMark(wxTreeItemId(), items, num, 1);
2152    
2153    delete[] items;
2154    
2155    tree->Refresh();
2156    SetWorking(false);
2157    
2158    listadd->Enable(false);
2159    listremove->Enable(true);
2160 }
2161
2162 void wxbRestorePanel::OnListRemove(wxCommandEvent& event) {
2163    if (IsWorking()) {
2164       AddPendingEvent(event);
2165       return;
2166    }
2167    
2168    SetWorking(true);
2169    
2170    long* items = new long[list->GetSelectedItemCount()];
2171    
2172    int num = 0;
2173    
2174    long item = list->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2175    while (item != -1) {
2176       items[num] = item;
2177       num++;
2178       item = list->GetNextItem(item, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
2179    }
2180      
2181    CmdMark(wxTreeItemId(), items, num, 0);
2182    
2183    delete[] items;
2184    
2185    tree->Refresh();
2186    SetWorking(false);
2187    
2188    listadd->Enable(true);
2189    listremove->Enable(false);
2190 }
2191
2192 void wxbRestorePanel::OnListRefresh(wxCommandEvent& event) {
2193    if (IsWorking()) {
2194       AddPendingEvent(event);
2195       return;
2196    }
2197    
2198    SetWorking(true);
2199    RefreshList();
2200    SetWorking(false);
2201 }
2202
2203 void wxbRestorePanel::OnConfigUpdated(wxCommandEvent& event) {
2204    if (status == entered) {
2205       if (event.GetId() == ConfigJobName) {
2206          if (IsWorking()) {
2207             return;
2208          }
2209          SetWorking(true);
2210          UpdateFirstConfig();
2211          SetWorking(false);
2212       }
2213       else if (event.GetId() == ConfigClient) {
2214          if (IsWorking()) {
2215             return;
2216          }
2217          SetWorking(true);
2218          configPanel->Enable(false);
2219          CmdListJobs();
2220          configPanel->Enable(true);
2221          SetWorking(false);
2222       }
2223       cfgUpdated = cfgUpdated | (1 << event.GetId());
2224    }
2225    else if (status == configuring) {
2226       restorePanel->EnableApply(true);
2227       cfgUpdated = cfgUpdated | (1 << event.GetId());
2228    }
2229 }
2230
2231 void wxbRestorePanel::OnConfigOk(wxCommandEvent& WXUNUSED(event)) {
2232    if (status != configuring) return;
2233    if (IsWorking()) {
2234       return;
2235    }
2236    SetWorking(true);
2237    CmdStart();
2238    SetWorking(false);
2239 }
2240
2241 void wxbRestorePanel::OnConfigApply(wxCommandEvent& WXUNUSED(event)) {
2242    if (status != configuring) return;
2243    if (IsWorking()) {
2244       return;
2245    }
2246    SetWorking(true);
2247    CmdConfigApply();
2248    if (cfgUpdated == 0) {
2249       restorePanel->EnableApply(false);
2250    }
2251    SetWorking(false);  
2252 }
2253
2254 void wxbRestorePanel::OnConfigCancel(wxCommandEvent& WXUNUSED(event)) {
2255    if (status != configuring) return;
2256    if (IsWorking()) {
2257       return;
2258    }
2259    SetWorking(true);
2260    CmdConfigCancel();
2261    SetWorking(false);
2262 }
2263