AsnShipperService.cs 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556
  1. namespace Admin.NET.Plugin.AiDOP.Order;
  2. using Admin.NET.Plugin.AiDOP.Entity.S0.Warehouse;
  3. using Admin.NET.Plugin.AiDOP.Universal;
  4. /// <summary>
  5. /// 订单发货服务 🚚
  6. /// 路由前缀:/api/Order/asnshipper/...
  7. /// </summary>
  8. [ApiDescriptionSettings(Order = 280, Description = "订单发货")]
  9. [Route("api/Order")]
  10. [AllowAnonymous]
  11. [NonUnify]
  12. public class AsnShipperService : IDynamicApiController, ITransient
  13. {
  14. private readonly ISqlSugarClient _db;
  15. private readonly SqlSugarRepository<AsnShipperMaster> _masterRep;
  16. private readonly SqlSugarRepository<AsnShipperDetail> _detailRep;
  17. private readonly UserManager _userManager;
  18. public AsnShipperService(
  19. ISqlSugarClient db,
  20. SqlSugarRepository<AsnShipperMaster> masterRep,
  21. SqlSugarRepository<AsnShipperDetail> detailRep,
  22. UserManager userManager)
  23. {
  24. _db = db;
  25. _masterRep = masterRep;
  26. _detailRep = detailRep;
  27. _userManager = userManager;
  28. }
  29. // ══════════════════════════════════════════════════════════════
  30. // 列表 GET /api/Order/asnshipper/list
  31. // ══════════════════════════════════════════════════════════════
  32. /// <summary>获取订单发货分页列表 🚚</summary>
  33. [DisplayName("获取订单发货列表")]
  34. [HttpGet("asnshipper/list")]
  35. public async Task<object> GetAsnShipperList([FromQuery] AsnShipperListInput input)
  36. {
  37. var tenantId = _userManager.TenantId;
  38. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  39. var conditions = new List<string>
  40. {
  41. "a.shtype = 'SH'",
  42. "a.Typed <> 'S'",
  43. "a.IsActive = 1",
  44. "a.tenant_id = @TenantId"
  45. };
  46. if (!string.IsNullOrWhiteSpace(input.Id))
  47. {
  48. conditions.Add("a.Id LIKE @Id");
  49. pars.Add(new SugarParameter("@Id", $"%{input.Id.Trim()}%"));
  50. }
  51. if (!string.IsNullOrWhiteSpace(input.OrdNbr))
  52. {
  53. conditions.Add("a.OrdNbr LIKE @OrdNbr");
  54. pars.Add(new SugarParameter("@OrdNbr", $"%{input.OrdNbr.Trim()}%"));
  55. }
  56. if (!string.IsNullOrWhiteSpace(input.Department))
  57. {
  58. conditions.Add("(b.Department LIKE @Dept OR b.Descr LIKE @Dept)");
  59. pars.Add(new SugarParameter("@Dept", $"%{input.Department.Trim()}%"));
  60. }
  61. if (!string.IsNullOrWhiteSpace(input.ShipDateFrom))
  62. {
  63. conditions.Add("a.ShipDate >= @ShipDateFrom");
  64. pars.Add(new SugarParameter("@ShipDateFrom", input.ShipDateFrom.Trim()));
  65. }
  66. var whereClause = "WHERE " + string.Join(" AND ", conditions);
  67. var baseSql = $"""
  68. SELECT
  69. a.RecID AS Id,
  70. a.Id AS Id1,
  71. a.OrdNbr,
  72. TRIM(CONCAT(IFNULL(b.Department,''), ' ', IFNULL(b.Descr,''))) AS DepartmentName,
  73. a.ShipDate,
  74. a.Status,
  75. a.Remark,
  76. CASE WHEN EXISTS (
  77. SELECT 1 FROM ASNBOLShipperDetail d
  78. WHERE d.ASNBOLShipperRecID = a.RecID
  79. AND d.tenant_id = a.tenant_id
  80. AND IFNULL(d.RealQty, 0) > 0
  81. ) THEN 1 ELSE 0 END AS HasShipped
  82. FROM ASNBOLShipperMaster a
  83. LEFT JOIN DepartmentMaster b
  84. ON a.Department = b.Department
  85. {whereClause}
  86. """;
  87. var offset = (input.Page - 1) * input.PageSize;
  88. var total = await _db.Ado.GetIntAsync(
  89. $"SELECT COUNT(*) FROM ({baseSql}) AS t", pars);
  90. var list = await _db.Ado.SqlQueryAsync<AsnShipperListRow>(
  91. $"SELECT * FROM ({baseSql}) AS t ORDER BY t.Id DESC LIMIT {input.PageSize} OFFSET {offset}",
  92. pars);
  93. return new { total, page = input.Page, pageSize = input.PageSize, list };
  94. }
  95. // ══════════════════════════════════════════════════════════════
  96. // 详情 GET /api/Order/asnshipper/{id}
  97. // ══════════════════════════════════════════════════════════════
  98. /// <summary>获取订单发货详情(含明细)🚚</summary>
  99. [DisplayName("获取订单发货详情")]
  100. [HttpGet("asnshipper/{id}")]
  101. public async Task<object> GetAsnShipperDetail(int id)
  102. {
  103. var tenantId = _userManager.TenantId;
  104. var master = await _masterRep.GetFirstAsync(u => u.RecID == id && u.TenantId == tenantId)
  105. ?? throw Oops.Oh("发货单不存在");
  106. // 查询客户名称,用于前端下拉回显"编号+名称"
  107. var soldToName = (await _db.Queryable<CustMaster>()
  108. .Where(u => u.Cust == master.SoldTo && u.TenantId == tenantId)
  109. .Select(u => u.SortName)
  110. .FirstAsync()) ?? string.Empty;
  111. // 查询销售单客户名称,用于前端下拉回显"编号_客户名称"
  112. var ordNbrName = string.IsNullOrWhiteSpace(master.OrdNbr) ? string.Empty
  113. : ((await _db.Queryable<SeOrder>()
  114. .Where(u => u.BillNo == master.OrdNbr && u.TenantId == tenantId)
  115. .Select(u => u.CustomName)
  116. .FirstAsync()) ?? string.Empty);
  117. // 查询部门描述,用于前端下拉回显"编号+描述"
  118. var deptName = string.IsNullOrWhiteSpace(master.Department) ? string.Empty
  119. : ((await _db.Queryable<AdoS0DepartmentMaster>()
  120. .Where(u => u.Department == master.Department && u.TenantId == tenantId)
  121. .Select(u => u.Descr)
  122. .FirstAsync()) ?? string.Empty);
  123. // 主键关联:ASNBOLShipperDetail.ASNBOLShipperRecID = 主表 RecID
  124. var details = await _detailRep.GetListAsync(u => u.AsnShipperRecID == id && u.TenantId == tenantId);
  125. // 兼容历史数据:若外键未写入或曾用错误方式取 RecID,则按业务单号 Id + 销售单号 OrdNbr 兜底
  126. if (details.Count == 0 && !string.IsNullOrWhiteSpace(master.Id) && !string.IsNullOrWhiteSpace(master.OrdNbr))
  127. {
  128. details = await _detailRep.GetListAsync(u =>
  129. u.Id == master.Id && u.OrdNbr == master.OrdNbr && u.TenantId == tenantId);
  130. }
  131. return new
  132. {
  133. recID = master.RecID,
  134. id = master.Id,
  135. ordNbr = master.OrdNbr,
  136. ordNbrName = ordNbrName,
  137. soldTo = master.SoldTo,
  138. soldToName = soldToName,
  139. department = master.Department,
  140. departmentName = deptName,
  141. shipDate = master.ShipDate?.ToString("yyyy-MM-dd"),
  142. status = master.Status ?? string.Empty,
  143. remark = master.Remark,
  144. details = details.OrderBy(d => d.Line).Select(d => new
  145. {
  146. recID = d.RecID,
  147. line = d.Line,
  148. ordNbr = d.OrdNbr,
  149. ordLine = d.OrdLine,
  150. containerItem = d.ContainerItem,
  151. descr = d.Descr,
  152. um = d.UM,
  153. location = d.Location,
  154. lotSerial = d.LotSerial,
  155. qtyToShip = d.QtyToShip,
  156. pickingQty = d.PickingQty,
  157. realQty = d.RealQty,
  158. status = d.Status,
  159. remark = d.Remark,
  160. })
  161. };
  162. }
  163. // ══════════════════════════════════════════════════════════════
  164. // 保存(新增/编辑)POST /api/Order/asnshipper/save
  165. // ══════════════════════════════════════════════════════════════
  166. /// <summary>保存订单发货(新增或编辑)🚚</summary>
  167. [DisplayName("保存订单发货")]
  168. [ApiDescriptionSettings(Name = "SaveAsnShipper"), HttpPost("asnshipper/save")]
  169. public async Task<object> SaveAsnShipper([FromBody] AsnShipperSaveInput input)
  170. {
  171. var now = DateTime.Now;
  172. var user = _userManager.Account ?? "system";
  173. // 验证明细行至少有一行数据
  174. if (input.Details == null || input.Details.Count == 0)
  175. {
  176. throw Oops.Oh("发货单明细行不能为空,请至少添加一行明细");
  177. }
  178. if (input.RecID is null or 0)
  179. {
  180. // ── 新增:参照 SysJobService.AddJobDetail ──
  181. // 若发货单号为空则自动生成 SH+当前日期+四位流水号
  182. var shipId = input.Id;
  183. if (string.IsNullOrWhiteSpace(shipId))
  184. {
  185. shipId = await GenerateShipIdAsync(_userManager.TenantId);
  186. }
  187. var entity = new AsnShipperMaster
  188. {
  189. Id = shipId,
  190. OrdNbr = input.OrdNbr,
  191. SoldTo = input.SoldTo,
  192. Department = input.Department,
  193. ShipDate = string.IsNullOrWhiteSpace(input.ShipDate) ? null : DateTime.Parse(input.ShipDate),
  194. Status = input.Status,
  195. Remark = input.Remark,
  196. ShType = "SH",
  197. Typed = string.Empty,
  198. IsActive = 1,
  199. IsConfirm = 0,
  200. CreateUser = user,
  201. CreateTime = now,
  202. };
  203. var inserted = await _masterRep.AsInsertable(entity).ExecuteReturnEntityAsync();
  204. var newId = inserted.RecID;
  205. await SaveDetailsAsync(newId, input.Id ?? string.Empty, input.Details, user, now, isNew: true);
  206. return new { id = newId, message = "新增成功" };
  207. }
  208. else
  209. {
  210. // ── 编辑:参照 SysJobService.UpdateJobDetail ──
  211. var tenantId = _userManager.TenantId;
  212. var entity = await _masterRep.GetFirstAsync(u => u.RecID == input.RecID.Value && u.TenantId == tenantId)
  213. ?? throw Oops.Oh("发货单不存在");
  214. entity.Id = input.Id;
  215. entity.OrdNbr = input.OrdNbr;
  216. entity.SoldTo = input.SoldTo;
  217. entity.Department = input.Department;
  218. entity.ShipDate = string.IsNullOrWhiteSpace(input.ShipDate) ? null : DateTime.Parse(input.ShipDate);
  219. entity.Status = input.Status;
  220. entity.Remark = input.Remark;
  221. entity.UpdateUser = user;
  222. entity.UpdateTime = now;
  223. await _masterRep.UpdateAsync(entity);
  224. await SaveDetailsAsync(input.RecID.Value, input.Id ?? string.Empty, input.Details, user, now, isNew: false);
  225. return new { id = input.RecID, message = "编辑成功" };
  226. }
  227. }
  228. /// <summary>
  229. /// 明细三路合并(参照 SeOrderService.SaveEntriesAsync):
  230. /// ① DB有且入参有 → 更新
  231. /// ② DB无入参有 → 新增
  232. /// ③ DB有入参无 → 删除
  233. /// </summary>
  234. private async Task SaveDetailsAsync(int masterRecId, string shipperId,
  235. List<AsnShipperDetailInput> details, string user, DateTime now, bool isNew)
  236. {
  237. var dbDetails = isNew ? new List<AsnShipperDetail>()
  238. : await _detailRep.GetListAsync(u => u.AsnShipperRecID == masterRecId && u.TenantId == _userManager.TenantId);
  239. var dbById = dbDetails.ToDictionary(u => u.RecID);
  240. var inputIds = new HashSet<int>(details.Where(d => d.RecID is > 0).Select(d => d.RecID!.Value));
  241. for (var i = 0; i < details.Count; i++)
  242. {
  243. var d = details[i];
  244. var lineNo = d.Line ?? (i + 1);
  245. if (d.RecID > 0 && dbById.TryGetValue(d.RecID!.Value, out var existing))
  246. {
  247. // ① 更新
  248. existing.Line = lineNo;
  249. existing.OrdNbr = d.OrdNbr;
  250. existing.OrdLine = d.OrdLine;
  251. existing.ContainerItem = d.ContainerItem;
  252. existing.Descr = d.Descr;
  253. existing.UM = d.UM;
  254. existing.Location = d.Location;
  255. existing.LotSerial = d.LotSerial;
  256. existing.QtyToShip = d.QtyToShip;
  257. existing.PickingQty = d.PickingQty;
  258. existing.RealQty = d.RealQty;
  259. existing.Status = d.Status;
  260. existing.Remark = d.Remark;
  261. existing.UpdateUser = user;
  262. existing.UpdateTime = now;
  263. await _detailRep.UpdateAsync(existing);
  264. }
  265. else
  266. {
  267. // ② 新增
  268. var detail = new AsnShipperDetail
  269. {
  270. Id = shipperId,
  271. AsnShipperRecID = masterRecId,
  272. Line = lineNo,
  273. OrdNbr = d.OrdNbr,
  274. OrdLine = d.OrdLine,
  275. ContainerItem = d.ContainerItem,
  276. Descr = d.Descr,
  277. UM = d.UM,
  278. Location = d.Location,
  279. LotSerial = d.LotSerial,
  280. QtyToShip = d.QtyToShip,
  281. PickingQty = d.PickingQty,
  282. RealQty = d.RealQty,
  283. Status = d.Status,
  284. Remark = d.Remark,
  285. ShType = "SH",
  286. IsActive = 1,
  287. CreateUser = user,
  288. CreateTime = now,
  289. };
  290. await _detailRep.InsertAsync(detail);
  291. }
  292. }
  293. // ③ 删除:DB有、入参无
  294. foreach (var toDelete in dbDetails.Where(u => !inputIds.Contains(u.RecID)))
  295. {
  296. await _detailRep.DeleteAsync(u => u.RecID == toDelete.RecID);
  297. }
  298. }
  299. // ══════════════════════════════════════════════════════════════
  300. // 删除 POST /api/Order/asnshipper/delete
  301. // ══════════════════════════════════════════════════════════════
  302. /// <summary>删除发货单(状态非关闭才可删除)🚚</summary>
  303. [DisplayName("删除订单发货")]
  304. [ApiDescriptionSettings(Name = "DeleteAsnShipper"), HttpPost("asnshipper/delete")]
  305. public async Task<object> DeleteAsnShipper([FromBody] AsnShipperDeleteInput input)
  306. {
  307. // 参照 SysJobService.DeleteJobDetail:先查再删
  308. var tenantId = _userManager.TenantId;
  309. var entity = await _masterRep.GetFirstAsync(u => u.RecID == input.RecID && u.TenantId == tenantId)
  310. ?? throw Oops.Oh("发货单不存在");
  311. if (entity.Status == "C")
  312. throw Oops.Oh("已关闭的发货单不允许删除");
  313. await _detailRep.DeleteAsync(u => u.AsnShipperRecID == input.RecID);
  314. await _masterRep.DeleteAsync(u => u.RecID == input.RecID);
  315. return new { message = "删除成功" };
  316. }
  317. // ══════════════════════════════════════════════════════════════
  318. // 销售单明细行 GET /api/Order/asnshipper/salesord/{ordNbr}/entries
  319. // ══════════════════════════════════════════════════════════════
  320. /// <summary>获取销售单明细行(用于发货单新建时自动填入)🚚</summary>
  321. [DisplayName("获取销售单明细行")]
  322. [HttpGet("asnshipper/salesord/{ordNbr}/entries")]
  323. public async Task<object> GetSalesOrdEntries(string ordNbr)
  324. {
  325. var tenantId = _userManager.TenantId;
  326. // 查询销售单主表,获取客户信息
  327. var order = await _db.Queryable<SeOrder>()
  328. .Where(u => u.BillNo == ordNbr && u.TenantId == tenantId && u.IsDeleted == 0)
  329. .Select(u => new { u.CustomNo, u.CustomName })
  330. .FirstAsync();
  331. // 查询销售单明细行(物料信息)
  332. var entries = await _db.Queryable<SeOrderEntry>()
  333. .Where(u => u.BillNo == ordNbr && u.TenantId == tenantId && u.IsDeleted == 0)
  334. .OrderBy(u => u.EntrySeq)
  335. .Select(u => new
  336. {
  337. entrySeq = u.EntrySeq,
  338. itemNumber = u.ItemNumber,
  339. itemName = u.ItemName,
  340. specification = u.Specification,
  341. unit = u.Unit,
  342. qty = u.Qty,
  343. })
  344. .ToListAsync();
  345. return new
  346. {
  347. ordNbr,
  348. customNo = order?.CustomNo,
  349. customName = order?.CustomName,
  350. entries,
  351. };
  352. }
  353. // ══════════════════════════════════════════════════════════════
  354. // 下拉数据源 GET /api/Order/asnshipper/options/salesords
  355. // ══════════════════════════════════════════════════════════════
  356. /// <summary>获取销售单下拉列表 🚚</summary>
  357. [DisplayName("获取销售单下拉")]
  358. [HttpGet("asnshipper/options/salesords")]
  359. public async Task<object> GetSalesOrdOptions([FromQuery] string? keyword)
  360. {
  361. var tenantId = _userManager.TenantId;
  362. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  363. var where = "s.IsDeleted = 0 AND s.tenant_id = @TenantId";
  364. if (!string.IsNullOrWhiteSpace(keyword))
  365. {
  366. where += " AND (s.bill_no LIKE @kw OR s.custom_name LIKE @kw)";
  367. pars.Add(new SugarParameter("@kw", $"%{keyword.Trim()}%"));
  368. }
  369. // MySQL:字符串拼接用 CONCAT(对应 SQL Server 的 bill_no+'_'+custom_name)
  370. var sql = $"""
  371. SELECT s.bill_no AS Value,
  372. CONCAT(s.bill_no, '_', IFNULL(s.custom_name, '')) AS Label
  373. FROM crm_seorder s
  374. WHERE {where}
  375. ORDER BY s.bill_no DESC
  376. LIMIT 100
  377. """;
  378. var list = await _db.Ado.SqlQueryAsync<SimpleKvRow>(sql, pars);
  379. return list;
  380. }
  381. // ══════════════════════════════════════════════════════════════
  382. // 下拉数据源 GET /api/Order/asnshipper/options/customers
  383. // ══════════════════════════════════════════════════════════════
  384. /// <summary>获取客户下拉列表 🚚</summary>
  385. [DisplayName("获取客户下拉")]
  386. [HttpGet("asnshipper/options/customers")]
  387. public async Task<object> GetCustomerOptions([FromQuery] string? keyword)
  388. {
  389. var tenantId = _userManager.TenantId;
  390. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  391. var where = "tenant_id = @TenantId";
  392. if (!string.IsNullOrWhiteSpace(keyword))
  393. {
  394. where += " AND (Cust LIKE @kw OR SortName LIKE @kw)";
  395. pars.Add(new SugarParameter("@kw", $"%{keyword.Trim()}%"));
  396. }
  397. var sql = $"""
  398. SELECT Cust AS Value,
  399. TRIM(CONCAT(Cust, ' ', IFNULL(SortName,''))) AS Label
  400. FROM CustMaster
  401. WHERE {where}
  402. ORDER BY Cust
  403. LIMIT 100
  404. """;
  405. var list = await _db.Ado.SqlQueryAsync<SimpleKvRow>(sql, pars);
  406. return list;
  407. }
  408. // ══════════════════════════════════════════════════════════════
  409. // 下拉数据源 GET /api/Order/asnshipper/options/departments
  410. // ══════════════════════════════════════════════════════════════
  411. /// <summary>获取部门下拉列表 🚚</summary>
  412. [DisplayName("获取部门下拉")]
  413. [HttpGet("asnshipper/options/departments")]
  414. public async Task<object> GetDepartmentOptions([FromQuery] string? keyword)
  415. {
  416. var tenantId = _userManager.TenantId;
  417. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  418. var where = "tenant_id = @TenantId";
  419. if (!string.IsNullOrWhiteSpace(keyword))
  420. {
  421. where += " AND (Department LIKE @kw OR Descr LIKE @kw)";
  422. pars.Add(new SugarParameter("@kw", $"%{keyword.Trim()}%"));
  423. }
  424. var sql = $"""
  425. SELECT Department AS Value,
  426. TRIM(CONCAT(Department, ' ', IFNULL(Descr,''))) AS Label
  427. FROM DepartmentMaster
  428. WHERE {where}
  429. ORDER BY Department
  430. LIMIT 100
  431. """;
  432. var list = await _db.Ado.SqlQueryAsync<SimpleKvRow>(sql, pars);
  433. return list;
  434. }
  435. // ══════════════════════════════════════════════════════════════
  436. // 下拉数据源 GET /api/Order/asnshipper/options/locations
  437. // ══════════════════════════════════════════════════════════════
  438. /// <summary>获取库位下拉列表 🚚</summary>
  439. [DisplayName("获取库位下拉")]
  440. [HttpGet("asnshipper/options/locations")]
  441. public async Task<object> GetLocationOptions([FromQuery] string? keyword)
  442. {
  443. var tenantId = _userManager.TenantId;
  444. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  445. var where = "tenant_id = @TenantId AND is_active = 1";
  446. if (!string.IsNullOrWhiteSpace(keyword))
  447. {
  448. where += " AND (location LIKE @kw OR descr LIKE @kw)";
  449. pars.Add(new SugarParameter("@kw", $"%{keyword.Trim()}%"));
  450. }
  451. var sql = $"""
  452. SELECT location AS Value,
  453. TRIM(CONCAT(location, ' ', IFNULL(descr, ''))) AS Label
  454. FROM LocationMaster
  455. WHERE {where}
  456. ORDER BY location
  457. LIMIT 100
  458. """;
  459. var list = await _db.Ado.SqlQueryAsync<SimpleKvRow>(sql, pars);
  460. return list;
  461. }
  462. // ──────────────── 号生成 ────────────────
  463. /// <summary>
  464. /// 生成发货单号:SH + yyyyMMdd + 4位流水号,例如 SH202605210001
  465. /// </summary>
  466. private async Task<string> GenerateShipIdAsync(long tenantId)
  467. {
  468. var today = DateTime.Now.ToString("yyyyMMdd");
  469. var prefix = $"SH{today}";
  470. // 查询当天已有的最大发货单号
  471. var maxId = await _db.Queryable<AsnShipperMaster>()
  472. .Where(u => u.Id!.StartsWith(prefix) && u.IsActive == 1 && u.TenantId == tenantId)
  473. .OrderBy(u => u.Id, OrderByType.Desc)
  474. .Select(u => u.Id)
  475. .FirstAsync();
  476. if (string.IsNullOrEmpty(maxId))
  477. return $"{prefix}0001";
  478. // 取后4位转为数字并加1,不足4位补零
  479. var last4 = maxId.Length >= 4 ? maxId[^4..] : maxId;
  480. if (int.TryParse(last4, out var serial))
  481. return $"{prefix}{(serial + 1).ToString().PadLeft(4, '0')}";
  482. return $"{prefix}0001";
  483. }
  484. // ──────────────── 内部查询结果映射类 ────────────────
  485. private sealed class AsnShipperListRow
  486. {
  487. public int Id { get; set; }
  488. public string? Id1 { get; set; }
  489. public string? OrdNbr { get; set; }
  490. public string? DepartmentName { get; set; }
  491. public DateTime? ShipDate { get; set; }
  492. public string? Status { get; set; }
  493. public string? Remark { get; set; }
  494. public bool HasShipped { get; set; }
  495. }
  496. private sealed class SimpleKvRow
  497. {
  498. public string? Value { get; set; }
  499. public string? Label { get; set; }
  500. }
  501. }