wormy 0.97.103
[wormy.git] / wormedit.pas
1 {WORMEDIT 0.5 OR SOMETHING... BY SHIAR
2  ---------------------------------w00t}
3
4 {PASCAL.... BECAUSE I'VE GOT A LOT OF EXPERIENCE USING IT, AND I JUST NEEDED A
5  QUICK SUCKY EDITOR. IF YOU'RE WILLING TO WRITE SOMETHING BETTER, JUST MAIL ME}
6
7 {WRITTEN USING ULTRAEDIT IN DVORAK WITH BORLAND PASCAL 7.0 FILES AND CRT PATCH
8  FOR DOS, HUZZAH. TESTED OK IN DOS 6, WINDOWS 98 AND XP, AND LINUX WITH DOSEMU}
9
10 {ALLCAPS BECAUSE. AND IT MAKES IT HARDER TO UNDERSTAND THE CODE}
11
12 {ENJOY! >:}
13
14 USES CRT,SHIAR,DOS;
15
16 TYPE ONELEVEL = RECORD
17        ID:STRING[22];
18        NAME:STRING[22];
19        SIZE:BYTE;
20        PEAS,DELAY:BYTE;
21        GROWTH,BSIZE:BYTE;
22        SPRSIZE:BYTE;
23        SPRITE:ARRAY[1..8]OF BYTE;
24        NRBALLS:BYTE;
25        BALLS:ARRAY[1..32]OF RECORD
26          Y,X,D:BYTE; {Z:%00=right+down, %11=left+up}
27        END; {BALLS}
28        W:ARRAY[1..4]OF RECORD
29          D,Y,X:BYTE;
30        END; {W(ORMS)}
31        FIELDX,FIELDY:BYTE;
32        FLAG1Y,FLAG1X:BYTE;
33        FLAG2Y,FLAG2X:BYTE;
34        NROBJ:BYTE;
35        OBJ:ARRAY[1..32]OF RECORD
36          TYP:SHORTINT;X1,Y1,X2,Y2:BYTE;
37        END; {OBJ}
38      END; {ONELEVEL}
39
40 CONST GAMETYPE:ARRAY[1..8]OF STRING[12] =
41         ('SINGLEPLAYER','PEAWORM','TRON','DEATHMATCH',
42          'FOODMATCH','TIMEMATCH','RACE','CTF');
43
44       OBJTYPE:ARRAY[0..4]OF STRING[8] =
45         ('NONE','LINE','FAT LINE','BAR','CIRCLE');
46
47       ENDTYPE:ARRAY[0..1]OF STRING[8] =
48         ('NONE','MESSAGE');
49
50       HEAD86S:STRING[53] =
51         ('**TI86**'+#$1A+#$A+#0+ {HEADER}
52          'Wormy�95 Levelfile made using WormEdit 0.5'); {COMMENT}
53
54       HEADEDIT = ('WormEdit�95'+#0+'LVL');
55       HEADVERSION = 95;
56
57       B:RECORD
58         HEAD:ARRAY[1..15]OF CHAR;
59         VERSION:BYTE;
60         NAME:STRING[32];
61         DESC:STRING[64];
62         LSINGLE,LMULTI,LRACE,LCTF,LTOTAL:BYTE;
63         LFROM:ARRAY[1..8]OF BYTE;
64         LTO:ARRAY[1..8]OF BYTE;
65         SPRSIZE:BYTE;
66         SPRITE:ARRAY[1..8]OF BYTE;
67         ENDTYPE:INTEGER;
68         ENDSTR:STRING[255];
69         ENDDATA:STRING[255];
70         HINAME:ARRAY[1..3]OF CHAR;
71       END {B} =
72         (HEAD:HEADEDIT;
73          VERSION:HEADVERSION;
74          NAME:'Custom Worm Level';
75          DESC:'Not done yet. Obviously.';
76          LSINGLE:0; LMULTI:0; LRACE:0; LCTF:0; LTOTAL:0;
77          LFROM:(0,0,0,0,0,0,0,0);
78          LTO:(0,0,0,0,0,0,0,0);
79          SPRSIZE:4;
80          SPRITE:(6*16,15*16,15*16,6*16,0,0,0,0);
81            {01100000
82             11110000
83             11110000
84             01100000}
85          ENDTYPE:1;
86          ENDSTR:'';
87          ENDDATA:#$C9;
88          HINAME:'WOR');
89
90       LVLPOS:BYTE = 1;
91       OBJPOS:BYTE = 1;
92
93         C_BAR   = 15;
94         CB_BAR  =  4;
95         C_MENU  = 14;
96         CB_TEXT =  0;
97         C_TEXT  =  7;
98         C_SUB   =  8;
99         C_HI    = 15;
100         C_VALUE =  3;
101         C_GRID  =  7;
102
103 VAR I,I2:WORD;
104     II,III,I4:SHORTINT;
105     C:CHAR;
106     S:STRING;
107     F,FF:FILE;
108     MX,MY:WORD;
109
110     L:ARRAY[1..64]OF ONELEVEL;
111     LTEMP:ONELEVEL;
112     USEMOUSE:BOOLEAN;
113     QUITOK:BOOLEAN;
114
115     FSIZE:LONGINT;
116     FILENAME:STRING[8];
117     LVLDISP:BYTE;
118
119 PROCEDURE MAINTITLE;
120 BEGIN
121   SETTEXT; SETPAL(3,44,44,32);
122   WINDOW(1,1,80,25); TEXTBACKGROUND(CB_TEXT); CLRSCR;
123   TEXTCOLOR(CB_BAR); WRITE('���������������������������<');
124   TEXTCOLOR(14); WRITE(' WormEdit � ');
125   TEXTCOLOR(15); WRITE(' by SHIAR ');
126   TEXTCOLOR(CB_BAR); WRITE('>�����������������������������');
127   WINDOW(41,2,41,25); TEXTCOLOR(C_TEXT); TEXTBACKGROUND(CB_TEXT); WRITE(CC('�',23));
128   WINDOW(1,25,80,25); TEXTBACKGROUND(CB_BAR); CLRSCR;
129   GOTOXY(75,1); TEXTCOLOR(0); WRITE('v0.5�'); TEXTBACKGROUND(CB_TEXT);
130 END; {MAINTITLE}
131
132 PROCEDURE PUTPIXEL(X,Y:INTEGER;COL:BYTE;WHERE:POINTER);
133 BEGIN {REPLACEMENT_PUTPIXEL}
134   MEM[SEG(WHERE^)+Y*20:OFS(WHERE^)+X]:=COL;
135 END; {PUTPIXEL}
136
137 PROCEDURE LINE(X,Y,XX,YY:INTEGER;COL:BYTE;WHERE:POINTER);
138 VAR LINE1,LINE2,SX,SY,SXY,SYX,DX1,DX2,DY1,DY2:INTEGER;
139 BEGIN {JUST_A_SIMPLE_LINE_PROCEDURE}
140   SX:=XX-X; SY:=YY-Y; DX1:=SGN(SX); DX2:=DX1; DY1:=SGN(SY); DY2:=0;
141   SXY:=ABS(SX); SYX:=ABS(SY);
142   IF SYX>SXY THEN BEGIN
143     DY2:=DY1; DX2:=0; LINE1:=SXY; SXY:=SYX; SYX:=LINE1;
144   END; {VERTICAL_LINE}
145   LINE1:=SXY SHR 1;
146   FOR LINE2:=0TO SXY DO BEGIN
147     PUTPIXEL(X,Y,COL,WHERE);
148     LINE1:=LINE1+SYX;
149     IF LINE1<SXY THEN BEGIN
150       INC(X,DX2); INC(Y,DY2);
151     END {STRAIGT_AHEAD} ELSE BEGIN
152       LINE1:=LINE1-SXY;
153       INC(X,DX1); INC(Y,DY1);
154     END; {NEXT_STEP}
155   END; {DRAW_THE_LINE}
156 END; {LINE}
157
158 PROCEDURE PPLINE(X,Y,XX,YY:INTEGER;COL:BYTE);
159 VAR LINE1,LINE2,SX,SY,SXY,SYX,DX1,DX2,DY1,DY2:INTEGER;
160 BEGIN {SPECIAL LINE}
161   SX:=XX-X; SY:=YY-Y; DX1:=SGN(SX); DX2:=DX1; DY1:=SGN(SY); DY2:=0;
162   SXY:=ABS(SX); SYX:=ABS(SY);
163   IF SYX>SXY THEN BEGIN
164     DY2:=DY1; DX2:=0; LINE1:=SXY; SXY:=SYX; SYX:=LINE1;
165   END; {VERTICAL_LINE}
166   LINE1:=SXY SHR 1;
167   FOR LINE2:=0TO SXY DO BEGIN
168     PUTPIXEL(X,Y,GETPIXEL(X,Y,VGA)OR 1-COL,PTR(VGA,0));
169     LINE1:=LINE1+SYX;
170     IF LINE1<SXY THEN BEGIN
171       INC(X,DX2); INC(Y,DY2);
172     END {STRAIGT_AHEAD} ELSE BEGIN
173       LINE1:=LINE1-SXY;
174       INC(X,DX1); INC(Y,DY1);
175     END; {NEXT_STEP}
176   END; {DRAW_THE_LINE}
177 END; {PPLINE}
178 PROCEDURE PPCIRCLE(X,Y,Z,COL:BYTE);
179 VAR XX,YY,ZZ:INTEGER;
180 BEGIN
181   XX:=-1; YY:=Z; ZZ:=-Z;
182   REPEAT
183     INC(XX);
184     PPLINE(X-XX,Y+YY,X+XX,Y+YY,COL); PPLINE(X-XX,Y-YY,X+XX,Y-YY,COL);
185     PPLINE(X-YY,Y+XX,X+YY,Y+XX,COL); PPLINE(X-YY,Y-XX,X+YY,Y-XX,COL);
186     IF ZZ>=0 THEN BEGIN ZZ:=ZZ-(YY SHL 1)+2; DEC(YY); END;
187     ZZ:=ZZ+(XX SHL 1)+3;
188   UNTIL XX>=YY;
189 END; {PPCIRCLE}
190
191 PROCEDURE PPDISPOBJ(OBJ:INTEGER;COL:BYTE);
192 VAR Y:BYTE;
193 BEGIN
194   CASE L[II].OBJ[OBJ].TYP OF
195     1:PPLINE(L[II].OBJ[OBJ].X1,L[II].OBJ[OBJ].Y1,L[II].OBJ[OBJ].X2,L[II].OBJ[OBJ].Y2,COL);
196     2:BEGIN
197       PPLINE(L[II].OBJ[OBJ].X1,L[II].OBJ[OBJ].Y1,L[II].OBJ[OBJ].X2,L[II].OBJ[OBJ].Y2,COL);
198       PPLINE(L[II].OBJ[OBJ].X1+1,L[II].OBJ[OBJ].Y1,L[II].OBJ[OBJ].X2+1,L[II].OBJ[OBJ].Y2,COL);
199       PPLINE(L[II].OBJ[OBJ].X1+1,L[II].OBJ[OBJ].Y1+1,L[II].OBJ[OBJ].X2+1,L[II].OBJ[OBJ].Y2+1,COL);
200       PPLINE(L[II].OBJ[OBJ].X1,L[II].OBJ[OBJ].Y1+1,L[II].OBJ[OBJ].X2,L[II].OBJ[OBJ].Y2+1,COL);
201     END; {FATLINE}
202     3:BEGIN
203       FOR Y:=L[II].OBJ[OBJ].Y1 TO L[II].OBJ[OBJ].Y1+L[II].OBJ[OBJ].Y2-1DO
204         PPLINE(L[II].OBJ[OBJ].X1,MIN(Y,191),L[II].OBJ[OBJ].X2,MIN(Y,191),COL);
205     END; {BOX}
206     4:PPCIRCLE(L[II].OBJ[OBJ].X1,L[II].OBJ[OBJ].Y1,L[II].OBJ[OBJ].X2,COL);
207   END; {OBJECT TYPE}
208 END; {DISPOBJ}
209
210 PROCEDURE DISPMAP(OBJ:INTEGER;WHERE:POINTER);
211 VAR OBJ2,TI:INTEGER;
212 PROCEDURE CIRCLE(X,Y,Z,COL:BYTE);
213 VAR XX,YY,ZZ:INTEGER;
214 BEGIN
215   XX:=-1; YY:=Z; ZZ:=-Z;
216   REPEAT
217     INC(XX);
218     LINE(X-XX,Y+YY,X+XX,Y+YY,COL,WHERE);
219     LINE(X-XX,Y-YY,X+XX,Y-YY,COL,WHERE);
220     LINE(X-YY,Y+XX,X+YY,Y+XX,COL,WHERE);
221     LINE(X-YY,Y-XX,X+YY,Y-XX,COL,WHERE);
222     IF ZZ>=0 THEN BEGIN ZZ:=ZZ-(YY SHL 1)+2; DEC(YY); END;
223     ZZ:=ZZ+(XX SHL 1)+3;
224   UNTIL XX>=YY;
225 END; {CIRCLE}
226 PROCEDURE DRAW4PIX(D4X,D4Y:INTEGER;D4XSIZE,D4YSIZE,COL:BYTE);
227 VAR D4LOOPX,D4LOOPY:BYTE;
228 BEGIN
229   FOR D4LOOPX:=0TO D4XSIZE-1DO FOR D4LOOPY:=0TO D4YSIZE-1DO
230     PUTPIXEL(D4X+D4LOOPX,D4Y+D4LOOPY,COL,WHERE);
231 END; {DRAW4PIX}
232 PROCEDURE DISPOBJ(OBJ:INTEGER;COL:BYTE);
233 VAR Y:BYTE;
234 BEGIN
235   CASE L[II].OBJ[OBJ].TYP OF
236     1:LINE(L[II].OBJ[OBJ].X1,L[II].OBJ[OBJ].Y1,L[II].OBJ[OBJ].X2,L[II].OBJ[OBJ].Y2,COL,WHERE);
237     2:BEGIN
238       LINE(L[II].OBJ[OBJ].X1,L[II].OBJ[OBJ].Y1,L[II].OBJ[OBJ].X2,L[II].OBJ[OBJ].Y2,COL,WHERE);
239       LINE(L[II].OBJ[OBJ].X1+1,L[II].OBJ[OBJ].Y1,L[II].OBJ[OBJ].X2+1,L[II].OBJ[OBJ].Y2,COL,WHERE);
240       LINE(L[II].OBJ[OBJ].X1+1,L[II].OBJ[OBJ].Y1+1,L[II].OBJ[OBJ].X2+1,L[II].OBJ[OBJ].Y2+1,COL,WHERE);
241       LINE(L[II].OBJ[OBJ].X1,L[II].OBJ[OBJ].Y1+1,L[II].OBJ[OBJ].X2,L[II].OBJ[OBJ].Y2+1,COL,WHERE);
242     END; {FATLINE}
243     3:BEGIN
244       FOR Y:=L[II].OBJ[OBJ].Y1 TO L[II].OBJ[OBJ].Y1+L[II].OBJ[OBJ].Y2-1DO
245         LINE(L[II].OBJ[OBJ].X1,MIN(Y,191),L[II].OBJ[OBJ].X2,MIN(Y,191),COL,WHERE);
246     END; {BOX}
247     4:CIRCLE(L[II].OBJ[OBJ].X1,L[II].OBJ[OBJ].Y1,L[II].OBJ[OBJ].X2,COL);
248   END; {OBJECT TYPE}
249 END; {DISPOBJ}
250 BEGIN {---}
251   TI:=3; WHILE TI<=255DO BEGIN LINE(TI,0,TI,191,2,WHERE); INC(TI,8); END; {GRID}
252   TI:=4; WHILE TI<=191DO BEGIN LINE(0,TI,255,TI,2,WHERE); INC(TI,8); END;
253   TI:=7; WHILE TI<=255DO BEGIN LINE(TI,0,TI,191,4,WHERE); INC(TI,8); END; {GRID}
254   TI:=0; WHILE TI<=191DO BEGIN LINE(0,TI,255,TI,4,WHERE); INC(TI,8); END;
255   TI:=127; WHILE TI<=255DO BEGIN LINE(TI,0,TI,191,8,WHERE); INC(TI,128); END;
256   TI:=0; WHILE TI<=191DO BEGIN LINE(0,TI,255,TI,8,WHERE); INC(TI,56); END;
257
258   FOR TI:=2TO L[II].FIELDX-2DO BEGIN
259     IF TI MOD 2=0THEN PUTPIXEL(TI,L[II].FIELDY SHR 1,6,WHERE)
260     ELSE PUTPIXEL(TI,L[II].FIELDY SHR 1,2,WHERE);
261   END; {MID}
262   IF L[II].FIELDY MOD 2=0THEN
263     FOR TI:=2TO L[II].FIELDX-2DO BEGIN
264       IF TI MOD 2=1THEN PUTPIXEL(TI,L[II].FIELDY SHR 1-1,6,WHERE)
265       ELSE PUTPIXEL(TI,L[II].FIELDY SHR 1-1,2,WHERE);
266     END; {DOUBLEMID}
267   FOR TI:=2TO L[II].FIELDY-2DO BEGIN
268     IF TI MOD 2=0THEN PUTPIXEL(L[II].FIELDX SHR 1,TI,6,WHERE)
269     ELSE PUTPIXEL(L[II].FIELDX SHR 1,TI,2,WHERE);
270   END; {CENTER}
271   IF L[II].FIELDX MOD 2=0THEN
272     FOR TI:=2TO L[II].FIELDY-2DO BEGIN
273       IF TI MOD 2=1THEN PUTPIXEL(L[II].FIELDX SHR 1-1,TI,6,WHERE)
274       ELSE PUTPIXEL(L[II].FIELDX SHR 1-1,TI,2,WHERE);
275     END; {DOUBLECENTER}
276   IF(II>B.LSINGLE+B.LMULTI)AND(II<=B.LSINGLE+B.LMULTI+B.LRACE)THEN BEGIN
277     LINE(L[II].FLAG1Y,0,L[II].FLAG1Y,L[II].FLAG1X,26,WHERE);
278     LINE(L[II].FLAG1Y,L[II].FLAG1X,L[II].FLAG1Y,L[II].FIELDY-1,28,WHERE);
279     LINE(0,L[II].FLAG1X,L[II].FIELDX-1,L[II].FLAG1X,16,WHERE);
280   END; {RACE}
281   IF II<=B.LSINGLE THEN TI:=1 ELSE TI:=4;
282   FOR TI:=1TO TI DO BEGIN
283     CIRCLE(L[II].W[TI].X,L[II].W[TI].Y,2,12);
284     LINE(L[II].W[TI].X,L[II].W[TI].Y,L[II].W[TI].X+ROUND(SIN(L[II].W[TI].D/128*PI)*20),
285       L[II].W[TI].Y+ROUND(COS(L[II].W[TI].D/128*PI)*20),14,WHERE);
286     LINE(L[II].W[TI].X,L[II].W[TI].Y,L[II].W[TI].X+ROUND(SIN((L[II].W[TI].D-13)/128*PI)*15),
287       L[II].W[TI].Y+ROUND(COS((L[II].W[TI].D-13)/128*PI)*15),12,WHERE);
288     LINE(L[II].W[TI].X,L[II].W[TI].Y,L[II].W[TI].X+ROUND(SIN((L[II].W[TI].D+13)/128*PI)*15),
289       L[II].W[TI].Y+ROUND(COS((L[II].W[TI].D+13)/128*PI)*15),12,WHERE);
290     PUTPIXEL(L[II].W[TI].X,L[II].W[TI].Y,0,WHERE);
291   END; {WORMS}
292   IF II>B.LSINGLE+B.LMULTI+B.LRACE THEN BEGIN
293     IF L[II].SPRSIZE>0THEN TI:=L[II].SPRSIZE ELSE TI:=B.SPRSIZE;
294     DRAW4PIX(L[II].FLAG1X,L[II].FLAG1Y,TI,TI,26); {FLAGS}
295     DRAW4PIX(L[II].FLAG2X,L[II].FLAG2Y,TI,TI,28);
296   END; {DRAW FLAGS}
297   LINE(0,0,255,0,22,WHERE); LINE(0,1,255,1,22,WHERE); {BORDERS}
298   LINE(0,2,0,190-3,22,WHERE); LINE(1,2,1,190-3,22,WHERE);
299   LINE(MAX(126,L[II].FIELDX-2),2,MAX(126,L[II].FIELDX-2),L[II].FIELDY-1,22,WHERE);
300   LINE(MAX(127,L[II].FIELDX-1),2,MAX(127,L[II].FIELDX-1),L[II].FIELDY-1,22,WHERE);
301   LINE(0,L[II].FIELDY-2,L[II].FIELDX-1,L[II].FIELDY-2,22,WHERE);
302   LINE(0,L[II].FIELDY-1,L[II].FIELDX-1,L[II].FIELDY-1,22,WHERE);
303   FOR OBJ2:=1TO L[II].NROBJ DO IF OBJ2<>OBJ THEN DISPOBJ(OBJ2,18); {OBJS}
304   IF(OBJ>0)AND(OBJ<=L[II].NROBJ)THEN DISPOBJ(OBJ,20); {HIGHLIGHTED_OBJ}
305   FOR OBJ2:=1TO L[II].NRBALLS DO
306     DRAW4PIX(L[II].BALLS[OBJ2].X,L[II].BALLS[OBJ2].Y,2,2,20); {BOUNCIES}
307 END; {DISPMAP}
308
309 PROCEDURE DISPMCGACOMMON;
310 BEGIN
311   SETMCGA;
312   SETPAL(0,0,0,0);              SETPAL(1,32,0,0); {BACK}
313   SETPAL(2,8,0,0);              SETPAL(3,35,0,0); {SUBGRID}
314   SETPAL(4,12,0,0);             SETPAL(5,38,0,0); {GRID}
315   SETPAL(6,20,0,0);             SETPAL(7,40,0,0); {MID}
316   SETPAL(8,28,0,0);             SETPAL(9,44,0,0); {MAINGRID}
317
318   SETPAL(12,48,32,0);           SETPAL(13,44,0,0); {STLN}
319   SETPAL(14,48,48,0);           SETPAL(15,52,0,0); {STARTPOS}
320   SETPAL(26,0,0,63);            SETPAL(27,63,0,0); {FLAG1}
321   SETPAL(28,0,63,0);            SETPAL(29,63,0,0); {FLAG2}
322
323   SETPAL(16,32,32,32);          SETPAL(17,48,0,0); {HALFOBJ}
324   SETPAL(18,48,48,48);          SETPAL(19,63,0,0); {OBJ}
325   SETPAL(20,63,63,20);          SETPAL(21,63,16,0); {HL_OBJ}
326   SETPAL(22,63,63,63);          SETPAL(23,63,0,0); {BORDER}
327 END; {DISPMCGACOMMON}
328
329 PROCEDURE DISP1MAP(OBJ:INTEGER);
330 VAR MXX,MYY:WORD;
331     M1,M2:BOOLEAN;
332 {TYPE SCRN = ARRAY[0..199,0..319]OF BYTE;
333 VAR VIRT:^SCRN;
334     I1,I1X,I1Y:BYTE;
335 BEGIN
336   NEW(VIRT);
337   FOR TI:=0 TO 199DO FOR I2:=0 TO 319DO VIRT^[I,I2]:=0; DISPMAP(OBJ,VIRT);
338   WINDOW(1,3,80,20);
339   FOR I:=0 TO 19DO FOR I2:=0 TO 79DO BEGIN
340     I1X:=ROUND(I2*L[II].FIELDX/80); I1Y:=ROUND(I*L[II].FIELDY/20);
341     CASE VIRT^[I1Y,I1X]OF
342       0:TEXTCOLOR(0);
343       1:TEXTCOLOR(8);
344       2:TEXTCOLOR(7);
345       3:TEXTCOLOR(15);
346       4,5:TEXTCOLOR(14);
347       6..9:TEXTCOLOR(9);
348       10..12:TEXTCOLOR(12);
349     END;
350     WRITE('�');
351   END;
352   DISPOSE(VIRT);}
353 BEGIN
354   DISPMCGACOMMON; DISPMAP(OBJ,PTR(VGA,0)); CLRKEY;
355   REPEAT
356     IF USEMOUSE THEN BEGIN
357       GETMOUSEPOS(MX,MY,M1,M2); MX:=MX SHR 1; {MY:=MY SHR 1;}
358       PUTPIXEL(MX,MY,GETPIXEL(MX,MY,VGA)XOR 1,PTR(VGA,0));
359       DELAY(10);
360       PUTPIXEL(MX,MY,GETPIXEL(MX,MY,VGA)XOR 1,PTR(VGA,0));
361       IF M1 THEN PPLINE(MXX,MYY,MX,MY,0);
362       IF M2 THEN PPLINE(MXX,MYY,MX,MY,1);
363       MXX:=MX; MYY:=MY;
364     END; {MOUSE}
365   UNTIL KEYPRESSED; WHILE KEYPRESSED DO READKEY; MAINTITLE;
366 END; {DISP1MAP}
367
368 PROCEDURE DISPMANYMAP(MAG:BYTE);
369 TYPE SCRN = ARRAY[0..199,0..319]OF BYTE;
370 VAR VIRT:^SCRN;
371 PROCEDURE DISP1MANYMAP(OFFSETX,OFFSETY,MAG:BYTE);
372 VAR I1,I1X,I1Y,I1XX,I1YY:BYTE;
373 BEGIN
374   FOR I:=0 TO 199DO FOR I2:=0 TO 255DO VIRT^[I,I2]:=0; DISPMAP(0,VIRT);
375   FOR I:=0 TO 255 DIV MAG DO FOR I2:=0 TO 199 DIV MAG DO BEGIN
376     {I1X:=ROUND(I2*L[II].FIELDX/80); I1Y:=ROUND(I*L[II].FIELDY/20);}
377     I1:=0;
378     FOR I1XX:=I*MAG TO(I+1)*MAG-1 DO FOR I1YY:=I2*MAG TO(I2+1)*MAG-1DO
379       IF VIRT^[I1YY,I1XX]>I1 THEN I1:=VIRT^[I1YY,I1XX];
380     PUTPIXEL(OFFSETX+I,OFFSETY+I2,I1,PTR(VGA,0));
381   END;
382 END; {DISP1MANYMAP}
383 VAR MXX,MYY:WORD; M1,M2:BOOLEAN;
384 BEGIN
385   DISPMCGACOMMON; NEW(VIRT);
386   MX:=255 DIV MAG+1; MY:=192 DIV MAG+1;
387   FOR II:=1TO MIN(B.LTOTAL,SQR(MAG))DO
388     DISP1MANYMAP(((II-1)MOD MAG)*MX,(II-1)DIV MAG*MY,MAG);
389   DISPOSE(VIRT); CLRKEY;
390   REPEAT
391     IF USEMOUSE THEN BEGIN
392       GETMOUSEPOS(MX,MY,M1,M2); MX:=MX SHR 1; {MY:=MY SHR 1;}
393       PUTPIXEL(MX,MY,GETPIXEL(MX,MY,VGA)XOR 1,PTR(VGA,0));
394       DELAY(10);
395       PUTPIXEL(MX,MY,GETPIXEL(MX,MY,VGA)XOR 1,PTR(VGA,0));
396       IF M1 THEN PPLINE(MXX,MYY,MX,MY,0);
397       IF M2 THEN PPLINE(MXX,MYY,MX,MY,1);
398       MXX:=MX; MYY:=MY;
399     END; {MOUSE}
400   UNTIL KEYPRESSED; WHILE KEYPRESSED DO READKEY; MAINTITLE;
401 END; {DISP1MAP}
402
403 PROCEDURE MAPOBJ(OBJ,OBJSUB:INTEGER);
404 VAR M1,M2:BOOLEAN;
405     MXX,MYY:WORD;
406     XMIN,XDEF,XMAX,YMIN,YDEF,YMAX:INTEGER;
407 BEGIN
408   DISPMCGACOMMON; DISPMAP(OBJ,PTR(VGA,0)); CLRKEY;
409   IF OBJSUB=0THEN BEGIN
410     XDEF:=L[II].OBJ[OBJ].X1; YDEF:=L[II].OBJ[OBJ].Y1;
411   END {FROM} ELSE BEGIN
412     XDEF:=L[II].OBJ[OBJ].X2; YDEF:=L[II].OBJ[OBJ].Y2;
413   END; {TO}
414   CASE L[II].OBJ[OBJ].TYP OF
415     1:BEGIN XMIN:=2; XMAX:=L[II].FIELDX-3; YMIN:=2; YMAX:=L[II].FIELDY-3;
416     END; {LINE}
417     2:BEGIN XMIN:=2; XMAX:=L[II].FIELDX-4; YMIN:=2; YMAX:=L[II].FIELDY-4;
418     END; {FATTY}
419     3:BEGIN XMIN:=2; XMAX:=L[II].FIELDX-1; YMIN:=2; YMAX:=L[II].FIELDY-2;
420       IF OBJSUB=0THEN YMAX:=L[II].OBJ[OBJ].Y1+L[II].OBJ[OBJ].Y2-1
421       ELSE YMIN:=L[II].OBJ[OBJ].Y1+1;
422     END; {BOX}
423     4:BEGIN XMIN:=0;
424       IF OBJSUB=0THEN BEGIN
425         XMAX:=255; YMIN:=0; YMAX:=255;
426       END {POS} ELSE BEGIN
427         XMAX:=120; XDEF:=ABS(INTEGER(L[II].OBJ[OBJ].X1)-L[II].OBJ[OBJ].X2);
428         YMIN:=L[II].OBJ[OBJ].Y1; YDEF:=L[II].OBJ[OBJ].Y1; YMAX:=L[II].OBJ[OBJ].Y1;
429       END; {SIZE}
430     END; {CIRCLE}
431   END; {TYPE}
432   MY:=YDEF;
433   IF USEMOUSE THEN BEGIN MX:=XDEF SHL 1; MOUSE(4,MX,MX,MX,MY); END; {INITMOUSE}
434   MX:=XDEF;
435   REPEAT
436     PPDISPOBJ(OBJ,0);
437     IF USEMOUSE THEN BEGIN
438       REPEAT GETMOUSEPOS(MX,MY,M1,M2);
439       UNTIL(MX<>MXX)OR(MY<>MYY)OR(M1)OR(M2)OR(KEYPRESSED);
440       MXX:=MX; MYY:=MY; MX:=MX SHR 1;
441       IF KEYPRESSED THEN BEGIN C:=READKEY;
442         IF(C=#13)OR(C=' ')THEN M1:=TRUE; IF(C=#27)OR(C='+')THEN M2:=TRUE;
443       END; {KEYS}
444     END {MOUSE} ELSE BEGIN
445       REPEAT UNTIL KEYPRESSED; C:=READKEY;
446       IF(C=#13)OR(C=' ')THEN M1:=TRUE ELSE M1:=FALSE;
447       IF(C=#27)OR(C='+')THEN M2:=TRUE ELSE M2:=FALSE;
448       IF C='2'THEN INC(MY); {DOWN} IF C='8'THEN DEC(MY); {UP}
449       IF C='4'THEN DEC(MX); {L}    IF C='6'THEN INC(MX); {R}
450       IF C=#0THEN BEGIN  C:=READKEY;
451         IF C=#80 THEN INC(MY); {DOWN} IF C=#72 THEN DEC(MY); {UP}
452         IF C=#75 THEN DEC(MX); {L}    IF C=#77 THEN INC(MX); {R}
453       END; {EXTENDED}
454     END; {KEYB}
455     IF NOT M1 THEN BEGIN
456       IF MX<XMIN THEN MX:=XMIN; IF MX>XMAX THEN MX:=XMAX;
457       IF MY<YMIN THEN MY:=YMIN; IF MY>YMAX THEN MY:=YMAX;
458       IF M2 THEN BEGIN MX:=XDEF; MY:=YDEF; END; {R_BUTTON}
459       IF(USEMOUSE)AND((MX<>MXX)OR(MY<>MYY))THEN BEGIN
460         MX:=MX SHL 1; MOUSE(4,MX,MX,MX,MY); MX:=MX SHR 1;
461       END; {SET_NEW_MOUSE_POS}
462       PPDISPOBJ(OBJ,1);
463       IF OBJSUB=0THEN CASE L[II].OBJ[OBJ].TYP OF
464         3:BEGIN
465           INC(L[II].OBJ[OBJ].Y2,L[II].OBJ[OBJ].Y1-MY);
466           L[II].OBJ[OBJ].X1:=MX; L[II].OBJ[OBJ].Y1:=MY;
467         END; {BOX}
468         ELSE BEGIN L[II].OBJ[OBJ].X1:=MX; L[II].OBJ[OBJ].Y1:=MY; END; {DEF}
469       END {FROM} ELSE  CASE L[II].OBJ[OBJ].TYP OF
470         3:BEGIN L[II].OBJ[OBJ].X2:=MX; L[II].OBJ[OBJ].Y2:=MY-L[II].OBJ[OBJ].Y1;
471         END; {BOX}
472         4:BEGIN L[II].OBJ[OBJ].X2:=ABS(MX-INTEGER(L[II].OBJ[OBJ].X1)); L[II].OBJ[OBJ].Y2:=0; END; {CIRCLE}
473         ELSE BEGIN L[II].OBJ[OBJ].X2:=MX; L[II].OBJ[OBJ].Y2:=MY; END; {DEF}
474       END; {TO}
475     END; {K}
476   UNTIL M1; MAINTITLE;
477   IF USEMOUSE THEN WHILE(M1)OR(M2)DO GETMOUSEPOS(MX,MY,M1,M2);
478   WHILE KEYPRESSED DO C:=READKEY;
479   IF(L[II].OBJ[OBJ].TYP=3)AND(L[II].OBJ[OBJ].Y1<0)THEN BEGIN
480     I4:=L[II].OBJ[OBJ].Y1; L[II].OBJ[OBJ].Y1:=L[II].OBJ[OBJ].Y2;
481     L[II].OBJ[OBJ].Y1:=-I4;
482   END; {SWAP Y FOR BOX}
483 END; {MAPOBJ}
484
485 FUNCTION NR(VALUE:LONGINT):STRING;
486 VAR NRSTR:STRING;
487 BEGIN
488   STR(VALUE,NRSTR); NR:=NRSTR;
489 END; {NR}
490
491 PROCEDURE HELPBAR(HELPPAGE:BYTE);
492 BEGIN
493   WINDOW(41,14,80,24);
494   IF HELPPAGE=0 THEN BEGIN
495     TEXTCOLOR(C_GRID); WRITE('�'); WINDOW(42,14,80,24); CLRSCR; EXIT;
496   END; {REMOVE}
497   TEXTBACKGROUND(CB_BAR); TEXTCOLOR(C_GRID); WRITE('�'); TEXTCOLOR(C_BAR);
498   WRITE('HELP',' ':39-4); WINDOW(42,15,80,24); TEXTBACKGROUND(CB_TEXT); TEXTCOLOR(C_TEXT);
499   CLRSCR; CASE HELPPAGE OF
500     1:WRITE('PRESS <F1> OR <H> FOR HELP');
501     2:WRITE('YOU UNDERSTAND EVERYTHING');
502     3:WRITE('THIS AIN''T A BUG, IT A FEATURE!');
503     4:WRITE('ALL YOUR WORMS ARE BELONG TO US!');
504     5:WRITE('PRESS UP AND DOWN TO VIEW ALL LEVELS');
505     6:WRITE('PRESS UP AND DOWN TO VIEW ALL OBJECTS');
506   END; {TEXT TO DISPLAY}
507 END; {HELPBAR}
508
509 PROCEDURE DISPLEVELS;
510 BEGIN
511   WINDOW(42,3,80,13); TEXTCOLOR(C_TEXT); TEXTBACKGROUND(CB_TEXT); CLRSCR;
512   IF B.LTOTAL<1 THEN WRITELN('NONE PRESENT')ELSE
513     FOR II:=LVLPOS TO MIN(B.LTOTAL,LVLPOS+9)DO BEGIN
514       IF LVLDISP=1 THEN WRITE(II) {CONTINOUS}
515       ELSE IF II>B.LSINGLE+B.LMULTI+B.LRACE THEN WRITE(II-B.LSINGLE-B.LMULTI-B.LRACE)ELSE
516         IF II>B.LSINGLE+B.LMULTI THEN WRITE(II-B.LSINGLE-B.LMULTI)ELSE
517           IF II>B.LSINGLE THEN WRITE(II-B.LSINGLE)ELSE WRITE(II); {SEPERATE}
518       IF II>=B.LFROM[1]THEN TEXTCOLOR(C_VALUE);
519       WRITELN(': '+L[II].ID); TEXTCOLOR(C_TEXT);
520     END; {DISP LEVELS}
521 END; {DISPLEVELS}
522
523 PROCEDURE DISPOBJECTS;
524 BEGIN
525   WINDOW(42,3,80,13); TEXTCOLOR(C_TEXT); TEXTBACKGROUND(CB_TEXT); CLRSCR;
526   IF L[II].NROBJ<1THEN WRITELN('NONE')
527   ELSE FOR I4:=OBJPOS TO MIN(L[II].NROBJ,OBJPOS+9)DO BEGIN
528     WRITE(I4); TEXTCOLOR(C_VALUE);
529     IF L[II].OBJ[I4].TYP>0 THEN BEGIN
530       WRITE(': ',OBJTYPE[L[II].OBJ[I4].TYP]);
531       WRITE(' (',L[II].OBJ[I4].X1,',',L[II].OBJ[I4].Y1,')-(');
532       CASE L[II].OBJ[I4].TYP OF
533         1,2:WRITELN(L[II].OBJ[I4].X2,',',L[II].OBJ[I4].Y2,')');
534         3:WRITELN(L[II].OBJ[I4].X2,',',L[II].OBJ[I4].Y1+L[II].OBJ[I4].Y2,')');
535         4:WRITELN(L[II].OBJ[I4].X2,')');
536       END; {TYPE}
537     END {ENABLED} ELSE WRITELN(': DISABLED'); TEXTCOLOR(C_TEXT);
538   END; {DISP OBJS}
539 END; {DISPOBJECTS}
540
541 FUNCTION EDITMENU(EDITMENUMAX,HELPPAGE:BYTE):SHORTINT;
542 VAR EDITOK,MENUPOS:SHORTINT;
543 BEGIN
544   MENUPOS:=1; EDITOK:=0;
545   WINDOW(2,3,3,20); GOTOXY(1,1); TEXTCOLOR(C_MENU); WRITE('>>');
546   REPEAT
547     GOTOXY(1,1); REPEAT UNTIL KEYPRESSED; C:=READKEY;
548     IF C=' 'THEN EDITOK:=10 {10} ELSE
549     IF(C>='1')AND(C<=CHR(EDITMENUMAX+48))AND(C<='9')THEN EDITOK:=ORD(C)-48 {1..<=9} ELSE
550     IF(EDITMENUMAX>9)AND(UPCASE(C)>='A')AND(UPCASE(C)<=CHR(EDITMENUMAX+55))AND(UPCASE(C)<='Z')THEN
551       EDITOK:=ORD(UPCASE(C))-55 {1..<=9} ELSE
552     IF C=#13 THEN EDITOK:=MENUPOS {ENTER} ELSE
553     IF(C=#27)OR(C='-')OR(C='+')THEN EDITOK:=-1 {EXIT} ELSE
554     IF UPCASE(C)='H'THEN HELPBAR(HELPPAGE) ELSE
555     IF C=#0THEN BEGIN  C:=READKEY;
556       IF(C=#80)AND(MENUPOS<EDITMENUMAX)THEN BEGIN
557         TEXTCOLOR(C_TEXT); GOTOXY(1,MENUPOS); WRITE(': ');
558         INC(MENUPOS); TEXTCOLOR(C_MENU); GOTOXY(1,MENUPOS); WRITE('>>');
559       END; {DOWN}
560       IF(C=#72)AND(MENUPOS>1)THEN BEGIN
561         TEXTCOLOR(C_TEXT); GOTOXY(1,MENUPOS); WRITE(': ');
562         DEC(MENUPOS); TEXTCOLOR(C_MENU); GOTOXY(1,MENUPOS); WRITE('>>');
563       END; {UP}
564       IF HELPPAGE=5THEN BEGIN
565         IF(C=#77)AND(LVLPOS<B.LTOTAL-9)THEN BEGIN INC(LVLPOS); DISPLEVELS; END {R} ELSE
566         IF(C=#75)AND(LVLPOS>1)THEN BEGIN DEC(LVLPOS); DISPLEVELS; END {L} ELSE
567       END {LEVELS DISPED} ELSE IF HELPPAGE=6THEN BEGIN
568         IF(C=#77)AND(OBJPOS<L[II].NROBJ-9)THEN BEGIN INC(OBJPOS); DISPOBJECTS; END {R} ELSE
569         IF(C=#75)AND(OBJPOS>1)THEN BEGIN DEC(OBJPOS); DISPOBJECTS; END {L} ELSE
570       END; {OBJS DISPED}
571       IF C=#59THEN HELPBAR(HELPPAGE);
572     END; {EXTENDED}
573   UNTIL EDITOK<>0; EDITMENU:=EDITOK; TEXTCOLOR(C_TEXT);
574 END; {EDITMENU}
575
576 FUNCTION COMPSIZE:WORD;
577 PROCEDURE COMPLEVEL;
578 BEGIN
579   INC(I,L[II].SIZE);
580   FOR I4:=1TO L[II].NROBJ DO IF L[II].OBJ[I4].TYP<=0THEN DEC(I,5);
581 END; {SIZE OF LEVEL II}
582 BEGIN
583   I:=$31+B.SPRSIZE+ORD(B.NAME[0])+ORD(B.DESC[0])+6;
584   IF B.LFROM[2]>0THEN I:=I+((1+B.LTO[2]-B.LFROM[2])*5); {HI_PW}
585   IF B.LFROM[3]>0THEN I:=I+((1+B.LTO[3]-B.LFROM[3])*5); {HI_TRON}
586   IF B.LFROM[1]>0THEN FOR II:=B.LFROM[1]TO B.LSINGLE DO COMPLEVEL; {SP}
587   IF B.LMULTI>0THEN FOR II:=B.LSINGLE+1TO B.LSINGLE+B.LMULTI DO COMPLEVEL; {MP}
588   IF B.LFROM[7]>0THEN
589    FOR II:=B.LSINGLE+B.LMULTI+B.LFROM[7]TO B.LSINGLE+B.LMULTI+B.LTO[7]DO COMPLEVEL; {RACE}
590   IF B.LFROM[8]>0THEN
591    FOR II:=B.LSINGLE+B.LMULTI+B.LRACE+B.LFROM[8]TO B.LSINGLE+B.LMULTI+B.LRACE+B.LTO[8]DO COMPLEVEL; {CTF}
592
593 {MAKE SP_END CODE}
594   CASE B.ENDTYPE OF
595     1:B.ENDDATA:=#$21+#$F4+#$00+#$C3+#$37+#$4A+#$CD+#$AA+#$55+B.ENDSTR;
596       {LD HL,MSG\JP _PUTS\.DB MSG}
597     ELSE B.ENDDATA:=#$C9; {RET}
598   END; {ENDTYPE}
599   INC(I,ORD(B.ENDDATA[0]));
600
601   COMPSIZE:=I;
602 END; {COMPSIZE}
603
604 PROCEDURE TITLEBAR(BAR:SHORTINT;TITLNAME:STRING);
605 BEGIN
606   HELPBAR(0); WINDOW(41,2,42,2);
607   IF BAR=0THEN TEXTBACKGROUND(CB_TEXT)ELSE TEXTBACKGROUND(CB_BAR);
608   TEXTCOLOR(C_GRID); WRITE('�'); WINDOW(42,2,80,13); TEXTBACKGROUND(CB_TEXT); CLRSCR;
609   TEXTCOLOR(C_BAR); TEXTBACKGROUND(CB_BAR);
610   CASE BAR OF
611     1:BEGIN
612       WRITE('LEVELS',' ':39-6); LVLDISP:=1; DISPLEVELS;
613     END; {1:SINGLVLS}
614     2:BEGIN
615       WRITE('LEVEL OBJECTS',' ':39-13); DISPOBJECTS;
616     END; {2:OBJS}
617     3:BEGIN
618       WRITE('LEVEL INFO',' ':39-10); TEXTCOLOR(C_TEXT); TEXTBACKGROUND(CB_TEXT);
619       WRITE('LEVEL #',II);
620       IF II>B.LSINGLE+B.LMULTI+B.LRACE THEN WRITELN(' CTF #',II-B.LSINGLE-B.LMULTI-B.LRACE)ELSE
621         IF II>B.LSINGLE+B.LMULTI THEN WRITELN(' RACE #',II-B.LSINGLE-B.LMULTI)ELSE
622           IF II>B.LSINGLE THEN WRITELN(' MP #',II-B.LSINGLE)ELSE WRITELN;
623       WRITELN('LEVEL SIZE: ',L[II].SIZE);
624       IF II<=B.LSINGLE THEN WRITELN('SINGLEPLAYER')ELSE FOR I:=2TO 6DO
625         IF(II-B.LSINGLE>=B.LFROM[I])AND(II-B.LSINGLE<=B.LTO[I])THEN
626           WRITELN(GAMETYPE[I]);
627     END; {3:LEVEL}
628     4:BEGIN
629       WRITE(FILENAME,'.LVL',' ':35-ORD(FILENAME[0])); TEXTCOLOR(C_TEXT);
630       TEXTBACKGROUND(CB_TEXT); WRITE('FILE VERSION: '); TEXTCOLOR(C_VALUE);
631       WRITELN(B.VERSION); TEXTCOLOR(C_TEXT); WRITE('FILE SIZE: '); TEXTCOLOR(C_VALUE);
632       WRITE(FSIZE SHR 10); TEXTCOLOR(C_TEXT); WRITELN('kB');
633       WRITE('COMPILED: '); TEXTCOLOR(C_VALUE); WRITE(COMPSIZE); TEXTCOLOR(C_TEXT);
634       WRITE(' BYTES');
635     END; {4:FILE}
636     5:BEGIN
637       WRITE('LEVELS',' ':39-6); LVLDISP:=2; DISPLEVELS;
638     END; {5:LEVELS}
639   END; {BAR}
640   WINDOW(1,2,40,24);
641   TEXTBACKGROUND(CB_BAR); TEXTCOLOR(C_BAR); WRITE(TITLNAME,' ':40-ORD(TITLNAME[0]));
642   WINDOW(1,3,40,24); TEXTBACKGROUND(CB_TEXT); CLRSCR;
643 END; {TITLEBAR}
644
645 FUNCTION INPUTSTR(ISTR,ICOM,IDEF:STRING;IMAX:BYTE):STRING;
646 VAR IVAL:STRING; IOK:BOOLEAN;
647 BEGIN
648   WINDOW(1,14,40,24);
649   TEXTBACKGROUND(CB_BAR); TEXTCOLOR(C_BAR); WRITE('INPUT',' ':40-5);
650   WINDOW(1,15,40,24); TEXTBACKGROUND(CB_TEXT); CLRSCR;
651   REPEAT
652     TEXTCOLOR(C_TEXT); WRITELN(ISTR); TEXTCOLOR(C_SUB); WRITELN(ICOM);
653     TEXTCOLOR(C_SUB); WRITELN('ENTER FOR DEFAULT:'); WRITELN(IDEF);
654     TEXTCOLOR(C_TEXT); WRITE('> '); READLN(IVAL);
655     IF IVAL=''THEN IVAL:=IDEF;
656     IOK:=FALSE; IF ORD(IVAL[0])<=IMAX THEN IOK:=TRUE
657       ELSE BEGIN CLRSCR; WRITELN('OH NO, THAT''S TOO MUCH!'); END; {TOOLONG}
658   UNTIL IOK; INPUTSTR:=IVAL;
659 END; {INPUTSTR}
660
661 FUNCTION INPUTINT(ISTR,ICOM:STRING;IMIN,IDEF,IMAX:INTEGER):INTEGER;
662 FUNCTION READ:INTEGER;
663 VAR REI1:INTEGER; REI2:BYTE;
664 BEGIN {INPUTS POSITIVE INTEGER VALUE FROM KEYBOARD}
665   REI1:=0; {VALUE} REI2:=0; {SIZE}
666   REPEAT
667     REPEAT C:=READKEY; UNTIL C<>'';
668     CASE C OF
669       #8,'+','-':IF REI2>0THEN BEGIN
670         REI1:=REI1 DIV 10; DEC(REI2); WRITE(#8+' '+#8);
671       END; {BACKSPACE}
672       '0'..'9':IF REI2<4THEN BEGIN
673         REI1:=REI1*10+ORD(C)-48; INC(REI2); WRITE(C);
674       END; {NUMBER}
675       #13:IF REI2=0THEN REI1:=-1; {ENTER}
676     END; {HANDLE INPUT}
677   UNTIL C=#13; {UNTIL ENTER PRESSED}
678   READ:=REI1;
679 END; {READLN}
680 VAR IVAL:INTEGER; IOK:BOOLEAN;
681 BEGIN
682   WINDOW(1,14,40,24);
683   TEXTBACKGROUND(CB_BAR); TEXTCOLOR(C_BAR); WRITE('INPUT',' ':40-5);
684   WINDOW(1,15,40,24); TEXTBACKGROUND(CB_TEXT); CLRSCR;
685   REPEAT
686     TEXTCOLOR(C_TEXT); WRITELN(ISTR); TEXTCOLOR(C_SUB); WRITELN(ICOM);
687     TEXTCOLOR(C_SUB); WRITE('(RANGE '); TEXTCOLOR(C_TEXT); WRITE(IMIN); TEXTCOLOR(C_SUB);
688     WRITE(' TO '); TEXTCOLOR(C_TEXT); WRITE(IMAX); TEXTCOLOR(C_SUB);
689     WRITE('; DEFAULT VALUE '); TEXTCOLOR(C_TEXT); WRITE(IDEF); TEXTCOLOR(C_SUB);
690     WRITELN(')'); TEXTCOLOR(C_TEXT); WRITE('> '); IVAL:=READ; IF IVAL=-1THEN IVAL:=IDEF;
691     IOK:=FALSE; IF(IVAL>=IMIN)AND(IVAL<=IMAX)THEN IOK:=TRUE
692       ELSE BEGIN CLRSCR; WRITELN('OUT OF RANGE!'); END; {OUTOFRANGE}
693   UNTIL IOK; INPUTINT:=IVAL;
694 END; {INPUTINT}
695
696 PROCEDURE EDITGAME;
697 LABEL EDITGAME1,EDITGAME2;
698 BEGIN
699   REPEAT
700     TITLEBAR(5,'EDIT GAMETYPES'); TEXTCOLOR(C_MENU); WRITE('1');
701     TEXTCOLOR(C_TEXT); WRITE(': SINGLEPLAYER ');
702     TEXTCOLOR(C_VALUE); IF(B.LSINGLE=0)OR(B.LFROM[1]=0)THEN WRITELN('(NO LEVELS)')
703       ELSE WRITELN('(',B.LTO[1]-B.LFROM[1]+1,' OF ',B.LSINGLE,')');
704     FOR II:=2TO 6DO BEGIN
705       TEXTCOLOR(C_MENU); WRITE(II); TEXTCOLOR(C_TEXT); WRITE(': '+GAMETYPE[II],' ');
706       TEXTCOLOR(C_VALUE); IF B.LFROM[II]=0THEN WRITELN('(NO LEVELS)')
707         ELSE WRITELN('(',B.LFROM[II],'-',B.LTO[II],')');
708     END; {MULTIP}
709     TEXTCOLOR(C_MENU); WRITE('7'); TEXTCOLOR(C_TEXT); WRITE(': RACE '); TEXTCOLOR(C_VALUE);
710     IF B.LFROM[7]=0THEN WRITELN('(NO LEVELS)')ELSE WRITELN('(',B.LTO[7]-B.LFROM[7]+1,' OF ',B.LRACE,')');
711     TEXTCOLOR(C_MENU); WRITE('8'); TEXTCOLOR(C_TEXT); WRITE(': CTF '); TEXTCOLOR(C_VALUE);
712     IF B.LFROM[8]=0THEN WRITE('(NO LEVELS)')ELSE WRITE('(',B.LTO[8]-B.LFROM[8]+1,' OF ',B.LCTF,')');
713
714     III:=EDITMENU(8,5);
715     IF III=1THEN BEGIN
716       B.LFROM[1]:=INPUTINT('ENTER LEVELS TO SKIP','USE THIS TO TEMPORARILY REMOVE THE FIRSTFEW LEVELS',
717         0,0,MAX(0,B.LSINGLE-1)); INC(B.LFROM[1]);
718       B.LTO[1]:=INPUTINT('ENTER LEVELS SELECTABLE','THE HIGHEST LEVEL SOMEONE CAN START IN  WITHOUT PLAYING IT EARLIER',
719         0,1,B.LSINGLE+1-B.LFROM[1]);
720       IF B.LTO[1]=0THEN B.LFROM[1]:=0 ELSE B.LTO[1]:=B.LTO[1]+B.LFROM[1]-1;
721     END {SINGLEPLAYER} ELSE IF III=7THEN BEGIN
722       B.LFROM[7]:=INPUTINT('ENTER STARTING LEVEL','USUALLY 1 UNLESS YOU WANT TO SKIP LEVELS0=NO LEVELS',0,1,B.LRACE);
723       IF B.LFROM[7]>0THEN
724         B.LTO[7]:=INPUTINT('ENTER LAST LEVEL','',B.LFROM[7],B.LRACE,B.LRACE);
725     END {RACE} ELSE IF III=8THEN BEGIN
726       B.LFROM[8]:=INPUTINT('ENTER STARTING LEVEL','USUALLY 1 UNLESS YOU WANT TO SKIP LEVELS0=NO LEVELS',0,1,B.LCTF);
727       IF B.LFROM[8]>0THEN
728         B.LTO[8]:=INPUTINT('ENTER LAST LEVEL','',B.LFROM[8],B.LCTF,B.LCTF);
729     END {CTF} ELSE IF III>0THEN BEGIN
730       B.LFROM[III]:=INPUTINT('ENTER NEW STARTING LEVEL',
731         'THAT LEVEL WILL BE LEVEL #1 IN GAMEMODE '+GAMETYPE[III]+'. 0=NO LEVELS',0,1,B.LMULTI);
732       IF B.LFROM[III]>0THEN
733         B.LTO[III]:=INPUTINT('ENTER LAST LEVEL NUMBER',NR(B.LMULTI)+' LEVELS PRESENT',
734           B.LFROM[III],B.LMULTI,B.LMULTI);
735     END; {MULTIPLAYER}
736   UNTIL III=-1; III:=0;
737 END; {EDITGAME}
738
739 PROCEDURE EDITOBJ;
740 LABEL EDITOBJ1;
741 BEGIN
742   REPEAT
743     TITLEBAR(2,'EDIT OBJECT #'+NR(I)+' IN LEVEL '+NR(II));
744     TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITE(': OBJECT TYPE ');
745     III:=L[II].OBJ[I].TYP; TEXTCOLOR(C_VALUE); WRITELN('(',ABS(III),')');
746     IF III<>0THEN BEGIN
747       IF L[II].OBJ[I].TYP>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT); WRITE('2'); TEXTCOLOR(C_TEXT);
748       IF ABS(L[II].OBJ[I].TYP)=4THEN WRITE(': OBJECT POSITION ')ELSE WRITE(': OBJECT BEGIN');
749       IF L[II].OBJ[I].TYP>0THEN BEGIN
750         TEXTCOLOR(C_VALUE); WRITELN(' (',L[II].OBJ[I].X1,',',L[II].OBJ[I].Y1,')');
751       END {PRESENT} ELSE WRITELN;
752       IF L[II].OBJ[I].TYP>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT);
753       WRITE('3'); TEXTCOLOR(C_TEXT); WRITELN(': SELECT');
754       IF L[II].OBJ[I].TYP>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT); WRITE('4'); TEXTCOLOR(C_TEXT);
755       IF ABS(L[II].OBJ[I].TYP)=4THEN WRITE(': OBJECT SIZE ')ELSE WRITE(': OBJECT END ');
756       TEXTCOLOR(C_VALUE); CASE L[II].OBJ[I].TYP OF
757         3:WRITELN('(',L[II].OBJ[I].X2,',',L[II].OBJ[I].Y1+L[II].OBJ[I].Y2,')');
758         4:WRITELN('(',L[II].OBJ[I].X2,')');
759         1..10:WRITELN('(',L[II].OBJ[I].X2,',',L[II].OBJ[I].Y2,')');
760         ELSE WRITELN;
761       END; {TYPE}
762       IF L[II].OBJ[I].TYP>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT);
763       WRITE('5'); TEXTCOLOR(C_TEXT); WRITELN(': SELECT');
764       TEXTCOLOR(C_MENU); WRITE('6'); TEXTCOLOR(C_TEXT); WRITELN(': PREVIEW');
765       TEXTCOLOR(C_MENU); WRITE('7'); TEXTCOLOR(C_TEXT);
766       IF L[II].OBJ[I].TYP<0THEN WRITE(': ENABLE')ELSE WRITE(': DISABLE');
767     END; {OBJECT PROPERTIES}
768
769     IF III<>0THEN III:=EDITMENU(7,6)ELSE III:=EDITMENU(1,6);
770     CASE III OF
771       1:BEGIN
772         S:=''; FOR I4:=0TO 4DO S:=S+NR(I4)+': '+OBJTYPE[I4]+CC(' ',37-ORD(OBJTYPE[I4,0]));
773         L[II].OBJ[I].TYP:=INPUTINT('ENTER OBJECT TYPE',S,0,0,4);
774         IF L[II].OBJ[I].TYP=4THEN BEGIN
775           L[II].OBJ[I].X1:=(L[II].FIELDX-1)SHR 1;
776           L[II].OBJ[I].Y1:=(L[II].FIELDY-1)SHR 1;
777           L[II].OBJ[I].X2:=MIN(L[II].FIELDY-3,L[II].FIELDX-3)SHR 2;
778           L[II].OBJ[I].Y2:=0;
779         END {CIRCLE} ELSE IF L[II].OBJ[I].TYP>0THEN BEGIN
780           IF(L[II].OBJ[I].TYP=1)OR(L[II].OBJ[I].TYP=3)THEN I4:=2 ELSE I4:=3;
781           L[II].OBJ[I].X1:=(L[II].FIELDX-1)SHR 1-(L[II].FIELDX-1)SHR 2;
782           L[II].OBJ[I].Y1:=(L[II].FIELDY-1)SHR 1-(L[II].FIELDY-1)SHR 2;
783           L[II].OBJ[I].X2:=(L[II].FIELDX-1)SHR 1+(L[II].FIELDX-1)SHR 2;
784           L[II].OBJ[I].Y2:=(L[II].FIELDY-1)SHR 1+(L[II].FIELDY-1)SHR 2;
785           IF L[II].OBJ[I].TYP=3THEN
786             L[II].OBJ[I].Y2:=L[II].OBJ[I].Y2-L[II].OBJ[I].Y1; {BOX}
787         END; {NORMAL OBJECT}
788       END; {TYPE}
789       2:BEGIN
790         S:='SCREEN=(2,2)-('+NR(L[II].FIELDX-3)+','+NR(L[II].FIELDY-3)+
791           '); CENTER=('+NR((L[II].FIELDX-1)SHR 1)+','+NR((L[II].FIELDY-1)SHR 1)+')';
792         IF L[II].OBJ[I].TYP=4THEN BEGIN
793           L[II].OBJ[I].X1:=INPUTINT('ENTER X',S+
794             '   BE CAREFUL DRAWING OFF-SCREEN!',0,L[II].OBJ[I].X1,255);
795           L[II].OBJ[I].Y1:=INPUTINT('ENTER Y',S+
796             '   DON''T TRY Y<0; MIGHT CRASH CALC!!!',0,L[II].OBJ[I].Y1,255);
797         END {CIRCLE} ELSE IF L[II].OBJ[I].TYP>0THEN BEGIN
798           IF L[II].OBJ[I].TYP=3THEN
799             L[II].OBJ[I].Y2:=L[II].OBJ[I].Y2+L[II].OBJ[I].Y1-3;
800           IF(L[II].OBJ[I].TYP=1)OR(L[II].OBJ[I].TYP=3)THEN I4:=2 ELSE I4:=3;
801           L[II].OBJ[I].X1:=INPUTINT('ENTER BEGIN X',S,2,L[II].OBJ[I].X1,L[II].FIELDX-I4);
802           L[II].OBJ[I].Y1:=INPUTINT('ENTER BEGIN Y',S,2,L[II].OBJ[I].Y1,L[II].FIELDY-I4);
803           IF L[II].OBJ[I].TYP=3THEN
804             L[II].OBJ[I].Y2:=L[II].OBJ[I].Y2-L[II].OBJ[I].Y1+3;
805         END; {NORMAL OBJECT}
806       END; {BEGIN}
807       3:MAPOBJ(I,0);
808       4:BEGIN
809         IF L[II].OBJ[I].TYP=4 THEN BEGIN
810           L[II].OBJ[I].X2:=INPUTINT('ENTER SIZE','CROSSING BOUNDARIES NOT CHECKED',
811            3,L[II].OBJ[I].X2,255{L[II].FIELDX-3)SHR 1});
812           L[II].OBJ[I].Y2:=0;
813         END {CIRCLE} ELSE IF L[II].OBJ[I].TYP>0THEN BEGIN
814           S:='SCREEN=(0,0)-('+NR(L[II].FIELDX-2)+','+NR(L[II].FIELDY-2)+
815             '); CENTER=('+NR((L[II].FIELDX-1)SHR 1)+','+NR((L[II].FIELDY-1)SHR 1)+')';
816           IF(L[II].OBJ[I].TYP=1)OR(L[II].OBJ[I].TYP=3)THEN I4:=2 ELSE I4:=3;
817           L[II].OBJ[I].X2:=INPUTINT('ENTER END X',S,2,L[II].OBJ[I].X2,L[II].FIELDX-I4);
818           L[II].OBJ[I].Y2:=INPUTINT('ENTER END Y',S,2,L[II].OBJ[I].Y2,L[II].FIELDY-I4);
819           IF L[II].OBJ[I].TYP=3THEN BEGIN
820             IF L[II].OBJ[I].Y2<L[II].OBJ[I].Y1 THEN BEGIN
821               I4:=L[II].OBJ[I].Y2; L[II].OBJ[I].Y2:=L[II].OBJ[I].Y1;
822               L[II].OBJ[I].Y1:=I4;
823             END; {END BEFORE START}
824             L[II].OBJ[I].Y2:=L[II].OBJ[I].Y2-L[II].OBJ[I].Y1;
825           END; {BOX}
826         END; {NORMAL OBJECT}
827       END; {END}
828       5:MAPOBJ(I,1);
829       6:DISP1MAP(I);
830       7:BEGIN
831         L[II].OBJ[I].TYP:=-L[II].OBJ[I].TYP;
832       END; {DISABLE/ENABLE}
833     END; {HANDLE}
834   UNTIL III=-1; III:=0;
835 END; {EDITOBJ}
836
837 PROCEDURE EDITOBJS;
838 LABEL EDITOBJS1,EDITOBJS2;
839 BEGIN
840   REPEAT
841     TITLEBAR(2,'EDIT LEVEL OBJECTS');
842     IF L[II].NROBJ>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT);
843     WRITE('1'); TEXTCOLOR(C_TEXT); WRITELN(': EDIT OBJECT');
844     IF L[II].NROBJ>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT);
845     WRITE('2'); TEXTCOLOR(C_TEXT); WRITELN(': DELETE OBJECT');
846     TEXTCOLOR(C_MENU); WRITE('3'); TEXTCOLOR(C_TEXT); WRITELN(': CREATE OBJECT');
847     TEXTCOLOR(C_MENU); WRITE('4'); TEXTCOLOR(C_TEXT); WRITELN(': PLACE OBJECT');
848     TEXTCOLOR(C_MENU); WRITE('5'); TEXTCOLOR(C_TEXT); WRITE(': VIEW OBJECT');
849
850     III:=EDITMENU(5,6);
851     CASE III OF
852       1:IF L[II].NROBJ>0THEN BEGIN
853         I:=INPUTINT('EDIT WHICH OBJECT','',1,1,L[II].NROBJ);
854         EDITOBJ;
855       END; {EDIT}
856       2:IF L[II].NROBJ>0THEN BEGIN
857         I:=INPUTINT('DELETE WHICH OBJECT','0=CANCEL',0,0,L[II].NROBJ);
858         IF I>0THEN BEGIN
859           I4:=INPUTINT('PLEASE CONFIRM OBJECT TO *REMOVE*','ENTER 0 TO CANCEL',0,0,L[II].NROBJ);
860           IF I=I4 THEN BEGIN
861             DEC(L[II].NROBJ); FOR I:=I4 TO L[II].NROBJ DO L[II].OBJ[I]:=L[II].OBJ[I+1];
862           END; {CONFIRMED}
863         END; {}
864       END; {DELETE}
865       3:BEGIN
866         INC(L[II].NROBJ); I:=L[II].NROBJ; L[II].OBJ[I].TYP:=0; EDITOBJ;
867       END; {CREATE}
868       4:BEGIN
869         INC(L[II].NROBJ); I:=L[II].NROBJ; L[II].OBJ[I].TYP:=0;
870         S:=''; FOR I4:=0TO 4DO S:=S+NR(I4)+': '+OBJTYPE[I4]+CC(' ',37-ORD(OBJTYPE[I4,0]));
871         L[II].OBJ[I].TYP:=INPUTINT('ENTER OBJECT TYPE',S,0,0,4);
872         L[II].OBJ[I].X1:=0; L[II].OBJ[I].Y1:=0; L[II].OBJ[I].X2:=0;
873         L[II].OBJ[I].Y2:=0; MAPOBJ(I,0); MAPOBJ(I,1);
874       END; {CREATE}
875       5:BEGIN
876         I4:=INPUTINT('ENTER OBJECT TO HIGHLIGHT','0 FOR NONE',0,0,L[II].NROBJ);
877         DISP1MAP(I4);
878       END; {DISP}
879     END; {HANDLE}
880   UNTIL III=-1; III:=0;
881 END; {EDITOBJS}
882
883 FUNCTION BYTE2STR(SPRITE,SPRSIZE:BYTE):STRING;
884 VAR SPRTEMP:STRING[8];
885     SPRTEMP2:BYTE;
886 BEGIN
887   SPRTEMP:='';
888   FOR SPRTEMP2:=1TO 8-SPRSIZE DO SPRITE:=SPRITE SHR 1;
889   FOR SPRTEMP2:=9-SPRSIZE TO 8DO BEGIN
890     IF SPRITE MOD 2=1THEN SPRTEMP:='X'+SPRTEMP ELSE SPRTEMP:='�'+SPRTEMP;
891     SPRITE:=SPRITE SHR 1;
892   END;
893   BYTE2STR:=SPRTEMP;
894 END; {BYTE2STR}
895
896 FUNCTION BYTE2BIN(BYI1:BYTE):LONGINT;
897 VAR BYI2:BYTE; BYI3:LONGINT;
898 BEGIN
899   BYI3:=0;
900   FOR BYI2:=0TO 7DO BEGIN
901     IF BYI1 AND 1=1THEN BYI3:=BYI3+POW(10,BYI2);
902     BYI1:=BYI1 SHR 1;
903   END;
904   BYTE2BIN:=BYI3;
905 END; {BYTE2BIN}
906
907 FUNCTION STR2BYTE(SPRITE:STRING):BYTE;
908 VAR SPRTEMP,SPRTEMP2:BYTE;
909 BEGIN
910   SPRTEMP:=0; SPRTEMP2:=1;
911   FOR SPRTEMP2:=1TO LENGTH(SPRITE)DO
912     IF SPRITE[SPRTEMP2]='1'THEN INC(SPRTEMP,1SHL(8-SPRTEMP2));
913   STR2BYTE:=SPRTEMP;
914 END; {STR2BYTE}
915
916 PROCEDURE EDITSPR;
917 BEGIN
918   REPEAT
919     TITLEBAR(0,'EDIT SPRITE');
920     TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITE(': SPRITE SIZE ');
921     TEXTCOLOR(C_VALUE); IF L[II].SPRSIZE=0THEN WRITELN('(',B.SPRSIZE,')')
922       ELSE WRITELN('(',L[II].SPRSIZE,')');
923     IF L[II].SPRSIZE=0THEN BEGIN
924       TEXTCOLOR(C_TEXT); WRITE('2: EDIT SPRITE'); TEXTCOLOR(C_VALUE); WRITELN(' (DEFAULT)');
925     END {DEF SPR} ELSE FOR I:=1TO L[II].SPRSIZE DO BEGIN
926       TEXTCOLOR(C_MENU); WRITE(I+1); TEXTCOLOR(C_TEXT); WRITE(': EDIT '); TEXTCOLOR(C_VALUE);
927       WRITELN(BYTE2STR(L[II].SPRITE[I],L[II].SPRSIZE));
928     END; {DISP SPRITE}
929
930     III:=EDITMENU(1+L[II].SPRSIZE,0);
931     IF III=1THEN BEGIN
932       I:=INPUTINT('ENTER NEW SPRITE SIZE','MAXIMUM HEIGHT AND WIDTH; 0=DEFAULT',0,B.SPRSIZE,8);
933       IF I>L[II].SPRSIZE THEN FOR I4:=L[II].SPRSIZE+1TO I DO L[II].SPRITE[I4]:=0;
934       L[II].SPRSIZE:=I;
935     END {SIZE} ELSE IF III>1THEN BEGIN
936       S:=INPUTSTR('ENTER NEW LINE #'+NR(III-1)+' OF SPRITE','EXAMPLE: 100101 = X��X�X',
937         ZERO(BYTE2BIN(L[II].SPRITE[III-1]),8),8);
938       L[II].SPRITE[III-1]:=STR2BYTE(S);
939     END; {APP}
940   UNTIL III=-1; III:=0;
941 END; {EDITSPR}
942
943 PROCEDURE EDITBOUNCIES;
944 BEGIN
945   REPEAT
946     TITLEBAR(0,'EDIT BALLS IN LEVEL #'+NR(II));
947     TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITELN(': CREATE BALL');
948     TEXTCOLOR(C_MENU); WRITE('2'); TEXTCOLOR(C_TEXT); WRITELN(': DELETE BALL');
949     FOR III:=1TO L[II].NRBALLS DO BEGIN
950       TEXTCOLOR(C_MENU); WRITE(III+2); TEXTCOLOR(C_TEXT); WRITE(': BALL #',III,' (');
951       TEXTCOLOR(C_VALUE); WRITE(L[II].BALLS[III].X); TEXTCOLOR(C_TEXT); WRITE(',');
952       TEXTCOLOR(C_VALUE); WRITE(L[II].BALLS[III].Y); TEXTCOLOR(C_TEXT); WRITE(') DIR ');
953       TEXTCOLOR(C_VALUE); WRITELN(L[II].BALLS[III].D);
954     END; {WORMS}
955
956     III:=EDITMENU(2+L[II].NRBALLS,0);
957     IF III=1THEN BEGIN
958         INC(L[II].NRBALLS);
959         L[II].BALLS[L[II].NRBALLS].X:=INPUTINT('ENTER X','WHERE THE BALL STARTS BOUNCING',0,63,255);
960         L[II].BALLS[L[II].NRBALLS].Y:=INPUTINT('ENTER Y','',0,27,255);
961         L[II].BALLS[L[II].NRBALLS].D:=INPUTINT('ENTER DIRECTION','0/1:DOWN; 2/3:UP; 1/3:RIGHT; 0/2:LEFT',0,0,3);
962     END {CREATE} ELSE IF III=2THEN BEGIN
963       III:=INPUTINT('ENTER THE BOUNCY BALL TO REMOVE','(0 CANCELS)',0,0,L[II].NRBALLS);
964       IF III>0THEN BEGIN
965         DEC(L[II].NRBALLS); FOR I4:=III TO L[II].NRBALLS DO L[II].BALLS[I4]:=L[II].BALLS[I4+1];
966       END; {OK2REMOVE}
967     END {DELETE} ELSE IF III>0THEN BEGIN
968         L[II].BALLS[III-2].X:=INPUTINT('ENTER X','NOTE THE BOUNCY BALLS WILL PROBABLY     BOLDLY GO SOME OTHER PLACE ANYWAY',
969           0,L[II].BALLS[III-2].X,255);
970         L[II].BALLS[III-2].Y:=INPUTINT('ENTER Y','',0,L[II].BALLS[III-2].Y,255);
971         L[II].BALLS[III-2].D:=INPUTINT('ENTER DIRECTION','0/1:DOWN; 2/3:UP; 1/3:RIGHT; 0/2:LEFT',0,L[II].BALLS[III-2].D,3);
972     END; {EDIT}
973   UNTIL III=-1; III:=0;
974 END; {EDITWORMS}
975
976 PROCEDURE EDITWORMS;
977 BEGIN
978   REPEAT
979     TITLEBAR(0,'EDIT WORMS IN LEVEL #'+NR(II));
980     IF II>B.LSINGLE THEN III:=4 ELSE III:=1;
981     FOR III:=1TO III DO BEGIN
982       TEXTCOLOR(C_MENU); WRITE(III SHL 1-1); TEXTCOLOR(C_TEXT);
983       WRITE(': WORM #',III,' DIRECTION '); TEXTCOLOR(C_VALUE);
984       WRITELN('(',L[II].W[III].D,')');
985       TEXTCOLOR(C_MENU); WRITE(III SHL 1); TEXTCOLOR(C_TEXT);
986       WRITE(': WORM #',III,' POSITION '); TEXTCOLOR(C_VALUE);
987       WRITELN('(',L[II].W[III].X,',',L[II].W[III].Y,')');
988     END; {WORMS}
989
990     IF III=4THEN III:=EDITMENU(8,4)ELSE III:=EDITMENU(2,4);
991     CASE(III+1)OF
992       2,4,6,8:BEGIN
993         I:=INPUTINT('ENTER WORM #'+NR(III SHR 1+1)+' STARTING DIRECTION',
994           '0=DOWN; 64=RIGHT; 128=UP; 192=LEFT',0,L[II].W[III SHR 1+1].D,255);
995         L[II].W[III SHR 1+1].D:=I;
996       END; {DIR}
997       3,5,7,9:BEGIN
998         I:=INPUTINT('ENTER WORM #'+NR(III SHR 1)+' BEGIN X-POSITION',
999           'SCREEN = 2-'+NR(L[II].FIELDX-3)+'; CENTER = '+NR((L[II].FIELDX-1)SHR 1),
1000           2,L[II].W[III SHR 1].X,L[II].FIELDX-3);
1001         L[II].W[III SHR 1].X:=I;
1002         I:=INPUTINT('ENTER WORM #'+NR(III SHR 1)+' BEGIN Y-POSITION',
1003           'SCREEN = 2-'+NR(L[II].FIELDY-3)+'; CENTER = '+NR((L[II].FIELDY-1)SHR 1),
1004           2,L[II].W[III SHR 1].Y,L[II].FIELDY-3);
1005         L[II].W[III SHR 1].Y:=I;
1006       END; {POS}
1007     END; {HANDLE}
1008   UNTIL III=-1; III:=0;
1009 END; {EDITWORMS}
1010
1011 PROCEDURE EDITLEVSETS;
1012 BEGIN
1013   REPEAT
1014     TITLEBAR(3,'LEVEL #'+NR(II)+' SETTINGS');
1015     IF II>B.LSINGLE+B.LMULTI+B.LRACE THEN BEGIN
1016       TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITE(': FLAG POSITIONS ');
1017        TEXTCOLOR(C_VALUE); WRITELN('(',L[II].FLAG1X,',',L[II].FLAG1Y,')(',
1018         L[II].FLAG2X,',',L[II].FLAG2Y,')');
1019     END {CTF} ELSE IF II>B.LSINGLE+B.LMULTI THEN BEGIN
1020       TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITE(': LAP BORDERS ');
1021        TEXTCOLOR(C_VALUE); WRITELN('(',L[II].FLAG1X,'; ',L[II].FLAG1Y,')');
1022     END {RACE} ELSE BEGIN
1023       IF II>B.LSINGLE THEN TEXTCOLOR(C_TEXT)ELSE TEXTCOLOR(C_MENU);
1024       WRITE('1'); TEXTCOLOR(C_TEXT); WRITE(': NUMBER OF PEAS ');
1025        TEXTCOLOR(C_VALUE); WRITELN('(',L[II].PEAS,')');
1026     END; {SP/MP}
1027     TEXTCOLOR(C_MENU); WRITE('2'); TEXTCOLOR(C_TEXT); WRITE(': GAME DELAY ');
1028       TEXTCOLOR(C_VALUE); WRITELN('(',L[II].DELAY,')');
1029     TEXTCOLOR(C_MENU); WRITE('3'); TEXTCOLOR(C_TEXT); WRITE(': STARTING LENGTH ');
1030       TEXTCOLOR(C_VALUE); WRITELN('(',L[II].BSIZE,')');
1031     TEXTCOLOR(C_MENU); WRITE('4'); TEXTCOLOR(C_TEXT); WRITE(': GROWTH ');
1032       TEXTCOLOR(C_VALUE); WRITE('(',L[II].GROWTH,')');
1033
1034     III:=EDITMENU(4,0);
1035     CASE III OF
1036       1:IF II<=B.LSINGLE THEN
1037         L[II].PEAS:=INPUTINT('ENTER TOTAL NUMBER OF PEAS','TOO MANY PEAS WILL MAKE A LEVEL BORING',1,L[II].PEAS,255)
1038       ELSE IF II>B.LSINGLE+B.LMULTI+B.LRACE THEN BEGIN
1039         IF L[II].SPRSIZE>0THEN I:=L[II].SPRSIZE ELSE I:=B.SPRSIZE;
1040         L[II].FLAG1X:=INPUTINT('ENTER X-COORDINATE OF FLAG #1','FLAG OF PLAYERS 1 AND 3',2,L[II].FLAG1X,L[II].FIELDX-2-I);
1041         L[II].FLAG1Y:=INPUTINT('ENTER Y-COORDINATE OF FLAG #1','FLAG OF PLAYERS 1 AND 3',2,L[II].FLAG1Y,L[II].FIELDY-2-I);
1042         L[II].FLAG2X:=INPUTINT('ENTER X-COORDINATE OF FLAG #2','FLAG OF PLAYERS 2 AND 4',2,L[II].FLAG2X,L[II].FIELDX-2-I);
1043         L[II].FLAG2Y:=INPUTINT('ENTER Y-COORDINATE OF FLAG #2','FLAG OF PLAYERS 2 AND 4',2,L[II].FLAG2Y,L[II].FIELDY-2-I);
1044       END {CTF} ELSE IF II>B.LSINGLE+B.LMULTI THEN BEGIN
1045         L[II].FLAG1X:=INPUTINT('ENTER HORIZONTAL LAP BORDER','USUALLY THE MIDDLE OF THE SCREEN',2,L[II].FIELDY SHR 1,L[II].FIELDY-2);
1046         L[II].FLAG1Y:=INPUTINT('ENTER VERTICAL LAP BORDER','USUALLY THE CENTER OF THE SCREEN',2,L[II].FIELDX SHR 1,L[II].FIELDX-2);
1047       END; {RACE}
1048       2:L[II].DELAY:=INPUTINT('ENTER NEW DELAY','0=FASTEST; 4=AVERAGE; 10=SLOW',0,L[II].DELAY,255);
1049       3:L[II].BSIZE:=INPUTINT('ENTER WORM BEGIN SIZE','15=NORMAL; TOO SHORT WORMS WILL BE HARD TO PLAY WITH',
1050           1,L[II].BSIZE,255);
1051       4:L[II].GROWTH:=INPUTINT('ENTER GROWTH PER PEA','0=NO GROWTH; 15=NORMAL',0,L[II].GROWTH,255);
1052     END; {HANDLE}
1053   UNTIL III=-1; III:=0;
1054 END; {EDITLEVSETS}
1055
1056 PROCEDURE EDITLEVEL;
1057 BEGIN
1058   REPEAT
1059     L[II].SIZE:=12+L[II].NRBALLS*3+L[II].SPRSIZE+L[II].NROBJ*5;
1060     IF II<=B.LSINGLE THEN L[II].NAME:=ZERO(II,2);
1061     IF II>B.LSINGLE THEN INC(L[II].SIZE,10+LENGTH(L[II].NAME));
1062     IF II>B.LSINGLE+B.LMULTI THEN INC(L[II].SIZE,2);
1063     IF II>B.LSINGLE+B.LMULTI+B.LRACE THEN INC(L[II].SIZE,2);
1064     TITLEBAR(3,'EDIT LEVEL #'+NR(II));
1065     TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITE(': DESCRIPTION ');
1066       TEXTCOLOR(C_VALUE); WRITELN('(',L[II].ID,')');
1067     TEXTCOLOR(C_MENU); WRITE('2'); TEXTCOLOR(C_TEXT); WRITE(': NAME ');
1068       TEXTCOLOR(C_VALUE); WRITELN('(',L[II].NAME,')');
1069     TEXTCOLOR(C_MENU); WRITE('3'); TEXTCOLOR(C_TEXT); WRITELN(': LEVEL SETTINGS');
1070     TEXTCOLOR(C_MENU); WRITE('4'); TEXTCOLOR(C_TEXT); WRITE(': SPRITE ');
1071       TEXTCOLOR(C_VALUE); IF L[II].SPRSIZE=0THEN WRITELN('(DEFAULT)')
1072                     ELSE WRITELN('(',L[II].SPRSIZE,')');
1073     TEXTCOLOR(C_MENU); WRITE('5'); TEXTCOLOR(C_TEXT); WRITE(': BOUNCY BALLS ');
1074       TEXTCOLOR(C_VALUE); WRITELN('(',L[II].NRBALLS,')');
1075     TEXTCOLOR(C_MENU); WRITE('6'); TEXTCOLOR(C_TEXT); WRITE(': WORMS '); TEXTCOLOR(C_VALUE);
1076       IF II>B.LSINGLE THEN BEGIN
1077         WRITELN('(',L[II].W[1].D,';',L[II].W[2].D,';',L[II].W[3].D,';',L[II].W[4].D,')');
1078       END {FOUR WORMS} ELSE
1079         WRITELN('(',L[II].W[1].X,',',L[II].W[1].Y,':',L[II].W[1].D,')');
1080     TEXTCOLOR(C_MENU); WRITE('7'); TEXTCOLOR(C_TEXT); WRITE(': FIELD SIZE ');
1081       TEXTCOLOR(C_VALUE); WRITELN('(',L[II].FIELDX,',',L[II].FIELDY,')');
1082     TEXTCOLOR(C_MENU); WRITE('8'); TEXTCOLOR(C_TEXT); WRITE(': OBJECTS ');
1083       TEXTCOLOR(C_VALUE); WRITELN('(',L[II].NROBJ,')');
1084     TEXTCOLOR(C_MENU); WRITE('9'); TEXTCOLOR(C_TEXT); WRITE(': PREVIEW LEVEL');
1085
1086     III:=EDITMENU(9,0);
1087     CASE III OF
1088       1:L[II].ID:=INPUTSTR('ENTER DESCRIPTION FOR THIS LEVEL',
1089         'THIS WON''T BE DISPLAYED DURING THE GAME,IT''S JUST TO MAKE EDITING EASIER',L[II].ID,22); {DESCR}
1090       2:L[II].NAME:=INPUTSTR('ENTER DESCRIPTION FOR THIS LEVEL',
1091         'MULTIPLAYER LEVELS CAN BE NAMED TO MAKE   LIFE EASIER FOR THE PLAYER',L[II].NAME,22); {NAME}
1092       3:EDITLEVSETS;
1093       4:EDITSPR;
1094       5:EDITBOUNCIES;
1095       6:EDITWORMS;
1096       7:BEGIN
1097         FOR III:=L[II].NROBJ DOWNTO 1DO
1098          IF L[II].OBJ[III].Y2+L[II].OBJ[III].Y1=255THEN BEGIN
1099           DEC(L[II].NROBJ); FOR I4:=III TO L[II].NROBJ DO L[II].OBJ[I4]:=L[II].OBJ[I4+1];
1100         END; {REMOVE OLD BARS}
1101         L[II].FIELDX:=INPUTINT('ENTER FIELD WIDTH','128=ONE SCREEN; MORE WILL SCROLL THE    SCREEN',5,L[II].FIELDX,255);
1102         IF L[II].FIELDX<128THEN BEGIN
1103           INC(L[II].NROBJ); L[II].OBJ[L[II].NROBJ].X1:=L[II].FIELDX;
1104           IF L[II].FIELDX<128THEN DEC(L[II].OBJ[L[II].NROBJ].X1,2);
1105           L[II].OBJ[L[II].NROBJ].Y2:=255; L[II].OBJ[L[II].NROBJ].TYP:=3;
1106           L[II].OBJ[L[II].NROBJ].X2:=127; L[II].OBJ[L[II].NROBJ].Y1:=0;
1107         END; {PUT BAR}
1108         L[II].FIELDY:=INPUTINT('ENTER FIELD HEIGHT','57=ONE SCREEN; MORE WILL SCROLL THE     SCREEN',5,L[II].FIELDY,255);
1109         IF L[II].FIELDY<57THEN BEGIN
1110           INC(L[II].NROBJ); L[II].OBJ[L[II].NROBJ].X1:=2;
1111           L[II].OBJ[L[II].NROBJ].Y2:=255-L[II].FIELDY;
1112           L[II].OBJ[L[II].NROBJ].X2:=L[II].FIELDX-1; L[II].OBJ[L[II].NROBJ].TYP:=3;
1113           L[II].OBJ[L[II].NROBJ].Y1:=L[II].FIELDY;
1114         END; {PUT BAR}
1115       END; {FIELD}
1116       8:EDITOBJS;
1117       9:DISP1MAP(0);
1118     END; {HANDLE}
1119   UNTIL III=-1; III:=0;
1120 END; {EDITLEVEL}
1121
1122 PROCEDURE EDITLVL;
1123 LABEL EDITHLVL1,EDITHLVL2;
1124 BEGIN
1125   REPEAT
1126     TITLEBAR(1,'EDIT LEVELS');
1127     IF B.LTOTAL>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT);
1128     WRITE('1'); TEXTCOLOR(C_TEXT); WRITELN(': EDIT LEVEL');
1129     IF B.LTOTAL>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT);
1130     WRITE('2'); TEXTCOLOR(C_TEXT); WRITELN(': COPY LEVEL');
1131     IF B.LTOTAL>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT);
1132     WRITE('3'); TEXTCOLOR(C_TEXT); WRITELN(': MOVE LEVEL');
1133     IF B.LTOTAL>0THEN TEXTCOLOR(C_MENU)ELSE TEXTCOLOR(C_TEXT);
1134     WRITE('4'); TEXTCOLOR(C_TEXT); WRITELN(': DELETE LEVEL');
1135     TEXTCOLOR(C_MENU); WRITE('5'); TEXTCOLOR(C_TEXT); WRITELN(': CREATE SINGLEPLAYER LEVEL');
1136     TEXTCOLOR(C_MENU); WRITE('6'); TEXTCOLOR(C_TEXT); WRITELN(': CREATE MULTIPLAYER LEVEL');
1137     TEXTCOLOR(C_MENU); WRITE('7'); TEXTCOLOR(C_TEXT); WRITELN(': CREATE RACE LEVEL');
1138     TEXTCOLOR(C_MENU); WRITE('8'); TEXTCOLOR(C_TEXT); WRITELN(': CREATE CTF LEVEL');
1139     TEXTCOLOR(C_MENU); WRITE('9'); TEXTCOLOR(C_TEXT); WRITELN(': PREVIEW LEVELS');
1140     TEXTCOLOR(C_TEXT); WRITELN; III:=EDITMENU(9,5);
1141
1142     CASE III OF
1143       1:IF B.LTOTAL>0THEN BEGIN
1144         II:=INPUTINT('EDIT WHICH LEVEL','0=CANCEL',0,0,B.LTOTAL);
1145         IF II<>0THEN EDITLEVEL;
1146       END; {EDIT}
1147       2:IF B.LTOTAL>0THEN BEGIN
1148         II:=INPUTINT('COPY WHICH LEVEL','0=CANCEL',0,0,B.LTOTAL);
1149         IF II<>0THEN BEGIN
1150           INC(B.LTOTAL);
1151           IF II<=B.LSINGLE THEN BEGIN
1152             INC(B.LSINGLE);
1153             FOR I:=B.LTOTAL DOWNTO B.LSINGLE+1DO L[I]:=L[I-1];
1154             L[B.LSINGLE]:=L[II]; II:=B.LSINGLE
1155           END {SP} ELSE IF II<=B.LSINGLE+B.LMULTI THEN BEGIN
1156             FOR I:=2TO 6DO IF B.LTO[I]=B.LMULTI THEN BEGIN
1157               INC(B.LTO[I]); IF B.LFROM[I]=0THEN INC(B.LFROM[I]);
1158             END; {MP}
1159             INC(B.LMULTI);
1160             FOR I:=B.LTOTAL DOWNTO B.LSINGLE+B.LMULTI+1DO L[I]:=L[I-1];
1161             L[B.LSINGLE+B.LMULTI]:=L[II]; II:=B.LSINGLE+B.LMULTI;
1162           END {MP} ELSE IF II<=B.LSINGLE+B.LMULTI+B.LRACE THEN BEGIN
1163             INC(B.LRACE); L[B.LTOTAL]:=L[II]; II:=B.LTOTAL;
1164           END {RACE} ELSE BEGIN
1165             INC(B.LCTF); L[B.LTOTAL]:=L[II]; II:=B.LTOTAL;
1166           END; {CTF}
1167           L[II].ID:='Copy of '+L[II].ID;
1168         END; {>0}
1169       END; {COPY}
1170       3:IF B.LTOTAL>0THEN BEGIN
1171         I:=INPUTINT('MOVE WHICH LEVEL','0=CANCEL',0,0,B.LTOTAL);
1172         IF II<>0THEN BEGIN
1173           II:=INPUTINT('NEW LEVEL NUMBER','ENTER 0 TO CANCEL',0,0,B.LTOTAL);
1174           IF II<>0THEN IF((I<=B.LSINGLE)AND(II<=B.LSINGLE))
1175            OR((I>B.LSINGLE)AND(I<=B.LMULTI+B.LSINGLE)AND(II>B.LSINGLE)AND(II<=B.LMULTI+B.LSINGLE))
1176             OR((I>B.LSINGLE+B.LMULTI)AND(I<=B.LMULTI+B.LSINGLE+B.LRACE)AND(II>B.LSINGLE+B.LMULTI)AND(II<=B.LMULTI+B.LSINGLE+B.LRACE))
1177              OR((I>B.LSINGLE+B.LMULTI+B.LRACE)AND(II>B.LSINGLE+B.LMULTI+B.LRACE))THEN BEGIN
1178               LTEMP:=L[I]; IF II>I THEN FOR I4:=I TO II-1DO L[I4]:=L[I4+1]
1179                 ELSE FOR I4:=I DOWNTO II+1DO L[I4]:=L[I4-1];
1180               L[II]:=LTEMP;
1181           END; {CONFIRMED}
1182         END; {}
1183       END; {MOVE}
1184       4:IF B.LTOTAL>0THEN BEGIN
1185         II:=INPUTINT('DELETE WHICH LEVEL','0=CANCEL',0,0,B.LTOTAL);
1186         IF II<>0THEN BEGIN
1187           I:=INPUTINT('PLEASE CONFIRM LEVEL TO *REMOVE*','ENTER 0 TO CANCEL',0,0,B.LTOTAL);
1188           IF I=II THEN BEGIN
1189             IF I<=B.LSINGLE THEN DEC(B.LSINGLE)ELSE
1190               IF I<=B.LMULTI+B.LSINGLE THEN DEC(B.LMULTI)ELSE
1191                 IF I<=B.LMULTI+B.LSINGLE+B.LRACE THEN DEC(B.LRACE)ELSE
1192                   DEC(B.LCTF);
1193             DEC(B.LTOTAL); FOR I:=II TO B.LTOTAL DO L[I]:=L[I+1];
1194             FOR I:=2TO 6DO BEGIN
1195               IF B.LTO[I]>B.LMULTI THEN B.LTO[I]:=B.LMULTI;
1196               IF B.LFROM[I]>B.LTO[I]THEN B.LFROM[I]:=B.LTO[I];
1197             END; {CHECK}
1198             IF B.LTO[7]>B.LMULTI+B.LRACE THEN B.LTO[7]:=B.LMULTI+B.LRACE;
1199             IF B.LFROM[7]>B.LTO[7]THEN B.LFROM[7]:=B.LTO[7];
1200             IF B.LTO[8]>B.LTOTAL THEN B.LTO[8]:=B.LTOTAL;
1201             IF B.LFROM[8]>B.LTO[8]THEN B.LFROM[8]:=B.LTO[8];
1202           END; {CONFIRMED}
1203         END; {}
1204       END; {DELETE}
1205       5:BEGIN
1206         INC(B.LSINGLE); INC(B.LTOTAL); II:=B.LSINGLE;
1207         IF B.LTO[1]=0THEN B.LTO[1]:=1; IF B.LFROM[1]=0THEN B.LFROM[1]:=1;
1208         FOR I:=B.LTOTAL DOWNTO II+1DO L[I]:=L[I-1];
1209         L[II].ID:='Level #'+NR(II);
1210         L[II].PEAS:=8; L[II].DELAY:=4; L[II].GROWTH:=15; L[II].BSIZE:=15;
1211         L[II].SPRSIZE:=0; L[II].NRBALLS:=0; L[II].W[1].D:=0; L[II].W[1].X:=63;
1212         L[II].W[1].Y:=2; L[II].FIELDX:=128; L[II].FIELDY:=57; L[II].NROBJ:=0;
1213         EDITLEVEL;
1214       END; {CREATE SP}
1215       6:BEGIN
1216         FOR I:=2TO 6DO IF B.LTO[I]=B.LMULTI THEN BEGIN
1217           INC(B.LTO[I]); IF B.LFROM[I]=0THEN INC(B.LFROM[I]);
1218         END; {}
1219         INC(B.LTOTAL); INC(B.LMULTI); II:=B.LSINGLE+B.LMULTI;
1220         FOR I:=B.LTOTAL DOWNTO II+1DO L[I]:=L[I-1];
1221         L[II].ID:='Arena #'+NR(B.LMULTI); L[II].NAME:=ZERO(B.LMULTI,2);
1222         L[II].DELAY:=4; L[II].GROWTH:=15; L[II].BSIZE:=15; L[II].SPRSIZE:=0;
1223         L[II].W[1].D:=$40; L[II].W[1].Y:=28; L[II].W[1].X:=2;
1224         L[II].W[2].D:=$C0; L[II].W[2].Y:=28; L[II].W[2].X:=125;
1225         L[II].W[3].D:=$00; L[II].W[3].Y:=02; L[II].W[3].X:=63;
1226         L[II].W[4].D:=$80; L[II].W[4].Y:=54; L[II].W[4].X:=63;
1227         L[II].NRBALLS:=0; L[II].FIELDX:=128; L[II].FIELDY:=57;
1228         L[II].NROBJ:=0; EDITLEVEL;
1229       END; {CREATE MP}
1230       7:BEGIN
1231         INC(B.LRACE); INC(B.LTOTAL); II:=B.LSINGLE+B.LMULTI+B.LRACE;
1232         FOR I:=B.LTOTAL DOWNTO II+1DO L[I]:=L[I-1];
1233         INC(B.LTO[7]); IF B.LFROM[7]=0THEN INC(B.LFROM[7]);
1234         L[II].ID:='Track #'+NR(II); L[II].NAME:=ZERO(B.LRACE,2);
1235         L[II].DELAY:=8; L[II].GROWTH:=15;
1236         L[II].BSIZE:=15; L[II].SPRSIZE:=0; L[II].NRBALLS:=0;
1237         L[II].W[1].D:=$40; L[II].W[1].X:=2; L[II].W[1].Y:=23;
1238         L[II].W[2].D:=$C0; L[II].W[2].X:=125; L[II].W[2].Y:=23;
1239         L[II].W[3].D:=$40; L[II].W[3].X:=2; L[II].W[3].Y:=35;
1240         L[II].W[4].D:=$C0; L[II].W[4].X:=125; L[II].W[4].Y:=35;
1241         L[II].FIELDX:=128; L[II].FIELDY:=57; L[II].NROBJ:=0;
1242         L[II].FLAG1X:=28; L[II].FLAG1Y:=63; L[II].FLAG2X:=0; L[II].FLAG2Y:=0;
1243         EDITLEVEL;
1244       END; {CREATE RACE}
1245       8:BEGIN
1246         INC(B.LCTF); INC(B.LTOTAL); II:=B.LTOTAL;
1247         INC(B.LTO[8]); IF B.LFROM[8]=0THEN INC(B.LFROM[8]);
1248         L[II].ID:='Field #'+NR(II); L[II].NAME:=ZERO(B.LCTF,2);
1249         L[II].DELAY:=5; L[II].GROWTH:=15;
1250         L[II].BSIZE:=15; L[II].SPRSIZE:=0; L[II].NRBALLS:=0;
1251         L[II].W[1].D:=$40; L[II].W[1].X:=2; L[II].W[1].Y:=23;
1252         L[II].W[2].D:=$C0; L[II].W[2].X:=125; L[II].W[2].Y:=23;
1253         L[II].W[3].D:=$40; L[II].W[3].X:=2; L[II].W[3].Y:=35;
1254         L[II].W[4].D:=$C0; L[II].W[4].X:=125; L[II].W[4].Y:=35;
1255         L[II].FIELDX:=128; L[II].FIELDY:=57; L[II].NROBJ:=0;
1256         L[II].FLAG1X:=8; L[II].FLAG2X:=128-8-B.SPRSIZE;
1257         L[II].FLAG1Y:=27; L[II].FLAG2Y:=27;
1258         EDITLEVEL;
1259       END; {CREATE CTF}
1260       9:BEGIN
1261         II:=INPUTINT('MAP SIZE','4=SHRINK 16 MAPS ON SCREEN',1,5,32);
1262         DISPMANYMAP(II);
1263       END; {PREVIEW}
1264     END; {HANDLE}
1265   UNTIL III=-1; III:=0;
1266 END; {EDITLVL}
1267
1268 PROCEDURE EDITSETS;
1269 BEGIN
1270   REPEAT
1271     TITLEBAR(0,'CHANGE DEFAULTS');
1272     TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITE(': HISCORE NAME ');
1273       TEXTCOLOR(C_VALUE); WRITELN('(',B.HINAME,')');
1274     TEXTCOLOR(C_MENU); WRITE('2'); TEXTCOLOR(C_TEXT); WRITE(': END TYPE ');
1275       TEXTCOLOR(C_VALUE); WRITELN('(',ENDTYPE[B.ENDTYPE],')');
1276     CASE B.ENDTYPE OF
1277       1:BEGIN
1278         TEXTCOLOR(C_MENU); WRITE('3'); TEXTCOLOR(C_TEXT); WRITE(': EDIT MESSAGE ');
1279         TEXTCOLOR(C_VALUE); WRITE('(',COPY(B.ENDSTR,1,18));
1280         IF LENGTH(B.ENDSTR)>18THEN WRITELN('...)')ELSE WRITELN(')');
1281       END; {MSG}
1282       ELSE BEGIN TEXTCOLOR(C_TEXT); WRITELN('3: CONFIGURE END'); END; {NOTN}
1283     END; {TYPE}
1284     TEXTCOLOR(C_MENU); WRITE('4'); TEXTCOLOR(C_TEXT); WRITE(': SPRITE SIZE ');
1285     TEXTCOLOR(C_VALUE); WRITELN('(',B.SPRSIZE,')');
1286     FOR I:=1TO B.SPRSIZE DO BEGIN
1287       TEXTCOLOR(C_MENU); IF I>5THEN WRITE(CHR(I+59))ELSE WRITE(I+4);
1288       TEXTCOLOR(C_TEXT); WRITE(': EDIT '); TEXTCOLOR(C_VALUE);
1289       WRITELN(BYTE2STR(B.SPRITE[I],B.SPRSIZE));
1290     END; {DISP SPRITE}
1291
1292     III:=EDITMENU(4+B.SPRSIZE,0);
1293     CASE III OF
1294       1:BEGIN
1295         S:=INPUTSTR('DEFAULT NAME OF HISCORES','LENGTH 3 CHARS',B.HINAME,3);
1296         FOR I:=1TO 3DO B.HINAME[I]:=S[I];
1297       END; {HINAME}
1298       2:BEGIN
1299         FOR II:=0TO 1DO WRITE(NR(II)+': '+ENDTYPE[II]+CC(' ',40-ORD(ENDTYPE[II,0])));
1300         B.ENDTYPE:=INPUTINT('ENTER NEW END TYPE',S,0,0,1);
1301         IF B.ENDTYPE=1THEN B.ENDSTR:='Congratulations!!';
1302       END; {ENDTYPE}
1303       3:BEGIN
1304         IF B.ENDTYPE=1THEN
1305           B.ENDSTR:=INPUTSTR('ENTER NEW END MESSAGE','MAX 255 CHARS','Congratulations!!',255);
1306       END; {ENDCONF}
1307       4:BEGIN
1308         I:=INPUTINT('ENTER THE DEFAULT SPRITE SIZE','CAN BE CHANGED PER LEVEL',1,B.SPRSIZE,8);
1309         IF I>B.SPRSIZE THEN FOR I4:=B.SPRSIZE+1TO I DO B.SPRITE[I4]:=0;
1310         B.SPRSIZE:=I;
1311       END; {SPRSIZE}
1312       ELSE IF III>4THEN BEGIN
1313         S:=INPUTSTR('ENTER NEW LINE #'+NR(III-4)+' OF SPRITE','EXAMPLE: 100101 = X��X�X',
1314           ZERO(BYTE2BIN(B.SPRITE[III-4]),8),8);
1315         B.SPRITE[III-4]:=STR2BYTE(S);
1316       END; {SPREDIT}
1317     END; {HANDLE}
1318   UNTIL III=-1; III:=0;
1319 END; {EDITSETS}
1320
1321 PROCEDURE EDITMAIN;
1322 BEGIN
1323   REPEAT
1324     TITLEBAR(4,UPCS(B.NAME));
1325     TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITELN(': EDIT TITLE');
1326     TEXTCOLOR(C_MENU); WRITE('2'); TEXTCOLOR(C_TEXT); WRITELN(': DESCRIPTION');
1327     TEXTCOLOR(C_MENU); WRITE('3'); TEXTCOLOR(C_TEXT); WRITELN(': DEFAULTS');
1328     TEXTCOLOR(C_MENU); WRITE('4'); TEXTCOLOR(C_TEXT); WRITELN(': GAME TYPES');
1329     TEXTCOLOR(C_MENU); WRITE('5'); TEXTCOLOR(C_TEXT); WRITE(': LEVELS');
1330       TEXTCOLOR(C_VALUE); WRITE(' (',B.LTOTAL,')');
1331
1332     III:=EDITMENU(5,3);
1333     CASE III OF
1334       1:B.NAME:=INPUTSTR('ENTER NEW TITLE FOR THIS LEVELFILE',
1335         'NOTE TI-86 DISPLAYS ONLY 22 CHARS ON THEFIRST LINE',B.NAME,32); {TITLE}
1336       2:B.DESC:=INPUTSTR('ENTER NEW DESCRIPTION FOR THIS LEVELFILE',
1337         'DISPLAYED AT THE LEVEL-SELECTION SCREEN.',B.DESC,64); {DESC}
1338       3:EDITSETS;
1339       4:EDITGAME;
1340       5:EDITLVL;
1341     END; {HANDLE}
1342   UNTIL III=-1; III:=0;
1343 END; {EDITMAIN}
1344
1345 PROCEDURE OPENFILE;
1346 BEGIN
1347   ASSIGN(F,FILENAME+'.LVL'); {$I-} RESET(F,1); {$I+} I:=IORESULT;
1348   IF I<>0THEN BEGIN
1349     TEXTCOLOR(C_HI); WRITE('WARNING!! '); TEXTCOLOR(C_TEXT);
1350     WRITELN('FAILED OPENING LEVEL FILE!'); WRITELN('RUNTIME ERROR #',I); HALT;
1351   END; {CAN'T OPEN}
1352   {$I-} BLOCKREAD(F,B,SIZEOF(B));
1353         FOR I:=1TO B.LTOTAL DO BLOCKREAD(F,L[I],SIZEOF(L[I])); {$I+}
1354   IF(IORESULT<>0)OR(B.HEAD<>HEADEDIT)THEN BEGIN
1355     TEXTCOLOR(C_HI); WRITE('WARNING!! '); TEXTCOLOR(C_TEXT);
1356     WRITELN('ERROR READING LEVEL FILE!');
1357     CLOSE(F); HALT;
1358   END; {ERROR}
1359   IF B.VERSION<>HEADVERSION THEN BEGIN
1360     WRITELN('CAN''T READ FILE VERSION ',B.VERSION,'.'); CLOSE(F); HALT;
1361   END; {VERSION}
1362   FSIZE:=FILESIZE(F); CLOSE(F);
1363 END; {OPENFILE}
1364
1365 PROCEDURE FILESAVE;
1366 BEGIN
1367   ASSIGN(F,FILENAME+'.LVL'); REWRITE(F,1);
1368   BLOCKWRITE(F,B,SIZEOF(B));
1369   FOR I:=1TO B.LTOTAL DO BLOCKWRITE(F,L[I],SIZEOF(L[I]));
1370   CLOSE(F);
1371 END; {FILESAVE}
1372
1373 PROCEDURE FILECOMP;
1374 VAR ISTACK:WORD;
1375 PROCEDURE SAVELEVEL(SAVEWORMS:BYTE);
1376 BEGIN
1377   IF II>B.LSINGLE THEN BEGIN
1378     BLOCKWRITE(FF,L[II].NAME[1],LENGTH(L[II].NAME)); I:=0; BLOCKWRITE(FF,I,1);
1379   END; {LEVEL_NAME}
1380   BLOCKWRITE(FF,L[II].PEAS,1);
1381   BLOCKWRITE(FF,L[II].DELAY,1);
1382   BLOCKWRITE(FF,L[II].GROWTH,1);
1383   BLOCKWRITE(FF,L[II].BSIZE,1);
1384   BLOCKWRITE(FF,L[II].SPRSIZE,1);
1385   BLOCKWRITE(FF,L[II].SPRITE,L[II].SPRSIZE);
1386   BLOCKWRITE(FF,L[II].NRBALLS,1);
1387   BLOCKWRITE(FF,L[II].BALLS,L[II].NRBALLS*3);
1388   BLOCKWRITE(FF,L[II].W[1].D,3*SAVEWORMS); {D,X,Y}
1389   BLOCKWRITE(FF,L[II].FIELDX,2); {FIELDX,FIELDY}
1390   IF II>B.LSINGLE+B.LMULTI THEN BLOCKWRITE(FF,L[II].FLAG1Y,2); {FLAG1Y,FLAG1X}
1391   IF II>B.LSINGLE+B.LMULTI+B.LRACE THEN BLOCKWRITE(FF,L[II].FLAG2Y,2); {FLAG2Y,FLAG2X}
1392   FOR I:=1TO L[II].NROBJ DO BEGIN
1393     IF L[II].OBJ[I].TYP>0THEN BEGIN
1394       BLOCKWRITE(FF,L[II].OBJ[I].TYP,4);
1395       IF(L[II].OBJ[I].TYP=3)AND(L[II].OBJ[I].Y2+L[II].OBJ[I].Y1=255)THEN BEGIN
1396         I4:=MAX(57,L[II].FIELDY)-L[II].OBJ[I].Y1; BLOCKWRITE(FF,I4,1);
1397       END {OFFSCREEN} ELSE BLOCKWRITE(FF,L[II].OBJ[I].Y2,1);
1398     END; {VALID}
1399   END; {OBJS}
1400   I:=0; BLOCKWRITE(FF,I,1); {LVL END}
1401 END; {COMMONSAVELEVEL}
1402 BEGIN
1403   ASSIGN(FF,FILENAME+'.86S'); REWRITE(FF,1);
1404
1405   BLOCKWRITE(FF,HEAD86S[1],ORD(HEAD86S[0])); {86S-HEADER}
1406   ISTACK:=COMPSIZE; BLOCKWRITE(FF,ISTACK,2); {FILE LENGTH (x+$12)}
1407   I:=$000C; BLOCKWRITE(FF,I,2); {STRING ID}
1408   DEC(ISTACK,$10); BLOCKWRITE(FF,ISTACK,2); {DATA LENGTH (x+2)}
1409
1410   I:=$080C; BLOCKWRITE(FF,I,2); {STRING ID}
1411   S:=LOCS(FILENAME); FOR I:=ORD(S[0])+1TO 8DO S[I]:=#0;
1412   BLOCKWRITE(FF,S[1],8); {PRGNAME}
1413   BLOCKWRITE(FF,ISTACK,2); {DATA LENGTH (x+2)}
1414   DEC(ISTACK,2); BLOCKWRITE(FF,ISTACK,2); {PROG LENGTH (x)}
1415
1416   C:='w'; BLOCKWRITE(FF,C,1); {WORM ID}
1417   I:=HEADVERSION; BLOCKWRITE(FF,I,1); {VERSION}
1418   I:=0; BLOCKWRITE(FF,B.NAME[1],ORD(B.NAME[0])); {TITLE}
1419   BLOCKWRITE(FF,I,1); {STRING TERMINATOR #0}
1420   BLOCKWRITE(FF,B.DESC[1],ORD(B.DESC[0])); {DESCRIPTION} BLOCKWRITE(FF,I,1);
1421   I:=1+B.SPRSIZE+2;
1422   FOR II:=B.LFROM[1]TO B.LSINGLE+B.LMULTI DO I:=I+L[II].SIZE;
1423   IF B.LFROM[7]>0THEN {RACE LEVELS}
1424     FOR II:=B.LSINGLE+B.LMULTI+B.LFROM[7]TO B.LSINGLE+B.LMULTI+B.LTO[7]
1425      DO I:=I+L[II].SIZE;
1426   IF B.LFROM[8]>0THEN {CTF LEVELS}
1427     FOR II:=B.LSINGLE+B.LMULTI+B.LRACE+B.LFROM[8]TO B.LSINGLE+B.LMULTI+B.LRACE+B.LTO[8]
1428      DO I:=I+L[II].SIZE;
1429   BLOCKWRITE(FF,I,2); {SIZE}
1430
1431   I2:=0; I:=$F401+B.SPRSIZE;
1432   IF B.LSINGLE=0THEN BLOCKWRITE(FF,I2,2)ELSE BEGIN
1433     BLOCKWRITE(FF,I,2); FOR II:=B.LFROM[1]TO B.LSINGLE DO INC(I,L[II].SIZE);
1434   END; {SINGLLVLS} INC(I,1+ORD(B.ENDDATA[0]));
1435   ISTACK:=I; FOR II:=2TO 6DO BEGIN
1436     I:=ISTACK; IF B.LFROM[II]=0THEN I:=0 ELSE
1437       FOR III:=B.LSINGLE TO B.LSINGLE+B.LFROM[II]-2DO
1438         INC(I,L[III].SIZE);
1439     BLOCKWRITE(FF,I,2);
1440   END; {LOC}
1441   FOR II:=B.LSINGLE+1TO B.LSINGLE+B.LMULTI DO INC(ISTACK,L[II].SIZE);
1442   I:=0; IF B.LRACE=0THEN BLOCKWRITE(FF,I,2)ELSE BLOCKWRITE(FF,ISTACK,2);
1443   FOR II:=B.LSINGLE+B.LMULTI+1TO B.LSINGLE+B.LMULTI+B.LRACE DO INC(ISTACK,L[II].SIZE);
1444   IF B.LCTF=0THEN BLOCKWRITE(FF,I,2)ELSE BLOCKWRITE(FF,ISTACK,2);
1445
1446   FOR II:=1TO 8DO BEGIN
1447     IF B.LFROM[II]=0THEN I:=0 ELSE I:=B.LTO[II]-B.LFROM[II]+1;
1448     BLOCKWRITE(FF,I,1);
1449   END; {NR}
1450
1451   BLOCKWRITE(FF,B.SPRSIZE,1);
1452   BLOCKWRITE(FF,B.SPRITE,B.SPRSIZE);
1453
1454   IF B.LFROM[1]>0THEN
1455     FOR II:=B.LFROM[1]TO B.LSINGLE DO SAVELEVEL(1); {LEVELS SINGLEPL}
1456   I:=255; BLOCKWRITE(FF,I,1); {END}
1457   BLOCKWRITE(FF,B.ENDDATA[1],ORD(B.ENDDATA[0])); {CODE}
1458
1459   IF B.LMULTI>0THEN
1460     FOR II:=B.LSINGLE+1TO B.LSINGLE+B.LMULTI DO SAVELEVEL(4); {LEVELS MULTIPL}
1461
1462   IF B.LFROM[7]>0THEN
1463     FOR II:=B.LSINGLE+B.LMULTI+B.LFROM[7]TO B.LSINGLE+B.LMULTI+B.LTO[7]
1464      DO SAVELEVEL(4); {LEVELS RACE}
1465
1466   IF B.LFROM[8]>0THEN
1467     FOR II:=B.LSINGLE+B.LMULTI+B.LRACE+B.LFROM[8]TO B.LSINGLE+B.LMULTI+B.LRACE+B.LTO[8]
1468      DO SAVELEVEL(4); {LEVELS CTF}
1469
1470   I:=0; I2:=1; IF B.LFROM[2]>0THEN I2:=I2+1+B.LTO[2]-B.LFROM[2];
1471   IF B.LFROM[3]>0THEN I2:=I2+1+B.LTO[3]-B.LFROM[3];
1472   FOR II:=1TO I2 DO BEGIN BLOCKWRITE(FF,I,2); BLOCKWRITE(FF,B.HINAME,3); END; {HI}
1473
1474   SEEK(FF,$37); I2:=0; I:=0;
1475   REPEAT INC(I,I2); BLOCKREAD(FF,I2,1,ISTACK); UNTIL ISTACK<>1;
1476   BLOCKWRITE(FF,I,2); {CHECKSUM}
1477 END; {FILECOMP}
1478
1479 PROCEDURE EDITFILE(STARTUP:BOOLEAN);
1480 VAR SFILE:SEARCHREC;
1481     FILESFOUND:INTEGER;
1482     SFILES:ARRAY[3..9]OF STRING[12];
1483 BEGIN
1484   FINDFIRST('*.LVL',$3F,SFILE); FILESFOUND:=2;
1485   WHILE DOSERROR=0DO BEGIN
1486     INC(FILESFOUND); IF FILESFOUND>9THEN BREAK;
1487     SFILES[FILESFOUND]:=SFILE.NAME; FINDNEXT(SFILE);
1488   END; {FIND MORE LEVELS}
1489   IF STARTUP THEN BEGIN
1490     TEXTCOLOR(C_MENU); WRITE('4'); TEXTCOLOR(C_TEXT); WRITELN(': ALL DONE!');
1491     DELAY(250);
1492   END; {STARTUP TEXT}
1493   REPEAT
1494     TITLEBAR(0,'CHOOSE LEVELFILE');
1495     TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITELN(': CREATE NEW FILE');
1496     TEXTCOLOR(C_MENU); WRITE('2'); TEXTCOLOR(C_TEXT); WRITELN(': EDIT EXISTING');
1497     FOR III:=3TO FILESFOUND DO BEGIN
1498       TEXTCOLOR(C_MENU); WRITE(III); TEXTCOLOR(C_TEXT); WRITE(': EDIT');
1499       TEXTCOLOR(C_VALUE); WRITELN(' ',SFILES[III]);
1500     END; {DISPLAY FOUND FILES}
1501
1502     III:=EDITMENU(FILESFOUND,2);
1503     CASE III OF
1504       -1:HALT;
1505       1:BEGIN
1506         FILENAME:=INPUTSTR('ENTER LEVELNAME','8 CHARS MAX; W/O EXTENSION','WORMLVL1',8);
1507         IF NOT FILEEXIST(FILENAME+'.LVL')THEN FILESAVE;
1508         OPENFILE;
1509       END; {NEW}
1510       2:BEGIN
1511         FILENAME:=INPUTSTR('ENTER FILENAME','W/O EXTENSION','WORMLVL1',8);
1512         OPENFILE;
1513       END; {EDIT}
1514       3..9:BEGIN
1515         WINDOW(1,15,40,24);
1516         FILENAME:=COPY(SFILES[III],1,POS('.LVL',SFILES[III])-1); OPENFILE;
1517       END; {WORMLVL1}
1518     END; {HANDLE}
1519   UNTIL III<>0;
1520 END; {EDITFILE}
1521
1522 PROCEDURE EDITQUIT;
1523 BEGIN
1524   REPEAT
1525     TITLEBAR(0,'CHOOSE LEVELFILE');
1526     TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITELN(': COMPILE, SAVE AND QUIT');
1527     TEXTCOLOR(C_MENU); WRITE('2'); TEXTCOLOR(C_TEXT); WRITE(': SAVE ');
1528       TEXTCOLOR(C_VALUE); WRITELN(FILENAME,'.LVL');
1529     TEXTCOLOR(C_MENU); WRITE('3'); TEXTCOLOR(C_TEXT); WRITELN(': QUIT');
1530     TEXTCOLOR(C_MENU); WRITE('4'); TEXTCOLOR(C_TEXT); WRITELN(': LOAD');
1531     TEXTCOLOR(C_MENU); WRITE('5'); TEXTCOLOR(C_TEXT); WRITELN(': CONTINUE EDITING');
1532     WRITELN; III:=EDITMENU(5,0);
1533     CASE III OF
1534       1:BEGIN
1535         FILESAVE; FILECOMP; QUITOK:=TRUE;
1536       END; {COMPILE}
1537       2:FILESAVE;
1538       3:BEGIN
1539         WINDOW(1,8,40,8);
1540         WRITE('REALLY QUIT '); TEXTCOLOR(C_SUB); WRITE('[Y/N]'); TEXTCOLOR(C_TEXT);
1541         WRITE('> '); REPEAT C:=READKEY; UNTIL(UPCASE(C)='Y')OR(UPCASE(C)='N');
1542         IF UPCASE(C)='Y'THEN QUITOK:=TRUE;
1543       END; {QUIT}
1544       4:EDITFILE(FALSE); {LOAD}
1545     END; {HANDLE}
1546   UNTIL(III>=4)OR(QUITOK);
1547 END; {EDITQUIT}
1548
1549
1550 BEGIN
1551  {$M 16384,0,655360}
1552   MAINTITLE; TITLEBAR(0,'LOADING WORMEDIT...');
1553   TEXTCOLOR(C_MENU); WRITE('1'); TEXTCOLOR(C_TEXT); WRITELN(': WORMEDIT STARTED');
1554   TEXTCOLOR(C_MENU); WRITE('2'); TEXTCOLOR(C_TEXT); WRITE(': MOUSE... ');
1555   MOUSE(0,I,I2,MX,MY); IF I<>0THEN BEGIN USEMOUSE:=TRUE; WRITELN(' FOUND');
1556   END {MOUSE} ELSE BEGIN
1557     WRITELN(' NOT FOUND!'); USEMOUSE:=FALSE; REPEAT UNTIL KEYPRESSED;
1558   END; {NO_MOUSE}
1559   TEXTCOLOR(C_MENU); WRITE('3'); TEXTCOLOR(C_TEXT); WRITELN(': SEARCHING FOR LEVELS');
1560   EDITFILE(TRUE); REPEAT EDITMAIN; EDITQUIT; UNTIL QUITOK=TRUE;
1561   WINDOW(1,1,80,25); TEXTCOLOR(7); TEXTBACKGROUND(0); CLRSCR;
1562   WRITE('EXITING '); TEXTCOLOR(15); WRITE('WORMEDIT 0.5 BY SHIAR');
1563   TEXTCOLOR(7); WRITELN('.'); WRITELN('BYEBYE..'); WRITELN;
1564 END.