CompanyDailyKpiRepository.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  1. using AutoMapper;
  2. using EyeSoft.Messanging;
  3. using MathNet.Numerics;
  4. using OASystem.Domain.Dtos.PersonnelModule;
  5. using OASystem.Domain.Entities.PersonnelModule;
  6. using OASystem.Domain.ViewModels.PersonnelModule;
  7. using OASystem.Domain.ViewModels.QiYeWeChat;
  8. using OASystem.Infrastructure.Repositories.System;
  9. namespace OASystem.Infrastructure.Repositories.PersonnelModule
  10. {
  11. /// <summary>
  12. /// 公司日常KPI考核仓储
  13. /// </summary>
  14. public class CompanyDailyKpiRepository : BaseRepository<Pm_CompanyDailyKpi, Pm_CompanyDailyKpi>
  15. {
  16. private readonly IMapper _mapper;
  17. private JsonView _jv;
  18. public CompanyDailyKpiRepository(SqlSugarClient sqlSugar,
  19. IMapper mapper,
  20. ApprovalProcessRepository approvalProcessRep)
  21. : base(sqlSugar)
  22. {
  23. _mapper = mapper;
  24. _jv = new JsonView() { Code = StatusCodes.Status400BadRequest, Msg = "操作失败!" };
  25. }
  26. /// <summary>
  27. /// 绩效模板数据(保持原方法签名)
  28. /// </summary>
  29. public async Task<(bool isSuccess, string msg, List<KpiTempTypeInfo> data)> KpiTemplateDataAsync(string depName, string jobName)
  30. {
  31. var result = new List<KpiTempTypeInfo>();
  32. // 获取配置
  33. var config = await _sqlSugar.Queryable<Sys_SetDataType>()
  34. .Where(x => x.IsDel == 0 && x.Id == 132)
  35. .FirstAsync();
  36. if (config?.Remark == null)
  37. return (false, "绩效考核-数据类型ID存储为空", result);
  38. var typeIds = JsonSerializer.Deserialize<List<int>>(config.Remark);
  39. if (typeIds?.Any() != true)
  40. return (false, "绩效考核-数据类型ID存储为空", result);
  41. // 构建查询
  42. var query = _sqlSugar.Queryable<Sys_SetData>()
  43. .LeftJoin<Sys_SetDataType>((sd, sdt) => sd.STid == sdt.Id)
  44. .Where((sd, sdt) => sd.IsDel == 0 && typeIds.Contains(sd.STid));
  45. // 条件过滤
  46. if (!string.IsNullOrWhiteSpace(depName))
  47. query = query.Where((sd, sdt) => sdt.Name.Contains(depName));
  48. if (!string.IsNullOrWhiteSpace(jobName))
  49. query = query.Where((sd, sdt) => sdt.Name.Contains(jobName));
  50. // 查询数据
  51. var data = await query
  52. .Select((sd, sdt) => new
  53. {
  54. KpiTypeId = sd.STid,
  55. KpiTypeName = sdt.Name,
  56. KpiId = sd.Id,
  57. KpiContent = sd.Remark,
  58. KpiSort = sd.Name
  59. })
  60. .ToListAsync();
  61. if (data?.Any() != true)
  62. return (false, "绩效考核模板信息为空", result);
  63. // 分组处理
  64. result = data
  65. .GroupBy(x => new { x.KpiTypeId, x.KpiTypeName })
  66. .Select(g => new KpiTempTypeInfo()
  67. {
  68. TypeId = g.Key.KpiTypeId,
  69. TypeName = g.Key.KpiTypeName,
  70. Contents = g
  71. .OrderBy(x => int.TryParse(x.KpiSort, out int s) ? s : int.MaxValue)
  72. .ThenBy(x => x.KpiSort)
  73. .Select(x => new KpiTempContentInfo(x.KpiId, x.KpiContent, x.KpiSort))
  74. .ToList()
  75. })
  76. .Where(g => g.Contents.Any())
  77. .OrderBy(g => g.TypeId)
  78. .ToList();
  79. return (true, "操作成功", result);
  80. }
  81. /// <summary>
  82. /// 获取部门和岗位
  83. /// </summary>
  84. /// <param name="depName">部门名称 空查询全部</param>
  85. /// <returns></returns>
  86. public async Task<(bool Success, string Message, List<string> Departments, List<KpiTempDepartmentInfo> DepartmentsWithJobs)>
  87. KpiDepartmentsAndJobs(string depName = "")
  88. {
  89. var departments = new List<string>();
  90. var departmentsWithJobs = new List<KpiTempDepartmentInfo>();
  91. var config = await _sqlSugar.Queryable<Sys_SetDataType>()
  92. .Where(x => x.IsDel == 0 && x.Id == 132)
  93. .FirstAsync();
  94. if (config == null) return (false, "配置为空", departments, departmentsWithJobs);
  95. var typeIds = JsonSerializer.Deserialize<List<int>>(config.Remark ?? "[]");
  96. if (typeIds == null || typeIds.Count < 1)
  97. return (false, "类型ID为空", departments, departmentsWithJobs);
  98. var names = await _sqlSugar.Queryable<Sys_SetData>()
  99. .LeftJoin<Sys_SetDataType>((sd, sdt) => sd.STid == sdt.Id)
  100. .Where((sd, sdt) => sd.IsDel == 0 && typeIds.Contains(sd.STid))
  101. .Select((sd, sdt) => sdt.Name)
  102. .Distinct()
  103. .ToListAsync();
  104. // 解析部门和岗位
  105. var deptJobList = names
  106. .Select(name =>
  107. {
  108. var clean = name.Split('(')[0].Trim();
  109. string dept, job;
  110. if (clean.Contains(" - "))
  111. {
  112. var parts = clean.Split(" - ");
  113. dept = parts[0].EndsWith("部") ? parts[0] : parts[0] + "部";
  114. job = parts.Length > 1 ? parts[1] : "通用";
  115. }
  116. else if (clean.Contains("-"))
  117. {
  118. var parts = clean.Split('-');
  119. dept = parts[0].EndsWith("部") ? parts[0] : parts[0] + "部";
  120. job = parts.Length > 1 ? parts[1] : "通用";
  121. }
  122. else
  123. {
  124. dept = clean.EndsWith("部") ? clean : "未分类";
  125. job = clean.EndsWith("部") ? "通用" : clean;
  126. }
  127. return new { Department = dept, Job = job };
  128. })
  129. .Where(x => x.Department != "未分类")
  130. .WhereIF(!string.IsNullOrEmpty(depName),x=> x.Department.Equals(depName))
  131. .GroupBy(x => x.Department)
  132. .Select(g => new KpiTempDepartmentInfo()
  133. {
  134. DepName = g.Key,
  135. JobNames = g.Select(x => x.Job).Distinct().Where(j => j != "通用").OrderBy(j => j).ToList()
  136. })
  137. .OrderBy(g => g.DepName)
  138. .ToList();
  139. departmentsWithJobs = deptJobList;
  140. departments = deptJobList.Select(g => g.DepName).ToList();
  141. return (true, "操作成功", departments, departmentsWithJobs);
  142. }
  143. /// <summary>
  144. /// 基础数据
  145. /// </summary>
  146. /// <returns></returns>
  147. public async Task<JsonView> InitAsync()
  148. {
  149. var result = await KpiDepartmentsAndJobs("财务部");
  150. if (!result.Success)
  151. {
  152. _jv.Msg = result.Message;
  153. return _jv;
  154. }
  155. var userInfos = await _sqlSugar.Queryable<Sys_Users>()
  156. .LeftJoin<Sys_JobPost>((u, jp) => u.JobPostId == jp.Id)
  157. .LeftJoin<Sys_Department>((u, jp, d) => u.DepId == d.Id)
  158. .LeftJoin<Sys_Company>((u, jp, d, c) => u.CompanyId == c.Id)
  159. .Where((u, jp, d, c) => u.IsDel == 0)
  160. .Select((u, jp, d, c) => new
  161. {
  162. UserId = u.Id,
  163. u.CompanyId,
  164. c.CompanyName,
  165. u.DepId,
  166. d.DepName,
  167. u.JobPostId,
  168. jp.JobName,
  169. UserName = u.CnName
  170. })
  171. .OrderBy(u => u.CompanyId)
  172. .ToListAsync();
  173. var jobNames = result.DepartmentsWithJobs
  174. .SelectMany(x => x.JobNames)
  175. .ToList();
  176. var view = userInfos.Where(x => jobNames.Contains(x.JobName))
  177. .Select(x => new KpiTempUserInfo() { Id = x.UserId, DepName = x.DepName, JobName = x.JobName, Name = x.UserName })
  178. .ToList();
  179. _jv.Code = StatusCodes.Status200OK;
  180. _jv.Data = view;
  181. _jv.Msg = $"操作成功";
  182. return _jv;
  183. }
  184. /// <summary>
  185. /// 考勤详情
  186. /// </summary>
  187. /// <param name="month">月份 2025-12</param>
  188. /// <param name="evaluator">考核人</param>
  189. /// <returns></returns>
  190. public async Task<JsonView> InfoAsync(string month, int evaluator)
  191. {
  192. if (string.IsNullOrEmpty(month))
  193. {
  194. _jv.Msg = $"请选择月份!";
  195. return _jv;
  196. }
  197. // 格式:2025-11
  198. const string pattern = @"^(\d{4})-(0[1-9]|1[0-2])$";
  199. if (!Regex.IsMatch(month, pattern))
  200. {
  201. _jv.Msg = $"请选择正确的月份!格式:2025-12";
  202. return _jv;
  203. }
  204. var evaluatorInfo = await _sqlSugar.Queryable<Sys_Users>()
  205. .LeftJoin<Sys_Company>((u, c) => u.CompanyId == c.Id)
  206. .LeftJoin<Sys_Department>((u, c, d) => u.DepId == d.Id)
  207. .LeftJoin<Sys_JobPost>((u, c, d, jp) => u.JobPostId == jp.Id)
  208. .Where((u, c, d, jp) => u.IsDel == 0 && u.Id == evaluator)
  209. .Select((u, c, d, jp) => new
  210. {
  211. UserId = u.Id,
  212. u.CompanyId,
  213. c.CompanyName,
  214. u.DepId,
  215. d.DepName,
  216. u.JobPostId,
  217. jp.JobName,
  218. UserName = u.CnName
  219. })
  220. .FirstAsync();
  221. if (evaluatorInfo == null)
  222. {
  223. _jv.Msg = $"请选择有效的考核人!";
  224. return _jv;
  225. }
  226. var info = await _sqlSugar
  227. .Queryable<Pm_CompanyDailyKpi>()
  228. .LeftJoin<Sys_Users>((cdk, u) => cdk.Evaluator == u.Id)
  229. .Where((cdk, u) => cdk.IsDel == 0 && cdk.Evaluator == evaluator && cdk.Month.Equals(month))
  230. .Select((cdk, u) => new CompanyDailyKpiView()
  231. {
  232. Id = cdk.Id,
  233. Month = cdk.Month,
  234. EvalContentOrder = cdk.EvalContentOrder,
  235. Evaluator = cdk.Evaluator,
  236. EvaluatorName = u.CnName,
  237. EvalContent = cdk.EvalContent,
  238. IsMistake = cdk.IsMistake,
  239. MistakeReason = cdk.MistakeReason
  240. })
  241. .ToListAsync();
  242. if (!info.Any())
  243. {
  244. var tempInfo = await KpiTemplateDataAsync(evaluatorInfo.DepName, evaluatorInfo.JobName);
  245. if (!tempInfo.isSuccess)
  246. {
  247. _jv.Msg = $"{evaluatorInfo.DepName}-{evaluatorInfo.JobName},未配置日常绩效考核。";
  248. return _jv;
  249. }
  250. tempInfo.data.ForEach(t =>
  251. {
  252. t.Contents.ForEach(c =>
  253. {
  254. _ = int.TryParse(c.KpiSort, out int no);
  255. info.Add(new CompanyDailyKpiView()
  256. {
  257. Month = month,
  258. EvalContentOrder = no,
  259. Evaluator = evaluatorInfo.UserId,
  260. EvaluatorName = evaluatorInfo.UserName,
  261. EvalContent = c.KpiContent,
  262. IsMistake = false,
  263. MistakeReason = string.Empty
  264. });
  265. });
  266. });
  267. }
  268. _jv.Code = StatusCodes.Status200OK;
  269. _jv.Data = info;
  270. _jv.Msg = $"操作成功";
  271. return _jv;
  272. }
  273. /// <summary>
  274. /// kpi Save
  275. /// </summary>
  276. /// <param name="dto">请求类</param>
  277. /// <returns></returns>
  278. public async Task<JsonView> SaveAsync(CompanyDailyKpiSaveDto dto)
  279. {
  280. if (dto.Infos == null || !dto.Infos.Any())
  281. {
  282. _jv.Msg = $"请选择需要新增或者编辑的数据!";
  283. return _jv;
  284. }
  285. var evaluatorInfo = await _sqlSugar.Queryable<Sys_Users>()
  286. .LeftJoin<Sys_Company>((u, c) => u.CompanyId == c.Id)
  287. .LeftJoin<Sys_Department>((u, c, d) => u.DepId == d.Id)
  288. .LeftJoin<Sys_JobPost>((u, c, d, jp) => u.JobPostId == jp.Id)
  289. .Where((u, c, d, jp) => u.IsDel == 0 && u.Id == dto.CurrUserId)
  290. .Select((u, c, d, jp) => new
  291. {
  292. UserId = u.Id,
  293. u.CompanyId,
  294. c.CompanyName,
  295. u.DepId,
  296. d.DepName,
  297. u.JobPostId,
  298. jp.JobName,
  299. UserName = u.CnName
  300. })
  301. .FirstAsync();
  302. if (evaluatorInfo == null)
  303. {
  304. _jv.Msg = $"请选择有效的考核人!";
  305. return _jv;
  306. }
  307. _ = dto.Infos.OrderBy(x => x.EvalContentOrder);
  308. var infos = _mapper.Map<List<Pm_CompanyDailyKpi>>(dto.Infos);
  309. infos.ForEach(x => {
  310. if (x.Id <= 0)
  311. {
  312. x.CreateUserId = dto.CurrUserId;
  313. }
  314. x.LastUpdateUserId = dto.CurrUserId;
  315. x.LastUpdateTime = DateTime.Now;
  316. });
  317. try
  318. {
  319. var storage = _sqlSugar.Storageable(infos)
  320. .WhereColumns(x => x.Id) // 根据Id判断
  321. .ToStorage();
  322. await storage.AsInsertable.ExecuteCommandAsync();
  323. await storage.AsUpdateable.ExecuteCommandAsync();
  324. _jv.Code = StatusCodes.Status200OK;
  325. //_jv.Data = infos;
  326. _jv.Msg = $"操作成功";
  327. return _jv;
  328. }
  329. catch (Exception ex)
  330. {
  331. _jv.Msg = $"操作失败!Ex:{ex.Message}";
  332. return _jv;
  333. }
  334. }
  335. }
  336. }