PurchaseReturnService.cs 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620
  1. using Admin.NET.Plugin.AiDOP.ProcurementExecution.Dto;
  2. using Admin.NET.Plugin.AiDOP.ProcurementExecution.Entity;
  3. using Admin.NET.Plugin.AiDOP.Supply;
  4. namespace Admin.NET.Plugin.AiDOP.ProcurementExecution;
  5. /// <summary>
  6. /// S4 采购退货单(PurOrdRctMaster / PurOrdRctDetail,RctType=pt)
  7. /// 路由:/api/ProcurementExecution/purchase-return/...
  8. /// </summary>
  9. [ApiDescriptionSettings(Order = 321, Description = "S4采购退货单")]
  10. [Route("api/ProcurementExecution")]
  11. [AllowAnonymous]
  12. [NonUnify]
  13. public class PurchaseReturnService : IDynamicApiController, ITransient
  14. {
  15. /// <summary>采购退货单号:PT + yyyyMMdd + 4 位流水,如 PT202605190001。</summary>
  16. private const int PtReceiverSerialWidth = 4;
  17. private readonly ISqlSugarClient _db;
  18. private readonly SqlSugarRepository<PurOrdRctMaster> _masterRep;
  19. private readonly SqlSugarRepository<PurOrdRctDetail> _detailRep;
  20. private readonly UserManager _userManager;
  21. public PurchaseReturnService(
  22. ISqlSugarClient db,
  23. SqlSugarRepository<PurOrdRctMaster> masterRep,
  24. SqlSugarRepository<PurOrdRctDetail> detailRep,
  25. UserManager userManager)
  26. {
  27. _db = db;
  28. _masterRep = masterRep;
  29. _detailRep = detailRep;
  30. _userManager = userManager;
  31. }
  32. private static string NormalizeOrderBy(string field, string order)
  33. {
  34. var col = (field ?? "").Trim();
  35. var ord = string.Equals(order?.Trim(), "desc", StringComparison.OrdinalIgnoreCase) ? "DESC" : "ASC";
  36. var map = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
  37. {
  38. // 注意:列表 SQL 外层包了 SELECT * FROM (...) t,因此 ORDER BY 只能引用外层可见列名
  39. ["receiver"] = "Receiver",
  40. ["rctdate"] = "RctDate",
  41. ["ordnbr"] = "OrdNbr",
  42. ["itemnum"] = "ItemNum",
  43. ["rctqty"] = "RctQty",
  44. ["suppname"] = "SuppName",
  45. };
  46. if (!map.TryGetValue(col, out var sqlCol))
  47. sqlCol = "RecID";
  48. return $"ORDER BY {sqlCol} {ord}, DetailRecId ASC";
  49. }
  50. [DisplayName("采购退货单分页列表")]
  51. [HttpGet("purchase-return/list")]
  52. public async Task<object> GetList([FromQuery] PurchaseReturnListInput input)
  53. {
  54. var page = input.Page <= 0 ? 1 : input.Page;
  55. var pageSize = input.PageSize <= 0 ? 10 : input.PageSize;
  56. var offset = (page - 1) * pageSize;
  57. var tenantId = _userManager.TenantId;
  58. var cond = new List<string> { "p.RctType='pt'", "p.IsActive=1", "p.tenant_id = @TenantId" };
  59. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  60. if (!string.IsNullOrWhiteSpace(input.Receiver))
  61. {
  62. cond.Add("p.Receiver LIKE @Receiver");
  63. pars.Add(new SugarParameter("@Receiver", $"%{input.Receiver.Trim()}%"));
  64. }
  65. if (!string.IsNullOrWhiteSpace(input.Supp))
  66. {
  67. cond.Add("p.Supp=@Supp");
  68. pars.Add(new SugarParameter("@Supp", input.Supp.Trim()));
  69. }
  70. if (!string.IsNullOrWhiteSpace(input.OrdNbr))
  71. {
  72. cond.Add("p.OrdNbr LIKE @OrdNbr");
  73. pars.Add(new SugarParameter("@OrdNbr", $"%{input.OrdNbr.Trim()}%"));
  74. }
  75. if (!string.IsNullOrWhiteSpace(input.ItemNum))
  76. {
  77. cond.Add("d1.ItemNum LIKE @ItemNum");
  78. pars.Add(new SugarParameter("@ItemNum", $"%{input.ItemNum.Trim()}%"));
  79. }
  80. var where = string.Join(" AND ", cond);
  81. var baseSql = $"""
  82. SELECT
  83. p.RecID AS RecID,
  84. IFNULL(d1.RecID, 0) AS DetailRecId,
  85. p.RctType AS RctType,
  86. p.Receiver AS Receiver,
  87. p.RctDate AS RctDate,
  88. p.OrdNbr AS OrdNbr,
  89. p.TaxClass AS TaxClass,
  90. IFNULL(p.TaxIn, 0) AS TaxIn,
  91. p.Curr AS Curr,
  92. p.Terms AS Terms,
  93. IFNULL(p.IsPlan, 0) AS IsPlan,
  94. TRIM(CONCAT(TRIM(IFNULL(p.Department,'')),' ',TRIM(IFNULL(d.Descr,'')))) AS DepartmentDescr,
  95. TRIM(CONCAT(TRIM(IFNULL(p.Supp,'')),' ',TRIM(IFNULL(s.SortName,'')))) AS SuppName,
  96. d1.ItemNum AS ItemNum,
  97. IFNULL(i.Descr,'') AS Descr,
  98. IFNULL(i.Descr1,'') AS Descr1,
  99. IFNULL(d1.RctQty, 0) AS RctQty
  100. FROM PurOrdRctMaster p
  101. LEFT JOIN PurOrdRctDetail d1 ON p.RecID = d1.PurOrdRctRecID AND d1.RctType='pt' AND d1.IsActive=1
  102. LEFT JOIN SuppMaster s ON p.Supp = s.Supp
  103. LEFT JOIN DepartmentMaster d ON p.Department = d.Department
  104. LEFT JOIN ItemMaster i ON d1.ItemNum = i.ItemNum
  105. WHERE {where}
  106. """;
  107. var orderBy = NormalizeOrderBy(input.SortField, input.SortOrder);
  108. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) FROM ({baseSql}) t", pars);
  109. var list = await _db.Ado.SqlQueryAsync<PurchaseReturnListRow>(
  110. $"SELECT * FROM ({baseSql}) t {orderBy} LIMIT {pageSize} OFFSET {offset}", pars);
  111. return new { total, page, pageSize, list };
  112. }
  113. [DisplayName("采购退货供应商下拉")]
  114. [HttpGet("purchase-return/options/suppliers")]
  115. public async Task<object> GetSuppliers()
  116. {
  117. var tenantId = _userManager.TenantId;
  118. var rows = await _db.Ado.SqlQueryAsync<PurchaseReturnKvItem>(
  119. """
  120. SELECT Supp AS `Value`,
  121. TRIM(CONCAT(TRIM(IFNULL(Supp,'')),' ',TRIM(IFNULL(SortName,'')))) AS `Label`
  122. FROM SuppMaster
  123. WHERE tenant_id = @TenantId
  124. ORDER BY Supp
  125. """,
  126. new SugarParameter("@TenantId", tenantId));
  127. return rows;
  128. }
  129. [DisplayName("收货单号下拉(rc)")]
  130. [HttpGet("purchase-return/options/rc-receivers")]
  131. public async Task<object> GetRcReceivers([FromQuery] string supp)
  132. {
  133. if (string.IsNullOrWhiteSpace(supp))
  134. return Array.Empty<object>();
  135. var tenantId = _userManager.TenantId;
  136. var rows = await _db.Ado.SqlQueryAsync<PurchaseReturnKvItem>(
  137. """
  138. SELECT p.Receiver AS `Value`,
  139. TRIM(CONCAT(TRIM(IFNULL(p.Receiver,'')),' ',
  140. TRIM(IFNULL(d.LotSerial,'')),' ',
  141. TRIM(IFNULL(d.ItemNum,'')),' ',
  142. TRIM(IFNULL(i.Descr,'')))) AS `Label`
  143. FROM PurOrdRctMaster p
  144. LEFT JOIN PurOrdRctDetail d
  145. ON p.RctType = d.RctType AND p.Receiver = d.Receiver
  146. LEFT JOIN ItemMaster i ON d.ItemNum = i.ItemNum
  147. WHERE p.RctType = 'rc' AND p.Supp = @Supp AND d.RecID > 0 AND IFNULL(p.TermsofTrade,'') <> ''
  148. AND p.tenant_id = @TenantId
  149. GROUP BY p.Receiver, d.LotSerial, d.ItemNum, i.Descr, p.Domain
  150. ORDER BY p.Receiver DESC
  151. """,
  152. new SugarParameter("@Supp", supp.Trim()),
  153. new SugarParameter("@TenantId", tenantId));
  154. return rows;
  155. }
  156. [DisplayName("按收货单加载明细模板")]
  157. [HttpGet("purchase-return/rc-detail-lines")]
  158. public async Task<object> GetRcDetailLines([FromQuery] string receiver)
  159. {
  160. if (string.IsNullOrWhiteSpace(receiver))
  161. throw Oops.Oh("请先选择收货单号");
  162. var tenantId = _userManager.TenantId;
  163. var rows = await _db.Ado.SqlQueryAsync<PurchaseReturnRcLineRow>(
  164. """
  165. SELECT
  166. p.Domain AS Domain,
  167. p.Department AS Department,
  168. p.Terms AS Terms,
  169. p.TermsofTrade AS TermsofTrade,
  170. d.ItemNum AS ItemNum,
  171. d.UM AS UM,
  172. d.OrdNbr AS OrdNbr,
  173. CAST(d.OrdLine AS SIGNED) AS OrdLine,
  174. d.QtyOrded AS QtyOrded,
  175. d.Supp AS Supp,
  176. d.LotSerial AS LotSerial,
  177. CASE WHEN UPPER(IFNULL(d.Potype,'')) = 'PW' THEN d.Supp ELSE d.Location END AS Location,
  178. d.QtyReceived AS QtyReceived,
  179. d.Potype AS Potype,
  180. d.Typed AS Typed,
  181. p.TaxClass AS TaxClass,
  182. IFNULL(p.TaxIn, 0) AS TaxIn,
  183. p.Curr AS Curr,
  184. DATE_FORMAT(CURDATE(), '%Y-%m-%d') AS RctDate,
  185. d.UMConversion AS UMConversion,
  186. d.POCost AS POCost,
  187. d.QtyReceived AS QtyReturn,
  188. d.Receiver AS RctNbr,
  189. CAST(d.Line AS SIGNED) AS RctLine,
  190. d.CustPO AS CustPO,
  191. d.BlanketLine AS BlanketLine,
  192. IFNULL(d.RctQty, 0) AS RctQty,
  193. CAST(d.Line AS SIGNED) AS Line
  194. FROM PurOrdRctMaster p
  195. LEFT JOIN PurOrdRctDetail d
  196. ON p.RctType = d.RctType AND p.Receiver = d.Receiver
  197. WHERE p.RctType = 'rc' AND p.Receiver = @Receiver AND d.RecID > 0
  198. AND p.tenant_id = @TenantId
  199. ORDER BY d.Line
  200. """,
  201. new SugarParameter("@Receiver", receiver.Trim()),
  202. new SugarParameter("@TenantId", tenantId));
  203. PurchaseReturnRcLineRow head = null;
  204. if (rows.Count > 0)
  205. head = rows[0];
  206. return new { head, lines = rows };
  207. }
  208. [DisplayName("部门下拉")]
  209. [HttpGet("purchase-return/options/departments")]
  210. public async Task<object> GetDepartments()
  211. {
  212. var tenantId = _userManager.TenantId;
  213. return await _db.Ado.SqlQueryAsync<PurchaseReturnKvItem>(
  214. """
  215. SELECT Department AS `Value`,
  216. TRIM(CONCAT(TRIM(IFNULL(Department,'')),' ',TRIM(IFNULL(Descr,'')))) AS `Label`
  217. FROM DepartmentMaster
  218. WHERE tenant_id = @TenantId
  219. ORDER BY Department
  220. """,
  221. new SugarParameter("@TenantId", tenantId));
  222. }
  223. [DisplayName("库位下拉")]
  224. [HttpGet("purchase-return/options/locations")]
  225. public async Task<object> GetLocations()
  226. {
  227. var tenantId = _userManager.TenantId;
  228. return await _db.Ado.SqlQueryAsync<PurchaseReturnKvItem>(
  229. """
  230. SELECT Location AS `Value`,
  231. TRIM(CONCAT(TRIM(IFNULL(Location,'')),' ',TRIM(IFNULL(Descr,'')))) AS `Label`
  232. FROM LocationMaster
  233. WHERE IFNULL(Typed,'') <> 'Supp' AND tenant_id = @TenantId
  234. ORDER BY Location
  235. """,
  236. new SugarParameter("@TenantId", tenantId));
  237. }
  238. [DisplayName("退货原因下拉")]
  239. [HttpGet("purchase-return/options/qc-reasons")]
  240. public async Task<object> GetQcReasons()
  241. {
  242. var tenantId = _userManager.TenantId;
  243. return await _db.Ado.SqlQueryAsync<PurchaseReturnKvItem>(
  244. """
  245. SELECT Val AS `Value`,
  246. CONCAT(IFNULL(Val,''),'_',IFNULL(Comments,'')) AS `Label`
  247. FROM GeneralizedCodeMaster
  248. WHERE FldName = 'SAPTransReason'
  249. AND IFNULL(Ufld1,'') <> ''
  250. AND LOCATE('122', IFNULL(Ufld1,'')) > 0
  251. AND tenant_id = @TenantId
  252. ORDER BY Val
  253. """,
  254. new SugarParameter("@TenantId", tenantId));
  255. }
  256. [DisplayName("物料选择分页")]
  257. [HttpGet("purchase-return/item-options")]
  258. public async Task<object> GetItems([FromQuery] KeywordPageInput input)
  259. {
  260. var page = input.Page <= 0 ? 1 : input.Page;
  261. var pageSize = input.PageSize <= 0 ? 20 : input.PageSize;
  262. var offset = (page - 1) * pageSize;
  263. var tenantId = _userManager.TenantId;
  264. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  265. var where = "1=1 AND tenant_id = @TenantId";
  266. if (!string.IsNullOrWhiteSpace(input.Keyword))
  267. {
  268. where += " AND (ItemNum LIKE @Kw OR Descr LIKE @Kw OR Descr1 LIKE @Kw)";
  269. pars.Add(new SugarParameter("@Kw", $"%{input.Keyword.Trim()}%"));
  270. }
  271. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) FROM ItemMaster WHERE {where}", pars);
  272. var list = await _db.Ado.SqlQueryAsync<PurchaseReturnKvItem>(
  273. $"""
  274. SELECT ItemNum AS `Value`,
  275. TRIM(CONCAT(IFNULL(Descr,''),' ',IFNULL(Descr1,''))) AS `Label`
  276. FROM ItemMaster
  277. WHERE {where}
  278. ORDER BY ItemNum
  279. LIMIT {pageSize} OFFSET {offset}
  280. """,
  281. pars);
  282. return new { total, page, pageSize, list };
  283. }
  284. [DisplayName("采购退货单详情")]
  285. [HttpGet("purchase-return/{id:int}")]
  286. public async Task<object> GetDetail(int id)
  287. {
  288. var tenantId = _userManager.TenantId;
  289. var master = await _masterRep.GetFirstAsync(m => m.RecID == id && m.RctType == "pt" && m.TenantId == tenantId)
  290. ?? throw Oops.Oh("单据不存在");
  291. var m = await _db.Ado.SqlQueryAsync<PurchaseReturnMasterDto>(
  292. """
  293. SELECT
  294. m.RecID AS RecID,
  295. m.Domain AS Domain,
  296. m.Receiver AS Receiver,
  297. DATE_FORMAT(m.RctDate,'%Y-%m-%d') AS RctDate,
  298. m.Supp AS Supp,
  299. m.OrdNbr AS OrdNbr,
  300. m.Department AS Department,
  301. IFNULL(m.Terms,'') AS Terms,
  302. IFNULL(m.Curr,'') AS Curr,
  303. IFNULL(m.TaxClass,'') AS TaxClass,
  304. IFNULL(m.TaxIn, 0) AS TaxIn,
  305. IFNULL(m.TermsofTrade,'') AS TermsofTrade,
  306. IFNULL(m.Remark,'') AS Remark,
  307. IFNULL(m.IsPlan, 0) AS IsPlan
  308. FROM PurOrdRctMaster m
  309. WHERE m.RecID = @Id AND m.tenant_id = @TenantId
  310. LIMIT 1
  311. """,
  312. new SugarParameter("@Id", id),
  313. new SugarParameter("@TenantId", tenantId));
  314. var head = m.FirstOrDefault() ?? throw Oops.Oh("单据不存在");
  315. var detailRows = await _db.Queryable<PurOrdRctDetail>()
  316. .Where(d => d.PurOrdRctRecID == id && d.RctType == "pt" && d.IsActive && d.TenantId == tenantId)
  317. .OrderBy(d => d.Line)
  318. .ToListAsync();
  319. var details = detailRows.Select(ToDetailDto).ToList();
  320. return new { master = head, details };
  321. }
  322. private static PurchaseReturnDetailDto ToDetailDto(PurOrdRctDetail d)
  323. {
  324. return new PurchaseReturnDetailDto
  325. {
  326. RecID = d.RecID,
  327. Line = d.Line,
  328. ItemNum = d.ItemNum,
  329. OrdNbr = d.OrdNbr,
  330. OrdLine = d.OrdLine,
  331. QtyOrded = d.QtyOrded,
  332. QtyReceived = d.QtyReceived,
  333. RctQty = d.RctQty,
  334. LotSerial = d.LotSerial,
  335. UM = d.UM,
  336. Location = d.Location,
  337. QCDescr = d.QCDescr,
  338. Remark = d.Remark,
  339. RctDate = d.RctDate?.ToString("yyyy-MM-dd"),
  340. Typed = d.Typed,
  341. TaxIn = d.TaxIn,
  342. TaxClass = d.TaxClass,
  343. Curr = d.Curr,
  344. Potype = d.Potype,
  345. IsActive = d.IsActive,
  346. IsConfirm = d.IsConfirm,
  347. UMConversion = d.UMConversion,
  348. POCost = d.POCost,
  349. RctNbr = d.RctNbr,
  350. RctLine = d.RctLine,
  351. CustPO = d.CustPO,
  352. BlanketLine = d.BlanketLine,
  353. QtyReturn = d.QtyReturn,
  354. Supp = d.Supp,
  355. };
  356. }
  357. [DisplayName("保存采购退货单")]
  358. [HttpPost("purchase-return/save")]
  359. public async Task<object> Save([FromBody] PurchaseReturnSaveInput input)
  360. {
  361. if (input == null) throw Oops.Oh("参数无效");
  362. if (string.IsNullOrWhiteSpace(input.Supp)) throw Oops.Oh("请选择供应商");
  363. if (string.IsNullOrWhiteSpace(input.OrdNbr)) throw Oops.Oh("请选择收货单号");
  364. if (string.IsNullOrWhiteSpace(input.Department)) throw Oops.Oh("请选择部门");
  365. if (input.Details == null || input.Details.Count == 0) throw Oops.Oh("请维护退货明细");
  366. var now = DateTime.Now;
  367. var account = _userManager.Account ?? "system";
  368. var tenantId = _userManager.TenantId;
  369. var domain = string.IsNullOrWhiteSpace(input.Domain)
  370. ? await ResolveRcDomainAsync(input.OrdNbr.Trim())
  371. : input.Domain.Trim();
  372. if (string.IsNullOrWhiteSpace(domain))
  373. domain = "100";
  374. if (input.Id is null or 0)
  375. {
  376. try
  377. {
  378. _db.Ado.BeginTran();
  379. var receiver = await AllocReceiverAsync();
  380. var rctDate = ParseDate(input.RctDate) ?? now.Date;
  381. var master = new PurOrdRctMaster
  382. {
  383. Domain = domain,
  384. RctType = "pt",
  385. Receiver = receiver,
  386. RctDate = rctDate,
  387. OrdNbr = input.OrdNbr.Trim(),
  388. Supp = input.Supp.Trim(),
  389. Department = input.Department.Trim(),
  390. Remark = input.Remark?.Trim(),
  391. Terms = input.Terms?.Trim(),
  392. Curr = input.Curr?.Trim(),
  393. TaxClass = input.TaxClass?.Trim(),
  394. TaxIn = input.TaxIn,
  395. TermsofTrade = input.TermsofTrade?.Trim(),
  396. Typed = "C",
  397. IsActive = true,
  398. IsConfirm = false,
  399. IsPlan = true,
  400. CreateUser = account,
  401. CreateTime = now,
  402. UpdateUser = account,
  403. UpdateTime = now,
  404. };
  405. var newId = await _db.Insertable(master).ExecuteReturnIdentityAsync();
  406. master.RecID = Convert.ToInt32(newId);
  407. await SaveDetailsAsync(master.RecID, domain, receiver, "pt", input.Details, account, now);
  408. _db.Ado.CommitTran();
  409. return new { id = master.RecID, receiver, message = "新增成功" };
  410. }
  411. catch
  412. {
  413. _db.Ado.RollbackTran();
  414. throw;
  415. }
  416. }
  417. var existing = await _masterRep.GetFirstAsync(m => m.RecID == input.Id!.Value && m.RctType == "pt" && m.TenantId == tenantId)
  418. ?? throw Oops.Oh("单据不存在");
  419. if (!existing.IsPlan)
  420. throw Oops.Oh("该单据不允许修改");
  421. existing.RctDate = ParseDate(input.RctDate) ?? existing.RctDate;
  422. existing.OrdNbr = input.OrdNbr.Trim();
  423. existing.Supp = input.Supp.Trim();
  424. existing.Department = input.Department.Trim();
  425. existing.Remark = input.Remark?.Trim();
  426. existing.Terms = input.Terms?.Trim();
  427. existing.Curr = input.Curr?.Trim();
  428. existing.TaxClass = input.TaxClass?.Trim();
  429. existing.TaxIn = input.TaxIn;
  430. existing.TermsofTrade = input.TermsofTrade?.Trim();
  431. existing.Domain = domain;
  432. existing.UpdateUser = account;
  433. existing.UpdateTime = now;
  434. await _masterRep.UpdateAsync(existing);
  435. await SaveDetailsAsync(existing.RecID, existing.Domain, existing.Receiver, "pt", input.Details, account, now);
  436. return new { id = existing.RecID, receiver = existing.Receiver, message = "保存成功" };
  437. }
  438. private async Task SaveDetailsAsync(
  439. int masterRecId,
  440. string domain,
  441. string receiver,
  442. string rctType,
  443. List<PurchaseReturnDetailInput> entries,
  444. string account,
  445. DateTime now)
  446. {
  447. var tenantId = _userManager.TenantId;
  448. var dbRows = await _detailRep.GetListAsync(d =>
  449. d.PurOrdRctRecID == masterRecId && d.RctType == rctType && d.TenantId == tenantId);
  450. var dbById = dbRows.ToDictionary(d => d.RecID);
  451. var inputIds = new HashSet<int>(entries.Where(e => e.RecID is > 0).Select(e => e.RecID!.Value));
  452. for (var i = 0; i < entries.Count; i++)
  453. {
  454. var e = entries[i];
  455. var lineNo = e.Line > 0 ? e.Line : (short)(i + 1);
  456. if (string.IsNullOrWhiteSpace(e.ItemNum))
  457. throw Oops.Oh($"第{i + 1}行:请填写物料");
  458. var loc = string.IsNullOrWhiteSpace(e.Location) ? "00" : e.Location.Trim();
  459. if (e.RecID is > 0 && dbById.TryGetValue(e.RecID.Value, out var row))
  460. {
  461. ApplyDetailInput(row, e, domain, receiver, masterRecId, account, now, lineNo, loc);
  462. row.UpdateUser = account;
  463. row.UpdateTime = now;
  464. await _detailRep.UpdateAsync(row);
  465. }
  466. else
  467. {
  468. var detailNew = new PurOrdRctDetail();
  469. ApplyDetailInput(detailNew, e, domain, receiver, masterRecId, account, now, lineNo, loc);
  470. detailNew.CreateUser = account;
  471. detailNew.CreateTime = now;
  472. detailNew.UpdateUser = account;
  473. detailNew.UpdateTime = now;
  474. await _detailRep.InsertAsync(detailNew);
  475. }
  476. }
  477. foreach (var del in dbRows.Where(d => !inputIds.Contains(d.RecID)))
  478. await _detailRep.DeleteAsync(d => d.RecID == del.RecID);
  479. }
  480. private static void ApplyDetailInput(
  481. PurOrdRctDetail row,
  482. PurchaseReturnDetailInput e,
  483. string domain,
  484. string receiver,
  485. int masterRecId,
  486. string account,
  487. DateTime now,
  488. short lineNo,
  489. string location)
  490. {
  491. row.Domain = domain;
  492. row.RctType = "pt";
  493. row.Receiver = receiver;
  494. row.PurOrdRctRecID = masterRecId;
  495. row.Line = lineNo;
  496. row.Dimension1 = "";
  497. row.Dimension2 = "";
  498. row.ItemNum = e.ItemNum?.Trim();
  499. row.OrdNbr = string.IsNullOrWhiteSpace(e.OrdNbr) ? "" : e.OrdNbr.Trim();
  500. row.OrdLine = (short)e.OrdLine;
  501. row.RctDate = ParseDate(e.RctDate);
  502. row.UM = e.UM;
  503. row.QtyOrded = e.QtyOrded;
  504. row.QtyReceived = e.QtyReceived;
  505. row.RctQty = e.RctQty;
  506. row.LotSerial = e.LotSerial;
  507. row.Location = location;
  508. row.QCDescr = e.QCDescr;
  509. row.Remark = e.Remark;
  510. row.Typed = string.IsNullOrWhiteSpace(e.Typed) ? "C" : e.Typed.Trim();
  511. row.TaxIn = e.TaxIn;
  512. row.TaxClass = e.TaxClass;
  513. row.Curr = e.Curr;
  514. row.Potype = string.IsNullOrWhiteSpace(e.Potype) ? "P" : e.Potype.Trim();
  515. row.IsActive = true;
  516. row.IsConfirm = e.IsConfirm;
  517. row.UMConversion = e.UMConversion;
  518. row.POCost = e.POCost;
  519. row.RctNbr = e.RctNbr;
  520. row.RctLine = (short)e.RctLine;
  521. row.CustPO = e.CustPO;
  522. row.BlanketLine = e.BlanketLine;
  523. row.QtyReturn = e.QtyReturn;
  524. row.Supp = e.Supp;
  525. }
  526. [DisplayName("删除采购退货单(软删除)")]
  527. [HttpPost("purchase-return/delete/{id:int}")]
  528. public async Task<object> SoftDelete(int id)
  529. {
  530. var tenantId = _userManager.TenantId;
  531. var m = await _masterRep.GetFirstAsync(x => x.RecID == id && x.RctType == "pt" && x.TenantId == tenantId)
  532. ?? throw Oops.Oh("单据不存在");
  533. if (!m.IsPlan)
  534. throw Oops.Oh("该单据不允许删除");
  535. m.IsActive = false;
  536. m.UpdateUser = _userManager.Account ?? "system";
  537. m.UpdateTime = DateTime.Now;
  538. await _masterRep.UpdateAsync(m);
  539. await _db.Ado.ExecuteCommandAsync(
  540. "UPDATE PurOrdRctDetail SET IsActive=0, UpdateUser=@U, UpdateTime=@T WHERE PurOrdRctRecID=@Id AND RctType='pt' AND tenant_id=@TenantId",
  541. new SugarParameter("@U", m.UpdateUser),
  542. new SugarParameter("@T", m.UpdateTime),
  543. new SugarParameter("@Id", id),
  544. new SugarParameter("@TenantId", tenantId));
  545. return new { message = "已删除" };
  546. }
  547. private async Task<string> ResolveRcDomainAsync(string rcReceiver)
  548. {
  549. var tenantId = _userManager.TenantId;
  550. var d = await _db.Ado.GetStringAsync(
  551. "SELECT Domain FROM PurOrdRctMaster WHERE RctType='rc' AND Receiver=@R AND tenant_id=@TenantId LIMIT 1",
  552. new SugarParameter("@R", rcReceiver),
  553. new SugarParameter("@TenantId", tenantId));
  554. return d;
  555. }
  556. private Task<string> AllocReceiverAsync() =>
  557. PurOrdNumberGenerator.NextRctReceiverAsync(_db, "PT", "pt", DateTime.Today, PtReceiverSerialWidth);
  558. private static DateTime? ParseDate(string raw)
  559. {
  560. if (string.IsNullOrWhiteSpace(raw)) return null;
  561. return DateTime.TryParse(raw.Trim(), out var d) ? d.Date : null;
  562. }
  563. }