DemandOrderService.cs 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666
  1. namespace Admin.NET.Plugin.AiDOP.Supply;
  2. /// <summary>
  3. /// 要货令服务(PurOrdMaster/PurOrdDetail,Potype=PO,ReqBy=DO)
  4. /// </summary>
  5. [ApiDescriptionSettings(Order = 311, Description = "要货令")]
  6. [Route("api/Supply")]
  7. [AllowAnonymous]
  8. [NonUnify]
  9. public class DemandOrderService : IDynamicApiController, ITransient
  10. {
  11. private const int PurOrdSerialWidth = 3;
  12. private readonly ISqlSugarClient _db;
  13. private readonly UserManager _userManager;
  14. public DemandOrderService(ISqlSugarClient db, UserManager userManager)
  15. {
  16. _db = db;
  17. _userManager = userManager;
  18. }
  19. [DisplayName("要货令列表")]
  20. [HttpGet("demand-order/list")]
  21. public async Task<object> GetList([FromQuery] DemandOrderListInput 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='PO'",
  29. "IFNULL(p.ReqBy,'')='DO'",
  30. "p.IsActive<=1"
  31. };
  32. var pars = new List<SugarParameter>();
  33. if (!string.IsNullOrWhiteSpace(input.PurOrd))
  34. {
  35. where.Add("p.PurOrd LIKE @PurOrd");
  36. pars.Add(new SugarParameter("@PurOrd", $"%{input.PurOrd.Trim()}%"));
  37. }
  38. if (!string.IsNullOrWhiteSpace(input.Buyer))
  39. {
  40. where.Add("p.Buyer=@Buyer");
  41. pars.Add(new SugarParameter("@Buyer", input.Buyer.Trim()));
  42. }
  43. if (!string.IsNullOrWhiteSpace(input.Supp))
  44. {
  45. where.Add("p.Supp=@Supp");
  46. pars.Add(new SugarParameter("@Supp", input.Supp.Trim()));
  47. }
  48. if (!string.IsNullOrWhiteSpace(input.Status))
  49. {
  50. var st = input.Status.Trim();
  51. if (string.Equals(st, "R", StringComparison.OrdinalIgnoreCase))
  52. where.Add("(IFNULL(LENGTH(p.Status),0)=0 OR p.Buyer IS NULL)");
  53. else
  54. {
  55. where.Add("p.Status=@Status");
  56. pars.Add(new SugarParameter("@Status", st));
  57. }
  58. }
  59. var fromSql = $"""
  60. FROM PurOrdMaster p
  61. LEFT JOIN SuppMaster s ON p.Supp=s.Supp
  62. LEFT JOIN DepartmentMaster d ON p.Department=d.Department
  63. LEFT JOIN EmployeeMaster e ON p.Buyer=e.Employee
  64. WHERE {string.Join(" AND ", where)}
  65. """;
  66. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) {fromSql}", pars);
  67. var list = await _db.Ado.SqlQueryAsync<DemandOrderListRow>(
  68. $"""
  69. SELECT
  70. p.RecID AS Id,
  71. p.PurOrd AS PurOrd,
  72. CONCAT(TRIM(IFNULL(p.Supp,'')),' ',TRIM(IFNULL(s.SortName,''))) AS SuppName,
  73. p.Potype AS Potype,
  74. p.ReqBy AS ReqBy,
  75. CONCAT(TRIM(IFNULL(p.Buyer,'')),' ',TRIM(IFNULL(e.Name,''))) AS Buyer,
  76. CONCAT(TRIM(IFNULL(p.Department,'')),' ',TRIM(IFNULL(d.Descr,''))) AS DepartmentDescr,
  77. p.OrdDate AS OrdDate,
  78. p.Contract AS Contract,
  79. p.DeliverTo AS DeliverTo,
  80. p.DueDate AS DueDate,
  81. p.Contact AS Contact,
  82. p.ShipTo AS ShipTo,
  83. p.Curr AS Curr,
  84. p.TaxClass AS TaxClass,
  85. p.TaxIn AS TaxIn,
  86. CASE
  87. WHEN IFNULL(LENGTH(p.Status),0)=0 OR p.Buyer IS NULL THEN 'R'
  88. ELSE p.Status
  89. END AS Status,
  90. p.Remark AS Remark,
  91. p.Supp AS Supp,
  92. p.Buyer AS BuyerCode,
  93. p.Department AS Department,
  94. p.`Usage` AS `Usage`,
  95. p.FSTID AS Fstid
  96. {fromSql}
  97. ORDER BY {BuildOrderBy(input.SortField, input.SortOrder)}
  98. LIMIT {pageSize} OFFSET {offset}
  99. """,
  100. pars);
  101. return new { total, page, pageSize, list };
  102. }
  103. [DisplayName("要货令详情")]
  104. [HttpGet("demand-order/{id:int}")]
  105. public async Task<object> GetDetail(int id)
  106. {
  107. var master = await GetMasterRowAsync(id);
  108. var details = await _db.Ado.SqlQueryAsync<DemandOrderDetailRow>(
  109. """
  110. SELECT
  111. d.RecID AS Id,
  112. d.Line AS Line,
  113. d.ItemNum AS ItemNum,
  114. d.UM AS UM,
  115. d.Location AS Location,
  116. d.QtyOrded AS QtyOrded,
  117. d.RctQty AS RctQty,
  118. d.ReceiptQty AS ReceiptQty,
  119. d.DueDate AS DueDate,
  120. d.Rev AS Rev,
  121. d.Drawing AS Drawing,
  122. d.LotSerial AS LotSerial,
  123. d.Potype AS Potype,
  124. d.PurOrd AS PurOrd,
  125. d.PurOrdRecID AS PurOrdRecID
  126. FROM PurOrdDetail d
  127. WHERE d.PurOrd=@PurOrd AND d.Potype='PO'
  128. ORDER BY d.Line
  129. """,
  130. new SugarParameter("@PurOrd", master.PurOrd));
  131. return new { master, details };
  132. }
  133. [DisplayName("保存要货令")]
  134. [HttpPost("demand-order/save")]
  135. public async Task<object> Save([FromBody] DemandOrderSaveInput input)
  136. {
  137. if (string.IsNullOrWhiteSpace(input.Supp)) throw Oops.Oh("请选择供应商");
  138. if (string.IsNullOrWhiteSpace(input.Department)) throw Oops.Oh("请选择部门");
  139. var now = DateTime.Now;
  140. var ordDate = ParseDate(input.OrdDate) ?? now.Date;
  141. var buyer = string.IsNullOrWhiteSpace(input.Buyer) ? "110" : input.Buyer.Trim();
  142. var reqBy = string.IsNullOrWhiteSpace(input.ReqBy) ? "DO" : input.ReqBy.Trim();
  143. var usage = string.IsNullOrWhiteSpace(input.Usage) ? "标准" : input.Usage.Trim();
  144. var fstid = string.Equals(usage, "VMI", StringComparison.OrdinalIgnoreCase) ? "3" : "1";
  145. if (!string.IsNullOrWhiteSpace(input.Fstid)) fstid = input.Fstid.Trim();
  146. try
  147. {
  148. _db.Ado.BeginTran();
  149. int masterId;
  150. string purOrd;
  151. if (input.Id is null or <= 0)
  152. {
  153. purOrd = await PurOrdNumberGenerator.NextAsync(_db, "DO", DateTime.Today, PurOrdSerialWidth);
  154. await _db.Ado.ExecuteCommandAsync(
  155. """
  156. INSERT INTO PurOrdMaster
  157. (PurOrd,OrdDate,Supp,ReqBy,Buyer,Department,Curr,`Usage`,Remark,Potype,FSTID,Status,IsActive,
  158. CreateUser,CreateTime,UpdateUser,UpdateTime,Domain,tenant_id)
  159. VALUES
  160. (@PurOrd,@OrdDate,@Supp,@ReqBy,@Buyer,@Department,@Curr,@Usage,@Remark,'PO',@FSTID,'R',1,
  161. @CreateUser,@CreateTime,@UpdateUser,@UpdateTime,@Domain,@TenantId)
  162. """,
  163. new SugarParameter("@PurOrd", purOrd),
  164. new SugarParameter("@OrdDate", ordDate),
  165. new SugarParameter("@Supp", input.Supp.Trim()),
  166. new SugarParameter("@ReqBy", reqBy),
  167. new SugarParameter("@Buyer", buyer),
  168. new SugarParameter("@Department", input.Department.Trim()),
  169. new SugarParameter("@Curr", input.Curr?.Trim()),
  170. new SugarParameter("@Usage", usage),
  171. new SugarParameter("@Remark", input.Remark?.Trim()),
  172. new SugarParameter("@FSTID", fstid),
  173. new SugarParameter("@CreateUser", _userManager.Account),
  174. new SugarParameter("@CreateTime", now),
  175. new SugarParameter("@UpdateUser", _userManager.Account),
  176. new SugarParameter("@UpdateTime", now),
  177. new SugarParameter("@Domain", "100"),
  178. new SugarParameter("@TenantId", _userManager.TenantId <= 0 ? null : _userManager.TenantId)
  179. );
  180. masterId = await _db.Ado.GetIntAsync(
  181. "SELECT IFNULL(MAX(RecID),0) FROM PurOrdMaster WHERE PurOrd=@PurOrd",
  182. new SugarParameter("@PurOrd", purOrd));
  183. }
  184. else
  185. {
  186. masterId = input.Id.Value;
  187. var exists = await _db.Ado.GetIntAsync(
  188. "SELECT COUNT(1) FROM PurOrdMaster WHERE RecID=@Id AND Potype='PO' AND IFNULL(ReqBy,'')='DO'",
  189. new SugarParameter("@Id", masterId));
  190. if (exists <= 0) throw Oops.Oh("记录不存在");
  191. purOrd = await _db.Ado.GetStringAsync(
  192. "SELECT PurOrd FROM PurOrdMaster WHERE RecID=@Id LIMIT 1",
  193. new SugarParameter("@Id", masterId)) ?? throw Oops.Oh("记录不存在");
  194. await _db.Ado.ExecuteCommandAsync(
  195. """
  196. UPDATE PurOrdMaster
  197. SET OrdDate=@OrdDate,Supp=@Supp,ReqBy=@ReqBy,Buyer=@Buyer,Department=@Department,
  198. Curr=@Curr,`Usage`=@Usage,Remark=@Remark,FSTID=@FSTID,
  199. UpdateUser=@UpdateUser,UpdateTime=@UpdateTime
  200. WHERE RecID=@Id
  201. """,
  202. new SugarParameter("@Id", masterId),
  203. new SugarParameter("@OrdDate", ordDate),
  204. new SugarParameter("@Supp", input.Supp.Trim()),
  205. new SugarParameter("@ReqBy", reqBy),
  206. new SugarParameter("@Buyer", buyer),
  207. new SugarParameter("@Department", input.Department.Trim()),
  208. new SugarParameter("@Curr", input.Curr?.Trim()),
  209. new SugarParameter("@Usage", usage),
  210. new SugarParameter("@Remark", input.Remark?.Trim()),
  211. new SugarParameter("@FSTID", fstid),
  212. new SugarParameter("@UpdateUser", _userManager.Account),
  213. new SugarParameter("@UpdateTime", now)
  214. );
  215. }
  216. await SaveDetailsAsync(masterId, purOrd, input.Details ?? new List<DemandOrderDetailInput>());
  217. _db.Ado.CommitTran();
  218. return new { id = masterId, purOrd, message = input.Id is null or <= 0 ? "新增成功" : "编辑成功" };
  219. }
  220. catch
  221. {
  222. _db.Ado.RollbackTran();
  223. throw;
  224. }
  225. }
  226. [DisplayName("删除要货令")]
  227. [HttpPost("demand-order/delete/{id:int}")]
  228. public async Task<object> Delete(int id, [FromQuery] string purOrd)
  229. {
  230. if (string.IsNullOrWhiteSpace(purOrd)) throw Oops.Oh("缺少要货令单号");
  231. var hasReceipt = await _db.Ado.GetIntAsync(
  232. """
  233. SELECT COUNT(1)
  234. FROM PurOrdRctDetail
  235. WHERE OrdNbr=@PurOrd AND IFNULL(QtyReceived,0)>0
  236. """,
  237. new SugarParameter("@PurOrd", purOrd.Trim()));
  238. if (hasReceipt > 0) throw Oops.Oh("存在收货数量,不允许删除");
  239. var hasDetailRct = await _db.Ado.GetIntAsync(
  240. """
  241. SELECT COUNT(1)
  242. FROM PurOrdDetail
  243. WHERE PurOrd=@PurOrd AND IFNULL(RctQty,0)>0
  244. """,
  245. new SugarParameter("@PurOrd", purOrd.Trim()));
  246. if (hasDetailRct > 0) throw Oops.Oh("存在收货数量,不允许删除");
  247. var hasShip = await _db.Ado.GetIntAsync(
  248. "SELECT COUNT(1) FROM scm_shdzb WHERE po_bill=@PurOrd",
  249. new SugarParameter("@PurOrd", purOrd.Trim()));
  250. if (hasShip > 0) throw Oops.Oh("存在发货单,不允许删除");
  251. try
  252. {
  253. _db.Ado.BeginTran();
  254. await _db.Ado.ExecuteCommandAsync(
  255. "DELETE FROM PurOrdDetail WHERE PurOrd=@PurOrd AND Potype='PO'",
  256. new SugarParameter("@PurOrd", purOrd.Trim()));
  257. await _db.Ado.ExecuteCommandAsync(
  258. "DELETE FROM PurOrdMaster WHERE RecID=@Id AND PurOrd=@PurOrd",
  259. new SugarParameter("@Id", id),
  260. new SugarParameter("@PurOrd", purOrd.Trim()));
  261. _db.Ado.CommitTran();
  262. }
  263. catch
  264. {
  265. _db.Ado.RollbackTran();
  266. throw;
  267. }
  268. return new { message = "删除成功" };
  269. }
  270. [DisplayName("选择物料列表")]
  271. [HttpGet("demand-order/items")]
  272. public async Task<object> GetItemList([FromQuery] DemandOrderItemListInput input)
  273. {
  274. var page = input.Page <= 0 ? 1 : input.Page;
  275. var pageSize = input.PageSize <= 0 ? 10 : input.PageSize;
  276. var offset = (page - 1) * pageSize;
  277. var where = new List<string> { "1=1" };
  278. var pars = new List<SugarParameter>();
  279. if (!string.IsNullOrWhiteSpace(input.ItemNum))
  280. {
  281. where.Add("ItemNum LIKE @ItemNum");
  282. pars.Add(new SugarParameter("@ItemNum", $"%{input.ItemNum.Trim()}%"));
  283. }
  284. if (!string.IsNullOrWhiteSpace(input.Descr))
  285. {
  286. where.Add("Descr LIKE @Descr");
  287. pars.Add(new SugarParameter("@Descr", $"%{input.Descr.Trim()}%"));
  288. }
  289. var fromSql = $"FROM ItemMaster WHERE {string.Join(" AND ", where)}";
  290. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) {fromSql}", pars);
  291. var list = await _db.Ado.SqlQueryAsync<ItemLookupRow>(
  292. $"""
  293. SELECT RecID, ItemNum, Descr, Descr1, UM, Location, Rev, Drawing
  294. {fromSql}
  295. ORDER BY {BuildItemOrderBy(input.SortField, input.SortOrder)}
  296. LIMIT {pageSize} OFFSET {offset}
  297. """,
  298. pars);
  299. return new { total, page, pageSize, list };
  300. }
  301. [DisplayName("采购组下拉")]
  302. [HttpGet("demand-order/options/buyers")]
  303. public async Task<object> GetBuyerOptions()
  304. {
  305. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  306. """
  307. SELECT Employee AS Value, CONCAT(TRIM(Employee),' ',TRIM(IFNULL(Name,''))) AS Label
  308. FROM EmployeeMaster
  309. ORDER BY Employee
  310. """);
  311. return new { list };
  312. }
  313. [DisplayName("供应商下拉")]
  314. [HttpGet("demand-order/options/suppliers")]
  315. public async Task<object> GetSupplierOptions()
  316. {
  317. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  318. """
  319. SELECT Supp AS Value, CONCAT(TRIM(Supp),' ',TRIM(IFNULL(SortName,''))) AS Label
  320. FROM SuppMaster
  321. ORDER BY Supp
  322. """);
  323. return new { list };
  324. }
  325. [DisplayName("部门下拉")]
  326. [HttpGet("demand-order/options/departments")]
  327. public async Task<object> GetDepartmentOptions()
  328. {
  329. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  330. """
  331. SELECT Department AS Value, CONCAT(TRIM(Department),' ',TRIM(IFNULL(Descr,''))) AS Label
  332. FROM DepartmentMaster
  333. ORDER BY Department
  334. """);
  335. return new { list };
  336. }
  337. [DisplayName("币别下拉")]
  338. [HttpGet("demand-order/options/curr")]
  339. public async Task<object> GetCurrOptions()
  340. {
  341. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  342. """
  343. SELECT Val AS Value, CONCAT(TRIM(Val),' ',TRIM(IFNULL(Comments,''))) AS Label
  344. FROM GeneralizedCodeMaster
  345. WHERE FldName='Curr'
  346. ORDER BY Val
  347. """);
  348. return new { list };
  349. }
  350. [DisplayName("库位下拉")]
  351. [HttpGet("demand-order/options/locations")]
  352. public async Task<object> GetLocationOptions()
  353. {
  354. var list = await _db.Ado.SqlQueryAsync<OptionRow>(
  355. """
  356. SELECT location AS Value, CONCAT(TRIM(location),' ',TRIM(IFNULL(descr,''))) AS Label
  357. FROM LocationMaster
  358. WHERE IFNULL(typed,'')<>'Supp'
  359. ORDER BY location
  360. """);
  361. return new { list };
  362. }
  363. private async Task SaveDetailsAsync(int masterId, string purOrd, List<DemandOrderDetailInput> details)
  364. {
  365. var dbDetails = await _db.Ado.SqlQueryAsync<DemandOrderDetailRow>(
  366. """
  367. SELECT RecID AS Id, Line AS Line
  368. FROM PurOrdDetail
  369. WHERE PurOrd=@PurOrd AND Potype='PO'
  370. """,
  371. new SugarParameter("@PurOrd", purOrd));
  372. var dbById = dbDetails.ToDictionary(d => d.Id);
  373. var inputIds = new HashSet<int>(details.Where(d => d.Id is > 0).Select(d => d.Id!.Value));
  374. for (var i = 0; i < details.Count; i++)
  375. {
  376. var d = details[i];
  377. if (string.IsNullOrWhiteSpace(d.ItemNum)) continue;
  378. if (!d.QtyOrded.HasValue || d.QtyOrded.Value <= 0) throw Oops.Oh("订单数量必须大于0");
  379. var item = (await _db.Ado.SqlQueryAsync<ItemLookupRow>(
  380. """
  381. SELECT ItemNum, Descr, UM, Location, Rev, Drawing
  382. FROM ItemMaster
  383. WHERE ItemNum=@ItemNum
  384. LIMIT 1
  385. """,
  386. new SugarParameter("@ItemNum", d.ItemNum.Trim()))).FirstOrDefault();
  387. if (item == null) throw Oops.Oh($"物料不存在:{d.ItemNum}");
  388. var um = d.UM ?? item.UM;
  389. var location = d.Location ?? item.Location;
  390. var rev = d.Rev ?? item.Rev;
  391. var drawing = d.Drawing ?? item.Drawing;
  392. var dueDate = ParseDate(d.DueDate);
  393. var now = DateTime.Now;
  394. if (d.Id is > 0 && dbById.ContainsKey(d.Id.Value))
  395. {
  396. await _db.Ado.ExecuteCommandAsync(
  397. """
  398. UPDATE PurOrdDetail
  399. SET ItemNum=@ItemNum,Descr=@Descr,UM=@UM,Location=@Location,QtyOrded=@QtyOrded,
  400. DueDate=@DueDate,Rev=@Rev,Drawing=@Drawing,LotSerial=@LotSerial,
  401. UpdateUser=@UpdateUser,UpdateTime=@UpdateTime
  402. WHERE RecID=@Id
  403. """,
  404. new SugarParameter("@Id", d.Id.Value),
  405. new SugarParameter("@ItemNum", item.ItemNum),
  406. new SugarParameter("@Descr", item.Descr),
  407. new SugarParameter("@UM", um),
  408. new SugarParameter("@Location", location),
  409. new SugarParameter("@QtyOrded", d.QtyOrded.Value),
  410. new SugarParameter("@DueDate", dueDate),
  411. new SugarParameter("@Rev", rev),
  412. new SugarParameter("@Drawing", drawing),
  413. new SugarParameter("@LotSerial", d.LotSerial?.Trim()),
  414. new SugarParameter("@UpdateUser", _userManager.Account),
  415. new SugarParameter("@UpdateTime", now)
  416. );
  417. }
  418. else
  419. {
  420. var line = d.Line ?? await _db.Ado.GetIntAsync(
  421. "SELECT IFNULL(MAX(Line),0)+1 FROM PurOrdDetail WHERE PurOrd=@PurOrd",
  422. new SugarParameter("@PurOrd", purOrd));
  423. await _db.Ado.ExecuteCommandAsync(
  424. """
  425. INSERT INTO PurOrdDetail
  426. (
  427. QtyBO, RctCost, CreditTermsInt, UpdateCurrentCost, CumReceived1, CumReceived2,
  428. CumReceived3, CumReceived4, Disc, FixedPrice, InspectReq, SingleLot, SupplyPer,
  429. PurOrd, PST, PackingSlipQty, PayUMConv, PurCost, RctQty, QtyOrded, QtyReceived,
  430. QtyReturned, Active, QtyReleased, RctUMConversion, Scheduled, ScheduledChanged,
  431. SchedMRPReq, SafetyDays, SafetyHours, StdCost, Taxable, TaxIn, MaxTaxableAmt,
  432. TransportHours, UMConversion, VAT, IsActive, IsConfirm, Potype, IsChanged,
  433. TaxRate, IsRounding, ReceiptQty, BarCodeQty, IsClosed, QtyReturnedRefund, CumQtyBO,
  434. Line, ItemNum, Descr, UM, Rev, Drawing, Location, DueDate, LotSerial, PurOrdRecID, Status,
  435. CreateUser, CreateTime, UpdateUser, UpdateTime, tenant_id
  436. )
  437. VALUES
  438. (
  439. 0, 0, 0, 0, 0, 0,
  440. 0, 0, 0, 0, 0, 0, 0,
  441. @PurOrd, 0, 0, 1, 0, 0, @QtyOrded, 0,
  442. 0, 1, 0, 1, 0, 0,
  443. 0, 0, 0, 0, 1, 1, 0,
  444. 0, 1, 0, 1, 1, 'PO', 0,
  445. 0, 0, 0, 0, 0, 0, 0,
  446. @Line, @ItemNum, @Descr, @UM, @Rev, @Drawing, @Location, @DueDate, @LotSerial, @PurOrdRecID, 'R',
  447. @CreateUser, @CreateTime, @UpdateUser, @UpdateTime, @TenantId
  448. )
  449. """,
  450. new SugarParameter("@PurOrd", purOrd),
  451. new SugarParameter("@Line", line),
  452. new SugarParameter("@ItemNum", item.ItemNum),
  453. new SugarParameter("@Descr", item.Descr),
  454. new SugarParameter("@UM", um),
  455. new SugarParameter("@Rev", rev),
  456. new SugarParameter("@Drawing", drawing),
  457. new SugarParameter("@Location", location),
  458. new SugarParameter("@QtyOrded", d.QtyOrded.Value),
  459. new SugarParameter("@DueDate", dueDate),
  460. new SugarParameter("@LotSerial", d.LotSerial?.Trim()),
  461. new SugarParameter("@PurOrdRecID", masterId),
  462. new SugarParameter("@CreateUser", _userManager.Account),
  463. new SugarParameter("@CreateTime", now),
  464. new SugarParameter("@UpdateUser", _userManager.Account),
  465. new SugarParameter("@UpdateTime", now),
  466. new SugarParameter("@TenantId", _userManager.TenantId <= 0 ? null : _userManager.TenantId)
  467. );
  468. }
  469. }
  470. foreach (var toDelete in dbDetails.Where(u => !inputIds.Contains(u.Id)))
  471. {
  472. var rctQty = await _db.Ado.GetDecimalAsync(
  473. "SELECT IFNULL(RctQty,0) FROM PurOrdDetail WHERE RecID=@Id",
  474. new SugarParameter("@Id", toDelete.Id));
  475. if (rctQty > 0) throw Oops.Oh("明细存在收货数量,不允许删除");
  476. await _db.Ado.ExecuteCommandAsync(
  477. "DELETE FROM PurOrdDetail WHERE RecID=@Id",
  478. new SugarParameter("@Id", toDelete.Id));
  479. }
  480. }
  481. private async Task<DemandOrderListRow> GetMasterRowAsync(int id)
  482. {
  483. var row = (await _db.Ado.SqlQueryAsync<DemandOrderListRow>(
  484. """
  485. SELECT
  486. p.RecID AS Id,
  487. p.PurOrd AS PurOrd,
  488. CONCAT(TRIM(IFNULL(p.Supp,'')),' ',TRIM(IFNULL(s.SortName,''))) AS SuppName,
  489. p.Potype AS Potype,
  490. p.ReqBy AS ReqBy,
  491. CONCAT(TRIM(IFNULL(p.Buyer,'')),' ',TRIM(IFNULL(e.Name,''))) AS Buyer,
  492. CONCAT(TRIM(IFNULL(p.Department,'')),' ',TRIM(IFNULL(d.Descr,''))) AS DepartmentDescr,
  493. p.OrdDate AS OrdDate,
  494. p.Contract AS Contract,
  495. p.DeliverTo AS DeliverTo,
  496. p.DueDate AS DueDate,
  497. p.Contact AS Contact,
  498. p.ShipTo AS ShipTo,
  499. p.Curr AS Curr,
  500. p.TaxClass AS TaxClass,
  501. p.TaxIn AS TaxIn,
  502. CASE WHEN IFNULL(LENGTH(p.Status),0)=0 OR p.Buyer IS NULL THEN 'R' ELSE p.Status END AS Status,
  503. p.Remark AS Remark,
  504. p.Supp AS Supp,
  505. p.Buyer AS BuyerCode,
  506. p.Department AS Department,
  507. p.`Usage` AS `Usage`,
  508. p.FSTID AS Fstid
  509. FROM PurOrdMaster p
  510. LEFT JOIN SuppMaster s ON p.Supp=s.Supp
  511. LEFT JOIN DepartmentMaster d ON p.Department=d.Department
  512. LEFT JOIN EmployeeMaster e ON p.Buyer=e.Employee
  513. WHERE p.RecID=@Id AND p.Potype='PO' AND IFNULL(p.ReqBy,'')='DO'
  514. LIMIT 1
  515. """,
  516. new SugarParameter("@Id", id))).FirstOrDefault();
  517. return row ?? throw Oops.Oh("记录不存在");
  518. }
  519. private static DateTime? ParseDate(string? v) => DateTime.TryParse(v, out var d) ? d : null;
  520. private static string BuildOrderBy(string? sortField, string? sortOrder)
  521. {
  522. var dir = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase) ? "ASC" : "DESC";
  523. return sortField?.ToLowerInvariant() switch
  524. {
  525. "purord" => $"p.PurOrd {dir}",
  526. "suppname" => $"p.Supp {dir}",
  527. "buyer" => $"p.Buyer {dir}",
  528. "departmentdescr" => $"p.Department {dir}",
  529. "orddate" => $"p.OrdDate {dir}",
  530. "contract" => $"p.Contract {dir}",
  531. "duedate" => $"p.DueDate {dir}",
  532. "contact" => $"p.Contact {dir}",
  533. "shipto" => $"p.ShipTo {dir}",
  534. "curr" => $"p.Curr {dir}",
  535. "taxclass" => $"p.TaxClass {dir}",
  536. "taxin" => $"p.TaxIn {dir}",
  537. "status" => $"p.Status {dir}",
  538. _ => "p.RecID DESC"
  539. };
  540. }
  541. private static string BuildItemOrderBy(string? sortField, string? sortOrder)
  542. {
  543. var dir = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase) ? "ASC" : "DESC";
  544. return sortField?.ToLowerInvariant() switch
  545. {
  546. "itemnum" => $"ItemNum {dir}",
  547. "descr" => $"Descr {dir}",
  548. "descr1" => $"Descr1 {dir}",
  549. "um" => $"UM {dir}",
  550. "location" => $"Location {dir}",
  551. "rev" => $"Rev {dir}",
  552. "drawing" => $"Drawing {dir}",
  553. _ => "RecID DESC"
  554. };
  555. }
  556. private sealed class DemandOrderListRow
  557. {
  558. public int Id { get; set; }
  559. public string? PurOrd { get; set; }
  560. public string? SuppName { get; set; }
  561. public string? Potype { get; set; }
  562. public string? ReqBy { get; set; }
  563. public string? Buyer { get; set; }
  564. public string? DepartmentDescr { get; set; }
  565. public DateTime? OrdDate { get; set; }
  566. public string? Contract { get; set; }
  567. public string? DeliverTo { get; set; }
  568. public DateTime? DueDate { get; set; }
  569. public string? Contact { get; set; }
  570. public string? ShipTo { get; set; }
  571. public string? Curr { get; set; }
  572. public string? TaxClass { get; set; }
  573. public bool TaxIn { get; set; }
  574. public string? Status { get; set; }
  575. public string? Remark { get; set; }
  576. public string? Supp { get; set; }
  577. public string? BuyerCode { get; set; }
  578. public string? Department { get; set; }
  579. public string? Usage { get; set; }
  580. public string? Fstid { get; set; }
  581. }
  582. private sealed class DemandOrderDetailRow
  583. {
  584. public int Id { get; set; }
  585. public int? Line { get; set; }
  586. public string? ItemNum { get; set; }
  587. public string? UM { get; set; }
  588. public string? Location { get; set; }
  589. public decimal? QtyOrded { get; set; }
  590. public decimal? RctQty { get; set; }
  591. public decimal? ReceiptQty { get; set; }
  592. public DateTime? DueDate { get; set; }
  593. public string? Rev { get; set; }
  594. public string? Drawing { get; set; }
  595. public string? LotSerial { get; set; }
  596. public string? Potype { get; set; }
  597. public string? PurOrd { get; set; }
  598. public int? PurOrdRecID { get; set; }
  599. }
  600. private sealed class ItemLookupRow
  601. {
  602. public long RecID { get; set; }
  603. public string? ItemNum { get; set; }
  604. public string? Descr { get; set; }
  605. public string? Descr1 { get; set; }
  606. public string? UM { get; set; }
  607. public string? Location { get; set; }
  608. public string? Rev { get; set; }
  609. public string? Drawing { get; set; }
  610. }
  611. private sealed class OptionRow
  612. {
  613. public string? Value { get; set; }
  614. public string? Label { get; set; }
  615. }
  616. }