AsnShipperService.cs 24 KB

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