using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.ComponentModel; using System.Data; using System.Linq; using System.Windows.Forms; using System.Windows.Input; using MECF.Framework.Common.CommonData; using MECF.Framework.Common.DataCenter; using OpenSEMI.ClientBase; using OpenSEMI.ClientBase.Command; using SciChart.Core.Extensions; namespace VirgoUI.Client.Models.DataLog.WaferHistory { public class WaferHistoryDBViewModel : BaseModel { public WaferHistoryDBViewModel() { SelectionChangedCommand = new BaseCommand(SelectionChanged); QueryCommand = new BaseCommand(QueryLots); ToChartCommand = new BaseCommand(GoToChart); HistoryData = new ObservableCollection>(); InitTime(); } private ObservableCollection> _historyData; public ObservableCollection> HistoryData { get { return _historyData; } set { _historyData = value; NotifyOfPropertyChange("HistoryData"); } } private WaferHistoryItem _selectedItem; public WaferHistoryItem SelectedItem { get { return _selectedItem; } set { _selectedItem = value; NotifyOfPropertyChange("SelectedItem"); } } private ObservableCollection _lots = new ObservableCollection(); public ObservableCollection Lots { get { return _lots; } set { _lots = value; NotifyOfPropertyChange("Lots"); } } private ObservableCollection _wafers = new ObservableCollection(); public ObservableCollection Wafers { get { return _wafers; } set { _wafers = value; NotifyOfPropertyChange("Wafers"); } } private ObservableCollection _movements = new ObservableCollection(); public ObservableCollection Movements { get { return _movements; } set { _movements = value; NotifyOfPropertyChange("Movements"); } } private WaferHistoryRecipe _recipe; public WaferHistoryRecipe Recipe { get { return _recipe; } set { _recipe = value; NotifyOfPropertyChange("Recipe"); } } public ObservableCollection _recipes = new ObservableCollection(); public ObservableCollection Recipes { get { return _recipes; } set { _recipes = value; NotifyOfPropertyChange("Recipes"); } } private DateTime _searchBeginTime; public DateTime SearchBeginTime { get { return _searchBeginTime; } set { _searchBeginTime = value; NotifyOfPropertyChange("SearchBeginTime"); } } private DateTime _searchEndTime; public DateTime SearchEndTime { get { return _searchEndTime; } set { _searchEndTime = value; NotifyOfPropertyChange("SearchEndTime"); } } private string keyWord; private WaferHistoryDBView view; public string KeyWord { get { return keyWord; } set { keyWord = value; NotifyOfPropertyChange("KeyWord"); } } public ICommand SelectionChangedCommand { get; set; } public ICommand QueryCommand { get; set; } public ICommand ToChartCommand { get; set; } void InitTime() { SearchBeginTime = DateTime.Now.Date; SearchEndTime = DateTime.Now.Date.AddDays(1).Date; } void QueryLots(object e) { this.SearchBeginTime = this.view.wfTimeFrom.Value; this.SearchEndTime = this.view.wfTimeTo.Value; if (SearchBeginTime > SearchEndTime) { MessageBox.Show("Time range invalid, start time should be early than end time"); return; } Lots = new ObservableCollection(QueryLot(SearchBeginTime, SearchEndTime, KeyWord));//.OrderByDescending(lot => lot.StartTime).ToArray(); HistoryData.Clear(); var lotsItem = new WaferHistoryItem() { Name = "Lots", }; var root = new LazyTreeItem(lotsItem, x => LoadSubItem(x)); root.SubItems = new ObservableCollection>(Lots.Select(x => new LazyTreeItem(new WaferHistoryItem() { ID = x.ID, Name = x.Name, StartTime = x.StartTime, EndTime = x.EndTime, Type = WaferHistoryItemType.Lot }, LoadSubItem))); root.IsExpanded = true; HistoryData.Add(root); SelectedItem = lotsItem; } /// /// Notify chart page to prepare datas /// /// void GoToChart(object o) { WaferHistoryItem item = o as WaferHistoryItem; if (o is WaferHistoryLot waferLot) { item.Type = WaferHistoryItemType.Lot; SelectionChanged(item); WaferHistoryItem selectItem = null; foreach (var root in HistoryData) { foreach (var subroot in root.SubItems) { var data = subroot.Data as WaferHistoryItem; if (data != null && data.Name != null && data.Name == item.Name) { selectItem = data; subroot.IsExpanded = true; break; } } if (selectItem != null) break; } } else if (o is WaferHistoryWafer wafer) { item.Type = WaferHistoryItemType.Wafer; //导航切换到chart页面 BaseApp.Instance.SwitchPage("DataLog", "ProcessHistory", wafer); return; } else if (o is WaferHistoryRecipe) { item.Type = WaferHistoryItemType.Recipe; } else if (o is string name) { var query = _recipes.FirstOrDefault(t => t.Recipe == name); if (query is null) return; WaferHistoryWafer wafer1 = new WaferHistoryWafer(); wafer1.StartTime = query.StartTime; wafer1.EndTime = query.EndTime; BaseApp.Instance.SwitchPage("DataLog", "ProcessHistory", wafer1); } else if (o is DateTime startTime) { var query = _recipes.FirstOrDefault(t => t.StartTime.CompareTo(startTime) == 0); if (query is null) return; WaferHistoryWafer wafer1 = new WaferHistoryWafer(); wafer1.StartTime = query.StartTime; wafer1.EndTime = query.EndTime; BaseApp.Instance.SwitchPage("DataLog", "ProcessHistory", wafer1); } // //WaferHistoryRecipe chartQuery = null; //if (o is string name) //{ // var query = _recipes.FirstOrDefault(t => t.Recipe == name); // if (query is null) return; // chartQuery = query; //} //if (o is WaferHistoryLot waferLot) //{ // DateTime start = waferLot.StartTime; // double.TryParse(waferLot.Duration, out double duration); // DateTime end = start.AddSeconds(duration); // chartQuery = new WaferHistoryRecipe() { StartTime = start, EndTime = end, /*Chamber = waferLot.CarrierID */}; //} //else if (o is WaferHistoryWafer wafer) //{ // chartQuery = new WaferHistoryRecipe() { StartTime = wafer.StartTime, EndTime = wafer.EndTime/*,Chamber = waferLot.CarrierID*/ }; //} //导航切换到chart页面 //BaseApp.Instance.SwitchPage("DataLog", "DataHistory", chartQuery); } public List QueryLot(DateTime from, DateTime to, string key) { List result = new List(); 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}' "; if (!string.IsNullOrWhiteSpace(key)) sql += $" and lower(\"name\") like '%{key.ToLower()}%'"; sql += "order by \"start_time\" ASC;"; //Application.Current.Dispatcher.BeginInvoke(new Action(() => //{ DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql); if (dbData != null && dbData.Rows.Count > 0) { for (int i = 0; i < dbData.Rows.Count; i++) { WaferHistoryLot item = new WaferHistoryLot(); string name = dbData.Rows[i]["name"].ToString(); string time = ""; item.WaferCount = (int)dbData.Rows[i]["total_wafer_count"]; item.ID = dbData.Rows[i]["guid"].ToString(); if (!dbData.Rows[i]["start_time"].Equals(DBNull.Value)) { item.StartTime = ((DateTime)dbData.Rows[i]["start_time"]); time = item.StartTime.ToString("yyyy-MM-dd HH:mm:ss"); } if (!dbData.Rows[i]["end_time"].Equals(DBNull.Value)) { item.EndTime = ((DateTime)dbData.Rows[i]["end_time"]); } item.InputPort = dbData.Rows[i]["input_port"].ToString(); item.Name = $"{name}-{time}"; item.LotID = $"{name}"; result.Add(item); } } //})); return result; } public List QueryLotWafer(WaferHistoryItem whItem) { List result = new List(); //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 )" + // $" LEFT JOIN process_data ON process_data.wafer_data_guid=data.wafer_id"; 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 " + $"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"; Wafers.Clear(); //Application.Current.Dispatcher.BeginInvoke(new Action(() => //{ DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql); Dictionary itemPtr = new Dictionary(); List itemNameList = new List(); if (dbData != null && dbData.Rows.Count > 0) { for (int i = 0; i < dbData.Rows.Count; i++) { WaferHistoryWafer item = new WaferHistoryWafer(); item.ID = dbData.Rows[i]["guid"].ToString(); if (!itemPtr.ContainsKey(item.ID)) { itemPtr.Add(item.ID, 0); } var itemLoadPort = dbData.Rows[i]["create_station"].ToString(); var itemSlot = dbData.Rows[i]["create_slot"].ToString(); //item.CarrierID = dbData.Rows[i]["rfid"].ToString(); item.LotID = dbData.Rows[i]["lot_id"].ToString(); item.SlotID = dbData.Rows[i]["create_slot"].ToString(); var itemWaferID = dbData.Rows[i]["wafer_id"].ToString(); item.Sequence = dbData.Rows[i]["sequence_name"].ToString(); item.Status = dbData.Rows[i]["status"].ToString(); 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;"; DataTable dbDataMovement = QueryDataClient.Instance.Service.QueryData(sql); if (dbDataMovement != null && dbDataMovement.Rows.Count >= 2 && dbDataMovement.Rows.Count > itemPtr[item.ID]) { item.StartTime = (DateTime)dbDataMovement.Rows[itemPtr[item.ID]]["arrive_time"]; for (int j = itemPtr[item.ID]; j < dbDataMovement.Rows.Count; j++) { if (dbDataMovement.Rows[j]["station"].ToString().StartsWith("LP")) { item.EndTime = (DateTime)dbDataMovement.Rows[j]["arrive_time"]; itemPtr[item.ID] = j + 1; break; } } } else { if (!dbData.Rows[i]["create_time"].Equals(DBNull.Value)) { item.StartTime = ((DateTime)dbData.Rows[i]["create_time"]); } if (!dbData.Rows[i]["delete_time"].Equals(DBNull.Value)) item.EndTime = ((DateTime)dbData.Rows[i]["delete_time"]); } item.Name = $"{itemLoadPort} - {itemSlot} - {itemWaferID}"; item.Type = WaferHistoryItemType.Wafer; if (!itemNameList.Contains(item.Name)) { Wafers.Add(item); result.Add(item); itemNameList.Add(item.Name); } } } //})); return result; } public List QueryWaferRecipe(WaferHistoryItem whItem) { List result = new List(); 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;"; Recipes.Clear(); //Application.Current.Dispatcher.BeginInvoke(new Action(() => //{ DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql); if (dbData != null && dbData.Rows.Count > 0) { for (int i = 0; i < dbData.Rows.Count; i++) { WaferHistoryRecipe item = new WaferHistoryRecipe(); item.ID = dbData.Rows[i]["guid"].ToString(); var itemName = dbData.Rows[i]["recipe_name"].ToString(); if (!dbData.Rows[i]["process_begin_time"].Equals(DBNull.Value)) item.StartTime = (DateTime)dbData.Rows[i]["process_begin_time"]; if (dbData.Rows[i].Table.Columns.Contains("process_end_time") && !dbData.Rows[i]["process_end_time"].Equals(DBNull.Value)) item.EndTime = (DateTime)dbData.Rows[i]["process_end_time"]; if (dbData.Rows[i].Table.Columns.Contains("recipe_setting_time") && !dbData.Rows[i]["recipe_setting_time"].Equals(DBNull.Value)) item.SettingTime = dbData.Rows[i]["recipe_setting_time"].ToString(); item.ActualTime = item.Duration; item.Recipe = itemName; item.Chamber = dbData.Rows[i]["process_in"].ToString(); item.Name = itemName; item.Type = WaferHistoryItemType.Recipe; Recipes.Add(item); result.Add(item); } } //})); return result; } public List QueryWaferMovement(WaferHistoryItem whItem) { List result = new List(); string sql = $"SELECT * FROM \"wafer_move_history\" where \"wafer_data_guid\" = '{whItem.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;"; Movements.Clear(); //Application.Current.Dispatcher.BeginInvoke(new Action(() => //{ DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql); if (dbData != null && dbData.Rows.Count > 0) { for (int i = 0; i < dbData.Rows.Count - 1; i++) { WaferHistoryMovement item = new WaferHistoryMovement(); item.Source = $"station : {dbData.Rows[i]["station"]} slot : {dbData.Rows[i]["slot"]}"; item.Destination = $"station : {dbData.Rows[i + 1]["station"]} slot : {dbData.Rows[i + 1]["slot"]}"; item.InTime = dbData.Rows[i]["arrive_time"].ToString(); result.Add(item); Movements.Add(item); } } //})); return result; } public WaferHistoryRecipe QueryRecipe(WaferHistoryItem whItem) { WaferHistoryRecipe result = new WaferHistoryRecipe(); 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}';"; //Application.Current.Dispatcher.BeginInvoke(new Action(() => //{ DataTable dbData = QueryDataClient.Instance.Service.QueryData(sql); if (dbData != null && dbData.Rows.Count > 0) { for (int i = 0; i < dbData.Rows.Count; i++) { WaferHistoryRecipe item = new WaferHistoryRecipe(); item.ID = dbData.Rows[i]["guid"].ToString(); var itemName = dbData.Rows[i]["recipe_name"].ToString(); if (!dbData.Rows[i]["process_begin_time"].Equals(DBNull.Value)) item.StartTime = (DateTime)dbData.Rows[i]["process_begin_time"]; if (!dbData.Rows[i]["process_end_time"].Equals(DBNull.Value)) item.EndTime = (DateTime)dbData.Rows[i]["process_end_time"]; if (dbData.Rows[i].Table.Columns.Contains("recipe_setting_time") && !dbData.Rows[i]["recipe_setting_time"].Equals(DBNull.Value)) item.SettingTime = dbData.Rows[i]["recipe_setting_time"].ToString(); item.ActualTime = item.Duration; item.Recipe = itemName; item.Chamber = dbData.Rows[i]["process_in"].ToString(); item.Name = itemName; item.Type = WaferHistoryItemType.Recipe; result = item; } } 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;"; dbData = QueryDataClient.Instance.Service.QueryData(sql); List steps = new List(); if (dbData != null && dbData.Rows.Count > 0) { for (int i = 0; i < dbData.Rows.Count; i++) { RecipeStep item = new RecipeStep(); item.No = int.Parse(dbData.Rows[i]["step_number"].ToString()); item.Name = dbData.Rows[i]["step_name"].ToString(); if (!dbData.Rows[i]["step_begin_time"].Equals(DBNull.Value)) item.StartTime = (DateTime)dbData.Rows[i]["step_begin_time"]; if (!dbData.Rows[i]["step_end_time"].Equals(DBNull.Value)) item.EndTime = (DateTime)dbData.Rows[i]["step_end_time"]; item.ActualTime = item.EndTime.CompareTo(item.StartTime) <= 0 ? "" : item.EndTime.Subtract(item.StartTime).TotalSeconds.ToString(); item.SettingTime = dbData.Rows[i]["step_time"].ToString(); steps.Add(item); } } 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;"; dbData = QueryDataClient.Instance.Service.QueryData(sql); List fdcs = new List(); if (dbData != null && dbData.Rows.Count > 0) { for (int i = 0; i < dbData.Rows.Count; i++) { RecipeStepFdcData item = new RecipeStepFdcData(); item.StepNumber = int.Parse(dbData.Rows[i]["step_number"].ToString()); item.Name = dbData.Rows[i]["parameter_name"].ToString(); foreach (var name in _mapGasNameKey) { if (item.Name.Contains(name.Value) && _mapGasNameValue.ContainsKey(name.Key)) item.Name = item.Name.Replace(name.Value, (string)_mapGasNameValue[name.Key]); } if (!dbData.Rows[i]["sample_count"].Equals(DBNull.Value)) item.SampleCount = (int)dbData.Rows[i]["sample_count"]; if (!dbData.Rows[i]["setpoint"].Equals(DBNull.Value)) item.SetPoint = (float)dbData.Rows[i]["setpoint"]; if (!dbData.Rows[i]["min_value"].Equals(DBNull.Value)) item.MinValue = (float)dbData.Rows[i]["min_value"]; if (!dbData.Rows[i]["max_value"].Equals(DBNull.Value)) item.MaxValue = (float)dbData.Rows[i]["max_value"]; if (!dbData.Rows[i]["std_value"].Equals(DBNull.Value)) item.StdValue = (float)dbData.Rows[i]["std_value"]; if (!dbData.Rows[i]["mean_value"].Equals(DBNull.Value)) item.MeanValue = (float)dbData.Rows[i]["mean_value"]; fdcs.Add(item); } } result.Steps = steps; result.Fdcs = fdcs; Recipe = result; return result; } private void SelectionChanged(WaferHistoryItem item) { switch (item.Type) { case WaferHistoryItemType.Lot: QueryLotWafer(item); break; case WaferHistoryItemType.Wafer: QueryWaferRecipe(item); QueryWaferMovement(item); break; case WaferHistoryItemType.Recipe: QueryRecipe(item); break; default: break; } SelectedItem = item; } protected List> LoadSubItem(WaferHistoryItem item) { switch (item.Type) { case WaferHistoryItemType.Lot: var wafers = QueryLotWafer(item); return wafers.Select(x => new LazyTreeItem(x, y => LoadSubItem(y))).OrderBy(s => s.Data.StartTime).ToList(); case WaferHistoryItemType.Wafer: var recipes = QueryWaferRecipe(item); return recipes.Select(x => new LazyTreeItem(x, y => LoadSubItem(y))).OrderBy(s => s.Data.StartTime).ToList(); default: break; } return new List> { }; } protected override void OnViewLoaded(object _view) { base.OnViewLoaded(_view); this.view = (WaferHistoryDBView)_view; this.view.wfTimeFrom.Value = this.SearchBeginTime; this.view.wfTimeTo.Value = this.SearchEndTime; QueryLots(new object()); SelectionChanged(SelectedItem); } private Dictionary _mapGasNameValue = new Dictionary(); private Dictionary _mapGasNameKey = new Dictionary() { {"PMA.MfcGas1.GasName", "PMA.IoMfc.MfcGas1"}, {"PMA.MfcGas2.GasName", "PMA.IoMfc.MfcGas2"}, {"PMA.MfcGas3.GasName", "PMA.IoMfc.MfcGas3"}, {"PMA.MfcGas4.GasName", "PMA.IoMfc.MfcGas4"}, {"PMA.MfcGas5.GasName", "PMA.IoMfc.MfcGas5"}, {"PMB.MfcGas1.GasName", "PMB.IoMfc.MfcGas1"}, {"PMB.MfcGas2.GasName", "PMB.IoMfc.MfcGas2"}, {"PMB.MfcGas3.GasName", "PMB.IoMfc.MfcGas3"}, {"PMB.MfcGas4.GasName", "PMB.IoMfc.MfcGas4"}, {"PMB.MfcGas5.GasName", "PMB.IoMfc.MfcGas5"}, }; protected override void OnActivate() { base.OnActivate(); _mapGasNameValue = QueryDataClient.Instance.Service.PollConfig(_mapGasNameKey.Keys); } } public class LazyTreeItem : INotifyPropertyChanged where T : ITreeItem, new() { private LazyTreeItem dummyChild; private T data; private Func>> loader; private LazyTreeItem() { data = new T(); data.ID = Guid.NewGuid().ToString(); } public LazyTreeItem(T data, Func>> loader) { this.data = data; this.loader = loader; dummyChild = new LazyTreeItem(); SubItems = new ObservableCollection>(); SubItems.Add(dummyChild); } public T Data { get { return data; } } public ObservableCollection> SubItems { get; set; } private bool isExpanded; public event PropertyChangedEventHandler PropertyChanged; private void OnPropertyChanged(string name) { if (PropertyChanged != null) { PropertyChanged.Invoke(this, new PropertyChangedEventArgs(name)); } } public bool HasDummyChild { get { return SubItems.Count == 1 && SubItems.First().data.ID == dummyChild.data.ID; } } public bool IsExpanded { get { return isExpanded; } set { if (value != isExpanded) { isExpanded = value; OnPropertyChanged("IsExpanded"); } if (HasDummyChild) { SubItems.Remove(dummyChild); var items = loader(data); items.ForEachDo(x => SubItems.Add(x)); } } } } }