OutsourceOrderService.cs 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773
  1. namespace Admin.NET.Plugin.AiDOP.Supply;
  2. /// <summary>
  3. /// 委外加工订单服务
  4. /// </summary>
  5. [ApiDescriptionSettings(Order = 310, Description = "委外加工订单")]
  6. [Route("api/Supply")]
  7. [AllowAnonymous]
  8. [NonUnify]
  9. public class OutsourceOrderService : IDynamicApiController, ITransient
  10. {
  11. private const int PwPurOrdSerialWidth = 4;
  12. private readonly ISqlSugarClient _db;
  13. private readonly UserManager _userManager;
  14. public OutsourceOrderService(ISqlSugarClient db, UserManager userManager)
  15. {
  16. _db = db;
  17. _userManager = userManager;
  18. }
  19. [DisplayName("委外加工订单列表")]
  20. [HttpGet("outsource-order/list")]
  21. public async Task<object> GetOrderList([FromQuery] OutsourceOrderListInput 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>
  27. {
  28. "p.Potype='PW'",
  29. "p.IsActive=1",
  30. "IFNULL(p.`Usage`,'')='委外加工'"
  31. };
  32. var pars = new List<SugarParameter>();
  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.Buyer))
  39. {
  40. where.Add("p.Buyer=@Buyer");
  41. pars.Add(new SugarParameter("@Buyer", input.Buyer.Trim()));
  42. }
  43. if (!string.IsNullOrWhiteSpace(input.Supp))
  44. {
  45. where.Add("p.Supp=@Supp");
  46. pars.Add(new SugarParameter("@Supp", input.Supp.Trim()));
  47. }
  48. var fromSql = $"""
  49. FROM PurOrdMaster p
  50. LEFT JOIN SuppMaster s ON p.Supp=s.Supp
  51. LEFT JOIN DepartmentMaster d ON p.Department=d.Department
  52. LEFT JOIN EmployeeMaster e ON p.Buyer=e.Employee
  53. WHERE {string.Join(" AND ", where)}
  54. """;
  55. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) {fromSql}", pars);
  56. var list = await _db.Ado.SqlQueryAsync<OutsourceOrderListRow>(
  57. $"""
  58. SELECT
  59. p.RecID AS Id,
  60. p.PurOrd AS PurOrd,
  61. CONCAT(TRIM(IFNULL(p.Supp,'')),' ',TRIM(IFNULL(s.SortName,''))) AS SuppName,
  62. p.Potype AS Potype,
  63. p.ReqBy AS ReqBy,
  64. CONCAT(TRIM(IFNULL(p.Buyer,'')),' ',TRIM(IFNULL(e.Name,''))) AS Buyer,
  65. CONCAT(TRIM(IFNULL(p.Department,'')),' ',TRIM(IFNULL(d.Descr,''))) AS DepartmentDescr,
  66. p.OrdDate AS OrdDate,
  67. p.DueDate AS DueDate,
  68. p.Curr AS Curr,
  69. CASE
  70. WHEN IFNULL(LENGTH(p.Status),0)=0 OR p.Buyer IS NULL THEN 'R'
  71. ELSE p.Status
  72. END AS Status,
  73. p.Supp AS Supp,
  74. p.Buyer AS BuyerCode,
  75. p.Department AS Department,
  76. p.`Usage` AS `Usage`,
  77. p.Remark AS Remark
  78. {fromSql}
  79. ORDER BY {BuildOrderBy(input.SortField, input.SortOrder)}
  80. LIMIT {pageSize} OFFSET {offset}
  81. """,
  82. pars);
  83. return new { total, page, pageSize, list };
  84. }
  85. [DisplayName("委外加工订单详情")]
  86. [HttpGet("outsource-order/{id:int}")]
  87. public async Task<object> GetOrderDetail(int id)
  88. {
  89. var row = (await _db.Ado.SqlQueryAsync<OutsourceOrderListRow>(
  90. """
  91. SELECT
  92. p.RecID AS Id,
  93. p.PurOrd AS PurOrd,
  94. CONCAT(TRIM(IFNULL(p.Supp,'')),' ',TRIM(IFNULL(s.SortName,''))) AS SuppName,
  95. p.Potype AS Potype,
  96. p.ReqBy AS ReqBy,
  97. CONCAT(TRIM(IFNULL(p.Buyer,'')),' ',TRIM(IFNULL(e.Name,''))) AS Buyer,
  98. CONCAT(TRIM(IFNULL(p.Department,'')),' ',TRIM(IFNULL(d.Descr,''))) AS DepartmentDescr,
  99. p.OrdDate AS OrdDate,
  100. p.DueDate AS DueDate,
  101. p.Curr AS Curr,
  102. CASE WHEN IFNULL(LENGTH(p.Status),0)=0 OR p.Buyer IS NULL THEN 'R' ELSE p.Status END AS Status,
  103. p.Supp AS Supp,
  104. p.Buyer AS BuyerCode,
  105. p.Department AS Department,
  106. p.`Usage` AS `Usage`,
  107. p.Remark AS Remark
  108. FROM PurOrdMaster p
  109. LEFT JOIN SuppMaster s ON p.Supp=s.Supp
  110. LEFT JOIN DepartmentMaster d ON p.Department=d.Department
  111. LEFT JOIN EmployeeMaster e ON p.Buyer=e.Employee
  112. WHERE p.RecID=@Id
  113. LIMIT 1
  114. """,
  115. new SugarParameter("@Id", id))).FirstOrDefault();
  116. return row ?? throw Oops.Oh("记录不存在");
  117. }
  118. [DisplayName("保存委外加工订单")]
  119. [HttpPost("outsource-order/save")]
  120. public async Task<object> SaveOrder([FromBody] OutsourceOrderSaveInput input)
  121. {
  122. var now = DateTime.Now;
  123. var ordDate = ParseDate(input.OrdDate) ?? now.Date;
  124. var buyer = string.IsNullOrWhiteSpace(input.Buyer) ? "110" : input.Buyer.Trim();
  125. var usage = string.IsNullOrWhiteSpace(input.Usage) ? "委外加工" : input.Usage.Trim();
  126. var reqBy = string.IsNullOrWhiteSpace(input.ReqBy) ? "DO" : input.ReqBy.Trim();
  127. if (string.IsNullOrWhiteSpace(input.Supp)) throw Oops.Oh("请选择供应商");
  128. if (string.IsNullOrWhiteSpace(input.Department)) throw Oops.Oh("请选择部门");
  129. if (input.Id is null or <= 0)
  130. {
  131. try
  132. {
  133. _db.Ado.BeginTran();
  134. var purOrd = await PurOrdNumberGenerator.NextAsync(_db, "PW", DateTime.Today, PwPurOrdSerialWidth);
  135. await _db.Ado.ExecuteCommandAsync(
  136. """
  137. INSERT INTO PurOrdMaster
  138. (
  139. Confirming, CreditTermsInt, Disc, ExchRate, EstVal, ExchRate1, ExchRate2,
  140. FixedPrice, FixedRate, Frt, PartialOK, AmtPrepaid, PrintPO, PST, Recurr,
  141. `Release`, Revision, Scheduled, ServiceCharge, SpecialCharge, Taxable,
  142. Tax1, Tax2, Tax3, TransportDays, IsActive, IsConfirm, Potype, IsChanged,
  143. TaxIn, Amt, IsPriceChanged,
  144. PurOrd, OrdDate, Supp, ReqBy, Buyer, Department, `Usage`, Remark, Status,
  145. Domain, FSTID, CreateUser, CreateTime, UpdateUser, UpdateTime, tenant_id
  146. )
  147. VALUES
  148. (
  149. 0, 0, 0, 1, 0, 1, 1,
  150. 0, 0, 0, 1, 0, 0, 0, 0,
  151. 0, 0, 0, 0, 0, 1,
  152. 0, 0, 0, 0, 1, 1, 'PW', 0,
  153. 1, 0, 0,
  154. @PurOrd, @OrdDate, @Supp, @ReqBy, @Buyer, @Department, @Usage, @Remark, 'R',
  155. @Domain, @FSTID, @CreateUser, @CreateTime, @UpdateUser, @UpdateTime, @TenantId
  156. )
  157. """,
  158. new SugarParameter("@PurOrd", purOrd),
  159. new SugarParameter("@OrdDate", ordDate),
  160. new SugarParameter("@Supp", input.Supp.Trim()),
  161. new SugarParameter("@ReqBy", reqBy),
  162. new SugarParameter("@Buyer", buyer),
  163. new SugarParameter("@Department", input.Department.Trim()),
  164. new SugarParameter("@Usage", usage),
  165. new SugarParameter("@Remark", input.Remark?.Trim()),
  166. new SugarParameter("@Domain", "100"),
  167. new SugarParameter("@FSTID", "1"),
  168. new SugarParameter("@CreateUser", _userManager.Account),
  169. new SugarParameter("@CreateTime", now),
  170. new SugarParameter("@UpdateUser", _userManager.Account),
  171. new SugarParameter("@UpdateTime", now),
  172. new SugarParameter("@TenantId", _userManager.TenantId <= 0 ? null : _userManager.TenantId)
  173. );
  174. var id = await _db.Ado.GetIntAsync(
  175. "SELECT IFNULL(MAX(RecID),0) FROM PurOrdMaster WHERE PurOrd=@PurOrd",
  176. new SugarParameter("@PurOrd", purOrd));
  177. _db.Ado.CommitTran();
  178. return new { id, purOrd, message = "新增成功" };
  179. }
  180. catch
  181. {
  182. _db.Ado.RollbackTran();
  183. throw;
  184. }
  185. }
  186. var exists = await _db.Ado.GetIntAsync("SELECT COUNT(1) FROM PurOrdMaster WHERE RecID=@Id", new SugarParameter("@Id", input.Id.Value));
  187. if (exists <= 0) throw Oops.Oh("记录不存在");
  188. await _db.Ado.ExecuteCommandAsync(
  189. """
  190. UPDATE PurOrdMaster
  191. SET OrdDate=@OrdDate,Supp=@Supp,ReqBy=@ReqBy,Buyer=@Buyer,Department=@Department,`Usage`=@Usage,Remark=@Remark,UpdateUser=@UpdateUser,UpdateTime=@UpdateTime
  192. WHERE RecID=@Id
  193. """,
  194. new SugarParameter("@Id", input.Id.Value),
  195. new SugarParameter("@OrdDate", ordDate),
  196. new SugarParameter("@Supp", input.Supp.Trim()),
  197. new SugarParameter("@ReqBy", reqBy),
  198. new SugarParameter("@Buyer", buyer),
  199. new SugarParameter("@Department", input.Department.Trim()),
  200. new SugarParameter("@Usage", usage),
  201. new SugarParameter("@Remark", input.Remark?.Trim()),
  202. new SugarParameter("@UpdateUser", _userManager.Account),
  203. new SugarParameter("@UpdateTime", now)
  204. );
  205. return new { id = input.Id, message = "编辑成功" };
  206. }
  207. [DisplayName("删除委外加工订单")]
  208. [HttpPost("outsource-order/delete/{id:int}")]
  209. public async Task<object> DeleteOrder(int id, [FromQuery] string purOrd)
  210. {
  211. if (string.IsNullOrWhiteSpace(purOrd)) throw Oops.Oh("缺少采购订单号");
  212. var hasReceipt = await _db.Ado.GetIntAsync(
  213. """
  214. SELECT COUNT(1)
  215. FROM PurOrdRctDetail
  216. WHERE OrdNbr=@PurOrd AND IFNULL(QtyReceived,0)>0
  217. """,
  218. new SugarParameter("@PurOrd", purOrd.Trim()));
  219. if (hasReceipt > 0) throw Oops.Oh("存在收货数量,不允许删除");
  220. var hasShip = await _db.Ado.GetIntAsync(
  221. "SELECT COUNT(1) FROM scm_shdzb WHERE po_bill=@PurOrd",
  222. new SugarParameter("@PurOrd", purOrd.Trim()));
  223. if (hasShip > 0) throw Oops.Oh("存在发货单,不允许删除");
  224. try
  225. {
  226. _db.Ado.BeginTran();
  227. await _db.Ado.ExecuteCommandAsync("DELETE FROM PurOrdDetailBatch WHERE PurOrd=@PurOrd", new SugarParameter("@PurOrd", purOrd.Trim()));
  228. await _db.Ado.ExecuteCommandAsync("DELETE FROM PurOrdDetail WHERE PurOrd=@PurOrd", new SugarParameter("@PurOrd", purOrd.Trim()));
  229. await _db.Ado.ExecuteCommandAsync("DELETE FROM PurOrdMaster WHERE RecID=@Id AND PurOrd=@PurOrd", new SugarParameter("@Id", id), new SugarParameter("@PurOrd", purOrd.Trim()));
  230. _db.Ado.CommitTran();
  231. }
  232. catch
  233. {
  234. _db.Ado.RollbackTran();
  235. throw;
  236. }
  237. return new { message = "删除成功" };
  238. }
  239. [DisplayName("委外采购明细列表")]
  240. [HttpGet("outsource-order/detail/list")]
  241. public async Task<object> GetDetailList([FromQuery] OutsourceOrderDetailListInput input)
  242. {
  243. var page = input.Page <= 0 ? 1 : input.Page;
  244. var pageSize = input.PageSize <= 0 ? 10 : input.PageSize;
  245. var offset = (page - 1) * pageSize;
  246. var where = new List<string> { "d.Potype='PW'" };
  247. var pars = new List<SugarParameter>();
  248. if (!string.IsNullOrWhiteSpace(input.PurOrd))
  249. {
  250. where.Add("d.PurOrd LIKE @PurOrd");
  251. pars.Add(new SugarParameter("@PurOrd", $"%{input.PurOrd.Trim()}%"));
  252. }
  253. if (!string.IsNullOrWhiteSpace(input.ItemNum))
  254. {
  255. where.Add("d.ItemNum LIKE @ItemNum");
  256. pars.Add(new SugarParameter("@ItemNum", $"%{input.ItemNum.Trim()}%"));
  257. }
  258. if (!string.IsNullOrWhiteSpace(input.Location))
  259. {
  260. where.Add("d.Location=@Location");
  261. pars.Add(new SugarParameter("@Location", input.Location.Trim()));
  262. }
  263. if (!string.IsNullOrWhiteSpace(input.WorkOrd))
  264. {
  265. where.Add("d.WorkOrd LIKE @WorkOrd");
  266. pars.Add(new SugarParameter("@WorkOrd", $"%{input.WorkOrd.Trim()}%"));
  267. }
  268. if (!string.IsNullOrWhiteSpace(input.LotSerial))
  269. {
  270. where.Add("d.LotSerial LIKE @LotSerial");
  271. pars.Add(new SugarParameter("@LotSerial", $"%{input.LotSerial.Trim()}%"));
  272. }
  273. var fromSql = $"""
  274. FROM PurOrdDetail d
  275. LEFT JOIN ItemMaster i ON d.ItemNum=i.ItemNum
  276. WHERE {string.Join(" AND ", where)}
  277. """;
  278. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) {fromSql}", pars);
  279. var list = await _db.Ado.SqlQueryAsync<OutsourceOrderDetailRow>(
  280. $"""
  281. SELECT
  282. d.PurOrd AS PurOrd,d.ItemNum AS ItemNum,d.QtyOrded AS QtyOrded,d.QtyReceived AS QtyReceived,d.QtyReleased AS QtyReleased,
  283. d.DueDate AS DueDate,d.Line AS Line,d.Location AS Location,d.UM AS UM,d.RecID AS Id,d.Rev AS Rev,d.Drawing AS Drawing,
  284. d.WorkOrd AS WorkOrd,d.LotSerial AS LotSerial,d.Status AS Status,IFNULL(i.Descr,'') AS Descr,d.PurOrdRecID AS PurOrdRecID
  285. {fromSql}
  286. ORDER BY {BuildDetailOrderBy(input.SortField, input.SortOrder)}
  287. LIMIT {pageSize} OFFSET {offset}
  288. """,
  289. pars);
  290. return new { total, page, pageSize, list };
  291. }
  292. [DisplayName("委外采购明细详情")]
  293. [HttpGet("outsource-order/detail/{id:int}")]
  294. public async Task<object> GetDetail(int id)
  295. {
  296. var detail = (await _db.Ado.SqlQueryAsync<OutsourceOrderDetailRow>(
  297. """
  298. SELECT
  299. d.PurOrd AS PurOrd,d.ItemNum AS ItemNum,d.QtyOrded AS QtyOrded,d.QtyReceived AS QtyReceived,d.QtyReleased AS QtyReleased,
  300. d.DueDate AS DueDate,d.Line AS Line,d.Location AS Location,d.UM AS UM,d.RecID AS Id,d.Rev AS Rev,d.Drawing AS Drawing,
  301. d.WorkOrd AS WorkOrd,d.LotSerial AS LotSerial,d.Status AS Status,IFNULL(i.Descr,'') AS Descr,d.PurOrdRecID AS PurOrdRecID
  302. FROM PurOrdDetail d
  303. LEFT JOIN ItemMaster i ON d.ItemNum=i.ItemNum
  304. WHERE d.RecID=@Id
  305. LIMIT 1
  306. """,
  307. new SugarParameter("@Id", id))).FirstOrDefault();
  308. if (detail == null) throw Oops.Oh("记录不存在");
  309. var batches = await _db.Ado.SqlQueryAsync<OutsourceOrderBatchRow>(
  310. """
  311. SELECT
  312. b.Batch AS Batch,b.ItemNum AS ItemNum,b.SuppItem AS SuppItem,b.UM AS UM,b.Location AS Location,b.QtyOrded AS QtyOrded,
  313. b.QtyBO AS QtyBO,b.QtyReleased AS QtyReleased,b.QtyReceived AS QtyReceived,b.QtyReturned AS QtyReturned,b.LotSerial AS LotSerial
  314. FROM PurOrdDetailBatch b
  315. WHERE b.PurOrdDetailRecID=@RecId
  316. ORDER BY b.Batch
  317. """,
  318. new SugarParameter("@RecId", detail.Id));
  319. return new { detail, batches };
  320. }
  321. [DisplayName("保存委外采购明细")]
  322. [HttpPost("outsource-order/detail/save")]
  323. public async Task<object> SaveDetail([FromBody] OutsourceOrderDetailSaveInput input)
  324. {
  325. if (string.IsNullOrWhiteSpace(input.PurOrd)) throw Oops.Oh("采购订单号不能为空");
  326. if (input.PurOrdRecID <= 0) throw Oops.Oh("采购订单主键值不能为空");
  327. if (string.IsNullOrWhiteSpace(input.ItemNum)) throw Oops.Oh("请选择物料");
  328. if (!input.QtyOrded.HasValue || input.QtyOrded.Value <= 0) throw Oops.Oh("订单数量必须大于0");
  329. var item = (await _db.Ado.SqlQueryAsync<ItemLookupRow>(
  330. """
  331. SELECT ItemNum,Descr,UM,Location,Rev,Drawing
  332. FROM ItemMaster
  333. WHERE ItemNum=@ItemNum
  334. LIMIT 1
  335. """,
  336. new SugarParameter("@ItemNum", input.ItemNum.Trim()))).FirstOrDefault();
  337. if (item == null) throw Oops.Oh("物料不存在");
  338. try
  339. {
  340. _db.Ado.BeginTran();
  341. int detailId;
  342. int line;
  343. if (input.Id is null or <= 0)
  344. {
  345. line = input.Line ?? await _db.Ado.GetIntAsync("SELECT IFNULL(MAX(Line),0)+1 FROM PurOrdDetail WHERE PurOrd=@PurOrd", new SugarParameter("@PurOrd", input.PurOrd.Trim()));
  346. await _db.Ado.ExecuteCommandAsync(
  347. """
  348. INSERT INTO PurOrdDetail
  349. (
  350. QtyBO, RctCost, CreditTermsInt, UpdateCurrentCost, CumReceived1, CumReceived2,
  351. CumReceived3, CumReceived4, Disc, FixedPrice, InspectReq, SingleLot, SupplyPer,
  352. PurOrd, PST, PackingSlipQty, PayUMConv, PurCost, RctQty, QtyOrded, QtyReceived,
  353. QtyReturned, Active, QtyReleased, RctUMConversion, Scheduled, ScheduledChanged,
  354. SchedMRPReq, SafetyDays, SafetyHours, StdCost, Taxable, TaxIn, MaxTaxableAmt,
  355. TransportHours, UMConversion, VAT, IsActive, IsConfirm, Potype, IsChanged,
  356. TaxRate, IsRounding, ReceiptQty, BarCodeQty, IsClosed, QtyReturnedRefund, CumQtyBO,
  357. Line, ItemNum, Descr, UM, Rev, Drawing, Location, DueDate, LotSerial, PurOrdRecID, Status,
  358. CreateUser, CreateTime, UpdateUser, UpdateTime, tenant_id
  359. )
  360. VALUES
  361. (
  362. 0, 0, 0, 0, 0, 0,
  363. 0, 0, 0, 0, 0, 0, 0,
  364. @PurOrd, 0, 0, 1, 0, 0, @QtyOrded, 0,
  365. 0, 1, 0, 1, 0, 0,
  366. 0, 0, 0, 0, 1, 1, 0,
  367. 0, 1, 0, 1, 1, 'PW', 0,
  368. 0, 0, 0, 0, 0, 0, 0,
  369. @Line, @ItemNum, @Descr, @UM, @Rev, @Drawing, @Location, @DueDate, @LotSerial, @PurOrdRecID, 'R',
  370. @CreateUser, @CreateTime, @UpdateUser, @UpdateTime, @TenantId
  371. )
  372. """,
  373. new SugarParameter("@PurOrd", input.PurOrd.Trim()),
  374. new SugarParameter("@Line", line),
  375. new SugarParameter("@ItemNum", item.ItemNum),
  376. new SugarParameter("@Descr", item.Descr),
  377. new SugarParameter("@UM", item.UM),
  378. new SugarParameter("@Rev", item.Rev),
  379. new SugarParameter("@Drawing", item.Drawing),
  380. new SugarParameter("@Location", item.Location),
  381. new SugarParameter("@QtyOrded", input.QtyOrded.Value),
  382. new SugarParameter("@DueDate", ParseDate(input.DueDate)),
  383. new SugarParameter("@LotSerial", string.Empty),
  384. new SugarParameter("@PurOrdRecID", input.PurOrdRecID),
  385. new SugarParameter("@CreateUser", _userManager.Account),
  386. new SugarParameter("@CreateTime", DateTime.Now),
  387. new SugarParameter("@UpdateUser", _userManager.Account),
  388. new SugarParameter("@UpdateTime", DateTime.Now),
  389. new SugarParameter("@TenantId", _userManager.TenantId <= 0 ? null : _userManager.TenantId)
  390. );
  391. detailId = await _db.Ado.GetIntAsync("SELECT IFNULL(MAX(RecID),0) FROM PurOrdDetail WHERE PurOrd=@PurOrd AND Line=@Line", new SugarParameter("@PurOrd", input.PurOrd.Trim()), new SugarParameter("@Line", line));
  392. }
  393. else
  394. {
  395. detailId = input.Id.Value;
  396. line = input.Line ?? await _db.Ado.GetIntAsync("SELECT IFNULL(Line,0) FROM PurOrdDetail WHERE RecID=@Id", new SugarParameter("@Id", detailId));
  397. await _db.Ado.ExecuteCommandAsync(
  398. """
  399. UPDATE PurOrdDetail
  400. SET ItemNum=@ItemNum,Descr=@Descr,UM=@UM,Rev=@Rev,Drawing=@Drawing,Location=@Location,QtyOrded=@QtyOrded,DueDate=@DueDate,UpdateUser=@UpdateUser,UpdateTime=@UpdateTime
  401. WHERE RecID=@Id
  402. """,
  403. new SugarParameter("@Id", detailId),
  404. new SugarParameter("@ItemNum", item.ItemNum),
  405. new SugarParameter("@Descr", item.Descr),
  406. new SugarParameter("@UM", item.UM),
  407. new SugarParameter("@Rev", item.Rev),
  408. new SugarParameter("@Drawing", item.Drawing),
  409. new SugarParameter("@Location", item.Location),
  410. new SugarParameter("@QtyOrded", input.QtyOrded.Value),
  411. new SugarParameter("@DueDate", ParseDate(input.DueDate)),
  412. new SugarParameter("@UpdateUser", _userManager.Account),
  413. new SugarParameter("@UpdateTime", DateTime.Now)
  414. );
  415. }
  416. await _db.Ado.ExecuteCommandAsync("DELETE FROM PurOrdDetailBatch WHERE PurOrdDetailRecID=@RecID", new SugarParameter("@RecID", detailId));
  417. var batchNo = 1;
  418. foreach (var b in input.Batches)
  419. {
  420. if (string.IsNullOrWhiteSpace(b.SuppItem) && string.IsNullOrWhiteSpace(b.ItemNum))
  421. continue;
  422. var batchItem = (await _db.Ado.SqlQueryAsync<ItemLookupRow>(
  423. """
  424. SELECT ItemNum,Descr,UM,Location,Rev,Drawing
  425. FROM ItemMaster
  426. WHERE ItemNum=@ItemNum
  427. LIMIT 1
  428. """,
  429. new SugarParameter("@ItemNum", (b.SuppItem ?? b.ItemNum)!.Trim()))).FirstOrDefault();
  430. await _db.Ado.ExecuteCommandAsync(
  431. """
  432. INSERT INTO PurOrdDetailBatch
  433. (
  434. Domain, PurOrd, Potype, Line, Batch, ItemNum, SuppItem, UM, Location,
  435. QtyOrded, QtyBO, QtyReleased, QtyReceived, QtyReturned, LotSerial, PurOrdDetailRecID,
  436. CreateUser, CreateTime, UpdateUser, UpdateTime, tenant_id
  437. )
  438. VALUES
  439. (
  440. '', @PurOrd, 'PW', @Line, @Batch, @ItemNum, @SuppItem, @UM, @Location,
  441. @QtyOrded, @QtyBO, @QtyReleased, @QtyReceived, @QtyReturned, @LotSerial, @PurOrdDetailRecID,
  442. @CreateUser, @CreateTime, @UpdateUser, @UpdateTime, @TenantId
  443. )
  444. """,
  445. new SugarParameter("@PurOrd", input.PurOrd.Trim()),
  446. new SugarParameter("@Line", line),
  447. new SugarParameter("@Batch", b.Batch ?? batchNo),
  448. new SugarParameter("@ItemNum", batchItem?.ItemNum ?? b.ItemNum?.Trim()),
  449. new SugarParameter("@SuppItem", b.SuppItem?.Trim()),
  450. new SugarParameter("@UM", b.UM ?? batchItem?.UM),
  451. new SugarParameter("@Location", b.Location ?? batchItem?.Location),
  452. new SugarParameter("@QtyOrded", b.QtyOrded ?? 0),
  453. new SugarParameter("@QtyBO", b.QtyBO.GetValueOrDefault(0)),
  454. new SugarParameter("@QtyReleased", b.QtyReleased ?? 0),
  455. new SugarParameter("@QtyReceived", b.QtyReceived ?? 0),
  456. new SugarParameter("@QtyReturned", b.QtyReturned ?? 0),
  457. new SugarParameter("@LotSerial", b.LotSerial?.Trim()),
  458. new SugarParameter("@PurOrdDetailRecID", detailId),
  459. new SugarParameter("@CreateUser", _userManager.Account),
  460. new SugarParameter("@CreateTime", DateTime.Now),
  461. new SugarParameter("@UpdateUser", _userManager.Account),
  462. new SugarParameter("@UpdateTime", DateTime.Now),
  463. new SugarParameter("@TenantId", _userManager.TenantId <= 0 ? null : _userManager.TenantId)
  464. );
  465. batchNo++;
  466. }
  467. _db.Ado.CommitTran();
  468. return new { id = detailId, message = "保存成功" };
  469. }
  470. catch
  471. {
  472. _db.Ado.RollbackTran();
  473. throw;
  474. }
  475. }
  476. [DisplayName("删除委外采购明细")]
  477. [HttpPost("outsource-order/detail/delete/{id:int}")]
  478. public async Task<object> DeleteDetail(int id)
  479. {
  480. var status = await _db.Ado.GetStringAsync("SELECT Status FROM PurOrdDetail WHERE RecID=@Id LIMIT 1", new SugarParameter("@Id", id));
  481. if (string.Equals(status, "C", StringComparison.OrdinalIgnoreCase))
  482. throw Oops.Oh("关闭状态不允许删除");
  483. try
  484. {
  485. _db.Ado.BeginTran();
  486. await _db.Ado.ExecuteCommandAsync("DELETE FROM PurOrdDetailBatch WHERE PurOrdDetailRecID=@Id", new SugarParameter("@Id", id));
  487. await _db.Ado.ExecuteCommandAsync("DELETE FROM PurOrdDetail WHERE RecID=@Id", new SugarParameter("@Id", id));
  488. _db.Ado.CommitTran();
  489. }
  490. catch
  491. {
  492. _db.Ado.RollbackTran();
  493. throw;
  494. }
  495. return new { message = "删除成功" };
  496. }
  497. [DisplayName("委外订单新增明细初始化")]
  498. [HttpGet("outsource-order/detail/init")]
  499. public async Task<object> GetDetailInit([FromQuery] string purOrd)
  500. {
  501. if (string.IsNullOrWhiteSpace(purOrd)) throw Oops.Oh("采购订单号不能为空");
  502. var row = (await _db.Ado.SqlQueryAsync<DetailInitRow>(
  503. """
  504. SELECT p.`Usage` AS `Usage`, IFNULL(MAX(d.Line),0)+1 AS Line, p.RecID AS PurOrdRecID, p.PurOrd AS PurOrd
  505. FROM PurOrdMaster p
  506. LEFT JOIN PurOrdDetail d ON p.PurOrd=d.PurOrd
  507. WHERE p.PurOrd=@PurOrd
  508. GROUP BY p.`Usage`, p.RecID, p.PurOrd
  509. LIMIT 1
  510. """,
  511. new SugarParameter("@PurOrd", purOrd.Trim()))).FirstOrDefault();
  512. return row ?? throw Oops.Oh("采购订单不存在");
  513. }
  514. [DisplayName("选择物料列表")]
  515. [HttpGet("outsource-order/items")]
  516. public async Task<object> GetItemList([FromQuery] ItemSelectListInput input)
  517. {
  518. var page = input.Page <= 0 ? 1 : input.Page;
  519. var pageSize = input.PageSize <= 0 ? 10 : input.PageSize;
  520. var offset = (page - 1) * pageSize;
  521. var where = new List<string> { "1=1" };
  522. var pars = new List<SugarParameter>();
  523. if (!string.IsNullOrWhiteSpace(input.ItemNum))
  524. {
  525. where.Add("ItemNum LIKE @ItemNum");
  526. pars.Add(new SugarParameter("@ItemNum", $"%{input.ItemNum.Trim()}%"));
  527. }
  528. if (!string.IsNullOrWhiteSpace(input.Descr))
  529. {
  530. where.Add("Descr LIKE @Descr");
  531. pars.Add(new SugarParameter("@Descr", $"%{input.Descr.Trim()}%"));
  532. }
  533. var fromSql = $"FROM ItemMaster WHERE {string.Join(" AND ", where)}";
  534. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) {fromSql}", pars);
  535. var list = await _db.Ado.SqlQueryAsync<ItemLookupRow>(
  536. $"""
  537. SELECT RecID, ItemNum, Descr, Descr1, UM, Location, Rev, Drawing
  538. {fromSql}
  539. ORDER BY {BuildItemOrderBy(input.SortField, input.SortOrder)}
  540. LIMIT {pageSize} OFFSET {offset}
  541. """,
  542. pars);
  543. return new { total, page, pageSize, list };
  544. }
  545. [DisplayName("采购组下拉")]
  546. [HttpGet("outsource-order/options/buyers")]
  547. public async Task<object> GetBuyerOptions()
  548. {
  549. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  550. """
  551. SELECT Employee AS Value, CONCAT(TRIM(Employee),' ',TRIM(IFNULL(Name,''))) AS Label
  552. FROM EmployeeMaster
  553. ORDER BY Employee
  554. """);
  555. return new { list };
  556. }
  557. [DisplayName("供应商下拉")]
  558. [HttpGet("outsource-order/options/suppliers")]
  559. public async Task<object> GetSupplierOptions()
  560. {
  561. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  562. """
  563. SELECT Supp AS Value, CONCAT(TRIM(Supp),' ',TRIM(IFNULL(SortName,''))) AS Label
  564. FROM SuppMaster
  565. ORDER BY Supp
  566. """);
  567. return new { list };
  568. }
  569. [DisplayName("部门下拉")]
  570. [HttpGet("outsource-order/options/departments")]
  571. public async Task<object> GetDepartmentOptions()
  572. {
  573. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  574. """
  575. SELECT Department AS Value, CONCAT(TRIM(Department),' ',TRIM(IFNULL(Descr,''))) AS Label
  576. FROM DepartmentMaster
  577. ORDER BY Department
  578. """);
  579. return new { list };
  580. }
  581. [DisplayName("库位下拉")]
  582. [HttpGet("outsource-order/options/locations")]
  583. public async Task<object> GetLocationOptions()
  584. {
  585. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  586. """
  587. SELECT location AS Value, CONCAT(TRIM(location),' ',TRIM(IFNULL(descr,''))) AS Label
  588. FROM LocationMaster
  589. WHERE IFNULL(typed,'')<>'Supp'
  590. ORDER BY location
  591. """);
  592. return new { list };
  593. }
  594. private static DateTime? ParseDate(string? v) => DateTime.TryParse(v, out var d) ? d : null;
  595. private static string BuildOrderBy(string? sortField, string? sortOrder)
  596. {
  597. var dir = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase) ? "ASC" : "DESC";
  598. return sortField?.ToLowerInvariant() switch
  599. {
  600. "purord" => $"p.PurOrd {dir}",
  601. "suppname" => $"p.Supp {dir}",
  602. "buyer" => $"p.Buyer {dir}",
  603. "orddate" => $"p.OrdDate {dir}",
  604. "duedate" => $"p.DueDate {dir}",
  605. "curr" => $"p.Curr {dir}",
  606. "status" => $"p.Status {dir}",
  607. _ => "p.RecID DESC"
  608. };
  609. }
  610. private static string BuildDetailOrderBy(string? sortField, string? sortOrder)
  611. {
  612. var dir = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase) ? "ASC" : "DESC";
  613. return sortField?.ToLowerInvariant() switch
  614. {
  615. "purord" => $"d.PurOrd {dir}",
  616. "line" => $"d.Line {dir}",
  617. "itemnum" => $"d.ItemNum {dir}",
  618. "descr" => $"i.Descr {dir}",
  619. "um" => $"d.UM {dir}",
  620. "rev" => $"d.Rev {dir}",
  621. "drawing" => $"d.Drawing {dir}",
  622. "qtyorded" => $"d.QtyOrded {dir}",
  623. "qtyreceived" => $"d.QtyReceived {dir}",
  624. "qtyreleased" => $"d.QtyReleased {dir}",
  625. "duedate" => $"d.DueDate {dir}",
  626. "location" => $"d.Location {dir}",
  627. "workord" => $"d.WorkOrd {dir}",
  628. "lotserial" => $"d.LotSerial {dir}",
  629. "status" => $"d.Status {dir}",
  630. _ => "d.RecID DESC"
  631. };
  632. }
  633. private static string BuildItemOrderBy(string? sortField, string? sortOrder)
  634. {
  635. var dir = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase) ? "ASC" : "DESC";
  636. return sortField?.ToLowerInvariant() switch
  637. {
  638. "itemnum" => $"ItemNum {dir}",
  639. "descr" => $"Descr {dir}",
  640. "descr1" => $"Descr1 {dir}",
  641. "um" => $"UM {dir}",
  642. "location" => $"Location {dir}",
  643. "rev" => $"Rev {dir}",
  644. "drawing" => $"Drawing {dir}",
  645. _ => "RecID DESC"
  646. };
  647. }
  648. private sealed class OutsourceOrderListRow
  649. {
  650. public int Id { get; set; }
  651. public string? PurOrd { get; set; }
  652. public string? SuppName { get; set; }
  653. public string? Potype { get; set; }
  654. public string? ReqBy { get; set; }
  655. public string? Buyer { get; set; }
  656. public string? DepartmentDescr { get; set; }
  657. public DateTime? OrdDate { get; set; }
  658. public DateTime? DueDate { get; set; }
  659. public string? Curr { get; set; }
  660. public string? Status { get; set; }
  661. public string? Supp { get; set; }
  662. public string? BuyerCode { get; set; }
  663. public string? Department { get; set; }
  664. public string? Usage { get; set; }
  665. public string? Remark { get; set; }
  666. }
  667. private sealed class OutsourceOrderDetailRow
  668. {
  669. public int Id { get; set; }
  670. public string? PurOrd { get; set; }
  671. public int? Line { get; set; }
  672. public string? ItemNum { get; set; }
  673. public string? Descr { get; set; }
  674. public string? UM { get; set; }
  675. public string? Rev { get; set; }
  676. public string? Drawing { get; set; }
  677. public decimal? QtyOrded { get; set; }
  678. public decimal? QtyReceived { get; set; }
  679. public decimal? QtyReleased { get; set; }
  680. public DateTime? DueDate { get; set; }
  681. public string? Location { get; set; }
  682. public string? WorkOrd { get; set; }
  683. public string? LotSerial { get; set; }
  684. public string? Status { get; set; }
  685. public int? PurOrdRecID { get; set; }
  686. }
  687. private sealed class OutsourceOrderBatchRow
  688. {
  689. public int? Batch { get; set; }
  690. public string? ItemNum { get; set; }
  691. public string? SuppItem { get; set; }
  692. public string? UM { get; set; }
  693. public string? Location { get; set; }
  694. public decimal? QtyOrded { get; set; }
  695. public decimal? QtyBO { get; set; }
  696. public decimal? QtyReleased { get; set; }
  697. public decimal? QtyReceived { get; set; }
  698. public decimal? QtyReturned { get; set; }
  699. public string? LotSerial { get; set; }
  700. }
  701. private sealed class ItemLookupRow
  702. {
  703. public long RecID { get; set; }
  704. public string? ItemNum { get; set; }
  705. public string? Descr { get; set; }
  706. public string? Descr1 { get; set; }
  707. public string? UM { get; set; }
  708. public string? Location { get; set; }
  709. public string? Rev { get; set; }
  710. public string? Drawing { get; set; }
  711. }
  712. private sealed class OptionRow
  713. {
  714. public string? Value { get; set; }
  715. public string? Label { get; set; }
  716. }
  717. private sealed class DetailInitRow
  718. {
  719. public string? Usage { get; set; }
  720. public int Line { get; set; }
  721. public int PurOrdRecID { get; set; }
  722. public string? PurOrd { get; set; }
  723. }
  724. }