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

  1. (********************************************************************)
  2. (*                         GRAPHIX TOOLBOX 4.0                      *)
  3. (*       Copyright (c) 1985, 87 by  Borland International, Inc.     *)
  4. (********************************************************************)
  5. unit GWindow;
  6.  
  7. interface
  8.  
  9. {$I Float.inc}  { Determines what type Float means. }
  10.  
  11. uses
  12.   Dos, Crt, GDriver, GKernel;
  13.  
  14. procedure MoveVer(Delta : integer; FillOut : boolean);
  15.  
  16. procedure MoveHor(Delta : integer; FillOut : boolean);
  17.  
  18. procedure CopyWindow(From, Tu : byte; X1, Y1 : integer);
  19.  
  20. procedure SaveWindow(N : integer; FileName : WrkString);
  21.  
  22. procedure LoadWindow(N, Xpos, Ypos : integer; FileName : WrkString);
  23.  
  24. function WindowSize(Win : integer) : word;
  25.  
  26. procedure ClearWindowStack(Win : integer);
  27.  
  28. procedure StoreWindow(Win : integer);
  29.  
  30. procedure RestoreWindow(Win, DeltaX, DeltaY : integer);
  31.  
  32. procedure SaveWindowStack(FileName : WrkString);
  33.  
  34. procedure LoadWindowStack(FileName : WrkString);
  35.  
  36. procedure ResetWindowStack;
  37.  
  38. procedure InvertWindow;
  39.  
  40. implementation
  41.  
  42. procedure MoveVer{(Delta : integer; FillOut : boolean)};
  43. var
  44.   Direction, Outer, XLen : integer;
  45.   FromBase, From, Tu : word;
  46.  
  47. procedure MoveVer1(VStep : integer);
  48. var
  49.   I : integer;
  50. begin
  51.   XLen := X2RefGlb - X1RefGlb + 1;
  52.   if Direction = -1 then
  53.     for I := Y1RefGlb to Y2RefGlb do
  54.     begin
  55.       if I > 0 then
  56.       begin
  57.         From := BaseAddress(I);
  58.         Tu := BaseAddress(I - VStep);
  59.         Move(Mem[GrafBase:From + X1RefGlb], Mem[GrafBase:Tu + X1RefGlb], XLen);
  60.       end;
  61.     end
  62.   else
  63.     for I := Y2RefGlb downto Y1RefGlb do
  64.       if I < YMaxGlb then
  65.         begin
  66.           From := BaseAddress(I);
  67.           Tu := BaseAddress(I + VStep);
  68.           Move(Mem[GrafBase:From + X1RefGlb], Mem[GrafBase:Tu + X1RefGlb], XLen);
  69.         end;
  70.   if not RamScreenGlb then
  71.     FillOut := false;
  72.   if not FillOut then
  73.     if Direction = -1 then
  74.       for I := Y2RefGlb downto Y2RefGlb - VStep + 1 do
  75.         FillChar(Mem[GrafBase:BaseAddress(I) + X1RefGlb], XLen, not ColorGlb)
  76.     else
  77.       for I := Y1RefGlb to Y1RefGlb + VStep - 1 do
  78.          FillChar(Mem[GrafBase:BaseAddress(I) + X1RefGlb], XLen, not ColorGlb)
  79.   else
  80.     begin
  81.       if GrafBase = HardwareGrafBase then
  82.         FromBase := Seg(ScreenGlb^)
  83.       else
  84.         FromBase:=HardwareGrafBase;
  85.       if Direction = -1 then
  86.         for I := Y2RefGlb downto Y2RefGlb - VStep + 1 do
  87.           Move(Mem[FromBase:BaseAddress(I) + X1RefGlb],
  88.                Mem[GrafBase:BaseAddress(I) + X1RefGlb], XLen)
  89.       else
  90.         for I := Y1RefGlb to Y1RefGlb + VStep - 1 do
  91.           Move(Mem[FromBase:BaseAddress(I) + X1RefGlb],
  92.                Mem[GrafBase:BaseAddress(I) + X1RefGlb], XLen);
  93.     end;
  94.   ReDefineWindow(WindowNdxGlb, X1RefGlb, Y1RefGlb + VStep * Direction,
  95.                  X2RefGlb, Y2RefGlb + VStep * Direction);
  96.   SelectWindow(WindowNdxGlb);
  97. end; { MoveVer1 }
  98.  
  99. begin { MoveVer }
  100.   if Delta <> 0 then
  101.   begin
  102.     Direction := 1;
  103.     if Delta < 0 then
  104.       Direction := -1;
  105.     with GrafWindow[WindowNdxGlb] do
  106.     if Drawn then
  107.       if Top then
  108.         Y1RefGlb := Y1RefGlb - HeaderSizeGlb
  109.       else
  110.         Y2RefGlb := Y2RefGlb + HeaderSizeGlb;
  111.     if (Y1RefGlb + Delta < 0) or (Y2RefGlb + Delta > YMaxGlb) then
  112.       Error(23, 7)
  113.     else
  114.       begin
  115.         for Outer := 1 to abs(Delta) div VStepGlb do
  116.           MoveVer1(VStepGlb);
  117.         if abs(Delta) mod VStepGlb <> 0 then
  118.           MoveVer1(abs(Delta) mod VStepGlb);
  119.       end;
  120.     with GrafWindow[WindowNdxGlb] do
  121.       if Drawn then
  122.         if Top then
  123.           Y1RefGlb := Y1RefGlb + HeaderSizeGlb
  124.         else
  125.           Y2RefGlb := Y2RefGlb - HeaderSizeGlb;
  126.   end;
  127. end; { MoveVer }
  128.  
  129. procedure MoveHor{(Delta : integer; FillOut : boolean)};
  130. var
  131.   Direction, Outer, XLen : integer;
  132.   I : integer;
  133.   FromBase, Y : word;
  134. begin
  135.   if Delta <> 0 then
  136.   begin
  137.     Direction := 1;
  138.     if Delta < 0 then
  139.       Direction := -1;
  140.     with GrafWindow[WindowNdxGlb] do
  141.     if Drawn then
  142.       if Top then
  143.         Y1RefGlb := Y1RefGlb - HeaderSizeGlb
  144.       else
  145.         Y2RefGlb := Y2RefGlb + HeaderSizeGlb;
  146.     if (X1RefGlb+Delta < 0) or (X2RefGlb+Delta > XMaxGlb) then
  147.       Error(24, 7)
  148.     else
  149.       begin
  150.         for Outer := 1 to abs(Delta) do
  151.         begin
  152.           XLen := X2RefGlb - X1RefGlb + 1;
  153.           for I := Y1RefGlb to Y2RefGlb do
  154.           begin
  155.             Y := BaseAddress(I);
  156.             Move(Mem[GrafBase:Y + X1RefGlb],
  157.                  Mem[GrafBase:Y + X1RefGlb + Direction], XLen);
  158.             if not RamScreenGlb then
  159.               FillOut := false;
  160.             if not FillOut then
  161.               if Direction < 0 then
  162.                 Mem[GrafBase:Y + X2RefGlb] := (not ColorGlb) and $FF
  163.               else
  164.                 { prevents range check errors }
  165.                 Mem[GrafBase:Y + X1RefGlb] := (not ColorGlb) and $FF
  166.               else
  167.                 begin
  168.                   if GrafBase = HardwareGrafBase then
  169.                     FromBase := Seg(ScreenGlb^)
  170.                   else
  171.                     FromBase := HardwareGrafBase;
  172.                   if Direction = -1 then
  173.                     Mem[GrafBase:Y + X2RefGlb] := Mem[FromBase:Y + X2RefGlb]
  174.                   else
  175.                     Mem[GrafBase:Y + X1RefGlb] := Mem[FromBase:Y + X1RefGlb];
  176.                 end;
  177.           end;
  178.           ReDefineWindow(WindowNdxGlb, X1RefGlb + Direction, Y1RefGlb,
  179.                          X2RefGlb + Direction, Y2RefGlb);
  180.           SelectWindow(WindowNdxGlb);
  181.         end;
  182.       end;
  183.       with GrafWindow[WindowNdxGlb] do
  184.       if Drawn then
  185.         if Top then
  186.           Y1RefGlb := Y1RefGlb + HeaderSizeGlb
  187.         else
  188.           Y2RefGlb := Y2RefGlb - HeaderSizeGlb;
  189.   end;
  190. end; { MoveHor }
  191.  
  192. procedure CopyWindow{(From, Tu : byte; X1, Y1 : integer)};
  193. var
  194.   XLen, YLen : integer;
  195.   FromBase, ToBase, I : word;
  196.  
  197. begin
  198.   if (X1 < 0) or (Y1 < 0) then
  199.     Error(17, 3)
  200.   else
  201.     begin
  202.       with GrafWindow[WindowNdxGlb] do
  203.         if Drawn then
  204.           if Top then
  205.             Y1RefGlb := Y1RefGlb - HeaderSizeGlb
  206.           else
  207.             Y2RefGlb := Y2RefGlb + HeaderSizeGlb;
  208.       if From = 2 then
  209.         FromBase := Seg(ScreenGlb^)
  210.       else
  211.         FromBase := HardwareGrafBase;
  212.       if Tu = 2 then
  213.         ToBase := Seg(ScreenGlb^)
  214.       else
  215.         ToBase := HardwareGrafBase;
  216.       XLen := X2RefGlb - X1RefGlb;
  217.       YLen := Y2RefGlb - Y1RefGlb;
  218.       if X1 + XLen > XMaxGlb then
  219.         XLen := XMaxGlb - X1;
  220.       if Y1 + YLen > YMaxGlb then
  221.         YLen := YMaxGlb - Y1;
  222.       XLen := XLen + 1;
  223.       for I := 0 to YLen do
  224.         Move(Mem[FromBase:BaseAddress(Y1RefGlb + I) + X1RefGlb],
  225.              Mem[ToBase:BaseAddress(Y1 + I) + X1], XLen);
  226.       with GrafWindow[WindowNdxGlb] do
  227.         if Drawn then
  228.           if Top then
  229.             Y1RefGlb := Y1RefGlb + HeaderSizeGlb
  230.           else
  231.             Y2RefGlb := Y2RefGlb - HeaderSizeGlb;
  232.     end;
  233. end; { CopyWindow }
  234.  
  235. procedure SaveWindow{(N : integer; FileName : WrkString)};
  236. type
  237.   Sector = array[0..127] of byte;
  238. var
  239.   I, J, SecPtr, Xlen : integer;
  240.   W : WindowType;
  241.   PictureFile : file of Sector;
  242.   Sec1 : array[0..1] of Sector;
  243.  
  244. begin
  245.   if not (N in [1..MaxWindowGlb]) then
  246.     Error(25, 2)
  247.   else if (FileName = '') then
  248.     Error(25, 5)
  249.   else
  250.     begin
  251.       W := GrafWindow[N];
  252.       Assign(PictureFile, FileName);
  253.       {$I-} Rewrite(PictureFile); {$I+}
  254.       if IOresult <> 0 then
  255.         Error(25, 5)
  256.       else
  257.         begin
  258.           Move(W, Sec1, SizeOf(W));
  259.           SecPtr := SizeOf(W);
  260.           with W do
  261.           begin
  262.             if Drawn then
  263.               if Top then
  264.                 Y1 := Y1 - HeaderSizeGlb
  265.               else
  266.                 Y2 := Y2 + HeaderSizeGlb;
  267.             Xlen := X2 - X1 + 1;
  268.             for I := Y1 to Y2 do
  269.             begin
  270.               Move(Mem[GrafBase:BaseAddress(I) + X1], Sec1[0, SecPtr], Xlen);
  271.               SecPtr := SecPtr + Xlen;
  272.               if SecPtr > 127 then
  273.               begin
  274.                 Write(PictureFile, Sec1[0]);
  275.                 Move(Sec1[1], Sec1[0], 128);
  276.                 SecPtr := SecPtr - 128;
  277.               end;
  278.             end;
  279.             if SecPtr <> 0 then
  280.               Write(PictureFile, Sec1[0]);
  281.           end;
  282.           Close(PictureFile);
  283.         end;
  284.     end;
  285. end; { SaveWindow }
  286.  
  287. procedure LoadWindow{(N, Xpos, Ypos : integer; FileName : WrkString)};
  288. type
  289.   Sector = array[0..127] of byte;
  290. var
  291.   I, SecPtr, Xlen : integer;
  292.   W : WindowType;
  293.   PictureFile : file of Sector;
  294.   Sec1 : array[0..1] of Sector;
  295.   Second : boolean;
  296.  
  297. begin
  298.   Assign(PictureFile, FileName);
  299.   {$I-} Reset(PictureFile); {$I+}
  300.   if (IOresult <> 0) or (FileName = '') then
  301.     Error(12, 5)
  302.   else
  303.     begin
  304.       Read(PictureFile, Sec1[0]);
  305.       Move(Sec1, W, SizeOf(W));
  306.       SecPtr := SizeOf(W);
  307.       Second := false;
  308.       GrafWindow[N] := W;
  309.       with W do
  310.       begin
  311.         if Drawn then
  312.           if Top then
  313.             Y1 := Y1 - HeaderSizeGlb
  314.           else
  315.             Y2 := Y2 + HeaderSizeGlb;
  316.         Xlen := X2 - X1 + 1;
  317.         if Xpos >= 0 then
  318.         begin
  319.           X2 := Xpos + X2 - X1;
  320.           X1 := Xpos;
  321.         end;
  322.         if Ypos >= 0 then
  323.         begin
  324.           Y2 := Ypos + Y2 - Y1;
  325.           Y1 := Ypos;
  326.         end;
  327.         if (X1 < 0) or (Y1 < 0) or (X2 > XMaxGlb) or (Y2 > YMaxGlb) then
  328.           Error(12, 3)
  329.         else
  330.           begin
  331.             for I := Y1 to Y2 do
  332.             begin
  333.               if (SecPtr + Xlen > 127) and
  334.                   not Second and not EOF(PictureFile) then
  335.               begin
  336.                 Read(PictureFile, Sec1[1]);
  337.                 Second := true;
  338.               end;
  339.               Move(Sec1[0, SecPtr], Mem[GrafBase:BaseAddress(I) + X1], Xlen);
  340.               SecPtr := SecPtr + Xlen;
  341.               if SecPtr > 127 then
  342.               begin
  343.                 Move(Sec1[1], Sec1[0], 128);
  344.                 SecPtr := SecPtr - 128;
  345.                 Second := false;
  346.               end;
  347.             end;
  348.           end;
  349.       end;
  350.       Close(PictureFile);
  351.     end;
  352. end; { LoadWindow }
  353.  
  354. function WindowSize{(Win : integer) : word};
  355. var
  356.   Ws : integer;
  357. begin
  358.   Ws := 0;
  359.   if not (Win in [1..MaxWindowsGlb]) then
  360.     Error(13, 2)
  361.   else
  362.     with GrafWindow[Win] do
  363.     begin
  364.       Ws := (Y2 - Y1 + 1) * (X2 - X1 + 1);
  365.       if Drawn then
  366.         Ws := Ws + HeaderSizeGlb * (X2 - X1 + 1);
  367.     end;
  368.   WindowSize := Ws;
  369. end; { WindowSize }
  370.  
  371. procedure ClearWindowStack{(Win : integer)};
  372. begin
  373.   if not (Win in [1..MaxWindowsGlb]) then
  374.     Error(14, 2)
  375.   else
  376.     with Stack[Win], W do
  377.     begin
  378.       if (Contents <> nil) then
  379.         FreeMem(Contents, Size);
  380.       Contents := nil;
  381.       Size := 0;
  382.     end;
  383. end; { ClearWindowStack }
  384.  
  385. procedure StoreWindow{(Win : integer)};
  386. var
  387.   XLen, Y0, Y9, A : integer;
  388.   I, Y : word;
  389.  
  390. begin
  391.   if not (Win in [1..MaxWindowsGlb]) then
  392.     Error(15, 2)
  393.   else
  394.     begin
  395.       if Stack[Win].Contents <> nil then
  396.         ClearWindowStack(Win);
  397.       if WindowSize(Win) > MaxAvail then
  398.         Error(15, 6)
  399.       else
  400.         with Stack[Win], W do
  401.         begin
  402.           W := GrafWindow[Win];
  403.           Size := WindowSize(Win);
  404.           GetMem(Contents, Size);
  405.           with W do
  406.           begin
  407.             Y0 := Y1;
  408.             Y9 := Y2;
  409.             if Drawn then
  410.               if Top then
  411.                 Y0 := Y0 - HeaderSizeGlb
  412.               else
  413.                 Y9 := Y9 + HeaderSizeGlb;
  414.             XLen := X2 - X1 + 1;
  415.             A := 0;
  416.             for I := Y0 to Y9 do
  417.             begin
  418.               Y := BaseAddress(I);
  419.               Move(Mem[GrafBase:Y + X1],
  420.                    Mem[Seg(Contents^):Ofs(Contents^) + A], XLen);
  421.               A := A + XLen;
  422.             end;
  423.           end;
  424.         end;
  425.     end;
  426. end; { StoreWindow }
  427.  
  428. procedure RestoreWindow{(Win, DeltaX, DeltaY : integer)};
  429. var
  430.   I, XLen, Y, Y0, Y9, A : integer;
  431.   W1 : WindowType;
  432. begin
  433.   if not (abs(Win) in [1..MaxWindowsGlb]) then
  434.     Error(16, 2)
  435.   else
  436.     with Stack[abs(Win)] do
  437.     begin
  438.       W1 := W;
  439.       if Contents = nil then
  440.         Error(16, 2)
  441.       else
  442.         with W1 do
  443.         begin
  444.           X1 := X1 + DeltaX;
  445.           X2 := X2 + DeltaX;
  446.           Y1 := Y1 + DeltaY;
  447.           Y2 := Y2 + DeltaY;
  448.           if (X1 >= 0) and (X1 <= XMaxGlb) and (X2 >= 0) and (X2 <= XMaxGlb) and
  449.              (Y1 >= 0) and (Y1 <= YMaxGlb) and (Y2 >= 0) and (Y2 <= YMaxGlb) then
  450.             begin
  451.               XLen := X2 - X1 + 1;
  452.               A := 0;
  453.               Y0 := Y1;
  454.               Y9 := Y2;
  455.               if Drawn then
  456.                 if Top then
  457.                   Y0 := Y0 - HeaderSizeGlb
  458.                 else
  459.                   Y9 := Y9 + HeaderSizeGlb;
  460.               for I := Y0 to Y9 do
  461.               begin
  462.                 Y := BaseAddress(I);
  463.                 with Stack[Abs(Win)] do
  464.                   Move(Mem[Seg(Contents^):Ofs(Contents^) + A],
  465.                        Mem[GrafBase:Y + X1], XLen);
  466.                 A := A + XLen;
  467.               end;
  468.               GrafWindow[Abs(Win)] := W1;
  469.               if Win < 0 then
  470.                 ClearWindowStack(abs(Win));
  471.               if Win = WindowNdxGlb then
  472.                 SelectWindow(Win);
  473.             end
  474.           else
  475.             Error(16, 3);
  476.         end;
  477.     end;
  478. end; { RestoreWindow }
  479.  
  480. procedure SaveWindowStack{(FileName : WrkString)};
  481. var
  482.   WindowFile : file;
  483.   PointerFile : file of WindowType;
  484.   I, NumWritten : word;
  485. begin
  486.   Assign(WindowFile, FileName + '.stk');
  487.   {$I-} Rewrite(WindowFile); {$I+}
  488.   if IOresult <> 0 then
  489.     Error(26, 5)
  490.   else
  491.     begin
  492.       for I := 1 to MaxWindowsGlb do
  493.       with Stack[I], W do
  494.         if Contents <> nil then
  495.           BlockWrite(WindowFile, Contents^, Size shr 7, NumWritten);
  496.       Close(WindowFile);
  497.       Assign(PointerFile, FileName + '.Ptr');
  498.       {$I-} Rewrite(PointerFile); {$I+}
  499.       if IOresult <> 0 then
  500.         Error(26, 5)
  501.       else
  502.         begin
  503.           for I := 1 to MaxWindowsGlb do
  504.             Write(PointerFile, Stack[I].W);
  505.           Close(PointerFile);
  506.         end;
  507.     end;
  508. end; { SaveWindowStack }
  509.  
  510. procedure LoadWindowStack{(FileName : WrkString)};
  511. var
  512.   WindowFile : file;
  513.   PointerFile : file of WindowType;
  514.   I, NumRead : word;
  515.  
  516. begin
  517.   Assign(PointerFile, FileName + '.Ptr');
  518.   {$I-} Reset(PointerFile); {$I+}
  519.   if IOresult = 0 then
  520.     begin
  521.       for I := 1 to MaxWindowsGlb do
  522.         Read(PointerFile, Stack[I].W);
  523.       Close(PointerFile);
  524.       Assign(WindowFile, FileName + '.stk');
  525.       {$I-} Reset(WindowFile); {$I+}
  526.       if IOresult = 0 then
  527.         begin
  528.           for I := 1 to MaxWindowsGlb do
  529.           with Stack[I], W do
  530.             if Size <> 0 then
  531.               begin
  532.                 GetMem(Contents, Size);
  533.                 BlockRead(WindowFile, Contents^, Size shr 7, NumRead);
  534.               end
  535.             else
  536.               Contents := nil;
  537.           Close(WindowFile);
  538.         end
  539.       else
  540.         Error(21, 5);
  541.     end
  542.   else
  543.     Error(21, 5);
  544. end; { LoadWindowStack }
  545.  
  546. procedure ResetWindowStack;
  547. var
  548.   I : word;
  549. begin
  550.   for I := 1 to MaxWindowsGlb do
  551.     ClearWindowStack(I);
  552. end; { ResetWindowStack }
  553.  
  554. procedure InvertWindow;
  555. var
  556.   I, B : word;
  557. begin
  558.   with GrafWindow[WindowNdxGlb] do
  559.     if Drawn then
  560.       if Top then
  561.         Y1RefGlb := Y1RefGlb - HeaderSizeGlb
  562.       else
  563.         Y2RefGlb := Y2RefGlb + HeaderSizeGlb;
  564.    for I := Y1RefGlb to Y2RefGlb do
  565.    begin
  566.      B := BaseAddress(I);
  567.      inline($8B/$86/B/        {          MOV   AX,[B]         }
  568.             $8B/$1E/X1RefGlb/ {          MOV   BX,[X1RefGlb]  }
  569.             $8B/$0E/X2RefGlb/ {          MOV   CX,[X2RefGlb]  }
  570.             $8B/$16/GrafBase/ {          MOV   DX,[GrafBase]  }
  571.             $1E/              {          PUSH  DS             }
  572.             $8E/$DA/          {          MOV   DS,DX          }
  573.             $29/$D9/          {          SUB   CX,BX          }
  574.             $41/              {          INC   CX             }
  575.             $01/$C3/          {          ADD   BX,AX          }
  576.             $F6/$17/          { InvByte: NOT   BYTE PTR [BX]  }
  577.             $43/              {          INC   BX             }
  578.             $E2/$FB/          {          LOOP  InvByte        }
  579.             $1F);             {          POP   DS             }
  580.    end;
  581.    with GrafWindow[WindowNdxGlb] do
  582.      if Drawn then
  583.        if Top then
  584.          Y1RefGlb := Y1RefGlb + HeaderSizeGlb
  585.        else
  586.          Y2RefGlb := Y2RefGlb - HeaderSizeGlb;
  587. end; { InvertWindow }
  588.  
  589. end. { GWindow }
  590.