DemandOrderService.cs 28 KB

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