WaferHistoryDBViewModel.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675
  1. using Prism.Mvvm;
  2. using System;
  3. using Prism.Commands;
  4. using System.Collections.Generic;
  5. using System.Linq;
  6. using System.Text;
  7. using System.Threading.Tasks;
  8. using CyberX8_MainPages.Views;
  9. using MECF.Framework.Common.CommonData;
  10. using System.Collections.ObjectModel;
  11. using Xceed.Wpf.Toolkit.Primitives;
  12. using CyberX8_MainPages.Unity;
  13. using System.Windows;
  14. using MECF.Framework.Common.DataCenter;
  15. using System.Data;
  16. using System.ComponentModel;
  17. using OpenSEMI.Ctrlib.Controls;
  18. using CyberX8_Core;
  19. using RecipeStep = MECF.Framework.Common.CommonData.RecipeStep;
  20. using QiHe.CodeLib;
  21. namespace CyberX8_MainPages.ViewModels
  22. {
  23. public class WaferHistoryDBViewModel : BindableBase
  24. {
  25. private ObservableCollection<WaferHistoryMovement> _movements = new ObservableCollection<WaferHistoryMovement>();
  26. public ObservableCollection<WaferHistoryMovement> Movements
  27. {
  28. get { return _movements; }
  29. set { SetProperty(ref _movements, value); }
  30. }
  31. private WaferHistoryItem _selectedItem;
  32. public WaferHistoryItem SelectedItem
  33. {
  34. get { return _selectedItem; }
  35. set { SetProperty(ref _selectedItem, value); }
  36. }
  37. private DateTime _searchBeginTime;
  38. public DateTime SearchBeginTime
  39. {
  40. get { return _searchBeginTime; }
  41. set { SetProperty(ref _searchBeginTime, value); }
  42. }
  43. private ObservableCollection<WaferHistoryLot> _lots = new ObservableCollection<WaferHistoryLot>();
  44. public ObservableCollection<WaferHistoryLot> Lots
  45. {
  46. get { return _lots; }
  47. set { SetProperty(ref _lots, value); }
  48. }
  49. public ObservableCollection<WaferHistoryRecipe> _recipes = new ObservableCollection<WaferHistoryRecipe>();
  50. public ObservableCollection<WaferHistoryRecipe> Recipes
  51. {
  52. get { return _recipes; }
  53. set { SetProperty(ref _recipes, value); }
  54. }
  55. private WaferHistoryRecipe _recipe;
  56. public WaferHistoryRecipe Recipe
  57. {
  58. get { return _recipe; }
  59. set { SetProperty(ref _recipe, value); }
  60. }
  61. private ObservableCollection<LazyTreeItem<WaferHistoryItem>> _historyData;
  62. public ObservableCollection<LazyTreeItem<WaferHistoryItem>> HistoryData
  63. {
  64. get { return _historyData; }
  65. set { SetProperty(ref _historyData, value); }
  66. }
  67. private ObservableCollection<WaferHistoryWafer> _wafers = new ObservableCollection<WaferHistoryWafer>();
  68. public ObservableCollection<WaferHistoryWafer> Wafers
  69. {
  70. get { return _wafers; }
  71. set { SetProperty(ref _wafers, value); }
  72. }
  73. private DateTime _searchEndTime;
  74. public DateTime SearchEndTime
  75. {
  76. get { return _searchEndTime; }
  77. set { SetProperty(ref _searchEndTime, value); }
  78. }
  79. private string keyWord;
  80. public string KeyWord
  81. {
  82. get { return keyWord; }
  83. set { SetProperty(ref keyWord, value); }
  84. }
  85. private WaferHistoryDBView view;
  86. public DateTime StartDateTime { get; set; }
  87. public DateTime EndDateTime { get; set; }
  88. private DelegateCommand<object> _QueryCommand;
  89. public DelegateCommand<object> QueryCommand =>
  90. _QueryCommand ?? (_QueryCommand = new DelegateCommand<object>(QueryLots));
  91. private DelegateCommand<object> _ToChartCommand;
  92. public DelegateCommand<object> ToChartCommand =>
  93. _ToChartCommand ?? (_ToChartCommand = new DelegateCommand<object>(Tocharts));
  94. private DelegateCommand<WaferHistoryItem> _SelectionChangedCommand;
  95. public DelegateCommand<WaferHistoryItem> SelectionChangedCommand =>
  96. _SelectionChangedCommand ?? (_SelectionChangedCommand = new DelegateCommand<WaferHistoryItem>(SelectionChanged));
  97. private DelegateCommand<object> _LoadCommandPD;
  98. public DelegateCommand<object> LoadCommandPD =>
  99. _LoadCommandPD ?? (_LoadCommandPD = new DelegateCommand<object>(OnLoadPd));
  100. public WaferHistoryDBViewModel()
  101. {
  102. HistoryData = new ObservableCollection<LazyTreeItem<WaferHistoryItem>>();
  103. }
  104. void Tocharts(object o)
  105. {
  106. WaferHistoryItem item = o as WaferHistoryItem;
  107. WaferHistoryWafer item1 = o as WaferHistoryWafer;
  108. if (o is WaferHistoryRecipe WaferRecipe)
  109. {
  110. WaferHistoryWafer wafer1 = new WaferHistoryWafer();
  111. wafer1.StartTime = WaferRecipe.StartTime;
  112. wafer1.EndTime = WaferRecipe.EndTime;
  113. string sql = string.Format($"SELECT * FROM \"lot_wafer_data\" where wafer_data_guid ='{WaferRecipe.RfID}'; ");
  114. DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql);
  115. string lot_data_guid = dbData.Rows[0]["guid"].ToString();
  116. wafer1.WaferGuid = lot_data_guid;
  117. SwitchPage("Process History", wafer1);
  118. }
  119. else if (o is WaferHistoryLot waferLot)
  120. {
  121. item.Type = WaferHistoryItemType.Lot;
  122. SelectionChanged(item);
  123. WaferHistoryItem selectItem = null;
  124. foreach (var root in HistoryData)
  125. {
  126. foreach (var subroot in root.SubItems)
  127. {
  128. var data = subroot.Data as WaferHistoryItem;
  129. if (data != null && data.Name != null && data.Name == item.Name)
  130. {
  131. selectItem = data;
  132. subroot.IsExpanded = true;
  133. break;
  134. }
  135. }
  136. if (selectItem != null)
  137. break;
  138. }
  139. }
  140. else if (o is WaferHistoryWafer wafer)
  141. {
  142. SwitchPage("Process History", item1);
  143. }
  144. }
  145. void SwitchPage(string MenuID, WaferHistoryWafer queryFilter)
  146. {
  147. Anychange.needchange = true;
  148. Anychange.menuname = MenuID;
  149. Anychange.args= queryFilter;
  150. }
  151. void QueryLots(object e)
  152. {
  153. this.SearchBeginTime = this.view.wfTimeFrom.Value;
  154. this.SearchEndTime = this.view.wfTimeTo.Value;
  155. if (SearchBeginTime > SearchEndTime)
  156. {
  157. MessageBox.Show("Time range invalid, start time should be early than end time");
  158. return;
  159. }
  160. Lots = new ObservableCollection<WaferHistoryLot>(QueryLot(SearchBeginTime, SearchEndTime, KeyWord));//.OrderByDescending(lot => lot.StartTime).ToArray();
  161. HistoryData.Clear();
  162. var lotsItem = new WaferHistoryItem() { Name = "Lots", };
  163. var root = new LazyTreeItem<WaferHistoryItem>(lotsItem, x => LoadSubItem(x));
  164. root.SubItems = new ObservableCollection<LazyTreeItem<WaferHistoryItem>>(Lots.Select(x => new LazyTreeItem<WaferHistoryItem>(new WaferHistoryItem() { ID = x.ID, Name = x.Name, StartTime = x.StartTime, EndTime = x.EndTime, Type = WaferHistoryItemType.Lot }, LoadSubItem)));
  165. root.IsExpanded = true;
  166. HistoryData.Add(root);
  167. SelectedItem = lotsItem;
  168. }
  169. private Dictionary<string, object> _mapGasNameValue = new Dictionary<string, object>();
  170. private Dictionary<string, string> _mapGasNameKey = new Dictionary<string, string>()
  171. {
  172. //{"PMA.MfcGas1.GasName", "PMA.IoMfc.MfcGas1"},
  173. //{"PMA.MfcGas2.GasName", "PMA.IoMfc.MfcGas2"},
  174. //{"PMA.MfcGas3.GasName", "PMA.IoMfc.MfcGas3"},
  175. //{"PMA.MfcGas4.GasName", "PMA.IoMfc.MfcGas4"},
  176. //{"PMA.MfcGas5.GasName", "PMA.IoMfc.MfcGas5"},
  177. //{"PMB.MfcGas1.GasName", "PMB.IoMfc.MfcGas1"},
  178. //{"PMB.MfcGas2.GasName", "PMB.IoMfc.MfcGas2"},
  179. //{"PMB.MfcGas3.GasName", "PMB.IoMfc.MfcGas3"},
  180. //{"PMB.MfcGas4.GasName", "PMB.IoMfc.MfcGas4"},
  181. //{"PMB.MfcGas5.GasName", "PMB.IoMfc.MfcGas5"},
  182. };
  183. //protected override void OnActivate()
  184. //{
  185. // base.OnActivate();
  186. // _mapGasNameValue = QueryDataClient.Instance.Service.PollConfig(_mapGasNameKey.Keys);
  187. //}
  188. private void SelectionChanged(WaferHistoryItem item)
  189. {
  190. switch (item.Type)
  191. {
  192. case WaferHistoryItemType.Lot:
  193. QueryLotWafer(item);
  194. break;
  195. case WaferHistoryItemType.Wafer:
  196. QueryWaferRecipe(item);
  197. QueryWaferMovement(item);
  198. break;
  199. case WaferHistoryItemType.Recipe:
  200. QueryRecipe(item);
  201. break;
  202. default:
  203. break;
  204. }
  205. SelectedItem = item;
  206. }
  207. public WaferHistoryRecipe QueryRecipe(WaferHistoryItem whItem)
  208. {
  209. WaferHistoryRecipe result = new WaferHistoryRecipe();
  210. string sql = $"SELECT * FROM \"process_data\" where \"guid\" = '{whItem.ID}'and \"process_begin_time\" >= '{whItem.StartTime:yyyy/MM/dd HH:mm:ss.fff}' and \"process_begin_time\" <= '{whItem.EndTime:yyyy/MM/dd HH:mm:ss.fff}';";
  211. //Application.Current.Dispatcher.BeginInvoke(new Action(() =>
  212. //{
  213. DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql);
  214. if (dbData != null && dbData.Rows.Count > 0)
  215. {
  216. for (int i = 0; i < dbData.Rows.Count; i++)
  217. {
  218. WaferHistoryRecipe item = new WaferHistoryRecipe();
  219. item.ID = dbData.Rows[i]["guid"].ToString();
  220. var itemName = dbData.Rows[i]["recipe_name"].ToString();
  221. if (!dbData.Rows[i]["process_begin_time"].Equals(DBNull.Value))
  222. item.StartTime = (DateTime)dbData.Rows[i]["process_begin_time"];
  223. if (!dbData.Rows[i]["process_end_time"].Equals(DBNull.Value))
  224. item.EndTime = (DateTime)dbData.Rows[i]["process_end_time"];
  225. if (dbData.Rows[i].Table.Columns.Contains("recipe_setting_time") && !dbData.Rows[i]["recipe_setting_time"].Equals(DBNull.Value))
  226. item.SettingTime = dbData.Rows[i]["recipe_setting_time"].ToString();
  227. item.ActualTime = item.Duration;
  228. item.Recipe = itemName;
  229. item.Chamber = dbData.Rows[i]["process_in"].ToString();
  230. item.Name = itemName;
  231. item.Type = WaferHistoryItemType.Recipe;
  232. result = item;
  233. }
  234. }
  235. sql = $"SELECT * FROM \"recipe_step_data\" where \"process_data_guid\" = '{whItem.ID}' and \"step_begin_time\" >= '{whItem.StartTime:yyyy/MM/dd HH:mm:ss.fff}' and \"step_begin_time\" <= '{whItem.EndTime:yyyy/MM/dd HH:mm:ss.fff}' order by step_number ASC;";
  236. dbData = QueryDataClient.Instance.Service.QueryData(sql);
  237. List<RecipeStep> steps = new List<RecipeStep>();
  238. if (dbData != null && dbData.Rows.Count > 0)
  239. {
  240. for (int i = 0; i < dbData.Rows.Count; i++)
  241. {
  242. RecipeStep item = new RecipeStep();
  243. item.No = int.Parse(dbData.Rows[i]["step_number"].ToString());
  244. item.Name = dbData.Rows[i]["step_name"].ToString();
  245. if (!dbData.Rows[i]["step_begin_time"].Equals(DBNull.Value))
  246. item.StartTime = (DateTime)dbData.Rows[i]["step_begin_time"];
  247. if (!dbData.Rows[i]["step_end_time"].Equals(DBNull.Value))
  248. item.EndTime = (DateTime)dbData.Rows[i]["step_end_time"];
  249. item.ActualTime = item.EndTime.CompareTo(item.StartTime) <= 0 ? "" : item.EndTime.Subtract(item.StartTime).TotalSeconds.ToString();
  250. item.SettingTime = dbData.Rows[i]["step_time"].ToString();
  251. steps.Add(item);
  252. }
  253. }
  254. sql = $"SELECT * FROM \"step_fdc_data\" where \"process_data_guid\" = '{whItem.ID}' and \"create_time\" >= '{whItem.StartTime:yyyy/MM/dd HH:mm:ss.fff}' and \"create_time\" <= '{whItem.EndTime:yyyy/MM/dd HH:mm:ss.fff}'order by step_number ASC;";
  255. dbData = QueryDataClient.Instance.Service.QueryData(sql);
  256. List<RecipeStepFdcData> fdcs = new List<RecipeStepFdcData>();
  257. if (dbData != null && dbData.Rows.Count > 0)
  258. {
  259. for (int i = 0; i < dbData.Rows.Count; i++)
  260. {
  261. RecipeStepFdcData item = new RecipeStepFdcData();
  262. item.StepNumber = int.Parse(dbData.Rows[i]["step_number"].ToString());
  263. item.Name = dbData.Rows[i]["parameter_name"].ToString();
  264. foreach (var name in _mapGasNameKey)
  265. {
  266. if (item.Name.Contains(name.Value) && _mapGasNameValue.ContainsKey(name.Key))
  267. item.Name = item.Name.Replace(name.Value, (string)_mapGasNameValue[name.Key]);
  268. }
  269. if (!dbData.Rows[i]["sample_count"].Equals(DBNull.Value))
  270. item.SampleCount = (int)dbData.Rows[i]["sample_count"];
  271. if (!dbData.Rows[i]["setpoint"].Equals(DBNull.Value))
  272. item.SetPoint = (float)dbData.Rows[i]["setpoint"];
  273. if (!dbData.Rows[i]["min_value"].Equals(DBNull.Value))
  274. item.MinValue = (float)dbData.Rows[i]["min_value"];
  275. if (!dbData.Rows[i]["max_value"].Equals(DBNull.Value))
  276. item.MaxValue = (float)dbData.Rows[i]["max_value"];
  277. if (!dbData.Rows[i]["std_value"].Equals(DBNull.Value))
  278. item.StdValue = (float)dbData.Rows[i]["std_value"];
  279. if (!dbData.Rows[i]["mean_value"].Equals(DBNull.Value))
  280. item.MeanValue = (float)dbData.Rows[i]["mean_value"];
  281. fdcs.Add(item);
  282. }
  283. }
  284. result.Steps = steps;
  285. result.Fdcs = fdcs;
  286. Recipe = result;
  287. return result;
  288. }
  289. public List<WaferHistoryMovement> QueryWaferMovement(WaferHistoryItem whItem)
  290. {
  291. List<WaferHistoryMovement> result = new List<WaferHistoryMovement>();
  292. string sql = $"SELECT * FROM \"wafer_move_history\" where \"wafer_data_guid\" = '{whItem.ID}'and \"arrive_time\" >= '{SearchBeginTime:yyyy/MM/dd HH:mm:ss.fff}' and \"arrive_time\" <= '{SearchEndTime:yyyy/MM/dd HH:mm:ss.fff}' order by \"arrive_time\" ASC limit 1000;";
  293. Movements.Clear();
  294. //Application.Current.Dispatcher.BeginInvoke(new Action(() =>
  295. //{
  296. DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql);
  297. if (dbData != null && dbData.Rows.Count > 0)
  298. {
  299. for (int i = 0; i < dbData.Rows.Count - 1; i++)
  300. {
  301. WaferHistoryMovement item = new WaferHistoryMovement();
  302. item.Source = $"station : {dbData.Rows[i]["station"]} slot : {dbData.Rows[i]["slot"]}";
  303. item.Destination = $"station : {dbData.Rows[i + 1]["station"]} slot : {dbData.Rows[i + 1]["slot"]}";
  304. item.InTime = dbData.Rows[i]["arrive_time"].ToString();
  305. result.Add(item);
  306. Movements.Add(item);
  307. }
  308. }
  309. //}));
  310. return result;
  311. }
  312. public List<WaferHistoryWafer> QueryLotWafer(WaferHistoryItem whItem)
  313. {
  314. List<WaferHistoryWafer> result = new List<WaferHistoryWafer>();
  315. //string sql = $"select data.*,process_data.process_status from (SELECT * FROM wafer_data,lot_wafer_data where lot_wafer_data.lot_data_guid = '{lotGuid}' and lot_wafer_data.wafer_data_guid = wafer_data.guid order by wafer_data.create_time ASC; as data )" +
  316. // $" LEFT JOIN process_data ON process_data.wafer_data_guid=data.wafer_id";
  317. //string sql = $"SELECT wafer_data.*,lot_wafer_data.*,process_data.process_status as status FROM wafer_data LEFT JOIN lot_wafer_data ON lot_wafer_data.wafer_data_guid = wafer_data.guid " +
  318. // $"LEFT JOIN process_data ON process_data.wafer_data_guid = wafer_data.guid where lot_wafer_data.lot_data_guid = '{whItem.ID}' order by wafer_data.create_time ASC";
  319. string sql = $"SELECT wafer_data.*,lot_wafer_data.*,wafer_data.process_status as status,wafer_data.reason FROM wafer_data LEFT JOIN lot_wafer_data ON lot_wafer_data.wafer_data_guid = wafer_data.guid " +
  320. $"LEFT JOIN process_data ON process_data.wafer_data_guid = wafer_data.guid where lot_wafer_data.lot_data_guid = '{whItem.ID}' order by wafer_data.create_time ASC";
  321. Wafers.Clear();
  322. //Application.Current.Dispatcher.BeginInvoke(new Action(() =>
  323. //{
  324. DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql);
  325. Dictionary<string, int> itemPtr = new Dictionary<string, int>();
  326. List<string> itemNameList = new List<string>();
  327. if (dbData != null && dbData.Rows.Count > 0)
  328. {
  329. for (int i = 0; i < dbData.Rows.Count; i++)
  330. {
  331. WaferHistoryWafer item = new WaferHistoryWafer();
  332. item.WaferGuid = dbData.Rows[i]["guid1"].ToString();
  333. item.ID = dbData.Rows[i]["guid"].ToString();
  334. if (!itemPtr.ContainsKey(item.ID))
  335. {
  336. itemPtr.Add(item.ID, 0);
  337. }
  338. var itemLoadPort = dbData.Rows[i]["create_station"].ToString();
  339. var itemSlot = dbData.Rows[i]["create_slot"].ToString();
  340. //item.CarrierID = dbData.Rows[i]["rfid"].ToString();
  341. item.LotID = dbData.Rows[i]["lot_id"].ToString();
  342. item.SlotID = dbData.Rows[i]["create_slot"].ToString();
  343. var itemWaferID = dbData.Rows[i]["wafer_id"].ToString();
  344. item.Sequence = dbData.Rows[i]["sequence_name"].ToString();
  345. item.Status = dbData.Rows[i]["status"].ToString();
  346. item.Reason = dbData.Rows[i]["reason"].ToString();
  347. sql = $"SELECT * FROM \"wafer_move_history\" where \"wafer_data_guid\" = '{item.ID}' and \"arrive_time\" >= '{whItem.StartTime:yyyy/MM/dd HH:mm:ss.fff}' and \"arrive_time\" <= '{whItem.EndTime:yyyy/MM/dd HH:mm:ss.fff}' order by \"arrive_time\" ASC limit 1000;";
  348. DataTable dbDataMovement = QueryDataClient.Instance.Service.QueryData(sql);
  349. if (dbDataMovement != null && dbDataMovement.Rows.Count >= 2 && dbDataMovement.Rows.Count > itemPtr[item.ID])
  350. {
  351. item.StartTime = (DateTime)dbDataMovement.Rows[itemPtr[item.ID]]["arrive_time"];
  352. for (int j = itemPtr[item.ID]; j < dbDataMovement.Rows.Count; j++)
  353. {
  354. if (dbDataMovement.Rows[j]["station"].ToString().StartsWith("LP"))
  355. {
  356. item.EndTime = (DateTime)dbDataMovement.Rows[j]["arrive_time"];
  357. itemPtr[item.ID] = j + 1;
  358. break;
  359. }
  360. }
  361. }
  362. else
  363. {
  364. if (!dbData.Rows[i]["create_time"].Equals(DBNull.Value))
  365. {
  366. item.StartTime = ((DateTime)dbData.Rows[i]["create_time"]);
  367. }
  368. if (!dbData.Rows[i]["delete_time"].Equals(DBNull.Value))
  369. item.EndTime = ((DateTime)dbData.Rows[i]["delete_time"]);
  370. }
  371. item.Name = $"{itemLoadPort} - {itemSlot} - {itemWaferID}";
  372. item.Type = WaferHistoryItemType.Wafer;
  373. if (!itemNameList.Contains(item.Name))
  374. {
  375. Wafers.Add(item);
  376. result.Add(item);
  377. itemNameList.Add(item.Name);
  378. }
  379. }
  380. }
  381. //}));
  382. return result;
  383. }
  384. public List<WaferHistoryRecipe> QueryWaferRecipe(WaferHistoryItem whItem)
  385. {
  386. List<WaferHistoryRecipe> result = new List<WaferHistoryRecipe>();
  387. string sql = $"SELECT * FROM \"process_data\" where \"wafer_data_guid\" = '{whItem.ID}'and \"process_begin_time\" >= '{whItem.StartTime:yyyy/MM/dd HH:mm:ss.fff}' and \"process_begin_time\" <= '{whItem.EndTime:yyyy/MM/dd HH:mm:ss.fff}' order by \"process_begin_time\" ASC;";
  388. Recipes.Clear();
  389. //Application.Current.Dispatcher.BeginInvoke(new Action(() =>
  390. //{
  391. DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql);
  392. if (dbData != null && dbData.Rows.Count > 0)
  393. {
  394. for (int i = 0; i < dbData.Rows.Count; i++)
  395. {
  396. WaferHistoryRecipe item = new WaferHistoryRecipe();
  397. item.RfID = dbData.Rows[i]["wafer_data_guid"].ToString();
  398. item.ID = dbData.Rows[i]["guid"].ToString();
  399. var itemName = dbData.Rows[i]["recipe_name"].ToString();
  400. if (!dbData.Rows[i]["process_begin_time"].Equals(DBNull.Value))
  401. item.StartTime = (DateTime)dbData.Rows[i]["process_begin_time"];
  402. if (dbData.Rows[i].Table.Columns.Contains("process_end_time") && !dbData.Rows[i]["process_end_time"].Equals(DBNull.Value))
  403. item.EndTime = (DateTime)dbData.Rows[i]["process_end_time"];
  404. if (dbData.Rows[i].Table.Columns.Contains("recipe_setting_time") && !dbData.Rows[i]["recipe_setting_time"].Equals(DBNull.Value))
  405. item.SettingTime = dbData.Rows[i]["recipe_setting_time"].ToString();
  406. item.ActualTime = item.Duration;
  407. item.Recipe = itemName;
  408. item.Chamber = dbData.Rows[i]["process_in"].ToString();
  409. item.Name = itemName;
  410. item.Type = WaferHistoryItemType.Recipe;
  411. Recipes.Add(item);
  412. result.Add(item);
  413. }
  414. }
  415. //}));
  416. return result;
  417. }
  418. public List<WaferHistoryLot> QueryLot(DateTime from, DateTime to, string key)
  419. {
  420. List<WaferHistoryLot> result = new List<WaferHistoryLot>();
  421. string sql = $"SELECT * FROM \"lot_data\" where \"start_time\" >= '{from:yyyy/MM/dd HH:mm:ss.fff}' and \"start_time\" <= '{to:yyyy/MM/dd HH:mm:ss.fff}' ";
  422. if (!string.IsNullOrWhiteSpace(key))
  423. sql += $" and lower(\"name\") like '%{key.ToLower()}%'";
  424. sql += "order by \"start_time\" ASC;";
  425. //Application.Current.Dispatcher.BeginInvoke(new Action(() =>
  426. //{
  427. DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql);
  428. if (dbData != null && dbData.Rows.Count > 0)
  429. {
  430. for (int i = 0; i < dbData.Rows.Count; i++)
  431. {
  432. WaferHistoryLot item = new WaferHistoryLot();
  433. string name = dbData.Rows[i]["name"].ToString();
  434. string time = "";
  435. item.WaferCount = (int)dbData.Rows[i]["total_wafer_count"];
  436. item.ID = dbData.Rows[i]["guid"].ToString();
  437. if (!dbData.Rows[i]["start_time"].Equals(DBNull.Value))
  438. {
  439. item.StartTime = ((DateTime)dbData.Rows[i]["start_time"]);
  440. time = item.StartTime.ToString("yyyy-MM-dd HH:mm:ss");
  441. }
  442. if (!dbData.Rows[i]["end_time"].Equals(DBNull.Value))
  443. {
  444. item.EndTime = ((DateTime)dbData.Rows[i]["end_time"]);
  445. }
  446. item.InputPort = dbData.Rows[i]["input_port"].ToString();
  447. item.Name = $"{name}-{time}";
  448. item.LotID = $"{name}";
  449. result.Add(item);
  450. }
  451. }
  452. return result;
  453. }
  454. private void OnLoadPd(Object eventView)
  455. {
  456. this.view = (WaferHistoryDBView)eventView;
  457. this.view.wfTimeFrom.Value = DateTime.Today;
  458. this.view.wfTimeTo.Value = new DateTime(DateTime.Today.Year, DateTime.Today.Month, DateTime.Today.Day, 23, 59, 59, 999);
  459. }
  460. protected List<LazyTreeItem<WaferHistoryItem>> LoadSubItem(WaferHistoryItem item)
  461. {
  462. switch (item.Type)
  463. {
  464. case WaferHistoryItemType.Lot:
  465. var wafers = QueryLotWafer(item);
  466. return wafers.Select(x => new LazyTreeItem<WaferHistoryItem>(x, y => LoadSubItem(y))).OrderBy(s => s.Data.StartTime).ToList();
  467. case WaferHistoryItemType.Wafer:
  468. var recipes = QueryWaferRecipe(item);
  469. return recipes.Select(x => new LazyTreeItem<WaferHistoryItem>(x, y => LoadSubItem(y))).OrderBy(s => s.Data.StartTime).ToList();
  470. default:
  471. break;
  472. }
  473. return new List<LazyTreeItem<WaferHistoryItem>> { };
  474. }
  475. }
  476. public class LazyTreeItem<T> : INotifyPropertyChanged where T : ITreeItem<T>, new()
  477. {
  478. private LazyTreeItem<T> dummyChild;
  479. private T data;
  480. private Func<T, List<LazyTreeItem<T>>> loader;
  481. private LazyTreeItem()
  482. {
  483. data = new T();
  484. data.ID = Guid.NewGuid().ToString();
  485. }
  486. public LazyTreeItem(T data, Func<T, List<LazyTreeItem<T>>> loader)
  487. {
  488. this.data = data;
  489. this.loader = loader;
  490. dummyChild = new LazyTreeItem<T>();
  491. SubItems = new ObservableCollection<LazyTreeItem<T>>();
  492. SubItems.Add(dummyChild);
  493. }
  494. public T Data
  495. {
  496. get
  497. {
  498. return data;
  499. }
  500. }
  501. public ObservableCollection<LazyTreeItem<T>> SubItems
  502. {
  503. get; set;
  504. }
  505. private bool isExpanded;
  506. public event PropertyChangedEventHandler PropertyChanged;
  507. private void OnPropertyChanged(string name)
  508. {
  509. if (PropertyChanged != null)
  510. {
  511. PropertyChanged.Invoke(this, new PropertyChangedEventArgs(name));
  512. }
  513. }
  514. public bool HasDummyChild
  515. {
  516. get { return SubItems.Count == 1 && SubItems.First().data.ID == dummyChild.data.ID; }
  517. }
  518. public bool IsExpanded
  519. {
  520. get { return isExpanded; }
  521. set
  522. {
  523. if (value != isExpanded)
  524. {
  525. isExpanded = value;
  526. OnPropertyChanged("IsExpanded");
  527. }
  528. if (HasDummyChild)
  529. {
  530. SubItems.Remove(dummyChild);
  531. var items = loader(data);
  532. foreach (var item in items)
  533. {
  534. SubItems.Add(item);
  535. }
  536. //items.ForEachDo(x => SubItems.Add(x));
  537. }
  538. }
  539. }
  540. }
  541. }