CompanyDailyKpiRepository.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  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. /// <returns></returns>
  188. public async Task<JsonView> PlanningInitAsync()
  189. {
  190. var result = await KpiDepartmentsAndJobs("策划部");
  191. if (!result.Success)
  192. {
  193. _jv.Msg = result.Message;
  194. return _jv;
  195. }
  196. var userInfos = await _sqlSugar.Queryable<Sys_Users>()
  197. .LeftJoin<Sys_JobPost>((u, jp) => u.JobPostId == jp.Id)
  198. .LeftJoin<Sys_Department>((u, jp, d) => u.DepId == d.Id)
  199. .LeftJoin<Sys_Company>((u, jp, d, c) => u.CompanyId == c.Id)
  200. .Where((u, jp, d, c) => u.IsDel == 0)
  201. .Select((u, jp, d, c) => new
  202. {
  203. UserId = u.Id,
  204. u.CompanyId,
  205. c.CompanyName,
  206. u.DepId,
  207. d.DepName,
  208. u.JobPostId,
  209. jp.JobName,
  210. UserName = u.CnName
  211. })
  212. .OrderBy(u => u.CompanyId)
  213. .ToListAsync();
  214. var jobNames = result.DepartmentsWithJobs
  215. .SelectMany(x => x.JobNames)
  216. .ToList();
  217. var view = userInfos.Where(x => jobNames.Contains(x.JobName))
  218. .Select(x => new KpiTempUserInfo() { Id = x.UserId, DepName = x.DepName, JobName = x.JobName, Name = x.UserName })
  219. .ToList();
  220. _jv.Code = StatusCodes.Status200OK;
  221. _jv.Data = view;
  222. _jv.Msg = $"操作成功";
  223. return _jv;
  224. }
  225. /// <summary>
  226. /// 考勤详情
  227. /// </summary>
  228. /// <param name="month">月份 2025-12</param>
  229. /// <param name="evaluator">考核人</param>
  230. /// <returns></returns>
  231. public async Task<JsonView> InfoAsync(string month, int evaluator)
  232. {
  233. if (string.IsNullOrEmpty(month))
  234. {
  235. _jv.Msg = $"请选择月份!";
  236. return _jv;
  237. }
  238. // 格式:2025-11
  239. const string pattern = @"^(\d{4})-(0[1-9]|1[0-2])$";
  240. if (!Regex.IsMatch(month, pattern))
  241. {
  242. _jv.Msg = $"请选择正确的月份!格式:2025-12";
  243. return _jv;
  244. }
  245. var evaluatorInfo = await _sqlSugar.Queryable<Sys_Users>()
  246. .LeftJoin<Sys_Company>((u, c) => u.CompanyId == c.Id)
  247. .LeftJoin<Sys_Department>((u, c, d) => u.DepId == d.Id)
  248. .LeftJoin<Sys_JobPost>((u, c, d, jp) => u.JobPostId == jp.Id)
  249. .Where((u, c, d, jp) => u.IsDel == 0 && u.Id == evaluator)
  250. .Select((u, c, d, jp) => new
  251. {
  252. UserId = u.Id,
  253. u.CompanyId,
  254. c.CompanyName,
  255. u.DepId,
  256. d.DepName,
  257. u.JobPostId,
  258. jp.JobName,
  259. UserName = u.CnName
  260. })
  261. .FirstAsync();
  262. if (evaluatorInfo == null)
  263. {
  264. _jv.Msg = $"请选择有效的考核人!";
  265. return _jv;
  266. }
  267. var info = await _sqlSugar
  268. .Queryable<Pm_CompanyDailyKpi>()
  269. .LeftJoin<Sys_Users>((cdk, u) => cdk.Evaluator == u.Id)
  270. .Where((cdk, u) => cdk.IsDel == 0 && cdk.Evaluator == evaluator && cdk.Month.Equals(month))
  271. .Select((cdk, u) => new CompanyDailyKpiView()
  272. {
  273. Id = cdk.Id,
  274. Month = cdk.Month,
  275. EvalContentOrder = cdk.EvalContentOrder,
  276. Evaluator = cdk.Evaluator,
  277. EvaluatorName = u.CnName,
  278. EvalContent = cdk.EvalContent,
  279. IsMistake = cdk.IsMistake,
  280. MistakeReason = cdk.MistakeReason
  281. })
  282. .ToListAsync();
  283. if (!info.Any())
  284. {
  285. var tempInfo = await KpiTemplateDataAsync(evaluatorInfo.DepName, evaluatorInfo.JobName);
  286. if (!tempInfo.isSuccess)
  287. {
  288. _jv.Msg = $"{evaluatorInfo.DepName}-{evaluatorInfo.JobName},未配置日常绩效考核。";
  289. return _jv;
  290. }
  291. tempInfo.data.ForEach(t =>
  292. {
  293. t.Contents.ForEach(c =>
  294. {
  295. _ = int.TryParse(c.KpiSort, out int no);
  296. info.Add(new CompanyDailyKpiView()
  297. {
  298. Month = month,
  299. EvalContentOrder = no,
  300. Evaluator = evaluatorInfo.UserId,
  301. EvaluatorName = evaluatorInfo.UserName,
  302. EvalContent = c.KpiContent,
  303. IsMistake = false,
  304. MistakeReason = string.Empty
  305. });
  306. });
  307. });
  308. }
  309. _jv.Code = StatusCodes.Status200OK;
  310. _jv.Data = info;
  311. _jv.Msg = $"操作成功";
  312. return _jv;
  313. }
  314. /// <summary>
  315. /// kpi Save
  316. /// </summary>
  317. /// <param name="dto">请求类</param>
  318. /// <returns></returns>
  319. public async Task<JsonView> SaveAsync(CompanyDailyKpiSaveDto dto)
  320. {
  321. if (dto.Infos == null || !dto.Infos.Any())
  322. {
  323. _jv.Msg = $"请选择需要新增或者编辑的数据!";
  324. return _jv;
  325. }
  326. var evaluatorInfo = await _sqlSugar.Queryable<Sys_Users>()
  327. .LeftJoin<Sys_Company>((u, c) => u.CompanyId == c.Id)
  328. .LeftJoin<Sys_Department>((u, c, d) => u.DepId == d.Id)
  329. .LeftJoin<Sys_JobPost>((u, c, d, jp) => u.JobPostId == jp.Id)
  330. .Where((u, c, d, jp) => u.IsDel == 0 && u.Id == dto.CurrUserId)
  331. .Select((u, c, d, jp) => new
  332. {
  333. UserId = u.Id,
  334. u.CompanyId,
  335. c.CompanyName,
  336. u.DepId,
  337. d.DepName,
  338. u.JobPostId,
  339. jp.JobName,
  340. UserName = u.CnName
  341. })
  342. .FirstAsync();
  343. if (evaluatorInfo == null)
  344. {
  345. _jv.Msg = $"请选择有效的考核人!";
  346. return _jv;
  347. }
  348. _ = dto.Infos.OrderBy(x => x.EvalContentOrder);
  349. var infos = _mapper.Map<List<Pm_CompanyDailyKpi>>(dto.Infos);
  350. infos.ForEach(x => {
  351. if (x.Id <= 0)
  352. {
  353. x.CreateUserId = dto.CurrUserId;
  354. }
  355. x.LastUpdateUserId = dto.CurrUserId;
  356. x.LastUpdateTime = DateTime.Now;
  357. });
  358. try
  359. {
  360. var storage = _sqlSugar.Storageable(infos)
  361. .WhereColumns(x => x.Id) // 根据Id判断
  362. .ToStorage();
  363. await storage.AsInsertable.ExecuteCommandAsync();
  364. await storage.AsUpdateable.ExecuteCommandAsync();
  365. _jv.Code = StatusCodes.Status200OK;
  366. //_jv.Data = infos;
  367. _jv.Msg = $"操作成功";
  368. return _jv;
  369. }
  370. catch (Exception ex)
  371. {
  372. _jv.Msg = $"操作失败!Ex:{ex.Message}";
  373. return _jv;
  374. }
  375. }
  376. }
  377. }