DemandOrderService.cs 28 KB

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