OutsourceOrderService.cs 36 KB

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