ide;GlobalSettingsDialog; fixed some ugly code repeated multiple times.
[sdk] / ide / src / dialogs / GlobalSettingsDialog.ec
1 import "IDESettings"
2
3 // import "SelectorBar"
4 import "CompilersDetectionDialog"
5 import "ide"
6
7 FileDialog settingsFileDialog { type = selectDir, text = $"Select directory" };
8
9 FileDialog toolchainFileDialog { type = open, text = $"Open"; mayNotExist = true; };
10
11 class GlobalSettingsDialog : Window
12 {
13    autoCreate = false;
14    tabCycle = true;
15    background = formColor;
16    hasClose = true;
17    borderStyle = sizable;
18    text = $"Global Settings";
19    minClientSize = { 560, 446 };
20    nativeDecorations = true;
21
22    IDESettings ideSettings;
23    IDESettingsContainer settingsContainer;
24    String workspaceActiveCompiler;
25    
26    TabControl tabControl { this, background = formColor, anchor = { left = 8, top = 8, right = 8, bottom = 40 } };
27    
28    EditorTab editorTab { this, tabControl = tabControl };
29    CompilersTab compilersTab { this, tabControl = tabControl };
30    ProjectOptionsTab projectOptionsTab { this, tabControl = tabControl };
31    WorkspaceOptionsTab workspaceOptionsTab { this, tabControl = tabControl };
32    
33    property bool settingsModified
34    {
35       get
36       {
37          return editorTab.modifiedDocument || compilersTab.modifiedDocument ||
38                projectOptionsTab.modifiedDocument || workspaceOptionsTab.modifiedDocument;
39       }
40    }
41
42    bool OnClose(bool parentClosing)
43    {
44       if(!settingsModified || MessageBox {
45          type = okCancel, master = ide,
46          text = $"Lose Changes?",
47          contents = $"Are you sure you wish to discard changes?"
48           }.Modal() == ok)
49          return true;
50       return false;
51    }
52
53    Button cancel
54    {
55       parent = this, hotKey = escape, text = $"Cancel", id = DialogResult::cancel;
56       position = { 290, 290 }, size = { 80 };
57       anchor = { right = 8, bottom = 8 };
58       NotifyClicked = ButtonCloseDialog;
59    };
60
61    Button ok
62    {
63       parent = this, isDefault = true, text = $"OK";
64       position = { 200, 290 }, size = { 90 };
65       anchor = { right = 96, bottom = 8 };
66
67       bool NotifyClicked(Button button, int x, int y, Modifiers mods)
68       {
69          if(settingsModified)
70          {
71             bool editorSettingsChanged = false;
72             bool compilerSettingsChanged = false;
73             bool projectOptionsChanged = false;
74             bool workspaceOptionsChanged = false;
75             
76             if(editorTab.modifiedDocument)
77             {
78                if(editorTab.useFreeCaret.checked != ideSettings.useFreeCaret ||
79                      editorTab.showLineNumbers.checked != ideSettings.showLineNumbers ||
80                      editorTab.caretFollowsScrolling.checked != ideSettings.caretFollowsScrolling)
81                {
82                   ideSettings.useFreeCaret = editorTab.useFreeCaret.checked;
83                   ideSettings.showLineNumbers = editorTab.showLineNumbers.checked;
84                   ideSettings.caretFollowsScrolling = editorTab.caretFollowsScrolling.checked;
85                   editorSettingsChanged = true;
86                }
87             }
88             
89             if(compilersTab.modifiedDocument)
90             {
91                bool foundActive = false;
92                Workspace workspace = ide.workspace;
93                if(strcmp(compilersTab.compilerConfigsDir.path, ideSettings.compilerConfigsDir))
94                   ideSettings.compilerConfigsDir = compilersTab.compilerConfigsDir.path;
95                ideSettings.compilerConfigs.Free();
96                for(compiler : compilersTab.compilerConfigs)
97                {
98                   ideSettings.compilerConfigs.Add(compiler.Copy());
99                   if(!foundActive && workspace && workspace.compiler && !strcmp(workspace.compiler, compiler.name))
100                      foundActive = true;
101                }
102                if(workspace && !foundActive)
103                   workspace.compiler = defaultCompilerName;
104
105                compilerSettingsChanged = true;
106             }
107
108             if(projectOptionsTab.modifiedDocument)
109             {
110                if(strcmp(projectOptionsTab.defaultTargetDir.path, ideSettings.projectDefaultTargetDir)
111                      || strcmp(projectOptionsTab.defaultIntermediateObjDir.path, ideSettings.projectDefaultIntermediateObjDir))
112                {
113                   ideSettings.projectDefaultTargetDir = projectOptionsTab.defaultTargetDir.path;
114                   ideSettings.projectDefaultIntermediateObjDir = projectOptionsTab.defaultIntermediateObjDir.path;
115                   projectOptionsChanged = true;
116                }
117             }
118
119             if(workspaceOptionsTab.modifiedDocument)
120             {
121                DataRow row = workspaceOptionsTab.defaultCompilerDropBox.currentRow;
122                if(row && row.string)
123                {
124                   if(!ideSettings.defaultCompiler || strcmp(row.string, ideSettings.defaultCompiler))
125                   {
126                      ideSettings.defaultCompiler = row.string;
127                      workspaceOptionsChanged = true;
128                   }
129                }
130             }
131
132             settingsContainer.Save();
133
134             if(compilerSettingsChanged)
135                OnGlobalSettingChange(GlobalSettingsChange::compilerSettings);
136             if(editorSettingsChanged)
137                OnGlobalSettingChange(GlobalSettingsChange::editorSettings);
138             if(projectOptionsChanged)
139                OnGlobalSettingChange(GlobalSettingsChange::projectOptions);
140
141             editorTab.modifiedDocument = false;
142             compilersTab.modifiedDocument = false;
143             projectOptionsTab.modifiedDocument = false;
144             workspaceOptionsTab.modifiedDocument = false;
145          }
146          
147          Destroy(DialogResult::ok);
148          return true;
149       }
150    };
151
152    /*
153    void Temp()
154    {
155       DirTypes c;
156       for(c = 0; c < DirTypes::enumSize; c++)
157       {
158          CompilerDir compilerDir;
159
160          for(systemDir : ideSettings.systemDirs[c])
161          {
162             compilerDir = CompilerDir { type = c; compilerConfig = null; path = CopyString(systemDir) };
163             dirs.Add(compilerDir);
164          }
165
166          row = compilersTab.dirs[c].AddRow();
167          row.SetData(null, "");
168          compilersTab.dirs[c].currentRow = compilersTab.dirs[c].firstRow;
169          compilersTab.dirs[c].modifiedDocument = false;
170       }
171    }
172    */
173
174    bool OnCreate()
175    {
176       CompilerConfig activateCompiler = null;
177       CompilerConfig readonlyCompiler = null;
178
179       // EditorTab
180       editorTab.useFreeCaret.checked = ideSettings.useFreeCaret;
181       editorTab.showLineNumbers.checked = ideSettings.showLineNumbers;
182       editorTab.caretFollowsScrolling.checked = ideSettings.caretFollowsScrolling;
183
184       // CompilersTab
185       if(workspaceActiveCompiler)
186       {
187          for(compiler : ideSettings.compilerConfigs)
188          {
189             if(!activateCompiler && !strcmp(workspaceActiveCompiler, compiler.name))
190                activateCompiler = compiler;
191             if(!readonlyCompiler && compiler.readOnly)
192                readonlyCompiler = compiler;
193             if(activateCompiler && readonlyCompiler)
194                break;
195          }
196       }
197       if(!activateCompiler && readonlyCompiler)
198          activateCompiler = readonlyCompiler;
199       if(!activateCompiler && ideSettings.compilerConfigs.count)
200          activateCompiler = ideSettings.compilerConfigs[0];
201       
202       for(compiler : ideSettings.compilerConfigs)
203          compilersTab.AddCompiler(compiler.Copy(), compiler == activateCompiler);
204       compilersTab.compilerConfigsDir.path = ideSettings.compilerConfigsDir;
205
206       // ProjectOptionsTab
207       projectOptionsTab.defaultTargetDir.path = ideSettings.projectDefaultTargetDir;
208       projectOptionsTab.defaultIntermediateObjDir.path = ideSettings.projectDefaultIntermediateObjDir;
209       
210       return true;
211    }
212
213    void OnDestroy()
214    {
215       editorTab.modifiedDocument = false;
216       compilersTab.modifiedDocument = false;
217       compilersTab.dirsTab.modifiedDocument = false;
218       compilersTab.toolchainTab.modifiedDocument = false;
219       compilersTab.optionsTab.modifiedDocument = false;
220       compilersTab.activeCompiler = null;
221       compilersTab.compilerConfigs.Free();
222       compilersTab.compilerSelector.Clear();
223       projectOptionsTab.modifiedDocument = false;
224       workspaceOptionsTab.modifiedDocument = false;
225    }
226
227    virtual void OnGlobalSettingChange(GlobalSettingsChange globalSettingsChange);
228 }
229
230 class EditorTab : GlobalSettingsSubTab
231 {
232    background = formColor;
233    text = $"Editor";
234
235    Button useFreeCaret
236    {
237       this, text = $"Move code editor caret freely past end of line", position = { 16, 68 }, isCheckbox = true;
238       NotifyClicked = NotifyClickedModifiedDocument;
239    };
240
241    Button caretFollowsScrolling
242    {
243       this, text = $"Keep caret visible (move along) when scrolling", position = { 16, 88 }, isCheckbox = true;
244       NotifyClicked = NotifyClickedModifiedDocument;
245    };
246
247    Button showLineNumbers
248    {
249       this, text = $"Show line numbers in code editor", position = { 16, 108 }, isCheckbox = true;
250       NotifyClicked = NotifyClickedModifiedDocument;
251    };
252
253    bool NotifyClickedModifiedDocument(Button button, int x, int y, Modifiers mods)
254    {
255       modifiedDocument = true;
256       return true;
257    }
258 }
259
260 static void DrawStipple(Surface surface, Size clientSize)
261 {
262    int x1 = 0;
263    int y1 = 0;
264    int x2 = clientSize.w - 1;
265    int y2 = clientSize.h - 1;
266    if((x2 - x1) & 1) x2--;
267    if((y2 - y1) & 1) y2--;
268
269    surface.LineStipple(0x5555);
270    surface.Rectangle(x1, y1, x2, y2);
271    surface.LineStipple(0);            
272 }
273
274 class CompilersTab : GlobalSettingsSubTab
275 {
276    background = formColor;
277    text = $"Compilers";
278
279    Label compilerConfigsDirLabel { this, position = { 8, 12 }, labeledWindow = compilerConfigsDir, tabCycle = false, inactive = true };
280    PathBox compilerConfigsDir
281    {
282       this, anchor = { left = 210, top = 8, right = 8 };
283       text = $"Compiler Configurations Directory", browseDialog = settingsFileDialog, NotifyModified = NotifyModifiedDocument;
284    };
285
286    SelectorBar compilerSelector
287    {
288       this, text = $"Compiler Configurations:", anchor = { left = 148, top = 38, right = 99 }; size = { 0, 26 };
289       opacity = 0;
290       direction = horizontal, scrollable = true;
291
292       bool OnKeyDown(Key key, unichar ch)
293       {
294          if(key == insert)
295          {
296             ((CompilersTab)parent).createCompiler.NotifyClicked(parent, ((CompilersTab)parent).createCompiler, 0, 0, 0);
297             return false;
298          }
299          else if(key == del)
300          {
301             ((CompilersTab)parent).deleteCompiler.NotifyClicked(parent, ((CompilersTab)parent).deleteCompiler, 0, 0, 0);
302             return false;
303          }
304          return SelectorBar::OnKeyDown(key, ch);
305       }
306       
307       bool OnActivate(bool active, Window previous, bool * goOnWithActivation, bool direct)
308       {
309          ((CompilersTab)master).labelCompilers.Update(null);
310          return true;
311       }
312
313       bool OnPostCreate()
314       {
315          CompilersTab compilers = (CompilersTab)parent;
316          SelectorBar::OnPostCreate();
317          if(compilers.selectedButton)
318          {
319             Button sb = compilers.selectedButton;
320             sb.Activate();
321             sb.checked = true;
322             // Why was this being set to null? On going back to compilers the 2nd time, the selectedButton was lost and so was not made visible...
323             // compilers.selectedButton = null;
324          }
325          return true;
326       }
327    };
328
329    TabControl tabControl { this, background = formColor, anchor = { left = 8, top = 68, right = 8, bottom = 8 } };
330    
331    CompilerDirectoriesTab dirsTab { this, tabControl = tabControl };
332    CompilerToolchainTab toolchainTab { this, tabControl = tabControl };
333    CompilerEnvironmentTab environmentTab { this, tabControl = tabControl };
334    CompilerOptionsTab optionsTab { this, tabControl = tabControl };
335
336    List<CompilerConfig> compilerConfigs { };
337    CompilerConfig activeCompiler;
338
339    Label labelCompilers
340    {
341       this, anchor = { left = 8, top = 44 }, labeledWindow = compilerSelector;
342
343       void OnRedraw(Surface surface)
344       {
345          Label::OnRedraw(surface);
346          if(labeledWindow.active)
347             DrawStipple(surface, clientSize);
348       }
349    };
350
351    void FindUniqueCompilerName(char * baseName, CompilerConfig compiler/*, bool startWithNumber*/, char * output)
352    {
353       int num = 0;
354       char tmp[MAX_F_STRING];
355       /*if(startWithNumber)
356          sprintf(tmp, "%s%d", baseName, num);
357       else*/
358          strcpy(tmp, baseName);
359       while(true)
360       {
361          CompilerConfig matchingCompiler = null;
362          for(c : compilerConfigs)
363          {     // TOFIX: Error when omitting these brackets, c not found
364             if((!compiler || c != compiler) && c.name && !strcmp(c.name, tmp))
365             {
366                matchingCompiler = c;
367                break;
368             }
369          }
370          if(matchingCompiler)
371          {
372             num++;
373             sprintf(tmp, "%s%d", baseName, num);
374          }
375          else
376             break;
377       }
378       strcpy(output, tmp);
379    }
380
381    Button createCompiler
382    {
383       parent = this, bevelOver = true, inactive = true;
384       size = { 22, 22 };
385       anchor = { top = 40, right = 77 };
386       hotKey = altC, bitmap = BitmapResource { fileName = ":actions/docNew.png", alphaBlend = true };
387
388       bool NotifyClicked(Button button, int x, int y, Modifiers mods)
389       {
390          char compilerName[MAX_F_STRING];
391          CompilerConfig newCompiler;
392          FindUniqueCompilerName("New Compiler", null, compilerName);
393          newCompiler = MakeDefaultCompiler(compilerName, false);
394          AddCompiler(newCompiler, true);
395          modifiedDocument = true;
396          return true;
397       }
398    };
399    Button detectCompiler
400    {
401       parent = this, bevelOver = true, inactive = true;
402       size = { 22, 22 };
403       anchor = { top = 40, right = 54 };
404       hotKey = altC, bitmap = BitmapResource { fileName = ":actions/attach.png", alphaBlend = true };
405
406       bool NotifyClicked(Button b, int x, int y, Modifiers mods)
407       {
408          CompilersDetectionDialog compilersDetectionDialog
409          {
410             dialog.parent;
411
412          };
413          if(compilersDetectionDialog.Modal())
414          {
415             if(compilersDetectionDialog.selectedCompilerType)
416             {
417                char uniqueName[MAX_F_STRING];
418                CompilerConfig newCompiler = compilersDetectionDialog.compilerConfig;
419                FindUniqueCompilerName(newCompiler.name, null, uniqueName);
420                newCompiler.name = uniqueName;
421                AddCompiler(newCompiler, true);
422                modifiedDocument = true;
423             }
424          }
425          return true;
426       }
427    };
428    Button duplicateCompiler
429    {
430       parent = this, bevelOver = true, inactive = true;
431       size = { 22, 22 };
432       anchor = { top = 40, right = 31 };
433       hotKey = altU, bitmap = BitmapResource { fileName = ":actions/editCopy.png", alphaBlend = true };
434
435       bool NotifyClicked(Button button, int x, int y, Modifiers mods)
436       {
437          char copyName[MAX_F_STRING];
438          CompilerConfig copyCompiler = activeCompiler.Copy();
439          FindUniqueCompilerName(copyCompiler.name, null, copyName);
440          copyCompiler.readOnly = false;
441          copyCompiler.name = copyName;
442          AddCompiler(copyCompiler, true);
443          modifiedDocument = true;
444          return true;
445       }
446    };
447    Button deleteCompiler
448    {
449       parent = this, bevelOver = true, inactive = true;
450       size = { 22, 22 };
451       anchor = { top = 40, right = 8 };
452       hotKey = altD, bitmap = BitmapResource { fileName = ":actions/delete2.png", alphaBlend = true };
453
454       bool NotifyClicked(Button button, int x, int y, Modifiers mods)
455       {
456          if(activeCompiler)
457          {
458             CompilerConfig compilerToDelete = activeCompiler;
459             String title = PrintString($"Delete ", compilerToDelete.name, $" Compiler Configuration");
460             String msg = PrintString($"Are you sure you wish to delete the ", compilerToDelete.name, $" compiler configuration?");
461             if(MessageBox { type = okCancel, text = title, contents = msg }.Modal() == ok)
462             {
463                SelectorButton button = compilerSelector.FindButtonByID((int)compilerToDelete);
464                if(button)
465                   compilerSelector.RemoveButton(button);
466                //DeleteCompiler(compilerToDelete);
467                {
468                   Iterator<CompilerConfig> it { compilerConfigs };
469                   if(it.Find(compilerToDelete))
470                      compilerConfigs.Delete(it.pointer);
471                }
472                modifiedDocument = true;
473             }
474             delete title;
475             delete msg;
476          }
477          return true;
478       }
479    };
480
481    void AddCompiler(CompilerConfig compiler, bool load)
482    {
483       SelectorButton selectButton;
484       if(compiler.readOnly)
485       {
486          SelectorButton button
487          {
488             compilerSelector, master = this, text = compiler.name, id = (int)compiler;
489             NotifyClicked = CompilerClicked;
490          };
491          selectButton = button;
492       }
493       else
494       {
495          EditableSelectorButton button
496          {
497             compilerSelector, master = this, renameable = true, text = compiler.name, id = (int)compiler;
498             NotifyClicked = CompilerClicked;
499
500             bool OnRename(EditableSelectorButton button, char ** oldName, char ** newName)
501             {
502                if(*newName && (*newName)[0])
503                {
504                   char compilerName[MAX_F_STRING];
505                   FindUniqueCompilerName(*newName, activeCompiler, compilerName);
506                   if(strcmp(*newName, compilerName))
507                   {
508                      delete *newName;
509                      *newName = CopyString(compilerName);
510                   }
511                   activeCompiler.name = compilerName;
512                   modifiedDocument = true;
513                   return true;
514                }
515                return false;
516             }
517          };
518          selectButton = (SelectorButton)button;
519       }
520       compilerConfigs.Add(compiler);
521       if(load)
522       {
523          LoadCompiler(compiler);
524          selectedButton = selectButton;
525          compilerSelector.Select(selectedButton);
526       }
527    }
528    SelectorButton selectedButton;
529
530    void LoadCompiler(CompilerConfig compiler)
531    {
532       bool modified = modifiedDocument;
533       activeCompiler = compiler;
534
535       dirsTab.Load();
536       toolchainTab.Load();
537       environmentTab.Load();
538       optionsTab.Load();
539
540       // Restore original modifiedDocument
541       modifiedDocument = modified;
542
543       deleteCompiler.disabled = compiler.readOnly;
544    }
545
546    bool CompilerClicked(Button clickedButton, int x, int y, Modifiers mods)
547    {
548       if(!eClass_IsDerived(clickedButton._class, class(EditableSelectorButton)) || !((EditableSelectorButton)clickedButton).editBox)
549       {
550          LoadCompiler((CompilerConfig)clickedButton.id);
551          selectedButton = (SelectorButton)clickedButton;
552       }
553       return true;
554    }
555
556    bool NotifyModifiedDocument(PathBox pathBox)
557    {
558       modifiedDocument = true;
559       return true;
560    }
561 }
562
563 Array<String> displayDirectoryNames
564 { [
565    $"Include Files",
566    $"Library Files",
567    $"Executable Files"
568 ] };
569
570 class CompilerDirectoriesTab : CompilersSubTab
571 {
572    background = formColor;
573    text = $"Directories";
574
575    Button dirTypeTglBtn[DirTypes];
576    DirectoriesBox dirs[DirTypes], currentDirs;
577
578    ~CompilerDirectoriesTab()
579    {
580       DirTypes c;
581       for(c = 0; c < DirTypes::enumSize; c++)
582       {
583          delete dirs[c];
584          delete dirTypeTglBtn[c];
585       }
586    }
587    CompilerDirectoriesTab()
588    {
589       DirTypes c;
590       for(c = 0; c < DirTypes::enumSize; c++)
591       {
592          dirs[c] = DirectoriesBox
593          {
594             this;//, alwaysHighLight = true
595             anchor = { left = 8, top = 8, right = 8, bottom = 8 };
596             id = c;
597
598    /*   MAKE SURE THINGS ARE DONE PROPERLY IN THE NEW DIRECTORIES BOX WHEN BROWSING FOR A DIR?
599             settingsFileDialog.filePath = directory;
600          if(settingsFileDialog.Modal())
601             row.SetData(null, (s = CopyUnixPath(settingsFileDialog.filePath)));
602    */
603
604             bool NotifyModified(DirectoriesBox dirsBox)
605             {
606                CompilerConfig compiler = loadedCompiler;
607                if(compiler)
608                {
609                   DirTypes dirType = (DirTypes)dirsBox.id;
610                   if(dirType == includes)
611                      compiler.includeDirs = dirsBox.strings;
612                   else if(dirType == libraries)
613                      compiler.libraryDirs = dirsBox.strings;
614                   else if(dirType == executables)
615                      compiler.executableDirs = dirsBox.strings;
616
617                   compilersTab.modifiedDocument = true;
618                }
619                return true;
620             }
621          };
622          incref dirs[c];
623          
624          if(c)
625             dirs[c].visible = false;
626          
627          // (width) Should be 324 for text...
628          //field[c] = { dataType = class(char *), editable = true };
629          //dirs[c].AddField(field[c]);
630
631          {
632          int v = (int)c * 100 + 8;
633          dirTypeTglBtn[c] = Button
634          {
635             this, inactive = true, text = displayDirectoryNames[c], bevelOver = true, isRadio = true, bitmap = null;
636             stayOnTop = true;
637             id = c;
638             size = { 99, 20 };
639             anchor = { left = v, top = 8 }; // ((int)c) * 100 + 8
640
641             bool NotifyClicked(Button button, int x, int y, Modifiers mods)
642             {
643                DirTypes dirType = (DirTypes)button.id;
644                currentDirs.visible = false;
645                dirs[dirType].visible = true;
646                currentDirs = dirs[dirType];
647                return true;
648             }
649          };
650          incref dirTypeTglBtn[c];
651
652          if(c == includes)
653             dirTypeTglBtn[c].hotKey = altI;
654          else if(c == libraries)
655             dirTypeTglBtn[c].hotKey = altL;
656          else if(c == executables)
657             dirTypeTglBtn[c].hotKey = altE;
658          }
659       }   
660       currentDirs = dirs[includes];
661       dirTypeTglBtn[includes].checked = true;
662       return true;
663    }
664
665    void Load()
666    {
667       CompilerConfig compiler = loadedCompiler;
668       if(compiler)
669       {
670          dirs[includes].strings = compiler.includeDirs;
671          dirs[libraries].strings = compiler.libraryDirs;
672          dirs[executables].strings = compiler.executableDirs;
673          dirs[includes].list.scroll = { 0, 0 };
674          dirs[libraries].list.scroll = { 0, 0 };
675          dirs[executables].list.scroll = { 0, 0 };
676       }
677    }
678 }
679
680 class CompilerToolchainTab : CompilersSubTab
681 {
682    background = formColor;
683    text = $"Toolchain";
684
685    Label ecpLabel { this, position = { 8, 12 }, labeledWindow = ecp, tabCycle = false, inactive = true };
686    PathBox ecp
687    {
688       this, anchor = { left = 120, top = 8, right = 8 };
689       text = $"eC Precompiler", browseDialog = toolchainFileDialog, NotifyModified = NotifyModifiedDocument;
690    };
691    Label eccLabel { this, position = { 8, 38 }, labeledWindow = ecc, tabCycle = false, inactive = true };
692    PathBox ecc
693    {
694       this, anchor = { left = 120, top = 34, right = 8 };
695       text = $"eC Compiler", browseDialog = toolchainFileDialog, NotifyModified = NotifyModifiedDocument;
696    };
697    Label ecsLabel { this, position = { 8, 64 }, labeledWindow = ecs, tabCycle = false, inactive = true };
698    PathBox ecs
699    {
700       this, anchor = { left = 120, top = 60, right = 8 };
701       text = $"eC Symbol Loader", browseDialog = toolchainFileDialog, NotifyModified = NotifyModifiedDocument;
702    };
703    Label earLabel { this, position = { 8, 90 }, labeledWindow = ear, tabCycle = false, inactive = true };
704    PathBox ear
705    {
706       this, anchor = { left = 120, top = 86, right = 8 };
707       text = $"Ecere Archiver", browseDialog = toolchainFileDialog, NotifyModified = NotifyModifiedDocument;
708    };
709    Label cppLabel { this, position = { 8, 116 }, labeledWindow = cpp, tabCycle = false, inactive = true };
710    PathBox cpp
711    {
712       this, anchor = { left = 120, top = 112, right = 8 };
713       text = $"C Preprocessor", browseDialog = toolchainFileDialog, NotifyModified = NotifyModifiedDocument;
714    };
715    Label ccLabel { this, position = { 8, 142 }, labeledWindow = cc, tabCycle = false, inactive = true };
716    PathBox cc
717    {
718       this, anchor = { left = 120, top = 138, right = 8 };
719       text = $"C Compiler", browseDialog = toolchainFileDialog, NotifyModified = NotifyModifiedDocument;
720    };
721    Label cxxLabel { this, position = { 8, 168 }, labeledWindow = cxx, tabCycle = false, inactive = true };
722    PathBox cxx
723    {
724       this, anchor = { left = 120, top = 164, right = 8 };
725       text = $"C++ Compiler", browseDialog = toolchainFileDialog, NotifyModified = NotifyModifiedDocument;
726    };
727    Label makeLabel { this, position = { 8, 194 }, labeledWindow = make, tabCycle = false, inactive = true };
728    PathBox make
729    {
730       this, anchor = { left = 120, top = 190, right = 8 };
731       text = $"GNU Make", browseDialog = toolchainFileDialog, NotifyModified = NotifyModifiedDocument;
732    };
733    Label execPrefixLabel { this, position = { 8, 220 }, labeledWindow = execPrefix, tabCycle = false, inactive = true };
734    PathBox execPrefix
735    {
736       this, anchor = { left = 120, top = 216, right = 8 };
737       text = $"Execution Prefix", browseDialog = toolchainFileDialog, NotifyModified = NotifyModifiedDocument;
738    };
739
740    bool NotifyModifiedDocument(PathBox pathBox)
741    {
742       CompilerConfig compiler = loadedCompiler;
743       if(compiler)
744       {
745          if(pathBox == ecp)
746             compiler.ecpCommand = pathBox.slashPath;
747          else if(pathBox == ecc)
748             compiler.eccCommand = pathBox.slashPath;
749          else if(pathBox == ecs)
750             compiler.ecsCommand = pathBox.slashPath;
751          else if(pathBox == ear)
752             compiler.earCommand = pathBox.slashPath;
753          else if(pathBox == cpp)
754             compiler.cppCommand = pathBox.slashPath;
755          else if(pathBox == cc)
756             compiler.ccCommand = pathBox.slashPath;
757          else if(pathBox == cxx)
758             compiler.cxxCommand = pathBox.slashPath;
759          else if(pathBox == make)
760             compiler.makeCommand = pathBox.slashPath;
761          else if(pathBox == execPrefix)
762             compiler.execPrefixCommand = pathBox.slashPath;
763          modifiedDocument = true;
764          compilersTab.modifiedDocument = true;
765       }
766       return true;
767    }
768
769    void Load()
770    {
771       CompilerConfig compiler = loadedCompiler;
772       if(compiler)
773       {
774          bool disabled = compiler.readOnly;
775          bool isVC = compiler.type.isVC;
776          ecp.path = compiler.ecpCommand;
777          ecc.path = compiler.eccCommand;
778          ecs.path = compiler.ecsCommand;
779          ear.path = compiler.earCommand;
780          cpp.path = compiler.cppCommand;
781          cc.path = compiler.ccCommand;
782          cxx.path = compiler.cxxCommand;
783          make.path = compiler.makeCommand;
784          execPrefix.path = compiler.execPrefixCommand;
785
786          ecpLabel.disabled = ecp.disabled = disabled;
787          eccLabel.disabled = ecc.disabled = disabled;
788          ecsLabel.disabled = ecs.disabled = disabled;
789          earLabel.disabled = ear.disabled = disabled;
790          cppLabel.disabled = cpp.disabled = isVC || disabled;
791          cxxLabel.disabled = cxx.disabled = isVC || disabled;
792          ccLabel.disabled = cc.disabled = isVC || disabled;
793          makeLabel.disabled = make.disabled = disabled;
794          execPrefixLabel.disabled = execPrefix.disabled = disabled;
795       }
796       modifiedDocument = false;
797    }
798 }
799
800 class CompilerEnvironmentTab : CompilersSubTab
801 {
802    background = formColor;
803    text = $"Environment";
804
805    Label labelEnvVars { envVars, labeledWindow = envVars, position = { 0, 6 }; };
806    NamedStringsBox envVars
807    {
808       this, size = { 290, 22 }, anchor = { left = 8, top = 8, right = 8, bottom = 8 };
809       text = $"Environment Variables", hotKey = altE; //, option = OPTION(postbuildCommands);
810
811       bool NotifyModified(NamedStringsBox stringsBox)
812       {
813          CompilerConfig compiler = loadedCompiler;
814          if(compiler)
815          {
816             compiler.environmentVars = stringsBox.namedStrings;
817             modifiedDocument = true;
818             compilersTab.modifiedDocument = true;
819          }
820          return true;
821       }
822    };
823
824    CompilerEnvironmentTab()
825    {
826    }
827
828    void Load()
829    {
830       CompilerConfig compiler = loadedCompiler;
831       if(compiler)
832       {
833          envVars.namedStrings = compiler.environmentVars;
834
835          modifiedDocument = false;
836       }
837    }
838 }
839
840 class CompilerOptionsTab : CompilersSubTab
841 {
842    background = formColor;
843    text = $"Options";
844
845    Label labelTargetPlatform { this, position = { 8, 12 }, labeledWindow = targetPlatform };   // TOCHECK: nameless instances dissapear when selecting tabs?
846    DropBox targetPlatform
847    {
848       this, position = { 110, 8 }, size = { 160 };
849       text = $"Target Platform", hotKey = altT;
850       bool NotifySelect(DropBox dropBox, DataRow row, Modifiers mods)
851       {
852          CompilerConfig compiler = loadedCompiler;
853          if(compiler && row)
854          {
855             compiler.targetPlatform = (Platform)row.tag;
856             modifiedDocument = true;
857             compilersTab.modifiedDocument = true;
858          }
859          return true;
860       }
861    };
862
863    int numJobs;
864    Label numJobsLabel { this, position = { 8, 40 }, labeledWindow = numJobsBox };
865    DataBox numJobsBox
866    {
867       this, text = $"Number of parallel build jobs", hotKey = altJ, borderStyle = deep;
868       position = { 184, 36 }, size = { 80, 20 }, type = class(int), data = &numJobs;
869
870       bool OnKeyDown(Key key, unichar ch)
871       {
872          if((SmartKey)key == enter)
873          {  
874             DataBox::OnKeyDown(key, ch);
875             return true;
876          }
877          else
878             return DataBox::OnKeyDown(key, ch);
879       }
880       
881       bool OnActivate(bool active, Window previous, bool * goOnWithActivation, bool direct)
882       {
883          if(!active)
884          {
885             if(!SaveData())
886                Refresh();
887          }
888          return true;
889       }
890
891       bool NotifyChanged(bool closingDropDown)
892       {
893          CompilerConfig compiler = loadedCompiler;
894          if(compiler)
895          {
896             compiler.numJobs = numJobs;
897             modifiedDocument = true;
898             compilersTab.modifiedDocument = true;
899          }
900          return true;
901       }
902    };
903
904    Button ccacheEnabled
905    {
906       this, text = $"Use ccache", hotKey = altC, position = { 8, 68 };
907       isCheckbox = true;
908
909       bool NotifyClicked(Button button, int x, int y, Modifiers mods)
910       {
911          CompilerConfig compiler = loadedCompiler;
912          if(compiler)
913          {
914             compiler.ccacheEnabled = button.checked;
915             modifiedDocument = true;
916             compilersTab.modifiedDocument = true;
917          }
918          return true;
919       }
920    };
921
922    Button distccEnabled
923    {
924       this, text = $"Use distcc", hotKey = altD, position = { 158, 68 };
925       isCheckbox = true;
926
927       bool NotifyClicked(Button button, int x, int y, Modifiers mods)
928       {
929          CompilerConfig compiler = loadedCompiler;
930          if(compiler)
931          {
932             distccHosts.disabled = !button.checked;
933             compiler.distccEnabled = button.checked;
934             modifiedDocument = true;
935             compilersTab.modifiedDocument = true;
936          }
937          return true;
938       }
939    };
940
941    Label distccHostsLabel { this, position = { 8, 96 }, labeledWindow = distccHosts };
942    EditBox distccHosts
943    {
944       this, text = $"distcc hosts", hotKey = altH;
945       position = { 88, 92 }, size = { 300 };
946
947       bool NotifyModified(EditBox editBox)
948       {
949          CompilerConfig compiler = loadedCompiler;
950          if(compiler)
951          {
952             compiler.distccHosts = editBox.contents;
953             modifiedDocument = true;
954             compilersTab.modifiedDocument = true;
955          }
956          return true;
957       }
958    }
959
960    Label lblPrepDefs { this, position = { 8, 124 }, labeledWindow = prepDefs };
961    StringListBox prepDefs
962    {
963       this, text = $"Preprocessor directives:", hotKey = altP;
964       position = { 148, 124 }, size = { 300 };
965
966       bool NotifyModified(EditBox editBox)
967       {
968          if(loadedCompiler)
969          {
970             CompilerConfig compiler = loadedCompiler;
971             compiler.prepDirectives = ((StringListBox)editBox).strings;
972             modifiedDocument = true;
973             compilersTab.modifiedDocument = true;
974          }
975          return true;
976       }
977    }
978
979    Label lblLinkerFlags { this, position = { 8, 152 }, labeledWindow = linkerFlags };
980    StringListBox linkerFlags
981    {
982       this, text = $"Additional Linker flags:", hotKey = altL;
983       position = { 148, 152 }, size = { 300 };
984
985       bool NotifyModified(EditBox editBox)
986       {
987          if(loadedCompiler)
988          {
989             CompilerConfig compiler = loadedCompiler;
990             compiler.linkerFlags = ((StringListBox)editBox).strings;
991             modifiedDocument = true;
992             compilersTab.modifiedDocument = true;
993          }
994          return true;
995       }
996    }
997
998    Label lblExcludedLibraries { this, position = { 8, 180 }, labeledWindow = excludedLibraries };
999    StringListBox excludedLibraries
1000    {
1001       this, text = $"Libraries to exclude:", hotKey = altX;
1002       position = { 148, 180 }, size = { 300 };
1003
1004       bool NotifyModified(EditBox editBox)
1005       {
1006          if(loadedCompiler)
1007          {
1008             CompilerConfig compiler = loadedCompiler;
1009             compiler.excludeLibs = ((StringListBox)editBox).strings;
1010             modifiedDocument = true;
1011             compilersTab.modifiedDocument = true;
1012          }
1013          return true;
1014       }
1015    }
1016
1017    Button supportsBitDepth
1018    {
1019       this, text = $"Supports forcing bit depth", hotKey = altD, position = { 8, 208 };
1020       isCheckbox = true;
1021
1022       bool NotifyClicked(Button button, int x, int y, Modifiers mods)
1023       {
1024          CompilerConfig compiler = loadedCompiler;
1025          if(compiler)
1026          {
1027             compiler.supportsBitDepth = button.checked;
1028             modifiedDocument = true;
1029             compilersTab.modifiedDocument = true;
1030          }
1031          return true;
1032       }
1033    };
1034
1035    CompilerOptionsTab()
1036    {
1037       Platform p;
1038       DataRow row;
1039       for(p = (Platform)1; p < Platform::enumSize; p++)
1040       {
1041          row = targetPlatform.AddRow();
1042          row.tag = p;
1043          row.string = p;
1044       }
1045    }
1046
1047    void Load()
1048    {
1049       CompilerConfig compiler = loadedCompiler;
1050       if(compiler)
1051       {
1052          bool disabled = compiler.readOnly;
1053          targetPlatform.currentRow = targetPlatform.FindRow(compiler.targetPlatform);
1054          numJobs = compiler.numJobs;
1055          numJobsBox.Refresh();
1056          ccacheEnabled.checked = compiler.ccacheEnabled;
1057          distccEnabled.checked = compiler.distccEnabled;
1058          distccHosts.disabled = !compiler.distccEnabled;
1059          distccHosts.contents = compiler.distccHosts;
1060          supportsBitDepth.checked = compiler.supportsBitDepth;
1061          prepDefs.strings = compiler.prepDirectives;
1062          excludedLibraries.strings = compiler.excludeLibs;
1063          linkerFlags.strings = compiler.linkerFlags;
1064          
1065          labelTargetPlatform.disabled = disabled;
1066          targetPlatform.disabled = disabled;
1067
1068       }
1069       modifiedDocument = false;
1070    }
1071 }
1072
1073 class CompilersSubTab : Tab
1074 {
1075    property CompilersTab compilersTab
1076    {
1077       get
1078       {
1079          CompilersTab tab = (CompilersTab)master;
1080          while(tab && tab._class != class(CompilersTab))
1081             tab = (CompilersTab)tab.master;
1082          return tab;
1083       }
1084    };
1085
1086    property CompilerConfig loadedCompiler
1087    {
1088       get
1089       {
1090          CompilersTab tab = compilersTab;
1091          return tab ? tab.activeCompiler : null;
1092       }
1093    };
1094 }
1095
1096 class ProjectOptionsTab : GlobalSettingsSubTab
1097 {
1098    background = formColor;
1099    text = $"Project";
1100
1101    Label defaultTargetDirLabel { this, position = { 8, 34 }, labeledWindow = defaultTargetDir };
1102    PathBox defaultTargetDir
1103    {
1104       this, size = { 160, 21 }, position = { 8, 52 }, anchor = { left = 8, top = 52, right = 8 };
1105       text = $"Default Target Directory", hotKey = altT;
1106
1107       bool NotifyModified(PathBox editBox)
1108       {
1109          modifiedDocument = true;
1110          return true;
1111       }
1112    };
1113
1114    Label defaultIntermediateObjDirLabel { this, position = { 8, 78 }, labeledWindow = defaultIntermediateObjDir };
1115    PathBox defaultIntermediateObjDir
1116    {
1117       this, size = { 160, 21 }, position = { 8, 96 }, anchor = { left = 8, top = 96, right = 8 };
1118       text = $"Default Intermediate Objects Directory", hotKey = altI;
1119
1120       bool NotifyModified(PathBox editBox)
1121       {
1122          modifiedDocument = true;
1123          return true;
1124       }
1125    };
1126 }
1127
1128 // COMPILER TOFIX: if class GlobalSettingsSubTab is after class WorkspaceOptionsTab the OnPostCreate 
1129 //                 of WorkspaceOptionsTab will *not* be called!
1130 class GlobalSettingsSubTab : Tab
1131 {
1132    property GlobalSettingsDialog dialog
1133    {
1134       get
1135       {
1136          GlobalSettingsDialog dialog = (GlobalSettingsDialog)master;
1137          while(dialog && dialog._class != class(GlobalSettingsDialog))
1138             dialog = (GlobalSettingsDialog)dialog.master;
1139          return dialog;
1140       }
1141    };
1142 }
1143
1144 class WorkspaceOptionsTab : GlobalSettingsSubTab
1145 {
1146    background = formColor;
1147    text = $"Workspace";
1148
1149    Label defaultCompilerLabel { this, position = { 8, 14 }, labeledWindow = defaultCompilerDropBox };
1150    DropBox defaultCompilerDropBox
1151    {
1152       this, position = { 140, 8 }, size = { 220 };
1153       text = $"Default Compiler", hotKey = altA;
1154
1155       bool NotifySelect(DropBox dropBox, DataRow row, Modifiers mods)
1156       {
1157          modifiedDocument = true;
1158          return true;
1159       }
1160    };
1161
1162    bool OnCreate()
1163    {
1164       GlobalSettingsDialog dialog = this.dialog;
1165       if(dialog && dialog.compilersTab.compilerConfigs && dialog.ideSettings)
1166       {
1167          DataRow row;
1168          for(compiler : dialog.ideSettings.compilerConfigs)
1169          {
1170             row = defaultCompilerDropBox.AddString(compiler.name);
1171             if(dialog.ideSettings.defaultCompiler && dialog.ideSettings.defaultCompiler[0] && 
1172                   !strcmp(compiler.name, dialog.ideSettings.defaultCompiler))
1173                defaultCompilerDropBox.currentRow = row;
1174          }
1175          if(!defaultCompilerDropBox.currentRow && defaultCompilerDropBox)
1176             defaultCompilerDropBox.currentRow = defaultCompilerDropBox.firstRow;
1177       }
1178       return true;
1179    }
1180
1181    void OnDestroy()
1182    {
1183       // TOFIX: The selection will be lost upon changing tab...
1184       // Should either warn, or leave it modified and put in place
1185       // checks to save/find the compiler by name
1186       defaultCompilerDropBox.Clear();
1187       modifiedDocument = false;
1188    }
1189 }
1190
1191 //static define app = ((GuiApplication)__thisModule);