home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 9 / 09.iso / l / l043 / 3.ddi / MCLIB.PAS < prev    next >
Encoding:
Pascal/Delphi Source File  |  1987-11-02  |  12.2 KB  |  503 lines

  1.  
  2. {           Copyright (c) 1985, 87 by Borland International, Inc.            }
  3.  
  4. unit MCLIB;
  5.  
  6. interface
  7.  
  8. uses Crt, Dos, MCVars, MCUtil, MCDisply, MCParser;
  9.  
  10. procedure DisplayCell(Col, Row : Word; Highlighting, Updating : Boolean);
  11. { Displays the contents of a cell }
  12.  
  13. function SetOFlags(Col, Row : Word; Display : Boolean) : Word;
  14. { Sets the overwrite flag on cells starting at (col + 1, row) - returns
  15.    the number of the column after the last column set.
  16. }
  17.  
  18. procedure ClearOFlags(Col, Row : Word; Display : Boolean);
  19. { Clears the overwrite flag on cells starting at (col, row) }
  20.  
  21. procedure UpdateOFlags(Col, Row : Word; Display : Boolean);
  22. { Starting in col, moves back to the last TEXT cell and updates all flags }
  23.  
  24. procedure DeleteCell(Col, Row : Word; Display : Boolean);
  25. { Deletes a cell }
  26.  
  27. procedure SetLeftCol;
  28. { Sets the value of LeftCol based on the value of RightCol }
  29.  
  30. procedure SetRightCol;
  31. { Sets the value of rightcol based on the value of leftcol }
  32.  
  33. procedure SetTopRow;
  34. { Figures out the value of toprow based on the value of bottomrow }
  35.  
  36. procedure SetBottomRow;
  37. { Figures out the value of bottomrow based on the value of toprow }
  38.  
  39. procedure SetLastCol;
  40. { Sets the value of lastcol based on the current value }
  41.  
  42. procedure SetLastRow;
  43. { Sets the value of lastrow based on the current value }
  44.  
  45. procedure ClearLastCol;
  46. { Clears any data left in the last column }
  47.  
  48. procedure DisplayCol(Col : Word; Updating : Boolean);
  49. { Displays a column on the screen }
  50.  
  51. procedure DisplayRow(Row : Word; Updating : Boolean);
  52. { Displays a row on the screen }
  53.  
  54. procedure DisplayScreen(Updating : Boolean);
  55. { Displays the current screen of the spreadsheet }
  56.  
  57. procedure RedrawScreen;
  58. { Displays the entire screen }
  59.  
  60. procedure FixFormula(Col, Row, Action, Place : Word);
  61. { Modifies a formula when its column or row designations need to change }
  62.  
  63. procedure ChangeAutoCalc(NewMode : Boolean);
  64. { Changes and prints the current AutoCalc value on the screen }
  65.  
  66. procedure ChangeFormDisplay(NewMode : Boolean);
  67. { Changes and prints the current formula display value on the screen }
  68.  
  69. procedure Recalc;
  70. { Recalculates all of the numbers in the speadsheet }
  71.  
  72. procedure Act(S : String);
  73. { Acts on a particular input }
  74.  
  75. implementation
  76.  
  77. procedure DisplayCell;
  78. var
  79.   Color : Word;
  80.   S : IString;
  81. begin
  82.   if Updating and
  83.       ((Cell[Col, Row] = Nil) or (Cell[Col, Row]^.Attrib <> FORMULA)) then
  84.     Exit;
  85.   S := CellString(Col, Row, Color, DOFORMAT);
  86.   if Highlighting then
  87.   begin
  88.     if Color = ERRORCOLOR then
  89.       Color := HIGHLIGHTERRORCOLOR
  90.     else
  91.       Color := HIGHLIGHTCOLOR;
  92.   end;
  93.   SetColor(Color);
  94.   WriteXY(S, ColStart[Succ(Col - LeftCol)], Row - TopRow + 3);
  95. end; { DisplayCell }
  96.  
  97. function SetOFlags;
  98. var
  99.   Len : Integer;
  100. begin
  101.   Len := Length(Cell[Col, Row]^.T) - ColWidth[Col];
  102.   Inc(Col);
  103.   while (Col <= MAXCOLS) and (Len > 0) and (Cell[Col, Row] = nil) do
  104.   begin
  105.     Format[Col, Row] := Format[Col, Row] or OVERWRITE;
  106.     Dec(Len, ColWidth[Col]);
  107.     if Display and (Col >= LeftCol) and (Col <= RightCol) then
  108.       DisplayCell(Col, Row, NOHIGHLIGHT, NOUPDATE);
  109.     Inc(Col);
  110.   end;
  111.   SetOFlags := Col;
  112. end; { SetOFlags }
  113.  
  114. procedure ClearOFlags;
  115. begin
  116.   while (Col <= MAXCOLS) and (Format[Col, Row] >= OVERWRITE) and
  117.         (Cell[Col, Row] = nil) do
  118.   begin
  119.     Format[Col, Row] := Format[Col, Row] and (not OVERWRITE);
  120.     if Display and (Col >= LeftCol) and (Col <= RightCol) then
  121.       DisplayCell(Col, Row, NOHIGHLIGHT, NOUPDATE);
  122.     Inc(Col);
  123.   end;
  124. end; { ClearOFlags }
  125.  
  126. procedure UpdateOFlags;
  127. var
  128.   Dummy : Word;
  129. begin
  130.   while (Cell[Col, Row] = nil) and (Col > 1) do
  131.     Dec(Col);
  132.   if (Cell[Col, Row]^.Attrib = TXT) and (Col >= 1) then
  133.     Dummy := SetOFlags(Col, Row, Display);
  134. end; { UpdateOFlags }
  135.  
  136. procedure DeleteCell;
  137. var
  138.   CPtr : CellPtr;
  139.   Size : Word;
  140. begin
  141.   CPtr := Cell[Col, Row];
  142.   if CPtr = nil then
  143.     Exit;
  144.   case CPtr^.Attrib of
  145.     TXT : begin
  146.       Size := Length(CPtr^.T) + 3;
  147.       ClearOFlags(Succ(Col), Row, Display);
  148.     end;
  149.     VALUE : Size := SizeOf(Real) + 2;
  150.     FORMULA : Size := SizeOf(Real) + Length(CPtr^.Formula) + 3;
  151.   end; { case }
  152.   Format[Col, Row] := Format[Col, Row] and (not OVERWRITE);
  153.   FreeMem(CPtr, Size);
  154.   Cell[Col, Row] := nil;
  155.   if Col = LastCol then
  156.     SetLastCol;
  157.   if Row = LastRow then
  158.     SetLastRow;
  159.   UpdateOFlags(Col, Row, Display);
  160.   Changed := True;
  161. end; { DeleteCell }
  162.  
  163. procedure SetLeftCol;
  164. var
  165.   Col : Word;
  166.   Total : Integer;
  167. begin
  168.   Total := 81;
  169.   Col := 0;
  170.   while (Total > LEFTMARGIN) and (RightCol - Col > 0) do
  171.   begin
  172.     Dec(Total, ColWidth[RightCol - Col]);
  173.     if Total > LEFTMARGIN then
  174.       ColStart[SCREENCOLS - Col] := Total;
  175.     Inc(Col);
  176.   end;
  177.   if Total > LEFTMARGIN then
  178.     Inc(Col);
  179.   Move(ColStart[SCREENCOLS - Col + 2], ColStart, Pred(Col));
  180.   LeftCol := RightCol - Col + 2;
  181.   Total := Pred(ColStart[1] - LEFTMARGIN);
  182.   if Total <> 0 then
  183.   begin
  184.     for Col := LeftCol to RightCol do
  185.       Dec(ColStart[Succ(Col - LeftCol)], Total);
  186.   end;
  187.   PrintCol;
  188. end; { SetLeftCol }
  189.  
  190. procedure SetRightCol;
  191. var
  192.   Total, Col : Word;
  193. begin
  194.   Total := Succ(LEFTMARGIN);
  195.   Col := 1;
  196.   repeat
  197.   begin
  198.     ColStart[Col] := Total;
  199.     Inc(Total, ColWidth[Pred(LeftCol + Col)]);
  200.     Inc(Col);
  201.   end;
  202.   until (Total > 81) or (Pred(LeftCol + Col) > MAXCOLS);
  203.   if Total > 81 then
  204.     Dec(Col);
  205.   RightCol := LeftCol + Col - 2;
  206.   PrintCol;
  207. end; { SetRightCol }
  208.  
  209. procedure SetTopRow;
  210. begin
  211.   if BottomRow < ScreenRows then
  212.     BottomRow := ScreenRows;
  213.   TopRow := Succ(BottomRow - ScreenRows);
  214.   PrintRow;
  215. end; { SetTopRow }
  216.  
  217. procedure SetBottomRow;
  218. begin
  219.   if TopRow + ScreenRows > Succ(MAXROWS) then
  220.     TopRow := Succ(MAXROWS - ScreenRows);
  221.   BottomRow := Pred(TopRow + ScreenRows);
  222.   PrintRow;
  223. end; { SetBottomRow }
  224.  
  225. procedure SetLastCol;
  226. var
  227.   Row, Col : Word;
  228. begin
  229.   for Col := LastCol downto 1 do
  230.   begin
  231.     for Row := 1 to LastRow do
  232.     begin
  233.       if Cell[Col, Row] <> nil then
  234.       begin
  235.         LastCol := Col;
  236.         Exit;
  237.       end;
  238.     end;
  239.   end;
  240.   LastCol := 1;
  241. end; { SetLastCol }
  242.  
  243. procedure SetLastRow;
  244. var
  245.   Row, Col : Word;
  246. begin
  247.   for Row := LastRow downto 1 do
  248.   begin
  249.     for Col := 1 to LastCol do
  250.     begin
  251.       if Cell[Col, Row] <> nil then
  252.       begin
  253.         LastRow := Row;
  254.         Exit;
  255.       end;
  256.     end;
  257.   end;
  258.   LastRow := 1;
  259. end; { SetLastRow }
  260.  
  261. procedure ClearLastCol;
  262. var
  263.   Col : Word;
  264. begin
  265.   Col := ColStart[Succ(RightCol - LeftCol)] + ColWidth[RightCol];
  266.   if (Col < 80) then
  267.     Scroll(UP, 0, Col, 3, 80, ScreenRows + 2, White);
  268. end; { ClearLastCol }
  269.  
  270. procedure DisplayCol;
  271. var
  272.   Row : Word;
  273. begin
  274.   for Row := TopRow to BottomRow do
  275.     DisplayCell(Col, Row, NOHIGHLIGHT, Updating);
  276. end; { DisplayCol }
  277.  
  278. procedure DisplayRow;
  279. var
  280.   Col : Word;
  281. begin
  282.   for Col := LeftCol to RightCol do
  283.     DisplayCell(Col, Row, NOHIGHLIGHT, Updating);
  284. end; { DisplayRow }
  285.  
  286. procedure DisplayScreen;
  287. var
  288.   Row : Word;
  289. begin
  290.   for Row := TopRow to BottomRow do
  291.     DisplayRow(Row, Updating);
  292.   ClearLastCol;
  293. end; { DisplayScreen }
  294.  
  295. procedure RedrawScreen;
  296. begin
  297.   CurRow := 1;
  298.   CurCol := 1;
  299.   LeftCol := 1;
  300.   TopRow := 1;
  301.   SetRightCol;
  302.   SetBottomRow;
  303.   GotoXY(1, 1);
  304.   SetColor(MSGMEMORYCOLOR);
  305.   Write(MSGMEMORY);
  306.   GotoXY(29, 1);
  307.   SetColor(PROMPTCOLOR);
  308.   Write(MSGCOMMAND);
  309.   ChangeAutocalc(Autocalc);
  310.   ChangeFormDisplay(FormDisplay);
  311.   PrintFreeMem;
  312.   DisplayScreen(NOUPDATE);
  313. end; { RedrawScreen }
  314.  
  315. procedure FixFormula;
  316. var
  317.   FormLen, ColStart, RowStart, CurPos, FCol, FRow : Word;
  318.   CPtr : CellPtr;
  319.   Value : Real;
  320.   S : String[5];
  321.   NewFormula : IString;
  322.   Good : Boolean;
  323. begin
  324.   CPtr := Cell[Col, Row];
  325.   CurPos := 1;
  326.   NewFormula := CPtr^.Formula;
  327.   while CurPos < Length(NewFormula) do
  328.   begin
  329.     if FormulaStart(NewFormula, CurPos, FCol, FRow, FormLen) then
  330.     begin
  331.       if FCol > 26 then
  332.       begin
  333.         RowStart := CurPos + 2;
  334.         ColStart := RowStart - 2;
  335.       end
  336.       else begin
  337.         RowStart := Succ(CurPos);
  338.         ColStart := Pred(RowStart);
  339.       end;
  340.       case Action of
  341.         COLADD : begin
  342.           if FCol >= Place then
  343.           begin
  344.             if FCol = 26 then
  345.             begin
  346.               if Length(NewFormula) = MAXINPUT then
  347.               begin
  348.                 DeleteCell(Col, Row, NOUPDATE);
  349.                 Good := AllocText(Col, Row, NewFormula);
  350.                 Exit;
  351.               end;
  352.             end;
  353.             S := ColString(FCol);
  354.             Delete(NewFormula, ColStart, Length(S));
  355.             S := ColString(Succ(FCol));
  356.             Insert(S, NewFormula, ColStart);
  357.           end;
  358.         end;
  359.         ROWADD : begin
  360.           if FRow >= Place then
  361.           begin
  362.             if RowWidth(Succ(FRow)) <> RowWidth(FRow) then
  363.             begin
  364.               if Length(NewFormula) = MAXINPUT then
  365.               begin
  366.                 DeleteCell(Col, Row, NOUPDATE);
  367.                 Good := AllocText(Col, Row, NewFormula);
  368.                 Exit;
  369.               end;
  370.             end;
  371.             S := WordToString(FRow, 1);
  372.             Delete(NewFormula, RowStart, Length(S));
  373.             S := WordToString(Succ(FRow), 1);
  374.             Insert(S, NewFormula, RowStart);
  375.           end;
  376.         end;
  377.         COLDEL : begin
  378.           if FCol > Place then
  379.           begin
  380.             S := ColString(FCol);
  381.             Delete(NewFormula, ColStart, Length(S));
  382.             S := ColString(Pred(FCol));
  383.             Insert(S, NewFormula, ColStart);
  384.           end;
  385.         end;
  386.         ROWDEL : begin
  387.           if FRow > Place then
  388.           begin
  389.             S := WordToString(FRow, 1);
  390.             Delete(NewFormula, RowStart, Length(S));
  391.             S := WordToString(Pred(FRow), 1);
  392.             Insert(S, NewFormula, RowStart);
  393.           end;
  394.         end;
  395.       end; { case }
  396.       Inc(CurPos, FormLen);
  397.     end
  398.     else
  399.       Inc(CurPos);
  400.   end;
  401.   if Length(NewFormula) <> Length(CPtr^.Formula) then
  402.   begin
  403.     Value := CPtr^.FValue;
  404.     DeleteCell(Col, Row, NOUPDATE);
  405.     Good := AllocFormula(Col, Row, NewFormula, Value);
  406.   end
  407.   else
  408.     CPtr^.Formula := NewFormula;
  409. end; { FixFormula }
  410.  
  411. procedure ChangeAutoCalc;
  412. var
  413.   S : String[15];
  414. begin
  415.   if (not AutoCalc) and NewMode then
  416.     Recalc;
  417.   AutoCalc := NewMode;
  418.   if AutoCalc then
  419.     S := MSGAUTOCALC
  420.   else
  421.     S := '';
  422.   SetColor(MSGAUTOCALCCOLOR);
  423.   GotoXY(73, 1);
  424.   Write(S:Length(MSGAUTOCALC));
  425. end; { ChangeAutoCalc }
  426.  
  427. procedure ChangeFormDisplay;
  428. var
  429.   S : String[15];
  430. begin
  431.   FormDisplay := NewMode;
  432.   if FormDisplay then
  433.     S := MSGFORMDISPLAY
  434.   else
  435.     S := '';
  436.   SetColor(MSGFORMDISPLAYCOLOR);
  437.   GotoXY(65, 1);
  438.   Write(S:Length(MSGFORMDISPLAY));
  439. end; { ChangeFormDisplay }
  440.  
  441. procedure Recalc;
  442. var
  443.   Col, Row, Attrib : Word;
  444. begin
  445.   for Col := 1 to LastCol do
  446.   begin
  447.     for Row := 1 to LastRow do
  448.     begin
  449.       if ((Cell[Col, Row] <> nil) and (Cell[Col, Row]^.Attrib = FORMULA)) then
  450.       begin
  451.         Cell[Col, Row]^.FValue := Parse(Cell[Col, Row]^.Formula, Attrib);
  452.         Cell[Col, Row]^.Error := Attrib >= 4;
  453.       end;
  454.     end;
  455.   end;
  456.   DisplayScreen(UPDATE);
  457. end; { Recalc }
  458.  
  459. procedure Act;
  460. var
  461.   Attrib, Dummy : Word;
  462.   Allocated : Boolean;
  463.   V : Real;
  464. begin
  465.   DeleteCell(CurCol, CurRow, UPDATE);
  466.   V := Parse(S, Attrib);
  467.   case (Attrib and 3) of
  468.     TXT : begin
  469.       Allocated := AllocText(CurCol, CurRow, S);
  470.       if Allocated then
  471.         DisplayCell(CurCol, CurRow, NOHIGHLIGHT, NOUPDATE);
  472.     end;
  473.     VALUE : Allocated := AllocValue(CurCol, CurRow, V);
  474.     FORMULA : Allocated := AllocFormula(CurCol, CurRow, UpperCase(S), V);
  475.   end; { case }
  476.   if Allocated then
  477.   begin
  478.     if Attrib >= 4 then
  479.     begin
  480.       Cell[CurCol, CurRow]^.Error := True;
  481.       Dec(Attrib, 4);
  482.     end
  483.     else
  484.       Cell[CurCol, CurRow]^.Error := False;
  485.     Format[CurCol, CurRow] := Format[CurCol, CurRow] and (not OVERWRITE);
  486.     ClearOFlags(Succ(CurCol), CurRow, UPDATE);
  487.     if Attrib = TXT then
  488.       Dummy := SetOFlags(CurCol, CurRow, UPDATE);
  489.     if CurCol > LastCol then
  490.       LastCol := CurCol;
  491.     if CurRow > LastRow then
  492.       LastRow := CurRow;
  493.     if AutoCalc then
  494.       Recalc;
  495.   end
  496.   else
  497.     ErrorMsg(MSGLOMEM);
  498.   PrintFreeMem;
  499. end; { Act }
  500.  
  501. begin
  502. end.
  503.