ProductionSchedulingActionService.cs 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683
  1. using System.Globalization;
  2. using Admin.NET.Plugin.AiDOP.Infrastructure;
  3. using Admin.NET.Plugin.AiDOP.Order;
  4. using Admin.NET.Plugin.AiDOP.WorkOrder;
  5. namespace Admin.NET.Plugin.AiDOP.Production;
  6. /// <summary>工单排产动作:生成排程、优先级调整并重检。</summary>
  7. [ApiDescriptionSettings(Order = 264, Description = "工单排产动作")]
  8. [Route("api/Production")]
  9. [AllowAnonymous]
  10. [NonUnify]
  11. public class ProductionSchedulingActionService : IDynamicApiController, ITransient
  12. {
  13. private readonly ISqlSugarClient _db;
  14. private readonly UserManager _userManager;
  15. private readonly OrderResourceCheckService _resourceCheck;
  16. private readonly ProductionScheduleGenerationService _scheduleGen;
  17. private readonly WorkOrderMaterialDetailSyncService _materialDetailSync;
  18. private readonly WorkOrderRoutingSyncService _routingSync;
  19. private readonly AidopActionRunLogWriter _runLog;
  20. public ProductionSchedulingActionService(
  21. ISqlSugarClient db,
  22. UserManager userManager,
  23. OrderResourceCheckService resourceCheck,
  24. ProductionScheduleGenerationService scheduleGen,
  25. WorkOrderMaterialDetailSyncService materialDetailSync,
  26. WorkOrderRoutingSyncService routingSync,
  27. AidopActionRunLogWriter runLog)
  28. {
  29. _db = db;
  30. _userManager = userManager;
  31. _resourceCheck = resourceCheck;
  32. _scheduleGen = scheduleGen;
  33. _materialDetailSync = materialDetailSync;
  34. _routingSync = routingSync;
  35. _runLog = runLog;
  36. }
  37. /// <summary>生成生产排程计划(写入 PeriodSequenceDet)。</summary>
  38. [DisplayName("生成生产排程")]
  39. [HttpPost("scheduling/generate")]
  40. public async Task<object> GenerateSchedule([FromQuery] string domain)
  41. {
  42. var tenantId = ResolveTenantId(domain);
  43. var account = _userManager.Account ?? "system";
  44. var logId = await _runLog.StartAsync("S2_SCHEDULE_GENERATE", tenantId, "WorkOrdMaster", null, $"domain={domain}");
  45. try
  46. {
  47. var result = await _scheduleGen.GenerateAsync(tenantId, domain, account);
  48. await _runLog.SuccessAsync(logId, result.Message, new
  49. {
  50. tenantId,
  51. domain,
  52. result.WorkOrderCount,
  53. result.ScheduleRowCount,
  54. result.UsedWorkCenterCalendar,
  55. result.SkippedWorkOrders
  56. });
  57. return new
  58. {
  59. message = result.Message,
  60. workOrderCount = result.WorkOrderCount,
  61. scheduleRowCount = result.ScheduleRowCount,
  62. usedWorkCenterCalendar = result.UsedWorkCenterCalendar,
  63. skippedWorkOrders = result.SkippedWorkOrders
  64. };
  65. }
  66. catch (Exception ex)
  67. {
  68. await _runLog.FailedAsync(logId, ex.Message, new { tenantId, domain });
  69. throw;
  70. }
  71. }
  72. /// <summary>优先级/数量/交期调整并重做资源检查。</summary>
  73. [DisplayName("优先级调整并重检")]
  74. [HttpPost("scheduling/update-priority-and-recheck")]
  75. public async Task<object> UpdatePriorityAndRecheck([FromBody] WorkOrderPriorityRecheckInput input)
  76. {
  77. var tenantId = AidopTenantHelper.Resolve(App.HttpContext);
  78. var account = string.IsNullOrWhiteSpace(input.UserAccount)
  79. ? (_userManager.Account ?? "system")
  80. : input.UserAccount.Trim();
  81. var workOrd = input.Workord.Trim();
  82. var logId = await _runLog.StartAsync("S2_PRIORITY_RECHECK", tenantId, "WorkOrdMaster", null, workOrd);
  83. try
  84. {
  85. var before = await LoadWorkOrderSnapshotAsync(tenantId, workOrd);
  86. DateTime? dueDate = null;
  87. if (!string.IsNullOrWhiteSpace(input.Instockdate))
  88. dueDate = DateTime.Parse(input.Instockdate.Trim(), CultureInfo.InvariantCulture).Date;
  89. decimal? qty = null;
  90. if (!string.IsNullOrWhiteSpace(input.Qty) && decimal.TryParse(input.Qty.Trim(), NumberStyles.Any, CultureInfo.InvariantCulture, out var q))
  91. qty = q;
  92. var pars = new List<SugarParameter>
  93. {
  94. new("@WorkOrd", workOrd),
  95. new("@TenantId", tenantId),
  96. new("@Priority", string.IsNullOrWhiteSpace(input.Priority) ? (object)DBNull.Value : input.Priority.Trim()),
  97. new("@LotSerial", string.IsNullOrWhiteSpace(input.LotSerial) ? (object)DBNull.Value : input.LotSerial.Trim()),
  98. new("@Qty", qty ?? (object)DBNull.Value),
  99. new("@DueDate", dueDate ?? (object)DBNull.Value),
  100. new("@User", account),
  101. new("@Now", DateTime.Now)
  102. };
  103. var affected = await _db.Ado.ExecuteCommandAsync(
  104. """
  105. UPDATE WorkOrdMaster
  106. SET Priority = COALESCE(@Priority, Priority),
  107. LotSerial = COALESCE(@LotSerial, LotSerial),
  108. QtyOrded = COALESCE(@Qty, QtyOrded),
  109. DueDate = COALESCE(@DueDate, DueDate),
  110. UpdateUser = @User,
  111. UpdateTime = @Now
  112. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  113. """,
  114. pars);
  115. if (affected == 0)
  116. throw Oops.Oh("工单不存在或未更新");
  117. var link = await LoadWorkOrderEntryLinkAsync(tenantId, workOrd);
  118. var warnings = new List<string>();
  119. var resourceRechecked = false;
  120. if (link is not null && (qty.HasValue || dueDate.HasValue))
  121. {
  122. if (qty.HasValue)
  123. link.Entry.Qty = qty;
  124. if (dueDate.HasValue)
  125. link.Entry.SysCapacityDate = dueDate;
  126. await _resourceCheck.RunForEntryAsync(link.Order, link.Entry, workOrd, account, warnings);
  127. resourceRechecked = true;
  128. // 资源检查后同步工单物料明细(WorkOrdDetail)和工艺路线(WorkOrdRouting)
  129. await _materialDetailSync.EnsureFromResourceCheckAsync(tenantId, workOrd, account);
  130. await _routingSync.EnsureFromRoutingAsync(tenantId, workOrd, account);
  131. }
  132. var qtyChanged = qty.HasValue && before?.QtyOrded != qty;
  133. var dueChanged = dueDate.HasValue && before?.DueDate?.Date != dueDate.Value.Date;
  134. var priorityChanged = !string.IsNullOrWhiteSpace(input.Priority)
  135. && !string.Equals(before?.Priority?.Trim(), input.Priority.Trim(), StringComparison.Ordinal);
  136. // 数量变更时同步更新 mes_morder 和 mes_moentry
  137. if (qtyChanged && qty.HasValue)
  138. {
  139. await UpdateMesOrderQuantityAsync(tenantId, workOrd, qty.Value, account);
  140. // 同步更新 WorkOrdRouting.QtyOrded(EnsureFromRoutingAsync 已有数据时不会更新数量)
  141. await _db.Ado.ExecuteCommandAsync(
  142. """
  143. UPDATE WorkOrdRouting
  144. SET QtyOrded = @Qty,
  145. UpdateUser = @User, UpdateTime = @Now
  146. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  147. AND IFNULL(IsActive, 0) = 1
  148. """,
  149. new SugarParameter("@Qty", qty.Value),
  150. new SugarParameter("@User", account),
  151. new SugarParameter("@Now", DateTime.Now),
  152. new SugarParameter("@TenantId", tenantId),
  153. new SugarParameter("@WorkOrd", workOrd));
  154. }
  155. // 数量变更时同步更新领料单明细(NbrDetail)
  156. if (qtyChanged)
  157. {
  158. await SyncPickingListDetailsAsync(tenantId, workOrd, account, warnings);
  159. }
  160. ProductionScheduleGenerationService.ScheduleGenerationResult? reschedule = null;
  161. if (qtyChanged || dueChanged)
  162. {
  163. reschedule = await _scheduleGen.RegenerateForWorkOrderAsync(tenantId, workOrd, input.Domain, account);
  164. warnings.Add(reschedule.Message);
  165. }
  166. else if (priorityChanged)
  167. {
  168. var woDomain = before?.Domain ?? input.Domain;
  169. await _scheduleGen.DeactivateExistingScheduleAsync(tenantId, workOrd, woDomain);
  170. warnings.Add("优先级已变更,原排程已失效,请重新执行批量排程");
  171. }
  172. await _runLog.SuccessAsync(logId, "优先级调整并重检完成", new
  173. {
  174. workOrd,
  175. before,
  176. after = new
  177. {
  178. qty,
  179. dueDate,
  180. priority = input.Priority,
  181. lotSerial = input.LotSerial
  182. },
  183. qtyChanged,
  184. dueChanged,
  185. priorityChanged,
  186. resourceRechecked,
  187. reschedule = reschedule is null
  188. ? null
  189. : new
  190. {
  191. reschedule.WorkOrderCount,
  192. reschedule.ScheduleRowCount,
  193. reschedule.UsedWorkCenterCalendar
  194. },
  195. warnings
  196. });
  197. return new
  198. {
  199. message = "ok",
  200. warnings,
  201. resourceRechecked,
  202. rescheduleTriggered = reschedule is not null,
  203. scheduleRowCount = reschedule?.ScheduleRowCount ?? 0
  204. };
  205. }
  206. catch (Exception ex)
  207. {
  208. await _runLog.FailedAsync(logId, ex.Message, new { workOrd, tenantId });
  209. throw;
  210. }
  211. }
  212. private long ResolveTenantId(string? domain)
  213. {
  214. if (!string.IsNullOrWhiteSpace(domain) && long.TryParse(domain.Trim(), out var tid) && tid > 0)
  215. return tid;
  216. return AidopTenantHelper.Resolve(App.HttpContext);
  217. }
  218. /// <summary>
  219. /// 数量变更时同步更新 mes_morder(制造工单)和 mes_moentry(工单明细)的数量字段。
  220. /// </summary>
  221. private async Task UpdateMesOrderQuantityAsync(long tenantId, string workOrd, decimal newQty, string account)
  222. {
  223. var now = DateTime.Now;
  224. // 更新 mes_morder
  225. await _db.Ado.ExecuteCommandAsync(
  226. """
  227. UPDATE mes_morder
  228. SET need_number = @Qty,
  229. morder_production_number = @Qty,
  230. update_by_name = @User,
  231. update_time = @Now
  232. WHERE morder_no = @MorderNo
  233. AND tenant_id = @TenantId
  234. AND IsDeleted = 0
  235. """,
  236. new SugarParameter("@Qty", newQty),
  237. new SugarParameter("@User", account),
  238. new SugarParameter("@Now", now),
  239. new SugarParameter("@MorderNo", workOrd),
  240. new SugarParameter("@TenantId", tenantId));
  241. // 更新 mes_moentry(remaining_number 仅在未开始生产时才更新)
  242. await _db.Ado.ExecuteCommandAsync(
  243. """
  244. UPDATE mes_moentry
  245. SET need_number = @Qty,
  246. morder_production_number = @Qty,
  247. remaining_number = CASE
  248. WHEN IFNULL(remaining_number, 0) = IFNULL(need_number, 0)
  249. OR IFNULL(morder_production_number, 0) = 0
  250. THEN @Qty
  251. ELSE remaining_number
  252. END,
  253. update_by_name = @User,
  254. update_time = @Now
  255. WHERE moentry_mono = @Mono
  256. AND tenant_id = @TenantId
  257. AND IsDeleted = 0
  258. """,
  259. new SugarParameter("@Qty", newQty),
  260. new SugarParameter("@User", account),
  261. new SugarParameter("@Now", now),
  262. new SugarParameter("@Mono", workOrd),
  263. new SugarParameter("@TenantId", tenantId));
  264. }
  265. /// <summary>
  266. /// 数量变更后同步领料单明细(NbrDetail),按物料逐行比对更新。
  267. /// 规则:
  268. /// 已发料(QtyRec>0) + 新需求>已发料 → 更新 QtyOrd/CurrQtyOpened
  269. /// 已发料(QtyRec>0) + 新需求≤已发料 → 关闭该行
  270. /// 未发料 → 直接修改 QtyOrd/CurrQtyOpened
  271. /// 工单明细有但领料单无 → 新增
  272. /// 领料单有但工单明细无 → 关闭
  273. /// </summary>
  274. private async Task SyncPickingListDetailsAsync(
  275. long tenantId, string workOrd, string account, List<string> warnings)
  276. {
  277. // 获取工单状态
  278. var status = await _db.Ado.GetStringAsync(
  279. """
  280. SELECT IFNULL(LOWER(TRIM(Status)), '') FROM WorkOrdMaster
  281. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  282. LIMIT 1
  283. """,
  284. new SugarParameter("@TenantId", tenantId),
  285. new SugarParameter("@WorkOrd", workOrd));
  286. // 仅处理 下达(R)/投产(W)/暂停(S) 状态的工单
  287. if (status is not ("r" or "w" or "s"))
  288. {
  289. // 初始(P)状态无领料单,无需同步
  290. return;
  291. }
  292. // 查找领料单主记录
  293. var nbrRows = await _db.Ado.SqlQueryAsync<NbrMasterRow>(
  294. """
  295. SELECT RecID, Nbr, `Domain` FROM NbrMaster
  296. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd AND Type = 'SM'
  297. AND IFNULL(TransType, '') = ''
  298. AND IFNULL(IsActive, 0) = 1
  299. """,
  300. new SugarParameter("@TenantId", tenantId),
  301. new SugarParameter("@WorkOrd", workOrd));
  302. if (nbrRows.Count == 0)
  303. {
  304. warnings.Add($"工单 {workOrd} 状态为 {status.ToUpper()} 但无领料单,跳过领料单同步");
  305. return;
  306. }
  307. var now = DateTime.Now;
  308. // 从 WorkOrdDetail 汇总最新物料需求
  309. var details = await _db.Ado.SqlQueryAsync<PickDetailRow>(
  310. """
  311. SELECT
  312. d.ItemNum,
  313. SUM(d.QtyRequired) AS QtyRequired,
  314. MAX(IFNULL(d.UM, im.Um)) AS Unit,
  315. MAX(im.Descr) AS ItemName
  316. FROM WorkOrdDetail d
  317. LEFT JOIN ItemMaster im ON d.ItemNum = im.ItemNum AND im.tenant_id = d.tenant_id
  318. WHERE d.tenant_id = @TenantId AND d.WorkOrd = @WorkOrd AND IFNULL(d.IsActive, 0) = 1
  319. GROUP BY d.ItemNum
  320. HAVING SUM(d.QtyRequired) > 0
  321. ORDER BY d.ItemNum
  322. """,
  323. new SugarParameter("@TenantId", tenantId),
  324. new SugarParameter("@WorkOrd", workOrd));
  325. foreach (var nbr in nbrRows)
  326. {
  327. var domain = string.IsNullOrWhiteSpace(nbr.Domain) ? tenantId.ToString() : nbr.Domain!.Trim();
  328. if (domain.Length > 8) domain = domain[..8];
  329. // 加载当前领料单明细行(仅未关闭的)
  330. var existingDetails = await _db.Ado.SqlQueryAsync<NbrDetailRow>(
  331. """
  332. SELECT RecID, ItemNum, QtyOrd, QtyRec, CurrQtyOpened, Line
  333. FROM NbrDetail
  334. WHERE tenant_id = @TenantId AND Nbr = @Nbr AND Type = 'SM'
  335. AND IFNULL(IsActive, 0) = 1
  336. """,
  337. new SugarParameter("@TenantId", tenantId),
  338. new SugarParameter("@Nbr", nbr.Nbr));
  339. var detailMap = details.ToDictionary(d => d.ItemNum.Trim(), d => d);
  340. var existingMap = existingDetails.ToDictionary(d => (d.ItemNum ?? "").Trim(), d => d);
  341. // ── 1. 处理已有明细行:按 ItemNum 匹配 ──
  342. foreach (var existing in existingDetails)
  343. {
  344. var key = (existing.ItemNum ?? "").Trim();
  345. if (detailMap.TryGetValue(key, out var newDetail))
  346. {
  347. var newQty = newDetail.QtyRequired;
  348. if (existing.QtyRec > 0)
  349. {
  350. if (newQty > existing.QtyRec)
  351. {
  352. // 新需求 > 已发料 → 更新需求数
  353. await _db.Ado.ExecuteCommandAsync(
  354. """
  355. UPDATE NbrDetail
  356. SET QtyOrd = @QtyOrd, CurrQtyOpened = @CurrQtyOpened,
  357. UM = @UM, ItemName = @ItemName,
  358. UpdateUser = @User, UpdateTime = @Now
  359. WHERE RecID = @RecId
  360. """,
  361. new SugarParameter("@QtyOrd", newQty),
  362. new SugarParameter("@CurrQtyOpened", newQty),
  363. new SugarParameter("@UM", (object?)newDetail.Unit ?? DBNull.Value),
  364. new SugarParameter("@ItemName", (object?)newDetail.ItemName ?? DBNull.Value),
  365. new SugarParameter("@User", account),
  366. new SugarParameter("@Now", now),
  367. new SugarParameter("@RecId", existing.RecID));
  368. }
  369. else
  370. {
  371. // 新需求 <= 已发料 → 关闭当前行
  372. await _db.Ado.ExecuteCommandAsync(
  373. """
  374. UPDATE NbrDetail
  375. SET Status = 'C', IsActive = 0,
  376. UpdateUser = @User, UpdateTime = @Now
  377. WHERE RecID = @RecId
  378. """,
  379. new SugarParameter("@User", account),
  380. new SugarParameter("@Now", now),
  381. new SugarParameter("@RecId", existing.RecID));
  382. }
  383. }
  384. else
  385. {
  386. // 未发料 → 直接修改需求数
  387. await _db.Ado.ExecuteCommandAsync(
  388. """
  389. UPDATE NbrDetail
  390. SET QtyOrd = @QtyOrd, CurrQtyOpened = @CurrQtyOpened,
  391. UM = @UM, ItemName = @ItemName,
  392. UpdateUser = @User, UpdateTime = @Now
  393. WHERE RecID = @RecId
  394. """,
  395. new SugarParameter("@QtyOrd", newQty),
  396. new SugarParameter("@CurrQtyOpened", newQty),
  397. new SugarParameter("@UM", (object?)newDetail.Unit ?? DBNull.Value),
  398. new SugarParameter("@ItemName", (object?)newDetail.ItemName ?? DBNull.Value),
  399. new SugarParameter("@User", account),
  400. new SugarParameter("@Now", now),
  401. new SugarParameter("@RecId", existing.RecID));
  402. }
  403. }
  404. else
  405. {
  406. // 物料明细中没有,但领料单明细有 → 关闭当前领料单明细行
  407. await _db.Ado.ExecuteCommandAsync(
  408. """
  409. UPDATE NbrDetail
  410. SET Status = 'C', IsActive = 0,
  411. UpdateUser = @User, UpdateTime = @Now
  412. WHERE RecID = @RecId
  413. """,
  414. new SugarParameter("@User", account),
  415. new SugarParameter("@Now", now),
  416. new SugarParameter("@RecId", existing.RecID));
  417. }
  418. }
  419. // ── 2. 新增:物料明细中有,领料单明细没有的 ──
  420. var nextDetailId = await _db.Ado.GetIntAsync("SELECT IFNULL(MAX(RecID), 0) + 1 FROM NbrDetail");
  421. short newLine = (short)(existingDetails.Count > 0 ? existingDetails.Max(d => d.Line) + 1 : 1);
  422. foreach (var d in details)
  423. {
  424. var key = d.ItemNum.Trim();
  425. if (existingMap.ContainsKey(key))
  426. continue;
  427. await _db.Ado.ExecuteCommandAsync(
  428. """
  429. INSERT INTO NbrDetail (
  430. RecID, `Domain`, Type, Nbr, Line, ItemNum, Dimension1, Dimension2,
  431. LocationFrom, LocationTo, QtyFrom, QtyTo, QtyOrd, QtyRec,
  432. CurrQtyOpened, UM, WorkOrd, ItemName, Status,
  433. IsActive, IsConfirm, IsChanged, BusinessID, NbrRecID,
  434. CreateUser, CreateTime, UpdateUser, UpdateTime, tenant_id
  435. ) VALUES (
  436. @RecId, @Domain, 'SM', @Nbr, @Line, @ItemNum, '', '',
  437. '', '', 0, 0, @QtyOrd, 0,
  438. @CurrQtyOpened, @UM, @WorkOrd, @ItemName, '',
  439. 1, 0, 1, 0, @NbrRecId,
  440. @User, @Now, @User, @Now, @TenantId
  441. )
  442. """,
  443. new SugarParameter("@RecId", nextDetailId++),
  444. new SugarParameter("@Domain", domain),
  445. new SugarParameter("@Nbr", nbr.Nbr),
  446. new SugarParameter("@Line", newLine++),
  447. new SugarParameter("@ItemNum", d.ItemNum),
  448. new SugarParameter("@QtyOrd", d.QtyRequired),
  449. new SugarParameter("@CurrQtyOpened", d.QtyRequired),
  450. new SugarParameter("@UM", (object?)d.Unit ?? DBNull.Value),
  451. new SugarParameter("@WorkOrd", workOrd),
  452. new SugarParameter("@ItemName", (object?)d.ItemName ?? DBNull.Value),
  453. new SugarParameter("@NbrRecId", nbr.RecID),
  454. new SugarParameter("@User", account.Length > 24 ? account[..24] : account),
  455. new SugarParameter("@Now", now),
  456. new SugarParameter("@TenantId", tenantId));
  457. }
  458. // ── 3. 更新领料单主记录的更新时间和数量 ──
  459. await _db.Ado.ExecuteCommandAsync(
  460. """
  461. UPDATE NbrMaster
  462. SET QtyOrd = (SELECT IFNULL(SUM(QtyOrd), 0) FROM NbrDetail
  463. WHERE Nbr = @Nbr AND Type = 'SM' AND IFNULL(IsActive, 1) = 1),
  464. UpdateUser = @User, UpdateTime = @Now
  465. WHERE RecID = @RecId
  466. """,
  467. new SugarParameter("@Nbr", nbr.Nbr),
  468. new SugarParameter("@User", account),
  469. new SugarParameter("@Now", now),
  470. new SugarParameter("@RecId", nbr.RecID));
  471. }
  472. }
  473. // ══════════════════════════════════════════════════════════════
  474. // 内部 DTO
  475. // ══════════════════════════════════════════════════════════════
  476. private sealed class NbrMasterRow
  477. {
  478. public int RecID { get; set; }
  479. public string Nbr { get; set; } = string.Empty;
  480. public string? Domain { get; set; }
  481. }
  482. private sealed class NbrDetailRow
  483. {
  484. public int RecID { get; set; }
  485. public string? ItemNum { get; set; }
  486. public decimal QtyOrd { get; set; }
  487. public decimal QtyRec { get; set; }
  488. public decimal CurrQtyOpened { get; set; }
  489. public short Line { get; set; }
  490. }
  491. private sealed class PickDetailRow
  492. {
  493. public string ItemNum { get; set; } = string.Empty;
  494. public decimal QtyRequired { get; set; }
  495. public string? Unit { get; set; }
  496. public string? ItemName { get; set; }
  497. }
  498. private async Task<WorkOrderSnapshotRow?> LoadWorkOrderSnapshotAsync(long tenantId, string workOrd)
  499. {
  500. var rows = await _db.Ado.SqlQueryAsync<WorkOrderSnapshotRow>(
  501. """
  502. SELECT WorkOrd, Priority, QtyOrded, DueDate, LotSerial, `Domain`
  503. FROM WorkOrdMaster
  504. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  505. LIMIT 1
  506. """,
  507. new SugarParameter("@TenantId", tenantId),
  508. new SugarParameter("@WorkOrd", workOrd));
  509. return rows.FirstOrDefault();
  510. }
  511. private async Task<WorkOrderEntryLink?> LoadWorkOrderEntryLinkAsync(long tenantId, string workOrd)
  512. {
  513. var rows = await _db.Ado.SqlQueryAsync<WorkOrderEntryLinkRow>(
  514. """
  515. SELECT
  516. w.BusinessID AS EntryId,
  517. e.seorder_id AS SeOrderId,
  518. e.bill_no AS BillNo,
  519. e.entry_seq AS EntrySeq,
  520. e.item_number AS ItemNumber,
  521. e.item_name AS ItemName,
  522. e.specification AS Specification,
  523. e.unit AS Unit,
  524. e.bom_number AS BomNumber,
  525. e.qty AS Qty,
  526. e.plan_date AS PlanDate,
  527. e.sys_capacity_date AS SysCapacityDate,
  528. e.progress AS Progress,
  529. e.urgent AS Urgent,
  530. e.factory_id AS FactoryId,
  531. e.company_id AS CompanyId,
  532. e.tenant_id AS TenantId,
  533. o.Id AS OrderId,
  534. o.bill_no AS OrderBillNo,
  535. o.custom_no AS CustomNo,
  536. o.urgent AS OrderUrgent,
  537. o.factory_id AS OrderFactoryId
  538. FROM WorkOrdMaster w
  539. INNER JOIN crm_seorderentry e ON e.Id = w.BusinessID AND e.tenant_id = w.tenant_id AND e.IsDeleted = 0
  540. INNER JOIN crm_seorder o ON o.Id = e.seorder_id AND o.tenant_id = e.tenant_id AND o.IsDeleted = 0
  541. WHERE w.tenant_id = @TenantId AND w.WorkOrd = @WorkOrd
  542. LIMIT 1
  543. """,
  544. new SugarParameter("@TenantId", tenantId),
  545. new SugarParameter("@WorkOrd", workOrd));
  546. var row = rows.FirstOrDefault();
  547. if (row is null || row.EntryId <= 0)
  548. return null;
  549. return new WorkOrderEntryLink
  550. {
  551. Order = new OrderWorkOrderGenerationService.OrderHeader
  552. {
  553. Id = row.OrderId,
  554. BillNo = row.OrderBillNo ?? row.BillNo,
  555. CustomNo = row.CustomNo,
  556. Urgent = row.OrderUrgent,
  557. FactoryId = row.OrderFactoryId,
  558. TenantId = tenantId
  559. },
  560. Entry = new OrderWorkOrderGenerationService.OrderEntryLine
  561. {
  562. Id = row.EntryId,
  563. SeOrderId = row.SeOrderId,
  564. BillNo = row.BillNo,
  565. EntrySeq = row.EntrySeq,
  566. ItemNumber = row.ItemNumber,
  567. ItemName = row.ItemName,
  568. Specification = row.Specification,
  569. Unit = row.Unit,
  570. BomNumber = row.BomNumber,
  571. Qty = row.Qty,
  572. PlanDate = row.PlanDate,
  573. SysCapacityDate = row.SysCapacityDate,
  574. Progress = row.Progress,
  575. Urgent = row.Urgent,
  576. FactoryId = row.FactoryId,
  577. CompanyId = row.CompanyId,
  578. TenantId = row.TenantId
  579. }
  580. };
  581. }
  582. private sealed class WorkOrderSnapshotRow
  583. {
  584. public string? WorkOrd { get; set; }
  585. public string? Priority { get; set; }
  586. public decimal? QtyOrded { get; set; }
  587. public DateTime? DueDate { get; set; }
  588. public string? LotSerial { get; set; }
  589. public string? Domain { get; set; }
  590. }
  591. private sealed class WorkOrderEntryLink
  592. {
  593. public OrderWorkOrderGenerationService.OrderHeader Order { get; set; } = new();
  594. public OrderWorkOrderGenerationService.OrderEntryLine Entry { get; set; } = new();
  595. }
  596. private sealed class WorkOrderEntryLinkRow
  597. {
  598. public long EntryId { get; set; }
  599. public long SeOrderId { get; set; }
  600. public string? BillNo { get; set; }
  601. public int? EntrySeq { get; set; }
  602. public string? ItemNumber { get; set; }
  603. public string? ItemName { get; set; }
  604. public string? Specification { get; set; }
  605. public string? Unit { get; set; }
  606. public string? BomNumber { get; set; }
  607. public decimal? Qty { get; set; }
  608. public DateTime? PlanDate { get; set; }
  609. public DateTime? SysCapacityDate { get; set; }
  610. public string? Progress { get; set; }
  611. public int? Urgent { get; set; }
  612. public long? FactoryId { get; set; }
  613. public long? CompanyId { get; set; }
  614. public long TenantId { get; set; }
  615. public long OrderId { get; set; }
  616. public string? OrderBillNo { get; set; }
  617. public string? CustomNo { get; set; }
  618. public int? OrderUrgent { get; set; }
  619. public long? OrderFactoryId { get; set; }
  620. }
  621. }
  622. public class WorkOrderPriorityRecheckInput
  623. {
  624. public string Workord { get; set; } = string.Empty;
  625. public string? Qty { get; set; }
  626. public string? Instockdate { get; set; }
  627. public string? Priority { get; set; }
  628. public string Domain { get; set; } = string.Empty;
  629. public string? UserAccount { get; set; }
  630. public string? LotSerial { get; set; }
  631. }