RealtimeProvider.cs 25 KB

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