/Math.Compiler/Classes/Variable.cs

# · C# · 414 lines · 366 code · 41 blank · 7 comment · 48 complexity · 2c38530c64b9714826a815fb5f91663b MD5 · raw file

  1. //-----------------------------------------------------------------------
  2. // <copyright>
  3. // Copyright (c) Artur Mustafin. All rights reserved.
  4. // </copyright>
  5. //-----------------------------------------------------------------------
  6. using System.Collections.Generic;
  7. using System.Globalization;
  8. using System.Text;
  9. using Math.Compiler;
  10. using System;
  11. using Math.Compiler.Expressions;
  12. using System.Linq.Expressions;
  13. namespace Math.Compiler.Classes
  14. {
  15. public class Variable
  16. {
  17. #if DEBUG
  18. private string _inputReferencesText;
  19. private string _outputReferencesText;
  20. #endif
  21. private int _id;
  22. private static Dictionary<string, Variable> _variables;
  23. static Variable()
  24. {
  25. _variables = new Dictionary<string, Variable>();
  26. }
  27. private bool _visited = false;
  28. private double _value;
  29. private MathExpression _expression;
  30. private readonly string _name;
  31. private Variable(int id, string name)
  32. {
  33. _id = id;
  34. _name = name;
  35. _inputReferences = new HashSet<Variable>();
  36. _outputReferences = new HashSet<Variable>();
  37. }
  38. public int Id
  39. {
  40. get
  41. {
  42. return _id;
  43. }
  44. }
  45. private static int _count = 0;
  46. public static Variable Load(string name)
  47. {
  48. if (!_variables.ContainsKey(name))
  49. {
  50. Variable variable = new Variable(_count++, name);
  51. _variables.Add(name, variable);
  52. return variable;
  53. }
  54. return _variables[name];
  55. }
  56. public bool HasExpression
  57. {
  58. get
  59. {
  60. return _expression != null;
  61. }
  62. }
  63. public void Eval()
  64. {
  65. if (_visited == false)
  66. {
  67. _visited = true;
  68. if (_expression != null)
  69. {
  70. //Func<double> f = _expression;
  71. //_value = f();
  72. _value = _expression.Eval();
  73. }
  74. else
  75. {
  76. _value = 0;
  77. }
  78. if (_outputReferences.Count > 0)
  79. {
  80. foreach (Variable cell in _outputReferences)
  81. {
  82. cell.Eval();
  83. }
  84. }
  85. }
  86. }
  87. public void Visit()
  88. {
  89. if (_visited == true)
  90. {
  91. _visited = false;
  92. if (_inputReferences.Count > 0)
  93. {
  94. foreach (var cell in _inputReferences)
  95. {
  96. cell.Visit();
  97. }
  98. }
  99. Eval();
  100. }
  101. }
  102. public double Double
  103. {
  104. get
  105. {
  106. return _value;
  107. }
  108. }
  109. public MathExpression MathExpression
  110. {
  111. get
  112. {
  113. return _expression;
  114. }
  115. set
  116. {
  117. if (_expression != value)
  118. {
  119. _expression = value;
  120. _visited = false;
  121. Eval();
  122. }
  123. }
  124. }
  125. public string Value
  126. {
  127. get
  128. {
  129. if (_expression != null)
  130. {
  131. return _value.ToString(CultureInfo.InvariantCulture.NumberFormat);
  132. }
  133. return string.Empty;
  134. }
  135. }
  136. private string _text;
  137. public string Text
  138. {
  139. get
  140. {
  141. return _text;
  142. }
  143. set
  144. {
  145. _text = value;
  146. }
  147. }
  148. private readonly HashSet<Variable> _outputReferences;
  149. private readonly HashSet<Variable> _inputReferences;
  150. private void Subscribe(Variable cell)
  151. {
  152. cell.SubscribeInput(this);
  153. SubscribeOutput(cell);
  154. }
  155. private void Unsubscribe(Variable cell)
  156. {
  157. cell.UnsubscribeInput(this);
  158. UnsubscribeOutput(cell);
  159. }
  160. private void SubscribeInput(Variable cell)
  161. {
  162. #if DEBUG
  163. _inputReferencesText = null;
  164. #endif
  165. _inputReferences.Add(cell);
  166. }
  167. private void UnsubscribeInput(Variable cell)
  168. {
  169. #if DEBUG
  170. _inputReferencesText = null;
  171. #endif
  172. _inputReferences.Remove(cell);
  173. }
  174. private void SubscribeOutput(Variable cell)
  175. {
  176. #if DEBUG
  177. _outputReferencesText = null;
  178. #endif
  179. _outputReferences.Add(cell);
  180. }
  181. private void UnsubscribeOutput(Variable cell)
  182. {
  183. #if DEBUG
  184. _outputReferencesText = null;
  185. #endif
  186. _outputReferences.Remove(cell);
  187. }
  188. public void Clear()
  189. {
  190. #if DEBUG
  191. _inputReferencesText = null;
  192. _outputReferencesText = null;
  193. #endif
  194. _inputReferences.Clear();
  195. _outputReferences.Clear();
  196. _expression = null;
  197. _text = null;
  198. _value = 0;
  199. }
  200. public void Subscribe()
  201. {
  202. HashSet<Variable> outputReferences = new HashSet<Variable>();
  203. Decompile(outputReferences, MathExpression);
  204. foreach (Variable cell in outputReferences)
  205. {
  206. cell.SubscribeInput(this);
  207. this.SubscribeOutput(cell);
  208. }
  209. }
  210. public void Unsubscribe()
  211. {
  212. HashSet<Variable> outputReferences = new HashSet<Variable>();
  213. Decompile(outputReferences, MathExpression);
  214. foreach (Variable cell in outputReferences)
  215. {
  216. cell.UnsubscribeInput(this);
  217. this.UnsubscribeOutput(cell);
  218. }
  219. }
  220. #if DEBUG
  221. public override string ToString()
  222. {
  223. if (_inputReferencesText == null)
  224. {
  225. _inputReferencesText = GetReferences(_inputReferences, _inputReferences.Count);
  226. }
  227. if (_outputReferencesText == null)
  228. {
  229. _outputReferencesText = GetReferences(_outputReferences, _outputReferences.Count);
  230. }
  231. bool inputIsNull = string.IsNullOrEmpty(_inputReferencesText);
  232. bool outputIsNull = string.IsNullOrEmpty(_outputReferencesText);
  233. if (!inputIsNull && !outputIsNull)
  234. {
  235. return string.Format("{0}->{1}->{2}", _outputReferencesText, _name, _inputReferencesText);
  236. }
  237. if (!inputIsNull)
  238. {
  239. return string.Format("{0}->{1}", _name, _inputReferencesText);
  240. }
  241. if (!outputIsNull)
  242. {
  243. return string.Format("{0}->{1}", _outputReferencesText, _name);
  244. }
  245. return _name;
  246. }
  247. #else
  248. public override string ToString()
  249. {
  250. return _name;
  251. }
  252. #endif
  253. private string GetReferences(IEnumerable<Variable> cells, int count)
  254. {
  255. IEnumerator<Variable> enumerator = cells.GetEnumerator();
  256. StringBuilder sb = new StringBuilder();
  257. if (enumerator.MoveNext())
  258. {
  259. Variable cell = enumerator.Current;
  260. sb.Append(cell.Name);
  261. int index = 0;
  262. while (enumerator.MoveNext())
  263. {
  264. if (index == 0 && index < count - 2)
  265. {
  266. sb.Append(",...");
  267. index++;
  268. continue;
  269. }
  270. if (index > 0 && index < count - 2)
  271. {
  272. index++;
  273. continue;
  274. }
  275. cell = enumerator.Current;
  276. sb.Append(',');
  277. sb.Append(cell.Name);
  278. index++;
  279. }
  280. }
  281. return sb.ToString();
  282. }
  283. public string Name
  284. {
  285. get
  286. {
  287. return _name;
  288. }
  289. }
  290. public bool Validate(MathExpression expression)
  291. {
  292. foreach (Variable cell in Decompile(expression))
  293. {
  294. if (cell == this)
  295. {
  296. return false;
  297. }
  298. }
  299. return true;
  300. }
  301. private HashSet<Variable> Decompile(MathExpression expression)
  302. {
  303. HashSet<Variable> cells = new HashSet<Variable>();
  304. Decompile(cells, expression);
  305. return cells;
  306. }
  307. public void Decompile(HashSet<Variable> cells, MathExpression expression)
  308. {
  309. if (expression != null)
  310. {
  311. Stack<MathExpression> expressions = new Stack<MathExpression>();
  312. expressions.Push(expression);
  313. do
  314. {
  315. MathExpression current = expressions.Pop();
  316. if (current.Children != null)
  317. {
  318. foreach (MathExpression node in current.Children)
  319. {
  320. VariableReference reference = node as VariableReference;
  321. if (reference != null)
  322. {
  323. if (cells.Add(reference.Variable))
  324. {
  325. if (reference.Variable.MathExpression != null)
  326. {
  327. expressions.Push(reference.Variable.MathExpression);
  328. }
  329. }
  330. }
  331. else
  332. {
  333. if (node != null)
  334. {
  335. expressions.Push(node);
  336. }
  337. }
  338. }
  339. }
  340. VariableReference referenced = current as VariableReference;
  341. if (referenced != null)
  342. {
  343. if (cells.Add(referenced.Variable))
  344. {
  345. if (referenced.Variable.MathExpression != null)
  346. {
  347. expressions.Push(referenced.Variable.MathExpression);
  348. }
  349. }
  350. }
  351. }
  352. while (expressions.Count > 0);
  353. }
  354. }
  355. private string _updateText;
  356. private MathExpression _updateExpression;
  357. public void BeginUpdate()
  358. {
  359. _updateText = _text;
  360. _updateExpression = _expression;
  361. }
  362. public void EndUpdate()
  363. {
  364. _updateText = null;
  365. _updateExpression = null;
  366. }
  367. public void CancelUpdate()
  368. {
  369. _text = _updateText;
  370. _expression = _updateExpression;
  371. }
  372. }
  373. }