CompanyDailyKpiRepository.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  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(List<string> depNames)
  88. {
  89. if (depNames?.Any() != true) return (false, "请求部门为空", new List<string>(), new List<KpiTempDepartmentInfo>());
  90. var departments = new List<string>();
  91. var departmentsWithJobs = new List<KpiTempDepartmentInfo>();
  92. var config = await _sqlSugar.Queryable<Sys_SetDataType>()
  93. .Where(x => x.IsDel == 0 && x.Id == 132)
  94. .FirstAsync();
  95. if (config == null) return (false, "配置为空", departments, departmentsWithJobs);
  96. var typeIds = JsonSerializer.Deserialize<List<int>>(config.Remark ?? "[]");
  97. if (typeIds == null || typeIds.Count < 1)
  98. return (false, "类型ID为空", departments, departmentsWithJobs);
  99. var names = await _sqlSugar.Queryable<Sys_SetData>()
  100. .LeftJoin<Sys_SetDataType>((sd, sdt) => sd.STid == sdt.Id)
  101. .Where((sd, sdt) => sd.IsDel == 0 && typeIds.Contains(sd.STid))
  102. .Select((sd, sdt) => sdt.Name)
  103. .Distinct()
  104. .ToListAsync();
  105. // 解析部门和岗位
  106. var deptJobList = names
  107. .Select(name =>
  108. {
  109. var clean = name.Split('(')[0].Trim();
  110. string dept, job;
  111. if (clean.Contains(" - "))
  112. {
  113. var parts = clean.Split(" - ");
  114. dept = parts[0].EndsWith("部") ? parts[0] : parts[0] + "部";
  115. job = parts.Length > 1 ? parts[1] : "通用";
  116. }
  117. else if (clean.Contains("-"))
  118. {
  119. var parts = clean.Split('-');
  120. dept = parts[0].EndsWith("部") ? parts[0] : parts[0] + "部";
  121. job = parts.Length > 1 ? parts[1] : "通用";
  122. }
  123. else
  124. {
  125. dept = clean.EndsWith("部") ? clean : "未分类";
  126. job = clean.EndsWith("部") ? "通用" : clean;
  127. }
  128. return new { Department = dept, Job = job };
  129. })
  130. .Where(x => x.Department != "未分类")
  131. .WhereIF(depNames.Any(), x => depNames.Contains(x.Department))
  132. .GroupBy(x => x.Department)
  133. .Select(g => new KpiTempDepartmentInfo()
  134. {
  135. DepName = g.Key,
  136. JobNames = g.Select(x => x.Job).Distinct().Where(j => j != "通用").OrderBy(j => j).ToList()
  137. })
  138. .OrderBy(g => g.DepName)
  139. .ToList();
  140. departmentsWithJobs = deptJobList;
  141. departments = deptJobList.Select(g => g.DepName).ToList();
  142. return (true, "操作成功", departments, departmentsWithJobs);
  143. }
  144. /// <summary>
  145. /// 基础数据
  146. /// </summary>
  147. /// <returns></returns>
  148. public async Task<JsonView> InitAsync()
  149. {
  150. var depNames = new List<string> { "财务部", "市场部" };
  151. var result = await KpiDepartmentsAndJobs(depNames);
  152. if (!result.Success)
  153. {
  154. _jv.Msg = result.Message;
  155. return _jv;
  156. }
  157. var userInfos = await _sqlSugar.Queryable<Sys_Users>()
  158. .LeftJoin<Sys_JobPost>((u, jp) => u.JobPostId == jp.Id)
  159. .LeftJoin<Sys_Department>((u, jp, d) => u.DepId == d.Id)
  160. .LeftJoin<Sys_Company>((u, jp, d, c) => u.CompanyId == c.Id)
  161. .Where((u, jp, d, c) => u.IsDel == 0)
  162. .Select((u, jp, d, c) => new
  163. {
  164. UserId = u.Id,
  165. u.CompanyId,
  166. c.CompanyName,
  167. u.DepId,
  168. d.DepName,
  169. u.JobPostId,
  170. jp.JobName,
  171. UserName = u.CnName
  172. })
  173. .OrderBy(u => u.CompanyId)
  174. .ToListAsync();
  175. var jobNames = result.DepartmentsWithJobs
  176. .SelectMany(x => x.JobNames)
  177. .ToList();
  178. var view = userInfos.Where(x => jobNames.Contains(x.JobName))
  179. .Select(x => new KpiTempUserInfo() { Id = x.UserId, DepName = x.DepName, JobName = x.JobName, Name = x.UserName })
  180. .ToList();
  181. _jv.Code = StatusCodes.Status200OK;
  182. _jv.Data = view;
  183. _jv.Msg = $"操作成功";
  184. return _jv;
  185. }
  186. /// <summary>
  187. /// 策划部基础数据
  188. /// </summary>
  189. /// <returns></returns>
  190. public async Task<JsonView> PlanningInitAsync()
  191. {
  192. var depNames = new List<string> { "策划部" };
  193. var result = await KpiDepartmentsAndJobs(depNames);
  194. if (!result.Success)
  195. {
  196. _jv.Msg = result.Message;
  197. return _jv;
  198. }
  199. var userInfos = await _sqlSugar.Queryable<Sys_Users>()
  200. .LeftJoin<Sys_JobPost>((u, jp) => u.JobPostId == jp.Id)
  201. .LeftJoin<Sys_Department>((u, jp, d) => u.DepId == d.Id)
  202. .LeftJoin<Sys_Company>((u, jp, d, c) => u.CompanyId == c.Id)
  203. .Where((u, jp, d, c) => u.IsDel == 0)
  204. .Select((u, jp, d, c) => new
  205. {
  206. UserId = u.Id,
  207. u.CompanyId,
  208. c.CompanyName,
  209. u.DepId,
  210. d.DepName,
  211. u.JobPostId,
  212. jp.JobName,
  213. UserName = u.CnName
  214. })
  215. .OrderBy(u => u.CompanyId)
  216. .ToListAsync();
  217. var jobNames = result.DepartmentsWithJobs
  218. .SelectMany(x => x.JobNames)
  219. .ToList();
  220. var view = userInfos.Where(x => jobNames.Contains(x.JobName))
  221. .Select(x => new KpiTempUserInfo() { Id = x.UserId, DepName = x.DepName, JobName = x.JobName, Name = x.UserName })
  222. .ToList();
  223. _jv.Code = StatusCodes.Status200OK;
  224. _jv.Data = view;
  225. _jv.Msg = $"操作成功";
  226. return _jv;
  227. }
  228. /// <summary>
  229. /// 考勤详情
  230. /// </summary>
  231. /// <param name="month">月份 2025-12</param>
  232. /// <param name="evaluator">考核人</param>
  233. /// <returns></returns>
  234. public async Task<JsonView> InfoAsync(string month, int evaluator)
  235. {
  236. if (string.IsNullOrEmpty(month))
  237. {
  238. _jv.Msg = $"请选择月份!";
  239. return _jv;
  240. }
  241. // 格式:2025-11
  242. const string pattern = @"^(\d{4})-(0[1-9]|1[0-2])$";
  243. if (!Regex.IsMatch(month, pattern))
  244. {
  245. _jv.Msg = $"请选择正确的月份!格式:2025-12";
  246. return _jv;
  247. }
  248. var evaluatorInfo = await _sqlSugar.Queryable<Sys_Users>()
  249. .LeftJoin<Sys_Company>((u, c) => u.CompanyId == c.Id)
  250. .LeftJoin<Sys_Department>((u, c, d) => u.DepId == d.Id)
  251. .LeftJoin<Sys_JobPost>((u, c, d, jp) => u.JobPostId == jp.Id)
  252. .Where((u, c, d, jp) => u.IsDel == 0 && u.Id == evaluator)
  253. .Select((u, c, d, jp) => new
  254. {
  255. UserId = u.Id,
  256. u.CompanyId,
  257. c.CompanyName,
  258. u.DepId,
  259. d.DepName,
  260. u.JobPostId,
  261. jp.JobName,
  262. UserName = u.CnName
  263. })
  264. .FirstAsync();
  265. if (evaluatorInfo == null)
  266. {
  267. _jv.Msg = $"请选择有效的考核人!";
  268. return _jv;
  269. }
  270. var info = await _sqlSugar
  271. .Queryable<Pm_CompanyDailyKpi>()
  272. .LeftJoin<Sys_Users>((cdk, u) => cdk.Evaluator == u.Id)
  273. .Where((cdk, u) => cdk.IsDel == 0 && cdk.Evaluator == evaluator && cdk.Month.Equals(month))
  274. .Select((cdk, u) => new CompanyDailyKpiView()
  275. {
  276. Id = cdk.Id,
  277. Month = cdk.Month,
  278. EvalContentOrder = cdk.EvalContentOrder,
  279. Evaluator = cdk.Evaluator,
  280. EvaluatorName = u.CnName,
  281. EvalContent = cdk.EvalContent,
  282. IsMistake = cdk.IsMistake,
  283. MistakeReason = cdk.MistakeReason
  284. })
  285. .ToListAsync();
  286. if (!info.Any())
  287. {
  288. var tempInfo = await KpiTemplateDataAsync(evaluatorInfo.DepName, evaluatorInfo.JobName);
  289. if (!tempInfo.isSuccess)
  290. {
  291. _jv.Msg = $"{evaluatorInfo.DepName}-{evaluatorInfo.JobName},未配置日常绩效考核。";
  292. return _jv;
  293. }
  294. tempInfo.data.ForEach(t =>
  295. {
  296. t.Contents.ForEach(c =>
  297. {
  298. _ = int.TryParse(c.KpiSort, out int no);
  299. info.Add(new CompanyDailyKpiView()
  300. {
  301. Month = month,
  302. EvalContentOrder = no,
  303. Evaluator = evaluatorInfo.UserId,
  304. EvaluatorName = evaluatorInfo.UserName,
  305. EvalContent = c.KpiContent,
  306. IsMistake = false,
  307. MistakeReason = string.Empty
  308. });
  309. });
  310. });
  311. }
  312. _jv.Code = StatusCodes.Status200OK;
  313. _jv.Data = info;
  314. _jv.Msg = $"操作成功";
  315. return _jv;
  316. }
  317. /// <summary>
  318. /// kpi Save
  319. /// </summary>
  320. /// <param name="dto">请求类</param>
  321. /// <returns></returns>
  322. public async Task<JsonView> SaveAsync(CompanyDailyKpiSaveDto dto)
  323. {
  324. if (dto.Infos == null || !dto.Infos.Any())
  325. {
  326. _jv.Msg = $"请选择需要新增或者编辑的数据!";
  327. return _jv;
  328. }
  329. var evaluatorInfo = await _sqlSugar.Queryable<Sys_Users>()
  330. .LeftJoin<Sys_Company>((u, c) => u.CompanyId == c.Id)
  331. .LeftJoin<Sys_Department>((u, c, d) => u.DepId == d.Id)
  332. .LeftJoin<Sys_JobPost>((u, c, d, jp) => u.JobPostId == jp.Id)
  333. .Where((u, c, d, jp) => u.IsDel == 0 && u.Id == dto.CurrUserId)
  334. .Select((u, c, d, jp) => new
  335. {
  336. UserId = u.Id,
  337. u.CompanyId,
  338. c.CompanyName,
  339. u.DepId,
  340. d.DepName,
  341. u.JobPostId,
  342. jp.JobName,
  343. UserName = u.CnName
  344. })
  345. .FirstAsync();
  346. if (evaluatorInfo == null)
  347. {
  348. _jv.Msg = $"请选择有效的考核人!";
  349. return _jv;
  350. }
  351. _ = dto.Infos.OrderBy(x => x.EvalContentOrder);
  352. var infos = _mapper.Map<List<Pm_CompanyDailyKpi>>(dto.Infos);
  353. infos.ForEach(x => {
  354. if (x.Id <= 0)
  355. {
  356. x.CreateUserId = dto.CurrUserId;
  357. }
  358. x.LastUpdateUserId = dto.CurrUserId;
  359. x.LastUpdateTime = DateTime.Now;
  360. });
  361. try
  362. {
  363. var storage = _sqlSugar.Storageable(infos)
  364. .WhereColumns(x => x.Id) // 根据Id判断
  365. .ToStorage();
  366. await storage.AsInsertable.ExecuteCommandAsync();
  367. await storage.AsUpdateable.ExecuteCommandAsync();
  368. _jv.Code = StatusCodes.Status200OK;
  369. //_jv.Data = infos;
  370. _jv.Msg = $"操作成功";
  371. return _jv;
  372. }
  373. catch (Exception ex)
  374. {
  375. _jv.Msg = $"操作失败!Ex:{ex.Message}";
  376. return _jv;
  377. }
  378. }
  379. }
  380. }