Formula.cs 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using Biff8Excel.Interfaces;
  5. using System.Runtime.InteropServices;
  6. using Biff8Excel.Excel;
  7. namespace Biff8Excel.Formulas
  8. {
  9. [StructLayoutAttribute(LayoutKind.Sequential, CharSet = CharSet.Auto, Pack = 1)]
  10. struct sFormula
  11. {
  12. public ushort opcode;
  13. public ushort length;
  14. public ushort row;
  15. public ushort col;
  16. public ushort xf;
  17. public double resul;
  18. public ushort flags;
  19. public uint notused;
  20. }
  21. public class Formula : IRecords
  22. {
  23. string m_Formula = "";
  24. IExternSheet m_externSheet;
  25. ushort m_thisSheetNumber;
  26. const int formulaLength = 24;
  27. sFormula formula;
  28. public Formula()
  29. {
  30. formula.opcode = 0x6;
  31. formula.notused = 0x0;
  32. //formula.flags = 0x1;
  33. }
  34. private byte[] SerializableTokens()
  35. {
  36. ReversePolishNotation rpn;
  37. List<string> tokens;
  38. Referencer reference;
  39. string token;
  40. byte[] b = new byte[1024 * 8]; // create a buffer 8k
  41. byte[] btmp;
  42. ushort offset = 2; // leave roon for the length
  43. string name;
  44. int id;
  45. double db;
  46. // parse the supplied formula
  47. tokens = Globals.ParseFormula(m_Formula);
  48. // Convert the tokens to reverse polish notation
  49. rpn = new ReversePolishNotation();
  50. tokens = rpn.ConverToReversePolishNotation(tokens);
  51. for (int i = 0; i < tokens.Count; i++)
  52. {
  53. token = tokens[i];
  54. // NUMERIC VALUE, Is it a integer or float
  55. if (double.TryParse(token, out db))
  56. {
  57. if (token.IndexOf(".") > -1) // Integer or float
  58. {
  59. b[offset] = Globals.tokenINT;
  60. offset += 1; // integer identifier
  61. BitConverter.GetBytes(ushort.Parse(token)).CopyTo(b, offset);
  62. offset += 2; // Value
  63. }
  64. else
  65. {
  66. b[offset] = Globals.tokenNUM;
  67. offset += 1; // float indentifier
  68. BitConverter.GetBytes(db).CopyTo(b, offset);
  69. offset += 8; // Value
  70. }
  71. }
  72. // FUNCTION VALUE,
  73. else if (Globals.g_sSupported.IndexOf(Globals.GetFunctionName(token)) > -1)
  74. {
  75. name = Globals.GetFunctionName(tokens[i]);
  76. switch (name)
  77. {
  78. case "SUM": id = Globals.IDENTIFIER_SUM; break;
  79. case "AVERAGE": id = Globals.IDENTIFIER_AVERAGE; break;
  80. case "COUNT": id = Globals.IDENTIFIER_COUNT; break;
  81. default:
  82. id = Globals.IDENTIFIER_AVERAGE ;
  83. break;
  84. }
  85. reference = new Referencer();
  86. reference.ExternSheet = m_externSheet;
  87. btmp = reference.Serialize(token);
  88. reference = null;
  89. //btmp = TArray<byte>.Preserve(btmp, (int)(btmp.Length + 4));
  90. Array.Resize<byte>(ref btmp, btmp.Length + 4);
  91. BitConverter.GetBytes(id).CopyTo(btmp, btmp.Length - 4);
  92. btmp.CopyTo(b, offset);
  93. offset += (ushort)(btmp.Length);
  94. }
  95. // CELL REFERENCE
  96. else if (token.Length > 1)
  97. {
  98. reference = new Referencer();
  99. reference.ExternSheet = m_externSheet;
  100. btmp = reference.Serialize(token);
  101. reference = null;
  102. // add the sum identifier to the end of array
  103. btmp.CopyTo(b, offset);
  104. offset += (ushort)(btmp.Length);
  105. }
  106. //OPERATOR
  107. else
  108. {
  109. switch (token)
  110. {
  111. case "+": b[offset] = Globals.tokenADD; offset++; break;
  112. case "-": b[offset] = Globals.tokenSUBTRACT; offset++; break;
  113. case "*": b[offset] = Globals.tokenMULTIPLY; offset++; break;
  114. case "/": b[offset] = Globals.tokenDIVIDE; offset++; break;
  115. case "^": b[offset] = Globals.tokenPOWER; offset++; break;
  116. default:
  117. break;
  118. }
  119. }
  120. } // end for
  121. byte[] tmp = new byte[offset];
  122. Array.Copy(b, tmp, offset);
  123. offset = (ushort)(tmp.Length - 2);
  124. // save the tokens length
  125. BitConverter.GetBytes(offset).CopyTo(tmp,0);
  126. return tmp;
  127. }
  128. public double CalculateCell(ExcelCellDictionary sheetCells)
  129. //public double CalculateCell(ExcelCellLink sheetCells)
  130. {
  131. ReversePolishNotation rpn = new ReversePolishNotation();
  132. Calculator oCalc = new Calculator();
  133. List<string> tokens = new List<string>();
  134. // parset the supplied formula
  135. tokens = Globals.ParseFormula(m_Formula);
  136. // Convert the tokens to reverse polis notation
  137. tokens = rpn.ConverToReversePolishNotation(tokens);
  138. // calculate the formula expression and return the result
  139. // returns empty if the formula could not be resloved
  140. oCalc.CellCollection = sheetCells;
  141. oCalc.ExternSheet = m_externSheet;
  142. oCalc.ThisSheetNumber = m_thisSheetNumber;
  143. return oCalc.Calculate(tokens);
  144. }
  145. // If there are no sheets in the function string, use
  146. // this value to make sure we only reference cells from the formulas sheet
  147. public ushort ThisSheetNumber
  148. {
  149. set { m_thisSheetNumber = value; }
  150. }
  151. public string[] SheetNames
  152. {
  153. set { Globals.g_arrSheetNames = value; }
  154. }
  155. public string FormulaString
  156. {
  157. set { m_Formula = value; }
  158. }
  159. public ushort Row
  160. {
  161. set { formula.row = value; }
  162. }
  163. public ushort Column
  164. {
  165. set { formula.col = value; }
  166. }
  167. public ushort CellStyle
  168. {
  169. set { formula.xf = value; }
  170. }
  171. public double Result
  172. {
  173. set { formula.resul = value; }
  174. }
  175. public ushort Flags
  176. {
  177. set { formula.flags = value; }
  178. }
  179. public IExternSheet ExternSheet
  180. {
  181. set { m_externSheet = value; }
  182. }
  183. #region IRecords ³ÉÔ±
  184. public byte[] GetByte()
  185. {
  186. byte[] tmp = SerializableTokens();
  187. formula.length = (ushort)(tmp.Length + formulaLength - 4);
  188. byte[] result = new byte[tmp.Length + formulaLength];
  189. Globals.GetStructToBytes(formula).CopyTo(result, 0);
  190. tmp.CopyTo(result, 24);
  191. return result;
  192. }
  193. #endregion
  194. }
  195. }