CompanyDailyKpiRepository.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  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(int currUserId)
  149. {
  150. var currUserInfo = await _sqlSugar.Queryable<Sys_Users>()
  151. .LeftJoin<Sys_JobPost>((u, jp) => u.JobPostId == jp.Id)
  152. .LeftJoin<Sys_Department>((u, jp, d) => u.DepId == d.Id)
  153. .LeftJoin<Sys_Company>((u, jp, d, c) => u.CompanyId == c.Id)
  154. .Where((u, jp, d, c) => u.IsDel == 0 && u.Id == currUserId)
  155. .Select((u, jp, d, c) => new
  156. {
  157. UserId = u.Id,
  158. u.CompanyId,
  159. c.CompanyName,
  160. u.DepId,
  161. d.DepName,
  162. u.JobPostId,
  163. jp.JobName,
  164. UserName = u.CnName
  165. })
  166. .OrderBy(u => u.CompanyId)
  167. .FirstAsync();
  168. if (currUserInfo == null)
  169. {
  170. _jv.Msg = $"当前登录用户信息不存在。";
  171. return _jv;
  172. }
  173. //验证是否是经理主管 是 可操作保存;是被考核人只能查看自己
  174. var isSave = !string.IsNullOrEmpty(currUserInfo.JobName) &&
  175. (currUserInfo.JobName.Contains("总监") ||
  176. currUserInfo.JobName.Contains("经理") ||
  177. currUserInfo.JobName.Contains("主管"));
  178. var depNames = new List<string> { currUserInfo?.DepName ?? "" };
  179. var result = await KpiDepartmentsAndJobs(depNames);
  180. if (!result.Success)
  181. {
  182. _jv.Msg = result.Message;
  183. return _jv;
  184. }
  185. var userInfos = await _sqlSugar.Queryable<Sys_Users>()
  186. .LeftJoin<Sys_JobPost>((u, jp) => u.JobPostId == jp.Id)
  187. .LeftJoin<Sys_Department>((u, jp, d) => u.DepId == d.Id)
  188. .LeftJoin<Sys_Company>((u, jp, d, c) => u.CompanyId == c.Id)
  189. .Where((u, jp, d, c) => u.IsDel == 0)
  190. .Select((u, jp, d, c) => new
  191. {
  192. UserId = u.Id,
  193. u.CompanyId,
  194. c.CompanyName,
  195. u.DepId,
  196. d.DepName,
  197. u.JobPostId,
  198. jp.JobName,
  199. UserName = u.CnName
  200. })
  201. .OrderBy(u => u.CompanyId)
  202. .ToListAsync();
  203. var jobNames = result.DepartmentsWithJobs
  204. .SelectMany(x => x.JobNames)
  205. .ToList();
  206. var view = new
  207. {
  208. isOp = isSave,
  209. userInfos = userInfos.Where(x => jobNames.Contains(x.JobName))
  210. .WhereIF(!isSave, x=> x.UserId == currUserId)
  211. .Select(x => new KpiTempUserInfo() { Id = x.UserId, DepName = x.DepName, JobName = x.JobName, Name = x.UserName })
  212. .ToList()
  213. };
  214. _jv.Code = StatusCodes.Status200OK;
  215. _jv.Data = view;
  216. _jv.Msg = $"操作成功";
  217. return _jv;
  218. }
  219. /// <summary>
  220. /// 策划部基础数据
  221. /// </summary>
  222. /// <returns></returns>
  223. public async Task<JsonView> PlanningInitAsync()
  224. {
  225. var depNames = new List<string> { "策划部" };
  226. var result = await KpiDepartmentsAndJobs(depNames);
  227. if (!result.Success)
  228. {
  229. _jv.Msg = result.Message;
  230. return _jv;
  231. }
  232. var userInfos = await _sqlSugar.Queryable<Sys_Users>()
  233. .LeftJoin<Sys_JobPost>((u, jp) => u.JobPostId == jp.Id)
  234. .LeftJoin<Sys_Department>((u, jp, d) => u.DepId == d.Id)
  235. .LeftJoin<Sys_Company>((u, jp, d, c) => u.CompanyId == c.Id)
  236. .Where((u, jp, d, c) => u.IsDel == 0)
  237. .Select((u, jp, d, c) => new
  238. {
  239. UserId = u.Id,
  240. u.CompanyId,
  241. c.CompanyName,
  242. u.DepId,
  243. d.DepName,
  244. u.JobPostId,
  245. jp.JobName,
  246. UserName = u.CnName
  247. })
  248. .OrderBy(u => u.CompanyId)
  249. .ToListAsync();
  250. var jobNames = result.DepartmentsWithJobs
  251. .SelectMany(x => x.JobNames)
  252. .ToList();
  253. var view = userInfos.Where(x => jobNames.Contains(x.JobName))
  254. .Select(x => new KpiTempUserInfo() { Id = x.UserId, DepName = x.DepName, JobName = x.JobName, Name = x.UserName })
  255. .ToList();
  256. _jv.Code = StatusCodes.Status200OK;
  257. _jv.Data = view;
  258. _jv.Msg = $"操作成功";
  259. return _jv;
  260. }
  261. /// <summary>
  262. /// 考勤详情
  263. /// </summary>
  264. /// <param name="month">月份 2025-12</param>
  265. /// <param name="evaluator">考核人</param>
  266. /// <returns></returns>
  267. public async Task<JsonView> InfoAsync(string month, int evaluator)
  268. {
  269. if (string.IsNullOrEmpty(month))
  270. {
  271. _jv.Msg = $"请选择月份!";
  272. return _jv;
  273. }
  274. // 格式:2025-11
  275. const string pattern = @"^(\d{4})-(0[1-9]|1[0-2])$";
  276. if (!Regex.IsMatch(month, pattern))
  277. {
  278. _jv.Msg = $"请选择正确的月份!格式:2025-12";
  279. return _jv;
  280. }
  281. var evaluatorInfo = await _sqlSugar.Queryable<Sys_Users>()
  282. .LeftJoin<Sys_Company>((u, c) => u.CompanyId == c.Id)
  283. .LeftJoin<Sys_Department>((u, c, d) => u.DepId == d.Id)
  284. .LeftJoin<Sys_JobPost>((u, c, d, jp) => u.JobPostId == jp.Id)
  285. .Where((u, c, d, jp) => u.IsDel == 0 && u.Id == evaluator)
  286. .Select((u, c, d, jp) => new
  287. {
  288. UserId = u.Id,
  289. u.CompanyId,
  290. c.CompanyName,
  291. u.DepId,
  292. d.DepName,
  293. u.JobPostId,
  294. jp.JobName,
  295. UserName = u.CnName
  296. })
  297. .FirstAsync();
  298. if (evaluatorInfo == null)
  299. {
  300. _jv.Msg = $"请选择有效的考核人!";
  301. return _jv;
  302. }
  303. var info = await _sqlSugar
  304. .Queryable<Pm_CompanyDailyKpi>()
  305. .LeftJoin<Sys_Users>((cdk, u) => cdk.Evaluator == u.Id)
  306. .Where((cdk, u) => cdk.IsDel == 0 && cdk.Evaluator == evaluator && cdk.Month.Equals(month))
  307. .Select((cdk, u) => new CompanyDailyKpiView()
  308. {
  309. Id = cdk.Id,
  310. Month = cdk.Month,
  311. EvalContentOrder = cdk.EvalContentOrder,
  312. Evaluator = cdk.Evaluator,
  313. EvaluatorName = u.CnName,
  314. EvalContent = cdk.EvalContent,
  315. IsMistake = cdk.IsMistake,
  316. MistakeReason = cdk.MistakeReason
  317. })
  318. .ToListAsync();
  319. if (!info.Any())
  320. {
  321. var tempInfo = await KpiTemplateDataAsync(evaluatorInfo.DepName, evaluatorInfo.JobName);
  322. if (!tempInfo.isSuccess)
  323. {
  324. _jv.Msg = $"{evaluatorInfo.DepName}-{evaluatorInfo.JobName},未配置日常绩效考核。";
  325. return _jv;
  326. }
  327. tempInfo.data.ForEach(t =>
  328. {
  329. t.Contents.ForEach(c =>
  330. {
  331. _ = int.TryParse(c.KpiSort, out int no);
  332. info.Add(new CompanyDailyKpiView()
  333. {
  334. Month = month,
  335. EvalContentOrder = no,
  336. Evaluator = evaluatorInfo.UserId,
  337. EvaluatorName = evaluatorInfo.UserName,
  338. EvalContent = c.KpiContent,
  339. IsMistake = false,
  340. MistakeReason = string.Empty
  341. });
  342. });
  343. });
  344. }
  345. _jv.Code = StatusCodes.Status200OK;
  346. _jv.Data = info;
  347. _jv.Msg = $"操作成功";
  348. return _jv;
  349. }
  350. /// <summary>
  351. /// kpi Save
  352. /// </summary>
  353. /// <param name="dto">请求类</param>
  354. /// <returns></returns>
  355. public async Task<JsonView> SaveAsync(CompanyDailyKpiSaveDto dto)
  356. {
  357. if (dto.Infos == null || !dto.Infos.Any())
  358. {
  359. _jv.Msg = $"请选择需要新增或者编辑的数据!";
  360. return _jv;
  361. }
  362. var evaluatorInfo = await _sqlSugar.Queryable<Sys_Users>()
  363. .LeftJoin<Sys_Company>((u, c) => u.CompanyId == c.Id)
  364. .LeftJoin<Sys_Department>((u, c, d) => u.DepId == d.Id)
  365. .LeftJoin<Sys_JobPost>((u, c, d, jp) => u.JobPostId == jp.Id)
  366. .Where((u, c, d, jp) => u.IsDel == 0 && u.Id == dto.CurrUserId)
  367. .Select((u, c, d, jp) => new
  368. {
  369. UserId = u.Id,
  370. u.CompanyId,
  371. c.CompanyName,
  372. u.DepId,
  373. d.DepName,
  374. u.JobPostId,
  375. jp.JobName,
  376. UserName = u.CnName
  377. })
  378. .FirstAsync();
  379. if (evaluatorInfo == null)
  380. {
  381. _jv.Msg = $"请选择有效的考核人!";
  382. return _jv;
  383. }
  384. _ = dto.Infos.OrderBy(x => x.EvalContentOrder);
  385. var infos = _mapper.Map<List<Pm_CompanyDailyKpi>>(dto.Infos);
  386. infos.ForEach(x => {
  387. if (x.Id <= 0)
  388. {
  389. x.CreateUserId = dto.CurrUserId;
  390. }
  391. x.LastUpdateUserId = dto.CurrUserId;
  392. x.LastUpdateTime = DateTime.Now;
  393. });
  394. try
  395. {
  396. var storage = _sqlSugar.Storageable(infos)
  397. .WhereColumns(x => x.Id) // 根据Id判断
  398. .ToStorage();
  399. await storage.AsInsertable.ExecuteCommandAsync();
  400. await storage.AsUpdateable.ExecuteCommandAsync();
  401. _jv.Code = StatusCodes.Status200OK;
  402. //_jv.Data = infos;
  403. _jv.Msg = $"操作成功";
  404. return _jv;
  405. }
  406. catch (Exception ex)
  407. {
  408. _jv.Msg = $"操作失败!Ex:{ex.Message}";
  409. return _jv;
  410. }
  411. }
  412. }
  413. }