| | |
| | | |
| | | /// <summary> |
| | | /// 更新机器时间并处理首检 |
| | | /// 变更说明: |
| | | /// 1) 最新首检结果为“不合格”时:清空 MA_SHOUT_TIME(送检呼叫时间) |
| | | /// 2) 若不存在首检单时或者首检单结果为“不合格”时:创建首检单 |
| | | /// 3) 首检合格且送检时间 >= 调机开始时间时:写入调机完成时间与开工时间 |
| | | /// 变更说明(新逻辑): |
| | | /// 1) 首检结果“不合格”时不立即重建,不清空送检时间,只提示需要再次送检 |
| | | /// 2) 当用户再次点击“送检呼叫”产生新的送检时间(与数据库原值不同)且上一张首检结果“不合格”,才重建首检单 |
| | | /// 3) 不存在首检单时仍直接创建 |
| | | /// 4) 首检合格且送检时间 >= 调机开始时间时:写入调机完成时间与开工时间 |
| | | /// </summary> |
| | | public bool ChangeMachineTime(MesOrderSta entity) |
| | | { |
| | |
| | | const string FirstCheckResultOK = "合格"; |
| | | const string FirstCheckResultNG = "不合格"; |
| | | |
| | | var womdaa = Db.Queryable<Womdaa>() |
| | | .Where(s => s.Id == entity.OrderId).First(); |
| | | var womdaa = Db.Queryable<Womdaa>().Where(s => s.Id == entity.OrderId).First(); |
| | | if (womdaa == null) throw new Exception("工单不存在"); |
| | | |
| | | // 读取当前状态行 |
| | | var currentSta = Db.Queryable<MesOrderSta>() |
| | | .Where(s => s.Id == entity.Id).First(); |
| | | // 读取现有状态(用于比较送检时间是否变化) |
| | | var dbSta = Db.Queryable<MesOrderSta>().Where(s => s.Id == entity.Id).First(); |
| | | |
| | | // 行级锁:串行化首检重建 |
| | | // 行级锁:避免并发重建 |
| | | Db.Ado.ExecuteCommand("SELECT ID FROM WOMDAA WHERE DAA001 = :BILL_NO FOR UPDATE", |
| | | new SugarParameter("BILL_NO", womdaa.Daa001)); |
| | | |
| | |
| | | .First(); |
| | | |
| | | var latestResult = latestFirst?.FcheckResu?.Trim(); |
| | | var needRebuild = latestFirst == null || string.Equals(latestResult, FirstCheckResultNG, StringComparison.OrdinalIgnoreCase); |
| | | var clearMaShout = false; |
| | | |
| | | // 送检时间是否发生变化(前端重新点击送检) |
| | | var isShoutTimeChanged = !string.IsNullOrEmpty(entity.MaShoutTime) |
| | | && entity.MaShoutTime != dbSta?.MaShoutTime; |
| | | |
| | | // 判定是否需要重建: |
| | | // 1) 首次(latestFirst == null) |
| | | // 2) 上一张是“不合格” 且 用户本次送检时间已发生变化(表示二次送检) |
| | | var needRebuild = latestFirst == null || |
| | | (string.Equals(latestResult, FirstCheckResultNG, StringComparison.OrdinalIgnoreCase) && isShoutTimeChanged); |
| | | |
| | | if (needRebuild) |
| | | { |
| | | // 若已有首检且结果不合格 => 先清空送检时间 |
| | | if (latestFirst != null && string.Equals(latestResult, FirstCheckResultNG, StringComparison.OrdinalIgnoreCase)) |
| | | { |
| | | clearMaShout = true; |
| | | entity.MaShoutTime = null; |
| | | entity.remark = $"最新首检结果“{FirstCheckResultNG}”,已清空送检时间并重建首检单"; |
| | | } |
| | | var previousState = latestResult == null ? "首次创建" : |
| | | (latestResult == FirstCheckResultNG ? "不合格后重建" : "重建"); |
| | | |
| | | // 调用存储过程重建 / 首次创建 |
| | | Db.Ado.ExecuteCommand( |
| | | "BEGIN AUTOMATIC_IPQC_FIRST_CHECK(:BILL_NO); END;", |
| | | new SugarParameter("BILL_NO", womdaa.Daa001, System.Data.DbType.String)); |
| | | |
| | | // 重取最新首检 |
| | | // 重取 |
| | | latestFirst = Db.Queryable<MesQaItemsDetect02>() |
| | | .Where(s => s.Aufnr == womdaa.Daa001 && s.Ftype == FirstCheckType && (s.Fcancel == null || s.Fcancel != "Y")) |
| | | .OrderBy(s => s.CreateDate, OrderByType.Desc) |
| | |
| | | if (latestFirst != null) |
| | | { |
| | | var ts = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"); |
| | | var remarkTag = latestResult == null ? "首次创建" : (latestResult == FirstCheckResultNG ? "不合格重建" : "重建"); |
| | | Db.Updateable<MesQaItemsDetect02>() |
| | | .SetColumns(s => s.Remeke == $"工控机于{ts}自动创建的首检单({remarkTag})") |
| | | .SetColumns(s => s.Remeke == $"工控机于{ts}自动创建的首检单({previousState})") |
| | | .Where(s => s.Id == latestFirst.Id) |
| | | .ExecuteCommand(); |
| | | } |
| | | |
| | | // 重建后不清空本次送检时间(保留用户新输入) |
| | | entity.remark = previousState.Contains("不合格") ? "不合格已重建,等待检验" : entity.remark; |
| | | } |
| | | else |
| | | { |
| | | // 不需要重建但上一张不合格且送检时间未变化 => 提示需要重新送检 |
| | | if (latestFirst != null && string.Equals(latestResult, FirstCheckResultNG, StringComparison.OrdinalIgnoreCase) && !isShoutTimeChanged) |
| | | { |
| | | entity.remark = "上一首检不合格,请重新点击送检呼叫以生成新的首检单"; |
| | | } |
| | | else if (latestFirst != null) |
| | | { |
| | | // 其它情况更新备注为当前结果 |
| | | var ts = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"); |
| | | Db.Updateable<MesQaItemsDetect02>() |
| | | .SetColumns(s => s.Remeke == $"工控机于{ts}首检结果:{latestResult}") |
| | | .Where(s => s.Id == latestFirst.Id) |
| | | .ExecuteCommand(); |
| | | } |
| | | } |
| | | else |
| | | { |
| | | // 不需要重建:如果前端主动清空(传空且原值有内容)也允许清空 |
| | | var requestWantClear = string.IsNullOrEmpty(entity.MaShoutTime) && !string.IsNullOrEmpty(currentSta?.MaShoutTime); |
| | | if (requestWantClear) |
| | | { |
| | | clearMaShout = true; |
| | | entity.MaShoutTime = null; |
| | | entity.remark = "前端请求手动清空送检时间"; |
| | | } |
| | | else |
| | | { |
| | | // 正常备注(保持最新结果说明) |
| | | if (latestFirst != null) |
| | | { |
| | | var ts = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"); |
| | | Db.Updateable<MesQaItemsDetect02>() |
| | | .SetColumns(s => s.Remeke == $"工控机于{ts}首检结果:{latestFirst.FcheckResu}") |
| | | .Where(s => s.Id == latestFirst.Id) |
| | | .ExecuteCommand(); |
| | | } |
| | | } |
| | | } |
| | | |
| | | // 后续报工锚点逻辑保持 |
| | | // 报工锚点逻辑保持 |
| | | var mesReporting = Db.Queryable<MesReporting>() |
| | | .Where(s => s.BillNo == womdaa.Daa001) |
| | | .OrderByDescending(s => s.Id) |
| | |
| | | }; |
| | | Db.Insertable<MesAnchors>(eAnchors).ExecuteCommand(); |
| | | |
| | | // 合格 & 调机逻辑 |
| | | if (!clearMaShout && !string.IsNullOrEmpty(entity.MaShoutTime)) |
| | | // 合格且送检时间有效 => 写调机完成时间与开工时间 |
| | | if (!string.IsNullOrEmpty(entity.MaShoutTime) |
| | | && DateTime.TryParse(entity.MaShoutTime, out var sjTime) |
| | | && DateTime.TryParse(entity.MaStartTime, out var startTime) |
| | | && sjTime >= startTime |
| | | && latestFirst != null |
| | | && string.Equals(latestFirst.FcheckResu?.Trim(), FirstCheckResultOK, StringComparison.OrdinalIgnoreCase)) |
| | | { |
| | | if (DateTime.TryParse(entity.MaShoutTime, out var sjTime) && |
| | | DateTime.TryParse(entity.MaStartTime, out var startTime) && |
| | | sjTime >= startTime) |
| | | QualifiedInspection(new OrderMachineDto |
| | | { |
| | | var sjRecord = latestFirst ?? |
| | | Db.Queryable<MesQaItemsDetect02>() |
| | | .Where(x => x.Aufnr == womdaa.Daa001 && x.Ftype == FirstCheckType && (x.Fcancel == null || x.Fcancel != "Y")) |
| | | .OrderBy(x => x.CreateDate, OrderByType.Desc) |
| | | .First(); |
| | | |
| | | if (sjRecord != null && string.Equals(sjRecord.FcheckResu?.Trim(), FirstCheckResultOK, StringComparison.OrdinalIgnoreCase)) |
| | | { |
| | | QualifiedInspection(new OrderMachineDto |
| | | { |
| | | OrderId = entity.OrderId, |
| | | orderNo = entity.OrderNo, |
| | | machineNo = entity.MachineNo |
| | | }); |
| | | entity.StartTime = entity.MaShoutTime; |
| | | entity.MaEndTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"); |
| | | } |
| | | } |
| | | OrderId = entity.OrderId, |
| | | orderNo = entity.OrderNo, |
| | | machineNo = entity.MachineNo |
| | | }); |
| | | entity.StartTime = entity.MaShoutTime; |
| | | entity.MaEndTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"); |
| | | } |
| | | |
| | | // 最终更新(优先清空) |
| | | var upd = Db.Updateable<MesOrderSta>() |
| | | .SetColumnsIF(clearMaShout, s => s.MaShoutTime == null) |
| | | .SetColumnsIF(!clearMaShout && entity.MaShoutTime != null, s => s.MaShoutTime == entity.MaShoutTime) |
| | | // 更新 |
| | | return Db.Updateable<MesOrderSta>() |
| | | .SetColumnsIF(entity.MaShoutTime != null, s => s.MaShoutTime == entity.MaShoutTime) |
| | | .SetColumnsIF(entity.MaStartTime != null, s => s.MaStartTime == entity.MaStartTime) |
| | | .SetColumnsIF(entity.MaEndTime != null, s => s.MaEndTime == entity.MaEndTime) |
| | | .SetColumnsIF(entity.StartTime != null, s => s.StartTime == entity.StartTime) |
| | | .SetColumnsIF(entity.Flag == 1 && entity.remark != null, s => s.remark == entity.remark) |
| | | .Where(s => s.Id == entity.Id); |
| | | |
| | | return upd.ExecuteCommand() > 0; |
| | | .Where(s => s.Id == entity.Id) |
| | | .ExecuteCommand() > 0; |
| | | } |
| | | |
| | | /// <summary> |