home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 9 / 09.iso / l / l045 / 2.ddi / GKERNEL.PAS < prev    next >
Encoding:
Pascal/Delphi Source File  |  1987-12-23  |  34.2 KB  |  1,419 lines

  1. (********************************************************************)
  2. (*                         GRAPHIX TOOLBOX 4.0                      *)
  3. (*       Copyright (c) 1985, 87 by  Borland International, Inc.     *)
  4. (********************************************************************)
  5. unit GKernel;
  6.  
  7. interface
  8.  
  9. {$I Float.inc}  { Determines what type Float means. }
  10.  
  11. uses
  12.   Dos, Crt, GDriver;
  13.  
  14. procedure GotoXY(X, Y : integer);
  15. { Set the text position }
  16.  
  17. procedure ClrEOL;
  18. { Clear from the current text position to the end of the line }
  19.  
  20. procedure SetBreakOff;
  21.  
  22. procedure SetBreakOn;
  23.  
  24. function GetErrorCode : integer;
  25.  
  26. procedure SetWindowModeOff;
  27.  
  28. procedure SetWindowModeOn;
  29.  
  30. procedure SetClippingOn;
  31.  
  32. procedure SetClippingOff;
  33.  
  34. procedure SetMessageOn;
  35.  
  36. procedure SetMessageOff;
  37.  
  38. procedure SetHeaderOn;
  39.  
  40. procedure SetHeaderOff;
  41.  
  42. procedure SetHeaderToTop;
  43.  
  44. procedure SetHeaderToBottom;
  45.  
  46. procedure RemoveHeader(I : integer);
  47.  
  48. procedure SetColorWhite;
  49.  
  50. procedure SetColorBlack;
  51.  
  52. function GetWindow : integer;
  53.  
  54. function GetColor : word;
  55.  
  56. function Clipping : boolean;
  57.  
  58. function WindowMode : boolean;
  59.  
  60. procedure SetScreenAspect(Aspect : Float);
  61.  
  62. function GetScreenAspect : Float;
  63.  
  64. procedure SetAspect(Aspect : Float);
  65.  
  66. function GetAspect : Float;
  67.  
  68. procedure SetLinestyle(Ls : word);
  69.  
  70. function GetLinestyle : word;
  71.  
  72. procedure SetVStep(Vs : word);
  73.  
  74. function GetVStep : word;
  75.  
  76. procedure DefineHeader(I : integer; Hdr : WrkString);
  77.  
  78. procedure SelectScreen(I : word);
  79.  
  80. function GetScreen : byte;
  81.  
  82. procedure DefineWorld(I : integer; X_1, Y_1, X_2, Y_2 : Float);
  83.  
  84. procedure SelectWorld(I : integer);
  85.  
  86. procedure ReDefineWindow(I, X_1, Y_1, X_2, Y_2 : integer);
  87.  
  88. procedure DefineWindow(I, X_1, Y_1, X_2, Y_2 : integer);
  89.  
  90. function TextLeft(TX, Boundary : integer) : integer;
  91.  
  92. function TextRight(TX, Boundary : integer) : integer;
  93.  
  94. function TextUp(TY, Boundary : integer) : integer;
  95.  
  96. function TextDown(TY, Boundary : integer) : integer;
  97.  
  98. procedure DefineTextWindow(I, X1, Y1, X2, Y2, B : integer);
  99.  
  100. procedure SelectWindow(I : integer);
  101.  
  102. function WindowX(X : Float) : integer;
  103.  
  104. function WindowY(Y : Float) : integer;
  105.  
  106. procedure InitGraphic;
  107.  
  108. procedure ResetWindows;
  109.  
  110. procedure ResetWorlds;
  111.  
  112. function Clip(var X1, Y1, X2, Y2 : integer) : boolean;
  113.  
  114. procedure DrawPoint(Xr, Yr : Float);
  115.  
  116. function PointDrawn(Xr, Yr : Float) : boolean;
  117.  
  118. procedure DrawLine(X1, Y1, X2, Y2 : Float);
  119.  
  120. procedure DrawLineClipped(X1, Y1, X2, Y2 : integer);
  121.  
  122. procedure DrawCrossDiag(X, Y, Scale : integer);
  123.  
  124. procedure DrawWye(X, Y, Scale : integer);
  125.  
  126. procedure DrawDiamond(X, Y, Scale : integer);
  127.  
  128. procedure DrawCircleDirect(Xr, Yr, R : integer; DirectModeLoc : boolean);
  129.  
  130. procedure DrawCircle(X_R, Y_R, Xradius : Float);
  131.  
  132. procedure DrawCross(X1, Y1, Scale : integer);
  133.  
  134. procedure DrawStar(X, Y, Scale : integer);
  135.  
  136. procedure DrawSquareC(X1, Y1, X2, Y2 : integer; Fill : boolean);
  137.  
  138. procedure DrawSquare(X1, Y1, X2, Y2 : Float; Fill : boolean);
  139.  
  140. procedure DrawAscii(var X, Y : integer; Size, Ch : byte);
  141.  
  142. procedure DrawText(X, Y, Scale : integer; Txt : WrkString);
  143.  
  144. procedure DrawTextW(X, Y : Float; Scale : integer; Txt : WrkString);
  145.  
  146. procedure DrawBorder;
  147.  
  148. procedure HardCopy(Inverse : boolean; Mode : byte);  { EPSON }
  149.  
  150. implementation
  151.  
  152. procedure GotoXY{(X, Y : integer)};
  153. { Set the text position }
  154. begin
  155.   if (X >= 1) and (X <= 80) and    { Ignore illegal values }
  156.      (Y >= 1) and (Y <= 25) then
  157.   begin
  158.     if GrafModeGlb then
  159.       begin
  160.         XTextGlb := X;      { Set text postion in graphics mode }
  161.         YTextGlb := Y;
  162.       end
  163.     else
  164.       Crt.GotoXY(X, Y);     { Set cursor position in text mode }
  165.   end;
  166. end; { GotoXY }
  167.  
  168. procedure ClrEOL;
  169. { Clear from the current text position to the end of the line }
  170. var
  171.   TempX : integer;
  172. begin
  173.   if GrafModeGlb then
  174.     begin
  175.       TempX := XTextGlb;
  176.       for XTextGlb := TempX to 79 do
  177.         DC(32);
  178.       XTextGlb := TempX;
  179.     end
  180.   else
  181.     Crt.ClrEOL;
  182. end; { ClrEOL }
  183.  
  184. procedure SetBreakOff;
  185. begin
  186.   BrkGlb := false;
  187. end; { SetBreakOff }
  188.  
  189. procedure SetBreakOn;
  190. begin
  191.   BrkGlb := true;
  192. end; { SetBreakOn }
  193.  
  194. function GetErrorCode{ : integer};
  195. begin
  196.   GetErrorCode := ErrCodeGlb;
  197.   ErrCodeGlb := -1; { Reset to No error }
  198. end; { GetErrorCode }
  199.  
  200. procedure SetWindowModeOff;
  201. begin
  202.   DirectModeGlb := true;
  203. end; { SetWindowModeOff }
  204.  
  205. procedure SetWindowModeOn;
  206. begin
  207.   DirectModeGlb := false;
  208. end; { SetWindowModeOn }
  209.  
  210. procedure SetClippingOn;
  211. begin
  212.   ClippingGlb := true;
  213. end; { SetClippingOn }
  214.  
  215. procedure SetClippingOff;
  216. begin
  217.   ClippingGlb := false;
  218. end; { SetClippingOff }
  219.  
  220. procedure SetMessageOn;
  221. begin
  222.   MessageGlb := true;
  223. end; { SetMessageOn }
  224.  
  225. procedure SetMessageOff;
  226. begin
  227.   MessageGlb := false;
  228. end; { SetMessageOff }
  229.  
  230. procedure SetHeaderOn;
  231. begin
  232.   HeaderGlb := true;
  233. end; { SetHeaderOn }
  234.  
  235. procedure SetHeaderOff;
  236. begin
  237.   HeaderGlb := false;
  238. end; { SetHeaderOff }
  239.  
  240. procedure SetHeaderToTop;
  241. begin
  242.   TopGlb := true;
  243. end; { SetHeaderToTop }
  244.  
  245. procedure SetHeaderToBottom;
  246. begin
  247.   TopGlb := false;
  248. end; { SetHeaderToBottom }
  249.  
  250. procedure RemoveHeader{(I : integer)};
  251. begin
  252.   if I in [1..MaxWindowsGlb] then
  253.     with GrafWindow[I] do
  254.     begin
  255.       if HeaderGlb then
  256.       begin
  257.         if Top then
  258.           begin
  259.             Dec(Y1, HeaderSizeGlb);
  260.             if Y1 < 0 then
  261.               Y1 := 0;
  262.             RedefineWindow(I, X1, Y1, X2, Y2);
  263.           end
  264.         else
  265.           begin
  266.             Inc(Y2, HeaderSizeGlb);
  267.             if Y2 > YMaxGlb then
  268.               Y2 := YMaxGlb;
  269.             RedefineWindow(I, X1, Y1, X2, Y2);
  270.           end;
  271.         if I = WindowNdxGlb then
  272.           SelectWindow(I);
  273.       end;
  274.       Drawn := false;
  275.       Top := true;
  276.       Header := '';
  277.     end
  278.   else
  279.     Error(22, 2);
  280. end; { RemoveHeader }
  281.  
  282. procedure SetColorWhite;
  283. begin
  284.   ColorGlb := 255;
  285. end; { SetColorWhite }
  286.  
  287. procedure SetColorBlack;
  288. begin
  289.   ColorGlb := 0;
  290. end; { SetColorBlack }
  291.  
  292. function GetWindow{ : integer};
  293. begin
  294.   GetWindow := WindowNdxGlb;
  295. end; { GetWindow }
  296.  
  297. function GetColor{ : word};
  298. begin
  299.   GetColor := ColorGlb;
  300. end; { GetColor }
  301.  
  302. function Clipping{ : boolean};
  303. begin
  304.   Clipping := ClippingGlb;
  305. end; { Clipping }
  306.  
  307. function WindowMode{ : boolean};
  308. begin
  309.   WindowMode := not DirectModeGlb;
  310. end; { WindowMode }
  311.  
  312. procedure SetScreenAspect{(Aspect : Float)};
  313. begin
  314.   if Aspect <> 0.0 then
  315.     AspectGlb := abs(Aspect);
  316. end; { SetScreenAspect }
  317.  
  318. function GetScreenAspect{ : Float};
  319. begin
  320.   GetScreenAspect := AspectGlb;
  321. end; { GetScreenAspect }
  322.  
  323. procedure SetAspect{(Aspect : Float)};
  324. begin
  325.   if Aspect <> 0.0 then
  326.     AspectGlb := abs(Aspect) * AspectFactor;
  327. end; { SetAspect }
  328.  
  329. function GetAspect{ : Float};
  330. begin
  331.   GetAspect := AspectGlb / AspectFactor;
  332. end; { GetAspect }
  333.  
  334. procedure SetLinestyle{(Ls : word)};
  335. var
  336.   I : integer;
  337. const
  338.   Lsa : array[0..4] of byte = ($FF,$88,$F8,$E4,$EE);
  339.  
  340. begin
  341.   if (Ls < 0) or (Ls > 5) then
  342.     Ls := Ls and $FF + $100;
  343.   LineStyleGlb := Ls;
  344.   if Ls < 5 then
  345.     Ls := Lsa[Ls];
  346.   for I := 0 to 7 do
  347.     LineStyleArrayGlb[7 - I] := ((Ls shr I) and 1) <> 0;
  348.   CntGlb := 7;
  349. end; { SetLinestyle }
  350.  
  351. function GetLinestyle{ : word};
  352. begin
  353.   GetLinestyle := LinestyleGlb;
  354. end; { GetLinestyle }
  355.  
  356. procedure SetVStep{(Vs : word)};
  357. begin
  358.   if Vs > 0 then
  359.     VStepGlb := Vs;
  360. end; { SetVStep }
  361.  
  362. function GetVStep{ : word};
  363. begin
  364.   GetVStep := VStepGlb;
  365. end; { GetVStep }
  366.  
  367. procedure DefineHeader{(I : integer; Hdr : WrkString)};
  368. begin
  369.   if (I in [1..MaxWindowsGlb]) then
  370.     GrafWindow[I].Header := Hdr
  371.   else
  372.     Error(3, 2);
  373. end; { DefineHeader }
  374.  
  375. procedure SelectScreen{(I : word)};
  376. begin
  377.   if RamScreenGlb and (I = 2) then
  378.     GrafBase := Seg(ScreenGlb^)
  379.   else
  380.     GrafBase := HardwareGrafBase;
  381. end; { SelectScreen }
  382.  
  383. function GetScreen{ : byte};
  384. begin
  385.   if GrafBase = HardwareGrafBase then
  386.     GetScreen := 1
  387.   else
  388.     GetScreen := 2;
  389. end; { GetScreen }
  390.  
  391. procedure DefineWorld{(I : integer; X_1, Y_1, X_2, Y_2 : Float)};
  392. begin
  393.   if ((X_1 <> X_2) and (Y_1 <> Y_2)) and (I in [1..MaxWorldsGlb]) then
  394.     with World[I] do
  395.     begin
  396.       X1 := X_1; Y1 := Y_1; X2 := X_2; Y2 := Y_2;
  397.       if I > MaxWorldGlb then
  398.         MaxWorldGlb := I;
  399.     end
  400.   else if I in [1..MaxWorldsGlb] then
  401.     Error(1, 3)
  402.   else
  403.     Error(1, 2);
  404. end; { DefineWorld }
  405.  
  406. procedure SelectWorld{(I : integer)};
  407. begin
  408.   if (I in [1..MaxWorldGlb]) then
  409.     with World[I] do
  410.     begin
  411.       WorldNdxGlb := I;
  412.       X1WldGlb := X1;
  413.       Y1WldGlb := Y1;
  414.       X2WldGlb := X2;
  415.       Y2WldGlb := Y2;
  416.     end
  417.   else
  418.     Error(2, 2);
  419. end; { SelectWorld }
  420.  
  421. procedure ReDefineWindow{(I, X_1, Y_1, X_2, Y_2 : integer)};
  422. begin
  423.   if (I in [1..MaxWindowsGlb]) and (X_1 <= X_2) and (Y_1 <= Y_2) and (X_1 >= 0)
  424.      and (X_2 <= XMaxGlb) and (Y_1 >= 0) and (Y_2 <= YMaxGlb) then
  425.     with GrafWindow[I] do
  426.     begin
  427.       Drawn := false;
  428.       X1 := X_1;
  429.       Y1 := Y_1;
  430.       X2 := X_2;
  431.       Y2 := Y_2;
  432.       if I > MaxWindowGlb then
  433.         MaxWindowGlb := I;
  434.     end
  435.   else if I in [1..MaxWindowsGlb] then
  436.     Error(3, 3)
  437.   else
  438.     Error(3, 2);
  439. end; { ReDefineWindow }
  440.  
  441. procedure DefineWindow{(I, X_1, Y_1, X_2, Y_2 : integer)};
  442. begin
  443.   ReDefineWindow(I, X_1, Y_1, X_2, Y_2);
  444.   if ErrCodeGlb = -1 then
  445.   begin
  446.     with GrafWindow[I] do
  447.     begin
  448.       Header := '';
  449.       Top := true;
  450.       Drawn := false;
  451.     end;
  452.   end;
  453. end; { DefineWindow }
  454.  
  455. function TextLeft{(TX, Boundary : integer) : integer};
  456. var
  457.   TL : integer;
  458. begin
  459.   TL := ((TX - 1) * ((XScreenMaxGlb + 1) div 80) - Boundary) div 8;
  460.   if TL < 0 then
  461.     TL := 0
  462.   else if TL > XMaxGlb then
  463.     TL := XMaxGlb;
  464.   TextLeft := TL;
  465. end; { TextLeft }
  466.  
  467. function TextRight{(TX, Boundary : integer) : integer};
  468. var
  469.   TR : integer;
  470. begin
  471.   TR := (XScreenMaxGlb + 1) div 80;
  472.   TR := (TX * TR + Boundary - 1) div 8;
  473.   if TR < 0 then
  474.     TR := 0
  475.   else if TR > XMaxGlb then
  476.     TR := XMaxGlb;
  477.   TextRight := TR;
  478. end; { TextRight }
  479.  
  480. function TextUp{(TY, Boundary : integer) : integer};
  481. var
  482.   TU : integer;
  483. begin
  484.   TU := (TY - 1) * ((YMaxGlb + 1) div 25) - Boundary;
  485.   if TU < 0 then
  486.     TU := 0
  487.   else if TU > YMaxGlb then
  488.     TU := YMaxGlb;
  489.   TextUp := TU;
  490. end; { TextUp }
  491.  
  492. function TextDown{(TY, Boundary : integer) : integer};
  493. var
  494.   TD : integer;
  495. begin
  496.   TD := TY * ((YMaxGlb + 1) div 25) + Boundary - 1;
  497.   if TD < 0 then
  498.     TD := 0
  499.   else if TD > YMaxGlb then
  500.     TD := YMaxGlb;
  501.   TextDown := TD;
  502. end; { TextDown }
  503.  
  504. procedure DefineTextWindow{(I, X1, Y1, X2, Y2, B : integer)};
  505. begin
  506.   DefineWindow(I, TextLeft(X1, B), TextUp(Y1, B),
  507.                TextRight(X2, B), TextDown(Y2, B));
  508. end; { DefineTextWindow }
  509.  
  510. procedure SelectWindow{(I : integer)};
  511. begin
  512.   if (I in [1..MaxWindowGlb]) then
  513.     with GrafWindow[I] do
  514.     begin
  515.       WindowNdxGlb := I;
  516.       X1RefGlb := X1;
  517.       Y1RefGlb := Y1;
  518.       X2RefGlb := X2;
  519.       Y2RefGlb := Y2;
  520.       BxGlb := ((X2 - X1) shl 3 + 7) / (X2WldGlb - X1WldGlb);
  521.       ByGlb := (Y2 - Y1) / (Y2WldGlb - Y1WldGlb);
  522.       AxGlb := (X1 shl 3) - X1WldGlb * BxGlb;
  523.       AyGlb := Y1 - Y1WldGlb * ByGlb;
  524.       if AxisGlb then
  525.       begin
  526.         AxisGlb := false;
  527.         X1Glb := 0;
  528.         Y1Glb := 0;
  529.         X2Glb := 0;
  530.         Y2Glb := 0;
  531.       end;
  532.     end
  533.   else
  534.     Error(4, 2);
  535. end; { SelectWindow }
  536.  
  537. function WindowX{(X : Float) : integer};
  538. var
  539.   Temp : Float;
  540. begin
  541.   Temp := AxGlb + BxGlb * X;
  542.   if Temp > MaxInt then
  543.     WindowX := MaxInt
  544.   else if Temp < -32767 then
  545.     WindowX := -32767
  546.   else
  547.     WindowX := trunc(Temp);
  548. end; { WindowX }
  549.  
  550. function WindowY{(Y : Float) : integer};
  551. var
  552.   Temp : Float;
  553. begin
  554.   Temp := AyGlb + ByGlb * Y;
  555.   if Temp > MaxInt then
  556.     WindowY := MaxInt
  557.   else if Temp < -32767 then
  558.     WindowY := -32767
  559.   else
  560.     WindowY := trunc(Temp);
  561. end; { WindowY }
  562.  
  563. procedure InitGraphic;
  564. var
  565.   Fil : file of CharArray;
  566.   Tfile : text;
  567.   Test : ^integer;
  568.   Temp : WrkString;
  569.   I : word;
  570. begin
  571.   MessageGlb := true;
  572.   BrkGlb := false;
  573.   GrafModeGlb := false;
  574.   GotoXY(1, 1);
  575.   if not HardwarePresent then
  576.   begin
  577.     ClrScr;
  578.     GotoXY(1, 2);
  579.     WriteLn('Fatal Error: graphics hardware not found or not properly activated');
  580.     Halt;
  581.   end;
  582.   GetMem(ErrorProc[0], 16);
  583.   GetMem(ErrorCode[0], 24);
  584.   ErrorProc[0]^ := 'InitGraphic';
  585.   ErrorCode[0]^ := 'Error.MSG missing';
  586.   Assign(Tfile, 'Error.msg');
  587.   {$I-} Reset(Tfile); {$I+}
  588.   if IOresult = 0 then
  589.     begin
  590.       for I := 0 to MaxProcsGlb do
  591.       begin
  592.         ReadLn(Tfile, Temp);
  593.         if I <> 0 then
  594.           GetMem(ErrorProc[I], Length(Temp) + 1);
  595.         ErrorProc[I]^ := Temp;
  596.       end;
  597.       for I := 0 to MaxErrsGlb do
  598.       begin
  599.         ReadLn(Tfile, Temp);
  600.         if I <> 0 then
  601.           GetMem(ErrorCode[I], Length(Temp) + 1);
  602.         ErrorCode[I]^ := Temp;
  603.       end;
  604.       ReadLn(Tfile, PcGlb);
  605.       Close(Tfile);
  606.     end
  607.   else
  608.     begin
  609.       GetMem(ErrorProc[1], 14);
  610.       ErrorProc[1]^ := '** UNKNOWN **';
  611.       for I := 2 to MaxProcsGlb do
  612.         ErrorProc[I] := ErrorProc[1];
  613.       for I := 1 to MaxErrsGlb do
  614.         ErrorCode[I] := ErrorProc[1];
  615.       Error(0, 0);
  616.     end;
  617.   for I := 1 to MaxWorldsGlb do
  618.     DefineWorld(I, 0, 0, XScreenMaxGlb, YMaxGlb);
  619.   MaxWorldGlb := 1;
  620.   for I := 1 to MaxWindowsGlb do
  621.   begin
  622.     DefineWindow(I, 0, 0, XMaxGlb, YMaxGlb);
  623.     with Stack[I] do
  624.     begin
  625.       W.Size := 0;
  626.       Contents := nil;
  627.     end;
  628.   end;
  629.   MaxWindowGlb := 1;
  630.   if CharFile <> '' then
  631.   begin
  632.     Assign(Fil, CharFile);
  633.     {$I-} Reset(Fil); {$I+}
  634.     if IOresult = 0 then
  635.       begin
  636.         Read(Fil, CharSet);
  637.         Close(Fil);
  638.       end
  639.     else
  640.       Error(0, 1);
  641.   end;
  642.   BrkGlb := true;
  643.   if RamScreenGlb then
  644.   begin
  645.     AllocateRAMScreen;
  646.     SelectScreen(2);
  647.     ClearScreen;
  648.   end;
  649.   SelectScreen(1);
  650.   WindowNdxGlb := 1;
  651.   SelectWorld(1);
  652.   SelectWindow(1);
  653.   SetColorWhite;
  654.   SetClippingOn;
  655.   SetAspect(AspectFactor);
  656.   DirectModeGlb := false;
  657.   PieGlb := false;
  658.   SetMessageOn;
  659.   SetHeaderOff;
  660.   SetHeaderToTop;
  661.   ErrCodeGlb := -1;
  662.   SetLineStyle(0);
  663.   VStepGlb := IVStepGlb;
  664.   EnterGraphic;
  665.   X1Glb := 0;
  666.   X2Glb := 0;
  667.   Y1Glb := 0;
  668.   Y2Glb := 0;
  669.   AxisGlb := false;
  670.   HatchGlb := false;
  671. end; { InitGraphic }
  672.  
  673. procedure ResetWindows;
  674. var
  675.   I : word;
  676. begin
  677.   for I := 1 to MaxWindowsGlb do
  678.   begin
  679.     DefineWindow(I, 0, 0, XMaxGlb, YMaxGlb);
  680.     RemoveHeader(I);
  681.   end;
  682.   SelectWindow(1);
  683. end; { ResetWindows }
  684.  
  685. procedure ResetWorlds;
  686. var
  687.   I : word;
  688. begin
  689.   for I := 1 to MaxWorldsGlb do
  690.     DefineWorld(I, 0, 0, XScreenMaxGlb, YMaxGlb);
  691.   SelectWorld(1);
  692.   SelectWindow(WindowNdxGlb);
  693. end; { ResetWorlds }
  694.  
  695. function Clip{(var X1, Y1, X2, Y2 : integer) : boolean};
  696. var
  697.   Ix1, Iy1, Ix2, Iy2, Dummy, X1Loc, X2Loc : integer;
  698.   ClipLoc : boolean;
  699.   Temp : Float;
  700.  
  701. function Inside(X, Xx1, Xx2 : integer) : integer;
  702. begin
  703.   Inside := 0;
  704.   if X < Xx1 then
  705.     Inside := -1
  706.   else if X > Xx2 then
  707.     Inside := 1;
  708. end; { Inside }
  709.  
  710. begin  { Clip }
  711.   Clip := true;
  712.   ClipLoc := true;
  713.   if ClippingGlb then
  714.     begin
  715.       if HatchGlb then
  716.         begin
  717.           X1Loc := X1RefGlb;
  718.           X2Loc := X2RefGlb;
  719.         end
  720.       else
  721.         begin
  722.           X1Loc := X1RefGlb shl 3;
  723.           X2Loc := X2RefGlb shl 3 + 7;
  724.         end;
  725.       Ix1 := Inside(X1, X1Loc, X2Loc);
  726.       Iy1 := Inside(Y1, Y1RefGlb, Y2RefGlb);
  727.       Ix2 := Inside(X2, X1Loc, X2Loc);
  728.       Iy2 := Inside(Y2, Y1RefGlb, Y2RefGlb);
  729.       if (Ix1 or Ix2 or Iy1 or Iy2) <> 0 then
  730.       begin
  731.         if X1 <> X2 then
  732.         begin
  733.           if Ix1 <>0 then
  734.           begin
  735.             if Ix1 < 0 then
  736.               Dummy := X1Loc
  737.             else
  738.               Dummy := X2Loc;
  739.             if Y2 <> Y1 then
  740.             begin
  741.               Temp := (Y2 - Y1) / (X2 - X1) * (Dummy - X1);
  742.               if Temp > MaxInt then
  743.                 Temp := MaxInt
  744.               else if Temp < -32767 then
  745.                 Temp := -32767;
  746.               Y1 := Y1 + trunc(Temp);
  747.             end;
  748.             X1 := Dummy;
  749.           end;
  750.           if (Ix2 <> 0) and (X1 <> X2) then
  751.           begin
  752.             if Ix2 < 0 then
  753.               Dummy := X1Loc
  754.             else
  755.               Dummy := X2Loc;
  756.             if Y2 <> Y1 then
  757.             begin
  758.               Temp := (Y2 - Y1) / (X2 - X1) * (Dummy - X1);
  759.               if Temp > MaxInt then
  760.                 Temp := MaxInt
  761.               else if Temp < -32767 then
  762.                 Temp := -32767;
  763.               Y2 := Y1 + trunc(Temp);
  764.             end;
  765.             X2 := Dummy;
  766.           end;
  767.           Iy1 := Inside(Y1, Y1RefGlb, Y2RefGlb);
  768.           Iy2 := Inside(Y2, Y1RefGlb, Y2RefGlb);
  769.         end;
  770.         if Y1 <> Y2 then
  771.         begin
  772.           if Iy1 <> 0 then
  773.           begin
  774.             if Iy1 < 0 then
  775.               Dummy := Y1RefGlb
  776.             else
  777.               Dummy := Y2RefGlb;
  778.             if X1 <> X2 then
  779.             begin
  780.               Temp := (X2 - X1) / (Y2 - Y1) * (Dummy - Y1);
  781.               if Temp > MaxInt then
  782.                 Temp := MaxInt
  783.               else if Temp < -32767 then
  784.                 Temp := -32767;
  785.               X1 := X1 + trunc(Temp);
  786.             end;
  787.             Y1 := Dummy;
  788.           end;
  789.           if Iy2 <> 0 then
  790.           begin
  791.             if Iy2 < 0 then
  792.               Dummy := Y1RefGlb
  793.             else
  794.               Dummy := Y2RefGlb;
  795.             if X1 <> X2 then
  796.             begin
  797.               Temp := (X2 - X1) / (Y2 - Y1) * (Dummy - Y1);
  798.               if Temp > MaxInt then
  799.                 Temp := MaxInt
  800.               else if Temp < -32767 then
  801.                 Temp := -32767;
  802.               X2 := X1 + trunc(Temp);
  803.             end;
  804.             Y2 := Dummy;
  805.           end;
  806.         end;
  807.         Iy1 := Inside(Y1, Y1RefGlb, Y2RefGlb);
  808.         Iy2 := Inside(Y2, Y1RefGlb, Y2RefGlb);
  809.         if (Iy1 <> 0) or (Iy2 <> 0) then
  810.           ClipLoc := false;
  811.         if ClipLoc then
  812.         begin
  813.           Ix1 := Inside(X1, X1Loc, X2Loc);
  814.           Ix2 := Inside(X2, X1Loc, X2Loc);
  815.           if (Ix2 <> 0) or (Ix1 <> 0) then
  816.             ClipLoc := false;
  817.         end;
  818.         Clip := ClipLoc;
  819.       end;
  820.     end;
  821. end; { Clip }
  822.  
  823. procedure DrawPoint{(Xr, Yr : Float)};
  824. var
  825.   X, Y : integer;
  826. begin
  827.   if DirectModeGlb then
  828.     DP(trunc(Xr), trunc(Yr))
  829.   else
  830.     begin
  831.       X := WindowX(Xr);
  832.       Y := WindowY(Yr);
  833.       if ClippingGlb then
  834.         begin
  835.           if (X >= X1RefGlb shl 3) and (X <= X2RefGlb shl 3 + 7) then
  836.             if (Y >= Y1RefGlb) and (Y <= Y2RefGlb) then
  837.               DP(X, Y);
  838.         end
  839.       else
  840.         DP(X, Y);
  841.     end;
  842. end; { DrawPoint }
  843.  
  844. function PointDrawn{(Xr, Yr : Float) : boolean};
  845. begin
  846.   if DirectModeGlb then
  847.     PointDrawn := PD(trunc(Xr), trunc(Yr))
  848.   else
  849.     PointDrawn := PD(WindowX(Xr), WindowY(Yr));
  850. end; { PointDrawn }
  851.  
  852. procedure DrawLine{(X1, Y1, X2, Y2 : Float)};
  853. var
  854.   X1Loc, Y1Loc, X2Loc, Y2Loc : integer;
  855.  
  856. procedure DrawLineDirect(X1, Y1, X2, Y2 : integer);
  857. var
  858.   X, Y, DeltaX, DeltaY, XStep, YStep, Direction : integer;
  859. begin
  860.   X := X1;
  861.   Y := Y1;
  862.   XStep := 1;
  863.   YStep := 1;
  864.   if X1 > X2 then
  865.     XStep := -1;
  866.   if Y1 > Y2 then
  867.     YStep := -1;
  868.   DeltaX := abs(X2 - X1);
  869.   DeltaY := abs(Y2 - Y1);
  870.   if DeltaX = 0 then
  871.     Direction := -1
  872.   else
  873.     Direction := 0;
  874.   while not ((X = X2) and (Y = Y2)) do
  875.   begin
  876.     if LinestyleGlb = 0 then
  877.       DP(X, Y)
  878.     else
  879.       begin
  880.         CntGlb := (CntGlb + 1) and 7;
  881.         if LineStyleArrayGlb[CntGlb] then
  882.           DP(X, Y);
  883.       end;
  884.     if Direction < 0 then
  885.       begin
  886.         Y := Y + YStep;
  887.         Direction := Direction + DeltaX;
  888.       end
  889.     else
  890.       begin
  891.         X := X + XStep;
  892.         Direction := Direction - DeltaY;
  893.       end;
  894.   end;
  895. end; { DrawLineDirect }
  896.  
  897. begin { DrawLine }
  898.   if DirectModeGlb then
  899.     DrawLineDirect(trunc(X1), trunc(Y1), trunc(X2), trunc(Y2))
  900.   else
  901.     begin
  902.       X1Loc := WindowX(X1);
  903.       Y1Loc := WindowY(Y1);
  904.       X2Loc := WindowX(X2);
  905.       Y2Loc := WindowY(Y2);
  906.       if Clip(X1Loc, Y1Loc, X2Loc, Y2Loc) then
  907.         DrawLineDirect(X1Loc, Y1Loc, X2Loc, Y2Loc);
  908.     end;
  909. end; { DrawLine }
  910.  
  911. procedure DrawLineClipped{(X1, Y1, X2, Y2 : integer)};
  912. var
  913.   Temp : boolean;
  914. begin
  915.   if Clip(X1, Y1, X2, Y2) then
  916.   begin
  917.     Temp := DirectModeGlb;
  918.     DirectModeGlb := true;
  919.     DrawLine(X1, Y1, X2, Y2);
  920.     DirectModeGlb := Temp;
  921.   end;
  922. end; { DrawLineClipped }
  923.  
  924. procedure DrawCrossDiag{(X, Y, Scale : integer)};
  925. begin
  926.   DrawLineClipped(X - Scale, Y + Scale, X + Scale + 1, Y - Scale - 1);
  927.   DrawLineClipped(X - Scale, Y - Scale, X + Scale + 1, Y + Scale + 1);
  928. end; { DrawCrossDiag }
  929.  
  930. procedure DrawWye{(X, Y, Scale : integer)};
  931. begin
  932.   DrawLineClipped(X - Scale, Y - Scale, X, Y);
  933.   DrawLineClipped(X + Scale, Y - Scale, X, Y);
  934.   DrawLineClipped(X, Y, X, Y + Scale);
  935. end; { DrawWye }
  936.  
  937. procedure DrawDiamond{(X, Y, Scale : integer)};
  938. begin
  939.   DrawLineClipped(X - Scale, Y, X, Y - Scale - 1);
  940.   DrawLineClipped(X, Y - Scale + 1, X + Scale, Y + 1);
  941.   DrawLineClipped(X + Scale, Y + 1, X, Y + Scale);
  942.   DrawLineClipped(X, Y + Scale, X - Scale, Y);
  943. end; { DrawDiamond }
  944.  
  945. procedure DrawCircleDirect{(Xr, Yr, R : integer; DirectModeLoc : boolean)};
  946. const
  947.   N = 14;
  948. type
  949.   Circ = array[1..N] of integer;
  950. const
  951.   X : Circ = (0,121,239,355,465,568,663,749,823,885,935,971,993,1000);
  952. var
  953.   Xk1, Xk2, Yk1, Yk2, Xp1, Yp1, Xp2, Yp2 : integer;
  954.   Xfact, Yfact : Float;
  955.   I : integer;
  956.  
  957. procedure DrawLinW(X1, Y1, X2, Y2 : integer);
  958. var
  959.   DrawIt     : boolean;
  960.   DirectSave : boolean;
  961. begin
  962.   DrawIt := DirectModeLoc;
  963.   if DrawIt then
  964.     begin
  965.       DrawIt := Clip(X1, Y1, X2, Y2);
  966.       if DrawIt then
  967.       begin
  968.         DirectSave := DirectModeGlb;
  969.         DirectModeGlb := true;
  970.         DrawLine(X1, Y1, X2, Y2);
  971.         DirectModeGlb := DirectSave;
  972.       end;
  973.     end
  974.   else
  975.     begin
  976.       DirectSave := DirectModeGlb;
  977.       DirectModeGlb := true;
  978.       DrawLine(X1, Y1, X2, Y2);
  979.       DirectModeGlb := DirectSave;
  980.     end;
  981. end; { DrawLinW }
  982.  
  983. begin { DrawCircleDirect }
  984.   Xfact := abs(R * 0.001);
  985.   Yfact := Xfact * AspectGlb;
  986.   if Xfact > 0.0 then
  987.     begin
  988.       Xk1 := trunc(X[1] * Xfact + 0.5);
  989.       Yk1 := trunc(X[N] * Yfact + 0.5);
  990.       for I := 2 to N do
  991.       begin
  992.         Xk2 := trunc(X[I] * Xfact + 0.5);
  993.         Yk2 := trunc(X[N - I + 1] * Yfact + 0.5);
  994.         Xp1 := Xr - Xk1;
  995.         Yp1 := Yr + Yk1;
  996.         Xp2 := Xr - Xk2;
  997.         Yp2 := Yr + Yk2;
  998.         DrawLinW(Xp1, Yp1, Xp2, Yp2);
  999.         Xp1 := Xr + Xk1;
  1000.         Xp2 := Xr + Xk2;
  1001.         DrawLinW(Xp1, Yp1, Xp2, Yp2);
  1002.         Yp1 := Yr - Yk1;
  1003.         Yp2 := Yr - Yk2;
  1004.         DrawLinW(Xp1, Yp1 + 1, Xp2, Yp2 + 1);
  1005.         Xp1 := Xr - Xk1;
  1006.         Xp2 := Xr - Xk2;
  1007.         DrawLinW(Xp1, Yp1 + 1, Xp2, Yp2 + 1);
  1008.         Xk1 := Xk2;
  1009.         Yk1 := Yk2;
  1010.       end;
  1011.     end
  1012.   else
  1013.     DP(Xr, Yr);
  1014. end; { DrawCircleDirect }
  1015.  
  1016. procedure DrawCircle{(X_R, Y_R, Xradius : Float)};
  1017. var
  1018.   DirectModeLoc : boolean;
  1019. begin
  1020.   DirectModeLoc := DirectModeGlb;
  1021.   DirectModeGlb := true;
  1022.   if DirectModeLoc then
  1023.     DrawCircleDirect(trunc(X_R), trunc(Y_R), trunc(Xradius), true)
  1024.   else
  1025.     DrawCircleDirect(WindowX(X_R), WindowY(Y_R), trunc(Xradius * 100.0), true);
  1026.   DirectModeGlb := DirectModeLoc;
  1027. end; { DrawCircle }
  1028.  
  1029. procedure DrawCross{(X1, Y1, Scale : integer)};
  1030. begin
  1031.   DrawLineClipped(X1 - Scale, Y1, X1 + Scale + 2, Y1);
  1032.   DrawLineClipped(X1, Y1 - Scale, X1, Y1 + Scale + 1);
  1033. end; { DrawCross }
  1034.  
  1035. procedure DrawStar{(X, Y, Scale : integer)};
  1036. begin
  1037.   DrawLineClipped(X - Scale, Y + Scale, X + Scale + 1, Y - Scale - 1);
  1038.   DrawLineClipped(X - Scale, Y - Scale, X + Scale + 1, Y + Scale + 1);
  1039.   DrawLineClipped(X - Scale - 2, Y, X + Scale + 4, Y);
  1040. end; { DrawStar }
  1041.  
  1042. procedure DrawSquareC{(X1, Y1, X2, Y2 : integer; Fill : boolean)};
  1043. var
  1044.   I : integer;
  1045.  
  1046. procedure DSC(X1, X2, Y : integer);
  1047. var
  1048.   DirectSave : boolean;
  1049. begin
  1050.   if Clip(X1, Y, X2, Y) then
  1051.     if LineStyleGlb = 0 then
  1052.       DrawStraight(X1, X2, Y)
  1053.     else
  1054.       begin
  1055.         DirectSave := DirectModeGlb;
  1056.         DirectModeGlb := true;
  1057.         DrawLine(X1, Y, X2, Y);
  1058.         DirectModeGlb := DirectSave;
  1059.       end;
  1060. end; { DSC }
  1061.  
  1062. begin { DrawSquareC }
  1063.   if not Fill then
  1064.     begin
  1065.       DSC(X1, X2, Y1);
  1066.       DrawLineClipped(X2, Y1, X2, Y2);
  1067.       DSC(X1, X2, Y2);
  1068.       DrawLineClipped(X1, Y2, X1, Y1);
  1069.     end
  1070.   else
  1071.     if Y2 > Y1 then
  1072.       for I := Y1 to Y2 do
  1073.         DSC(X1, X2, I)
  1074.     else
  1075.       for I := Y2 to Y1 do
  1076.         DSC(X1, X2, I);
  1077. end; { DrawSquareC }
  1078.  
  1079. procedure DrawSquare{(X1, Y1, X2, Y2 : Float; Fill : boolean)};
  1080. var
  1081.   I, X1Loc, Y1Loc, X2Loc, Y2Loc : integer;
  1082.   DirectModeLoc : boolean;
  1083.  
  1084. procedure DS(X1, X2, Y : integer);
  1085. begin
  1086.   if LineStyleGlb = 0 then
  1087.     DrawStraight(X1, X2, Y)
  1088.   else
  1089.     DrawLine(X1, Y, X2, Y);
  1090. end; { DS }
  1091.  
  1092. procedure DSC(X1, X2, Y : integer);
  1093. begin
  1094.   if Clip(X1, Y, X2, Y) then
  1095.     DS(X1, X2, Y);
  1096. end; { DSC }
  1097.  
  1098. procedure DrawSqr(X1, Y1, X2, Y2 : integer; Fill : boolean);
  1099. var
  1100.   I : integer;
  1101. begin
  1102.   if not Fill then
  1103.     begin
  1104.       DS(X1, X2, Y1);
  1105.       DrawLine(X2, Y1, X2, Y2);
  1106.       DS(X1, X2, Y2);
  1107.       DrawLine(X1, Y2, X1, Y1);
  1108.     end
  1109.   else
  1110.     for I := Y1 to Y2 do
  1111.       DS(X1, X2, I);
  1112. end; { DrawSqr }
  1113.  
  1114. begin { DrawSquare }
  1115.   if DirectModeGlb then
  1116.     DrawSqr(trunc(X1), trunc(Y1), trunc(X2), trunc(Y2), Fill)
  1117.   else
  1118.     begin
  1119.       DirectModeLoc := DirectModeGlb;
  1120.       DirectModeGlb := true;
  1121.       X1Loc := WindowX(X1);
  1122.       Y1Loc := WindowY(Y1);
  1123.       X2Loc := WindowX(X2);
  1124.       Y2Loc := WindowY(Y2);
  1125.       if not Fill then
  1126.         begin
  1127.           DSC(X1Loc, X2Loc, Y1Loc);
  1128.           DrawLineClipped(X2Loc, Y1Loc, X2Loc, Y2Loc);
  1129.           DSC(X1Loc, X2Loc, Y2Loc);
  1130.           DrawLineClipped(X1Loc, Y2Loc, X1Loc, Y1Loc);
  1131.         end
  1132.       else
  1133.         for I := Y1Loc to Y2Loc do
  1134.           DSC(X1Loc, X2Loc, I);
  1135.       DirectModeGlb := DirectModeLoc;
  1136.     end;
  1137. end; { DrawSquare }
  1138.  
  1139. procedure DrawAscii{(var X, Y : integer; Size, Ch : byte)};
  1140. var
  1141.   X1Ref, X2Ref, Xpos, Ypos, Xstart, Ystart, Xend, Yend, Xx, Yy : integer;
  1142.   CharByte : byte;
  1143. begin
  1144.   X1Ref := X1RefGlb shl 3;
  1145.   X2Ref := X2RefGlb shl 3 + 7;
  1146.   for Ypos := 0 to 5 do
  1147.   begin
  1148.     CharByte := (CharSet[Ch, (7 - Ypos) shr 1] shr ((Ypos and 1) shl 2)) and $0F;
  1149.     for Xpos := 0 to 3 do
  1150.       if (CharByte shr (3 - Xpos)) and 1 <> 0 then
  1151.       begin
  1152.         Xstart := X + Xpos * Size;
  1153.         Xend := Xstart + Size - 1;
  1154.         Ystart := Y + 1 + (Ypos - 2) * Size;
  1155.         Yend := Ystart + Size - 1;
  1156.         if ClippingGlb then
  1157.         begin
  1158.           if Xstart < X1Ref then
  1159.             Xstart := X1Ref;
  1160.           if Xend > X2Ref then
  1161.             Xend := X2Ref;
  1162.           if Ystart < Y1RefGlb then
  1163.             Ystart := Y1RefGlb;
  1164.           if Yend > Y2RefGlb then
  1165.             Yend := Y2RefGlb;
  1166.         end;
  1167.         for Yy := Ystart to Yend do
  1168.           for Xx := Xstart to Xend do
  1169.             DP(Xx, Yy);
  1170.       end;
  1171.   end;
  1172.   X := X + Size * 6;
  1173. end; { DrawAscii }
  1174.  
  1175. procedure DrawText{(X, Y, Scale : integer; Txt : WrkString)};
  1176. var
  1177.   LineStyleLoc, Code, AsciiValue, StringLen,
  1178.   I, SymbolScale, SymbolCode : integer;
  1179.   DirectModeLoc : boolean;
  1180. begin
  1181.   DirectModeLoc := DirectModeGlb;
  1182.   DirectModeGlb := true;
  1183.   LineStyleLoc := LinestyleGlb;
  1184.   SetLineStyle(0);
  1185.   StringLen := Length(Txt);
  1186.   I := 1;
  1187.   while I <= StringLen do
  1188.   begin
  1189.     AsciiValue := Ord(Txt[I]);
  1190.     if AsciiValue = 27 then
  1191.       begin
  1192.         SymbolScale := Scale;
  1193.         I := I + 1;
  1194.         if I <= StringLen then
  1195.         begin
  1196.           Val(Txt[I], SymbolCode, Code);
  1197.           if (I + 2 <= StringLen) and (Ord(Txt[I + 1]) = 64) then
  1198.           begin
  1199.             Val(Txt[I + 2], SymbolScale, Code);
  1200.             I := I + 2;
  1201.           end;
  1202.           case SymbolCode of
  1203.             1   : DrawCross(X + SymbolScale, Y + Scale, SymbolScale);
  1204.             2   : DrawCrossDiag(X + SymbolScale, Y + Scale, SymbolScale);
  1205.             3,4 : DrawSquareC(X, Y + (SymbolScale shl 1) - 1,
  1206.                               X + (SymbolScale shl 1), Y - 1, (SymbolCode = 4));
  1207.             5   : begin
  1208.                     DrawDiamond(X + trunc(1.5 * SymbolScale),
  1209.                                 Y + SymbolScale - 1, SymbolScale + 1);
  1210.                     X := X + SymbolScale;
  1211.                   end;
  1212.             6   : DrawWye(X + SymbolScale, Y + SymbolScale - 1, SymbolScale);
  1213.             7   : begin
  1214.                     DrawStar(X + SymbolScale shl 1, Y + SymbolScale - 1, SymbolScale);
  1215.                     X := X + SymbolScale shl 1;
  1216.                   end;
  1217.             8   : DrawCircleDirect(X + SymbolScale, Y + (SymbolScale shr 1),
  1218.                                    SymbolScale + 1, true);
  1219.           end;
  1220.           X := X + 3 * SymbolScale;
  1221.           SymbolScale := Scale;
  1222.         end;
  1223.       end
  1224.     else
  1225.       DrawAscii(X, Y, Scale, AsciiValue);
  1226.     I := I + 1;
  1227.   end;
  1228.   DirectModeGlb := DirectModeLoc;
  1229.   SetLineStyle(LineStyleLoc);
  1230. end; { DrawText }
  1231.  
  1232. procedure DrawTextW{(X, Y : Float; Scale : integer; Txt : WrkString)};
  1233. begin
  1234.   if DirectModeGlb then
  1235.     DrawText(trunc(X), trunc(Y), Scale, Txt)
  1236.   else
  1237.     DrawText(WindowX(X), WindowY(Y), Scale, Txt);
  1238. end; { DrawTextW }
  1239.  
  1240. procedure DrawBorder;
  1241. var
  1242.   ClipLoc, DirectModeLoc : boolean;
  1243.   Xl1, Xl2 : integer;
  1244.  
  1245. procedure DrawHeaderBackground(Y1, Y2 : integer);
  1246. var
  1247.   I : integer;
  1248. begin
  1249.   for I := Y1 to Y2 do
  1250.     DrawStraight(Xl1, Xl2, I);
  1251. end; { DrawHeaderBackground }
  1252.  
  1253. procedure DrawHeader;
  1254. var
  1255.   Y1Hdr, Y2Hdr, Yl1, Yl2 : integer;
  1256. begin
  1257.   with GrafWindow[WindowNdxGlb] do
  1258.   begin
  1259.     if Drawn then
  1260.       if Top then
  1261.         begin
  1262.           ReDefineWindow(WindowNdxGlb, X1RefGlb, Y1RefGlb - HeaderSizeGlb,
  1263.                          X2RefGlb, Y2RefGlb);
  1264.           SelectWindow(WindowNdxGlb);
  1265.         end
  1266.       else
  1267.         begin
  1268.           ReDefineWindow(WindowNdxGlb, X1RefGlb, Y1RefGlb, X2RefGlb,
  1269.                          Y2RefGlb + HeaderSizeGlb);
  1270.           SelectWindow(WindowNdxGlb);
  1271.         end;
  1272.     if TopGlb then
  1273.       begin
  1274.         Yl1 := Y1RefGlb + HeaderSizeGlb;
  1275.         Yl2 := Y2RefGlb;
  1276.         Y1Hdr := Y1RefGlb;
  1277.         Y2Hdr := Y1RefGlb + HeaderSizeGlb - 1;
  1278.       end
  1279.     else
  1280.       begin
  1281.         Yl1 := Y1RefGlb;
  1282.         Yl2 := Y2RefGlb - HeaderSizeGlb;
  1283.         Y1Hdr := Y2RefGlb - HeaderSizeGlb + 1;
  1284.         Y2Hdr := Y2RefGlb;
  1285.       end;
  1286.     Top := TopGlb;
  1287.     ReDefineWindow(WindowNdxGlb, X1RefGlb, Yl1, X2RefGlb, Yl2);
  1288.     SelectWindow(WindowNdxGlb);
  1289.     DrawHeaderBackground(Y1Hdr, Y2Hdr);
  1290.     ColorGlb := 255 - ColorGlb;
  1291.     DrawText(Xl1 + 2 + (Xl2 - Xl1 - Length(Header) * 6) shr 1,
  1292.              Y1Hdr + 3, 1, Header);
  1293.     DrawSquare(Xl1, Y1Hdr, Xl2, Y2Hdr, false);
  1294.     ColorGlb := 255 - ColorGlb;
  1295.     DrawSquare(Xl1, Y1RefGlb, Xl2, Y2RefGlb, false);
  1296.     Drawn := true;
  1297.   end;
  1298. end; { DrawHeader }
  1299.  
  1300. begin { DrawBorder }
  1301.   DirectModeLoc := DirectModeGlb;
  1302.   DirectModeGlb := true;
  1303.   ClipLoc := ClippingGlb;
  1304.   ClippingGlb := false;
  1305.   Xl1 := X1RefGlb shl 3;
  1306.   Xl2 := X2RefGlb shl 3 + 7;
  1307.   with GrafWindow[WindowNdxGlb] do
  1308.     if ((HeaderGlb) and (Length(Header) > 0)) and (Y2 - Y1 > HeaderSizeGlb) and
  1309.        ((Length(Header) * 6) < abs(Xl2 - Xl1) - 4) then
  1310.       DrawHeader
  1311.     else
  1312.       begin
  1313.         Drawn := false;
  1314.         DrawSquare(Xl1, Y1RefGlb, Xl2, Y2RefGlb, false);
  1315.       end;
  1316.     DirectModeGlb := DirectModeLoc;
  1317.     ClippingGlb := ClipLoc;
  1318. end; { DrawBorder }
  1319.  
  1320. procedure HardCopy(Inverse : boolean; Mode : byte);
  1321. { Graphics screen dump routine for EPSON compatible    }
  1322. { printers. Pre-FX series of EPSON printers should     }
  1323. { only use Mode 1.                                     }
  1324. {                                                      }
  1325. { Mode: 1       = Double-Density 120 dots per inch     }
  1326. {       2       = High-Speed D-D 120 dots per inch     }
  1327. {       3       = Quadruple-Density 240 dots per inch  }
  1328. {       0, 4, 5 = 80 dots per inch                     }
  1329. {       6       = 90 dots per inch                     }
  1330.  
  1331. const
  1332.   Esc   = 27;
  1333. var
  1334.   ScanLine : integer; { The current scan line }
  1335.   n1, n2   : byte;    { 2 byte printer control code }
  1336.  
  1337. procedure SendByte(B : byte);
  1338. { Send one byte to the printer }
  1339. const
  1340.   LPTPortNum = 1; { Defaults to LPT1. 2 = LPT2 }
  1341. var
  1342.   Regs : Registers;
  1343. begin
  1344.   Regs.AH := 0;
  1345.   Regs.AL := B;
  1346.   Regs.DX := Pred(LPTPortNum);
  1347.   Intr($17, Regs);
  1348. end; { SendByte }
  1349.  
  1350. {$B+}  { Turn off short circuit boolean evaluation }
  1351.  
  1352. function ConstructByte(X, Y : integer) : byte;
  1353. { Construct a print byte by reading bits from the graphics screen buffer }
  1354. const
  1355.   Bits : array[0..7] of byte = (128,64,32,16,8,4,2,1);
  1356. var
  1357.   CByte, Bit : byte;
  1358. begin
  1359.   Y := Y shl 3;  { Y := Y * 8 }
  1360.   CByte := 0;
  1361.   for Bit := 0 to 7 do
  1362.   if ((Mem[GrafBase:BaseAddress(Y+Bit) + X shr 3] and Bits[X and 7]) <> 0) then
  1363.     CByte := CByte + Bits[Bit];
  1364.   ConstructByte := CByte;
  1365. end; { ConstructByte }
  1366.  
  1367. {$B-}  { Turn on short circuit boolean evaluation }
  1368.  
  1369. procedure DoLine;
  1370. { Dumps one print line to the printer }
  1371. var
  1372.   XPixel    : integer;
  1373.   PrintByte : byte;
  1374. begin
  1375.   if Mode = 1 then
  1376.     begin
  1377.       SendByte(Esc);        { Select double-density graphics print mode }
  1378.       SendByte(Ord('L'));
  1379.     end
  1380.   else
  1381.     begin                   { Select 8-Pin graphics print mode }
  1382.       SendByte(Esc);
  1383.       SendByte(Ord('*'));
  1384.       SendByte(Mode);
  1385.     end;
  1386.   SendByte(n1);             { Send 2 byte control code }
  1387.   SendByte(n2);
  1388.   for XPixel := 0 to XScreenMaxGlb do
  1389.   begin
  1390.     PrintByte := ConstructByte(XPixel, ScanLine);
  1391.     if Inverse then
  1392.       PrintByte := not PrintByte;
  1393.     SendByte(PrintByte);    { Send print byte }
  1394.   end;
  1395.   SendByte(10);             { Send line feed }
  1396. end; { DoLine }
  1397.  
  1398. begin { HardCopy }
  1399.   Mode := Mode mod 7;              { Modes 0 through 6 supported }
  1400.   if (Mode = 0) or (Mode = 5) then
  1401.     Mode := 4;                     { Modes 0 and 5 use Mode 4 }
  1402.  
  1403.   SendByte(Esc);                   { Select 24/216-inch line spacing }
  1404.   SendByte(Ord('3'));
  1405.   SendByte(24);
  1406.  
  1407.   n1 := Lo(Succ(XScreenMaxGlb));   { Determine 2 byte control code for }
  1408.   n2 := Hi(Succ(XScreenMaxGlb));   { the number of dots per print line }
  1409.  
  1410.   for ScanLine := 0 to (YMaxGlb div 8) do
  1411.     DoLine;                        { Do a print line }
  1412.  
  1413.   SendByte(Esc); SendByte(2);      { Select 1/6-inch line spacing }
  1414. end; { HardCopy }
  1415.  
  1416. begin
  1417.   GrafModeGlb := false;
  1418. end. { GKernel }
  1419.