TaskController.cs 43 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.Globalization;
  5. using System.Linq;
  6. using OASystem.Domain.Dtos.Task;
  7. using OASystem.Domain.Entities.Groups;
  8. using OASystem.Domain.Entities.WorkOrder;
  9. using OASystem.Domain.ViewModels.JuHeExchangeRate;
  10. using OASystem.API.OAMethodLib.QiYeWeChatAPI.AppNotice;
  11. using Dm.util;
  12. using OASystem.API.OAMethodLib.DoubaoAPI;
  13. using OASystem.API.OAMethodLib.QiYeWeChatAPI;
  14. using OASystem.Domain.Dtos.QiYeWeChat;
  15. namespace OASystem.API.Controllers
  16. {
  17. /// <summary>
  18. /// 部门任务相关
  19. /// </summary>
  20. //[Authorize]
  21. [Route("api/[controller]/[action]")]
  22. public class TaskController : ControllerBase
  23. {
  24. readonly SqlSugarClient _sqlsugar;
  25. readonly IDoubaoService _doubaoService;
  26. readonly IQiYeWeChatApiService _qiYeWeChatApiService;
  27. public TaskController(SqlSugarClient sqlsugar, IDoubaoService doubaoService, IQiYeWeChatApiService qiYeWeChatApiService)
  28. {
  29. _sqlsugar = sqlsugar;
  30. _doubaoService = doubaoService;
  31. _qiYeWeChatApiService = qiYeWeChatApiService;
  32. }
  33. /// <summary>
  34. /// 数据列表
  35. /// </summary>
  36. /// <param name="dto"></param>
  37. /// <returns></returns>
  38. [HttpPost]
  39. public async Task<IActionResult> GetTaskList(GetTaskListDto dto)
  40. {
  41. var jw = JsonView(false);
  42. if (dto.TypeId < 1 || dto.UserId < 1)
  43. {
  44. jw.Msg = "参数有误!";
  45. return Ok(jw);
  46. }
  47. DateTime startTime;
  48. DateTime endTime = DateTime.Now;
  49. var expr = Expressionable.Create<Task_WorkOrder>().And(x => x.IsDel == 0)
  50. .And(x => x.TypeId == dto.TypeId)
  51. .And(x => (x.CreateUserId == dto.UserId || x.AssignedUserId == dto.UserId))
  52. .AndIF(!string.IsNullOrEmpty(dto.Search_Name), x => x.Name.Contains(dto.Search_Name))
  53. // .AndIF(dto.Search_UserId > 0, x => (x.CreateUserId == dto.Search_UserId || x.AssignedUserId == dto.Search_UserId))
  54. .AndIF(DateTime.TryParse(dto.Seach_StartTime, out startTime) && DateTime.TryParse(dto.Seach_EndTime, out endTime), x => x.StartTime >= startTime && x.StartTime <= endTime);
  55. var query = _sqlsugar.Queryable<Task_WorkOrder>()
  56. .LeftJoin<Grp_DelegationInfo>((x, y) => x.GroupId == y.Id && y.IsDel == 0)
  57. .LeftJoin<Sys_SetData>((x, y, z) => x.ForeignOptionId == z.Id && z.IsDel == 0)
  58. .LeftJoin<Sys_Users>((x, y, z, w) => x.AssignedUserId == w.Id && w.IsDel == 0)
  59. .Where(expr.ToExpression());
  60. var count = query.Count();
  61. var temp = query.Select((x, y, z, w) => new
  62. {
  63. Id = x.Id,
  64. TaskName = x.Name,
  65. StartTime = x.StartTime.ToString("yyyy-MM-dd HH:mm"),
  66. Assignee = w.CnName,
  67. ExternalOption = z.Name,
  68. Team = y.TeamName,
  69. Active = x.ProgressId,
  70. CreateUserid = x.CreateUserId
  71. });
  72. var list = temp
  73. .OrderByIF(dto.isLv == 1,
  74. x => SqlFunc.IsNull(
  75. SqlFunc.Subqueryable<Task_WorkTask>()
  76. .Where(twt => twt.IsDel == 0 && twt.IsUrgent && twt.WorkOrderId == x.Id)
  77. .Min(twt => twt.StartTime),
  78. DateTime.MaxValue // 把 null 替换成极大值
  79. )
  80. )
  81. .OrderBy(x => x.StartTime)
  82. .ToPageList(dto.PageIndex, dto.PageSize);
  83. var data = list.Select(x =>
  84. {
  85. var taskArr = _sqlsugar.Queryable<Task_WorkTask>()
  86. .Where(y => y.IsDel == 0 && y.WorkOrderId == x.Id)
  87. .ToList();
  88. var ExtraStep = taskArr.Where(x => x.IsExtraTask == true);
  89. var historyStep = _sqlsugar.Queryable<Task_WorkTaskReceipt>()
  90. .Where(y => y.IsDel == 0 && y.WorkOrderId == x.Id
  91. && ExtraStep.Select(z => z.Id).Contains(y.WorkTaskId)
  92. && y.IsApproved == 1)
  93. .Select(x => x.WorkTaskId)
  94. .ToList();
  95. return new WorkOrdeView
  96. {
  97. Id = x.Id,
  98. TaskName = x.TaskName,
  99. StartTime = x.StartTime,
  100. Assignee = x.Assignee,
  101. ExternalOption = x.ExternalOption,
  102. Team = string.IsNullOrWhiteSpace(x.Team) ? "未选择" : x.Team,
  103. StepInfo = new StepInfo
  104. {
  105. Active = x.Active,
  106. ExtraActive = 1,
  107. Steps = taskArr.Where(x => x.IsExtraTask == false)
  108. .OrderBy(x => x.Sort)
  109. .Select(x => new Steps
  110. {
  111. Id = x.Id,
  112. Name = x.Name,
  113. }).ToList(),
  114. ExtraStep = ExtraStep
  115. .OrderBy(x => x.Sort)
  116. .Select(x => new ExtraSteps
  117. {
  118. Id = x.Id,
  119. Name = x.Name,
  120. Approved = historyStep.Contains(x.Id) ? 1 : 0,
  121. }).ToList()
  122. },
  123. isReview = x.CreateUserid == dto.UserId,
  124. UrgentCount = taskArr.Where(x => x.IsUrgent).Count()
  125. };
  126. }).ToList();
  127. jw.Data = data;
  128. jw.Count = count;
  129. jw.Msg = "查询成功!";
  130. jw.Code = 200;
  131. return Ok(jw);
  132. }
  133. /// <summary>
  134. /// 根据工单ID获取任务详情
  135. /// </summary>
  136. /// <param name="dto"></param>
  137. /// <returns></returns>
  138. [HttpPost]
  139. public async Task<IActionResult> GetTaskListByTaskId(GetTaskListByTaskIdDto dto)
  140. {
  141. var jw = JsonView(false);
  142. if (dto.WorkOrderId < 1 || dto.UserId < 1)
  143. {
  144. jw.Msg = "参数有误!";
  145. return Ok(jw);
  146. }
  147. // 查询工单基本信息
  148. var query = _sqlsugar.Queryable<Task_WorkOrder>()
  149. .LeftJoin<Grp_DelegationInfo>((x, y) => x.GroupId == y.Id && y.IsDel == 0)
  150. .LeftJoin<Sys_SetData>((x, y, z) => x.ForeignOptionId == z.Id && z.IsDel == 0)
  151. .LeftJoin<Sys_Users>((x, y, z, w) => x.AssignedUserId == w.Id && w.IsDel == 0)
  152. .Where(x => x.Id == dto.WorkOrderId && x.IsDel == 0);
  153. var workOrderInfo = query.Select((x, y, z, w) => new
  154. {
  155. Id = x.Id,
  156. TaskName = x.Name,
  157. StartTime = x.StartTime.ToString("yyyy-MM-dd HH:mm"),
  158. Assignee = w.CnName,
  159. ExternalOption = z.Name,
  160. Team = y.TeamName,
  161. Active = x.ProgressId,
  162. CreateUserid = x.CreateUserId
  163. }).First();
  164. if (workOrderInfo == null)
  165. {
  166. jw.Msg = "工单不存在或已删除!";
  167. return Ok(jw);
  168. }
  169. // 查询工单的所有任务
  170. var taskArr = _sqlsugar.Queryable<Task_WorkTask>()
  171. .Where(y => y.IsDel == 0 && y.WorkOrderId == workOrderInfo.Id)
  172. .ToList();
  173. // 分离额外任务
  174. var ExtraStep = taskArr.Where(x => x.IsExtraTask == true);
  175. // 查询已审核通过的额外任务回执
  176. var historyStep = _sqlsugar.Queryable<Task_WorkTaskReceipt>()
  177. .Where(y => y.IsDel == 0 && y.WorkOrderId == workOrderInfo.Id
  178. && ExtraStep.Select(z => z.Id).Contains(y.WorkTaskId)
  179. && y.IsApproved == 1)
  180. .Select(x => x.WorkTaskId)
  181. .ToList();
  182. // 构建返回数据
  183. var data = new WorkOrdeView
  184. {
  185. Id = workOrderInfo.Id,
  186. TaskName = workOrderInfo.TaskName,
  187. StartTime = workOrderInfo.StartTime,
  188. Assignee = workOrderInfo.Assignee,
  189. ExternalOption = workOrderInfo.ExternalOption,
  190. Team = string.IsNullOrWhiteSpace(workOrderInfo.Team) ? "未选择" : workOrderInfo.Team,
  191. StepInfo = new StepInfo
  192. {
  193. Active = workOrderInfo.Active,
  194. ExtraActive = 1,
  195. Steps = taskArr.Where(x => x.IsExtraTask == false)
  196. .OrderBy(x => x.Sort)
  197. .Select(x => new Steps
  198. {
  199. Id = x.Id,
  200. Name = x.Name,
  201. }).ToList(),
  202. ExtraStep = ExtraStep
  203. .OrderBy(x => x.Sort)
  204. .Select(x => new ExtraSteps
  205. {
  206. Id = x.Id,
  207. Name = x.Name,
  208. Approved = historyStep.Contains(x.Id) ? 1 : 0,
  209. }).ToList()
  210. },
  211. isReview = workOrderInfo.CreateUserid == dto.UserId,
  212. UrgentCount = taskArr.Where(x => x.IsUrgent).Count()
  213. };
  214. jw.Data = data;
  215. jw.Msg = "查询成功!";
  216. jw.Code = 200;
  217. return Ok(jw);
  218. }
  219. /// <summary>
  220. /// 计算两个时间之间的工作小时数
  221. /// 工作时间:周一至周五,上午9:00-12:00,下午13:30-18:00
  222. /// </summary>
  223. /// <param name="startStr">开始时间 "yyyy-MM-dd HH:mm"</param>
  224. /// <param name="endStr">结束时间 "yyyy-MM-dd HH:mm"</param>
  225. /// <returns>工作小时数</returns>
  226. private double CalcWorkHoursExact(string startStr, string endStr)
  227. {
  228. DateTime start = DateTime.ParseExact(startStr, "yyyy-MM-dd HH:mm", CultureInfo.InvariantCulture);
  229. DateTime end = DateTime.ParseExact(endStr, "yyyy-MM-dd HH:mm", CultureInfo.InvariantCulture);
  230. if (end <= start) return 0;
  231. // 每天工作时间段(分钟)
  232. (int Start, int End)[] workPeriods = {
  233. (9 * 60, 12 * 60), // 上午 9:00-12:00
  234. (13 * 60 + 30, 18 * 60) // 下午 13:30-18:00
  235. };
  236. double totalMinutes = 0;
  237. DateTime current = start;
  238. while (current < end)
  239. {
  240. DayOfWeek day = current.DayOfWeek;
  241. if (day >= DayOfWeek.Monday && day <= DayOfWeek.Friday) // 工作日
  242. {
  243. foreach (var period in workPeriods)
  244. {
  245. DateTime periodStart = current.Date.AddMinutes(period.Start);
  246. DateTime periodEnd = current.Date.AddMinutes(period.End);
  247. DateTime effectiveStart = periodStart > current ? periodStart : current;
  248. DateTime effectiveEnd = periodEnd < end ? periodEnd : end;
  249. if (effectiveEnd > effectiveStart)
  250. {
  251. totalMinutes += (effectiveEnd - effectiveStart).TotalMinutes;
  252. }
  253. }
  254. }
  255. // 下一天 0 点
  256. current = current.Date.AddDays(1);
  257. }
  258. return totalMinutes / 60.0; // 转为小时
  259. }
  260. /// <summary>
  261. /// 根据开始时间和工作小时数,计算结束日期时间
  262. /// </summary>
  263. /// <param name="start">开始时间</param>
  264. /// <param name="workHours">工作小时数</param>
  265. /// <returns>结束时间</returns>
  266. private DateTime CalcEndDate(DateTime start, double workHours)
  267. {
  268. // 工作时间段(分钟)
  269. (int Start, int End)[] workPeriods = {
  270. (9 * 60, 12 * 60), // 上午 9:00-12:00
  271. (13 * 60 + 30, 18 * 60) // 下午 13:30-18:00
  272. };
  273. double remainingMinutes = workHours * 60;
  274. DateTime current = start;
  275. while (remainingMinutes > 0)
  276. {
  277. DayOfWeek day = current.DayOfWeek;
  278. if (day >= DayOfWeek.Monday && day <= DayOfWeek.Friday) // 工作日
  279. {
  280. foreach (var period in workPeriods)
  281. {
  282. DateTime periodStart = current.Date.AddMinutes(period.Start);
  283. DateTime periodEnd = current.Date.AddMinutes(period.End);
  284. // 如果当前时间超过本段结束,则跳过
  285. if (current >= periodEnd) continue;
  286. // 当前段开始时间
  287. DateTime effectiveStart = current > periodStart ? current : periodStart;
  288. double availableMinutes = (periodEnd - effectiveStart).TotalMinutes;
  289. if (availableMinutes >= remainingMinutes)
  290. {
  291. // 剩余工作时间在本段可以完成
  292. return effectiveStart.AddMinutes(remainingMinutes);
  293. }
  294. else
  295. {
  296. remainingMinutes -= availableMinutes;
  297. }
  298. }
  299. }
  300. // 下一天 0点
  301. current = current.Date.AddDays(1);
  302. current = current.Date.AddHours(0);
  303. }
  304. return current; // 理论上不会到这里
  305. }
  306. private string GetAdjustedStartTime()
  307. {
  308. // 获取当前时间
  309. DateTime now = DateTime.Now;
  310. // 获取年月日、小时和分钟
  311. int year = now.Year;
  312. int month = now.Month;
  313. int day = now.Day;
  314. int hours = now.Hour;
  315. int minutes = now.Minute;
  316. // 调整分钟数,使其能被10整除
  317. if (minutes % 10 != 0)
  318. {
  319. minutes = (int)Math.Ceiling(minutes / 10.0) * 10;
  320. // 如果分钟超过59,则进位到下一小时
  321. if (minutes > 59)
  322. {
  323. minutes = 0;
  324. hours++;
  325. // 如果小时进位到24,则日期需要增加一天
  326. if (hours >= 24)
  327. {
  328. hours = 0;
  329. // 如果小时为24,日期需要增加一天
  330. DateTime nextDay = now.AddDays(1);
  331. year = nextDay.Year;
  332. month = nextDay.Month;
  333. day = nextDay.Day;
  334. }
  335. }
  336. }
  337. // 格式化时间为 "yyyy-MM-dd HH:mm"
  338. return new DateTime(year, month, day, hours, minutes, 0).ToString("yyyy-MM-dd HH:mm");
  339. }
  340. /// <summary>
  341. /// 新增/编辑操作基本数据
  342. /// </summary>
  343. /// <param name="dto"></param>
  344. /// <returns></returns>
  345. [HttpPost]
  346. public async Task<IActionResult> TaskInit(TaskInitDto dto)
  347. {
  348. if (dto.TypeId < 1)
  349. {
  350. return Ok(JsonView(false, "参数有误!"));
  351. }
  352. var jw = JsonView(false);
  353. var groupList = _sqlsugar.Queryable<Grp_DelegationInfo>().Where(x => x.IsDel == 0).OrderByDescending(x => x.VisitStartDate).Select(x => new { x.Id, x.TeamName }).ToList();
  354. groupList.Insert(0, new { Id = -1, TeamName = "未选择" });
  355. var setdataids = new List<int>() { 122, 123 };
  356. var setData = _sqlsugar.Queryable<Sys_SetData>().Where(x => x.IsDel == 0 && setdataids.Contains(x.STid)).Select(x => new
  357. {
  358. x.Id,
  359. x.STid,
  360. x.Name
  361. }).ToList();
  362. var taskLv = setData.Where(x => x.STid == 122);
  363. var foreignLv = setData.Where(x => x.STid == 123);
  364. var users = _sqlsugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0).Select(x => new { x.Id, x.CnName }).ToList();
  365. users.Insert(0, new { Id = 0, CnName = "未选择" });
  366. var defaultTask = new ArrayList();
  367. var startTime = GetAdjustedStartTime();
  368. if (DateTime.TryParse(startTime, out DateTime thisTime))
  369. {
  370. var taskArr = _sqlsugar.Queryable<Task_WorkOrderDefault>()
  371. .Where(x => x.IsDel == 0 && x.TypeId == dto.TypeId)
  372. .OrderBy(x => x.Sort).ToList();
  373. var i = 1;
  374. foreach (var item in taskArr)
  375. {
  376. var endTime = CalcEndDate(thisTime, item.DurationHours);
  377. defaultTask.Add(new WorkTaskView
  378. {
  379. Name = item.Name,
  380. PriorityId = item.PriorityId,
  381. IsUrgent = item.IsUrgent,
  382. StartTime = thisTime,
  383. EndTime = endTime,
  384. DurationHours = item.DurationHours,
  385. IsExtraTask = false,
  386. Sort = i,
  387. Remark = item.Remark
  388. });
  389. i++;
  390. thisTime = endTime;
  391. }
  392. }
  393. var data = new
  394. {
  395. GroupList = groupList,
  396. TaskLv = taskLv,
  397. ForeignLv = foreignLv,
  398. Users = users,
  399. DefaultTask = defaultTask,
  400. startTime
  401. };
  402. jw.Data = data;
  403. jw.Code = 200;
  404. jw.Msg = "SUCCESS!";
  405. return Ok(jw);
  406. }
  407. /// <summary>
  408. /// 工单详情
  409. /// </summary>
  410. /// <param name="dto"></param>
  411. /// <returns></returns>
  412. [HttpPost]
  413. public async Task<IActionResult> TaskDetail(TaskDetailDto dto)
  414. {
  415. var jw = JsonView(false);
  416. var groupList = _sqlsugar.Queryable<Grp_DelegationInfo>().Where(x => x.IsDel == 0).OrderByDescending(x => x.VisitStartDate).Select(x => new { x.Id, x.TeamName }).ToList();
  417. groupList.Insert(0, new { Id = -1, TeamName = "未选择" });
  418. var setdataids = new List<int>() { 122, 123 };
  419. var setData = _sqlsugar.Queryable<Sys_SetData>().Where(x => x.IsDel == 0 && setdataids.Contains(x.STid)).Select(x => new
  420. {
  421. x.Id,
  422. x.STid,
  423. x.Name
  424. }).ToList();
  425. var taskLv = setData.Where(x => x.STid == 122);
  426. var foreignLv = setData.Where(x => x.STid == 123);
  427. var users = _sqlsugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0).Select(x => new { x.Id, x.CnName }).ToList();
  428. users.Insert(0, new { Id = 0, CnName = "未选择" });
  429. var workOrder = _sqlsugar.Queryable<Task_WorkOrder>().First(x => x.Id == dto.Id && x.IsDel == 0);
  430. var tasks = _sqlsugar.Queryable<Task_WorkTask>()
  431. .Where(x => x.IsDel == 0 && x.WorkOrderId == workOrder.Id)
  432. .OrderBy(x => x.Sort)
  433. .Select<WorkTaskView>()
  434. .ToList();
  435. var data = new
  436. {
  437. GroupList = groupList,
  438. TaskLv = taskLv,
  439. ForeignLv = foreignLv,
  440. Users = users,
  441. workOrder = new
  442. {
  443. workOrder.Id,
  444. workOrder.Name,
  445. workOrder.Remark,
  446. workOrder.AssignedUserId,
  447. workOrder.ForeignOptionId,
  448. workOrder.GroupId,
  449. workOrder.StartTime,
  450. Action = workOrder.ProgressId
  451. },
  452. mainTask = tasks.Where(x => x.IsExtraTask == false).ToList(),
  453. extraTask = tasks.Where(x => x.IsExtraTask == true).ToList()
  454. };
  455. jw.Data = data;
  456. jw.Code = 200;
  457. jw.Msg = "SUCCESS!";
  458. return Ok(jw);
  459. }
  460. /// <summary>
  461. /// 新增/编辑操作
  462. /// </summary>
  463. /// <param name="dto"></param>
  464. /// <returns></returns>
  465. [HttpPost]
  466. public async Task<IActionResult> TaskOperation(TaskOperationDto dto)
  467. {
  468. var jw = JsonView(false);
  469. jw.Data = "";
  470. if (dto.TypeId < 1 || dto.AssignedUserId < 1)
  471. {
  472. return Ok(JsonView(false, "参数有误!"));
  473. }
  474. if (!DateTime.TryParse(dto.StartTime, out DateTime startTime))
  475. {
  476. return Ok(JsonView(false, "时间格式不正确!"));
  477. }
  478. Task_WorkOrder workOrder = new Task_WorkOrder
  479. {
  480. Id = dto.Id,
  481. AssignedUserId = dto.AssignedUserId,
  482. CreateTime = DateTime.Now,
  483. CreateUserId = dto.CreateUserId,
  484. ForeignOptionId = dto.ForeignOptionId,
  485. GroupId = dto.GroupId,
  486. ProgressId = dto.Action,
  487. TypeId = dto.TypeId,
  488. IsDel = 0,
  489. Name = dto.Name,
  490. StartTime = startTime,
  491. };
  492. if (dto.Id == 0)
  493. {
  494. _sqlsugar.BeginTran();
  495. try
  496. {
  497. var workOrderId = _sqlsugar.Insertable<Task_WorkOrder>(workOrder).ExecuteReturnIdentity();
  498. workOrder.Id = workOrderId;
  499. List<Task_WorkTask> workTasks = dto.Tasks.Select(x => new Task_WorkTask
  500. {
  501. AssignedUserId = x.AssignedUserId,
  502. CreateTime = DateTime.Now,
  503. CreateUserId = dto.CreateUserId,
  504. DurationHours = x.DurationHours,
  505. IsUrgent = x.IsUrgent,
  506. PriorityId = x.PriorityId,
  507. WorkOrderId = workOrderId,
  508. IsDel = 0,
  509. StartTime = x.StartTime,
  510. EndTime = x.EndTime,
  511. IsExtraTask = x.IsExtraTask,
  512. Name = x.Name,
  513. Sort = x.Sort,
  514. Remark = x.Remark
  515. }).ToList();
  516. _sqlsugar.Insertable(workTasks).ExecuteCommand();
  517. _sqlsugar.CommitTran();
  518. jw.Code = 200;
  519. jw.Msg = "SUCCESS!";
  520. }
  521. catch (Exception ex)
  522. {
  523. _sqlsugar.RollbackTran();
  524. jw.Code = 500;
  525. jw.Msg = ex.Message;
  526. }
  527. }
  528. else
  529. {
  530. _sqlsugar.BeginTran();
  531. try
  532. {
  533. var workOrderId = workOrder.Id;
  534. _sqlsugar.Updateable<Task_WorkOrder>(workOrder)
  535. .IgnoreColumns(x => new { x.CreateTime, x.CreateUserId })
  536. .ExecuteCommand();
  537. List<Task_WorkTask> workTasks = dto.Tasks.Select(x => new Task_WorkTask
  538. {
  539. Id = x.Id,
  540. AssignedUserId = x.AssignedUserId,
  541. DurationHours = x.DurationHours,
  542. IsUrgent = x.IsUrgent,
  543. PriorityId = x.PriorityId,
  544. WorkOrderId = workOrderId,
  545. IsDel = 0,
  546. StartTime = x.StartTime,
  547. EndTime = x.EndTime,
  548. IsExtraTask = x.IsExtraTask,
  549. Name = x.Name,
  550. Sort = x.Sort,
  551. Remark = x.Remark
  552. }).ToList();
  553. var insertArr = workTasks.Where(x => x.Id == 0).ToList();
  554. var updateArr = workTasks.Where(x => x.Id > 0).ToList();
  555. var notdeleteArr = updateArr.Select(x => x.Id);
  556. _sqlsugar.Updateable<Task_WorkTask>().Where(x => x.WorkOrderId == workOrderId && !notdeleteArr.Contains(x.Id))
  557. .SetColumns(x => new Task_WorkTask
  558. {
  559. IsDel = 1,
  560. DeleteTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm"),
  561. DeleteUserId = dto.CreateUserId
  562. })
  563. .ExecuteCommand();
  564. if (insertArr.Any())
  565. {
  566. insertArr.ForEach(x =>
  567. {
  568. x.CreateUserId = dto.CreateUserId;
  569. x.CreateTime = DateTime.Now;
  570. });
  571. _sqlsugar.Insertable(insertArr).ExecuteCommand();
  572. }
  573. if (updateArr.Any())
  574. {
  575. _sqlsugar.Updateable(updateArr)
  576. .IgnoreColumns(x => new { x.CreateTime, x.CreateUserId })
  577. .ExecuteCommand();
  578. }
  579. _sqlsugar.CommitTran();
  580. jw.Code = 200;
  581. jw.Msg = "SUCCESS!";
  582. }
  583. catch (Exception ex)
  584. {
  585. _sqlsugar.RollbackTran();
  586. jw.Code = 500;
  587. jw.Msg = ex.Message;
  588. }
  589. }
  590. _ = AppNoticeLibrary.SendUserMsg_Task_ToUser(workOrder.Id,
  591. new List<string>() { dto.AssignedUserId.toString(), "235" }
  592. , dto.Id == 0);
  593. return Ok(jw);
  594. }
  595. /// <summary>
  596. /// 获取历史回执数据
  597. /// </summary>
  598. /// <param name="dto"></param>
  599. /// <returns></returns>
  600. [HttpPost]
  601. public async Task<IActionResult> GetWorkTaskReceipt(GetWorkTaskReceiptDto dto)
  602. {
  603. var jw = JsonView(false);
  604. if (dto.OrderId < 1 || dto.TaskId < 1)
  605. {
  606. jw.Msg = "参数有误!";
  607. return Ok(jw);
  608. }
  609. var data = _sqlsugar.Queryable<Task_WorkTaskReceipt>()
  610. .Where(x => x.WorkOrderId == dto.OrderId && x.WorkTaskId == dto.TaskId && x.IsDel == 0)
  611. .Select<WorkTaskReceiptView>()
  612. .ToList()
  613. .Select(x =>
  614. {
  615. var Files = JsonConvert.DeserializeObject<List<string>>(x.FilePath) ?? new
  616. List<string>();
  617. return new
  618. {
  619. x.WorkTaskId,
  620. x.WorkOrderId,
  621. x.Remark,
  622. x.Id,
  623. x.RejectReason,
  624. x.Content,
  625. x.IsApproved,
  626. x.IsCompleted,
  627. Files = Files.Select(y =>
  628. {
  629. var sp = y.Split('/');
  630. string filename = sp[sp.Length - 1];
  631. string ftpUrl = AppSettingsHelper.Get("TaskFileBaseUrl") + y.Replace(AppSettingsHelper.Get("TaskFileBasePath"), AppSettingsHelper.Get("TaskFileFtpPath"));
  632. return new
  633. {
  634. FileName = filename,
  635. Url = ftpUrl,
  636. };
  637. })
  638. };
  639. });
  640. jw.Data = data;
  641. jw.Code = 200;
  642. jw.Msg = "SUCCESS";
  643. return Ok(jw);
  644. }
  645. /// <summary>
  646. /// 任务回执提交
  647. /// </summary>
  648. /// <param name="dto"></param>
  649. /// <returns></returns>
  650. [HttpPost]
  651. public async Task<IActionResult> SubmitWorkTaskReceipt([FromForm] SubmitWorkTaskReceiptDto dto)
  652. {
  653. var jw = JsonView(false);
  654. jw.Data = "";
  655. var orderInfo = _sqlsugar.Queryable<Task_WorkOrder>().First(x => x.Id == dto.WorkOrderId && x.IsDel == 0);
  656. if (orderInfo == null)
  657. {
  658. jw.Msg = "参数有误!";
  659. return Ok(jw);
  660. }
  661. var filePathArr = new List<string>();
  662. if (dto.Files != null && dto.Files.Count > 0)
  663. {
  664. string winPath = AppSettingsHelper.Get("TaskFileBasePath") + $"{orderInfo.Name}/";
  665. if (!Directory.Exists(winPath))
  666. {
  667. Directory.CreateDirectory(winPath);
  668. }
  669. //保存文件
  670. foreach (var item in dto.Files)
  671. {
  672. var fileName = item.FileName;
  673. var saveFilePath = winPath + fileName;
  674. using (FileStream fs = System.IO.File.Create(saveFilePath))
  675. {
  676. item.CopyTo(fs);
  677. fs.Flush();
  678. }
  679. filePathArr.Add(saveFilePath);
  680. }
  681. }
  682. var task = _sqlsugar.Queryable<Task_WorkTask>().
  683. First(x => x.Id == dto.WorkTaskId && x.IsDel == 0);
  684. if (task == null)
  685. {
  686. jw.Msg = "任务不存在或已删除!";
  687. return Ok(jw);
  688. }
  689. Task_WorkTaskReceipt data = new Task_WorkTaskReceipt
  690. {
  691. Content = dto.Content,
  692. CreateTime = DateTime.Now,
  693. CreateUserId = dto.UserId,
  694. IsApproved = 0,
  695. IsCompleted = dto.IsCompleted,
  696. WorkTaskId = dto.WorkTaskId,
  697. WorkOrderId = dto.WorkOrderId,
  698. FilePath = JsonConvert.SerializeObject(filePathArr)
  699. };
  700. var count = _sqlsugar.Insertable(data).ExecuteCommand();
  701. if (count > 0)
  702. {
  703. _ = AppNoticeLibrary.SendUserMsg_TaskSubMit_ToUser(task.Id,
  704. new List<string>() { task.CreateUserId.toString() });
  705. }
  706. jw.Code = 200;
  707. jw.Msg = $"{count}条回执单、{dto.Files}个文件保存成功!";
  708. return Ok(jw);
  709. }
  710. /// <summary>
  711. /// 无记录通过
  712. /// </summary>
  713. /// <param name="dto"></param>
  714. /// <returns></returns>
  715. [HttpPost]
  716. public async Task<IActionResult> SubmitWorkTaskNotRecord(SubmitWorkTaskNotRecord dto)
  717. {
  718. var jw = JsonView(false);
  719. jw.Data = "";
  720. var orderInfo = _sqlsugar.Queryable<Task_WorkOrder>().First(x => x.Id == dto.WorkOrderId && x.IsDel == 0);
  721. if (orderInfo == null)
  722. {
  723. jw.Msg = "参数有误!";
  724. return Ok(jw);
  725. }
  726. var task = _sqlsugar.Queryable<Task_WorkTask>().
  727. First(x => x.Id == dto.WorkTaskId && x.IsDel == 0);
  728. if (task == null)
  729. {
  730. jw.Msg = "任务不存在或已删除!";
  731. return Ok(jw);
  732. }
  733. //设置工单回执
  734. Task_WorkTaskReceipt data = new Task_WorkTaskReceipt
  735. {
  736. Content = string.Empty,
  737. CreateTime = DateTime.Now,
  738. CreateUserId = dto.UserId,
  739. IsApproved = 1,
  740. IsCompleted = 1,
  741. WorkTaskId = dto.WorkTaskId,
  742. WorkOrderId = dto.WorkOrderId,
  743. FilePath = string.Empty
  744. };
  745. _sqlsugar.BeginTran();
  746. try
  747. {
  748. var ReceiptId = _sqlsugar.Insertable(data)
  749. .ExecuteReturnIdentity();
  750. var existsApproved = _sqlsugar.Queryable<Task_WorkTaskReceipt>()
  751. .First(x => x.WorkOrderId == dto.WorkOrderId
  752. && x.WorkTaskId == dto.WorkTaskId
  753. && x.IsDel == 0
  754. && x.IsApproved == 1
  755. && x.Id != ReceiptId);
  756. if (ReceiptId > 0 && existsApproved == null && !task.IsExtraTask)
  757. {
  758. _sqlsugar.Updateable<Task_WorkOrder>()
  759. .SetColumns(x => new Task_WorkOrder
  760. {
  761. ProgressId = x.ProgressId + 1
  762. })
  763. .Where(x => x.Id == orderInfo.Id)
  764. .ExecuteCommand();
  765. }
  766. _sqlsugar.CommitTran();
  767. jw.Code = 200;
  768. jw.Msg = "SUCCESS!";
  769. jw.Data = ReceiptId;
  770. }
  771. catch (Exception ex)
  772. {
  773. _sqlsugar.RollbackTran();
  774. jw.Code = 500;
  775. jw.Msg = ex.Message;
  776. }
  777. return Ok(jw);
  778. }
  779. /// <summary>
  780. /// 软删除工作订单
  781. /// </summary>
  782. /// <param name="dto">删除参数</param>
  783. /// <returns></returns>
  784. [HttpPost]
  785. public async Task<IActionResult> SoftDeleteWorkOrder(SoftDeleteWorkOrderDto dto)
  786. {
  787. var jw = JsonView(false);
  788. if (dto.WorkOrderId < 1 || dto.UserId < 1)
  789. {
  790. jw.Msg = "参数有误!";
  791. return Ok(jw);
  792. }
  793. // 检查工作订单是否存在且未被删除
  794. var workOrder = _sqlsugar.Queryable<Task_WorkOrder>()
  795. .First(x => x.Id == dto.WorkOrderId && x.IsDel == 0);
  796. if (workOrder == null)
  797. {
  798. jw.Msg = "工作订单不存在或已被删除!";
  799. return Ok(jw);
  800. }
  801. // 检查权限:只有创建者或管理员可以删除
  802. if (workOrder.CreateUserId != dto.UserId)
  803. {
  804. jw.Msg = "权限不足,无法删除此工作订单!";
  805. return Ok(jw);
  806. }
  807. _sqlsugar.BeginTran();
  808. try
  809. {
  810. // 软删除工作订单
  811. var updateWorkOrderResult = _sqlsugar.Updateable<Task_WorkOrder>()
  812. .SetColumns(x => new Task_WorkOrder
  813. {
  814. IsDel = 1,
  815. DeleteTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  816. DeleteUserId = dto.UserId
  817. })
  818. .Where(x => x.Id == dto.WorkOrderId)
  819. .ExecuteCommand();
  820. // 软删除相关的工作任务
  821. var updateWorkTaskResult = _sqlsugar.Updateable<Task_WorkTask>()
  822. .SetColumns(x => new Task_WorkTask
  823. {
  824. IsDel = 1,
  825. DeleteTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  826. DeleteUserId = dto.UserId
  827. })
  828. .Where(x => x.WorkOrderId == dto.WorkOrderId)
  829. .ExecuteCommand();
  830. // 软删除相关的回执单
  831. var updateReceiptResult = _sqlsugar.Updateable<Task_WorkTaskReceipt>()
  832. .SetColumns(x => new Task_WorkTaskReceipt
  833. {
  834. IsDel = 1,
  835. DeleteTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  836. DeleteUserId = dto.UserId
  837. })
  838. .Where(x => x.WorkOrderId == dto.WorkOrderId)
  839. .ExecuteCommand();
  840. _sqlsugar.CommitTran();
  841. jw.Code = 200;
  842. jw.Msg = "工作订单删除成功!";
  843. jw.Data = new
  844. {
  845. WorkOrderId = dto.WorkOrderId,
  846. DeletedWorkTasks = updateWorkTaskResult,
  847. DeletedReceipts = updateReceiptResult,
  848. DeleteTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  849. };
  850. }
  851. catch (Exception ex)
  852. {
  853. _sqlsugar.RollbackTran();
  854. jw.Code = 500;
  855. jw.Msg = $"删除失败:{ex.Message}";
  856. }
  857. return Ok(jw);
  858. }
  859. /// <summary>
  860. /// 回执审核
  861. /// </summary>
  862. /// <param name="dto"></param>
  863. /// <returns></returns>
  864. [HttpPost]
  865. public async Task<IActionResult> AuditWorkTaskReceipt(AuditWorkTaskReceiptDto dto)
  866. {
  867. var jw = JsonView(false);
  868. if (dto.ReceiptId < 1 || dto.ApproverId < 1)
  869. {
  870. jw.Msg = "参数有误!";
  871. return Ok(jw);
  872. }
  873. if (dto.Approve == -1 && string.IsNullOrWhiteSpace(dto.RejectReason))
  874. {
  875. jw.Msg = "驳回必须填写原因!";
  876. return Ok(jw);
  877. }
  878. var receipt = _sqlsugar.Queryable<Task_WorkTaskReceipt>()
  879. .Includes<Task_WorkTask>(o => o.WorkTask)
  880. .First(x => x.Id == dto.ReceiptId && x.IsDel == 0);
  881. if (receipt == null)
  882. {
  883. jw.Msg = "回执单不存在或已删除!";
  884. return Ok(jw);
  885. }
  886. try
  887. {
  888. _sqlsugar.BeginTran();
  889. var existsApproved = _sqlsugar.Queryable<Task_WorkTaskReceipt>()
  890. .First(x => x.WorkOrderId == receipt.WorkOrderId && x.WorkTaskId == receipt.WorkTaskId && x.IsDel == 0 && x.IsApproved == 1);
  891. var affected = _sqlsugar.Updateable<Task_WorkTaskReceipt>()
  892. .SetColumns(x => new Task_WorkTaskReceipt
  893. {
  894. IsApproved = dto.Approve,
  895. RejectReason = dto.Approve == -1 ? dto.RejectReason : ""
  896. })
  897. .Where(x => x.Id == dto.ReceiptId)
  898. .ExecuteCommand();
  899. if (existsApproved == null && dto.Approve == 1 && !receipt.WorkTask.IsExtraTask)
  900. {
  901. _sqlsugar.Updateable<Task_WorkOrder>()
  902. .SetColumns(x => new Task_WorkOrder
  903. {
  904. ProgressId = x.ProgressId + 1
  905. })
  906. .Where(x => x.Id == receipt.WorkOrderId)
  907. .ExecuteCommand();
  908. }
  909. _sqlsugar.CommitTran();
  910. //企微通知
  911. _ = AppNoticeLibrary.SendUserMsg_TaskAudit_ToUser
  912. (receipt.WorkTaskId, dto.Approve == 1,
  913. dto.RejectReason,
  914. new List<string>() { receipt.WorkTask.AssignedUserId.toString() });
  915. jw.Code = 200;
  916. jw.Msg = "审核完成!";
  917. jw.Data = new
  918. {
  919. dto.ReceiptId,
  920. Status = dto.Approve == 1 ? "审核通过" : "审核驳回",
  921. dto.RejectReason,
  922. receipt.WorkOrderId,
  923. Affected = affected
  924. };
  925. }
  926. catch (Exception ex)
  927. {
  928. _sqlsugar.RollbackTran();
  929. jw.Code = 500;
  930. jw.Msg = $"审核失败:{ex.Message}";
  931. }
  932. return Ok(jw);
  933. }
  934. /// <summary>
  935. /// 设置回执备注信息
  936. /// </summary>
  937. /// <param name="dto"></param>
  938. /// <returns></returns>
  939. [HttpPost]
  940. public async Task<IActionResult> SetWorkTaskReceiptRemark(SetWorkTaskReceiptRemarkDto dto)
  941. {
  942. var jw = JsonView(false);
  943. if (dto.ReceiptId < 1 || string.IsNullOrWhiteSpace(dto.Remark))
  944. {
  945. jw.Msg = "参数有误!";
  946. return Ok(jw);
  947. }
  948. var receipt = _sqlsugar.Queryable<Task_WorkTaskReceipt>()
  949. .First(x => x.Id == dto.ReceiptId && x.IsDel == 0);
  950. if (receipt == null)
  951. {
  952. jw.Msg = "回执单不存在或已删除!";
  953. return Ok(jw);
  954. }
  955. _sqlsugar.Updateable<Task_WorkTaskReceipt>()
  956. .SetColumns(x => new Task_WorkTaskReceipt
  957. {
  958. Remark = dto.Remark
  959. })
  960. .Where(x => x.Id == dto.ReceiptId)
  961. .ExecuteCommand();
  962. jw.Code = 200;
  963. jw.Msg = "设置回执备注信息成功!";
  964. jw.Data = new
  965. {
  966. dto.ReceiptId,
  967. dto.Remark
  968. };
  969. return Ok(jw);
  970. }
  971. [HttpPost]
  972. public async Task<IActionResult> TestDoubao(string messages)
  973. {
  974. var jw = JsonView(false);
  975. var result = await _doubaoService.CompleteChatAsync
  976. (
  977. new List<DouBaoChatMessage>()
  978. {
  979. new DouBaoChatMessage() { Role = DouBaoRole.system, Content = "你是一个专业的AI助手,请根据用户的问题给出回答" },
  980. new DouBaoChatMessage() { Role = DouBaoRole.user, Content = messages }
  981. },
  982. new CompleteChatOptions()
  983. {
  984. ThinkingOptions = new thinkingOptions()
  985. { IsThinking = true, ReasoningEffort = ReasoningEffort.High }
  986. }
  987. );
  988. jw.Code = 200;
  989. jw.Msg = "SUCCESS!";
  990. jw.Data = result;
  991. return Ok(jw);
  992. }
  993. [HttpPost]
  994. public async Task<IActionResult> TestQiYeWeChat(DateTime startDt, DateTime endDt, int cursor = 0, int limit = 10)
  995. {
  996. var jw = JsonView(false);
  997. List<JournalFilter> filters = new List<JournalFilter>()
  998. {
  999. new JournalFilter() { key = JournalFilterKey.template_id, value = "C4Rby4W2jqwk9mfiN3fZbQZHaawzwNZgn8vdt4g8y" }
  1000. };
  1001. var result = await _qiYeWeChatApiService.GetJournalRecordListAsync(startDt, endDt, cursor, limit, filters);
  1002. jw.Code = 200;
  1003. jw.Msg = "SUCCESS!";
  1004. jw.Data = result;
  1005. return Ok(jw);
  1006. }
  1007. [HttpPost]
  1008. public async Task<IActionResult> TestQiYeWeChatDetail(string journaluuid)
  1009. {
  1010. var jw = JsonView(false);
  1011. var result = await _qiYeWeChatApiService.GetJournalRecordDetailAsync(journaluuid);
  1012. jw.Code = 200;
  1013. jw.Msg = "SUCCESS!";
  1014. jw.Data = result;
  1015. return Ok(jw);
  1016. }
  1017. }
  1018. }