home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 9 / 09.iso / l / l048 / 1.ddi / GRAPHIX.Z10 < prev    next >
Encoding:
Text File  |  1986-03-21  |  10.8 KB  |  391 lines

  1. (***********************************************************)
  2. (*                                                         *)
  3. (*                TURBO GRAPHIX version 1.06A              *)
  4. (*                                                         *)
  5. (*  Graphics module for Heath/Zenith 100 series computers  *)
  6. (*                                                         *)
  7. (*                  Copyright (C) 1985 by                  *)
  8. (*                  BORLAND International                  *)
  9. (*                                                         *)
  10. (***********************************************************)
  11.  
  12. const
  13.   XMaxGlb       = 79;    { Number of bytes-1 in one screen line }
  14.   XScreenMaxGlb = 639;   { Number of pixels-1 in one screen line }
  15.   YMaxGlb       = 224;   { Number of lines-1 on the screen }
  16.   IVStepGlb     = 3;     { Initial value of VStepGlb }
  17.  
  18.   ScreenSizeGlb = 25127; { Total size-1 in integers of the screen }
  19.   { Note: Scan lines are numbered from 0 to 392. The 128 bytes per scan
  20.     line are numbered from 0 to 127. Only 80 bytes of each scan line are
  21.     displayed and, scan lines 9 thru 16 are not displayed at all. Because
  22.     of this hole in the video RAM it takes a total of 393 scan lines to
  23.     get 225 scan lines displayed. Bytes 80 thru 127 of scan line 392 are
  24.     not added to ScreenSizeGlb since these bytes are not displayed.       }
  25.  
  26.   HardwareGrafBase     = $E000; { Segment location of the hardware screen }
  27.   FontLoaded : boolean = false; { Flag: has font been loaded yet? }
  28.   MinForeground        = 1;     { Lowest allowable foreground color }
  29.   MaxForeground        = 7;     { Highest allowable foreground color }
  30.   MinBackground        = 0;     { Lowest allowable background color }
  31.   MaxBackground        = 0;     { Highest allowable background color }
  32.   AspectFactor         = 0.495; { Aspect ratio for a true circle }
  33.   GrafPort             = $D8;   { Port address of the video control register }
  34.   White                = 0;
  35.   Cyan                 = 1;
  36.   Magenta              = 2;
  37.   Blue                 = 3;
  38.   Yellow               = 4;
  39.   Green                = 5;
  40.   Red                  = 6;
  41.   Black                = 7;
  42.  
  43. type
  44.   ScreenType        = array[0..ScreenSizeGlb] of integer;
  45.   ScreenPointer     = ^ScreenType;
  46.   FontChar          = array[0..8] of byte;
  47.   Z100Font          = array[0..255] of FontChar;
  48.   WindowStackRecord = record
  49.                         W : WindowType;
  50.                         Contents : ScreenPointer;
  51.                       end;
  52.   Stacks            = array[1..MaxWindowsGlb] of WindowStackRecord;
  53.  
  54. var
  55.   ScreenGlb     : ScreenPointer;
  56.   ConOutPtrSave : integer;
  57.   Font          : Z100Font;
  58.   Stack         : Stacks;
  59.  
  60. procedure Error(ErrProc, ErrCode : integer); forward; { Code in KERNEL.SYS }
  61.  
  62. procedure AllocateRAMScreen;
  63. { Allocates the RAM screen and makes sure that
  64.   ScreenGlb is on a segment (16 byte) boundary }
  65. var
  66.   Test : ^integer;
  67. begin
  68.   New(ScreenGlb);
  69.   while Ofs(ScreenGlb^) <> 0 do
  70.   begin
  71.     Dispose(ScreenGlb);
  72.     New(Test);
  73.     New(ScreenGlb);
  74.   end;
  75. end; { AllocateRAMScreen }
  76.  
  77. function HardwarePresent : boolean;
  78. { Graphics hardware is present in all H/Z100's }
  79. begin
  80.   HardwarePresent := true;
  81. end; { HardwarePresent }
  82.  
  83. function BaseAddress(Y : integer) : integer;
  84. { Calculate address of scan line Y }
  85. begin
  86.   BaseAddress := ((Y div 9) shl 4 + Y mod 9) shl 7;
  87. end; { BaseAddress }
  88.  
  89. procedure LeaveGraphic;
  90. { Exit from graphics mode and clear the screen }
  91. begin
  92.   Port[GrafPort] := 8;
  93.   if GrafModeGlb then
  94.     ConOutPtr := ConOutPtrSave;
  95.   GrafModeGlb := false;
  96.   Write(#27, 'y', 5); { Turn cursor on }
  97.   Write(#27, 'y', 1); { Disable 25th line }
  98.   ClrScr;
  99. end; { LeaveGraphic }
  100.  
  101. procedure DC(C : byte);
  102. { Draws a character whose ASCII code is C at
  103.   text coordinates XtextGlb,YTextGlb         }
  104. var
  105.   X, Y, I : integer;
  106. begin
  107.   X := XTextGlb - 1;
  108.   Y := (YTextGlb - 1) * 9;
  109.   for I := 0 to 8 do
  110.     Mem[GrafBase:BaseAddress(Y + I) + X] := Font[C][I];
  111. end; { DC }
  112.  
  113. procedure DisplayChar(C : byte);
  114. { Same as DC. Intended for internal use by the graphics system }
  115. begin
  116.   if C = 8 then
  117.   begin
  118.     if XTextGlb > 1 then
  119.       XTextGlb := XTextGlb - 1;
  120.   end
  121.   else
  122.     if C = 10 then
  123.     begin
  124.       if YTextGlb < 25 then
  125.         YTextGlb := YTextGlb + 1;
  126.     end
  127.   else
  128.     if C = 13 then
  129.       XTextGlb := 1
  130.     else
  131.     begin
  132.       DC(C);
  133.       if XTextGlb < 80 then
  134.         XTextGlb := XTextGlb + 1;
  135.     end;
  136. end; { DisplayChar }
  137.  
  138. procedure SetForegroundColor(Color : byte);
  139. { Set the foreground color. Colors range from 0 to 7 }
  140. begin
  141.   if Color in [0..7] then
  142.     Port[GrafPort] := Color + 8
  143.   else
  144.     Port[GrafPort] := White + 8;
  145. end; { SetForegroundColor }
  146.  
  147. procedure SetBackgroundColor(Color : integer);
  148. { Background color is always black }
  149. begin
  150. end; { SetBackgroundColor }
  151.  
  152. procedure ClearScreen;
  153. { Clear the displayed screen }
  154. var
  155.   ScanLine : byte;
  156. begin
  157.   ClrScr;
  158.   for ScanLine := 0 to YMaxGlb do
  159.     FillChar(Mem[GrafBase:BaseAddress(ScanLine)], XMaxGlb + 1, 0);
  160. end; { ClearScreen }
  161.  
  162. procedure EnterGraphic;
  163. { Enter graphics mode and clear the screen }
  164. var
  165.   FontFile : file of Z100Font;
  166. begin
  167.   if not FontLoaded then
  168.   begin
  169.     Assign(FontFile, '8x9.FON');
  170.     {$I-} Reset(FontFile); {$I+}
  171.     if IOResult = 0 then
  172.       begin
  173.         Read(FontFile, Font);
  174.         Close(FontFile);
  175.       end
  176.     else
  177.       FillChar(Font, SizeOf(Font), 0);
  178.     FontLoaded := true;
  179.   end;
  180.   Write(#27, 'x', 5);  { Turn cursor off }
  181.   Write(#27, 'x', 1);  { Enable 25th line }
  182.   ClearScreen;
  183.   Port[GrafPort] := 8;
  184.   if not GrafModeGlb then
  185.     ConOutPtrSave := ConOutPtr;
  186.   ConOutPtr := Ofs(DisplayChar);
  187.   GrafModeGlb := true;
  188. end; { EnterGraphic }
  189.  
  190. procedure DP(X, Y : integer);
  191. { Plot a pixel at X,Y. GrafBase (a global variable) contains
  192.   the address of the current screen (hardware or RAM).
  193.   ColorGlb is 0 for black or 255 for white }
  194. var
  195.   I : integer;
  196. begin
  197.   I := BaseAddress(Y) + X shr 3;
  198.   if ColorGlb = 255 then
  199.     Mem[GrafBase:I] := Mem[GrafBase:I] or 128 shr (X and 7)
  200.   else
  201.     Mem[GrafBase:I] := Mem[GrafBase:I] and ($FF7F shr (X and 7));
  202. end; { DP }
  203.  
  204. function PD(X, Y : integer) : boolean;
  205. { Return true if the color of the pixel at X,Y matches ColorGlb }
  206. begin
  207.   PD := (ColorGlb = 0) xor (Mem[GrafBase:BaseAddress(Y) + X shr 3]
  208.                        and (128 shr (X and 7)) <> 0);
  209. end; { PD }
  210.  
  211. procedure SetBackground8(Background : BackgroundArray);
  212. { Fills the active display with the specified bit pattern }
  213. var
  214.   I : integer;
  215. begin
  216.   for I := Y1RefGlb to Y2RefGlb do
  217.     FillChar(Mem[GrafBase:BaseAddress(I) + X1RefGlb], X2RefGlb - X1RefGlb + 1,
  218.              Background[I and 7]);
  219. end; { SetBackground8 }
  220.  
  221. procedure SetBackground(Byt : byte);
  222. { Determines the background pattern of the active window }
  223. var
  224.   Bk : BackgroundArray;
  225. begin
  226.   FillChar(Bk, 8, Byt);
  227.   SetBackground8(Bk);
  228. end; { SetBackground }
  229.  
  230. procedure DrawStraight(X1, X2, Y : integer);
  231. { Draw a horizontal line from X1,Y to X2,Y }
  232. var
  233.   I, X : integer;
  234.   DirectModeLoc : boolean;
  235. begin
  236.   if (not ((X1 < 0) or (X1 > XMaxGlb shl 3 + 7)) and not ((X2 < 0) or
  237.      (X2 > XMaxGlb shl 3 + 7)) and ((Y >= 0) and (Y <= YMaxGlb))) then
  238.   begin
  239.     DirectModeLoc := DirectModeGlb;
  240.     DirectModeGlb := true;
  241.     if X1 > X2 then
  242.     begin
  243.       X := X1;
  244.       X1 := X2;
  245.       X2 := X;
  246.     end;
  247.     if X2 - X1 < 16 then
  248.       for X := X1 to X2 do
  249.         DP(X, Y)
  250.     else
  251.       begin
  252.         X1 := X1 + 8;
  253.         for I := (X1 - 8) to (X1 and -8) do
  254.           DP(I, Y);
  255.         for I := (X2 and -8) to X2 do
  256.           DP(I, Y);
  257.         FillChar(Mem[GrafBase:BaseAddress(Y) + (X1 shr 3)],
  258.                 (X2 shr 3) - (X1 shr 3), ColorGlb);
  259.       end;
  260.     DirectModeGlb := DirectModeLoc;
  261.   end;
  262. end; { DrawStraight }
  263.  
  264. procedure SaveScreen(FileName : wrkstring);
  265. { Save the current screen on disk using FileName }
  266. type
  267.   BufferTyp = array[0..XMaxGlb] of byte;
  268.   PicFile = file of BufferTyp;
  269. var
  270.   PictureFile : PicFile;
  271.   Buffer : BufferTyp;
  272.   ScanLine : integer;
  273.   IOErr : boolean;
  274.  
  275. procedure IOCheck;
  276. begin
  277.   IOErr := IOResult <> 0;
  278.   if IOErr then
  279.     Error(27, 5);
  280. end; { IOCheck }
  281.  
  282. begin
  283.   Assign(PictureFile, FileName);
  284.   {$I-} Rewrite(PictureFile); {$I+}
  285.   IOCheck;
  286.   if not IOErr then
  287.   for ScanLine := 0 to YMaxGlb Do
  288.   begin
  289.     Move(Mem[GrafBase:BaseAddress(ScanLine)], Buffer, XMaxGlb + 1);
  290.     Write(PictureFile, Buffer);
  291.   end;
  292.   Close(PictureFile);
  293. end; { SaveScreen }
  294.  
  295. procedure LoadScreen(FileName : wrkstring);
  296. { Load screen from file FileName }
  297. type
  298.   BufferTyp = array[0..XMaxGlb] of byte;
  299.   PicFile = file of BufferTyp;
  300. var
  301.   PictureFile : PicFile;
  302.   Buffer : BufferTyp;
  303.   ScanLine : integer;
  304.   IOErr : boolean;
  305.  
  306. procedure IOCheck;
  307. begin
  308.   IOErr := IOResult <> 0;
  309.   if IOErr then
  310.     Error(11, 5);
  311. end; { IOCheck }
  312.  
  313. begin
  314.   Assign(PictureFile, FileName);
  315.   {$I-} Reset(PictureFile); {$I+}
  316.   IOCheck;
  317.   if not IOErr then
  318.   begin
  319.     for ScanLine := 0 to YMaxGlb do
  320.     begin
  321.       Read(PictureFile, Buffer);
  322.       Move(Buffer, Mem[GrafBase:BaseAddress(ScanLine)], XMaxGlb + 1);
  323.     end;
  324.     Close(PictureFile);
  325.   end;
  326. end; { LoadScreen }
  327.  
  328. procedure SwapScreen;
  329. { Exchanges the contents the of the displayed
  330.   screen with the contents of the RAM screen  }
  331. var
  332.   ScanLine,
  333.   LineBase,
  334.   RamScrSeg  : integer;
  335.   LineBuffer : array[0..XMaxGlb] of byte;
  336. begin
  337.   if RamScreenGlb then
  338.   begin
  339.     RamScrSeg := Seg(ScreenGlb^);
  340.     for ScanLine := 0 to YMaxGlb do
  341.     begin
  342.       LineBase := ((ScanLine div 9) shl 4 + ScanLine mod 9) shl 7;
  343.       Move(Mem[GrafBase:LineBase], LineBuffer, XMaxGlb + 1);
  344.       Move(Mem[RamScrSeg:LineBase], Mem[GrafBase:LineBase], XMaxGlb + 1);
  345.       Move(LineBuffer, Mem[RamScrSeg:LineBase], XMaxGlb + 1);
  346.     end;
  347.   end;
  348. end; { SwapScreen }
  349.  
  350. procedure CopyScreen;
  351. { Copies the active screen onto the inactive screen }
  352. var
  353.   ToBase,
  354.   LineBase : integer;
  355.   ScanLine : byte;
  356. begin
  357.   if RamScreenGlb then
  358.   begin
  359.     if GrafBase = HardwareGrafBase then
  360.       ToBase := Seg(ScreenGlb^)
  361.     else
  362.       ToBase := HardwareGrafBase;
  363.     for ScanLine := 0 to YMaxGlb do
  364.     begin
  365.       LineBase := BaseAddress(ScanLine);
  366.       Move(Mem[GrafBase:LineBase], Mem[ToBase:LineBase], XMaxGlb + 1);
  367.     end;
  368.   end;
  369. end; { CopyScreen }
  370.  
  371. procedure InvertScreen;
  372. { Inverts the image on the active screen }
  373. var
  374.   ScanLine,
  375.   LineBase,
  376.   ScreenWord,
  377.   AddrSum : integer;
  378. begin
  379.   for ScanLine := 0 to YMaxGlb do
  380.   begin
  381.     LineBase := ((ScanLine div 9) shl 4 + ScanLine mod 9) shl 7;
  382.     ScreenWord := 0;
  383.     while ScreenWord < XMaxGlb do
  384.     begin
  385.       AddrSum := LineBase + ScreenWord;
  386.       MemW[GrafBase:AddrSum] := not(MemW[GrafBase:AddrSum]);
  387.       ScreenWord := ScreenWord + 2;
  388.     end;
  389.   end;
  390. end; { InvertScreen }
  391.