SeOrderService.cs 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. using Yitter.IdGenerator;
  2. namespace Admin.NET.Plugin.AiDOP.Order;
  3. /// <summary>
  4. /// 销售订单评审服务 🗂️
  5. /// 路由前缀:/api/Order/seorder/...
  6. /// </summary>
  7. [ApiDescriptionSettings(Order = 300, Description = "销售订单评审")]
  8. [Route("api/Order")]
  9. [AllowAnonymous]
  10. [NonUnify]
  11. public class SeOrderService : IDynamicApiController, ITransient
  12. {
  13. private readonly ISqlSugarClient _db;
  14. private readonly SqlSugarRepository<SeOrder> _seOrderRep;
  15. private readonly SqlSugarRepository<SeOrderEntry> _seOrderEntryRep;
  16. private readonly SqlSugarRepository<SeOrderChange> _seOrderChangeRep;
  17. private readonly UserManager _userManager;
  18. private readonly SeOrderOperationLogService _orderOpLog;
  19. public SeOrderService(
  20. ISqlSugarClient db,
  21. SqlSugarRepository<SeOrder> seOrderRep,
  22. SqlSugarRepository<SeOrderEntry> seOrderEntryRep,
  23. SqlSugarRepository<SeOrderChange> seOrderChangeRep,
  24. UserManager userManager,
  25. SeOrderOperationLogService orderOpLog)
  26. {
  27. _db = db;
  28. _seOrderRep = seOrderRep;
  29. _seOrderEntryRep = seOrderEntryRep;
  30. _seOrderChangeRep = seOrderChangeRep;
  31. _userManager = userManager;
  32. _orderOpLog = orderOpLog;
  33. }
  34. // ══════════════════════════════════════════════════════════════
  35. // 订单列表 GET /api/Order/seorder/list
  36. // ══════════════════════════════════════════════════════════════
  37. /// <summary>获取订单评审分页列表 🗂️</summary>
  38. [DisplayName("获取订单评审列表")]
  39. [HttpGet("seorder/list")]
  40. public async Task<object> GetSeOrderList([FromQuery] SeOrderListInput input)
  41. {
  42. var tenantId = _userManager.TenantId;
  43. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  44. var innerConditions = new List<string> { "a.IsDeleted = 0", "a.tenant_id = @TenantId" };
  45. if (!string.IsNullOrWhiteSpace(input.BillNo))
  46. {
  47. innerConditions.Add("a.bill_no LIKE @BillNo");
  48. pars.Add(new SugarParameter("@BillNo", $"%{input.BillNo.Trim()}%"));
  49. }
  50. if (!string.IsNullOrWhiteSpace(input.CustomNo))
  51. {
  52. innerConditions.Add("a.custom_no LIKE @CustomNo");
  53. pars.Add(new SugarParameter("@CustomNo", $"%{input.CustomNo.Trim()}%"));
  54. }
  55. if (input.OrderType.HasValue)
  56. {
  57. innerConditions.Add("a.order_type = @OrderType");
  58. pars.Add(new SugarParameter("@OrderType", input.OrderType.Value));
  59. }
  60. var baseSql = BuildListBaseSql(string.Join(" AND ", innerConditions));
  61. // state 是 CASE 计算列,需包一层再过滤
  62. var outerWhere = "";
  63. if (!string.IsNullOrWhiteSpace(input.State))
  64. {
  65. outerWhere = "WHERE t.State = @State";
  66. pars.Add(new SugarParameter("@State", input.State.Trim()));
  67. }
  68. var offset = (input.Page - 1) * input.PageSize;
  69. var total = await _db.Ado.GetIntAsync(
  70. $"SELECT COUNT(*) FROM ({baseSql}) AS t {outerWhere}", pars);
  71. var list = await _db.Ado.SqlQueryAsync<SeOrderListRow>(
  72. $"SELECT * FROM ({baseSql}) AS t {outerWhere} ORDER BY t.Id DESC LIMIT {input.PageSize} OFFSET {offset}", pars);
  73. return new { total, page = input.Page, pageSize = input.PageSize, list };
  74. }
  75. private static string BuildListBaseSql(string innerWhere) => $"""
  76. SELECT
  77. a.id AS Id,
  78. a.bill_no AS BillNo,
  79. a.order_type AS OrderType,
  80. a.custom_no AS CustomNo,
  81. b.SortName AS SortName,
  82. a.rdate AS RDate,
  83. a.flowstate AS FlowState,
  84. entry.progress AS Progress,
  85. CASE
  86. WHEN entry.progress = 1 THEN '新建'
  87. WHEN entry.progress = 2 THEN '评审'
  88. WHEN entry.progress = 0 THEN '再评审'
  89. WHEN entry.progress = 3 THEN '确认'
  90. ELSE '确认'
  91. END AS State,
  92. chg.change_content AS ChangeContent,
  93. cN.changeNum AS ChangeNum
  94. FROM crm_seorder a
  95. LEFT JOIN CustMaster b ON a.custom_no = b.Cust
  96. LEFT JOIN (
  97. SELECT sorderid, MAX(create_time) AS create_time
  98. FROM b_examine_result
  99. GROUP BY sorderid
  100. ) d ON a.id = d.sorderid
  101. LEFT JOIN (
  102. SELECT bill_no, change_content
  103. FROM (
  104. SELECT bill_no, change_content, update_time,
  105. ROW_NUMBER() OVER (PARTITION BY bill_no ORDER BY update_time DESC) AS RowNum
  106. FROM crm_seorder_change
  107. WHERE bill_no IS NOT NULL
  108. ) ranked
  109. WHERE RowNum = 1
  110. ) chg ON chg.bill_no = a.bill_no
  111. LEFT JOIN (
  112. SELECT COUNT(*) AS changeNum, bill_no
  113. FROM crm_seorder_change
  114. WHERE bill_no IS NOT NULL
  115. GROUP BY bill_no
  116. ) cN ON cN.bill_no = a.bill_no
  117. LEFT JOIN (
  118. SELECT seorder_id, progress,
  119. ROW_NUMBER() OVER (PARTITION BY seorder_id ORDER BY Id) AS rn
  120. FROM crm_seorderentry
  121. WHERE IsDeleted = 0
  122. ) entry ON a.id = entry.seorder_id AND entry.rn = 1
  123. WHERE {innerWhere}
  124. """;
  125. // ══════════════════════════════════════════════════════════════
  126. // 客户列表 GET /api/Order/seorder/customers
  127. // 注:字面量路由优先于 {id} 参数路由,无需额外排序
  128. // ══════════════════════════════════════════════════════════════
  129. /// <summary>获取客户选择列表 🗂️</summary>
  130. [DisplayName("获取客户列表")]
  131. [HttpGet("seorder/customers")]
  132. public async Task<object> GetCustomers([FromQuery] KeywordPageInput input)
  133. {
  134. var pars = new List<SugarParameter>();
  135. var where = "1=1";
  136. if (!string.IsNullOrWhiteSpace(input.Keyword))
  137. {
  138. where = "(Cust LIKE @Keyword OR SortName LIKE @Keyword)";
  139. pars.Add(new SugarParameter("@Keyword", $"%{input.Keyword.Trim()}%"));
  140. }
  141. var offset = (input.Page - 1) * input.PageSize;
  142. var total = await _db.Ado.GetIntAsync(
  143. $"SELECT COUNT(*) FROM CustMaster WHERE {where}", pars);
  144. var list = await _db.Ado.SqlQueryAsync<SimpleKvRow>(
  145. $"SELECT Cust AS Value, SortName AS Label, NULL AS Extra, NULL AS Id FROM CustMaster WHERE {where} ORDER BY Cust LIMIT {input.PageSize} OFFSET {offset}",
  146. pars);
  147. return new { total, page = input.Page, pageSize = input.PageSize, list };
  148. }
  149. // ══════════════════════════════════════════════════════════════
  150. // 物料列表 GET /api/Order/seorder/items
  151. // TODO: 将 bd_material 替换为实际物料主数据表名
  152. // ══════════════════════════════════════════════════════════════
  153. /// <summary>获取物料选择列表 🗂️</summary>
  154. [DisplayName("获取物料列表")]
  155. [HttpGet("seorder/items")]
  156. public async Task<object> GetItems([FromQuery] KeywordPageInput input)
  157. {
  158. var pars = new List<SugarParameter>();
  159. var where = "IsDeleted = 0";
  160. if (!string.IsNullOrWhiteSpace(input.Keyword))
  161. {
  162. where += " AND (item_number LIKE @Keyword OR item_name LIKE @Keyword)";
  163. pars.Add(new SugarParameter("@Keyword", $"%{input.Keyword.Trim()}%"));
  164. }
  165. var offset = (input.Page - 1) * input.PageSize;
  166. var total = await _db.Ado.GetIntAsync(
  167. $"SELECT COUNT(*) FROM bd_material WHERE {where}", pars);
  168. var list = await _db.Ado.SqlQueryAsync<SimpleKvRow>(
  169. $"SELECT item_number AS Value, item_name AS Label, CONCAT(IFNULL(specification,''),'|',IFNULL(unit,'')) AS Extra, NULL AS Id FROM bd_material WHERE {where} ORDER BY item_number LIMIT {input.PageSize} OFFSET {offset}",
  170. pars);
  171. return new { total, page = input.Page, pageSize = input.PageSize, list };
  172. }
  173. // ══════════════════════════════════════════════════════════════
  174. // 订单详情 GET /api/Order/seorder/{id}
  175. // ══════════════════════════════════════════════════════════════
  176. /// <summary>获取订单详情(含明细)🗂️</summary>
  177. [DisplayName("获取订单详情")]
  178. [HttpGet("seorder/{id}")]
  179. public async Task<object> GetSeOrderDetail(long id)
  180. {
  181. const string orderSql = """
  182. SELECT Id, bill_no AS BillNo, order_type AS OrderType,
  183. custom_id AS CustomId, custom_no AS CustomNo, custom_name AS CustomName,
  184. date AS Date, custom_level AS CustomLevel, urgent AS Urgent,
  185. bill_from AS BillFrom, country AS Country, rdate AS RDate, flowstate AS FlowState
  186. FROM crm_seorder
  187. WHERE Id = @Id AND IsDeleted = 0
  188. LIMIT 1
  189. """;
  190. var order = (await _db.Ado.SqlQueryAsync<SeOrderDetailRow>(orderSql, new { Id = id }))
  191. .FirstOrDefault()
  192. ?? throw Oops.Oh("订单不存在");
  193. const string entrySql = """
  194. SELECT Id, entry_seq AS EntrySeq, item_number AS ItemNumber, item_name AS ItemName,
  195. specification AS Specification, qty AS Qty, plan_date AS PlanDate,
  196. sys_capacity_date AS SysCapacityDate, date AS Date, remark AS Remark,
  197. unit AS Unit, deliver_count AS DeliverCount, progress AS Progress
  198. FROM crm_seorderentry
  199. WHERE seorder_id = @Id AND IsDeleted = 0
  200. ORDER BY entry_seq
  201. """;
  202. var entries = await _db.Ado.SqlQueryAsync<SeOrderEntryRow>(entrySql, new { Id = id });
  203. return new
  204. {
  205. order.Id, order.BillNo, order.OrderType,
  206. order.CustomId, order.CustomNo, order.CustomName,
  207. order.Date, order.CustomLevel, order.Urgent,
  208. order.BillFrom, order.Country, order.RDate, order.FlowState,
  209. entries
  210. };
  211. }
  212. // ══════════════════════════════════════════════════════════════
  213. // 新增 / 编辑 POST /api/Order/seorder/save
  214. // ══════════════════════════════════════════════════════════════
  215. /// <summary>保存销售订单(新增或编辑)🗂️</summary>
  216. [DisplayName("保存销售订单")]
  217. [ApiDescriptionSettings(Name = "SaveSeOrder"), HttpPost("seorder/save")]
  218. public async Task<object> SaveSeOrder([FromBody] SeOrderSaveInput input)
  219. {
  220. if (input.Id is null or 0)
  221. {
  222. // ── 新增:参照 SysJobService.AddJobDetail ──
  223. var tenantId = _userManager.TenantId;
  224. var isExist = await _seOrderRep.IsAnyAsync(u => u.BillNo == input.BillNo && u.IsDeleted == 0 && u.TenantId == tenantId);
  225. if (isExist) throw Oops.Oh("订单编号已存在");
  226. var entity = input.Adapt<SeOrder>();
  227. entity.Id = YitIdHelper.NextId();
  228. entity.IsDeleted = 0;
  229. entity.CreateTime = DateTime.Now;
  230. entity.TenantId = tenantId;
  231. await _seOrderRep.InsertAsync(entity);
  232. await SaveEntriesAsync(entity.Id, input.BillNo, input.Entries);
  233. var orderAfter = await _seOrderRep.GetFirstAsync(u => u.Id == entity.Id && u.IsDeleted == 0 && u.TenantId == tenantId);
  234. var entriesAfter = await _seOrderEntryRep.GetListAsync(u => u.SeOrderId == entity.Id && u.IsDeleted == 0 && u.TenantId == tenantId);
  235. var afterJson = SeOrderAuditHelper.BuildCombinedSnapshotJson(orderAfter!, entriesAfter);
  236. await _orderOpLog.WriteAsync(new SeOrderOperationLogDto
  237. {
  238. UserId = _userManager.UserId,
  239. Action = "新增订单",
  240. TableName = "crm_seorder",
  241. KeyValue = entity.Id.ToString(),
  242. Content = $"新增订单,订单编号:{entity.BillNo},订单主键:{entity.Id}。",
  243. BeforeJson = null,
  244. AfterJson = afterJson
  245. });
  246. return new { id = entity.Id, message = "新增成功" };
  247. }
  248. else
  249. {
  250. // ── 编辑:参照 SysJobService.UpdateJobDetail ──
  251. var tenantId = _userManager.TenantId;
  252. var entity = await _seOrderRep.GetFirstAsync(u => u.Id == input.Id.Value && u.IsDeleted == 0)
  253. ?? throw Oops.Oh("订单不存在");
  254. var beforeEntries = await _seOrderEntryRep.GetListAsync(u => u.SeOrderId == input.Id.Value && u.IsDeleted == 0);
  255. var orderBefore = entity.Adapt<SeOrder>();
  256. var entriesBefore = beforeEntries.Select(e => e.Adapt<SeOrderEntry>()).ToList();
  257. input.Adapt(entity);
  258. entity.UpdateTime = DateTime.Now;
  259. await _seOrderRep.UpdateAsync(entity);
  260. await SaveEntriesAsync(input.Id.Value, input.BillNo, input.Entries);
  261. var afterOrder = await _seOrderRep.GetFirstAsync(u => u.Id == input.Id.Value && u.IsDeleted == 0 && u.TenantId == tenantId);
  262. var afterEntries = await _seOrderEntryRep.GetListAsync(u => u.SeOrderId == input.Id.Value && u.IsDeleted == 0 && u.TenantId == tenantId);
  263. var (content, beforeJson, afterJson) = SeOrderAuditHelper.BuildUpdateDiff(
  264. orderBefore!, entriesBefore, afterOrder!, afterEntries);
  265. await _orderOpLog.WriteAsync(new SeOrderOperationLogDto
  266. {
  267. UserId = _userManager.UserId,
  268. Action = "修改订单",
  269. TableName = "crm_seorder",
  270. KeyValue = input.Id.Value.ToString(),
  271. Content = content,
  272. BeforeJson = beforeJson,
  273. AfterJson = afterJson
  274. });
  275. return new { id = input.Id, message = "编辑成功" };
  276. }
  277. }
  278. /// <summary>
  279. /// 明细三路合并:
  280. /// ① DB有且入参有(按 Id 匹配)→ 更新
  281. /// ② DB无但入参有 → 新增(补填 bill_no / progress=0 / create_by)
  282. /// ③ DB有但入参无 → 删除
  283. /// </summary>
  284. private async Task SaveEntriesAsync(long orderId, string billNo, List<SeOrderEntryInput> entries)
  285. {
  286. var tenantId = _userManager.TenantId;
  287. // 取 DB 现有明细,以 Id 为 key 建索引
  288. var dbEntries = await _seOrderEntryRep.GetListAsync(u => u.SeOrderId == orderId && u.IsDeleted == 0 && u.TenantId == tenantId);
  289. var dbById = dbEntries.ToDictionary(u => u.Id);
  290. // 入参中携带 Id 的集合(用于判断 DB 行是否需要删除)
  291. var inputIds = new HashSet<long>(entries.Where(e => e.Id > 0).Select(e => e.Id!.Value));
  292. for (var i = 0; i < entries.Count; i++)
  293. {
  294. var e = entries[i];
  295. var seq = e.EntrySeq ?? (i + 1);
  296. if (e.Id > 0 && dbById.TryGetValue(e.Id.Value, out var existing))
  297. {
  298. // ① DB有、参数有 → 更新(参照 SysJobService.UpdateJobDetail)
  299. e.Adapt(existing);
  300. existing.EntrySeq = seq;
  301. existing.UpdateTime = DateTime.Now;
  302. await _seOrderEntryRep.UpdateAsync(existing);
  303. }
  304. else
  305. {
  306. // ② DB无、参数有 → 新增(参照 SysJobService.AddJobDetail)
  307. var entry = e.Adapt<SeOrderEntry>();
  308. entry.Id = YitIdHelper.NextId();
  309. entry.SeOrderId = orderId;
  310. entry.BillNo = billNo;
  311. entry.EntrySeq = seq;
  312. entry.Progress ??= 0;
  313. entry.CreateBy = _userManager.UserId;
  314. entry.IsDeleted = 0;
  315. entry.CreateTime = DateTime.Now;
  316. entry.TenantId = tenantId;
  317. await _seOrderEntryRep.InsertAsync(entry);
  318. }
  319. }
  320. // ③ DB有、参数无 → 删除(参照 SysJobService.DeleteJobDetail)
  321. foreach (var toDelete in dbEntries.Where(u => !inputIds.Contains(u.Id)))
  322. {
  323. await _seOrderEntryRep.DeleteAsync(u => u.Id == toDelete.Id);
  324. }
  325. }
  326. // ══════════════════════════════════════════════════════════════
  327. // 订单评审 / 交期确认 — 已改为前端直接调用外部接口,此处仅保留存根
  328. // ══════════════════════════════════════════════════════════════
  329. // ══════════════════════════════════════════════════════════════
  330. // 资源解锁 POST /api/Order/seorder/{id}/unlock
  331. // ══════════════════════════════════════════════════════════════
  332. /// <summary>资源解锁 🔓</summary>
  333. [DisplayName("资源解锁")]
  334. [ApiDescriptionSettings(Name = "UnlockSeOrder"), HttpPost("seorder/{id}/unlock")]
  335. public async Task<object> UnlockSeOrder(long id)
  336. {
  337. return new { message = "解锁成功" };
  338. }
  339. // ══════════════════════════════════════════════════════════════
  340. // 变更申请 POST /api/Order/seorder/{id}/change
  341. // ══════════════════════════════════════════════════════════════
  342. /// <summary>提交订单变更申请 🔖</summary>
  343. [DisplayName("提交订单变更申请")]
  344. [ApiDescriptionSettings(Name = "CreateSeOrderChange"), HttpPost("seorder/{id}/change")]
  345. public async Task<object> CreateChange(long id, [FromBody] SeOrderChangeSaveInput input)
  346. {
  347. var tenantId = _userManager.TenantId;
  348. // 参照 SysJobService.AddJobDetail:Adapt → 设主键与审计字段 → InsertAsync
  349. var entity = input.Adapt<SeOrderChange>();
  350. entity.Id = YitIdHelper.NextId();
  351. entity.SeOrderId = id;
  352. entity.IsDeleted = 0;
  353. entity.CreateTime = DateTime.Now;
  354. entity.TenantId = tenantId;
  355. await _seOrderChangeRep.InsertAsync(entity);
  356. return new { id = entity.Id, message = "变更申请已保存" };
  357. }
  358. // ──────────────── 内部查询结果映射类 ────────────────
  359. private sealed class SeOrderListRow
  360. {
  361. public long Id { get; set; }
  362. public string? BillNo { get; set; }
  363. public int? OrderType { get; set; }
  364. public string? CustomNo { get; set; }
  365. public string? SortName { get; set; }
  366. public DateTime? RDate { get; set; }
  367. public int? Progress { get; set; }
  368. public string? State { get; set; }
  369. public string? ChangeContent { get; set; }
  370. public int? ChangeNum { get; set; }
  371. public string? FlowState { get; set; }
  372. }
  373. private sealed class SeOrderDetailRow
  374. {
  375. public long Id { get; set; }
  376. public string? BillNo { get; set; }
  377. public int? OrderType { get; set; }
  378. public long? CustomId { get; set; }
  379. public string? CustomNo { get; set; }
  380. public string? CustomName { get; set; }
  381. public DateTime? Date { get; set; }
  382. public int? CustomLevel { get; set; }
  383. public int? Urgent { get; set; }
  384. public string? BillFrom { get; set; }
  385. public string? Country { get; set; }
  386. public DateTime? RDate { get; set; }
  387. public string? FlowState { get; set; }
  388. }
  389. private sealed class SeOrderEntryRow
  390. {
  391. public long Id { get; set; }
  392. public int? EntrySeq { get; set; }
  393. public string? ItemNumber { get; set; }
  394. public string? ItemName { get; set; }
  395. public string? Specification { get; set; }
  396. public decimal? Qty { get; set; }
  397. public DateTime? PlanDate { get; set; }
  398. public DateTime? SysCapacityDate { get; set; }
  399. public DateTime? Date { get; set; }
  400. public string? Remark { get; set; }
  401. public string? Unit { get; set; }
  402. public decimal? DeliverCount { get; set; }
  403. public int? Progress { get; set; }
  404. }
  405. private sealed class SimpleKvRow
  406. {
  407. public string? Value { get; set; }
  408. public string? Label { get; set; }
  409. public string? Extra { get; set; }
  410. public long? Id { get; set; }
  411. }
  412. // ══════════════════════════════════════════════════════════════
  413. // 提交订单评审审批
  414. // ══════════════════════════════════════════════════════════════
  415. /// <summary>提交订单评审审批流程</summary>
  416. [DisplayName("提交订单评审审批")]
  417. [HttpPost("seorder/submitReview")]
  418. public async Task<object> SubmitOrderReview([FromBody] SubmitReviewInput input)
  419. {
  420. var order = await _seOrderRep.GetByIdAsync(input.OrderId)
  421. ?? throw Oops.Oh("订单不存在");
  422. var engine = App.GetRequiredService<Admin.NET.Plugin.ApprovalFlow.Service.FlowEngineService>();
  423. var instanceId = await engine.StartFlow(new Admin.NET.Plugin.ApprovalFlow.Service.StartFlowInput
  424. {
  425. BizType = "ORDER_REVIEW",
  426. BizId = order.Id,
  427. BizNo = order.BillNo,
  428. Title = $"订单评审-{order.BillNo}",
  429. Comment = input.Comment,
  430. });
  431. return new { instanceId };
  432. }
  433. public class SubmitReviewInput
  434. {
  435. public long OrderId { get; set; }
  436. public string? Comment { get; set; }
  437. }
  438. }