ContractReviewService.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339
  1. namespace Admin.NET.Plugin.AiDOP.Order;
  2. /// <summary>
  3. /// 合同评审服务 📋
  4. /// 路由前缀:/api/Order/contract/...
  5. /// 审批:由审批流插件(BizType=CONTRACT_REVIEW)+ 前端 ApprovalPanel 发起与办理;本服务仅保留列表/详情/保存/删除。
  6. /// </summary>
  7. [ApiDescriptionSettings(Order = 250, Description = "合同评审")]
  8. [Route("api/Order")]
  9. [AllowAnonymous]
  10. [NonUnify]
  11. public class ContractReviewService : IDynamicApiController, ITransient
  12. {
  13. private readonly ISqlSugarClient _db;
  14. private readonly SqlSugarRepository<ContractReview> _reviewRep;
  15. private readonly SqlSugarRepository<ContractReviewFlow> _flowRep;
  16. private readonly UserManager _userManager;
  17. public ContractReviewService(
  18. ISqlSugarClient db,
  19. SqlSugarRepository<ContractReview> reviewRep,
  20. SqlSugarRepository<ContractReviewFlow> flowRep,
  21. UserManager userManager)
  22. {
  23. _db = db;
  24. _reviewRep = reviewRep;
  25. _flowRep = flowRep;
  26. _userManager = userManager;
  27. }
  28. // ══════════════════════════════════════════════════════════════
  29. // 列表 GET /api/Order/contract/list
  30. // ══════════════════════════════════════════════════════════════
  31. /// <summary>获取合同评审分页列表 📋</summary>
  32. [DisplayName("获取合同评审列表")]
  33. [HttpGet("contract/list")]
  34. public async Task<object> GetContractReviewList([FromQuery] ContractReviewListInput input)
  35. {
  36. var tenantId = _userManager.TenantId;
  37. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  38. var conditions = new List<string> { "tenant_id = @TenantId" };
  39. if (!string.IsNullOrWhiteSpace(input.BillNo))
  40. {
  41. conditions.Add("BillNo LIKE @BillNo");
  42. pars.Add(new SugarParameter("@BillNo", $"%{input.BillNo.Trim()}%"));
  43. }
  44. if (!string.IsNullOrWhiteSpace(input.Title))
  45. {
  46. conditions.Add("Title LIKE @Title");
  47. pars.Add(new SugarParameter("@Title", $"%{input.Title.Trim()}%"));
  48. }
  49. if (!string.IsNullOrWhiteSpace(input.CustomerName))
  50. {
  51. conditions.Add("CustomerName LIKE @CustomerName");
  52. pars.Add(new SugarParameter("@CustomerName", $"%{input.CustomerName.Trim()}%"));
  53. }
  54. if (!string.IsNullOrWhiteSpace(input.FlowStatus))
  55. {
  56. conditions.Add("FlowStatus = @FlowStatus");
  57. pars.Add(new SugarParameter("@FlowStatus", input.FlowStatus.Trim()));
  58. }
  59. var whereClause = "WHERE " + string.Join(" AND ", conditions);
  60. var baseSql = $"SELECT * FROM ado_contract_review {whereClause}";
  61. var offset = (input.Page - 1) * input.PageSize;
  62. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(*) FROM ({baseSql}) AS t", pars);
  63. var list = await _db.Ado.SqlQueryAsync<ContractReviewListRow>(
  64. $"SELECT * FROM ({baseSql}) AS t ORDER BY t.RecID DESC LIMIT {input.PageSize} OFFSET {offset}",
  65. pars);
  66. return new { total, page = input.Page, pageSize = input.PageSize, list };
  67. }
  68. // ══════════════════════════════════════════════════════════════
  69. // 详情 GET /api/Order/contract/{id}
  70. // ══════════════════════════════════════════════════════════════
  71. /// <summary>获取合同评审详情(含流程节点)📋</summary>
  72. [DisplayName("获取合同评审详情")]
  73. [HttpGet("contract/{id}")]
  74. public async Task<object> GetContractReviewDetail(int id)
  75. {
  76. var tenantId = _userManager.TenantId;
  77. var review = await _reviewRep.GetFirstAsync(u => u.RecID == id && u.TenantId == tenantId)
  78. ?? throw Oops.Oh("合同评审记录不存在");
  79. var flows = await _flowRep.GetListAsync(u => u.ReviewRecID == id && u.TenantId == tenantId);
  80. return new
  81. {
  82. recID = review.RecID,
  83. billNo = review.BillNo,
  84. title = review.Title,
  85. customerName = review.CustomerName,
  86. customerNo = review.CustomerNo,
  87. salesCompany = review.SalesCompany,
  88. salesArea = review.SalesArea,
  89. projectCode = review.ProjectCode,
  90. crmNo = review.CrmNo,
  91. responsibleAccount = review.ResponsibleAccount,
  92. responsibleName = review.ResponsibleName,
  93. projectStartDate = review.ProjectStartDate?.ToString("yyyy-MM-dd"),
  94. currentStage = review.CurrentStage ?? 0,
  95. currentDept = review.CurrentDept,
  96. projectStatus = review.ProjectStatus,
  97. winRate = review.WinRate,
  98. expectedOrderMonth = review.ExpectedOrderMonth?.ToString("yyyy-MM"),
  99. expectedDeliveryDate = review.ExpectedDeliveryDate?.ToString("yyyy-MM-dd"),
  100. projectRequirement = review.ProjectRequirement,
  101. remark = review.Remark,
  102. flowStatus = review.FlowStatus ?? "draft",
  103. createUser = review.CreateUser,
  104. createTime = review.CreateTime?.ToString("yyyy-MM-dd HH:mm:ss"),
  105. updateTime = review.UpdateTime?.ToString("yyyy-MM-dd HH:mm:ss"),
  106. flows = flows.OrderBy(f => f.StageNo).ThenBy(f => f.Seq).Select(f => new
  107. {
  108. recID = f.RecID,
  109. reviewRecID = f.ReviewRecID,
  110. reviewBillNo = f.ReviewBillNo,
  111. stageNo = f.StageNo,
  112. stageName = f.StageName,
  113. department = f.Department,
  114. deptNo = f.DeptNo,
  115. seq = f.Seq,
  116. reviewerAccount = f.ReviewerAccount,
  117. reviewerName = f.ReviewerName,
  118. opinion = f.Opinion,
  119. startTime = f.StartTime?.ToString("yyyy-MM-dd HH:mm:ss"),
  120. completeTime = f.CompleteTime?.ToString("yyyy-MM-dd HH:mm:ss"),
  121. actualDays = f.ActualDays,
  122. nodeStatus = f.NodeStatus ?? "pending",
  123. })
  124. };
  125. }
  126. // ══════════════════════════════════════════════════════════════
  127. // 保存(新增/编辑)POST /api/Order/contract/save
  128. // ══════════════════════════════════════════════════════════════
  129. /// <summary>保存合同评审(草稿新增或编辑)📋</summary>
  130. [DisplayName("保存合同评审")]
  131. [ApiDescriptionSettings(Name = "SaveContractReview"), HttpPost("contract/save")]
  132. public async Task<object> SaveContractReview([FromBody] ContractReviewSaveInput input)
  133. {
  134. // 必填校验
  135. if (string.IsNullOrWhiteSpace(input.ProjectCode))
  136. throw Oops.Oh("合同编号不能为空");
  137. if (string.IsNullOrWhiteSpace(input.ProjectStartDate))
  138. throw Oops.Oh("合同开始时间不能为空");
  139. if (string.IsNullOrWhiteSpace(input.SalesCompany))
  140. throw Oops.Oh("销售公司不能为空");
  141. if (string.IsNullOrWhiteSpace(input.CustomerName))
  142. throw Oops.Oh("客户名称不能为空");
  143. if (string.IsNullOrWhiteSpace(input.Title))
  144. throw Oops.Oh("合同标题不能为空");
  145. var tenantId = _userManager.TenantId;
  146. // 合同编号唯一性校验
  147. var existingId = input.RecID ?? 0;
  148. var codeExists = await _reviewRep.IsAnyAsync(u =>
  149. u.ProjectCode == input.ProjectCode.Trim()
  150. && u.TenantId == tenantId
  151. && u.RecID != existingId);
  152. if (codeExists)
  153. throw Oops.Oh($"合同编号「{input.ProjectCode.Trim()}」已存在,请使用其他编号");
  154. var now = DateTime.Now;
  155. var user = _userManager.Account ?? "system";
  156. if (input.RecID is null or 0)
  157. {
  158. // ── 新增:参照 SysJobService.AddJobDetail ──
  159. var month = now.ToString("yyyyMM");
  160. var maxSeq = await _db.Ado.GetIntAsync(
  161. $"SELECT IFNULL(MAX(CAST(SUBSTRING(BillNo, 9) AS UNSIGNED)), 0) FROM ado_contract_review WHERE BillNo LIKE 'CR{month}%' AND tenant_id = @TenantId",
  162. new SugarParameter("@TenantId", tenantId));
  163. var billNo = $"CR{month}{(maxSeq + 1):D4}";
  164. var entity = new ContractReview
  165. {
  166. BillNo = billNo,
  167. Title = input.Title,
  168. CustomerName = input.CustomerName,
  169. CustomerNo = input.CustomerNo,
  170. SalesCompany = input.SalesCompany,
  171. SalesArea = input.SalesArea,
  172. ProjectCode = input.ProjectCode,
  173. CrmNo = input.CrmNo,
  174. ResponsibleAccount = input.ResponsibleAccount,
  175. ResponsibleName = input.ResponsibleName,
  176. ProjectStartDate = string.IsNullOrWhiteSpace(input.ProjectStartDate) ? null : DateTime.Parse(input.ProjectStartDate),
  177. ProjectStatus = input.ProjectStatus,
  178. WinRate = input.WinRate,
  179. ExpectedOrderMonth = string.IsNullOrWhiteSpace(input.ExpectedOrderMonth) ? null : DateTime.Parse(input.ExpectedOrderMonth + "-01"),
  180. ExpectedDeliveryDate = string.IsNullOrWhiteSpace(input.ExpectedDeliveryDate) ? null : DateTime.Parse(input.ExpectedDeliveryDate),
  181. ProjectRequirement = input.ProjectRequirement,
  182. Remark = input.Remark,
  183. FlowStatus = "draft",
  184. CurrentStage = 0,
  185. CreateUser = user,
  186. CreateTime = now,
  187. IsActive = 1,
  188. TenantId = tenantId,
  189. };
  190. await _reviewRep.InsertAsync(entity);
  191. // 取刚插入行的 RecID
  192. var newId = await _db.Ado.GetIntAsync(
  193. "SELECT MAX(RecID) FROM ado_contract_review WHERE BillNo = @BillNo",
  194. new SugarParameter("@BillNo", billNo));
  195. return new { id = newId, billNo, message = "新增成功" };
  196. }
  197. else
  198. {
  199. // ── 编辑:参照 SysJobService.UpdateJobDetail ──
  200. var entity = await _reviewRep.GetFirstAsync(u => u.RecID == input.RecID.Value && u.TenantId == tenantId)
  201. ?? throw Oops.Oh("合同评审记录不存在");
  202. if (entity.FlowStatus == "reviewing")
  203. throw Oops.Oh("审批中的合同评审不允许编辑,如需修改请先驳回");
  204. entity.Title = input.Title;
  205. entity.CustomerName = input.CustomerName;
  206. entity.CustomerNo = input.CustomerNo;
  207. entity.SalesCompany = input.SalesCompany;
  208. entity.SalesArea = input.SalesArea;
  209. entity.ProjectCode = input.ProjectCode;
  210. entity.CrmNo = input.CrmNo;
  211. entity.ResponsibleAccount = input.ResponsibleAccount;
  212. entity.ResponsibleName = input.ResponsibleName;
  213. entity.ProjectStartDate = string.IsNullOrWhiteSpace(input.ProjectStartDate) ? null : DateTime.Parse(input.ProjectStartDate);
  214. entity.ProjectStatus = input.ProjectStatus;
  215. entity.WinRate = input.WinRate;
  216. entity.ExpectedOrderMonth = string.IsNullOrWhiteSpace(input.ExpectedOrderMonth) ? null : DateTime.Parse(input.ExpectedOrderMonth + "-01");
  217. entity.ExpectedDeliveryDate = string.IsNullOrWhiteSpace(input.ExpectedDeliveryDate) ? null : DateTime.Parse(input.ExpectedDeliveryDate);
  218. entity.ProjectRequirement = input.ProjectRequirement;
  219. entity.Remark = input.Remark;
  220. entity.UpdateUser = user;
  221. entity.UpdateTime = now;
  222. await _reviewRep.UpdateAsync(entity);
  223. return new { id = entity.RecID, billNo = entity.BillNo, message = "编辑成功" };
  224. }
  225. }
  226. // ══════════════════════════════════════════════════════════════
  227. // 删除 POST /api/Order/contract/delete
  228. // ══════════════════════════════════════════════════════════════
  229. /// <summary>删除合同评审(审批中不允许删除)📋</summary>
  230. [DisplayName("删除合同评审")]
  231. [ApiDescriptionSettings(Name = "DeleteContractReview"), HttpPost("contract/delete")]
  232. public async Task<object> DeleteContractReview([FromBody] ContractReviewDeleteInput input)
  233. {
  234. // 参照 SysJobService.DeleteJobDetail:先查再删,同时删关联数据
  235. var tenantId = _userManager.TenantId;
  236. var entity = await _reviewRep.GetFirstAsync(u => u.RecID == input.RecID && u.TenantId == tenantId)
  237. ?? throw Oops.Oh("合同评审记录不存在");
  238. if (entity.FlowStatus == "reviewing")
  239. throw Oops.Oh("审批中的合同评审不允许删除");
  240. await _flowRep.DeleteAsync(u => u.ReviewRecID == input.RecID);
  241. await _reviewRep.DeleteAsync(u => u.RecID == input.RecID);
  242. return new { message = "删除成功" };
  243. }
  244. // ══════════════════════════════════════════════════════════════
  245. // 合同编号唯一性校验 GET /api/Order/contract/check-project-code
  246. // ══════════════════════════════════════════════════════════════
  247. /// <summary>校验合同编号唯一性 📋</summary>
  248. [DisplayName("校验合同编号唯一性")]
  249. [HttpGet("contract/check-project-code")]
  250. public async Task<object> CheckProjectCode([FromQuery] string projectCode, [FromQuery] int excludeId = 0)
  251. {
  252. if (string.IsNullOrWhiteSpace(projectCode))
  253. return new { exists = false };
  254. var tenantId = _userManager.TenantId;
  255. var exists = await _reviewRep.IsAnyAsync(u =>
  256. u.ProjectCode == projectCode.Trim()
  257. && u.TenantId == tenantId
  258. && u.RecID != excludeId);
  259. return new { exists };
  260. }
  261. // ══════════════════════════════════════════════════════════════
  262. // 合同负责人下拉框 GET /api/Order/contract/responsible-users
  263. // ══════════════════════════════════════════════════════════════
  264. /// <summary>获取合同负责人下拉选项(当前租户用户列表)📋</summary>
  265. [DisplayName("获取合同负责人下拉选项")]
  266. [HttpGet("contract/responsible-users")]
  267. public async Task<List<object>> GetResponsibleUsers([FromQuery] string? keyword)
  268. {
  269. var tenantId = _userManager.TenantId;
  270. var pars = new List<SugarParameter> { new("@TenantId", tenantId) };
  271. var conditions = new List<string> { "tenant_id = @TenantId" };
  272. if (!string.IsNullOrWhiteSpace(keyword))
  273. {
  274. conditions.Add("(Account LIKE @Keyword OR RealName LIKE @Keyword)");
  275. pars.Add(new SugarParameter("@Keyword", $"%{keyword.Trim()}%"));
  276. }
  277. var whereClause = "WHERE " + string.Join(" AND ", conditions);
  278. var sql = $"SELECT Account, RealName FROM sys_user {whereClause} ORDER BY Account LIMIT 100";
  279. var rows = await _db.Ado.SqlQueryAsync<ResponsibleUserRow>(sql, pars);
  280. return rows.Select(r => (object)new { value = r.Account, label = r.RealName }).ToList();
  281. }
  282. // ──────────────── 内部查询结果映射类 ────────────────
  283. private sealed class ResponsibleUserRow
  284. {
  285. public string? Account { get; set; }
  286. public string? RealName { get; set; }
  287. }
  288. private sealed class ContractReviewListRow
  289. {
  290. public int RecID { get; set; }
  291. public string? BillNo { get; set; }
  292. public string? Title { get; set; }
  293. public string? CustomerName { get; set; }
  294. public string? SalesCompany { get; set; }
  295. public int? CurrentStage { get; set; }
  296. public string? CurrentDept { get; set; }
  297. public string? FlowStatus { get; set; }
  298. public string? ResponsibleAccount { get; set; }
  299. public string? ResponsibleName { get; set; }
  300. public DateTime? CreateTime { get; set; }
  301. public DateTime? UpdateTime { get; set; }
  302. }
  303. }