RealtimeProvider.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.ObjectModel;
  4. using System.Data;
  5. using System.Diagnostics;
  6. using System.Linq;
  7. using Aitex.Core.RT.Log;
  8. using Aitex.Core.RT.SCCore;
  9. using DocumentFormat.OpenXml.Drawing;
  10. using ExtendedGrid.Microsoft.Windows.Controls;
  11. using MECF.Framework.Common.DataCenter;
  12. using MECF.Framework.Common.Device;
  13. using MECF.Framework.Common.Equipment;
  14. using MECF.Framework.Common.Utilities;
  15. using MECF.Framework.UI.Client.CenterViews.DataLogs.ProcessHistory;
  16. using MECF.Framework.UI.Client.ClientBase.Tree;
  17. using OpenSEMI.ClientBase.ServiceProvider;
  18. using SciChart.Core.Extensions;
  19. namespace MECF.Framework.UI.Client.CenterViews.Operations.RealTime
  20. {
  21. public class RealtimeProvider : IProvider
  22. {
  23. ObservableCollection<ParameterNode> _rootNode = new ObservableCollection<ParameterNode>();
  24. Dictionary<string, ParameterNode> _indexer = new Dictionary<string, ParameterNode>();
  25. public void Clear()
  26. {
  27. _indexer.Clear();
  28. _rootNode.Clear();
  29. }
  30. /// <summary>
  31. /// 使用字符串队列创建树结构的分支。
  32. /// </summary>
  33. /// <param name="branchString"></param>
  34. /// <param name="node"></param>
  35. private static TreeNode CreateTreeBranch(Queue<string> branchString, TreeNode node)
  36. {
  37. var rootNode = node;
  38. while (true)
  39. {
  40. if (branchString.Count <= 0)
  41. return rootNode;
  42. var name = branchString.Dequeue();
  43. if (node == null)
  44. {
  45. // 根节点
  46. node = new TreeNode(name);
  47. rootNode = node;
  48. }
  49. else
  50. {
  51. if (node.Name == name)
  52. continue;
  53. // 向ChildNode插入
  54. var subNode = node.ChildNodes.FirstOrDefault(x => x.Name == name);
  55. if (subNode == null)
  56. {
  57. subNode = new TreeNode(name);
  58. node.ChildNodes.Add(subNode);
  59. }
  60. node = subNode;
  61. }
  62. }
  63. }
  64. public void Create()
  65. {
  66. }
  67. public ObservableCollection<string> GetUserDefineParameters()
  68. {
  69. var typedContents = ((string)QueryDataClient.Instance.Service.GetTypedConfigContent("UserDefine", ""));
  70. ObservableCollection<string> dataList = new ObservableCollection<string>();
  71. if (typedContents != null)
  72. {
  73. var contentList = typedContents.Split(',').ToList();
  74. contentList.ForEach(x =>
  75. {
  76. if (!string.IsNullOrEmpty(x)) dataList.Add($"{x}");
  77. });
  78. }
  79. return dataList;
  80. }
  81. public Dictionary<string, string> GetSystemDeviceNameDict()
  82. {
  83. return new Dictionary<string, string>()
  84. {
  85. {ModuleName.CarrierRobot.ToString(),ModuleName.Robot.ToString() },
  86. {ModuleName.WaferRobot.ToString(),ModuleName.Robot.ToString() },
  87. {ModuleName.FIMS.ToString(),ModuleName.FIMS.ToString() },
  88. {ModuleName.LP.ToString(),ModuleName.LP.ToString() },
  89. {ModuleName.Stocker.ToString(),ModuleName.Stocker.ToString() },
  90. };
  91. }
  92. /// <summary>
  93. /// 筛选数据库字段
  94. /// </summary>
  95. /// <param name="dataList"></param>
  96. /// <returns></returns>
  97. private Dictionary<string, string> FilterNumericDataList(List<string> dataList, Dictionary<string, string> displayXmlDict)
  98. {
  99. //这里包含了项目中注册到数据库中的所有字段,需要从中筛选出来有效字段并归类
  100. Dictionary<string, string> result = new Dictionary<string, string>();
  101. var effectiveDataList = dataList.Where(a => a.Split('.').Count() > 1);
  102. var pm1DataList = effectiveDataList.Where(a => a.StartsWith(ModuleName.PM1.ToString()) && !displayXmlDict.Keys.Contains(a)).ToDictionary(a => a, a => a);
  103. var systemDataList = effectiveDataList.Where(a => a.StartsWith(ModuleName.System.ToString()) && !displayXmlDict.Keys.Contains(a)).ToDictionary(a => a, a => a);
  104. result.AddRange(pm1DataList);
  105. result.AddRange(systemDataList);
  106. var systemDeviceNameDict = GetSystemDeviceNameDict();
  107. // var NameKeyDict = QueryDataClient.Instance.Service.GetData("System.NameKeyDict") as Dictionary<string, List<string>>;
  108. foreach (var item in systemDeviceNameDict)
  109. {
  110. var data = effectiveDataList.Where(a => (a.StartsWith(item.Key))).ToDictionary(a => a, a => $"{item.Value}.{a}");
  111. result.AddRange(data);
  112. }
  113. return result;
  114. }
  115. /// <summary>
  116. /// 获取指定日期数据库PM/System table 列
  117. /// </summary>
  118. /// <param name="dbTableName"></param>
  119. /// <returns></returns>
  120. public List<string> GetDBCol(string dbTableName)
  121. {
  122. var columnsql = $"select column_name from information_schema.columns where table_name = '{dbTableName}'";
  123. var columnTable = QueryDataClient.Instance.Service.QueryData(columnsql);
  124. return columnTable.Rows.Cast<DataRow>().Select(x => x.ItemArray[0].ToString()).ToList();
  125. }
  126. /// <summary>
  127. /// 获取DataHistory 左侧树字典(既包含配置文件字段,也包含PM/System表字段)
  128. /// </summary>
  129. /// <returns></returns>
  130. public Dictionary<string, string> GetTreeNameDict()
  131. {
  132. Dictionary<string, string> displayDic = QueryDataClient.Instance.Service.GetData("System.Display") as Dictionary<string, string>;
  133. List<string> dataList = (List<string>)QueryDataClient.Instance.Service.GetConfig("System.NumericDataList");
  134. var dataListDict = FilterNumericDataList(dataList, displayDic);
  135. displayDic.AddRange(dataListDict);
  136. return displayDic;
  137. }
  138. /// <summary>
  139. /// 获取ProcessHistory 左侧树结构(不包含PM/System,只根据配置文件字段生成)
  140. /// </summary>
  141. /// <param name="processDetailDisplayDic"></param>
  142. /// <param name="isSelected"></param>
  143. /// <returns></returns>
  144. public Dictionary<string, Dictionary<string, string>> GetProcessProcessDetailAttributeDict()
  145. {
  146. return QueryDataClient.Instance.Service.GetData("System.ProcessDetailInfoDict") as Dictionary<string, Dictionary<string, string>>;
  147. }
  148. public ObservableCollection<ParameterNode> GetParameters(out Dictionary<string, string> processDetailDisplayDic, bool isSelected = false)
  149. {
  150. processDetailDisplayDic = new Dictionary<string, string>();
  151. ObservableCollection<ParameterNode> rootNode = new ObservableCollection<ParameterNode>();
  152. try
  153. {
  154. Dictionary<string, string> displayDic = QueryDataClient.Instance.Service.GetData("System.ProcessDetailDisplay") as Dictionary<string, string>;
  155. if (displayDic == null)
  156. return rootNode;
  157. List<string> dataList = new List<string>();
  158. foreach (var itemDisplay in displayDic)
  159. {
  160. string[] keyItems = itemDisplay.Key.Split('.');
  161. string[] valueItems = itemDisplay.Value.Split('.');
  162. if (keyItems != null)
  163. {
  164. if (keyItems.Length == 2)
  165. {
  166. var key = $"{valueItems[1]}";
  167. var keyValue = itemDisplay.Key;
  168. if (!processDetailDisplayDic.ContainsKey(key))
  169. {
  170. processDetailDisplayDic.Add(key, keyValue);
  171. }
  172. dataList.Add($"{valueItems[0]}.{valueItems[1]}");
  173. }
  174. else if (keyItems.Length == 3)
  175. {
  176. var key = $"{valueItems[1]} {valueItems[2]}";
  177. var keyValue = itemDisplay.Key;
  178. if (!processDetailDisplayDic.ContainsKey(key))
  179. {
  180. processDetailDisplayDic.Add(key, keyValue);
  181. }
  182. dataList.Add($"{valueItems[0]}.{valueItems[1]}.{valueItems[2]}");
  183. }
  184. }
  185. }
  186. Dictionary<string, ParameterNode> indexer = new Dictionary<string, ParameterNode>();
  187. foreach (string dataName in dataList)
  188. {
  189. string[] nodeName = dataName.Split('.');
  190. ParameterNode parentNode = null;
  191. string pathName = "";
  192. for (int i = 0; i < nodeName.Length; i++)
  193. {
  194. pathName = (i == 0 || i == 1) ? nodeName[i] : (pathName + " " + nodeName[i]);
  195. if (!indexer.ContainsKey(pathName))
  196. {
  197. indexer[pathName] = new ParameterNode() { Name = pathName, ChildNodes = new ObservableCollection<ParameterNode>(), ParentNode = parentNode };
  198. if (parentNode == null)
  199. {
  200. rootNode.Add(indexer[pathName]);
  201. }
  202. else
  203. {
  204. parentNode.ChildNodes.Add(indexer[pathName]);
  205. }
  206. }
  207. parentNode = indexer[pathName];
  208. }
  209. }
  210. SortParameterNode(rootNode,
  211. new Dictionary<string, int>() {
  212. { "Heater",0},
  213. { "MFC",1},
  214. { "APC",2},
  215. { "Boat",3},
  216. { "Valve",4} },
  217. isSelected);
  218. }
  219. catch (Exception ex)
  220. {
  221. LOG.Write(ex);
  222. }
  223. return rootNode;
  224. }
  225. /// <summary>
  226. /// 获取DataHistory 左侧树结构(既包含配置文件字段,也包含PM/System表字段)
  227. /// </summary>
  228. /// <returns></returns>
  229. public List<TreeNode> GetTreeNodeParameters()
  230. {
  231. var root = new List<TreeNode>();
  232. try
  233. {
  234. var displayDic = GetTreeNameDict();
  235. foreach (var dataName in displayDic.Values)
  236. {
  237. var nodeName = new Queue<string>(dataName.Split('.'));
  238. if (nodeName.Count <= 0)
  239. continue;
  240. // 开始创建Tree分支
  241. var rootNode = root.FirstOrDefault(x => x.Name == nodeName.Peek());
  242. var node = CreateTreeBranch(nodeName, rootNode);
  243. if (rootNode == null)
  244. {
  245. root.Add(node);
  246. }
  247. }
  248. SortParameterNode(root, "PM1", 0);
  249. SortParameterNode(root, "System", 1);
  250. SortParameterNode(root, "Heater", 2);
  251. SortParameterNode(root, "MFC", 3);
  252. SortParameterNode(root, "APC", 4);
  253. SortParameterNode(root, "Boat", 5);
  254. SortParameterNode(root, "Valve", 6);
  255. foreach (var rootNodes in root)
  256. {
  257. var childNode = rootNodes.ChildNodes.ToList();
  258. childNode.Sort(new NameComparer());
  259. for (int i = 0; i < childNode.Count; i++)
  260. {
  261. rootNodes.ChildNodes.Move(rootNodes.ChildNodes.IndexOf(childNode[i]), i);
  262. }
  263. }
  264. }
  265. catch (Exception ex)
  266. {
  267. LOG.Write(ex);
  268. }
  269. return root;
  270. }
  271. #region Sort 排序
  272. void SortParameterNode(List<TreeNode> rootNode, string Name, int Index)
  273. {
  274. if (Index >= rootNode.Count)
  275. {
  276. return;
  277. }
  278. for (int i = 0; i < rootNode.Count; i++)
  279. {
  280. if (rootNode[i].Name == Name && i != Index)
  281. {
  282. TreeNode node = rootNode[i];
  283. rootNode.RemoveAt(i);
  284. rootNode.Insert(Index, node);
  285. break;
  286. }
  287. }
  288. // Recursively sort children
  289. }
  290. void SortParameterNode(ObservableCollection<ParameterNode> rootNode, Dictionary<string, int> sortDict, bool isSelectd = false)
  291. {
  292. foreach (var item in sortDict)
  293. {
  294. SortParameterNode(rootNode, item.Key, item.Value, isSelectd);
  295. }
  296. }
  297. void SortParameterNode(ObservableCollection<ParameterNode> rootNode, string Name, int Index, bool isSelectd = false)
  298. {
  299. if (Index >= rootNode.Count)
  300. {
  301. return;
  302. }
  303. if (rootNode[Index].Name != Name)
  304. {
  305. for (int i = 0; i < rootNode.Count; i++)
  306. {
  307. rootNode[i].Selected = isSelectd;
  308. if (rootNode[i].Name == Name)
  309. {
  310. ParameterNode node = rootNode[i];
  311. rootNode.RemoveAt(i);
  312. rootNode.Insert(Index, node);
  313. }
  314. }
  315. }
  316. }
  317. #endregion
  318. public ObservableCollection<ParameterNode> GetParameterNodeParameters()
  319. {
  320. try
  321. {
  322. List<string> dataList =
  323. (List<string>)QueryDataClient.Instance.Service.GetConfig("System.NumericDataList");
  324. var typedContents = ((string)QueryDataClient.Instance.Service.GetTypedConfigContent("UserDefine", ""));
  325. for (int i = 0; i < dataList.Count; i++)
  326. {
  327. if (dataList[i].StartsWith("SMIF"))
  328. dataList[i] = $"SMIFs.{dataList[i]}";
  329. if (dataList[i].StartsWith("Stocker"))
  330. dataList[i] = $"Stockers.{dataList[i]}";
  331. if (dataList[i].StartsWith("Stage"))
  332. dataList[i] = $"Stages.{dataList[i]}";
  333. dataList[i] = dataList[i].Replace("PM1", "Tube");
  334. }
  335. if (string.IsNullOrEmpty(typedContents))
  336. {
  337. dataList.Add($"UserDefine");
  338. }
  339. else
  340. {
  341. var contentList = typedContents.Split(',').ToList();
  342. contentList.ForEach(x =>
  343. {
  344. if (!string.IsNullOrEmpty(x)) dataList.Add($"UserDefine.{x}");
  345. });
  346. }
  347. dataList.Sort();
  348. //List<string> removeList = _indexer.Keys.ToList();
  349. foreach (string dataName in dataList)
  350. {
  351. string[] nodeName = dataName.Split('.');
  352. ParameterNode parentNode = null;
  353. string pathName = "";
  354. for (int i = 0; i < nodeName.Length; i++)
  355. {
  356. pathName = (i == 0) ? nodeName[i] : (pathName + "." + nodeName[i]);
  357. //removeList.Remove(pathName);
  358. if (!_indexer.ContainsKey(pathName))
  359. {
  360. _indexer[pathName] = new ParameterNode()
  361. {
  362. Name = pathName.Replace("UserDefine.", "").Replace("SMIFs.", "").Replace("Stages.", "")
  363. .Replace("Stockers.", ""),
  364. ChildNodes = new ObservableCollection<ParameterNode>(),
  365. ParentNode = parentNode
  366. };
  367. if (parentNode == null)
  368. {
  369. _rootNode.Add(_indexer[pathName]);
  370. }
  371. else
  372. {
  373. parentNode.ChildNodes.Add(_indexer[pathName]);
  374. }
  375. }
  376. parentNode = _indexer[pathName];
  377. //removeList.Remove(pathName);
  378. }
  379. }
  380. //foreach (var key in removeList)
  381. //{
  382. // if (_indexer[key].ParentNode == null)
  383. // _rootNode.Remove(_indexer[key]);
  384. // else
  385. // {
  386. // _indexer[key].ParentNode.ChildNodes.Remove(_indexer[key]);
  387. // }
  388. //}
  389. return _rootNode;
  390. }
  391. catch (Exception ex)
  392. {
  393. LOG.Write(ex);
  394. }
  395. #region Test code
  396. ObservableCollection<ParameterNode> result = new ObservableCollection<ParameterNode>();
  397. ParameterNode node1 = new ParameterNode() { Name = "Para Node 1", Selected = false, ChildNodes = new ObservableCollection<ParameterNode>() };
  398. ParameterNode node2 = new ParameterNode() { Name = "Para Node 2", Selected = false, ChildNodes = new ObservableCollection<ParameterNode>() };
  399. ParameterNode node3 = new ParameterNode() { Name = "Para Node 3", Selected = false, ChildNodes = new ObservableCollection<ParameterNode>() };
  400. for (int i = 0; i < 5; i++)
  401. {
  402. ParameterNode node = new ParameterNode() { Name = node1.Name + "_" + i.ToString(), Selected = false, ChildNodes = new ObservableCollection<ParameterNode>() };
  403. node1.ChildNodes.Add(node);
  404. }
  405. for (int i = 0; i < 3; i++)
  406. {
  407. ParameterNode node = new ParameterNode() { Name = node2.Name + "_" + i.ToString(), Selected = false, ChildNodes = new ObservableCollection<ParameterNode>() };
  408. node2.ChildNodes.Add(node);
  409. }
  410. for (int i = 0; i < 4; i++)
  411. {
  412. ParameterNode node = new ParameterNode() { Name = node3.Name + "_" + i.ToString(), Selected = false, ChildNodes = new ObservableCollection<ParameterNode>() };
  413. node3.ChildNodes.Add(node);
  414. }
  415. result.Add(node1);
  416. result.Add(node2);
  417. result.Add(node3);
  418. return result;
  419. #endregion
  420. }
  421. }
  422. public class NameComparer : IComparer<TreeNode>
  423. {
  424. public int Compare(TreeNode x, TreeNode y)
  425. {
  426. if (x == null || y == null)
  427. throw new ArgumentException("Parameters can't be null");
  428. string fileA = x.DisplayName;
  429. string fileB = y.DisplayName;
  430. char[] arr1 = fileA.ToCharArray();
  431. char[] arr2 = fileB.ToCharArray();
  432. int i = 0, j = 0;
  433. while (i < arr1.Length && j < arr2.Length)
  434. {
  435. if (char.IsDigit(arr1[i]) && char.IsDigit(arr2[j]))
  436. {
  437. string s1 = "", s2 = "";
  438. while (i < arr1.Length && char.IsDigit(arr1[i]))
  439. {
  440. s1 += arr1[i];
  441. i++;
  442. }
  443. while (j < arr2.Length && char.IsDigit(arr2[j]))
  444. {
  445. s2 += arr2[j];
  446. j++;
  447. }
  448. if (int.Parse(s1) > int.Parse(s2))
  449. {
  450. return 1;
  451. }
  452. if (int.Parse(s1) < int.Parse(s2))
  453. {
  454. return -1;
  455. }
  456. }
  457. else
  458. {
  459. if (arr1[i] > arr2[j])
  460. {
  461. return 1;
  462. }
  463. if (arr1[i] < arr2[j])
  464. {
  465. return -1;
  466. }
  467. i++;
  468. j++;
  469. }
  470. }
  471. if (arr1.Length == arr2.Length)
  472. {
  473. return 0;
  474. }
  475. else
  476. {
  477. return arr1.Length > arr2.Length ? 1 : -1;
  478. }
  479. }
  480. }
  481. }