OutsourceOrderService.cs 37 KB

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