OutsourceOrderService.cs 31 KB

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