3 (* THIS IS TEXT COPY OF BlackBox 1.6-rc6 System/Mod/Dialog.odc *)
6 IMPORT SYSTEM, Kernel, Files;
9 pressed* = 1; released* = 2; changed* = 3; included* = 5; excluded* = 6; set* = 7; (** notify ops **)
10 ok* = 1; yes* = 2; no* = 3; cancel* = 4; (** GetOK forms & results **)
11 persistent* = TRUE; nonPersistent* = FALSE; (** constants for SetLanguage **)
17 stringFile = "Strings";
19 update = 2; (* notify options *)
40 String* = ARRAY stringLen OF CHAR;
42 Buf = POINTER TO RECORD
44 s: ARRAY bufLen OF CHAR
48 len, max: INTEGER; (* number of items, max number of items *)
49 strings: Buf; (* string buffer list. strings[0] = 0X -> uninitialized items appear as empty *)
50 end: INTEGER; (* next free position in string buffer list *)
51 scnt: INTEGER; (* number of strings in list, including unused entries *)
52 items: POINTER TO ARRAY OF INTEGER (* indices into string buffer list *)
56 index*: INTEGER; (** val IN [0, n-1] **)
69 sel: POINTER TO ARRAY OF SET;
73 Currency* = RECORD (* number = val * 10^-scale *)
82 TreeNode* = POINTER TO LIMITED RECORD
85 parent, next, prev, firstChild: TreeNode;
86 viewAsFolder, expanded: BOOLEAN;
92 nofRoots, nofNodes: INTEGER;
93 firstRoot, selected: TreeNode
96 (** command procedure types**)
98 Par* = RECORD (** parameter for guard procedures **)
99 disabled*: BOOLEAN; (** OUT, preset to FALSE **)
100 checked*: BOOLEAN; (** OUT, preset to default **)
101 undef*: BOOLEAN; (** OUT, preset to default **)
102 readOnly*: BOOLEAN; (** OUT, preset to default **)
103 label*: String (** OUT, preset to "" **)
106 GuardProc* = PROCEDURE (VAR par: Par);
107 NotifierProc* = PROCEDURE (op, from, to: INTEGER);
109 StringPtr = POINTER TO ARRAY [untagged] OF CHAR;
110 StringTab = POINTER TO RECORD
113 key: POINTER TO ARRAY OF StringPtr;
114 str: POINTER TO ARRAY OF StringPtr;
115 data: POINTER TO ARRAY OF CHAR
118 LangNotifier* = POINTER TO ABSTRACT RECORD next: LangNotifier END;
119 Language* = ARRAY 3 OF CHAR;
121 LangTrapCleaner = POINTER TO RECORD (Kernel.TrapCleaner) END;
123 GetHook* = POINTER TO ABSTRACT RECORD (Kernel.Hook) END;
124 ShowHook* = POINTER TO ABSTRACT RECORD (Kernel.Hook) END;
125 CallHook* = POINTER TO ABSTRACT RECORD (Kernel.Hook) END;
126 NotifyHook* = POINTER TO ABSTRACT RECORD (Kernel.Hook) END;
127 LanguageHook* = POINTER TO ABSTRACT RECORD (Kernel.Hook) END;
130 metricSystem*: BOOLEAN;
131 showsStatus*: BOOLEAN;
133 commandLinePars*: String;
135 appName*: ARRAY 32 OF CHAR;
137 user*: ARRAY 32 OF CHAR;
138 caretPeriod*: INTEGER;
139 thickCaret*: BOOLEAN;
142 langNotifiers: LangNotifier;
143 currentNotifier: LangNotifier;
148 notifyHook: NotifyHook;
149 languageHook: LanguageHook;
151 PROCEDURE (h: GetHook) GetOK* (IN str, p0, p1, p2: ARRAY OF CHAR; form: SET;
152 OUT res: INTEGER), NEW, ABSTRACT;
153 PROCEDURE (h: GetHook) GetColor* (in: INTEGER; OUT out: INTEGER;
154 OUT set: BOOLEAN), NEW, ABSTRACT;
155 PROCEDURE (h: GetHook) GetIntSpec* (IN defType: Files.Type; VAR loc: Files.Locator;
156 OUT name: Files.Name), NEW, ABSTRACT;
157 PROCEDURE (h: GetHook) GetExtSpec* (IN defName: Files.Name; IN defType: Files.Type;
158 VAR loc: Files.Locator; OUT name: Files.Name), NEW, ABSTRACT;
160 PROCEDURE SetGetHook*(h: GetHook);
165 PROCEDURE (h: ShowHook) ShowParamMsg* (IN str, p0, p1, p2: ARRAY OF CHAR), NEW, ABSTRACT;
166 PROCEDURE (h: ShowHook) ShowParamStatus* (IN str, p0, p1, p2: ARRAY OF CHAR), NEW, ABSTRACT;
168 PROCEDURE SetShowHook* (h: ShowHook);
173 PROCEDURE (h: CallHook) Call* (IN proc, errorMsg: ARRAY OF CHAR; VAR res: INTEGER), NEW, ABSTRACT;
175 PROCEDURE SetCallHook* (h: CallHook);
180 PROCEDURE (h: NotifyHook) Notify* (id0, id1: INTEGER; opts: SET), NEW, ABSTRACT;
182 PROCEDURE SetNotifyHook* (h: NotifyHook);
187 PROCEDURE (h: LanguageHook) SetLanguage* (lang: Language; persistent: BOOLEAN;
188 OUT ok: BOOLEAN), NEW, ABSTRACT;
189 PROCEDURE (h: LanguageHook) GetPersistentLanguage* (OUT lang: Language), NEW, ABSTRACT;
191 PROCEDURE SetLanguageHook* (h: LanguageHook);
196 PROCEDURE ReadStringFile (subsys: Files.Name; f: Files.File; VAR tab: StringTab);
197 VAR i, j, h, n, s, x, len, next, down, end: INTEGER; in, in1: Files.Reader;
198 ch: CHAR; b: BYTE; p, q: StringPtr;
200 PROCEDURE ReadInt (OUT x: INTEGER);
203 in.ReadByte(b); x := b MOD 256;
204 in.ReadByte(b); x := x + (b MOD 256) * 100H;
205 in.ReadByte(b); x := x + (b MOD 256) * 10000H;
206 in.ReadByte(b); x := x + b * 1000000H
209 PROCEDURE ReadHead (OUT next, down, end: INTEGER);
210 VAR b, t: BYTE; n: INTEGER;
215 IF t = -14 THEN ReadInt(n)
217 REPEAT in.ReadByte(b) UNTIL b = 0
221 ReadInt(next); next := next + in.Pos();
222 ReadInt(down); down := down + in.Pos();
223 ReadInt(end); end := end + in.Pos()
228 IF f # NIL THEN (* read text file *)
229 in := f.NewReader(NIL); in1 := f.NewReader(NIL);
230 IF (in # NIL) & (in1 # NIL) THEN
231 in.SetPos(8); ReadHead(next, down, end); (* document view *)
232 in.SetPos(down); ReadHead(next, down, end); (* document model *)
233 in.SetPos(down); ReadHead(next, down, end); (* text view *)
234 in.SetPos(down); ReadHead(next, down, end); (* text model *)
235 in.ReadByte(b); in.ReadByte(b); in.ReadByte(b); (* versions *)
236 in.ReadByte(b); in.ReadByte(b); in.ReadByte(b);
237 ReadInt(x); in1.SetPos(in.Pos() + x); (* text offset *)
239 NEW(tab); tab.name := subsys$;
240 NEW(tab.data, f.Length());
241 n := 0; i := 0; s := 0; in.ReadByte(b);
243 IF next = in.Pos() THEN ReadHead(next, down, end); in.SetPos(end) END; (* skip attributes *)
245 IF len > 0 THEN (* shortchar run *)
247 in1.ReadByte(b); ch := CHR(b MOD 256);
249 IF s = 0 THEN j := i; s := 1 END; (* start of left part *)
250 tab.data[j] := ch; INC(j)
251 ELSIF (s = 1) & (ch = TAB) THEN
252 tab.data[j] := 0X; INC(j);
253 s := 2 (* start of right part *)
254 ELSIF (s = 2) & (ch = CR) THEN
255 tab.data[j] := 0X; INC(j);
256 INC(n); i := j; s := 0 (* end of line *)
262 ELSIF len < 0 THEN (* longchar run *)
264 in1.ReadByte(b); x := b MOD 256; in1.ReadByte(b); ch := CHR(x + 256 * (b + 128));
265 IF s = 0 THEN j := i; s := 1 END; (* start of left part *)
266 tab.data[j] := ch; INC(j);
270 ReadInt(x); ReadInt(x); in1.ReadByte(b); (* ignore *)
272 IF next = in.Pos() THEN ReadHead(next, down, end); in.SetPos(end) END; (* skip view data *)
276 NEW(tab.key, n); NEW(tab.str, n); i := 0; j := 0;
278 tab.key[j] := SYSTEM.VAL(StringPtr, SYSTEM.ADR(tab.data[i]));
279 WHILE tab.data[i] >= " " DO INC(i) END;
281 tab.str[j] := SYSTEM.VAL(StringPtr, SYSTEM.ADR(tab.data[i]));
282 WHILE tab.data[i] >= " " DO INC(i) END;
285 (* sort keys (shellsort) *)
286 h := 1; REPEAT h := h*3 + 1 UNTIL h > n;
287 REPEAT h := h DIV 3; i := h;
288 WHILE i < n DO p := tab.key[i]; q := tab.str[i]; j := i;
289 WHILE (j >= h) & (tab.key[j-h]^ > p^) DO
290 tab.key[j] := tab.key[j-h]; tab.str[j] := tab.str[j-h]; j := j-h
292 tab.key[j] := p; tab.str[j] := q; INC(i)
300 PROCEDURE MergeTabs (VAR master, extra: StringTab): StringTab;
301 VAR tab: StringTab; nofKeys, datalength, di, mi, ei, ml, el, ti, i: INTEGER;
303 IF (extra = NIL) OR (extra.key = NIL) THEN RETURN master END;
304 IF (master = NIL) OR (master.key = NIL) THEN RETURN extra END;
305 ml := LEN(master.key); el := LEN(extra.key);
306 mi := 0; ei := 0; datalength := 0; nofKeys := 0;
307 (* find out how big the resulting table will be *)
308 WHILE (mi < ml) OR (ei < el) DO
310 IF (mi < ml) & (ei < el) & (master.key[mi]$ = extra.key[ei]$) THEN
311 datalength := datalength + LEN(master.key[mi]$) + LEN(master.str[mi]$) + 2; INC(mi); INC(ei)
312 ELSIF (ei < el) & ((mi >= ml) OR (master.key[mi]$ > extra.key[ei]$)) THEN
313 datalength := datalength + LEN(extra.key[ei]$) + LEN(extra.str[ei]$) + 2; INC(ei)
315 datalength := datalength + LEN(master.key[mi]$) + LEN(master.str[mi]$) + 2; INC(mi)
318 NEW(tab); tab.name := master.name;
319 NEW(tab.key, nofKeys); NEW(tab.str, nofKeys); NEW(tab.data, datalength);
320 mi := 0; ei := 0; di := 0; ti := 0;
322 WHILE (mi < ml) OR (ei < el) DO
323 IF (mi < ml) & (ei < el) & (master.key[mi]$ = extra.key[ei]$) THEN
324 i := 0; tab.key[ti] := SYSTEM.VAL(StringPtr, SYSTEM.ADR(tab.data[di]));
325 WHILE master.key[mi][i] # 0X DO tab.data[di] := master.key[mi][i]; INC(di); INC(i) END;
326 tab.data[di] :=0X; INC(di); i := 0;
327 tab.str[ti] := SYSTEM.VAL(StringPtr, SYSTEM.ADR(tab.data[di]));
328 WHILE master.str[mi][i] # 0X DO tab.data[di] := master.str[mi][i]; INC(di); INC(i) END;
329 tab.data[di] :=0X; INC(di);
331 ELSIF (ei < el) & ((mi >= ml) OR (master.key[mi]$ > extra.key[ei]$)) THEN
332 i := 0; tab.key[ti] := SYSTEM.VAL(StringPtr, SYSTEM.ADR(tab.data[di]));
333 WHILE extra.key[ei][i] # 0X DO tab.data[di] := extra.key[ei][i]; INC(di); INC(i) END;
334 tab.data[di] :=0X; INC(di); i := 0;
335 tab.str[ti] := SYSTEM.VAL(StringPtr, SYSTEM.ADR(tab.data[di]));
336 WHILE extra.str[ei][i] # 0X DO tab.data[di] := extra.str[ei][i]; INC(di); INC(i) END;
337 tab.data[di] :=0X; INC(di);
340 i := 0; tab.key[ti] := SYSTEM.VAL(StringPtr, SYSTEM.ADR(tab.data[di]));
341 WHILE master.key[mi][i] # 0X DO tab.data[di] := master.key[mi][i]; INC(di); INC(i) END;
342 tab.data[di] :=0X; INC(di); i := 0;
343 tab.str[ti] := SYSTEM.VAL(StringPtr, SYSTEM.ADR(tab.data[di]));
344 WHILE master.str[mi][i] # 0X DO tab.data[di] := master.str[mi][i]; INC(di); INC(i) END;
345 tab.data[di] :=0X; INC(di);
353 PROCEDURE LoadStringTab (subsys: Files.Name; VAR tab: StringTab);
354 VAR loc: Files.Locator; f: Files.File; name: Files.Name; ltab: StringTab;
357 name := stringFile; Kernel.MakeFileName(name, "");
358 loc := Files.dir.This(subsys); loc := loc.This(rsrcDir);
360 f := Files.dir.Old(loc, name, Files.shared);
361 ReadStringFile(subsys, f, tab);
362 IF language # "" THEN
363 loc := loc.This(language);
365 f := Files.dir.Old(loc, name, Files.shared);
366 ReadStringFile(subsys, f, ltab);
367 tab := MergeTabs(ltab, tab)
370 IF tab # NIL THEN tab.next := tabList; tabList := tab END
374 PROCEDURE SearchString (VAR in: ARRAY OF CHAR; OUT out: ARRAY OF CHAR);
375 VAR i, j, k, len: INTEGER; ch: CHAR; subsys: Files.Name; tab: StringTab;
380 WHILE (ch # 0X) (* & (ch # ".") *) & (ch # ":") DO subsys[i] := ch; INC(i); ch := in[i + 1] END;
383 INC(i, 2); ch := in[i]; j := 0;
384 WHILE (ch # 0X) DO in[j] := ch; INC(i); INC(j); ch := in[i] END;
390 WHILE (tab # NIL) & (tab.name # subsys) DO tab := tab.next END;
391 IF tab = NIL THEN LoadStringTab(subsys, tab) END;
394 IF tab.key = NIL THEN j := 0 ELSE j := LEN(tab.key^) END;
395 WHILE i < j DO (* binary search *)
397 IF tab.key[k]^ < in THEN i := k + 1 ELSE j := k END
399 IF (tab.key # NIL) & (j < LEN(tab.key^)) & (tab.key[j]^ = in) THEN
400 k := 0; len := LEN(out)-1;
401 WHILE (k < len) & (tab.str[j][k] # 0X) DO
402 out[k] := tab.str[j][k]; INC(k)
411 PROCEDURE Init (VAR l: StrList);
413 l.len := 0; l.max := 0; l.end := 0; l.scnt := 0
416 PROCEDURE Compact (VAR l: StrList);
417 VAR i, j, k: INTEGER; ibuf, jbuf: Buf; ch: CHAR;
419 i := 1; ibuf := l.strings; j := 1; jbuf := l.strings;
421 (* find index entry k pointing to position j *)
422 k := 0; WHILE (k < l.len) & (l.items[k] # j) DO INC(k) END;
423 IF k < l.len THEN (* copy string *)
426 ch := jbuf.s[j MOD bufLen]; INC(j);
427 IF j MOD bufLen = 0 THEN jbuf := jbuf.next END;
428 ibuf.s[i MOD bufLen] := ch; INC(i);
429 IF i MOD bufLen = 0 THEN ibuf := ibuf.next END
431 ELSE (* skip next string *)
433 ch := jbuf.s[j MOD bufLen]; INC(j);
434 IF j MOD bufLen = 0 THEN jbuf := jbuf.next END
438 ibuf.next := NIL; (* release superfluous buffers *)
439 l.end := i; l.scnt := l.len
442 PROCEDURE SetLen (VAR l: StrList; len: INTEGER);
444 VAR i, newmax: INTEGER;
445 items: POINTER TO ARRAY OF INTEGER;
447 IF l.items = NIL THEN Init(l) END;
448 IF (l.max - D < len) & (len <= l.max) THEN
449 (* we do not reallocate anything *)
451 newmax := (len + D-1) DIV D * D;
453 IF l.strings = NIL THEN NEW(l.strings); (* l.strings[0] := 0X; *) l.end := 1 END;
455 IF len < l.len THEN i := len ELSE i := l.len END;
456 WHILE i > 0 DO DEC(i); items[i] := l.items[i] END;
462 IF (l.scnt > 32) & (l.scnt > 2 * l.len) THEN Compact(l) END
465 PROCEDURE GetItem (VAR l: StrList; index: INTEGER; VAR item: String);
466 VAR i, j, k: INTEGER; b: Buf; ch: CHAR;
468 IF l.items = NIL THEN Init(l) END;
469 IF (index >= 0) & (index < l.len) THEN
470 i := l.items[index]; j := i MOD bufLen; i := i DIV bufLen;
471 b := l.strings; WHILE i # 0 DO b := b.next; DEC(i) END;
474 ch := b.s[j]; INC(j); IF j = bufLen THEN j := 0; b := b.next END;
475 item[k] := ch; INC(k)
482 PROCEDURE SetItem (VAR l: StrList; index: INTEGER; item: ARRAY OF CHAR);
483 VAR len, i, j, k: INTEGER; b: Buf; ch: CHAR;
485 IF l.items = NIL THEN Init(l) END;
486 IF index >= l.len THEN SetLen(l, index + 1) END;
487 IF (l.scnt > 32) & (l.scnt > 2 * l.len) THEN Compact(l) END;
488 len := 0; WHILE item[len] # 0X DO INC(len) END;
489 IF len >= stringLen THEN len := stringLen - 1; item[len] := 0X END; (* clip long strings *)
490 l.items[index] := l.end;
491 i := l.end; j := i MOD bufLen; i := i DIV bufLen;
492 b := l.strings; WHILE i # 0 DO b := b.next; DEC(i) END;
495 ch := item[k]; INC(k); INC(l.end);
496 b.s[j] := ch; INC(j); IF j = bufLen THEN j := 0; NEW(b.next); b := b.next END
501 PROCEDURE SetResources (VAR l: StrList; IN key: ARRAY OF CHAR);
502 VAR i, k, j, x: INTEGER; ch: CHAR; s, a: ARRAY 16 OF CHAR; h, item: ARRAY 256 OF CHAR;
504 IF l.items = NIL THEN Init(l) END;
508 j := 0; REPEAT a[j] := CHR(x MOD 10 + ORD("0")); x := x DIV 10; INC(j) UNTIL x = 0;
509 k := 0; REPEAT DEC(j); ch := a[j]; s[k] := ch; INC(k) UNTIL j = 0;
511 h := key + "[" + s + "]";
512 SearchString(h, item);
513 IF item # "" THEN SetItem(l, i, item) END;
521 PROCEDURE (VAR l: List) SetLen* (len: INTEGER), NEW;
523 ASSERT(len >= 0, 20);
528 PROCEDURE (VAR l: List) GetItem* (index: INTEGER; OUT item: String), NEW;
530 GetItem(l.l, index, item);
534 PROCEDURE (VAR l: List) SetItem* (index: INTEGER; IN item: ARRAY OF CHAR), NEW;
536 ASSERT(index >= 0, 20); ASSERT(item # "", 21);
537 SetItem(l.l, index, item);
541 PROCEDURE (VAR l: List) SetResources* (IN key: ARRAY OF CHAR), NEW;
543 ASSERT(key # "", 20);
544 SetResources(l.l, key);
551 PROCEDURE (VAR s: Selection) SetLen* (len: INTEGER), NEW;
552 VAR sel: POINTER TO ARRAY OF SET; i: INTEGER;
554 ASSERT(len >= 0, 20);
556 len := len + (MAX(SET) - 1) DIV MAX(SET);
557 IF len = 0 THEN s.sel := NIL
558 ELSIF s.sel = NIL THEN NEW(s.sel, len)
559 ELSIF LEN(s.sel^) # len THEN
561 IF LEN(s.sel^) < len THEN len := LEN(s.sel^) END;
562 i := 0; WHILE i < len DO sel[i] := s.sel[i]; INC(i) END;
568 PROCEDURE (VAR s: Selection) GetItem* (index: INTEGER; OUT item: String), NEW;
570 GetItem(s.l, index, item);
574 PROCEDURE (VAR s: Selection) SetItem* (index: INTEGER; IN item: ARRAY OF CHAR), NEW;
576 ASSERT(index >= 0, 20); (*ASSERT(index < s.l.len, 21);*) ASSERT(item # "", 21);
577 SetItem(s.l, index, item);
578 IF s.l.len > s.len THEN s.SetLen(s.l.len) END
581 PROCEDURE (VAR s: Selection) SetResources* (IN key: ARRAY OF CHAR), NEW;
583 ASSERT(key # "", 20);
584 SetResources(s.l, key);
585 IF s.l.len > s.len THEN s.SetLen(s.l.len) END
588 PROCEDURE (VAR s: Selection) In* (index: INTEGER): BOOLEAN, NEW;
590 IF s.l.items = NIL THEN Init(s.l); s.len := s.l.len END;
591 IF s.sel # NIL THEN RETURN (index MOD 32) IN (s.sel[index DIV 32]) ELSE RETURN FALSE END
594 PROCEDURE (VAR s: Selection) Excl* (from, to: INTEGER), NEW;
596 IF s.l.items = NIL THEN Init(s.l); s.len := s.l.len END;
597 IF from < 0 THEN from := 0 END;
598 IF to >= s.l.len THEN to := s.l.len - 1 END;
599 WHILE from <= to DO EXCL(s.sel[from DIV 32], from MOD 32); INC(from) END
602 PROCEDURE (VAR s: Selection) Incl* (from, to: INTEGER), NEW;
604 IF s.l.items = NIL THEN Init(s.l); s.len := s.l.len END;
605 IF from < 0 THEN from := 0 END;
606 IF to >= s.l.len THEN to := s.l.len - 1 END;
607 WHILE from <= to DO INCL(s.sel[from DIV 32], from MOD 32); INC(from) END
613 PROCEDURE (VAR c: Combo) SetLen* (len: INTEGER), NEW;
615 ASSERT(len >= 0, 20);
620 PROCEDURE (VAR c: Combo) GetItem* (index: INTEGER; OUT item: String), NEW;
622 GetItem(c.l, index, item);
626 PROCEDURE (VAR c: Combo) SetItem* (index: INTEGER; IN item: ARRAY OF CHAR), NEW;
628 ASSERT(index >= 0, 20); ASSERT(item # "", 21);
629 SetItem(c.l, index, item);
633 PROCEDURE (VAR c: Combo) SetResources* (IN key: ARRAY OF CHAR), NEW;
635 ASSERT(key # "", 20);
636 SetResources(c.l, key);
641 (* Tree and TreeNode *)
643 PROCEDURE (tn: TreeNode) SetName* (name: String), NEW;
648 PROCEDURE (tn: TreeNode) GetName* (OUT name: String), NEW;
653 PROCEDURE (tn: TreeNode) SetData* (data: ANYPTR), NEW;
658 PROCEDURE (tn: TreeNode) Data* (): ANYPTR, NEW;
663 PROCEDURE (tn: TreeNode) NofChildren* (): INTEGER, NEW;
665 RETURN tn.nofChildren
668 PROCEDURE (tn: TreeNode) SetExpansion* (expanded: BOOLEAN), NEW;
670 tn.expanded := expanded
673 PROCEDURE (tn: TreeNode) IsExpanded* (): BOOLEAN, NEW;
678 PROCEDURE (tn: TreeNode) IsFolder* (): BOOLEAN, NEW;
680 IF (~tn.viewAsFolder) & (tn.firstChild = NIL) THEN
687 PROCEDURE (tn: TreeNode) ViewAsFolder* (isFolder: BOOLEAN), NEW;
689 tn.viewAsFolder := isFolder
692 PROCEDURE (VAR t: Tree) NofNodes* (): INTEGER, NEW;
694 IF t.firstRoot = NIL THEN
697 RETURN MAX(0, t.nofNodes)
701 PROCEDURE (VAR t: Tree) NofRoots* (): INTEGER, NEW;
703 IF t.firstRoot = NIL THEN
706 RETURN MAX(0, t.nofRoots)
710 PROCEDURE (VAR t: Tree) Parent* (node: TreeNode): TreeNode, NEW;
712 ASSERT(node # NIL, 20); ASSERT(node.tree = SYSTEM.ADR(t), 21);
716 PROCEDURE (VAR t: Tree) Next* (node: TreeNode): TreeNode, NEW;
718 ASSERT(node # NIL, 20); ASSERT(node.tree = SYSTEM.ADR(t), 21);
722 PROCEDURE (VAR t: Tree) Prev* (node: TreeNode): TreeNode, NEW;
724 ASSERT(node # NIL, 20); ASSERT(node.tree = SYSTEM.ADR(t), 21);
728 PROCEDURE (VAR t: Tree) Child* (node: TreeNode; pos: INTEGER): TreeNode, NEW;
731 ASSERT(pos >= lastPos, 20); ASSERT((node = NIL) OR (node.tree = SYSTEM.ADR(t)), 21);
732 IF node = NIL THEN cur := t.firstRoot
733 ELSE cur := node.firstChild END;
734 IF pos = lastPos THEN
735 WHILE (cur # NIL) & (cur.next # NIL) DO cur := cur.next END
737 WHILE (cur # NIL) & (pos > 0) DO cur := cur.next; DEC(pos) END
742 PROCEDURE (VAR t: Tree) Selected* (): TreeNode, NEW;
747 PROCEDURE (VAR t: Tree) Select* (node: TreeNode), NEW;
749 ASSERT((node = NIL) OR (node.tree = SYSTEM.ADR(t)), 20);
750 IF (node # NIL) OR (t.nofRoots = 0) THEN
753 t.selected := t.Child(NIL, 0)
757 PROCEDURE Include (IN t: Tree; node: TreeNode);
760 ASSERT(node # NIL, 20); ASSERT(node.tree = 0, 100);
761 node.tree := SYSTEM.ADR(t);
762 c := node.firstChild;
763 WHILE c # NIL DO Include(t, c); c := c.next END
766 PROCEDURE (VAR t: Tree) InsertAt (parent: TreeNode; pos: INTEGER; node: TreeNode), NEW;
770 ASSERT(node # NIL, 20); ASSERT(pos >= lastPos, 21);
771 ASSERT((parent = NIL) OR (parent.tree = SYSTEM.ADR(t)), 22); ASSERT(node.tree = 0, 23);
773 IF parent = NIL THEN (* Add new root *)
774 IF (t.firstRoot = NIL) OR (pos = 0) THEN
775 node.next := t.firstRoot; node.prev := NIL;
776 IF t.firstRoot # NIL THEN t.firstRoot.prev := node END;
780 IF pos = lastPos THEN pos := t.nofRoots END;
781 WHILE (cur # NIL) & (pos > 0) DO
782 prev := cur; cur := t.Next(cur); DEC(pos)
785 prev.next := node; node.prev := prev
787 node.next := cur; node.prev := cur.prev; cur.prev := node; prev.next := node
792 IF pos = lastPos THEN pos := parent.nofChildren END;
793 IF (parent.firstChild = NIL) OR (pos = 0) THEN
794 IF parent.firstChild # NIL THEN parent.firstChild.prev := node END;
795 node.prev := NIL; node.next := parent.firstChild; parent.firstChild := node
797 cur := parent.firstChild;
798 WHILE (cur # NIL) & (pos > 0) DO
799 prev := cur; cur := t.Next(cur); DEC(pos)
802 prev.next := node; node.prev := prev
804 node.next := cur; node.prev := cur.prev; cur.prev := node; prev.next := node
807 INC(parent.nofChildren)
809 node.parent := parent;
813 PROCEDURE (VAR t: Tree) NewChild* (parent: TreeNode; pos: INTEGER; name: String): TreeNode, NEW;
817 NEW(new); new.tree := 0;
818 new.SetName(name); new.expanded := FALSE; new.nofChildren := 0;
819 new.viewAsFolder := FALSE;
820 t.InsertAt(parent, pos, new);
824 PROCEDURE (VAR t: Tree) CountChildren (node: TreeNode): INTEGER, NEW;
825 VAR tot, nofc, i: INTEGER;
829 nofc := node.nofChildren; tot := nofc;
830 FOR i := 0 TO nofc -1 DO
831 tot := tot + t.CountChildren(t.Child(node, i))
837 PROCEDURE Exclude (IN t: Tree; node: TreeNode);
840 ASSERT(node # NIL, 20); ASSERT(node.tree = SYSTEM.ADR(t), 100);
841 IF t.Selected() = node THEN t.Select(NIL) END;
843 c := node.firstChild;
844 WHILE c # NIL DO Exclude(t, c); c := c.next END
847 PROCEDURE (VAR t: Tree) Delete* (node: TreeNode): INTEGER, NEW;
851 ASSERT(node # NIL, 20); ASSERT(node.tree = SYSTEM.ADR(t), 21);
852 ndel := t.CountChildren(node);
853 IF node.parent = NIL THEN (* root node *)
854 IF node.prev = NIL THEN
855 IF node.next # NIL THEN
856 t.firstRoot := node.next;
857 node.next.prev := NIL
862 node.prev.next := node.next;
863 IF node.next # NIL THEN node.next.prev := node.prev END
867 IF node.prev = NIL THEN
868 IF node.next # NIL THEN
869 node.parent.firstChild := node.next;
870 node.next.prev := NIL
872 node.parent.firstChild := NIL
875 node.prev.next := node.next;
876 IF node.next # NIL THEN node.next.prev := node.prev END
878 DEC(node.parent.nofChildren)
880 node.parent := NIL; node.next := NIL; node.prev := NIL;
883 t.nofNodes := t.nofNodes - ndel;
887 PROCEDURE (VAR t: Tree) Move* (node, parent: TreeNode; pos: INTEGER), NEW;
888 VAR ndel, nofn: INTEGER; s: TreeNode;
890 ASSERT(node # NIL, 20); ASSERT(pos >= lastPos, 21);
891 ASSERT(node.tree = SYSTEM.ADR(t), 22);
892 nofn := t.NofNodes();
894 ndel := t.Delete(node); t.InsertAt(parent, pos, node);
895 t.nofNodes := t.nofNodes + ndel - 1;
896 IF (s # NIL) & (t.Selected() # s) THEN t.Select(s) END;
897 ASSERT(nofn = t.NofNodes(), 60)
900 PROCEDURE (VAR t: Tree) DeleteAll*, NEW;
902 t.nofRoots := 0; t.nofNodes := 0; t.firstRoot := NIL; t.selected := NIL
906 PROCEDURE Notify* (id0, id1: INTEGER; opts: SET);
908 ASSERT(notifyHook # NIL, 100);
909 notifyHook.Notify(id0, id1, opts)
912 PROCEDURE Update* (IN x: ANYREC);
913 VAR type: Kernel.Type; adr, size: INTEGER;
915 adr := SYSTEM.ADR(x);
916 type := Kernel.TypeOf(x);
918 IF size = 0 THEN size := 1 END;
919 Notify(adr, adr + size, {update, guardCheck})
922 PROCEDURE UpdateBool* (VAR x: BOOLEAN);
925 adr := SYSTEM.ADR(x);
926 Notify(adr, adr + SIZE(BOOLEAN), {update, guardCheck})
929 PROCEDURE UpdateSChar* (VAR x: SHORTCHAR);
932 adr := SYSTEM.ADR(x);
933 Notify(adr, adr + SIZE(SHORTCHAR), {update, guardCheck})
936 PROCEDURE UpdateChar* (VAR x: CHAR);
939 adr := SYSTEM.ADR(x);
940 Notify(adr, adr + SIZE(CHAR), {update, guardCheck})
943 PROCEDURE UpdateByte* (VAR x: BYTE);
946 adr := SYSTEM.ADR(x);
947 Notify(adr, adr + SIZE(BYTE), {update, guardCheck})
950 PROCEDURE UpdateSInt* (VAR x: SHORTINT);
953 adr := SYSTEM.ADR(x);
954 Notify(adr, adr + SIZE(SHORTINT), {update, guardCheck})
957 PROCEDURE UpdateInt* (VAR x: INTEGER);
960 adr := SYSTEM.ADR(x);
961 Notify(adr, adr + SIZE(INTEGER), {update, guardCheck})
964 PROCEDURE UpdateLInt* (VAR x: LONGINT);
967 adr := SYSTEM.ADR(x);
968 Notify(adr, adr + SIZE(LONGINT), {update, guardCheck})
971 PROCEDURE UpdateSReal* (VAR x: SHORTREAL);
974 adr := SYSTEM.ADR(x);
975 Notify(adr, adr + SIZE(SHORTREAL), {update, guardCheck})
978 PROCEDURE UpdateReal* (VAR x: REAL);
981 adr := SYSTEM.ADR(x);
982 Notify(adr, adr + SIZE(REAL), {update, guardCheck})
985 PROCEDURE UpdateSet* (VAR x: SET);
988 adr := SYSTEM.ADR(x);
989 Notify(adr, adr + SIZE(SET), {update, guardCheck})
992 PROCEDURE UpdateSString* (IN x: ARRAY OF SHORTCHAR);
995 adr := SYSTEM.ADR(x);
996 Notify(adr, adr + LEN(x) * SIZE(SHORTCHAR), {update, guardCheck})
999 PROCEDURE UpdateString* (IN x: ARRAY OF CHAR);
1002 adr := SYSTEM.ADR(x);
1003 Notify(adr, adr + LEN(x) * SIZE(CHAR), {update, guardCheck})
1006 PROCEDURE UpdateList* (IN x: ANYREC);
1007 VAR type: Kernel.Type; adr, size: INTEGER;
1009 adr := SYSTEM.ADR(x);
1010 type := Kernel.TypeOf(x);
1012 IF size = 0 THEN size := 1 END;
1013 Notify(adr, adr + size, {listUpdate, guardCheck})
1017 PROCEDURE GetOK* (IN str, p0, p1, p2: ARRAY OF CHAR; form: SET; OUT res: INTEGER);
1019 ASSERT(((yes IN form) = (no IN form)) & ((yes IN form) # (ok IN form)), 20);
1020 ASSERT(gethook # NIL, 100);
1021 gethook.GetOK(str, p0, p1, p2, form, res)
1024 PROCEDURE GetIntSpec* (defType: Files.Type; VAR loc: Files.Locator; OUT name: Files.Name);
1026 ASSERT(gethook # NIL, 100);
1027 gethook.GetIntSpec(defType, loc, name)
1030 PROCEDURE GetExtSpec* (defName: Files.Name; defType: Files.Type; VAR loc: Files.Locator;
1031 OUT name: Files.Name);
1033 ASSERT(gethook # NIL, 100);
1034 gethook.GetExtSpec(defName, defType, loc, name)
1037 PROCEDURE GetColor* (in: INTEGER; OUT out: INTEGER; OUT set: BOOLEAN);
1039 ASSERT(gethook # NIL, 100);
1040 gethook.GetColor(in, out, set)
1044 PROCEDURE Subst (in: ARRAY OF CHAR; IN p0, p1, p2: ARRAY OF CHAR; VAR out: ARRAY OF CHAR);
1045 VAR len, i, j, k: INTEGER; ch, c: CHAR;
1047 i := 0; ch := in[i]; j := 0; len := LEN(out) - 1;
1048 WHILE (ch # 0X) & (j < len) DO
1050 INC(i); ch := in[i];
1053 WHILE (c # 0X) & (j < len) DO out[j] := c; INC(j); INC(k); c := p0[k] END;
1057 WHILE (c # 0X) & (j < len) DO out[j] := c; INC(j); INC(k); c := p1[k] END;
1061 WHILE (c # 0X) & (j < len) DO out[j] := c; INC(j); INC(k); c := p2[k] END;
1063 ELSE out[j] := "^"; INC(j)
1065 ELSE out[j] := ch; INC(j); INC(i); ch := in[i]
1071 PROCEDURE FlushMappings*;
1076 PROCEDURE MapParamString* (in, p0, p1, p2: ARRAY OF CHAR; OUT out: ARRAY OF CHAR);
1077 (* use in as key in string table file, and return corresponding string in out.
1078 If the resource lookup fails, return in in out *)
1080 SearchString(in, out);
1081 IF out # "" THEN Subst(out, p0, p1, p2, out)
1082 ELSE Subst(in, p0, p1, p2, out)
1086 PROCEDURE MapString* (in: ARRAY OF CHAR; OUT out: ARRAY OF CHAR);
1087 VAR len, k: INTEGER;
1089 SearchString(in, out);
1091 k := 0; len := LEN(out)-1;
1092 WHILE (k < len) & (in[k] # 0X) DO out[k] := in[k]; INC(k) END;
1097 PROCEDURE ShowMsg* (IN str: ARRAY OF CHAR);
1099 ASSERT(str # "", 20);
1100 ASSERT(showHook # NIL, 100);
1101 showHook.ShowParamMsg(str, "", "", "")
1104 PROCEDURE ShowParamMsg* (IN str, p0, p1, p2: ARRAY OF CHAR);
1106 ASSERT(str # "", 20);
1107 ASSERT(showHook # NIL, 100);
1108 showHook.ShowParamMsg(str,p0, p1, p2)
1111 PROCEDURE ShowStatus* (IN str: ARRAY OF CHAR);
1113 ASSERT(showHook # NIL, 100);
1114 showHook.ShowParamStatus(str, "", "", "")
1117 PROCEDURE ShowParamStatus* (IN str, p0, p1, p2: ARRAY OF CHAR);
1119 ASSERT(showHook # NIL, 100);
1120 showHook.ShowParamStatus(str, p0, p1, p2)
1121 END ShowParamStatus;
1124 PROCEDURE Call* (IN proc, errorMsg: ARRAY OF CHAR; OUT res: INTEGER);
1126 ASSERT(callHook # NIL, 100);
1127 callHook.Call(proc, errorMsg, res)
1135 PROCEDURE (n: LangNotifier) Notify-(), NEW, ABSTRACT;
1137 PROCEDURE RegisterLangNotifier* (notifier: LangNotifier);
1138 VAR nl: LangNotifier;
1140 ASSERT(notifier # NIL, 20);
1141 nl := langNotifiers;
1142 WHILE (nl # NIL) & (nl # notifier) DO nl := nl.next END;
1144 notifier.next := langNotifiers; langNotifiers := notifier
1146 END RegisterLangNotifier;
1148 PROCEDURE RemoveLangNotifier* (notifier: LangNotifier);
1149 VAR nl, prev: LangNotifier;
1151 ASSERT(notifier # NIL, 20);
1152 nl := langNotifiers; prev := NIL;
1153 WHILE (nl # NIL) & (nl # notifier) DO prev := nl; nl := nl.next END;
1155 IF prev = NIL THEN langNotifiers := langNotifiers.next ELSE prev.next := nl.next END;
1158 END RemoveLangNotifier;
1160 PROCEDURE Exec (a, b, c: INTEGER);
1161 VAR nl: LangNotifier;
1163 nl := currentNotifier; currentNotifier := NIL;
1165 currentNotifier := nl
1168 PROCEDURE SetLanguage* (lang: Language; persistent: BOOLEAN);
1169 VAR nl, t: LangNotifier; ok: BOOLEAN;
1171 ASSERT((lang = "") OR (LEN(lang$) = 2), 20);
1172 ASSERT(languageHook # NIL, 100);
1173 IF lang # language THEN
1174 languageHook.SetLanguage(lang, persistent, ok);
1176 language := lang; FlushMappings;
1177 nl := langNotifiers;
1179 currentNotifier := nl;
1180 Kernel.Try(Exec, 0, 0, 0);
1181 IF currentNotifier = NIL THEN
1182 t := nl; nl := nl.next; RemoveLangNotifier(t) (* Notifier trapped, remove it *)
1188 currentNotifier := NIL
1192 PROCEDURE ResetLanguage*;
1195 ASSERT(languageHook # NIL, 100);
1196 languageHook.GetPersistentLanguage(lang);
1197 SetLanguage(lang, nonPersistent)
1201 appName := "BlackBox"; showsStatus := FALSE; caretPeriod := 500; thickCaret := FALSE; user := ""