ProcessOutsourceOrderService.cs 40 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910
  1. namespace Admin.NET.Plugin.AiDOP.Supply;
  2. /// <summary>
  3. /// 工序外协订单服务
  4. /// </summary>
  5. [ApiDescriptionSettings(Order = 311, Description = "工序外协订单")]
  6. [Route("api/Supply")]
  7. [AllowAnonymous]
  8. [NonUnify]
  9. public class ProcessOutsourceOrderService : IDynamicApiController, ITransient
  10. {
  11. private const int PwPurOrdSerialWidth = 4;
  12. private readonly ISqlSugarClient _db;
  13. private readonly UserManager _userManager;
  14. public ProcessOutsourceOrderService(ISqlSugarClient db, UserManager userManager)
  15. {
  16. _db = db;
  17. _userManager = userManager;
  18. }
  19. [DisplayName("工序外协订单列表")]
  20. [HttpGet("process-outsource-order/list")]
  21. public async Task<object> GetList([FromQuery] ProcessOutsourceOrderListInput input)
  22. {
  23. var page = input.Page <= 0 ? 1 : input.Page;
  24. var pageSize = input.PageSize <= 0 ? 10 : input.PageSize;
  25. var offset = (page - 1) * pageSize;
  26. var where = new List<string> { "p.Potype='PW'", "p.IsActive<=1", "IFNULL(p.`Usage`,'')='外协'" };
  27. var pars = new List<SugarParameter>();
  28. if (_userManager.TenantId > 0)
  29. {
  30. where.Add("p.tenant_id = @TenantId");
  31. pars.Add(new SugarParameter("@TenantId", _userManager.TenantId));
  32. }
  33. if (!string.IsNullOrWhiteSpace(input.PurOrd))
  34. {
  35. where.Add("p.PurOrd LIKE @PurOrd");
  36. pars.Add(new SugarParameter("@PurOrd", $"%{input.PurOrd.Trim()}%"));
  37. }
  38. if (!string.IsNullOrWhiteSpace(input.WorkOrd))
  39. {
  40. where.Add("p.WorkOrd LIKE @WorkOrd");
  41. pars.Add(new SugarParameter("@WorkOrd", $"%{input.WorkOrd.Trim()}%"));
  42. }
  43. if (!string.IsNullOrWhiteSpace(input.Buyer))
  44. {
  45. where.Add("p.Buyer=@Buyer");
  46. pars.Add(new SugarParameter("@Buyer", input.Buyer.Trim()));
  47. }
  48. if (!string.IsNullOrWhiteSpace(input.Supp))
  49. {
  50. where.Add("p.Supp=@Supp");
  51. pars.Add(new SugarParameter("@Supp", input.Supp.Trim()));
  52. }
  53. var baseSql = $"""
  54. SELECT
  55. p.RecID AS Id,
  56. p.PurOrd AS PurOrd,
  57. p.WorkOrd AS WorkOrd,
  58. p.ERPWorkOrd AS ERPWorkOrd,
  59. CONCAT(TRIM(IFNULL(p.Supp,'')),' ',TRIM(IFNULL(s.SortName,''))) AS SuppName,
  60. p.Buyer AS BuyerCode,
  61. CONCAT(TRIM(IFNULL(p.Buyer,'')),' ',TRIM(IFNULL(e.Name,''))) AS Buyer,
  62. f.ItemNum AS ItemNum,
  63. i.Descr AS ItemName,
  64. CONCAT(TRIM(CAST(IFNULL(r.Op,0) AS CHAR(20))),' ',TRIM(IFNULL(r.Descr,''))) AS Op,
  65. p.OrdDate AS OrdDate,
  66. f.QtyOrded AS QtyOrded,
  67. f.CumQtyBO AS CumQtyBO,
  68. f.DueDate AS DueDate,
  69. n.`Date` AS FlDate,
  70. CASE
  71. WHEN IFNULL(LENGTH(p.Status),0)=0 OR p.Buyer IS NULL THEN 'R'
  72. ELSE p.Status
  73. END AS Status,
  74. p.Supp AS Supp,
  75. p.ReqBy AS ReqBy,
  76. p.Department AS Department,
  77. p.`Usage` AS `Usage`,
  78. p.Curr AS Curr,
  79. p.Remark AS Remark
  80. FROM PurOrdMaster p
  81. LEFT JOIN SuppMaster s ON p.Supp=s.Supp
  82. LEFT JOIN DepartmentMaster d ON p.Department=d.Department
  83. LEFT JOIN EmployeeMaster e ON p.Buyer=e.Employee
  84. LEFT JOIN PurOrdDetail f ON p.PurOrd=f.PurOrd
  85. LEFT JOIN RoutingOpDetail r ON f.Op=r.Op AND f.ItemNum=r.RoutingCode
  86. LEFT JOIN ItemMaster i ON f.ItemNum=i.ItemNum
  87. LEFT JOIN PurOrdRctMaster o ON o.OrdNbr=p.PurOrd
  88. LEFT JOIN NbrMaster n ON p.WorkOrd=n.WorkOrd AND n.Type='CA'
  89. WHERE {string.Join(" AND ", where)}
  90. """;
  91. var outerWhere = string.Empty;
  92. if (!string.IsNullOrWhiteSpace(input.Status))
  93. {
  94. outerWhere = "WHERE t.Status=@Status";
  95. pars.Add(new SugarParameter("@Status", input.Status.Trim()));
  96. }
  97. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) FROM ({baseSql}) t {outerWhere}", pars);
  98. var list = await _db.Ado.SqlQueryAsync<ProcessOutsourceOrderListRow>(
  99. $"SELECT * FROM ({baseSql}) t {outerWhere} ORDER BY {BuildOrderBy(input.SortField, input.SortOrder)} LIMIT {pageSize} OFFSET {offset}",
  100. pars);
  101. return new { total, page, pageSize, list };
  102. }
  103. [DisplayName("工序外协订单详情")]
  104. [HttpGet("process-outsource-order/{id:int}")]
  105. public async Task<object> GetDetail(int id)
  106. {
  107. var tenantWhere = _userManager.TenantId > 0 ? " AND p.tenant_id=@TenantId" : string.Empty;
  108. var masterPars = _userManager.TenantId > 0
  109. ? new[] { new SugarParameter("@Id", id), new SugarParameter("@TenantId", _userManager.TenantId) }
  110. : new[] { new SugarParameter("@Id", id) };
  111. var master = (await _db.Ado.SqlQueryAsync<ProcessOutsourceOrderMasterRow>(
  112. $"""
  113. SELECT
  114. p.RecID AS Id,p.PurOrd AS PurOrd,p.OrdDate AS OrdDate,p.Supp AS Supp,p.ReqBy AS ReqBy,p.WorkOrd AS WorkOrd,
  115. p.Buyer AS Buyer,p.Department AS Department,p.`Usage` AS `Usage`,p.Curr AS Curr,p.Remark AS Remark,
  116. CASE WHEN IFNULL(LENGTH(p.Status),0)=0 OR p.Buyer IS NULL THEN 'R' ELSE p.Status END AS Status
  117. FROM PurOrdMaster p
  118. WHERE p.RecID=@Id{tenantWhere}
  119. LIMIT 1
  120. """,
  121. masterPars)).FirstOrDefault();
  122. if (master == null) throw Oops.Oh("记录不存在");
  123. var details = await _db.Ado.SqlQueryAsync<ProcessOutsourceOrderDetailRow>(
  124. """
  125. SELECT
  126. d.RecID AS Id,
  127. d.Line AS Line,
  128. d.ItemNum AS ItemNum,
  129. IFNULL(i.Descr,'') AS ItemName,
  130. d.Op AS Op,
  131. d.UM AS UM,
  132. d.Location AS Location,
  133. d.QtyOrded AS QtyOrded,
  134. d.QtyReceived AS QtyReceived,
  135. d.ReceiptQty AS ReceiptQty,
  136. d.DueDate AS DueDate,
  137. d.LotSerial AS LotSerial,
  138. d.Potype AS Potype,
  139. d.PurOrd AS PurOrd,
  140. d.PurOrdRecID AS PurOrdRecID
  141. FROM PurOrdDetail d
  142. LEFT JOIN ItemMaster i ON d.ItemNum=i.ItemNum
  143. WHERE d.PurOrdRecID=@PurOrdRecID
  144. ORDER BY d.Line,d.RecID
  145. """,
  146. new SugarParameter("@PurOrdRecID", id));
  147. return new { master, details };
  148. }
  149. [DisplayName("按工单生成工序外协订单")]
  150. [HttpPost("process-outsource-order/create-by-workord")]
  151. public async Task<object> CreateByWorkOrd([FromBody] ProcessOutsourceOrderCreateInput input)
  152. {
  153. if (string.IsNullOrWhiteSpace(input.WorkOrd)) throw Oops.Oh("请选择工单");
  154. var workOrd = input.WorkOrd.Trim();
  155. // 先筛出“当前工单且未生成采购单”的外协供应商清单(等价于存储过程内 @v_temp_supp)。
  156. var suppliers = await _db.Ado.SqlQueryAsync<SuppRow>(
  157. """
  158. SELECT DISTINCT d.SupplierCode AS Supp
  159. FROM WorkOrdMaster m
  160. LEFT JOIN WorkOrdRouting r ON m.WorkOrd=r.WorkOrd
  161. LEFT JOIN RoutingOpDetail d ON r.Op=d.Op AND r.ItemNum=d.RoutingCode
  162. LEFT JOIN PurOrdMaster p ON m.WorkOrd=p.WorkOrd AND d.SupplierCode=p.Supp
  163. WHERE r.ProcessOut=1
  164. AND m.WorkOrd=@WorkOrd
  165. AND p.RecID IS NULL
  166. AND IFNULL(TRIM(d.SupplierCode),'')<>''
  167. ORDER BY d.SupplierCode
  168. """,
  169. new SugarParameter("@WorkOrd", workOrd));
  170. if (suppliers.Count == 0)
  171. throw Oops.Oh("当前工单无需生成外协采购订单。");
  172. var domain = await _db.Ado.GetStringAsync(
  173. """
  174. SELECT IFNULL(TRIM(Domain),'')
  175. FROM WorkOrdMaster
  176. WHERE WorkOrd=@WorkOrd
  177. LIMIT 1
  178. """,
  179. new SugarParameter("@WorkOrd", workOrd)) ?? string.Empty;
  180. var created = new List<IdPurOrdRow>();
  181. try
  182. {
  183. _db.Ado.BeginTran();
  184. var purOrdNumbers = await PurOrdNumberGenerator.NextBatchAsync(
  185. _db, "PW", DateTime.Today, PwPurOrdSerialWidth, suppliers.Count);
  186. for (var i = 0; i < suppliers.Count; i++)
  187. {
  188. var supp = suppliers[i].Supp!.Trim();
  189. var purOrd = purOrdNumbers[i].Trim();
  190. var now = DateTime.Now;
  191. await _db.Ado.ExecuteCommandAsync(
  192. """
  193. INSERT INTO PurOrdMaster
  194. (
  195. Confirming, CreditTermsInt, Disc, ExchRate, EstVal, ExchRate1, ExchRate2,
  196. FixedPrice, FixedRate, Frt, PartialOK, AmtPrepaid, PrintPO, PST, Recurr,
  197. `Release`, Revision, Scheduled, ServiceCharge, SpecialCharge, Taxable,
  198. Tax1, Tax2, Tax3, TransportDays, IsActive, IsConfirm, Potype, IsChanged,
  199. TaxIn, Amt, IsPriceChanged,
  200. Buyer, Domain, PurOrd, OrdDate, ReqBy, Status, Supp, CreateUser, CreateTime,
  201. Department, `Usage`, FSTID, ERPWorkOrd, WorkOrd, tenant_id
  202. )
  203. SELECT
  204. 0, 0, 0, 1, 0, 1, 1,
  205. 0, 0, 0, 1, 0, 0, 0, 0,
  206. 0, 0, 0, 0, 0, 1,
  207. 0, 0, 0, 0, 1, 1, 'PW', 0,
  208. 1, 0, 0,
  209. @Buyer,
  210. IFNULL(TRIM(m.Domain),''),
  211. @PurOrd,
  212. @Now,
  213. 'PO',
  214. '',
  215. @Supp,
  216. @CreateUser,
  217. @Now,
  218. m.Department,
  219. '外协',
  220. 1,
  221. m.Batch,
  222. m.WorkOrd,
  223. @TenantId
  224. FROM WorkOrdMaster m
  225. WHERE m.WorkOrd=@WorkOrd
  226. LIMIT 1
  227. """,
  228. new SugarParameter("@Buyer", _userManager.Account),
  229. new SugarParameter("@PurOrd", purOrd),
  230. new SugarParameter("@Now", now),
  231. new SugarParameter("@Supp", supp),
  232. new SugarParameter("@CreateUser", _userManager.Account),
  233. new SugarParameter("@WorkOrd", workOrd),
  234. new SugarParameter("@TenantId", _userManager.TenantId <= 0 ? null : _userManager.TenantId));
  235. var masterId = await _db.Ado.GetIntAsync(
  236. "SELECT IFNULL(MAX(RecID),0) FROM PurOrdMaster WHERE PurOrd=@PurOrd",
  237. new SugarParameter("@PurOrd", purOrd));
  238. if (masterId <= 0)
  239. throw Oops.Oh("采购订单生成失败。");
  240. var lines = await _db.Ado.SqlQueryAsync<CreateLineRow>(
  241. """
  242. SELECT
  243. IFNULL(i.PMBOM,'') AS PMBOM,
  244. IFNULL(r.Op,0) AS Op,
  245. r.ItemNum AS ItemNum,
  246. IFNULL(i.UM,'') AS UM,
  247. IFNULL(m.QtyOrded,0) AS QtyOrded,
  248. IFNULL(i.Rev,'') AS Rev,
  249. IFNULL(m.Department,'') AS Department,
  250. m.WorkOrd AS WorkOrd,
  251. IFNULL(i.Drawing,'') AS Drawing,
  252. IFNULL(m.RecID,0) AS WorkOrdID,
  253. IFNULL(d.OutsourcedLeadTime,0) AS ProcessOutDay
  254. FROM WorkOrdMaster m
  255. LEFT JOIN WorkOrdRouting r ON m.WorkOrd=r.WorkOrd
  256. LEFT JOIN RoutingOpDetail d ON r.Op=d.Op AND r.ItemNum=d.RoutingCode
  257. LEFT JOIN ItemMaster i ON r.ItemNum=i.ItemNum
  258. WHERE r.ProcessOut=1
  259. AND m.WorkOrd=@WorkOrd
  260. AND d.SupplierCode=@Supp
  261. ORDER BY r.RecID
  262. """,
  263. new SugarParameter("@WorkOrd", workOrd),
  264. new SugarParameter("@Supp", supp));
  265. for (var lineIndex = 0; lineIndex < lines.Count; lineIndex++)
  266. {
  267. var row = lines[lineIndex];
  268. await _db.Ado.ExecuteCommandAsync(
  269. """
  270. INSERT INTO PurOrdDetail
  271. (
  272. QtyBO, RctCost, CreditTermsInt, UpdateCurrentCost, CumReceived1, CumReceived2,
  273. CumReceived3, CumReceived4, Disc, FixedPrice, InspectReq, SingleLot, SupplyPer,
  274. PurOrd, PST, PackingSlipQty, PayUMConv, PurCost, RctQty, QtyOrded, QtyReceived,
  275. QtyReturned, Active, QtyReleased, RctUMConversion, Scheduled, ScheduledChanged,
  276. SchedMRPReq, SafetyDays, SafetyHours, StdCost, Taxable, TaxIn, MaxTaxableAmt,
  277. TransportHours, UMConversion, VAT, IsActive, IsConfirm, Potype, IsChanged,
  278. TaxRate, IsRounding, ReceiptQty, BarCodeQty, IsClosed, QtyReturnedRefund, CumQtyBO,
  279. ActiveRlseID1, Domain, DueDate, Line, Location, Op, ItemNum, Status, UM, WorkOrdID,
  280. CreateUser, CreateTime, `Usage`, Rev, Department, WorkOrd, Drawing, PurOrdRecID, tenant_id
  281. )
  282. VALUES
  283. (
  284. 0, 0, 0, 0, 0, 0,
  285. 0, 0, 0, 0, 0, 0, 0,
  286. @PurOrd, 0, 0, 1, 0, 0, @QtyOrded, 0,
  287. 0, 1, 0, 1, 0, 0,
  288. 0, 0, 0, 0, 1, 1, 0,
  289. 0, 1, 0, 1, 1, 'PW', 0,
  290. 0, 0, 0, 0, 0, 0, 0,
  291. @ActiveRlseID1, @Domain, DATE_ADD(@Now, INTERVAL @ProcessOutDay DAY), @Line, '5008', @Op, @ItemNum, '', @UM, @WorkOrdID,
  292. @CreateUser, @Now, '外协', @Rev, @Department, @WorkOrd, @Drawing, @PurOrdRecID, @TenantId
  293. )
  294. """,
  295. new SugarParameter("@ActiveRlseID1", row.PMBOM),
  296. new SugarParameter("@Domain", domain),
  297. new SugarParameter("@Now", now),
  298. new SugarParameter("@ProcessOutDay", row.ProcessOutDay),
  299. new SugarParameter("@Line", lineIndex + 1),
  300. new SugarParameter("@PurOrd", purOrd),
  301. new SugarParameter("@Op", row.Op),
  302. new SugarParameter("@ItemNum", row.ItemNum),
  303. new SugarParameter("@QtyOrded", row.QtyOrded),
  304. new SugarParameter("@UM", row.UM),
  305. new SugarParameter("@WorkOrdID", row.WorkOrdID),
  306. new SugarParameter("@CreateUser", _userManager.Account),
  307. new SugarParameter("@Rev", row.Rev),
  308. new SugarParameter("@Department", row.Department),
  309. new SugarParameter("@WorkOrd", row.WorkOrd),
  310. new SugarParameter("@Drawing", row.Drawing),
  311. new SugarParameter("@PurOrdRecID", masterId),
  312. new SugarParameter("@TenantId", _userManager.TenantId <= 0 ? null : _userManager.TenantId));
  313. }
  314. await UpsertProcessOutsourceMdpAsync(purOrd, now);
  315. created.Add(new IdPurOrdRow { Id = masterId, PurOrd = purOrd });
  316. }
  317. _db.Ado.CommitTran();
  318. }
  319. catch
  320. {
  321. _db.Ado.RollbackTran();
  322. throw Oops.Oh("采购订单生成失败。");
  323. }
  324. var latest = created.OrderByDescending(x => x.Id).FirstOrDefault();
  325. if (latest == null) throw Oops.Oh("采购订单生成失败。");
  326. return new { id = latest.Id, purOrd = latest.PurOrd, message = "采购订单生成成功。" };
  327. }
  328. private async Task UpsertProcessOutsourceMdpAsync(string purOrd, DateTime now)
  329. {
  330. var batchId = $"S3_OUTSOURCE_{now:yyyyMMddHHmmss}";
  331. await _db.Ado.ExecuteCommandAsync(
  332. """
  333. INSERT INTO mdp_std_purchase_order
  334. (tenant_id, source_system, po_no, po_line, po_type, supplier_code, item_code, item_name, order_qty, received_qty, returned_qty, due_date, need_date, order_date, status, buyer, work_order, source_biz_key, sync_batch_id, sync_time)
  335. SELECT
  336. COALESCE(m.tenant_id, 0),
  337. 'AIDOP',
  338. IFNULL(m.PurOrd, ''),
  339. CAST(IFNULL(d.Line, 0) AS CHAR),
  340. IFNULL(m.Potype, ''),
  341. IFNULL(m.Supp, ''),
  342. IFNULL(d.ItemNum, ''),
  343. IFNULL(i.Descr, ''),
  344. IFNULL(d.QtyOrded, 0),
  345. IFNULL(d.QtyReceived, IFNULL(d.RctQty, 0)),
  346. IFNULL(d.QtyReturned, 0),
  347. d.DueDate,
  348. d.NeedDate,
  349. m.OrdDate,
  350. CASE WHEN IFNULL(LENGTH(m.Status), 0) = 0 THEN 'R' ELSE m.Status END,
  351. m.Buyer,
  352. m.WorkOrd,
  353. CONCAT(IFNULL(m.PurOrd,''), '|', IFNULL(d.Line, 0)),
  354. @BatchId,
  355. @Now
  356. FROM PurOrdMaster m
  357. JOIN PurOrdDetail d ON m.RecID = d.PurOrdRecID
  358. LEFT JOIN ItemMaster i ON d.ItemNum = i.ItemNum
  359. WHERE m.PurOrd = @PurOrd
  360. AND IFNULL(d.ItemNum, '') <> ''
  361. ON DUPLICATE KEY UPDATE
  362. po_type=VALUES(po_type),
  363. supplier_code=VALUES(supplier_code),
  364. item_code=VALUES(item_code),
  365. item_name=VALUES(item_name),
  366. order_qty=VALUES(order_qty),
  367. received_qty=VALUES(received_qty),
  368. returned_qty=VALUES(returned_qty),
  369. due_date=VALUES(due_date),
  370. need_date=VALUES(need_date),
  371. order_date=VALUES(order_date),
  372. status=VALUES(status),
  373. buyer=VALUES(buyer),
  374. work_order=VALUES(work_order),
  375. sync_batch_id=VALUES(sync_batch_id),
  376. sync_time=VALUES(sync_time),
  377. update_time=CURRENT_TIMESTAMP
  378. """,
  379. new SugarParameter("@PurOrd", purOrd),
  380. new SugarParameter("@BatchId", batchId),
  381. new SugarParameter("@Now", now));
  382. await _db.Ado.ExecuteCommandAsync(
  383. """
  384. INSERT INTO mdp_std_process_outsource_order
  385. (tenant_id, source_system, work_order, op_code, routing_code, supplier_code, po_no, po_line, order_qty, completed_qty, due_date, status, source_biz_key, sync_batch_id, sync_time)
  386. SELECT
  387. COALESCE(m.tenant_id, 0),
  388. 'AIDOP',
  389. IFNULL(m.WorkOrd, ''),
  390. CAST(IFNULL(d.Op, 0) AS CHAR),
  391. IFNULL(d.ItemNum, ''),
  392. IFNULL(m.Supp, ''),
  393. IFNULL(m.PurOrd, ''),
  394. CAST(IFNULL(d.Line, 0) AS CHAR),
  395. IFNULL(d.QtyOrded, 0),
  396. IFNULL(d.QtyReceived, IFNULL(d.RctQty, 0)),
  397. d.DueDate,
  398. CASE WHEN IFNULL(LENGTH(m.Status), 0) = 0 THEN 'R' ELSE m.Status END,
  399. CONCAT(IFNULL(m.PurOrd,''), '|', IFNULL(d.Line, 0)),
  400. @BatchId,
  401. @Now
  402. FROM PurOrdMaster m
  403. JOIN PurOrdDetail d ON m.RecID = d.PurOrdRecID
  404. WHERE m.PurOrd = @PurOrd
  405. AND m.Potype = 'PW'
  406. AND IFNULL(m.WorkOrd, '') <> ''
  407. AND IFNULL(d.ItemNum, '') <> ''
  408. ON DUPLICATE KEY UPDATE
  409. supplier_code=VALUES(supplier_code),
  410. po_no=VALUES(po_no),
  411. po_line=VALUES(po_line),
  412. order_qty=VALUES(order_qty),
  413. completed_qty=VALUES(completed_qty),
  414. due_date=VALUES(due_date),
  415. status=VALUES(status),
  416. sync_batch_id=VALUES(sync_batch_id),
  417. sync_time=VALUES(sync_time),
  418. update_time=CURRENT_TIMESTAMP
  419. """,
  420. new SugarParameter("@PurOrd", purOrd),
  421. new SugarParameter("@BatchId", batchId),
  422. new SugarParameter("@Now", now));
  423. await _db.Ado.ExecuteCommandAsync(
  424. """
  425. INSERT INTO dwd_supplier_delivery
  426. (tenant_id, stat_date, po_no, po_line, po_type, supplier_code, supplier_name, item_code, item_name, order_qty, schedule_qty, delivery_qty, receipt_qty, return_qty, remaining_qty, due_date, need_date, delivery_status, source_system, sync_batch_id, calc_time)
  427. SELECT
  428. COALESCE(m.tenant_id, 0),
  429. @StatDate,
  430. IFNULL(m.PurOrd, ''),
  431. CAST(IFNULL(d.Line, 0) AS CHAR),
  432. IFNULL(m.Potype, ''),
  433. IFNULL(m.Supp, ''),
  434. IFNULL(s.SortName, ''),
  435. IFNULL(d.ItemNum, ''),
  436. IFNULL(i.Descr, ''),
  437. IFNULL(d.QtyOrded, 0),
  438. 0,
  439. 0,
  440. IFNULL(d.QtyReceived, IFNULL(d.RctQty, 0)),
  441. IFNULL(d.QtyReturned, 0),
  442. GREATEST(IFNULL(d.QtyOrded, 0) - IFNULL(d.QtyReceived, IFNULL(d.RctQty, 0)) - IFNULL(d.QtyReturned, 0), 0),
  443. d.DueDate,
  444. d.NeedDate,
  445. 'OPEN',
  446. 'AIDOP',
  447. @BatchId,
  448. @Now
  449. FROM PurOrdMaster m
  450. JOIN PurOrdDetail d ON m.RecID = d.PurOrdRecID
  451. LEFT JOIN ItemMaster i ON d.ItemNum = i.ItemNum
  452. LEFT JOIN SuppMaster s ON m.Domain = s.Domain AND m.Supp = s.Supp
  453. WHERE m.PurOrd = @PurOrd
  454. AND IFNULL(d.ItemNum, '') <> ''
  455. ON DUPLICATE KEY UPDATE
  456. po_type=VALUES(po_type),
  457. supplier_code=VALUES(supplier_code),
  458. supplier_name=VALUES(supplier_name),
  459. item_code=VALUES(item_code),
  460. item_name=VALUES(item_name),
  461. order_qty=VALUES(order_qty),
  462. receipt_qty=VALUES(receipt_qty),
  463. return_qty=VALUES(return_qty),
  464. remaining_qty=VALUES(remaining_qty),
  465. due_date=VALUES(due_date),
  466. need_date=VALUES(need_date),
  467. delivery_status=VALUES(delivery_status),
  468. sync_batch_id=VALUES(sync_batch_id),
  469. calc_time=VALUES(calc_time),
  470. update_time=CURRENT_TIMESTAMP
  471. """,
  472. new SugarParameter("@PurOrd", purOrd),
  473. new SugarParameter("@StatDate", now.Date),
  474. new SugarParameter("@BatchId", batchId),
  475. new SugarParameter("@Now", now));
  476. }
  477. [DisplayName("保存工序外协订单")]
  478. [HttpPost("process-outsource-order/save")]
  479. public async Task<object> Save([FromBody] ProcessOutsourceOrderSaveInput input)
  480. {
  481. if (input.Id <= 0) throw Oops.Oh("缺少主键");
  482. if (string.IsNullOrWhiteSpace(input.PurOrd)) throw Oops.Oh("采购单号不能为空");
  483. if (string.IsNullOrWhiteSpace(input.Supp)) throw Oops.Oh("供应商不能为空");
  484. if (string.IsNullOrWhiteSpace(input.Department)) throw Oops.Oh("部门不能为空");
  485. var exists = await _db.Ado.GetIntAsync("SELECT COUNT(1) FROM PurOrdMaster WHERE RecID=@Id" + (_userManager.TenantId > 0 ? " AND tenant_id=@TenantId" : ""),
  486. _userManager.TenantId > 0
  487. ? new[] { new SugarParameter("@Id", input.Id), new SugarParameter("@TenantId", _userManager.TenantId) }
  488. : new[] { new SugarParameter("@Id", input.Id) });
  489. if (exists <= 0) throw Oops.Oh("记录不存在");
  490. try
  491. {
  492. _db.Ado.BeginTran();
  493. await _db.Ado.ExecuteCommandAsync(
  494. """
  495. UPDATE PurOrdMaster
  496. SET OrdDate=@OrdDate,Supp=@Supp,ReqBy=@ReqBy,WorkOrd=@WorkOrd,Buyer=@Buyer,Department=@Department,`Usage`=@Usage,Curr=@Curr,Remark=@Remark,UpdateUser=@UpdateUser,UpdateTime=@UpdateTime
  497. WHERE RecID=@Id
  498. """,
  499. new SugarParameter("@Id", input.Id),
  500. new SugarParameter("@OrdDate", ParseDate(input.OrdDate)),
  501. new SugarParameter("@Supp", input.Supp!.Trim()),
  502. new SugarParameter("@ReqBy", string.IsNullOrWhiteSpace(input.ReqBy) ? "PO" : input.ReqBy.Trim()),
  503. new SugarParameter("@WorkOrd", input.WorkOrd?.Trim()),
  504. new SugarParameter("@Buyer", string.IsNullOrWhiteSpace(input.Buyer) ? "110" : input.Buyer.Trim()),
  505. new SugarParameter("@Department", input.Department!.Trim()),
  506. new SugarParameter("@Usage", string.IsNullOrWhiteSpace(input.Usage) ? "外协" : input.Usage.Trim()),
  507. new SugarParameter("@Curr", input.Curr?.Trim()),
  508. new SugarParameter("@Remark", input.Remark?.Trim()),
  509. new SugarParameter("@UpdateUser", _userManager.Account),
  510. new SugarParameter("@UpdateTime", DateTime.Now)
  511. );
  512. await SaveDetailsAsync(input.Id, input.PurOrd!.Trim(), input.WorkOrd, input.Details);
  513. _db.Ado.CommitTran();
  514. }
  515. catch
  516. {
  517. _db.Ado.RollbackTran();
  518. throw;
  519. }
  520. return new { id = input.Id, message = "保存成功" };
  521. }
  522. /// <summary>
  523. /// 明细三路合并:
  524. /// ① DB有且入参有(按 Id 匹配)→ 更新
  525. /// ② DB无但入参有 → 新增
  526. /// ③ DB有但入参无 → 删除
  527. /// </summary>
  528. private async Task SaveDetailsAsync(int masterId, string purOrd, string? workOrd, List<ProcessOutsourceOrderDetailInput> details)
  529. {
  530. var dbDetails = await _db.Ado.SqlQueryAsync<DbDetailRow>(
  531. "SELECT RecID AS Id FROM PurOrdDetail WHERE PurOrdRecID=@PurOrdRecID",
  532. new SugarParameter("@PurOrdRecID", masterId));
  533. var dbById = dbDetails.ToDictionary(x => x.Id);
  534. var inputIds = new HashSet<int>(details.Where(x => x.Id.HasValue && x.Id.Value > 0).Select(x => x.Id!.Value));
  535. var maxLine = await _db.Ado.GetIntAsync("SELECT IFNULL(MAX(Line),0) FROM PurOrdDetail WHERE PurOrdRecID=@PurOrdRecID", new SugarParameter("@PurOrdRecID", masterId));
  536. var lineSeed = maxLine;
  537. for (var i = 0; i < details.Count; i++)
  538. {
  539. var item = details[i];
  540. if (string.IsNullOrWhiteSpace(item.ItemNum)) continue;
  541. var line = item.Line ?? (i + 1);
  542. if (line <= 0)
  543. {
  544. lineSeed++;
  545. line = lineSeed;
  546. }
  547. var itemInfo = (await _db.Ado.SqlQueryAsync<ItemLookupRow>(
  548. """
  549. SELECT ItemNum,Descr,UM,Location
  550. FROM ItemMaster
  551. WHERE ItemNum=@ItemNum
  552. LIMIT 1
  553. """,
  554. new SugarParameter("@ItemNum", item.ItemNum.Trim()))).FirstOrDefault();
  555. if (item.Id.HasValue && item.Id.Value > 0 && dbById.ContainsKey(item.Id.Value))
  556. {
  557. await _db.Ado.ExecuteCommandAsync(
  558. """
  559. UPDATE PurOrdDetail
  560. SET Line=@Line,ItemNum=@ItemNum,Descr=@Descr,Op=@Op,UM=@UM,Location=@Location,QtyOrded=@QtyOrded,QtyReceived=@QtyReceived,ReceiptQty=@ReceiptQty,DueDate=@DueDate,LotSerial=@LotSerial,Potype='PW',PurOrd=@PurOrd,PurOrdRecID=@PurOrdRecID,WorkOrd=@WorkOrd,UpdateUser=@UpdateUser,UpdateTime=@UpdateTime
  561. WHERE RecID=@Id
  562. """,
  563. new SugarParameter("@Id", item.Id.Value),
  564. new SugarParameter("@Line", line),
  565. new SugarParameter("@ItemNum", item.ItemNum.Trim()),
  566. new SugarParameter("@Descr", itemInfo?.Descr),
  567. new SugarParameter("@Op", item.Op ?? 0),
  568. new SugarParameter("@UM", string.IsNullOrWhiteSpace(item.UM) ? itemInfo?.UM : item.UM!.Trim()),
  569. new SugarParameter("@Location", string.IsNullOrWhiteSpace(item.Location) ? itemInfo?.Location : item.Location!.Trim()),
  570. new SugarParameter("@QtyOrded", item.QtyOrded ?? 0),
  571. new SugarParameter("@QtyReceived", item.QtyReceived ?? 0),
  572. new SugarParameter("@ReceiptQty", item.ReceiptQty ?? 0),
  573. new SugarParameter("@DueDate", ParseDate(item.DueDate)),
  574. new SugarParameter("@LotSerial", item.LotSerial?.Trim()),
  575. new SugarParameter("@PurOrd", purOrd),
  576. new SugarParameter("@PurOrdRecID", masterId),
  577. new SugarParameter("@WorkOrd", workOrd?.Trim()),
  578. new SugarParameter("@UpdateUser", _userManager.Account),
  579. new SugarParameter("@UpdateTime", DateTime.Now)
  580. );
  581. }
  582. else
  583. {
  584. await _db.Ado.ExecuteCommandAsync(
  585. """
  586. INSERT INTO PurOrdDetail
  587. (PurOrd,Line,ItemNum,Descr,Op,UM,Location,QtyOrded,QtyReceived,ReceiptQty,DueDate,LotSerial,Potype,PurOrdRecID,WorkOrd,Status,CreateUser,CreateTime,UpdateUser,UpdateTime,tenant_id)
  588. VALUES
  589. (@PurOrd,@Line,@ItemNum,@Descr,@Op,@UM,@Location,@QtyOrded,@QtyReceived,@ReceiptQty,@DueDate,@LotSerial,'PW',@PurOrdRecID,@WorkOrd,'R',@CreateUser,@CreateTime,@UpdateUser,@UpdateTime,@TenantId)
  590. """,
  591. new SugarParameter("@PurOrd", purOrd),
  592. new SugarParameter("@Line", line),
  593. new SugarParameter("@ItemNum", item.ItemNum.Trim()),
  594. new SugarParameter("@Descr", itemInfo?.Descr),
  595. new SugarParameter("@Op", item.Op ?? 0),
  596. new SugarParameter("@UM", string.IsNullOrWhiteSpace(item.UM) ? itemInfo?.UM : item.UM!.Trim()),
  597. new SugarParameter("@Location", string.IsNullOrWhiteSpace(item.Location) ? itemInfo?.Location : item.Location!.Trim()),
  598. new SugarParameter("@QtyOrded", item.QtyOrded ?? 0),
  599. new SugarParameter("@QtyReceived", item.QtyReceived ?? 0),
  600. new SugarParameter("@ReceiptQty", item.ReceiptQty ?? 0),
  601. new SugarParameter("@DueDate", ParseDate(item.DueDate)),
  602. new SugarParameter("@LotSerial", item.LotSerial?.Trim()),
  603. new SugarParameter("@PurOrdRecID", masterId),
  604. new SugarParameter("@WorkOrd", workOrd?.Trim()),
  605. new SugarParameter("@CreateUser", _userManager.Account),
  606. new SugarParameter("@CreateTime", DateTime.Now),
  607. new SugarParameter("@UpdateUser", _userManager.Account),
  608. new SugarParameter("@UpdateTime", DateTime.Now),
  609. new SugarParameter("@TenantId", _userManager.TenantId <= 0 ? null : _userManager.TenantId)
  610. );
  611. }
  612. }
  613. foreach (var toDelete in dbDetails.Where(x => !inputIds.Contains(x.Id)))
  614. {
  615. await _db.Ado.ExecuteCommandAsync("DELETE FROM PurOrdDetail WHERE RecID=@Id", new SugarParameter("@Id", toDelete.Id));
  616. }
  617. }
  618. [DisplayName("删除工序外协订单")]
  619. [HttpPost("process-outsource-order/delete/{id:int}")]
  620. public async Task<object> Delete(int id, [FromQuery] string purOrd)
  621. {
  622. if (string.IsNullOrWhiteSpace(purOrd)) throw Oops.Oh("缺少采购单号");
  623. var ordNo = purOrd.Trim();
  624. var hasReceipt = await _db.Ado.GetIntAsync(
  625. "SELECT COUNT(1) FROM PurOrdRctDetail WHERE OrdNbr=@PurOrd AND IFNULL(QtyReceived,0)>0" + (_userManager.TenantId > 0 ? " AND tenant_id=@TenantId" : ""),
  626. _userManager.TenantId > 0
  627. ? new[] { new SugarParameter("@PurOrd", ordNo), new SugarParameter("@TenantId", _userManager.TenantId) }
  628. : new[] { new SugarParameter("@PurOrd", ordNo) });
  629. if (hasReceipt > 0) throw Oops.Oh("存在收货数量,不允许删除");
  630. var hasShip = await _db.Ado.GetIntAsync(
  631. "SELECT COUNT(1) FROM scm_shdzb WHERE po_bill=@PurOrd" + (_userManager.TenantId > 0 ? " AND tenant_id=@TenantId" : ""),
  632. _userManager.TenantId > 0
  633. ? new[] { new SugarParameter("@PurOrd", ordNo), new SugarParameter("@TenantId", _userManager.TenantId) }
  634. : new[] { new SugarParameter("@PurOrd", ordNo) });
  635. if (hasShip > 0) throw Oops.Oh("存在发货单,不允许删除");
  636. try
  637. {
  638. _db.Ado.BeginTran();
  639. await _db.Ado.ExecuteCommandAsync("DELETE FROM PurOrdDetail WHERE PurOrd=@PurOrd", new SugarParameter("@PurOrd", ordNo));
  640. await _db.Ado.ExecuteCommandAsync("DELETE FROM PurOrdMaster WHERE RecID=@Id AND PurOrd=@PurOrd", new SugarParameter("@Id", id), new SugarParameter("@PurOrd", ordNo));
  641. _db.Ado.CommitTran();
  642. }
  643. catch
  644. {
  645. _db.Ado.RollbackTran();
  646. throw;
  647. }
  648. return new { message = "删除成功" };
  649. }
  650. [DisplayName("采购组下拉")]
  651. [HttpGet("process-outsource-order/options/buyers")]
  652. public async Task<object> GetBuyerOptions()
  653. {
  654. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  655. """
  656. SELECT Employee AS Value, CONCAT(TRIM(Employee),' ',TRIM(IFNULL(Name,''))) AS Label
  657. FROM EmployeeMaster
  658. ORDER BY Employee
  659. """);
  660. return new { list };
  661. }
  662. [DisplayName("供应商下拉")]
  663. [HttpGet("process-outsource-order/options/suppliers")]
  664. public async Task<object> GetSupplierOptions()
  665. {
  666. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  667. """
  668. SELECT Supp AS Value, CONCAT(TRIM(Supp),' ',TRIM(IFNULL(SortName,''))) AS Label
  669. FROM SuppMaster
  670. ORDER BY Supp
  671. """);
  672. return new { list };
  673. }
  674. [DisplayName("部门下拉")]
  675. [HttpGet("process-outsource-order/options/departments")]
  676. public async Task<object> GetDepartmentOptions()
  677. {
  678. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  679. """
  680. SELECT Department AS Value, CONCAT(TRIM(Department),' ',TRIM(IFNULL(Descr,''))) AS Label
  681. FROM DepartmentMaster
  682. ORDER BY Department
  683. """);
  684. return new { list };
  685. }
  686. [DisplayName("币别下拉")]
  687. [HttpGet("process-outsource-order/options/currencies")]
  688. public async Task<object> GetCurrencyOptions()
  689. {
  690. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  691. """
  692. SELECT Val AS Value, CONCAT(TRIM(Val),' ',TRIM(IFNULL(Comments,''))) AS Label
  693. FROM GeneralizedCodeMaster
  694. WHERE FldName='Curr'
  695. ORDER BY Val
  696. """);
  697. return new { list };
  698. }
  699. [DisplayName("库位下拉")]
  700. [HttpGet("process-outsource-order/options/locations")]
  701. public async Task<object> GetLocationOptions()
  702. {
  703. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  704. """
  705. SELECT Location AS Value, CONCAT(TRIM(Location),' ',TRIM(IFNULL(Descr,''))) AS Label
  706. FROM LocationMaster
  707. WHERE IFNULL(Typed,'')<>'Supp'
  708. ORDER BY Location
  709. """);
  710. return new { list };
  711. }
  712. [DisplayName("工序下拉")]
  713. [HttpGet("process-outsource-order/options/ops")]
  714. public async Task<object> GetOpOptions([FromQuery] int purOrdRecId)
  715. {
  716. if (purOrdRecId <= 0) return new { list = new List<OptionRow>() };
  717. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  718. """
  719. SELECT DISTINCT CAST(Op AS CHAR(50)) AS Value, CONCAT(TRIM(CAST(Op AS CHAR(50))),' ',TRIM(IFNULL(Descr,''))) AS Label
  720. FROM RoutingOpDetail
  721. WHERE UDeci5!=0
  722. AND RoutingCode IN (SELECT ItemNum FROM PurOrdDetail WHERE PurOrdRecID=@PurOrdRecID)
  723. ORDER BY Value
  724. """,
  725. new SugarParameter("@PurOrdRecID", purOrdRecId));
  726. return new { list };
  727. }
  728. private static DateTime? ParseDate(string? v) => DateTime.TryParse(v, out var d) ? d : null;
  729. private static string BuildOrderBy(string? sortField, string? sortOrder)
  730. {
  731. var dir = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase) ? "ASC" : "DESC";
  732. return sortField?.ToLowerInvariant() switch
  733. {
  734. "purord" => $"t.PurOrd {dir}",
  735. "workord" => $"t.WorkOrd {dir}",
  736. "suppname" => $"t.SuppName {dir}",
  737. "buyer" => $"t.Buyer {dir}",
  738. "itemnum" => $"t.ItemNum {dir}",
  739. "itemname" => $"t.ItemName {dir}",
  740. "op" => $"t.Op {dir}",
  741. "orddate" => $"t.OrdDate {dir}",
  742. "qtyorded" => $"t.QtyOrded {dir}",
  743. "cumqtybo" => $"t.CumQtyBO {dir}",
  744. "duedate" => $"t.DueDate {dir}",
  745. "fldate" => $"t.FlDate {dir}",
  746. "status" => $"t.Status {dir}",
  747. _ => "t.Id DESC"
  748. };
  749. }
  750. private sealed class ProcessOutsourceOrderListRow
  751. {
  752. public int Id { get; set; }
  753. public string? PurOrd { get; set; }
  754. public string? WorkOrd { get; set; }
  755. public string? ERPWorkOrd { get; set; }
  756. public string? SuppName { get; set; }
  757. public string? Buyer { get; set; }
  758. public string? BuyerCode { get; set; }
  759. public string? ItemNum { get; set; }
  760. public string? ItemName { get; set; }
  761. public string? Op { get; set; }
  762. public DateTime? OrdDate { get; set; }
  763. public decimal? QtyOrded { get; set; }
  764. public decimal? CumQtyBO { get; set; }
  765. public DateTime? DueDate { get; set; }
  766. public DateTime? FlDate { get; set; }
  767. public string? Status { get; set; }
  768. public string? Supp { get; set; }
  769. public string? ReqBy { get; set; }
  770. public string? Department { get; set; }
  771. public string? Usage { get; set; }
  772. public string? Curr { get; set; }
  773. public string? Remark { get; set; }
  774. }
  775. private sealed class ProcessOutsourceOrderMasterRow
  776. {
  777. public int Id { get; set; }
  778. public string? PurOrd { get; set; }
  779. public DateTime? OrdDate { get; set; }
  780. public string? Supp { get; set; }
  781. public string? ReqBy { get; set; }
  782. public string? WorkOrd { get; set; }
  783. public string? Buyer { get; set; }
  784. public string? Department { get; set; }
  785. public string? Usage { get; set; }
  786. public string? Curr { get; set; }
  787. public string? Remark { get; set; }
  788. public string? Status { get; set; }
  789. }
  790. private sealed class ProcessOutsourceOrderDetailRow
  791. {
  792. public int Id { get; set; }
  793. public int? Line { get; set; }
  794. public string? ItemNum { get; set; }
  795. public string? ItemName { get; set; }
  796. public int? Op { get; set; }
  797. public string? UM { get; set; }
  798. public string? Location { get; set; }
  799. public decimal? QtyOrded { get; set; }
  800. public decimal? QtyReceived { get; set; }
  801. public decimal? ReceiptQty { get; set; }
  802. public DateTime? DueDate { get; set; }
  803. public string? LotSerial { get; set; }
  804. public string? Potype { get; set; }
  805. public string? PurOrd { get; set; }
  806. public int? PurOrdRecID { get; set; }
  807. }
  808. private sealed class OptionRow
  809. {
  810. public string? Value { get; set; }
  811. public string? Label { get; set; }
  812. }
  813. private sealed class IdPurOrdRow
  814. {
  815. public int Id { get; set; }
  816. public string? PurOrd { get; set; }
  817. }
  818. private sealed class SuppRow
  819. {
  820. public string? Supp { get; set; }
  821. }
  822. private sealed class CreateLineRow
  823. {
  824. public string? PMBOM { get; set; }
  825. public int Op { get; set; }
  826. public string? ItemNum { get; set; }
  827. public string? UM { get; set; }
  828. public decimal QtyOrded { get; set; }
  829. public string? Rev { get; set; }
  830. public string? Department { get; set; }
  831. public string? WorkOrd { get; set; }
  832. public string? Drawing { get; set; }
  833. public long WorkOrdID { get; set; }
  834. public int ProcessOutDay { get; set; }
  835. }
  836. private sealed class DbDetailRow
  837. {
  838. public int Id { get; set; }
  839. }
  840. private sealed class ItemLookupRow
  841. {
  842. public string? ItemNum { get; set; }
  843. public string? Descr { get; set; }
  844. public string? UM { get; set; }
  845. public string? Location { get; set; }
  846. }
  847. }